require 'hanami/utils/class_attribute'
require 'hanami/http/status'
module Hanami
module Action
# Throw API
#
# @since 0.1.0
#
# @see Hanami::Action::Throwable::ClassMethods#handle_exception
# @see Hanami::Action::Throwable#halt
# @see Hanami::Action::Throwable#status
module Throwable
# @since 0.2.0
# @api private
RACK_ERRORS = 'rack.errors'.freeze
# This isn't part of Rack SPEC
#
# Exception notifiers use rack.exception instead of
# rack.errors, so we need to support it.
#
# @since 0.5.0
# @api private
#
# @see Hanami::Action::Throwable::RACK_ERRORS
# @see http://www.rubydoc.info/github/rack/rack/file/SPEC#The_Error_Stream
# @see https://github.com/hanami/controller/issues/133
RACK_EXCEPTION = 'rack.exception'.freeze
# @since 0.1.0
# @api private
def self.included(base)
base.extend ClassMethods
end
# Throw API class methods
#
# @since 0.1.0
# @api private
module ClassMethods
private
# Handle the given exception with an HTTP status code.
#
# When the exception is raise during #call execution, it will be
# translated into the associated HTTP status.
#
# This is a fine grained control, for a global configuration see
# Hanami::Action.handled_exceptions
#
# @param exception [Hash] the exception class must be the key and the
# HTTP status the value of the hash
#
# @since 0.1.0
#
# @see Hanami::Action.handled_exceptions
#
# @example
# require 'hanami/controller'
#
# class Show
# include Hanami::Action
# handle_exception RecordNotFound => 404
#
# def call(params)
# # ...
# raise RecordNotFound.new
# end
# end
#
# Show.new.call({id: 1}) # => [404, {}, ['Not Found']]
def handle_exception(exception)
configuration.handle_exception(exception)
end
end
protected
# Halt the action execution with the given HTTP status code and message.
#
# When used, the execution of a callback or of an action is interrupted
# and the control returns to the framework, that decides how to handle
# the event.
#
# If a message is provided, it sets the response body with the message.
# Otherwise, it sets the response body with the default message associated
# to the code (eg 404 will set `"Not Found"`).
#
# @param code [Fixnum] a valid HTTP status code
# @param message [String] the response body
#
# @since 0.2.0
#
# @see Hanami::Controller#handled_exceptions
# @see Hanami::Action::Throwable#handle_exception
# @see Hanami::Http::Status:ALL
#
# @example Basic usage
# require 'hanami/controller'
#
# class Show
# def call(params)
# halt 404
# end
# end
#
# # => [404, {}, ["Not Found"]]
#
# @example Custom message
# require 'hanami/controller'
#
# class Show
# def call(params)
# halt 404, "This is not the droid you're looking for."
# end
# end
#
# # => [404, {}, ["This is not the droid you're looking for."]]
def halt(code, message = nil)
message ||= Http::Status.message_for(code)
status(code, message)
throw :halt
end
# Sets the given code and message for the response
#
# @param code [Fixnum] a valid HTTP status code
# @param message [String] the response body
#
# @since 0.1.0
# @see Hanami::Http::Status:ALL
def status(code, message)
self.status = code
self.body = message
end
private
# @since 0.1.0
# @api private
def _rescue
catch :halt do
begin
yield
rescue => exception
_reference_in_rack_errors(exception)
_handle_exception(exception)
end
end
end
# @since 0.2.0
# @api private
def _reference_in_rack_errors(exception)
return if configuration.handled_exception?(exception)
@_env[RACK_EXCEPTION] = exception
if errors = @_env[RACK_ERRORS]
errors.write(_dump_exception(exception))
errors.flush
end
end
# @since 0.2.0
# @api private
def _dump_exception(exception)
[[exception.class, exception.message].compact.join(": "), *exception.backtrace].join("\n\t")
end
# @since 0.1.0
# @api private
def _handle_exception(exception)
raise unless configuration.handle_exceptions
instance_exec(
exception,
&_exception_handler(exception)
)
end
# @since 0.3.0
# @api private
def _exception_handler(exception)
handler = configuration.exception_handler(exception)
if respond_to?(handler.to_s, true)
method(handler)
else
->(ex) { halt handler }
end
end
end
end
end