require 'logger' require 'logstash-logger/tagged_logging' require 'logstash-logger/silenced_logging' module LogStashLogger autoload :MultiLogger, 'logstash-logger/multi_logger' def self.new(*args) opts = extract_opts(*args) build_logger(opts) end def self.extended(base) base.instance_eval do class << self attr_reader :device end def flush !!(@device.flush if @device.respond_to?(:flush)) end end end protected def self.extract_opts(*args) args.flatten! if args.length > 1 if args.all?{|arg| arg.is_a?(Hash)} # Deprecated array of hashes warn "[LogStashLogger] Passing an array of hashes to the constructor is deprecated. Please replace with an options hash: { type: :multi_delegator, outputs: [...] }" { type: :multi_delegator, outputs: args } else # Deprecated host/port/type constructor warn "[LogStashLogger] The (host, port, type) constructor is deprecated. Please use an options hash instead." host, port, type = *args { host: host, port: port, type: type } end elsif Hash === args[0] args[0] else fail ArgumentError, "Invalid LogStashLogger options" end end def self.build_logger(opts) formatter = Formatter.new(opts.delete(:formatter)) logger = case opts[:type] when :multi_logger build_multi_logger(opts) when :syslog build_syslog_logger(opts) else build_default_logger(opts) end logger.formatter = formatter if formatter logger end private def self.build_default_logger(opts) device = Device.new(opts) ::Logger.new(device).tap do |logger| logger.instance_variable_set(:@device, device) logger.extend(self) logger.extend(TaggedLogging) logger.extend(SilencedLogging) end end def self.build_multi_logger(opts) loggers = opts[:outputs].map { |logger_opts| build_logger(logger_opts) } MultiLogger.new(loggers) end def self.build_syslog_logger(opts) logger = begin require 'syslog/logger' Syslog::Logger.new(opts[:program_name], opts[:facility]) rescue ArgumentError Syslog::Logger.new(opts[:program_name]) end logger.extend(self) logger.extend(TaggedLogging) logger.extend(SilencedLogging) end end