Dropped Ice Cream

Sarah Kilian

Use a custom validator

Active Model’s validations, used by calling validates inside your model with various options, can be supplemented with your own custom classes. The Rails Guides contains a section on custom validations.

Instead of…

…using complex code in your validates calls:

class Invite
  validates :invitee_email, format: {
    with: /\A([^@\s]+)@((?:[-a-z0-9]+\.)+[a-z]{2,})\z/i,
    message: "is not an email"
  }
end

Use…

…a validator class and extract the logic into it:

# app/validators/email_validator.rb
class EmailValidator < ActiveModel::EachValidator
  def validate_each(record, attribute, value)
    unless value =~ /\A([^@\s]+)@((?:[-a-z0-9]+\.)+[a-z]{2,})\z/i
      record.errors[attribute] << (options[:message] || "is not an email")
    end
  end
end
class Invite
  # Rails "magic" infers the name of the validation
  validates :invitee_email, email: true
end

Why?

If you have a validation you might reuse, or that contains complex formatting rules, it’s a good idea to break out that functionality into its own object.

Now you can test that logic in isolation and, at the point of use, it’s easier to work out what is going on.

Why not?

For email validation in particular, there’s a little trick I’ve used in the past, which is succinct and doesn’t need a validator, but you need to be using devise for your application’s authentication:

class Invite
  validates :invitee_email, format: {
    with: Devise.email_regexp,
    allow_blank: false
  }
end

This approach still works:

class EmailValidator < ActiveModel::EachValidator
  def validate_each(record, attribute, value)
    unless value =~ Devise.email_regexp
      record.errors[attribute] << (options[:message] || "is not an email")
    end
  end
end

Last updated on September 8th, 2019