require 'erb' module ActionController # The Failsafe middleware is usually the top-most middleware in the Rack # middleware chain. It returns the underlying middleware's response, but if # the underlying middle raises an exception then Failsafe will log the # exception into the Rails log file, and will attempt to return an error # message response. # # Failsafe is a last resort for logging errors and for telling the HTTP # client that something went wrong. Do not confuse this with the # ActionController::Rescue module, which is responsible for catching # exceptions at deeper levels. Unlike Failsafe, which is as simple as # possible, Rescue provides features that allow developers to hook into # the error handling logic, and can customize the error message response # based on the HTTP client's IP. class Failsafe cattr_accessor :error_file_path self.error_file_path = Rails.public_path if defined?(Rails.public_path) def initialize(app) @app = app end def call(env) @app.call(env) rescue Exception => exception # Reraise exception in test environment if defined?(Rails) && Rails.env.test? raise exception else failsafe_response(exception) end end private def failsafe_response(exception) log_failsafe_exception(exception) [500, {'Content-Type' => 'text/html'}, [failsafe_response_body]] rescue Exception => failsafe_error # Logger or IO errors $stderr.puts "Error during failsafe response: #{failsafe_error}" end def failsafe_response_body error_template_path = "#{self.class.error_file_path}/500.html" if File.exist?(error_template_path) begin result = render_template(error_template_path) rescue Exception result = nil end else result = nil end if result.nil? result = "<html><body><h1>500 Internal Server Error</h1>" << "If you are the administrator of this website, then please read this web " << "application's log file to find out what went wrong.</body></html>" end result end # The default 500.html uses the h() method. def h(text) # :nodoc: ERB::Util.h(text) end def render_template(filename) ERB.new(File.read(filename)).result(binding) end def log_failsafe_exception(exception) message = "/!\\ FAILSAFE /!\\ #{Time.now}\n Status: 500 Internal Server Error\n" message << " #{exception}\n #{exception.backtrace.join("\n ")}" if exception failsafe_logger.fatal(message) failsafe_logger.flush if failsafe_logger.respond_to?(:flush) end def failsafe_logger if defined?(Rails) && Rails.logger Rails.logger else Logger.new($stderr) end end end end