NAME
Mock::MonkeyPatch - Monkey patching with test mocking in mind
SYNOPSIS
{
package MyApp;
sub gen_item_id {
my $type = shift;
# calls external service and gets id for $type
}
sub build_item {
my $type = shift;
my $item = Item->new(type => $type);
$item->id(gen_item_id($type));
return $item;
}
}
use Test::More;
use MyApp;
use Mock::MonkeyPatch;
my $mock = Mock::MonkeyPatch->patch(
'MyApp::gen_item_id' => sub { 'abcd' }
);
my $item = MyApp::build_item('rubber_chicken');
is $item->id, 'abcd', 'building item calls MyApp::gen_random_id';
ok $mock->called, 'the mock was indeed called';
is_deeply $mock->arguments, ['rubber_chicken'], 'the mock was called with expected arguments';
DESCRIPTION
Mocking is a common tool, especially for testing. By strategically
replacing a subroutine, one can isolate segments (units) of code to
test individually. When this is done it is important to know that the
mocked sub was actually called and with what arguments it was called.
Mock::MonkeyPatch injects a subroutine in the place of an existing one.
It returns an object by which you can revisit the manner in which the
mocked subroutine was called. Further when the object goes out of scope
(or when the "restore" method is called) the original subroutine is
replaced.
CONSTRUCTOR
patch
my $mock = Mock::MonkeyPatch->patch('MyPackage::foo' => sub { ... });
my $mock = Mock::MonkeyPatch->patch('MyPackage::foo' => sub { ... }, \%options);
Mock a subroutine and return a object to represent it. Takes a fully
qualifed subroutine name, a subroutine reference to call in its place,
and optionally a hash reference of additional constructor arguments.
The replacement subroutine will be wrapped in a one that will store
calling data, then injected in place of the original. Within the
replacement subroutine the original is available as the fully qualified
subroutine Mock::MonkeyPatch::ORIGINAL. This can be used to inject
behavior before, after, or even around the original. This includes
munging the arguments passed to the origial (though the actual
arguments are what are stored). For example usage, see "COOKBOOK".
Currently the optional hashref only accepts one option, an initial
value for "store_arguments" which is true if not given.
The wrapper will have the same prototype as the mocked function if one
exists. The replacement need not have any prototype, the arguments
received by the wrapper will be passed to the given sub as they were
received. (If this doesn't make any sense to you, don't worry about
it.)
METHODS
arguments
my $args = $mock->arguments;
my $args_second_time = $mock->arguments(1);
Returns an array reference containing the arguments that were passed to
the mocked subroutine (but see also "store_arguments"). Optionally an
integer may be passed which designates the call number to fetch
arguments in the same manner of indexing an array (zero indexed). If
not given, 0 is assumed, representing the first time the mock was
called. Returns undef if the mocked subroutine was not called (or was
not called enough times).
use Test::More;
is_deeply $mock->arguments, [1, 2, 3], 'called with the right arguments';
called
my $time_called = $mock->called;
Returns the number of times the mocked subroutine was called. This
means that that there should be values available from "arguments" up to
the value of $mock->called - 1.
use Test::More;
ok $mock->called, 'mock was called';
is $mock->called, 3, 'mock was called three times';
method_arguments
my $args = $mock->method_arguments;
my $args_third_time = $mock->method_arguments(2, 'MyClass');
A wrapper around "arguments" convenient for when the mocked subroutine
is called as a method. Like "arguments" it returns a subroutine
reference, though it removes the first arguments which is the invocant.
It also can take a call number designation.
Additionally it takes a class name to test against the invocant as
$invocant->isa('Class::Name'). If the invocant is not an instance of
the class or a subclass thereof it returns undef.
use Test::More;
is_deeply $mock->method_arguments(0, 'FrobberCo::Employee'),
['some', 'arguments'], 'mock method called with known arguments on a FrobberCo::Employee instance';
reset
$mock = $mock->reset;
Reset the historical information stored in the mock, including
"arguments" and "called". Returns the mock instance for chaining if
desired.
Note that this does not restore the original method. for that, see
"restore".
use Test::More;
is $mock->called, 3, 'called 3 times';
is $mock->reset->called, 0, 'called zero times after reset';
restore
$mock = $mock->restore;
Restore the original method to its original place in the symbol table.
This method is also called automatically when the object goes out of
scope and is garbage collected. Returns the mock instance for chaining
if desired. This method can only be called once!
Note that this does not reset historical information stored in the
mock, for that, see "reset".
store_arguments
$mock = $mock->store_arguments(0);
When true, the default if not passed to the constructor, arguments
passed to the mocked subroutine are stored and accessible later via
"arguments" and "method_arguments". However sometimes this isn't
desirable, especially in cases where the reference count of items in
the arguments matter; notably when an object should be destroyed and
the destructor's behavior is important. When this is true set
store_arguments to a false value and only an empty array reference will
be stored.
When used as a setter, it returns the mock instance for chaining if
desired.
COOKBOOK
Run code before the original
The original version of the mocked function (read: the code that was
available via the symbol at the time the mock was initiated) is
available via the fully qualified symbol Mock::MonkeyPatch::ORIGINAL.
You can call this in your mock if for example you want to do some setup
before calling the function.
my $mock = $self->patch($symbol, sub {
# do some stuff before the original
do_mocked_stuff(@_);
# then call the original function/method
Mock::MonkeyPatch::ORIGINAL(@_);
});
Using ORIGINAL in a nonblocking environment
Since the ORIGINAL symbol is implemented via local if you want to call
it after leaving the scope you need to store a reference to the
function in a lexical.
my $mock = $self->patch($symbol, sub {
my @args = @_;
my $orig = \&Mock::MonkeyPatch::ORIGINAL;
Mojo::IOLoop->timer(1 => sub { $orig->(@args) });
});
SEE ALSO
* Test::MockObject
* Mock::Quick
* Mock::Sub
SOURCE REPOSITORY
http://github.com/jberger/Mock-MonkeyPatch
AUTHOR
Joel Berger, <
[email protected]>
CONTRIBUTORS
* Doug Bell (preaction)
* Brian Medley (bpmedley)
COPYRIGHT AND LICENSE
Copyright (C) 2016 by Joel Berger and "CONTRIBUTORS"
This library is free software; you can redistribute it and/or modify it
under the same terms as Perl itself.