Rack errors raised during parameter parsing (malformed multipart, parameter type conflicts, encoding issues, etc.) are now wrapped in Grape::Exceptions::RequestError instead of their previous specific exception classes (Grape::Exceptions::EmptyMessageBody, Grape::Exceptions::TooManyMultipartFiles, Grape::Exceptions::TooDeepParameters, Grape::Exceptions::ConflictingTypes, Grape::Exceptions::InvalidParameters). Those classes have been removed.
If you rescue any of these specific exceptions, update your rescue clauses to use Grape::Exceptions::RequestError:
# Before
rescue Grape::Exceptions::ConflictingTypes, Grape::Exceptions::TooDeepParameters => e
# ...
# After
rescue Grape::Exceptions::RequestError => e
# ...The error message is now forwarded directly from Rack rather than translated through Grape's locale system. On Rack 3, all Rack bad-request errors share the Rack::BadRequest marker module and are covered by a single rescue.
ActiveSupport::Notifications subscribers listening to endpoint_run_validators.grape will no longer receive an event for endpoints that have no validators. If you rely on this notification to measure every request, subscribe to endpoint_run.grape instead, which always fires.
Validators are now instantiated once at definition time and frozen. Any setup should happen in initialize, not in validate_param!.
If your custom validator did work in validate_param! that only depends on the validator's options (not the param value), move it to initialize. A common case is compiling a value derived from options — for example, building a Regexp. Previously this may have been cached back into @options, which now raises FrozenError since @options and its nested values are deep-frozen by the base class:
Before:
class MyValidator < Grape::Validations::Validators::Base
def validate_param!(attr_name, params)
# raises FrozenError: @options is frozen, cannot store compiled pattern back into it
@options[:compiled] ||= Regexp.new(@options[:pattern])
validation_error!(attr_name) unless params[attr_name].match?(@options[:compiled])
end
endAfter:
class MyValidator < Grape::Validations::Validators::Base
def initialize(attrs, options, required, scope, opts)
super
@pattern = Regexp.new(@options[:pattern]).freeze
end
def validate_param!(attr_name, params)
validation_error!(attr_name) unless params[attr_name].match?(@pattern)
end
endAny Array or Hash derived from options and stored in an ivar should be frozen, since the validator instance is shared across requests. @options itself (and any nested Hash/Array/String values within it) is deep-frozen by the base class, so mutations like @options[:values] << 'extra' will also raise a FrozenError.
The instance variable holding the validator's option value has been renamed from @option to @options. @option remains as an alias for backwards compatibility but will be removed in the next major release. Update any custom validators to use @options instead.
Several new helpers are available — see Available helpers in the README for full documentation and examples.
The with DSL method now uses **opts instead of a positional hash. Calls using bare keyword syntax are unaffected:
# still works
with(type: String, documentation: { in: 'body' }) { ... }However, passing an explicit hash literal will now raise an ArgumentError:
# raises ArgumentError
with({ type: String }) { ... }See #2663 for more information.
Grape::Util::Translation is now included in Grape::Validations::Validators::Base. Custom validators that previously called I18n.t or I18n.translate directly should switch to the translate, which provides the same :en fallback logic used by all built-in validators.
Key points:
scopedefaults to'grape.errors.messages'— no need to specify it for standard error message keys.- Interpolation variables are passed directly to I18n.
formatis no longer needed —translatereturns the fully interpolated string.
# Before
raise Grape::Exceptions::Validation.new(
params: [@scope.full_name(attr_name)],
message: format(I18n.t(:my_key, scope: 'grape.errors.messages'), min: 2, max: 10)
)
# After
raise Grape::Exceptions::Validation.new(
params: [@scope.full_name(attr_name)],
message: translate(:my_key, min: 2, max: 10)
)See #2662 for more information.
The API#namespace and route_param methods are now defined with **options instead of options = {}. In addtion, requirements in explicitly defined so it's not in options anymore. You can still call requirements like before but options[:requirements] will be empty. For route_param, type is also an explicit parameter so it's not in options anymore. See #2647 for more information.
Deprecated ParamsBuilder's extensions have been removed.
Endpoints are now "compiled" instead of lazy loaded. Historically, when calling YourAPI.compile! in config.ru (or just receiving the first API call), only routing was compiled see Grape::Router#compile! and endpoints were lazy loaded. Now, it's part of the API compilation. See #2645 for more information.
Grape has been modernized to use Ruby 3+'s preferred argument delegation patterns. This change replaces args.extract_options! with explicit **kwargs parameters throughout the codebase.
- All DSL methods now use explicit keyword arguments (
**kwargs) instead of extracting options from mixed argument lists - Method signatures are now more explicit and follow Ruby 3+ best practices
- The
active_support/core_ext/array/extract_optionsdependency has been removed
This is a modernization effort that improves code quality while maintaining full backward compatibility.
See #2618 for more information.
Grape has migrated from ActiveSupport::Configurable to Dry::Configurable for its configuration system since its deprecated.
See #2617 for more information.
Executing a endpoint's block has been simplified and calling return in it has been deprecated. Use next instead.
See #2577 for more information.
rack_responsehas been removed in favor of usingerror!.Grape::Exceptions::MissingGroupTypeandGrape::Exceptions::UnsupportedGroupTypealiasesMissingGroupTypeError andUnsupportedGroupType` have been removed.Grape::Validations::Basehas been removed in favor ofGrape::Validations::Validators::Base.
See 2573 for more information.
type is now validated at compile time and will raise a Grape::Exceptions::UnknownAuthStrategy if unknown.
- Second argument
optionsis now a double splat (**) instead of single splat (*). If you're redefininginitializein your middleware and/or callingsuperin it, you might have to adapt the signature and thesupercall. Also, you might have to remove{}if you're passoptionsas a literalHashor add**if you're using a variable. Grape::Middleware::Helpershas been removed. The equivalent methodcontextis now part ofGrape::Middleware::Base.
Both have been removed. See 2554. Here are the notable changes:
- Constants like
HTTP_ACCEPThave been replaced by their literal value. SUPPORTED_METHODShas been moved toGrapemodule.HTTP_HEADERShas been moved toGrape::Requestand renamedKNOWN_HEADERS. The last has been refreshed with new headers, and it's not lazy anymore.SUPPORTED_METHODS_WITHOUT_OPTIONSandfind_supported_methodhave been removed.
- Constant
TEXT_HTMLhas been removed in favor of using literal string 'text/html'. rack_requestandquery_paramshave been added. Feel free to call these in your middlewares.
- Passing a class to
build_withorGrape.config.param_builderhas been deprecated in favor of a symbolized short_name. SeeSHORTNAME_LOOKUPin params_builder. - Including Grape's extensions like
Grape::Extensions::Hashie::Mash::ParamBuilderhas been deprecated in favor of usingbuild_withat the route level.
Accept header is now fully interpreted through Rack::Utils.best_q_match which is following RFC2616 14.1. Since Grape 2.1.0, the header versioning strategy was adhering to it, but Grape::Middleware::Formatter never did.
Your API might act differently since it will strictly follow the RFC2616 14.1 when interpreting the Accept header. Here are the differences:
The following used to yield application/xml and now will yield application/json as the preferred media type:
application/json;q=invalid,application/xml;q=0.5application/json,application/xml;q=1.0
For the invalid case, the value invalid was automatically to_f and invalid.to_f equals 0.0. Now, since it doesn't match Rack's regex, its interpreted as non provided and its quality ranking equals 1.0.
For the non provided case, 1.0 was automatically assigned and in a case of multiple best matches, the first was returned based on Ruby's sort_by quality. Now, 1.0 is still assigned and the last is returned in case of multiple best matches. See Rack's implementation of the RFC.
Excluding the header versioning strategy, whenever a media type with the vendor tree leading facet vnd. like application/vnd.api+json was provided, Grape would also consider its closest generic when negotiating. In that case, application/json was added to the negotiation. Now, it will just consider the provided media types without considering any closest generics, and you'll need to register it.
You can find the official vendor tree registrations on IANA
If you now receive an error of 'Grape::Validations.require_validator': unknown validator: your_custom_validation (Grape::Exceptions::UnknownValidator) after upgrading to 2.4.0 then you will need to ensure that you require the your_custom_validation file before your Grape API code is loaded.
See 2533 for more information.
Before 2.3.0, content_type had priority over env['api.format'] when set in an API, which was incorrect. The priority has been flipped and env['api.format'] will be checked first.
In addition, the function api_format has been added. Instead of setting env['api.format'] directly, you can call api_format.
See #2506 for more information.
-
Deprecated
filemethod has been removed. Usesend_fileorstream. See #2500 for more information. -
The
exceptandprocoptions have been removed from thevaluesvalidator. Useexcept_valuesvalidator or assignprocdirectly tovalues. See #2501 for more information. -
Passing an options hash and a block to 'desc'deprecation has been removed. Move all hash options to block instead. See #2502 for more information.
After Grape 2.2.0, length validator will only take effect for parameters with types that support #length method, will not throw ArgumentError exception.
See #2464 for more information.
The builder gem dependency has been made optional as it's only used when generating XML. If your code does, add builder to your Gemfile.
See #2445 for more information.
Grape now uses deep_merge to combine parameter attributes within the with method. Previously, attributes defined at the parameter level would override those defined at the group level.
With deep merge, attributes are now combined, allowing for more detailed and nuanced API specifications.
For example:
with(documentation: { in: 'body' }) do
optional :vault, documentation: { default: 33 }
endBefore it was equivalent to:
optional :vault, documentation: { default: 33 }After it is an equivalent of:
optional :vault, documentation: { in: 'body', default: 33 }See #2432 for more information.
Grape's autoloader has been updated and it's now based on Zeitwerk. If you MP (Monkey Patch) some files and you're not following the file structure, you might end up with a Zeitwerk error.
See #2363 for more information.
The rack_response method has been deprecated and the error_response method has been removed. Use error! instead.
See #2414 for more information.
When using together with Grape::Extensions::Hash::ParamBuilder, route_param takes higher precedence over a regular parameter defined with same name, which now matches the default param builder behavior.
This was a regression introduced by #2326 in Grape v1.8.0.
Grape.configure do |config|
config.param_builder = Grape::Extensions::Hash::ParamBuilder
end
params do
requires :foo, type: String
end
route_param :foo do
get do
{ value: params[:foo] }
end
endRequest:
curl -X POST -H "Content-Type: application/json" localhost:9292/bar -d '{"foo": "baz"}'Response prior to v1.8.0:
{
"value": "bar"
}v1.8.0..v2.0.0:
{
"value": "baz"
}v2.1.0+:
{
"value": "bar"
}See #2378 for details.
route_methodis accessible throughrequest_methodroute_pathis accessible throughpath- Any other
route_xyzare accessible throughoptions[xyz]
Due to the changes done in #2377, the instance variables defined inside each of the endpoints (or inside a before validator) are now accessible inside the rescue_from. The behavior of the instance variables was undefined until 2.1.0.
If you were using the same variable name defined inside an endpoint or before validator inside a rescue_from handler, you need to take in mind that you can start getting different values or you can be overriding values.
Before:
class TwitterAPI < Grape::API
before do
@var = 1
end
get '/' do
puts @var # => 1
raise
end
rescue_from :all do
puts @var # => nil
end
endAfter:
class TwitterAPI < Grape::API
before do
@var = 1
end
get '/' do
puts @var # => 1
raise
end
rescue_from :all do
puts @var # => 1
end
endGrape now considers the types of the configured route_params in order to determine the endpoint that matches with the performed request.
So taking into account this Grape::API class
class Books < Grape::API
resource :books do
route_param :id, type: Integer do
# GET /books/:id
get do
#...
end
end
resource :share do
# POST /books/share
post do
# ....
end
end
end
endBefore:
API.recognize_path '/books/1' # => /books/:id
API.recognize_path '/books/share' # => /books/:id
API.recognize_path '/books/other' # => /books/:idAfter:
API.recognize_path '/books/1' # => /books/:id
API.recognize_path '/books/share' # => /books/share
API.recognize_path '/books/other' # => nilThis implies that before this changes, when you performed /books/other and it matched with the /books/:id endpoint, you get a 400 Bad Request response because the type of the provided :id param was not an Integer. However, after upgrading to version 2.1.0 you will get a 404 Not Found response, because there is not a defined endpoint that matches with /books/other.
See #2379 for more information.
As per rack/rack#1592 Rack 3 is following the HTTP/2+ semantics which require header names to be lower case. To avoid compatibility issues, starting with Grape 1.9.0, headers will be cased based on what version of Rack you are using.
Given this request:
curl -H "Content-Type: application/json" -H "Secret-Password: foo" ...If you are using Rack 3 in your application then the headers will be set to:
{ "content-type" => "application/json", "secret-password" => "foo"}This means if you are checking for header values in your application, you would need to change your code to use downcased keys.
get do
# This would use headers['Secret-Password'] in Rack < 3
error!('Unauthorized', 401) unless headers['secret-password'] == 'swordfish'
endSee #2355 for more information.
Digest auth has been removed along with the deprecation of Rack::Auth::Digest in Rack 3.
See #2294 for more information.
The following exceptions has been renamed for consistency through exceptions naming :
MissingGroupTypeError=>MissingGroupTypeUnsupportedGroupTypeError=>UnsupportedGroupType
See #2227 for more information.
Rack supports a configurable limit on the number of files created from multipart parameters (Rack::Utils.multipart_part_limit) and raises an error if params are received that create too many files. If you were handling the Rack error directly, Grape now wraps that error in Grape::Execeptions::TooManyMultipartFiles. Additionally, Grape will return a 413 status code if the exception goes unhandled.
Prior to 1.6.0 the parameter renaming with :as was directly touching the request payload (#params) while duplicating the old and the new key to be both available in the hash. This allowed clients to bypass any validation in case they knew the internal name of the parameter. Unfortunately, in combination with grape-swagger the internal name (name set with :as) of the parameters were documented.
This behavior was fixed. Parameter renaming is now done when using the #declared(params) parameters helper. This stops confusing validation/coercion behavior.
Here comes an illustration of the old and new behaviour as code:
# (1) Rename a to b, while client sends +a+
optional :a, type: Integer, as: :b
params = { a: 1 }
declared(params, include_missing: false)
# expected => { b: 1 }
# actual => { b: 1 }
# (2) Rename a to b, while client sends +b+
optional :a, type: Integer, as: :b, values: [1, 2, 3]
params = { b: '5' }
declared(params, include_missing: false)
# expected => { } (>= 1.6.0)
# actual => { b: '5' } (uncasted, unvalidated, <= 1.5.3)Another implication of this change is the dependent parameter resolution. Prior to 1.6.0 the following code produced a Grape::Exceptions::UnknownParameter because :a was replaced by :b:
params do
optional :a, as: :b
given :a do # (<= 1.5.3 you had to reference +:b+ here to make it work)
requires :c
end
endThis code now works without any errors, as the renaming is just an internal behaviour of the #declared(params) parameter helper.
See #2189 for more information.
Prior to 1.2.5 version passing a nil value for a parameter with a custom coercer would invoke the coercer, and not passing a parameter would not invoke it.
This behavior was not tested or documented. Version 1.3.0 quietly changed this behavior, in that nil values skipped the coercion. Version 1.5.3 fixes and documents this as follows:
class Api < Grape::API
params do
optional :value, type: Integer, coerce_with: ->(val) { val || 0 }
end
get 'example' do
params[:my_param]
end
get '/example', params: { value: nil }
# 1.5.2 = nil
# 1.5.3 = 0
get '/example', params: {}
# 1.5.2 = nil
# 1.5.3 = nil
endSee #2164 for more information.
If you use dependent params with
Grape::Extensions::Hash::ParamBuilder, make sure a parameter to be dependent on is set as a Symbol.
If a String is given, a parameter that other parameters depend on won't be found even if it is present.
Correct:
given :matrix do
# dependent params
endWrong:
given 'matrix' do
# dependent params
endPrior to 1.3.3, the declared helper would always return the complete params structure if include_missing=true was set. In 1.3.3 a regression was introduced such that a missing Hash with or without nested parameters would always resolve to {}.
In 1.5.0 this behavior is reverted, so the whole params structure will always be available via declared, regardless of whether any params are passed.
The following rules now apply to the declared helper when params are missing and include_missing=true:
- Hash params with children will resolve to a Hash with keys for each declared child.
- Hash params with no children will resolve to
{}. - Set params will resolve to
Set.new. - Array params will resolve to
[]. - All other params will resolve to
nil.
class Api < Grape::API
params do
optional :outer, type: Hash do
optional :inner, type: Hash do
optional :value, type: String
end
end
end
get 'example' do
declared(params, include_missing: true)
end
endget '/example'
# 1.3.3 = {}
# 1.5.0 = {outer: {inner: {value:null}}}
For more information see #2103.
Previously in 0.16 stream-like objects were deprecated. This release restores their functionality for use-cases other than file streaming.
This release deprecated file in favor of sendfile to better document its purpose.
To deliver a file via the Sendfile support in your web server and have the Rack::Sendfile middleware enabled. See Rack::Sendfile.
class API < Grape::API
get '/' do
sendfile '/path/to/file'
end
endUse stream to stream file content in chunks.
class API < Grape::API
get '/' do
stream '/path/to/file'
end
endOr use stream to stream other kinds of content. In the following example a streamer class
streams paginated data from a database.
class MyObject
attr_accessor :result
def initialize(query)
@result = query
end
def each
yield '['
# Do paginated DB fetches and return each page formatted
first = false
result.find_in_batches do |records|
yield process_records(records, first)
first = false
end
yield ']'
end
def process_records(records, first)
buffer = +''
buffer << ',' unless first
buffer << records.map(&:to_json).join(',')
buffer
end
end
class API < Grape::API
get '/' do
stream MyObject.new(Sprocket.all)
end
endNil values have always been a special case when dealing with types, especially with the following structures:
- Array
- Hash
- Set
The behavior for these structures has changed throughout the latest releases. For example:
class Api < Grape::API
params do
require :my_param, type: Array[Integer]
end
get 'example' do
params[:my_param]
end
get '/example', params: { my_param: nil }
# 1.3.1 = []
# 1.3.2 = nil
endFor now on, nil values stay nil values for all types, including arrays, sets and hashes.
If you want to have the same behavior as 1.3.1, apply a default validator:
class Api < Grape::API
params do
require :my_param, type: Array[Integer], default: []
end
get 'example' do
params[:my_param]
end
get '/example', params: { my_param: nil } # => []
endDefault validator is now applied for nil values.
class Api < Grape::API
params do
requires :my_param, type: Integer, default: 0
end
get 'example' do
params[:my_param]
end
get '/example', params: { my_param: nil } #=> before: nil, after: 0
endYou will need to upgrade to this version if you depend on rack >= 2.1.0.
After adding dry-types, Ruby 2.4 or newer is required.
Virtus has been replaced by dry-types for parameter coercion. If your project depends on Virtus outside of Grape, explicitly add it to your Gemfile.
Here's an example of how to migrate a custom type from Virtus to dry-types:
# Legacy Grape parser
class SecureUriType < Virtus::Attribute
def coerce(input)
URI.parse value
end
def value_coerced?(input)
value.is_a? String
end
end
params do
requires :secure_uri, type: SecureUri
endTo use dry-types, we need to:
- Remove the inheritance of
Virtus::Attribute - Rename
coercetoself.parse - Rename
value_coerced?toself.parsed?
The custom type must have a class-level parse method to the model. A class-level parsed? is needed if the parsed type differs from the defined type. In the example below, since SecureUri is not the same as URI::HTTPS, self.parsed? is needed:
# New dry-types parser
class SecureUri
def self.parse(value)
URI.parse value
end
def self.parsed?(value)
value.is_a? URI::HTTPS
end
end
params do
requires :secure_uri, type: SecureUri
endPrevious Grape versions allowed this, though it wasn't documented:
requires :true_value, type: TrueClass
requires :bool_value, types: [FalseClass, TrueClass]This is no longer supported, if you do this, your values will never be valid. Instead you should do this:
requires :true_value, type: Boolean # in your endpoint you should validate if this is actually `true`
requires :bool_value, type: BooleanUnlike Virtus, dry-types does not perform any implict coercions. If you have any uses of Array[String], Array[Integer], etc. be sure they use a coerce_with block. For example:
requires :values, type: Array[String]It's quite common to pass a comma-separated list, such as tag1,tag2 as values. Previously Virtus would implicitly coerce this to Array(values) so that ["tag1,tag2"] would pass the type checks, but with dry-types the values are no longer coerced for you. To fix this, you might do:
requires :values, type: Array[String], coerce_with: ->(val) { val.split(',').map(&:strip) }Likewise, for Array[Integer], you might do:
requires :values, type: Array[Integer], coerce_with: ->(val) { val.split(',').map(&:strip).map(&:to_i) }For more information see #1920.
Headers in error! will be merged with headers hash. If any header need to be cleared on error! call, make sure to move it to the after block.
class SampleApi < Grape::API
before do
header 'X-Before-Header', 'before_call'
end
get 'ping' do
header 'X-App-Header', 'on_call'
error! :pong, 400, 'X-Error-Details' => 'Invalid token'
end
endFormer behaviour
response.headers['X-Before-Header'] # => nil
response.headers['X-App-Header'] # => nil
response.headers['X-Error-Details'] # => Invalid tokenCurrent behaviour
response.headers['X-Before-Header'] # => 'before_call'
response.headers['X-App-Header'] # => 'on_call'
response.headers['X-Error-Details'] # => Invalid tokenIn order to make obtaining the name of a mounted class simpler, we've delegated .to_s to base.name
Deprecated in 1.2.0
payload[:endpoint].options[:for].nameNew
payload[:endpoint].options[:for].to_sIn an effort to make APIs re-mountable, The class Grape::API no longer refers to an API instance, rather, what used to be Grape::API is now Grape::API::Instance and Grape::API was replaced with a class that can contain several instances of Grape::API.
This changes were done in such a way that no code-changes should be required. However, if experiencing problems, or relying on private methods and internal behaviour too deeply, it is possible to restore the prior behaviour by replacing the references from Grape::API to Grape::API::Instance.
Note, this is particularly relevant if you are opening the class Grape::API for modification.
Deprecated
class Grape::API
# your patched logic
...
endNew
class Grape::API::Instance
# your patched logic
...
endAfter the patch, the mounted API is no longer a Named class inheriting from Grape::API, it is an anonymous class which inherit from Grape::API::Instance.
What this means in practice, is:
- Generally: you can access the named class from the instance calling the getter
base. - In particular: If you need the
name, you can usebase.name.
Deprecated
payload[:endpoint].options[:for].nameNew
payload[:endpoint].options[:for].base.nameGrape will now check the object returned from rescue_from and ensure that it is a Rack::Response. That makes sure response is valid and avoids exposing service information. Change any code that invoked Rack::Response.new(...).finish in a custom rescue_from block to Rack::Response.new(...) to comply with the validation.
class Twitter::API < Grape::API
rescue_from :all do |e|
# version prior to 1.2.0
Rack::Response.new([ e.message ], 500, { 'Content-type' => 'text/error' }).finish
# 1.2.0 version
Rack::Response.new([ e.message ], 500, { 'Content-type' => 'text/error' })
end
endSee #1757 and #1776 for more information.
For PUT, POST, PATCH, and DELETE requests where a non-empty body and a "Content-Type" header is supplied that is not supported by the Grape API, Grape will no longer return a 406 "Not Acceptable" HTTP status code and will instead return a 415 "Unsupported Media Type" so that the usage of HTTP status code falls more in line with the specification of RFC 2616.
Grape no longer uses multi_json or multi_xml by default and uses JSON and ActiveSupport::XmlMini instead. This has no visible impact on JSON processing, but the default behavior of the XML parser has changed. For example, an XML POST containing <user>Bobby T.</user> was parsed as Bobby T. with multi_xml, and as now parsed as {"__content__"=>"Bobby T."} with XmlMini.
If you were using MultiJson.load, MultiJson.dump or MultiXml.parse, you can substitute those with Grape::Json.load, Grape::Json.dump, ::Grape::Xml.parse, or directly with JSON.load, JSON.dump, XmlMini.parse, etc.
To restore previous behavior, add multi_json or multi_xml to your Gemfile and require it.
See #1623 for more information.
The default class for params has changed from Hashie::Mash to ActiveSupport::HashWithIndifferentAccess and the hashie dependency has been removed. This means that by default you can no longer access parameters by method name.
class API < Grape::API
params do
optional :color, type: String
end
get do
params[:color] # use params[:color] instead of params.color
end
endTo restore the behavior of prior versions, add hashie to your Gemfile and include Grape::Extensions::Hashie::Mash::ParamBuilder in your API.
class API < Grape::API
include Grape::Extensions::Hashie::Mash::ParamBuilder
params do
optional :color, type: String
end
get do
# params.color works
end
endThis behavior can also be overridden on individual parameter blocks using build_with.
params do
build_with Grape::Extensions::Hash::ParamBuilder
optional :color, type: String
endIf you're constructing your own Grape::Request in a middleware, you can pass different parameter handlers to create the desired params class with build_params_with.
def request
Grape::Request.new(env, build_params_with: Grape::Extensions::Hashie::Mash::ParamBuilder)
endSee #1610 for more information.
The new except_values validator should be used in place of the except and except_message options of the values validator.
Arity one Procs may now be used directly as the values option to explicitly test param values.
Deprecated
params do
requires :a, values: { value: 0..99, except: [3] }
requires :b, values: { value: 0..99, except: [3], except_message: 'not allowed' }
requires :c, values: { except: ['admin'] }
requires :d, values: { proc: -> (v) { v.even? } }
endNew
params do
requires :a, values: 0..99, except_values: [3]
requires :b, values: 0..99, except_values: { value: [3], message: 'not allowed' }
requires :c, except_values: ['admin']
requires :d, values: -> (v) { v.even? }
endSee #1616 for more information.
Prior to this version, DELETE requests defaulted to a status code of 204 No Content, even when the response included content. This behavior confused some clients and prevented the formatter middleware from running properly. As of this version, DELETE requests will only default to a 204 No Content status code if no response body is provided, and will default to 200 OK otherwise.
Specifically, DELETE behaviour has changed as follows:
- In versions < 0.19.0, all DELETE requests defaulted to a 200 OK status code.
- In version 0.19.0, all DELETE requests defaulted to a 204 No Content status code, even when content was included in the response.
- As of version 0.19.1, DELETE requests default to a 204 No Content status code, unless content is supplied, in which case they default to a 200 OK status code.
To achieve the old behavior, one can specify the status code explicitly:
delete :id do
status 204 # or 200, for < 0.19.0 behavior
'foo successfully deleted'
endOne can also use the new return_no_content helper to explicitly return a 204 status code and an empty body for any request type:
delete :id do
return_no_content
'this will not be returned'
endSee #1550 for more information.
Prior to this version, :any routes were searched after matching first route and 405 routes. This behavior has changed and :any routes are now searched before 405 processing. In the following example the :any route will match first when making a request with an unsupported verb.
post :example do
'example'
end
route :any, '*path' do
error! :not_found, 404
end
get '/example' #=> before: 405, after: 404When a request is made to the built-in OPTIONS handler, only the before and after callbacks associated with the resource will be run. The before_validation and after_validation callbacks and parameter validations will be skipped.
See #1505 for more information.
Grape now correctly returns validation errors for all params when multiple params are passed to a requires.
The following code will return one is missing, two is missing when calling the endpoint without parameters.
params do
requires :one, :two
endPrior to this version the response would be one is missing.
See #1510 for more information.
Breaking change: Sets the default response status code for a delete request to 204. A status of 204 makes the response more distinguishable and therefore easier to handle on the client side, particularly because a DELETE request typically returns an empty body as the resource was deleted or voided.
To achieve the old behavior, one has to set it explicitly:
delete :id do
status 200
'foo successfully deleted'
endFor more information see: #1532.
Grape is no longer automatically tested against versions of Ruby prior to 2.2.2. This is because of its dependency on activesupport which, with version 5.0.0, now requires at least Ruby 2.2.2.
See #1441 for nmore information.
The rescue_from clauses declared inside a namespace would take a priority over ones declared in the root scope.
This could possibly affect those users who use different rescue_from clauses in root scope and in namespaces.
See #1405 for more information.
Helper methods are injected inside rescue_from may cause undesirable effects. For example, definining a helper method called error! will take precendence over the built-in error! method and should be renamed.
See #1451 for an example.
The Route#route_xyz methods have been deprecated since 0.15.1.
Please use Route#xyz instead.
Note that the Route#route_method was replaced by Route#request_method.
The following code would work correctly.
TwitterAPI::versions # yields [ 'v1', 'v2' ]
TwitterAPI::routes # yields an array of Grape::Route objects
TwitterAPI::routes[0].version # => 'v1'
TwitterAPI::routes[0].description # => 'Includes custom settings.'
TwitterAPI::routes[0].settings[:custom] # => { key: 'value' }
TwitterAPI::routes[0].request_method # => 'GET'Now to serve files via Grape just pass the path to the file. Functionality with FileStreamer-like objects is deprecated.
Please, replace your FileStreamer-like objects with paths of served files.
Old style:
class FileStreamer
def initialize(file_path)
@file_path = file_path
end
def each(&blk)
File.open(@file_path, 'rb') do |file|
file.each(10, &blk)
end
end
end
# ...
class API < Grape::API
get '/' do
file FileStreamer.new('/path/to/file')
end
endNew style:
class API < Grape::API
get '/' do
file '/path/to/file'
end
endThe :with option of rescue_from does not accept value except Proc, String or Symbol now.
If you have been depending the old behavior, you should use lambda block instead.
class API < Grape::API
rescue_from :all, with: -> { Rack::Response.new('rescued with a method', 400) }
endThe after method of the middleware is now also called on error. The following code would work correctly.
class ErrorMiddleware < Grape::Middleware::Base
def after
return unless @app_response && @app_response[0] == 500
env['rack.logger'].debug("Raised error on #{env['PATH_INFO']}")
end
endSee #1147 and #1240 for discussion of the issues.
A warning will be logged if an exception is raised in an after callback, which points you to middleware that was not called in the previous version and is called now.
caught error of type NoMethodError in after callback inside Api::Middleware::SomeMiddleware : undefined method `headers' for nil:NilClass
See #1285 for more information.
A 405 Method Not Allowed error now causes Grape::Exceptions::MethodNotAllowed to be raised, which will be rescued via rescue_from :all. Restore old behavior with the following error handler.
rescue_from Grape::Exceptions::MethodNotAllowed do |e|
error! e.message, e.status, e.headers
endSee #1283 for more information.
When raising Grape::Exceptions::Validation explicitly, replace message_key with message.
For example,
fail Grape::Exceptions::Validation, params: [:oauth_token_secret], message_key: :presencebecomes
fail Grape::Exceptions::Validation, params: [:oauth_token_secret], message: :presenceSee #1295 for more information.
The #declared method of the route DSL is no longer available in the before filter. Using declared in a before filter will now raise Grape::DSL::InsideRoute::MethodNotYetAvailable.
See #1074 for discussion of the issue.
Identical endpoints with different versions now work correctly. A regression introduced in Grape 0.11.0 caused all but the first-mounted version for such an endpoint to wrongly throw an InvalidAcceptHeader. As a side effect, requests with a correct vendor but invalid version can no longer be rescued from a rescue_from block.
See #1114 for more information.
To be consistent with rack and it's handling of standard responses associated with no content, both default and custom formatters will now be bypassed when processing responses for status codes defined by rack
See #1190 for more information.
#redirect now uses text/plain regardless of whether that format has been enabled. This prevents formatters from attempting to serialize the message body and allows for a descriptive message body to be provided - and optionally overridden - that better fulfills the theme of the HTTP spec.
See #1194 for more information.
The Rack response object is no longer converted to an array by the formatter, enabling streaming. If your custom middleware is accessing @app_response, update it to expect a Rack::Response instance instead of an array.
For example,
class CacheBusterMiddleware < Grape::Middleware::Base
def after
@app_response[1]['Expires'] = Time.at(0).utc.to_s
@app_response
end
endbecomes
class CacheBusterMiddleware < Grape::Middleware::Base
def after
@app_response.headers['Expires'] = Time.at(0).utc.to_s
@app_response
end
endSee #1029 for more information.
There is a known issue because of this change. When Grape is used with an older than 1.2.4 version of warden there may be raised the following exception having the rack-mount gem's lines as last ones in the backtrace:
NoMethodError: undefined method `[]' for nil:NilClass
The issue can be solved by upgrading warden to 1.2.4 version.
See #1151 for more information.
Using present with objects that responded to merge would cause early evaluation of the represented object, with unexpected side-effects, such as missing parameters or environment within rendering code. Grape now only merges represented objects with a previously rendered body, usually when multiple present calls are made in the same route.
See grape-with-roar#5 and #1023.
Parameters with nil value will now pass regexp validation. To disallow nil value for an endpoint, add allow_blank: false.
params do
requires :email, allow_blank: false, regexp: /.+@.+/
endSee #957 for more information.
Note: error_response is being deprecated, not removed.
def error!(message, status = options[:default_status], headers = {}, backtrace = [])
headers = { 'Content-Type' => content_type }.merge(headers)
rack_response(format_message(message, backtrace), status, headers)
endFor example,
error_response({ message: { message: 'No such page.', id: 'missing_page' }, status: 404, headers: { 'Content-Type' => 'api/error' })
becomes
error!({ message: 'No such page.', id: 'missing_page' }, 404, { 'Content-Type' => 'api/error' })
error! also supports just passing a message. error!('Server error.') and format: :json returns the following JSON response
{ 'error': 'Server error.' }
with a status code of 500 and a Content Type of text/error.
Optionally, also replace Rack::Response.new with error!.
The following are equivalent:
Rack::Response.new([ e.message ], 500, { "Content-type" => "text/error" }).finish
error!(e)
See #889 for more information.
Version 0.10.0 has introduced a change via #809 whereas routes no longer got file-type suffixes added if you declared a single API format. This has been reverted, it's now again possible to call API with proper suffix when single format is defined:
class API < Grape::API
format :json
get :hello do
{ hello: 'world' }
end
endWill respond with JSON to /hello and /hello.json.
Will respond with 404 to /hello.xml, /hello.txt etc.
See the #1001 and #914 for more info.
Grape now supports, but doesn't require Rack 1.6.0. If you encounter an issue with parsing requests larger than 128KB, explictly require Rack 1.6.0 in your Gemfile.
gem 'rack', '~> 1.6.0'See #559 for more information.
Key route_info is excluded from params.
See #879 for more information.
Callbacks defined in a version block are only called for the routes defined in that block. This was a regression introduced in Grape 0.10.0, and is fixed in this version.
See #901 for more information.
Groups of parameters now require their type to be set explicitly as Array or Hash. Not setting the type now results in MissingGroupTypeError, unsupported type will raise UnsupportedTypeError.
See #886 for more information.
Attributes with nil values or with values that evaluate to false are no longer considered missing and will be returned when include_missing is set to false.
See #864 for more information.
The following content-types have been removed:
- atom (application/atom+xml)
- rss (application/rss+xml)
- jsonapi (application/jsonapi)
This is because they have never been properly supported.
New block syntax:
Former:
desc "some descs",
detail: 'more details',
entity: API::Entities::Entity,
params: API::Entities::Status.documentation,
named: 'a name',
headers: [XAuthToken: {
description: 'Valdates your identity',
required: true
}
get nil, http_codes: [
[401, 'Unauthorized', API::Entities::BaseError],
[404, 'not found', API::Entities::Error]
] doNow:
desc "some descs" do
detail 'more details'
params API::Entities::Status.documentation
success API::Entities::Entity
failure [
[401, 'Unauthorized', API::Entities::BaseError],
[404, 'not found', API::Entities::Error]
]
named 'a name'
headers [
XAuthToken: {
description: 'Valdates your identity',
required: true
},
XOptionalHeader: {
description: 'Not really needed',
required: false
}
]
endA common hack to extend Grape with custom DSL methods was manipulating @last_description.
module Grape
module Extensions
module SortExtension
def sort(value)
@last_description ||= {}
@last_description[:sort] ||= {}
@last_description[:sort].merge! value
value
end
end
Grape::API.extend self
end
endYou could access this value from within the API with route.route_sort or, more generally, via env['api.endpoint'].options[:route_options][:sort].
This will no longer work, use the documented and supported route_setting.
module Grape
module Extensions
module SortExtension
def sort(value)
route_setting :sort, sort: value
value
end
end
Grape::API.extend self
end
endTo retrieve this value at runtime from within an API, use env['api.endpoint'].route_setting(:sort) and when introspecting a mounted API, use route.route_settings[:sort].
It used to be possible to fetch an API class variable from a helper function. For example:
@@static_variable = 42
helpers do
def get_static_variable
@@static_variable
end
end
get do
get_static_variable
endThis will no longer work. Use a class method instead of a helper.
@@static_variable = 42
def self.get_static_variable
@@static_variable
end
get do
get_static_variable
endFor more information see #836.
To implement a custom validator, you need to inherit from Grape::Validations::Base instead of Grape::Validations::Validator.
For more information see Custom Validators in the documentation.
In previous versions raising Grape::Exceptions::Validation required a single param.
raise Grape::Exceptions::Validation, param: :id, message_key: :presenceThe param argument has been deprecated and is now an array of params, accepting multiple values.
raise Grape::Exceptions::Validation, params: [:id], message_key: :presenceRoutes will no longer get file-type suffixes added if you declare a single API format. For example,
class API < Grape::API
format :json
get :hello do
{ hello: 'world' }
end
endPre-0.10.0, this would respond with JSON to /hello, /hello.json, /hello.xml, /hello.txt, etc.
Now, this will only respond with JSON to /hello, but will be a 404 when trying to access /hello.json, /hello.xml, /hello.txt, etc.
If you declare further content_types, this behavior will be circumvented. For example, the following API will respond with JSON to /hello, /hello.json, /hello.xml, /hello.txt, etc.
class API < Grape::API
format :json
content_type :json, 'application/json'
get :hello do
{ hello: 'world' }
end
endSee the the updated API Formats documentation and #809 for more info.
Permitted and default parameter values are now only evaluated lazily for each request when declared as a proc. The following code would raise an error at startup time.
params do
optional :v, values: -> { [:x, :y] }, default: -> { :z }
endRemove the proc to get the previous behavior.
params do
optional :v, values: [:x, :y], default: :z
endSee #801 for more information.
If version is used with a block, the callbacks defined within that version block are not scoped to that individual block. In other words, the callback would be inherited by all versions blocks that follow the first one e.g
class API < Grape::API
resource :foo do
version 'v1', :using => :path do
before do
@output ||= 'hello1'
end
get '/' do
@output += '-v1'
end
end
version 'v2', :using => :path do
before do
@output ||= 'hello2'
end
get '/:id' do
@output += '-v2'
end
end
end
endwhen making a API call GET /foo/v2/1, the API would set instance variable @output to hello1-v2
See #898 for more information.
The following middleware classes have been removed:
Grape::Middleware::Auth::BasicGrape::Middleware::Auth::DigestGrape::Middleware::Auth::OAuth2
When you use theses classes directly like:
module API
class Root < Grape::API
class Protected < Grape::API
use Grape::Middleware::Auth::OAuth2,
token_class: 'AccessToken',
parameter: %w(access_token api_key)you have to replace these classes.
As replacement can be used
Grape::Middleware::Auth::Basic=>Rack::Auth::BasicGrape::Middleware::Auth::Digest=>Rack::Auth::Digest::MD5Grape::Middleware::Auth::OAuth2=> warden-oauth2 or rack-oauth2
If this is not possible you can extract the middleware files from grape v0.7.0 and host these files within your application
See #703 for more information.
Assume you have the following exception classes defined.
class ParentError < StandardError; end
class ChildError < ParentError; endIn Grape <= 0.6.1, the rescue_from keyword only handled the exact exception being raised. The following code would rescue ParentError, but not ChildError.
rescue_from ParentError do |e|
# only rescue ParentError
endThis made it impossible to rescue an exception hieararchy, which is a more sensible default. In Grape 0.7.0 or newer, both ParentError and ChildError are rescued.
rescue_from ParentError do |e|
# rescue both ParentError and ChildError
endTo only rescue the base exception class, set rescue_subclasses: false.
rescue_from ParentError, rescue_subclasses: false do |e|
# only rescue ParentError
endSee #544 for more information.
In Grape <= 0.6.1, the default status code returned from error! was 403.
error! "You may not reticulate this spline!" # yields HTTP error 403This was a bad default value, since 403 means "Forbidden". Change any call to error! that does not specify a status code to specify one. The new default value is a more sensible default of 500, which is "Internal Server Error".
error! "You may not reticulate this spline!", 403 # yields HTTP error 403You may also use default_error_status to change the global default.
default_error_status 400See #525 for more information.
In Grape <= 0.6.1, group, optional and requires keywords with a block accepted either an Array or a Hash.
params do
requires :id, type: Integer
group :name do
requires :first_name
requires :last_name
end
endThis caused the ambiguity and unexpected errors described in #543.
In Grape 0.7.0, the group, optional and requires keywords take an additional type attribute which defaults to Array. This means that without a type attribute, these nested parameters will no longer accept a single hash, only an array (of hashes).
Whereas in 0.6.1 the API above accepted the following json, it no longer does in 0.7.0.
{
"id": 1,
"name": {
"first_name": "John",
"last_name" : "Doe"
}
}The params block should now read as follows.
params do
requires :id, type: Integer
requires :name, type: Hash do
requires :first_name
requires :last_name
end
endSee #545 for more information.
In Grape <= 0.5.0, only the first validation error was raised and processing aborted. Validation errors are now collected and a single Grape::Exceptions::ValidationErrors exception is raised. You can access the collection of validation errors as .errors.
rescue_from Grape::Exceptions::Validations do |e|
Rack::Response.new({
status: 422,
message: e.message,
errors: e.errors
}.to_json, 422)
endFor more information see #462.