Yielding

Use and_yield to make the test double yield the provided arguments when it receives the message. If the caller does not provide a block, or the caller’s block does not accept the provided arguments, an error will be raised. If you want to yield multiple times, chain multiple and_yield calls together.

Yield an argument

Given a file named “yieldargumentsspec.rb” with:

RSpec.describe "Making it yield arguments" do
  it "yields the provided args" do
    dbl = double
    allow(dbl).to receive(:foo).and_yield(2, 3)

    x = y = nil
    dbl.foo { |a, b| x, y = a, b }
    expect(x).to eq(2)
    expect(y).to eq(3)
  end
end

When I run rspec yield_arguments_spec.rb

Then the examples should all pass.

It fails when the caller does not provide a block

Given a file named “nocallerblock_spec.rb” with:

RSpec.describe "Making it yield" do
  it "fails when the caller does not provide a block" do
    dbl = double
    allow(dbl).to receive(:foo).and_yield(2, 3)
    dbl.foo
  end
end

When I run rspec no_caller_block_spec.rb

Then it should fail with:

#<Double (anonymous)> asked to yield |[2, 3]| but no block was passed

It fails when the caller’s block does not accept the provided arguments

Given a file named “argmismatchspec.rb” with:

RSpec.describe "Making it yield" do
  it "fails when the caller's block does not accept the provided arguments" do
    dbl = double
    allow(dbl).to receive(:foo).and_yield(2, 3)
    dbl.foo { |x| }
  end
end

When I run rspec arg_mismatch_spec.rb

Then it should fail with:

#<Double (anonymous)> yielded |2, 3| to block with arity of 1

Yield multiple times

Given a file named “yieldmultipletimes_spec.rb” with:

RSpec.describe "Making it yield multiple times" do
  it "yields the specified args in succession" do
    yielded = []

    dbl = double
    allow(dbl).to receive(:foo).and_yield(1).and_yield(2).and_yield(3)
    dbl.foo { |x| yielded << x }

    expect(yielded).to eq([1, 2, 3])
  end
end

When I run rspec yield_multiple_times_spec.rb

Then the examples should all pass.