Stubs, Spies and Fakes. Mocks are not supported, because they are dumb (yes, really).
The model is fairly simple and intuitive. I don't think many developers will have trouble understanding what is going on.
A word on the interface:
The interface is intended to be simple and non-intrusive (I do not, for instance, monkey-patch a "stubs" method onto Object, which would be an abomination before God and Man).
Anything in the public interface can be called using the module, for instance:
RDouble::create_stub(:a => 1)
If this is intrusive to you, you can include RDouble in your class, which would give you access to everything without being explicit with the RDouble:
class TestThisThing
include RDouble
def test_1
a = create_stub(:a => 1)
end
end
My background is in Python, so I prefer explicit to implicit. However, you really should include RDouble in any test class, because it automatically includes a teardown that rolls back all the swaps at the end. This is more than just a nice-to-have, it is essential for keeping your tests isolated.
Example class:
My examples will generally be using this class:
class KlassA
def self.klass_method
return "KlassA.klass_method"
end
def klass_a_instance_method
return "KlassA.klass_a_instance_method"
end
def method_with_arguments(arg1)
return "KlassA.method_with_arguments"
end
end
def fake_method_b(this)
return "fake_method_b"
end
class FakeKlass
def fake_method(this)
return [this, self]
end
end
Notice the "this" argument to my fake method b. Yes, it is peculiar. We'll talk about it later.
Fakes: Everything in ruby-doubles is built on the notion of swapping an existing function with another function at run-time, with an easily-understood syntax, that can be easily undone later. In ruby, this is traditionally accomplished on a one-off basis by "opening up" the class and changing the implementation. If you want the ability to roll it back later, you need to alias the old implementation to another name for safe-keeping. I'm not going to demonstrate that, because it is ugly and dumb (yes, really).
I should note, when it comes to test doubles you usually want a Stub or a Spy. A Fake is used in those special situations where you require some sophistication. If you are not a sophisticated person, and just want stubs and spies (which is perfectly reasonable), and you don't care about what Stubs and Spies are based on in ruby-doubles, you can just skip the following section on Fakes.
Here is how you swap Fakes using ruby-doubles:
Swapping a class method:
result = KlassA.klass_method
#result is now "KlassA.klass_method"
RDouble::swap_double(KlassA, :klass_method, method(:fake_method_b))
result = KlassA.klass_method
#result is now "fake_method_b"
Swapping an instance method:
a = KlassA.new()
result = a.klass_a_instance_method
#result is now "KlassA.klass_a_instance_method"
RDouble::swap_double(a, :klass_a_instance_method, method(:fake_method_b))
result = a.klass_a_instance_method
#result is now "fake_method_b
Swapping for all instances of a class:
a1 = KlassA.new()
a2 = KlassA.new()
#a1 and a2 both return "KlassA.klass_a_instance_method" if you call klass_a_instance_method on them
RDouble::swap_double(KlassA, :klass_a_instance_method, method(:fake_method_b), :all_instances => true)
a3 = KlassA.new()
#a1, a2, and a3 now both return "fake_method_b" when klass_a_instance_method is called
Let's talk about the "this" argument now. Any fake method that we want to swap in requires a "this" argument. This is the only way that I could figure out to give the method access to the instance or class that was calling it. The original method has full access to "self", and any instance variables or private methods contained therein. The fake method only has access to the public interface of "this". This is annoying, but acceptable, as Ruby gives you ways to access both the instance variables and the private methods of "this".
"this" will be set to the receiver of whatever method you are replacing. If you are replacing a class method, "this" is the class. If you are replacing an instance method, "this" is the instance.
As you might expect, "self" in the fake method is whatever it was before the swap. It is
- Undefined if your fake method is not attached to a class or instance
- Refers to the class or instance that it is attached to, otherwise.
Here is an example that illustrates this:
a = KlassA.new()
RDouble::swap_double(a, :klass_a_instance_method, FakeKlass.instance_method(:fake_method))
result = a.klass_a_instance_method
#result is [KlassA, FakeKlass]
Unswapping:
At the end of your test, it's important to restore the original implementations. This is really most vital for non-transient subjects - in other words, things whose lifespan will surpass the execution of the current test-case. For instance, local variables declared in the test case will not be accessible after the test-case has finished, but classes, modules and global variables will. Unswapping is important because, of course, you do not want global changes made in one test case to "leak" into another, thereby affecting its success or failure. This would lead to test cases that passed individually, but failed in the suite if they were unlucky enough to be preceded by the wrong test case. Here is how you unswap using ruby-doubles:
#unswaps everything
RDouble::unswap_doubles()
#unswaps only changes made on KlassA
RDouble::unswap_doubles(KlassA)
#unswaps only KlassA.klass_method
RDouble::unswap_doubles(KlassA, :klass_method)
Of course, this isn't the way you should generally do it in a test case. If the test case fails,
the unswap will never be run. You could include an "ensure" block, but the standard way is to do
all your teardown in the "teardown" method. Even better, because it's not good to have to think
about doing this (and the price of forgetting is often too high), is to include RDouble in your Test Class.
This "mixes in" a teardown function which reverts everything for you.
Namespaces:
You generally do not need to worry about the namespaces feature, as, if they are omitted, everything just
defaults to :standard. The point of this feature is to make it possible to install fakes at the beginning
of the run of your test suite, and not have them rolled back anytime someone calls unswap_doubles().
The namespace is only used for keeping track of what needs to be unswapped. An example is more
illustrative:
RDouble::swap_double(KlassA, :klass_method, method(:fake_method_b), :namespace => :some_other_namespace)
#KlassA.klass_method is swapped as you'd expect, so when called, it returns "fake_method_b"
RDouble::unswap_doubles()
#That just unswapped everything in the :standard namespace, so KlassA.klass_method still returns "fake_method_b"
RDouble::unswap_doubles(:namespace => :some_other_namespace)
#Now everything has been returned to the way it was.
The takeaway is this: namespaces are only
Stubs:
Stubs are just generic objects that are set up to have certain attributes set to known values.
Here is how you make one with ruby-doubles:
mystub = RDouble::create_stub(:a => 1, :b => 2)
result = mystub.a
# result is now 1
result = mystub.b
# result is now 2
"Stubbing" a function:
A function can be "swapped" with another function that just returns a given value. This is called "stubbing" the function:
result = KlassA.klass_method
#result is now "KlassA.klass_method"
RDouble::install_stub(KlassA, :klass_method, :returns => "stubbed_klass_method")
result = KlassA.klass_method
#result is now "stubbed_klass_method"
Spy Functions:
Spy Functions are like Stubs (In my implementation, SpyFunction inherits from StubFunction). They return whatever you want them to. However, they also remember each time they are called, and with what arguments. This is extremely useful if you have an external service that you are trying to use (like a Database or some HTTP server somewhere) that you don't want to actually use in your test (usually because of performance). In cases like these, what you want is to capture the calls to these services and make assertions about them. Here is an example of spies in ruby-doubles:
a = KlassA.new
spy = RDouble::install_spy(KlassA, :klass_method, :returns => "stubbed_klass_method")
result = a.method_with_arguments(1)
#result is now "stubbed_klass_method"
a.method_with_arguments(2)
result = spy.calls.map {|c| c.arguments}
#result is [[1], [2]]
If you are installing a spy somewhere deep in your code and don't have the luxury of saving the spy instance somewhere where you can access it from your test case, you can do this:
RDouble::install_spy(KlassA, :klass_method, :returns => "stubbed_class_method")
#some time passes, and your scope changes, and you don't have access to the return value anymore
spy = RDouble::get_double(KlassA, :klass_method)
a.method_with_arguments(1)
result = spy.calls.map {|c| c.arguments}
#result is [[1], [2]]
A Word about Special Options:
Special options passed to the install_stub and install_spy functions that pertain to how the double is installed, not how the stub behaves (eg. :all_instances and :namespace) will be passed on to the swap_double function, as you might expect:
RDouble::install_stub(KlassA, :klass_a_instance_method, :returns => "stubbed_instance_method",
:all_instances => true,
:namespace => :not_standard)
So, does all of this work?
It works for me in all my uses of it (and I use it fairly heavily). In addition, all my test cases pass (if you want to understand more specific behavior, you should peruse the test cases).
There is some special logic present for ruby 1.8 related to how 1.8 handles inherited singleton methods. I'm not going to claim that my solution is particularly elegant, or that it is fool-proof (I only know that I have not yet produced a test-case that can make it fail). I do know that it incurs a noticeable performance hit on 1.8. I may get around to improving that at some point, but please understand that 1.8 is not vital to any of the projects that I am currently working on.
As an aside, the other test-double frameworks I've looked at all have special logic that they have to execute for ruby 1.8 to get around this issue. Their solution may be more appropriate for you if your project is entrenched in 1.8.
To clarify, all of the special logic I wrote to sidestep this issue is skipped entirely on 1.9 or above.