Rails 日志中的缩进

Indentation in Rails logs

记录自定义信息时,例如在 rake 任务中,我想缩进日志行以提高可读性。例如:

Seeding database...
  Importing xyz.csv...
    Skipping row 5 due to invalid value 'Unknown' in column 'year'
  Finished importing xyz.csv
Finished seeding database

在我的 seeds.rb 中,我使用以下内容进行日志记录:

logger = Logger.new(STDOUT)
logger.info('Seeding database...')

然而,Skipping row 5... 的日志记录发生在服务中,不一定必须从 seeds.rb 调用,但可以从任何地方调用。因此,我无法对正确的缩进进行硬编码(无论如何,这听起来是个坏主意)。

一种可能是保留 "indentation counter",我可以在开始导入文件时增加它,在完成时减少它。我不确定如何从我的应用程序中的任何地方访问它,或者这是否是最佳解决方案。有什么想法吗?

更好的解决方案是为您的服务对象提供记录器对象以及要处理的数据。这样它就不必知道有关您的记录器首选项的任何信息。

并使用类似的东西:

require 'logger'

class IndentedLogger < Logger

  INDENTATION_STR = '  '.freeze
  attr_accessor :indentation

  def initialize(io, *args, **params)
    self.indentation = 0
    super
  end

  def indented
    return self.dup.tap{|l| l.indentation += 1 } unless block_given?
    self.indentation += 1
    yield self
    self.indentation -= 1
  end

  protected

  def format_message(severity, datetime, progname, msg)
    (@formatter || @default_formatter).call(severity, datetime, progname, "#{INDENTATION_STR*indentation}#{msg}")
  end

end

# Example:
logger = IndentedLogger.new(STDOUT)

logger.info "Foo"
logger.indented{
  logger.info "Foo"
  logger.indented.info "Indented even more"
}
logger.info "Foo"

以及服务调用 - YourService.new(some_data, logger: logger.indented).process_or_whatever