require 'spec_helper' require 'active_support/core_ext/hash' describe Grape::Middleware::Error do # raises a text exception class ExceptionApp class << self def call(env) raise "rain!" end end end # raises a hash error class ErrorHashApp class << self def error!(message, status = 403) throw :error, message: { error: message, detail: "missing widget" }, status: status end def call(env) error!("rain!", 401) end end end # raises an error! class AccessDeniedApp class << self def error!(message, status = 403) throw :error, message: message, status: status end def call(env) error!("Access Denied", 401) end end end # raises a custom error class CustomError < Grape::Exceptions::Base end class CustomErrorApp class << self def call(env) raise CustomError, status: 400, message: 'failed validation' end end end attr_reader :app it 'does not trap errors by default' do @app ||= Rack::Builder.app do use Grape::Middleware::Error run ExceptionApp end lambda { get '/' }.should raise_error end context 'with rescue_all set to true' do it 'sets the message appropriately' do @app ||= Rack::Builder.app do use Grape::Middleware::Error, rescue_all: true run ExceptionApp end get '/' last_response.body.should == "rain!" end it 'defaults to a 403 status' do @app ||= Rack::Builder.app do use Grape::Middleware::Error, rescue_all: true run ExceptionApp end get '/' last_response.status.should == 403 end it 'is possible to specify a different default status code' do @app ||= Rack::Builder.app do use Grape::Middleware::Error, rescue_all: true, default_status: 500 run ExceptionApp end get '/' last_response.status.should == 500 end it 'is possible to return errors in json format' do @app ||= Rack::Builder.app do use Grape::Middleware::Error, rescue_all: true, format: :json run ExceptionApp end get '/' last_response.body.should == '{"error":"rain!"}' end it 'is possible to return hash errors in json format' do @app ||= Rack::Builder.app do use Grape::Middleware::Error, rescue_all: true, format: :json run ErrorHashApp end get '/' ['{"error":"rain!","detail":"missing widget"}', '{"detail":"missing widget","error":"rain!"}'].should include(last_response.body) end it 'is possible to return errors in jsonapi format' do @app ||= Rack::Builder.app do use Grape::Middleware::Error, rescue_all: true, format: :jsonapi run ExceptionApp end get '/' last_response.body.should == '{"error":"rain!"}' end it 'is possible to return hash errors in jsonapi format' do @app ||= Rack::Builder.app do use Grape::Middleware::Error, rescue_all: true, format: :jsonapi run ErrorHashApp end get '/' ['{"error":"rain!","detail":"missing widget"}', '{"detail":"missing widget","error":"rain!"}'].should include(last_response.body) end it 'is possible to return errors in xml format' do @app ||= Rack::Builder.app do use Grape::Middleware::Error, rescue_all: true, format: :xml run ExceptionApp end get '/' last_response.body.should == "\n\n rain!\n\n" end it 'is possible to return hash errors in xml format' do @app ||= Rack::Builder.app do use Grape::Middleware::Error, rescue_all: true, format: :xml run ErrorHashApp end get '/' ["\n\n missing widget\n rain!\n\n", "\n\n rain!\n missing widget\n\n"].should include(last_response.body) end it 'is possible to specify a custom formatter' do @app ||= Rack::Builder.app do use Grape::Middleware::Error, rescue_all: true, format: :custom, error_formatters: { custom: lambda { |message, backtrace, options, env| { custom_formatter: message }.inspect } } run ExceptionApp end get '/' last_response.body.should == '{:custom_formatter=>"rain!"}' end it 'does not trap regular error! codes' do @app ||= Rack::Builder.app do use Grape::Middleware::Error run AccessDeniedApp end get '/' last_response.status.should == 401 end it 'responds to custom Grape exceptions appropriately' do @app ||= Rack::Builder.app do use Grape::Middleware::Error, rescue_all: false run CustomErrorApp end get '/' last_response.status.should == 400 last_response.body.should == 'failed validation' end end end