# Copyright (c) 2001-2008, Aruba Networks, Inc. # This library is free software; you can redistribute it and/or modify it # under the same terms as Perl itself. use strict; use warnings; package Test::Resub; use base qw(Exporter); use Carp qw(croak); use Class::Std; use Storable qw(dclone); BEGIN { local $@; eval { require Scalar::Util; Scalar::Util->import(qw(set_prototype)); }; if ($@) { *set_prototype = sub { $_[0] }; } } our @EXPORT_OK = qw(resub); our $VERSION = '1.06'; my %name :ATTR( :init_arg ); my %capture :ATTR( :init_arg, :default(0) ); my %call_mode :ATTR( :init_arg, :default('required') ); my %deep_copy :ATTR( :init_arg, :default(1) ); my %old_method :ATTR; my %new_method :ATTR; my %called :ATTR( :default(0) ); my %was_called :ATTR( :default(0) ); my %method_args :ATTR; sub resub { my ($name, $code, %args) = @_; return Test::Resub->new({ name => $name, code => $code, %args, }); } sub default_replacement_sub { sub {} } sub _get_prototype_safely { my ($func) = @_; return undef unless defined $func; my $prototype; local $@; eval { $prototype = prototype $func }; return $prototype; } sub BUILD { my ($self, $ident, $args) = @_; local $Carp::Internal{'Class::Std'} = 1; local $Carp::Internal{ do {__PACKAGE__} } = 1; my $code = $args->{code} || $self->default_replacement_sub; my $method = $args->{name}; unless ($method =~ /^(\w+::)+\w+$/) { croak qq{bad method name "$method"!}; } my $call_mode = $args->{call}; my %is_valid = map { ($_ => 1) } qw(forbidden optional required); if ($call_mode and not $is_valid{$call_mode}) { croak sprintf(q{bad 'call' argument: %s (valid arguments are %s)}, $call_mode, (join q{, }, sort keys %is_valid)); } my ($package, $func) = _split_package_method($method); if ((not UNIVERSAL::can($package, $func)) && (not $args->{create})) { croak sprintf(q{Package %s doesn't have a %s; Test::Resub will not create} . q{ a method without the 'create' flag.}, $package, $func, ); } $new_method{$ident} = $code; $method_args{$ident} = []; } sub START { my ($self, $ident, $args) = @_; my $method = $name{$ident}; my $wrapper_code = set_prototype(sub { $called{$ident}++; $was_called{$ident}++; if ($capture{$ident}) { local $Storable::Deparse = 1; local $Storable::Eval = 1; my $saved_args = $deep_copy{$ident} ? dclone([@_]) : [@_]; push @{$method_args{$ident}}, $saved_args; } return $new_method{$ident}->(@_); }, _get_prototype_safely($method)); { no strict 'refs'; no warnings 'redefine'; my $orig_data = _save_variables($method); if (defined *$method{CODE}) { $old_method{$ident} = \&$method; } *$method = $wrapper_code; _restore_variables($method, $orig_data); } } sub DEMOLISH { my ($self, $ident) = @_; my $method = $name{$ident}; return unless $method; # happens if BUILD throws exception my $was_called = $was_called{$ident}; if ( ($call_mode{$ident} eq 'forbidden' and $was_called) || ($call_mode{$ident} eq 'required' and not $was_called) ) { print "not ok 1000\n"; local $Carp::CarpLevel = $Carp::CarpLevel + 1; print '# ' . $method . ' ' . ($was_called ? 'called' : 'not called') . "\n" . Carp::longmess; } { no strict 'refs'; no warnings 'redefine'; my $existing_data = _save_variables($method); if (not defined $old_method{$ident}) { my ($package, $name) = _split_package_method($method); delete ${ "$package\::" }{$name}; } else { *$method = $old_method{$ident}; } _restore_variables($method, $existing_data); } } sub called { my ($self) = @_; return $called{ident $self}; } sub was_called { my ($self) = @_; return $was_called{ident $self}; } sub not_called { my ($self) = @_; return not $self->called; } sub reset { my ($self) = @_; my $ident = ident $self; $called{$ident} = 0; $method_args{$ident} = []; } sub make_key { my ($item) = @_; my $ref = ref($item); return $item unless $ref; return $$item if $ref eq 'SCALAR'; return join ',', map { make_key($_) } @$item if $ref eq 'ARRAY'; return join ',', map { $_ => make_key($item->{$_}) } sort keys %$item if $ref eq 'HASH'; return $ref; # hokey; all objects of same class get treated as same thing } sub unique { my (@list) = @_; my %seen; return grep { my $key = make_key($_); ! $seen{$key}++ } @list; } sub unique_args { return unique @{shift->args} } sub args { my ($self) = @_; my $ident = ident $self; warn "Must use the 'capture' flag to capture arguments" unless $capture{$ident}; return $method_args{$ident}; } sub unique_named_args { return unique @{shift->named_args} } sub named_args { my ($self, %args) = @_; return $self->_named_things( sub { $self->args() }, %args, ); } sub unique_method_args { return unique @{shift->method_args} } sub method_args { my ($self) = @_; my $args = $self->args; return [map { my @tmp = @$_; shift @tmp; \@tmp } @$args]; } sub unique_named_method_args { return unique @{shift->named_method_args} } sub named_method_args { my ($self, %args) = @_; return $self->_named_things( sub { $self->method_args() }, %args, ); } sub _named_things { my ($self, $retriever, %args) = @_; my $index = $args{arg_start_index} || $args{scalars} || 0; my $start_index = exists $args{arg_start_index} ? $index : 0; return [ map { (@$_[$start_index..$index-1], { @$_[$index..$#$_] }) } @{$retriever->()} ]; } sub _save_variables { my ($varname) = @_; no strict 'refs'; return { scalar => $$varname, array => \@$varname, hash => \%$varname, }; } sub _restore_variables { my ($varname, $data) = @_; no strict 'refs'; $$varname = $data->{scalar}; @$varname = @{$data->{array}}; %$varname = %{$data->{hash}}; } sub _split_package_method { my ($method) = @_; my ($package, $name) = $method =~ /^(.+)::([^:]+)$/; return ($package, $name); } 1; __END__ =head1 NAME Test::Resub - Lexically scoped subroutine replacement for testing =head1 SYNOPSIS #!/usr/bin/perl use Test::More tests => 4; use Test::Resub qw(resub); { package Somewhere; sub show { my ($class, $message) = @_; return "$class, $message"; } } # sanity is( Somewhere->show('beyond the sea'), 'Somewhere, beyond the sea' ); # scoped replacement of subroutine with argument capturing { my $rs = resub 'Somewhere::show', sub { 'hi' }, capture => 1; is( Somewhere->show('over the rainbow'), 'hi' ); is_deeply( $rs->method_args, [['over the rainbow']] ); } # scope ends, resub goes away, original code returns is( Somewhere->show('waiting for me'), 'Somewhere, waiting for me' ); =head1 DESCRIPTION This module allows you to temporarily replace a subroutine/method with arbitrary code. Later, you can tell how many times was it called and with what arguments each time. You can also specify that the subroutine/method must get called, must not get called, or may be optionally called. =head1 CONSTRUCTOR use Test::Resub qw(resub); my $rs = resub 'package::method', sub { ... }, %args; is equivalent to: use Test::Resub; my $rs = Test::Resub->new({ name => 'package::method', code => sub { ... }, %args, }); C<%args> can be any of the following named arguments: =over 4 =item B The function/method which is to be replaced. =item B The code reference which will replace C. Defaults to C =item B Boolean which indicates whether or not arguments should be captured. A warning is emitted if you try to look at args without specifying a "true" C. Defaults to 0. =item B One of the following values (defaults to 'required'): =over 4 =item B If the subroutine/method was never called when the Test::Resub object is destroyed, "not ok 1000" is printed to STDOUT. =item B If the subroutine/method was called when the Test::Resub object is destroyed, "not ok 1000" is printed to STDOUT. =item B It doesn't matter if the subroutine/method gets called. As a general rule, your tests should know whether or not a subroutine/method is going to get called, so avoid using this option if you can. =back =item B Boolean which indicates whether or not a function will be created if none exists. If the package can't resolve the method (i.e. ! UNIVERSAL::can($package, $method)), then an exception will be thrown unless 'create' is true. Defaults to false. This is mainly useful to catch typos. =item B Whether or not to make a deep copy of saved-off arguments. Default is 1. Normally, one wants deep copies, but there is an associated performance penalty, e.g. for large objects. =back =head1 METHODS =over 4 =item B Returns the number of times the replaced subroutine/method was called. The C method clears this data. =item B Returns the total number of times the replaced subroutine/method was called. This data is B cleared by the C method. =item B Returns true if the replaced subroutine/method was never called. The C method clears this data. =item B Clears the C, C, and C data. =item B Returns data on how the replaced subroutine/method was invoked. Examples: Invocations: C returns: ---------------------------- ------------------------- (none) [] foo('a'); [['a']] foo('a', 'b'); foo('d'); [['a', 'b'], ['d']] =item B Similar to C, but only returns unique argument calls. Given: foo('a', 'b'); foo(1, 2); foo('a', 'b'); C returns: [['a', 'b'], [1, 2]]; Objects of the same class are considered the same object. =item B Like C, but each invocation's arguments are returned in a hashref. Examples: Invocations: C returns: ---------------------------- ------------------------- (none) [] foo(a => 'b'); [{a => 'b'}] foo(a => 'b', c => 'd'); foo(e => 'f'); [{ a => 'b', c => 'd', }, { e => 'f', }] The C argument specifes that a certain number of arguments are to be discarded. For example: my $rs = resub 'some_sub'; ... some_sub('one', 'two', a => 1, b => 2); ... $rs->named_args(arg_start_index => 1); # returns ['two', {a => 1, b => 2}] $rs->named_args(arg_start_index => 2); # returns [{a => 1, b => 2}] The C argument specifies that a certain number of scalar arguments precede the key/value arguments. For example: my $rs = resub 'some_sub'; ... some_sub(3306, a => 'b', c => 123); some_sub(9158, a => 'z', c => 456); ... $rs->named_args(scalars => 1); # returns [3306, {a => 'b', c => 123}, # 9158, {a => 'z', c => 456}] Note that C N)> will yield N scalars plus one hashref per call regardless of how many arguments were passed to the subroutine/method. For example: my $rs = Test::Resub->new({name => 'some_sub'}); ... some_sub('one argument only'); some_sub('many', 'arguments', a => 1, b => 2); ... $rs->named_args(scalars => 2); # returns ['one argument only', undef, {}, # 'many', 'arguments', {a => 1, b => 2}] =item B The C version of C =item B Like C, but the first argument of each invocation is thrown away. This is used when you're resub'ing an object or class method and you're not interested in testing the object or class argument. Examples: Invocations: C returns: ---------------------------- ------------------------- (none) [] $obj->foo('a'); [['a']] Class->foo('a', 'b'); Class->foo('d'); [['a', 'b'], ['d']] =item B The C version of C =item B Like C, but the first argument of each invocation is thrown away. This is used when you're resub'ing an object or class method and the arguments are name/value pairs. Examples: Invocations: C returns: ---------------------------- ------------------------- (none) [] $obj->foo(a => 'b'); [{a => 'b'}] $obj->foo(a => 'b', c => 'd'); [{ Class->foo(e => 'f'); a => 'b', c => 'd', }, { e => 'f', }] C also takes a "scalars" named argument which specifies a number of scalar arguments preceding the name/value pairs of each invocation. It works just like C except that the first argument of each invocation is automatically discarded. For example: my $rs = resub 'SomeClass::some_sub'; ... SomeClass->some_sub(3306, a => 'b', c => 123); SomeClass->some_sub(9158, a => 'z', c => 456); ... $rs->named_method_args(scalars => 1); # returns [3306, {a => 'b', c => 123}, # 9158, {a => 'z', c => 456}] Note: the first argument is automatically discarded B the optional C parameter is applied. That is, my $rs = resub 'SomeClass::some_sub'; ... SomeClass->some_sub('first', b => 2); ... $rs->named_method_args(arg_start_index => 1); # returns [{b => 2}] =item B The C version of C =back =head1 HISTORY Written at AirWave Wireless for internal testing, 2001-2007. Tidied up and released to CPAN in 2007. =head1 AUTHORS The development team at AirWave Wireless, L. Please direct questions, comments, bugs, patches, etc. to F. =cut