use Test;
use Test::Mock;
plan 2;
class Foo {
method lol() { 'rofl' }
method wtf() { 'oh ffs' }
}
my $x = mocked(Foo);
$x.lol();
$x.lol();
check-mock($x,
*.called('lol', times => 2),
*.never-called('wtf'),
);
Test::Mock is a module that works alongside the standard Test module to help you write tests when you want to verify what methods are called on an object, while still having calls to undefined methods die.
You get started just as normal with the test file, but also add a use statement for Test::Mock.
use Test;
use Test::Mock;
plan 2;
Imagine we have some class Foo:
class Foo {
method lol() { 'rofl' }
method wtf() { 'oh ffs' }
}
We then arrange to have a mocked instance of this class. This means that instead of calls to lol and wtf actually resulting in the methods being invoked, it simply logs the invocations.
my $x = mocked(Foo);
We can then take the actions that should result in some method calls. Here we just make them directly, but you'd probably pass the mock to other bits of code that will make calls on it.
$x.lol();
$x.lol();
When you're done, you assert that the things you expected to happen actually happened.
check-mock($x,
*.called('lol', times => 2),
*.never-called('wtf'),
);
And it's as easy as that. Of course, you may also be interested to check that the arguments passed to the mocked method were as expected. For our second example, here's a class representing one of my favorite places.
class Pub {
method order_beer($pints) { }
method throw($what) { }
}
We'll also declare a couple of other classes:
class Glass { }
class Party { }
Our test file would have started with the same boilerplate - use Test and Test::Mock, and set a plan. We then produce a mock instance of Pub:
my $p = mocked(Pub);
And do our stuff:
$p.throw(Party.new);
$p.order_beer(2);
$p.order_beer(1);
After our excruciatingly low on beer party, we can now do some checks. Of course, we ordered beer twice, so we can check this as before:
check-mock($p,
*.called('order_beer', times => 2),
But what if we wanted to check the arguments passed to the method? In that case, you can simply pass along the parameter "with". We may pass a Capture here, which contains the exact arguments we expected to be passed; this will be tested against the actual passed Capture for equivalance.
*.called('order_beer', times => 1, with => \(1)),
*.called('order_beer', times => 1, with => \(2)),
*.never-called('order_beer', with => \(10)),
That's going to cover some cases, but what if we wanted to check if things of the correct type were passed? In that case, write a Signature literal, and the args Capture will be smart-matched against it, which conveniently happens to check if the Capture could have bound to this Signature.
*.called('throw', with => :(Party)),
*.never-called('throw', with => :(Glass)),
Of course, now the gloves are off: if you have a Signature you can do all kinds of matching, with constraints and sub-signatures. Here's an easy but not so creative example (I need at least 3 pints to be creative...)
*.called('order_beer', times => 2, with => :($ where { $^n < 10 })),
*.never-called('order_beer', with => :($ where { $^n >= 10 })),
And if all that isn't enough, since we just smart-match against anything else you may pass as the with argument, you may also pass a block that takes a Capture as a parameter and implement whatever fancier checks you may wish to.
In some cases, just logging method calls on your mock may not be enough; you may wish them to return some fake data from the method call. For example, we may have a yak shaving class that we dependency-inject with a yak provider and a yak shaver.
class Yak {
has $.shaved;
}
class Shaver {
method shave($yak) {
...
}
}
class YakStore {
method get-all-yaks() {
...
}
}
class YakShaving {
has $.yak-store;
has $.yak-shaver;
method proccess() {
for $!yak-store.get-all-yaks() -> $yak {
unless $yak.shaved {
$!yak-shaver.shave($yak);
}
}
}
}
We want to check that our the shave method from the Shaver class is only invoked for yaks that need shaving. We set up our mock of the Shaver class just as normal:
my $shaver = mocked(Shaver);
However, for the Yak store we want to provide some fake yaks in various states of shavenness.
my $store = mocked(YakStore, returning => {
get-all-yaks => (Yak.new(:!shaved), Yak.new(:shaved), Yak.new(:!shaved))
});
Now we can inject our mocks to the YakShaving class and and get it to do it's thing.
my $yaktivity = YakShaving.new(
yak-store => $store,
yak-shaver => $shaver
);
$yaktivity.proccess();
And finally, it's time to write our tests. We expect just one call on the store:
check-mock($store,
*.called('get-all-yaks', times => 1)
);
On the shaver, we expect two calls in total to the shave method with yaks that are unshaven, and no calls at all with shaven yaks.
check-mock($shaver,
*.called('shave', times => 2, with => :($ where { !$^y.shaved })),
*.never-called('shave', with => :($ where { $^y.shaved }))
);
This is the first example where we're really made good use of mock testing; if absolutely every object involved in the test is mocked, then you'd not be testing any of the actual real code. Of course, being able to do this easily somewhat depends on good de-coupled code design, where objects are given instances of other objects to work on rather than directly instantiating objects of other classes.
Sometimes, it would be nice to fake an exception or do some other computation
rather than return a literal value. For that, use computing
instead of using
returning
. Our previous example would look like this:
my $store = mocked(YakStore, computing => {
get-all-yaks => { Yak.new(:!shaved), Yak.new(:shaved), Yak.new(:!shaved) }
});
However, unlike with returning
, it's possible to instead do:
my $store = mocked(YakStore, computing => {
get-all-yaks => { die "Could not connect to yak-db" }
});
Occasionally, it is also desirable to have access to the arguments passed to
method. In this case, use overriding
:
my $shaver = mocked(Shaver, overriding => {
shave => -> $yak { die 'Already shaven' if $yak.shaved }
});
Of course, you can use returning
, computing
and overriding
in a call to mocked
to get different behaviours for different methods, as long as each method name only occurs once.
Feature requests, bug reports and patches on this module are welcome; use the GitHub issues tracker.