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.