From 09beb2100db10cbf4773ed0e71ddf79280a2f275 Mon Sep 17 00:00:00 2001 From: sanfrecce-osaka Date: Tue, 31 Dec 2024 12:31:25 +0900 Subject: [PATCH] activemodel: ActiveModel::Errors#add accepts String and Proc for type (#717) * refactor(activemodel): move ActiveModel::Errors#add from generated * fix(activemodel): ActiveModel::Errors#add accepts String and Proc for type The behavior of passing String or Proc for type should be allowed. FYI: It is documented in 6.1, but It is already available in 6.0. cf. https://github.com/rails/rails/commit/fcd1e41e82d3c9993f96e1763ac49a196678f931#diff-4deafad6fefcf6aa6626a1f7e3cd0ca1cf95e358f55a10a69e02be84d926bb31R368-R374 cf. https://github.com/rails/rails/commit/d9011e39357300fe78720227af4c13b4bc4ac4dd#diff-4deafad6fefcf6aa6626a1f7e3cd0ca1cf95e358f55a10a69e02be84d926bb31R370 Also, the name of the second argument was changed from `message` to `type` in 6.1. * fix(activemodel): Divide type definition of ActiveModel::Errors#add to 6.0 and 7.0 ActiveModel::Error was added on 6.1. cf. https://github.com/rails/rails/commit/ef68d3e35cb58f9f491993eeec6e7de99442dd06 On 6.1 or later, ActiveModel::Errors#add returns ActiveModel::Error. On the other hand, `ActiveModel::Errors#add` returns messages array on rails 6.0, because last evaluated is `Array#<<`. cf. https://github.com/rails/rails/blob/v6.0.0/activemodel/lib/active_model/errors.rb#L311-L322 Currently in gem_rbs_collection, activemodel type definitions are defined 6.0 and 7.0, but not 6.1. cf. https://github.com/ruby/gem_rbs_collection/pull/615 So I divide type definition of ActiveModel::Errors#add to 6.0 and 7.0. --- gems/activemodel/6.0/_test/test.rb | 9 ++++ gems/activemodel/6.0/_test/test.rbs | 1 + gems/activemodel/6.0/activemodel-6.0.rbs | 43 +++++++++++++++++ .../activemodel/6.0/activemodel-generated.rbs | 43 ----------------- gems/activemodel/6.0/activemodel.rbs | 2 +- gems/activemodel/7.0/_test/test.rb | 22 +++++++-- gems/activemodel/7.0/_test/test.rbs | 4 ++ gems/activemodel/7.0/activemodel-7.0.rbs | 47 +++++++++++++++++++ 8 files changed, 124 insertions(+), 47 deletions(-) diff --git a/gems/activemodel/6.0/_test/test.rb b/gems/activemodel/6.0/_test/test.rb index 7d0c7864..a3d3b406 100644 --- a/gems/activemodel/6.0/_test/test.rb +++ b/gems/activemodel/6.0/_test/test.rb @@ -5,8 +5,17 @@ class Person validates :name, presence: true, length: { maximum: 100 } validates :email, presence: true, if: [:foo?, -> { age >= 20 }] + validate :should_be_satisfied_special_email_rule def foo? = true + + def should_be_satisfied_special_email_rule + if Time.current >= Time.zone.local(2024, 10) + errors.add(:email, -> (_person, _options) { "must be satisfied at least 3 rules after #{Time.zone.local(2024, 10)}" }) if [/a-z/, /A-Z/, /0-9/, /[+]/].count {|rule| email.match?(rule) } > 3 + else + errors.add(:email, 'must be satisfied at least 2 rules') if [/a-z/, /A-Z/, /0-9/].count {|rule| email.match?(rule) } > 2 + end + end end person = Person.new(name: 'John Doe') diff --git a/gems/activemodel/6.0/_test/test.rbs b/gems/activemodel/6.0/_test/test.rbs index b3c14da2..7387aced 100644 --- a/gems/activemodel/6.0/_test/test.rbs +++ b/gems/activemodel/6.0/_test/test.rbs @@ -8,4 +8,5 @@ class Person attr_accessor email: String def foo?: () -> bool + def should_be_satisfied_special_email_rule: () -> void end diff --git a/gems/activemodel/6.0/activemodel-6.0.rbs b/gems/activemodel/6.0/activemodel-6.0.rbs index d62fc2df..d547b3b8 100644 --- a/gems/activemodel/6.0/activemodel-6.0.rbs +++ b/gems/activemodel/6.0/activemodel-6.0.rbs @@ -6,5 +6,48 @@ module ActiveModel # person.errors.delete(:name) # => ["cannot be nil"] # person.errors[:name] # => [] def delete: (untyped key) -> untyped + + # Adds +message+ to the error messages and used validator type to +details+ on +attribute+. + # More than one error can be added to the same +attribute+. + # If no +message+ is supplied, :invalid is assumed. + # + # person.errors.add(:name) + # # => ["is invalid"] + # person.errors.add(:name, :not_implemented, message: "must be implemented") + # # => ["is invalid", "must be implemented"] + # + # person.errors.messages + # # => {:name=>["is invalid", "must be implemented"]} + # + # person.errors.details + # # => {:name=>[{error: :not_implemented}, {error: :invalid}]} + # + # If +message+ is a symbol, it will be translated using the appropriate + # scope (see +generate_message+). + # + # If +message+ is a proc, it will be called, allowing for things like + # Time.now to be used within an error. + # + # If the :strict option is set to +true+, it will raise + # ActiveModel::StrictValidationFailed instead of adding the error. + # :strict option can also be set to any other exception. + # + # person.errors.add(:name, :invalid, strict: true) + # # => ActiveModel::StrictValidationFailed: Name is invalid + # person.errors.add(:name, :invalid, strict: NameIsInvalid) + # # => NameIsInvalid: Name is invalid + # + # person.errors.messages # => {} + # + # +attribute+ should be set to :base if the error is not + # directly associated with a single attribute. + # + # person.errors.add(:base, :name_or_email_blank, + # message: "either name or email must be present") + # person.errors.messages + # # => {:base=>["either name or email must be present"]} + # person.errors.details + # # => {:base=>[{error: :name_or_email_blank}]} + def add: (untyped attribute, ?::Symbol | ::String | ^(untyped base, ::Hash[untyped, untyped]) -> (::Symbol | ::String) type, ?::Hash[untyped, untyped] options) -> Array[String] end end diff --git a/gems/activemodel/6.0/activemodel-generated.rbs b/gems/activemodel/6.0/activemodel-generated.rbs index 0334de32..565c61ae 100644 --- a/gems/activemodel/6.0/activemodel-generated.rbs +++ b/gems/activemodel/6.0/activemodel-generated.rbs @@ -1493,49 +1493,6 @@ module ActiveModel # person.errors.to_hash(true) # => {:name=>["name cannot be nil"]} def to_hash: (?bool full_messages) -> untyped - # Adds +message+ to the error messages and used validator type to +details+ on +attribute+. - # More than one error can be added to the same +attribute+. - # If no +message+ is supplied, :invalid is assumed. - # - # person.errors.add(:name) - # # => ["is invalid"] - # person.errors.add(:name, :not_implemented, message: "must be implemented") - # # => ["is invalid", "must be implemented"] - # - # person.errors.messages - # # => {:name=>["is invalid", "must be implemented"]} - # - # person.errors.details - # # => {:name=>[{error: :not_implemented}, {error: :invalid}]} - # - # If +message+ is a symbol, it will be translated using the appropriate - # scope (see +generate_message+). - # - # If +message+ is a proc, it will be called, allowing for things like - # Time.now to be used within an error. - # - # If the :strict option is set to +true+, it will raise - # ActiveModel::StrictValidationFailed instead of adding the error. - # :strict option can also be set to any other exception. - # - # person.errors.add(:name, :invalid, strict: true) - # # => ActiveModel::StrictValidationFailed: Name is invalid - # person.errors.add(:name, :invalid, strict: NameIsInvalid) - # # => NameIsInvalid: Name is invalid - # - # person.errors.messages # => {} - # - # +attribute+ should be set to :base if the error is not - # directly associated with a single attribute. - # - # person.errors.add(:base, :name_or_email_blank, - # message: "either name or email must be present") - # person.errors.messages - # # => {:base=>["either name or email must be present"]} - # person.errors.details - # # => {:base=>[{error: :name_or_email_blank}]} - def add: (untyped attribute, ?::Symbol message, ?::Hash[untyped, untyped] options) -> untyped - # Returns +true+ if an error on the attribute with the given message is # present, or +false+ otherwise. +message+ is treated the same as for +add+. # diff --git a/gems/activemodel/6.0/activemodel.rbs b/gems/activemodel/6.0/activemodel.rbs index a5731cdc..c8b1f77e 100644 --- a/gems/activemodel/6.0/activemodel.rbs +++ b/gems/activemodel/6.0/activemodel.rbs @@ -251,7 +251,7 @@ module ActiveModel # person = Person.new # person.valid? # => false # person.errors # => # - def errors: () -> untyped + def errors: () -> Errors # Runs all the specified validations and returns +true+ if no errors were # added otherwise +false+. diff --git a/gems/activemodel/7.0/_test/test.rb b/gems/activemodel/7.0/_test/test.rb index 9e3268ba..b19b3118 100644 --- a/gems/activemodel/7.0/_test/test.rb +++ b/gems/activemodel/7.0/_test/test.rb @@ -1,16 +1,32 @@ require "active_model" class Person + include ActiveModel::Model include ActiveModel::Validations - # @dynamic name, name= - attr_accessor :name + # @dynamic name, name=, email, email= + attr_accessor :name, :email + + validate :should_be_satisfied_special_email_rule + + def should_be_satisfied_special_email_rule + return unless email + + if Time.current >= Time.zone.local(2024, 10) + errors.add(:email, -> (_person, _options) { "must be satisfied at least 3 rules after #{Time.zone.local(2024, 10)}" }) if [/a-z/, /A-Z/, /0-9/, /[+]/].count {|rule| email.match?(rule) } > 3 + else + errors.add(:email, 'must be satisfied at least 2 rules') if [/a-z/, /A-Z/, /0-9/].count {|rule| email.match?(rule) } > 2 + end + end end ActiveModel::Error::CALLBACKS_OPTIONS ActiveModel::Error::MESSAGE_OPTIONS -error = ActiveModel::Error.new(Person.new, :name, :too_short, count: 5) +person = Person.new(name: 'John Doe') +person.valid? + +error = ActiveModel::Error.new(person, :name, :too_short, count: 10) error.attribute error.type error.options diff --git a/gems/activemodel/7.0/_test/test.rbs b/gems/activemodel/7.0/_test/test.rbs index a2961a80..91fbe939 100644 --- a/gems/activemodel/7.0/_test/test.rbs +++ b/gems/activemodel/7.0/_test/test.rbs @@ -1,6 +1,10 @@ class Person + include ActiveModel::Model include ActiveModel::Validations extend ActiveModel::Validations::ClassMethods attr_accessor name: String? + attr_accessor email: String? + + def should_be_satisfied_special_email_rule: () -> void end diff --git a/gems/activemodel/7.0/activemodel-7.0.rbs b/gems/activemodel/7.0/activemodel-7.0.rbs index 19a25142..c9d82404 100644 --- a/gems/activemodel/7.0/activemodel-7.0.rbs +++ b/gems/activemodel/7.0/activemodel-7.0.rbs @@ -11,6 +11,53 @@ module ActiveModel end end +module ActiveModel + class Errors + # Adds +message+ to the error messages and used validator type to +details+ on +attribute+. + # More than one error can be added to the same +attribute+. + # If no +message+ is supplied, :invalid is assumed. + # + # person.errors.add(:name) + # # => ["is invalid"] + # person.errors.add(:name, :not_implemented, message: "must be implemented") + # # => ["is invalid", "must be implemented"] + # + # person.errors.messages + # # => {:name=>["is invalid", "must be implemented"]} + # + # person.errors.details + # # => {:name=>[{error: :not_implemented}, {error: :invalid}]} + # + # If +message+ is a symbol, it will be translated using the appropriate + # scope (see +generate_message+). + # + # If +message+ is a proc, it will be called, allowing for things like + # Time.now to be used within an error. + # + # If the :strict option is set to +true+, it will raise + # ActiveModel::StrictValidationFailed instead of adding the error. + # :strict option can also be set to any other exception. + # + # person.errors.add(:name, :invalid, strict: true) + # # => ActiveModel::StrictValidationFailed: Name is invalid + # person.errors.add(:name, :invalid, strict: NameIsInvalid) + # # => NameIsInvalid: Name is invalid + # + # person.errors.messages # => {} + # + # +attribute+ should be set to :base if the error is not + # directly associated with a single attribute. + # + # person.errors.add(:base, :name_or_email_blank, + # message: "either name or email must be present") + # person.errors.messages + # # => {:base=>["either name or email must be present"]} + # person.errors.details + # # => {:base=>[{error: :name_or_email_blank}]} + def add: (untyped attribute, ?::Symbol | ::String | ^(untyped base, ::Hash[untyped, untyped]) -> (::Symbol | ::String) type, ?::Hash[untyped, untyped] options) -> Error + end +end + module ActiveModel class Error CALLBACKS_OPTIONS: ::Array[:if | :unless | :on | :allow_nil | :allow_blank | :strict]