# frozen_string_literal: true require_relative 'null_io' require_relative 'error_logger' require 'stringio' require 'io/wait' unless Puma::HAS_NATIVE_IO_WAIT module Puma # Handles logging concerns for both standard messages # (+stdout+) and errors (+stderr+). class LogWriter class DefaultFormatter def call(str) str end end class PidFormatter def call(str) "[#{$$}] #{str}" end end LOG_QUEUE = Queue.new attr_reader :stdout, :stderr attr_accessor :formatter, :custom_logger # Create a LogWriter that prints to +stdout+ and +stderr+. def initialize(stdout, stderr) @formatter = DefaultFormatter.new @custom_logger = nil @stdout = stdout @stderr = stderr @debug = ENV.key?('PUMA_DEBUG') @error_logger = ErrorLogger.new(@stderr) end DEFAULT = new(STDOUT, STDERR) # Returns an LogWriter object which writes its status to # two StringIO objects. def self.strings LogWriter.new(StringIO.new, StringIO.new) end def self.stdio LogWriter.new($stdout, $stderr) end def self.null n = NullIO.new LogWriter.new(n, n) end # Write +str+ to +@stdout+ def log(str) if @custom_logger&.respond_to?(:write) @custom_logger.write(format(str)) else internal_write "#{@formatter.call str}\n" end end def write(str) internal_write @formatter.call(str) end def internal_write(str) LOG_QUEUE << str while (w_str = LOG_QUEUE.pop(true)) do begin @stdout.is_a?(IO) and @stdout.wait_writable(1) @stdout.write w_str @stdout.flush unless @stdout.sync rescue Errno::EPIPE, Errno::EBADF, IOError, Errno::EINVAL # 'Invalid argument' (Errno::EINVAL) may be raised by flush end end rescue ThreadError end private :internal_write def debug? @debug end def debug(str) log("% #{str}") if @debug end # Write +str+ to +@stderr+ def error(str) @error_logger.info(text: @formatter.call("ERROR: #{str}")) exit 1 end def format(str) formatter.call(str) end # An HTTP connection error has occurred. # +error+ a connection exception, +req+ the request, # and +text+ additional info # @version 5.0.0 def connection_error(error, req, text="HTTP connection error") @error_logger.info(error: error, req: req, text: text) end # An HTTP parse error has occurred. # +error+ a parsing exception, # and +req+ the request. def parse_error(error, req) @error_logger.info(error: error, req: req, text: 'HTTP parse error, malformed request') end # An SSL error has occurred. # @param error # @param ssl_socket def ssl_error(error, ssl_socket) peeraddr = ssl_socket.peeraddr.last rescue "" peercert = ssl_socket.peercert subject = peercert ? peercert.subject : nil @error_logger.info(error: error, text: "SSL error, peer: #{peeraddr}, peer cert: #{subject}") end # An unknown error has occurred. # +error+ an exception object, +req+ the request, # and +text+ additional info def unknown_error(error, req=nil, text="Unknown error") @error_logger.info(error: error, req: req, text: text) end # Log occurred error debug dump. # +error+ an exception object, +req+ the request, # and +text+ additional info # @version 5.0.0 def debug_error(error, req=nil, text="") @error_logger.debug(error: error, req: req, text: text) end end end