module Remarkable module ActiveModel # Holds ActiveModel matchers. # # == Validations matchers # # Remarkable supports all ActiveModel validations, and the only options # not supported in those matchers is the :on options. So whenever you have # to test that a validation runs on update, you have to do reproduce the # state in your tests: # # describe Project do # describe 'validations on create' do # should_validate_presence_of :title # end # # describe 'validations on update' do # subject { Post.create!(@valid_attributes) } # should_validate_presence_of :updated_at # end # end # # Another behavior in validations is the :message option. Whenever you change # the message in your model, it must be given in your tests too: # # class Post < ActiveModel::Base # validates_presence_of :title, :message => 'must be filled' # end # # describe Post do # should_validate_presence_of :title #=> fails # should_validate_presence_of :title, :message => 'must be filled' # end # # However, if you change the title using the I18n API, you don't need to # specify the message in your tests, because it's retrieved properly. # module Matchers class ValidatePresenceOfMatcher < Remarkable::ActiveModel::Base #:nodoc: arguments :collection => :attributes, :as => :attribute optional :message collection_assertions :allow_nil? default_options :message => :blank protected def allow_nil? bad?(blank_value, :message) end def blank_value collection? ? [] : nil end def collection? if reflection = find_reflection [:has_many, :has_and_belongs_to_many].include?(reflection.macro) else false end end private def find_reflection if subject_class.respond_to?(:reflect_on_association) subject_class.reflect_on_association(@attribute) end end end # Ensures that the model cannot be saved if one of the attributes listed is not present. # # == Options # # * :message - value the test expects to find in errors[:attribute]. # Regexp, string or symbol. Default = I18n.translate('activerecord.errors.messages.blank') # # == Examples # # should_validate_presence_of :name, :phone_number # it { should validate_presence_of(:name, :phone_number) } # def validate_presence_of(*args, &block) ValidatePresenceOfMatcher.new(*args, &block).spec(self) end end end end