RSpec.describe Schema, 'using high-level rules' do context 'composing rules' do subject(:schema) do Dry::Validation.Schema do configure do def self.messages Messages.default.merge( en: { errors: { destiny: 'you must select either red or blue' } } ) end end optional(:red).maybe optional(:blue).maybe rule(destiny: [:red, :blue]) do |red, blue| red.filled? | blue.filled? end end end it 'passes when only red is filled' do expect(schema.(red: '1')).to be_success end it 'fails when keys are missing' do expect(schema.({})).to be_failure end it 'fails when red and blue are not filled ' do expect(schema.(red: nil, blue: nil).messages).to eql( destiny: ['you must select either red or blue'] ) end end context 'composing specific predicates' do let(:schema) do Dry::Validation.Schema do configure do def self.messages Messages.default.merge( en: { errors: { email_presence: 'must be present when login is set to true', email_absence: 'must not be present when login is set to false' } } ) end end required(:login).filled(:bool?) required(:email).maybe rule(:email_presence) { value(:login).true?.then(value(:email).filled?) } rule(:email_absence) { value(:login).false?.then(value(:email).none?) } end end it 'passes when login is false and email is nil' do expect(schema.(login: false, email: nil)).to be_success end it 'fails when login is false and email is present' do expect(schema.(login: false, email: 'jane@doe').messages).to eql( email_absence: ['must not be present when login is set to false'] ) end it 'passes when login is true and email is present' do expect(schema.(login: true, email: 'jane@doe')).to be_success end it 'fails when login is true and email is not present' do expect(schema.(login: true, email: nil).messages).to eql( email_presence: ['must be present when login is set to true'] ) end end describe 'with nested schemas' do subject(:schema) do Dry::Validation.Schema do required(:command).filled(:str?, included_in?: %w(First Second)) required(:args).filled(:hash?) rule(first_args: [:command, :args]) do |command, args| command.eql?('First') .then(args.schema { required(:first).filled(:bool?) }) end rule(second_args: [:command, :args]) do |command, args| command.eql?('Second') .then(args.schema { required(:second).filled(:bool?) }) end end end it 'generates check rule matching on value' do expect(schema.(command: 'First', args: { first: true })).to be_success expect(schema.(command: 'Second', args: { second: true })).to be_success expect(schema.(command: 'oops', args: { such: 'validation' }).messages).to eql( command: ['must be one of: First, Second'] ) expect(schema.(command: 'First', args: { second: true }).messages).to eql( args: { first: ['is missing'] } ) expect(schema.(command: 'Second', args: { first: true }).messages).to eql( args: { second: ['is missing'] } ) end end end