|  | @ -9,72 +9,68 @@ require 'json' | 
		
	
		
			
				|  |  | class AnsiFormatter < SemanticLogger::Formatters::Color |  |  | class AnsiFormatter < SemanticLogger::Formatters::Color | 
		
	
		
			
				|  |  |   include AnsiColors |  |  |   include AnsiColors | 
		
	
		
			
				|  |  | 
 |  |  | 
 | 
		
	
		
			
				|  |  |   ANSI_DEBUG = CLEAR + TEXT_GRAY_400 |  |  |  | 
		
	
		
			
				|  |  |   ANSI_INFO = CLEAR + TEXT_GRAY_100 |  |  |  | 
		
	
		
			
				|  |  |   ANSI_WARN = CLEAR + BG_YELLOW + TEXT_BLACK |  |  |  | 
		
	
		
			
				|  |  |   ANSI_ERROR = CLEAR + BG_RED + TEXT_WHITE |  |  |  | 
		
	
		
			
				|  |  |   ANSI_FATAL = CLEAR + BG_MAGENTA + BOLD + TEXT_WHITE |  |  |  | 
		
	
		
			
				|  |  |   NAME_MAX_SIZE = 20 |  |  |  | 
		
	
		
			
				|  |  |  |  |  |   CENTER_SIZE = 20 | 
		
	
		
			
				|  |  |   FOREMAN_PREFIX_LENGTH = 18 |  |  |   FOREMAN_PREFIX_LENGTH = 18 | 
		
	
		
			
				|  |  |   DEFAULT_UNDETECTED_WRAP = 80 |  |  |  | 
		
	
		
			
				|  |  |  |  |  |   FAILOVER_WRAP = 80 | 
		
	
		
			
				|  |  | 
 |  |  | 
 | 
		
	
		
			
				|  |  |   def initialize |  |  |   def initialize | 
		
	
		
			
				|  |  |     super(color_map: ColorMap.new( |  |  |  | 
		
	
		
			
				|  |  |       debug: ANSI_DEBUG, |  |  |  | 
		
	
		
			
				|  |  |       info: ANSI_INFO, |  |  |  | 
		
	
		
			
				|  |  |       warn: ANSI_WARN, |  |  |  | 
		
	
		
			
				|  |  |       error: ANSI_ERROR, |  |  |  | 
		
	
		
			
				|  |  |       fatal: ANSI_FATAL |  |  |  | 
		
	
		
			
				|  |  |     )) |  |  |  | 
		
	
		
			
				|  |  |  |  |  |     super(color_map: | 
		
	
		
			
				|  |  |  |  |  |       ColorMap.new( | 
		
	
		
			
				|  |  |  |  |  |         debug: CLEAR + TEXT_GRAY_400, | 
		
	
		
			
				|  |  |  |  |  |         info: CLEAR + TEXT_GRAY_100, | 
		
	
		
			
				|  |  |  |  |  |         warn: CLEAR + BG_YELLOW + TEXT_BLACK, | 
		
	
		
			
				|  |  |  |  |  |         error: CLEAR + BG_RED + TEXT_WHITE, | 
		
	
		
			
				|  |  |  |  |  |         fatal: CLEAR + BG_MAGENTA + BOLD + TEXT_WHITE | 
		
	
		
			
				|  |  |  |  |  |       )) | 
		
	
		
			
				|  |  |   end |  |  |   end | 
		
	
		
			
				|  |  | 
 |  |  | 
 | 
		
	
		
			
				|  |  |   def call(log, logger) |  |  |   def call(log, logger) | 
		
	
		
			
				|  |  |  |  |  |     log = AnsiFilter.alter(log) | 
		
	
		
			
				|  |  |  |  |  | 
 | 
		
	
		
			
				|  |  |     self.log = log |  |  |     self.log = log | 
		
	
		
			
				|  |  |     self.logger = logger |  |  |     self.logger = logger | 
		
	
		
			
				|  |  |     self.color = color_map[log.level] |  |  |     self.color = color_map[log.level] | 
		
	
		
			
				|  |  |     wrap_length = compute_useful_length |  |  |  | 
		
	
		
			
				|  |  |     wrap_level(wrap_length, message, payload, exception) |  |  |  | 
		
	
		
			
				|  |  |   end |  |  |  | 
		
	
		
			
				|  |  | 
 |  |  | 
 | 
		
	
		
			
				|  |  |   def origin |  |  |  | 
		
	
		
			
				|  |  |     origin = log.name.truncate(NAME_MAX_SIZE).center(NAME_MAX_SIZE) |  |  |  | 
		
	
		
			
				|  |  |     colorize(origin, AnsiColors::TEXT_CYAN) |  |  |  | 
		
	
		
			
				|  |  |  |  |  |     wrap_level(compute_useful_length, message, payload, exception) | 
		
	
		
			
				|  |  |   end |  |  |   end | 
		
	
		
			
				|  |  | 
 |  |  | 
 | 
		
	
		
			
				|  |  |   def message |  |  |  | 
		
	
		
			
				|  |  |     return unless log.message |  |  |  | 
		
	
		
			
				|  |  |  |  |  |   private | 
		
	
		
			
				|  |  | 
 |  |  | 
 | 
		
	
		
			
				|  |  |     colorize(log.message) |  |  |  | 
		
	
		
			
				|  |  |   end |  |  |  | 
		
	
		
			
				|  |  |  |  |  |   def origin                          = colorize(centerize(log.name), TEXT_CYAN) | 
		
	
		
			
				|  |  |  |  |  |   def build_prefix(char)              = "#{origin} ╣#{colorize(char)}╠  " | 
		
	
		
			
				|  |  |  |  |  |   def centerize(text)                 = text.truncate(CENTER_SIZE).center(CENTER_SIZE) | 
		
	
		
			
				|  |  |  |  |  |   def colorize(text, tint = color)    = "#{tint}#{text}#{CLEAR}" | 
		
	
		
			
				|  |  |  |  |  |   def stackisize(items)               = items.length.positive? ? "\n#{items.join("\n")}" : '' | 
		
	
		
			
				|  |  | 
 |  |  | 
 | 
		
	
		
			
				|  |  |   def exception |  |  |  | 
		
	
		
			
				|  |  |     return unless log.exception |  |  |  | 
		
	
		
			
				|  |  |  |  |  |   def build_dimensions(dimensions) | 
		
	
		
			
				|  |  |  |  |  |     "#{origin} #{dimensions.rails}  " | 
		
	
		
			
				|  |  |  |  |  |   end | 
		
	
		
			
				|  |  | 
 |  |  | 
 | 
		
	
		
			
				|  |  |     clazz = "#{log.exception.class}\n" |  |  |  | 
		
	
		
			
				|  |  |     message = log.exception.message.chomp('') |  |  |  | 
		
	
		
			
				|  |  |     backtrace = clean_backtrace(log.exception.backtrace) |  |  |  | 
		
	
		
			
				|  |  |  |  |  |   def compute_useful_length | 
		
	
		
			
				|  |  |  |  |  |     IO.console.winsize[1] - FOREMAN_PREFIX_LENGTH | 
		
	
		
			
				|  |  |  |  |  |   rescue StandardError | 
		
	
		
			
				|  |  |  |  |  |     FAILOVER_WRAP | 
		
	
		
			
				|  |  |  |  |  |   end | 
		
	
		
			
				|  |  | 
 |  |  | 
 | 
		
	
		
			
				|  |  |     "#{colorize(clazz, ANSI_FATAL)}#{colorize(message, ANSI_ERROR)}#{backtrace}" |  |  |  | 
		
	
		
			
				|  |  |  |  |  |   def message | 
		
	
		
			
				|  |  |  |  |  |     colorize(log.message) if log.message | 
		
	
		
			
				|  |  |   end |  |  |   end | 
		
	
		
			
				|  |  | 
 |  |  | 
 | 
		
	
		
			
				|  |  |   private |  |  |  | 
		
	
		
			
				|  |  |  |  |  |   def exception | 
		
	
		
			
				|  |  |  |  |  |     return unless log.exception | 
		
	
		
			
				|  |  | 
 |  |  | 
 | 
		
	
		
			
				|  |  |   def colorize(text, tint = color)    = "#{tint}#{text}#{AnsiColors::CLEAR}" |  |  |  | 
		
	
		
			
				|  |  |   def build_prefix                    = "#{origin} #{colorize(draw_rails(level_char))}  " |  |  |  | 
		
	
		
			
				|  |  |   def build_continuation              = "#{origin} #{colorize(draw_rails('┆'))}  " |  |  |  | 
		
	
		
			
				|  |  |   def draw_rails(char)                = "╣#{char}╠" |  |  |  | 
		
	
		
			
				|  |  |  |  |  |     exc = log.exception | 
		
	
		
			
				|  |  |  |  |  |     clazz = colorize("#{exc.class}\n", color_map[:fatal]) | 
		
	
		
			
				|  |  |  |  |  |     message = colorize(exc.message.chomp(''), color_map[:error]) | 
		
	
		
			
				|  |  |  |  |  |     backtrace = stackisize(clean_backtrace(exc.backtrace)) | 
		
	
		
			
				|  |  | 
 |  |  | 
 | 
		
	
		
			
				|  |  |   def compute_useful_length |  |  |  | 
		
	
		
			
				|  |  |     IO.console.winsize[1] - FOREMAN_PREFIX_LENGTH |  |  |  | 
		
	
		
			
				|  |  |   rescue StandardError |  |  |  | 
		
	
		
			
				|  |  |     DEFAULT_UNDETECTED_WRAP |  |  |  | 
		
	
		
			
				|  |  |  |  |  |     "#{clazz}#{message}#{backtrace}" | 
		
	
		
			
				|  |  |   end |  |  |   end | 
		
	
		
			
				|  |  | 
 |  |  | 
 | 
		
	
		
			
				|  |  |   def clean_backtrace(backtrace) |  |  |   def clean_backtrace(backtrace) | 
		
	
		
			
				|  |  |     root_path = Rails.root.to_s |  |  |     root_path = Rails.root.to_s | 
		
	
		
			
				|  |  |     backtrace = backtrace.select { |line| line.starts_with?(root_path) } |  |  |     backtrace = backtrace.select { |line| line.starts_with?(root_path) } | 
		
	
		
			
				|  |  |     backtrace = backtrace.map { |line| line.delete_prefix("#{root_path}/") } |  |  |  | 
		
	
		
			
				|  |  |     backtrace.compact.join("\n") |  |  |  | 
		
	
		
			
				|  |  |  |  |  |     backtrace.map { |line| line.delete_prefix("#{root_path}/") } | 
		
	
		
			
				|  |  |   end |  |  |   end | 
		
	
		
			
				|  |  | 
 |  |  | 
 | 
		
	
		
			
				|  |  |   def level_char |  |  |   def level_char | 
		
	
	
		
			
				|  | @ -85,8 +81,11 @@ class AnsiFormatter < SemanticLogger::Formatters::Color | 
		
	
		
			
				|  |  |   end |  |  |   end | 
		
	
		
			
				|  |  | 
 |  |  | 
 | 
		
	
		
			
				|  |  |   def wrap_level(length, *items) |  |  |   def wrap_level(length, *items) | 
		
	
		
			
				|  |  |     prefix = build_prefix |  |  |  | 
		
	
		
			
				|  |  |     continuation = build_continuation |  |  |  | 
		
	
		
			
				|  |  |  |  |  |     prefix = log.dimensions ? build_dimensions(log.dimensions) : build_prefix(level_char) | 
		
	
		
			
				|  |  |  |  |  |     continuation = build_prefix('┆') | 
		
	
		
			
				|  |  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |  |     log.dimensions&.before&.times { items.unshift('') } | 
		
	
		
			
				|  |  |  |  |  |     log.dimensions&.after&.times { items << '' } | 
		
	
		
			
				|  |  |     items.map do |item| |  |  |     items.map do |item| | 
		
	
		
			
				|  |  |       AnsiWrapper.wrap(item, length, prefix, continuation) |  |  |       AnsiWrapper.wrap(item, length, prefix, continuation) | 
		
	
		
			
				|  |  |     end.compact.join("\n") |  |  |     end.compact.join("\n") | 
		
	
	
		
			
				|  | 
 |