lib/rspec/rails/matchers/be_a_new.rb
module RSpec::Rails::Matchers class BeANew < RSpec::Matchers::BuiltIn::BaseMatcher def initialize(expected) @expected = expected end # @api private def matches?(actual) @actual = actual actual.is_a?(expected) && actual.new_record? && attributes_match?(actual) end # Use this to specify the specific attributes to match on the new record. # # @example # # it "assigns a new Thing with the submitted attributes" do # post :create, :thing => { :name => "Illegal Value" } # assigns(:thing).should be_a_new(Thing).with(:name => nil) # end def with(expected_attributes) attributes.merge!(expected_attributes) self end # @api private def failure_message_for_should [].tap do |message| unless actual.is_a?(expected) && actual.new_record? message << "expected #{actual.inspect} to be a new #{expected.inspect}" end unless attributes_match?(actual) if unmatched_attributes.size > 1 message << "attributes #{unmatched_attributes.inspect} were not set on #{actual.inspect}" else message << "attribute #{unmatched_attributes.inspect} was not set on #{actual.inspect}" end end end.join(' and ') end private def attributes @attributes ||= {} end def attributes_match?(actual) attributes.stringify_keys.all? do |key, value| actual.attributes[key].eql?(value) end end def unmatched_attributes attributes.stringify_keys.reject do |key, value| actual.attributes[key].eql?(value) end end end # Passes if actual is an instance of `model_class` and returns `false` for # `persisted?`. Typically used to specify instance variables assigned to # views by controller actions # # @example # # get :new # assigns(:thing).should be_a_new(Thing) # # post :create, :thing => { :name => "Illegal Value" } # assigns(:thing).should be_a_new(Thing).with(:name => nil) def be_a_new(model_class) BeANew.new(model_class) end end