Skip to main content

Dynamic classes

Verifying instance doubles do not support methods which the class reports to not exist since an actual instance of the class would be required to verify against. This is commonly the case when method_missing is used.

There are a few ways to work around this. If the object has already been loaded you may consider using an object_double, but that cannot work if you are testing in isolation. Alternatively you could implement the methods directly (calling super to return the method_missing definition).

Some of these classes may have methods to define these methods on the objects at runtime. (For example, ActiveRecord does this to define methods from database columns.) For these cases we provide an API which can be used to customise verifying doubles on creation. We use this ourselves in rspec-rails to set up some niceties for you.

These types of methods are supported at class level (with class_double) however, since respond_to? can be queried directly on the class.

Background

Given a file named "lib/fake_active_record.rb" with:

class FakeActiveRecord
COLUMNS = %w[name email]

def respond_to_missing?(method_name)
COLUMNS.include?(method_name.to_s) || super
end

def method_missing(method_name, *args)
if respond_to?(method_name)
instance_variable_get("@#{method_name}")
else
super
end
end

def self.define_attribute_methods
COLUMNS.each do |name|
define_method(name) { instance_variable_get("@#{name}") }
end
end
end

Given a file named "spec/user_spec.rb" with:

require 'user'

RSpec.describe User do
it 'can be doubled' do
instance_double("User", :name => "Don")
end
end

Fails with method missing

Given a file named "lib/user.rb" with:

require 'fake_active_record'

class User < FakeActiveRecord
end

When I run rspec spec/user_spec.rb

Then the output should contain "1 example, 1 failure".

Workaround with explicit definitions

Given a file named "lib/user.rb" with:

require 'fake_active_record'

class User < FakeActiveRecord
def name; super end
def email; super end
end

When I run rspec spec/user_spec.rb

Then the examples should all pass.

Workaround using callback

Given a file named "lib/user.rb" with:

require 'fake_active_record'

class User < FakeActiveRecord
end

And a file named "spec/fake_record_helper.rb" with:

RSpec.configuration.mock_with(:rspec) do |config|
config.before_verifying_doubles do |reference|
reference.target.define_attribute_methods
end
end
#
# or you can use:
#
# RSpec::Mocks.configuration.before_verifying_doubles do |reference|
# reference.target.define_attribute_methods
# end

When I run rspec -r fake_record_helper spec/user_spec.rb

Then the examples should all pass.