Created
July 2, 2018 17:28
-
-
Save TechFounder/070423aef8dcc551955d0a9b81098704 to your computer and use it in GitHub Desktop.
RSpec 3 Matchers
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
describe 'Expectation Matchers' do | |
describe 'equivalence matchers' do | |
it 'will match loose equality with #eq' do | |
a = "2 cats" | |
b = "2 cats" | |
expect(a).to eq(b) | |
expect(a).to be == b # synonym for #eq | |
c = 17 | |
d = 17.0 | |
expect(c).to eq(d) # different types, but "close enough" | |
end | |
it 'will match value equality with #eql' do | |
a = "2 cats" | |
b = "2 cats" | |
expect(a).to eql(b) # just a little stricter | |
c = 17 | |
d = 17.0 | |
expect(c).not_to eql(d) # not the same, close doesn't count | |
end | |
it 'will match identity equality with #equal' do | |
a = "2 cats" | |
b = "2 cats" | |
expect(a).not_to equal(b) # same value, but different object | |
c = b | |
expect(b).to equal(c) # same object | |
expect(b).to be(c) # synonym for #equal | |
end | |
end | |
describe 'truthiness matchers' do | |
it 'will match true/false' do | |
expect(1 < 2).to be(true) # do not use 'be_true' | |
expect(1 > 2).to be(false) # do not use 'be_false' | |
expect('foo').not_to be(true) # the string is not exactly true | |
expect(nil).not_to be(false) # nil is not exactly false | |
expect(0).not_to be(false) # 0 is not exactly false | |
end | |
it 'will match truthy/falsey' do | |
expect(1 < 2).to be_truthy | |
expect(1 > 2).to be_falsey | |
expect('foo').to be_truthy # any value counts as true | |
expect(nil).to be_falsey # nil counts as false | |
expect(0).not_to be_falsey # but 0 is still not falsey enough | |
end | |
it 'will match nil' do | |
expect(nil).to be_nil | |
expect(nil).to be(nil) # either way works | |
expect(false).not_to be_nil # nil only, just like #nil? | |
expect(0).not_to be_nil # nil only, just like #nil? | |
end | |
end | |
describe 'numeric comparison matchers' do | |
it 'will match less than/greater than' do | |
expect(10).to be > 9 | |
expect(10).to be >= 10 | |
expect(10).to be <= 10 | |
expect(9).to be < 10 | |
end | |
it 'will match numeric ranges' do | |
expect(10).to be_between(5, 10).inclusive | |
expect(10).not_to be_between(5, 10).exclusive | |
expect(10).to be_within(1).of(11) | |
expect(5..10).to cover(9) | |
end | |
end | |
describe 'collection matchers' do | |
it 'will match arrays' do | |
array = [1,2,3] | |
expect(array).to include(3) | |
expect(array).to include(1,3) | |
expect(array).to start_with(1) | |
expect(array).to end_with(3) | |
expect(array).to match_array([3,2,1]) | |
expect(array).not_to match_array([1,2]) | |
expect(array).to contain_exactly(3,2,1) # similar to match_array | |
expect(array).not_to contain_exactly(1,2) # but use individual args | |
end | |
it 'will match strings' do | |
string = 'some string' | |
expect(string).to include('ring') | |
expect(string).to include('so', 'ring') | |
expect(string).to start_with('so') | |
expect(string).to end_with('ring') | |
end | |
it 'will match hashes' do | |
hash = {:a => 1, :b => 2, :c => 3} | |
expect(hash).to include(:a) | |
expect(hash).to include(:a => 1) | |
expect(hash).to include(:a => 1, :c => 3) | |
expect(hash).to include({:a => 1, :c => 3}) | |
expect(hash).not_to include({'a' => 1, 'c' => 3}) | |
end | |
end | |
describe 'other useful matchers' do | |
it 'will match strings with a regex' do | |
# This matcher is a good way to "spot check" strings | |
string = 'The order has been received.' | |
expect(string).to match(/order(.+)received/) | |
expect('123').to match(/\d{3}/) | |
expect(123).not_to match(/\d{3}/) # only works with strings | |
email = 'someone@somewhere.com' | |
expect(email).to match(/\A\w+@\w+\.\w{3}\Z/) | |
end | |
it 'will match object types' do | |
expect('test').to be_instance_of(String) | |
expect('test').to be_an_instance_of(String) # alias of #be_instance_of | |
expect('test').to be_kind_of(String) | |
expect('test').to be_a_kind_of(String) # alias of #be_kind_of | |
expect('test').to be_a(String) # alias of #be_kind_of | |
expect([1,2,3]).to be_an(Array) # alias of #be_kind_of | |
end | |
it 'will match objects with #respond_to' do | |
string = 'test' | |
expect(string).to respond_to(:length) | |
expect(string).not_to respond_to(:sort) | |
end | |
it 'will match class instances with #have_attributes' do | |
class Car | |
attr_accessor :make, :year, :color | |
end | |
car = Car.new | |
car.make = 'Dodge' | |
car.year = 2010 | |
car.color = 'green' | |
expect(car).to have_attributes(:color => 'green') | |
expect(car).to have_attributes( | |
:make => 'Dodge', :year => 2010, :color => 'green' | |
) | |
end | |
it 'will match anything with #satisfy' do | |
# This is the most flexible matcher | |
expect(10).to satisfy do |value| | |
(value >= 5) && (value <=10) && (value % 2 == 0) | |
end | |
end | |
end | |
describe 'predicate matchers' do | |
it 'will match be_* to custom methods ending in ?' do | |
# drops "be_", adds "?" to end, calls method on object | |
# Can use these when methods end in "?", require no arguments, | |
# and return true/false. | |
# with built-in methods | |
expect([]).to be_empty # [].empty? | |
expect(1).to be_integer # 1.integer? | |
expect(0).to be_zero # 0.zero? | |
expect(1).to be_nonzero # 1.nonzero? | |
expect(1).to be_odd # 1.odd? | |
expect(2).to be_even # 1.even? | |
# be_nil is actually an example of this too | |
# with custom methods | |
class Product | |
def visible?; true; end | |
end | |
product = Product.new | |
expect(product).to be_visible # product.visible? | |
expect(product.visible?).to be true # exactly the same as this | |
end | |
it 'will match have_* to custom methods like has_*?' do | |
# changes "have_" to "has_", adds "?" to end, calls method on object | |
# Can use these when methods start with "has_", end in "?", | |
# and return true/false. Can have arguments, but not required. | |
# with built-in methods | |
hash = {:a => 1, :b => 2} | |
expect(hash).to have_key(:a) # hash.has_key? | |
expect(hash).to have_value(2) # hash.has_value? | |
# with custom methods | |
class Customer | |
def has_pending_order?; true; end | |
end | |
customer = Customer.new | |
expect(customer).to have_pending_order # customer.has_pending_order? | |
expect(customer.has_pending_order?).to be true # same as this | |
end | |
end | |
describe 'observation matchers' do | |
# Note that all of these use "expect {}", not "expect()". | |
# It is a special block format that allows a | |
# process to take place inside of the expectation. | |
it 'will match when events change object attributes' do | |
# calls the test before the block, | |
# then again after the block | |
array = [] | |
expect { array << 1 }.to change(array, :empty?).from(true).to(false) | |
class WebsiteHits | |
attr_accessor :count | |
def initialize; @count = 0; end | |
def increment; @count += 1; end | |
end | |
hits = WebsiteHits.new | |
expect { hits.increment }.to change(hits, :count).from(0).to(1) | |
end | |
it 'will match when events change any values' do | |
# calls the test before the block, | |
# then again after the block | |
# notice the "{}" after "change", | |
# can be used on simple variables | |
x = 10 | |
expect { x += 1 }.to change {x}.from(10).to(11) | |
expect { x += 1 }.to change {x}.by(1) | |
expect { x += 1 }.to change {x}.by_at_least(1) | |
expect { x += 1 }.to change {x}.by_at_most(1) | |
# notice the "{}" after "change", | |
# can contain any block of code | |
z = 11 | |
expect { z += 1 }.to change { z % 3 }.from(2).to(0) | |
# Must have a value before the block | |
# Must change the value inside the block | |
end | |
it 'will match when errors are raised' do | |
# observes any errors raised by the block | |
expect { raise StandardError }.to raise_error | |
expect { raise StandardError }.to raise_exception | |
expect { 1 / 0 }.to raise_error(ZeroDivisionError) | |
expect { 1 / 0 }.to raise_error.with_message("divided by 0") | |
expect { 1 / 0 }.to raise_error.with_message(/divided/) | |
# Note that the negative form does | |
# not accept arguments | |
expect { 1 / 1 }.not_to raise_error | |
end | |
it 'will match when output is generated' do | |
# observes output sent to $stdout or $stderr | |
expect { print('hello') }.to output.to_stdout | |
expect { print('hello') }.to output('hello').to_stdout | |
expect { print('hello') }.to output(/ll/).to_stdout | |
expect { warn('problem') }.to output(/problem/).to_stderr | |
end | |
end | |
describe 'compound expectations' do | |
it 'will match using: and, or, &, |' do | |
expect([1,2,3,4]).to start_with(1).and end_with(4) | |
expect([1,2,3,4]).to start_with(1) & include(2) | |
expect(10 * 10).to be_odd.or be > 50 | |
array = ['hello', 'goodbye'].shuffle | |
expect(array.first).to eq("hello") | eq("goodbye") | |
end | |
end | |
describe 'composing matchers' do | |
# some matchers accept matchers as arguments. (new in rspec3) | |
it 'will match all collection elements using a matcher' do | |
array = [1,2,3] | |
expect(array).to all( be < 5 ) | |
end | |
it 'will match by sending matchers as arguments to matchers' do | |
string = "hello" | |
expect { string = "goodbye" }.to change { string }. | |
from( match(/ll/) ).to( match(/oo/) ) | |
hash = {:a => 1, :b => 2, :c => 3} | |
expect(hash).to include(:a => be_odd, :b => be_even, :c => be_odd) | |
expect(hash).to include(:a => be > 0, :b => be_within(2).of(4)) | |
end | |
it 'will match using noun-phrase aliases for matchers' do | |
# These are built-in aliases that make | |
# specs read better by using noun-based | |
# phrases instead of verb-based phrases. | |
# valid but awkward example | |
fruits = ['apple', 'banana', 'cherry'] | |
expect(fruits).to start_with( start_with('a') ) & | |
include( match(/a.a.a/) ) & | |
end_with( end_with('y') ) | |
# improved version of the previous example | |
# "start_with" becomes "a_string_starting_with" | |
# "end_with" becomes "a_string_ending_with" | |
# "match" becomes "a_string_matching" | |
fruits = ['apple', 'banana', 'cherry'] | |
expect(fruits).to start_with( a_string_starting_with('a') ) & | |
include( a_string_matching(/a.a.a/) ) & | |
end_with( a_string_ending_with('y') ) | |
# valid but awkward example | |
array = [1,2,3,4] | |
expect(array).to start_with( be <= 2 ) | | |
end_with( be_within(1).of(5) ) | |
# improved version of the previous example | |
# "be <= 2" becomes "a_value <= 2" | |
# "be_within" becomes "a_value_within" | |
array = [1,2,3,4] | |
expect(array).to start_with( a_value <= 2 ) | | |
end_with( a_value_within(1).of(5) ) | |
end | |
end | |
end |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment