# frozen_string_literal: true module Grape class API class Boolean def self.build(val) return nil if val != true && val != false new end end class Instance Boolean = Grape::API::Boolean end end module Validations class CoerceValidator < Base def initialize(*_args) super @converter = if type.is_a?(Grape::Validations::Types::VariantCollectionCoercer) type else Types.build_coercer(type, method: @option[:method]) end end def validate(request) super end def validate_param!(attr_name, params) raise validation_exception(attr_name) unless params.is_a? Hash new_value = coerce_value(params[attr_name]) raise validation_exception(attr_name) unless valid_type?(new_value) # Don't assign a value if it is identical. It fixes a problem with Hashie::Mash # which looses wrappers for hashes and arrays after reassigning values # # h = Hashie::Mash.new(list: [1, 2, 3, 4]) # => #> # list = h.list # h[:list] = list # h # => # return if params[attr_name].class == new_value.class && params[attr_name] == new_value params[attr_name] = new_value end private # @!attribute [r] converter # Object that will be used for parameter coercion and type checking. # # See {Types.build_coercer} # # @return [Object] attr_reader :converter def valid_type?(val) !val.is_a?(Types::InvalidValue) end def coerce_value(val) val.nil? ? coerce_nil(val) : converter.call(val) # Some custom types might fail, so it should be treated as an invalid value rescue StandardError Types::InvalidValue.new end def coerce_nil(val) # define default values for structures, the dry-types lib which is used # for coercion doesn't accept nil as a value, so it would fail return [] if type == Array return Set.new if type == Set return {} if type == Hash val end # Type to which the parameter will be coerced. # # @return [Class] def type @option[:type].is_a?(Hash) ? @option[:type][:value] : @option[:type] end def validation_exception(attr_name) Grape::Exceptions::Validation.new(params: [@scope.full_name(attr_name)], message: message(:coerce)) end end end end