Working with schemas

A schema is an object which contains a list of rules that will be applied to its input when you call a schema. It returns a result object which provides an API to retrieve error messages and access to the validation output.

Schema definition best practices:

  • Be specific about the exact shape of the data, define all the keys that you expect to be present
  • Specify optional keys too, even if you don't need additional rules to be applied to their values
  • Specify type specs for all the values
  • Assign schema objects to constants for convenient access
  • Define a base schema for your application with common configuration

Calling a schema

Calling a schema will apply all its rules to the input. High-level rules defined with the rule API are applied in a second step and they are guarded, which means if the values they depend on are not valid, nothing will crash and a high-level rule will not be applied.

Example:

schema = Dry::Schema.Params do
  required(:email).filled(:string)
  required(:age).filled(:integer)
end

result = schema.call(email: 'jane@doe.org', age: 21)

# access validation output data
result.to_h
# => {:email=>'jane@doe.org', :age=>21}

# check if all rules passed
result.success?
# => true

# check if any of the rules failed
result.failure?
# => false

Defining base schema class

class AppSchema < Dry::Schema::Params
  config.messages.load_paths << '/my/app/config/locales/en.yml'
  config.messages.backend = :i18n

  define do
    # define common rules, if any
  end
end

# now you can build other schemas on top of the base one:
class MySchema < AppSchema
  # define your rules
end

my_schema = MySchema.new

Working with error messages

The result object returned by Schema#call provides an API to convert error objects to human-friendly messages.

result = schema.call(email: nil, age: 21)

# get default errors
result.errors.to_h
# => {:email=>['must be filled']}

# get full errors
result.errors(full: true).to_h
# => {:email=>['email must be filled']}

# get errors in another language
result.errors(locale: :pl).to_h
# => {:email=>['musi być wypełniony']}

Checking presence of errors

You can ask result object if there are any errors under given path.

schema = Dry::Schema.Params do
  required(:name).filled(:string)
  optional(:tags).array(:str?)
end

result = schema.call(name: "", tags: ["red", 123])

result.error?(:name)
# => true

result.error?(:tags)
# => true

result.error?([:tags, 0])
# => false

result.error?([:tags, 1])
# => true

Learn more

octocatEdit on GitHub