require 'spec_helper' describe Volt::Model do let(:model) { test_model_class.new } let(:test_model_class) do Class.new(Volt::Model) do set_collection_name :test_model4_classes validate :count, numericality: { min: 5, max: 10 } validate :description, length: { message: 'needs to be longer', length: 50 } validate :email, email: true validate :name, length: 4 validate :phone_number, phone_number: true validate :username, presence: true end end it 'should return errors for all failed validations' do model.validate! expect(model.errors).to eq( count: ['must be a number'], description: ['needs to be longer'], email: ['must be an email address'], name: ['must be at least 4 characters'], phone_number: ['must be a phone number with area or country code'], username: ['must be specified'] ) end it 'should show all fields in marked errors once saved' do buffer = model.buffer buffer.save! expect(buffer.marked_errors.keys).to eq( [:count, :description, :email, :name, :phone_number, :username] ) end describe 'builtin validations' do shared_examples_for 'a built in validation' do |field, message| specify do expect { model.mark_field! field } .to change { model.marked_errors } .from({}).to(field => [message]) end end describe 'numericality' do message = 'must be a number' it_should_behave_like 'a built in validation', :count, message end describe 'length' do message = 'needs to be longer' it_should_behave_like 'a built in validation', :description, message end describe 'email' do message = 'must be an email address' it_should_behave_like 'a built in validation', :email, message end describe 'name' do message = 'must be at least 4 characters' it_should_behave_like 'a built in validation', :name, message end describe 'phone_number' do message = 'must be a phone number with area or country code' it_should_behave_like 'a built in validation', :phone_number, message end describe 'presence' do message = 'must be specified' it_should_behave_like 'a built in validation', :username, message end it 'should fail on non-numbers' do model._count = 'not a number' expect(model.errors[:count]).to eq(['must be a number']) end end describe 'validators with multiple criteria' do let(:regex_message) { 'regex failed' } let(:proc_message) { 'proc failed' } let(:test_model_class) do Class.new(Volt::Model) do set_collection_name :test_model5_classes validate :special_field, format: [ { with: /regex/, message: 'regex failed' }, { with: ->(x) { x == false }, message: 'proc failed' } ] end end context 'when multiple fail' do before { model._special_field = 'nope' } it 'returns an array of errors' do expect(model.errors).to eq(special_field: [regex_message, proc_message]) end end context 'when one fails' do before do # Prevent rollback for testing allow(model).to receive(:revert_changes!) model._special_field = 'regex' end it 'returns an array with a single error' do expect(model.errors.to_h).to eq(special_field: [proc_message]) end end end it 'should report if errors have happened in changed attributes' do # Prevent revert_changes! so it doesn't revert on failed values allow(model).to receive(:revert_changes!) expect(model.error_in_changed_attributes?).to eq(false) model._not_validated_attr = 'yes' expect(model.error_in_changed_attributes?).to eq(false) model._name = '5' # fail, too short expect(model.changed?(:name)).to eq(true) expect(model.error_in_changed_attributes?).to eq(true) model._name = 'Jimmy' expect(model.error_in_changed_attributes?).to eq(false) end it 'should revert changes which fail a validation' do model._name = 'bob' # fails too short validation expect(model._name).to eq(nil) model._name = 'Jimmy' # long enough, passes expect(model._name).to eq('Jimmy') model._name = 'ok' # fails again expect(model._name).to eq('Jimmy') end end