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