Using an instance double

An instance_double is the most common type of verifying double. It takes a class name or object as its first argument, then verifies that any methods being stubbed would be present on an instance of that class. In addition, when it receives messages, it verifies that the provided arguments are supported by the method signature, both in terms of arity and allowed or required keyword arguments, if any. The same argument verification happens when you constrain the arguments using with.

For methods handled by method_missing, see dynamic classes.

Background

Given a file named “app/models/user.rb” with:

class User < Struct.new(:notifier)
  def suspend!
    notifier.notify("suspended as")
  end
end

Given a file named “spec/unit_helper.rb” with:

$LOAD_PATH.unshift("app/models")

Given a file named “spec/spec_helper.rb” with:

require 'unit_helper'

require 'user'
require 'console_notifier'

RSpec.configure do |config|
  config.mock_with :rspec do |mocks|

    # This option should be set when all dependencies are being loaded
    # before a spec run, as is the case in a typical spec helper. It will
    # cause any verifying double instantiation for a class that does not
    # exist to raise, protecting against incorrectly spelt names.
    mocks.verify_doubled_constant_names = true

  end
end

Given a file named “spec/unit/user_spec.rb” with:

require 'unit_helper'

require 'user'

RSpec.describe User, '#suspend!' do
  it 'notifies the console' do
    notifier = instance_double("ConsoleNotifier")

    expect(notifier).to receive(:notify).with("suspended as")

    user = User.new(notifier)
    user.suspend!
  end
end

Spec passes in isolation

When I run rspec spec/unit/user_spec.rb

Then the examples should all pass.

Spec passes with dependencies loaded and method implemented

Given a file named “app/models/console_notifier.rb” with:

class ConsoleNotifier
  def notify(msg)
    puts message
  end
end

When I run rspec -r./spec/spec_helper spec/unit/user_spec.rb

Then the examples should all pass.

Spec fails with dependencies loaded and method unimplemented

Given a file named “app/models/console_notifier.rb” with:

class ConsoleNotifier
end

When I run rspec -r./spec/spec_helper spec/unit/user_spec.rb

Then the output should contain “1 example, 1 failure”

And the output should contain “ConsoleNotifier class does not implement the instance method:”.

Spec fails with dependencies loaded and incorrect arity

Given a file named “app/models/console_notifier.rb” with:

class ConsoleNotifier
  def notify(msg, color)
    puts color + message
  end
end

When I run rspec -r./spec/spec_helper spec/unit/user_spec.rb

Then the output should contain “1 example, 1 failure”

And the output should contain “Wrong number of arguments.”.