Hook::LexWrap allows you to attach code to the start and end of subroutines that will be called every time that subroutine is run. While this has a multitude of uses, I use it mainly for debugging.
Sometimes, just sometimes, we inherit a whole bunch of terrible code that resembles the metaphorical plateful of spaghetti, something with an execution path as hard to follow as the cabling mess stuffed behind the television. You simply can't tell what's calling what, and what calls what.
The standard Perl approach to untangling this - short of actually breaking out the perl debugger - is to perforate the code with a smattering of print statements that print out "We're here" and "Now we're here" and maybe what values certain variables have at the time. This can be tedious and slow to develop, and can introduce as many bugs as you're trying to remove. By placing this debugging code in Hook::LexWrap wrappers, you can easily debug your code and speed up your development process.
So let's pretend we've got a whole bunch of code, of which one of the methods looks like this:
sub get_conf { my $self = shift; my $var = shift;
# check the user variables first return $self->{user_conf}{ $var } if exists $self->{user_conf}{ $var }; # if there wasn't one, return the normal config return $self->{conf}{ $var }; }
It's a simple chunk of code that checks first in one hash stored in our object for a value (the user_conf) and if it doesn't find a value in there, returns the value from the main conf hash. This code is called from all over our code like so:
my $filename = $obj->get_conf("filename"); my $vector = $obj->get_conf($direction);
As always when we write programs though, we don't get it right the
very first time. In this case we start noticing that some of the
configuration variables don't have the values in we expect. There's
several reasons this could be - our program could be calling
get_conf
with the wrong name (for example $direction
in the
above example could have an unexpected value.) Or our
various configuration hashes could have the wrong values within them.
One thing we could do is temporarily alter our code so that it prints out the package name and parameters whenever it's called.
sub get_conf { print STDERR "get_conf called from ".caller.", var='$_[1]'\n;"; my $self = shift; my $var = shift;
We also could alter our code so that it prints out whatever it returns:
# check the user variables first if (exists $self->{user_conf}{ $var }) { print STDERR "returning '$self->{user_conf}{ $var }'\n"; return $self->{user_conf}{ $var } } # if there wasn't one, return the normal config print STDERR "returning '$self->{conf}{ $var }'\n"; return $self->{conf}{ $var }; }
Note how we're printing to STDERR. This has two advantages - firstly, it means we can use shell shenanigans to redirect this debug information to a separate file so it doesn't interfere with our main code output, and secondly it, unlike STDOUT, is automatically printed out immediately, without any buffering, as soon as the print statement is called.
This technique will work - though it does have some disadvantages. The first two of these come from the fact that you're editing existing code. There are situations where it might be very hard to do this; if you're debugging your interactions with an existing module the code for that module's methods could be located in a file you don't have write permissions for, or it could be locked away inside a PAR archive. Worryingly, when you change code like this you might forget to put it back again (I know I always do) and you'll end up with everyone on the entire system having debug information printing out for their code evermore.
The second disadvantage we can see from the above example is that sometimes to put in debug statements we have to change the very logic of the code itself. In the case of the first return statement we had to reorder the if statement. And in both cases we had to copy the thing we were returning into the print statement. It's very easy to make a mistake when doing this and alter the very thing we're trying to examine. What we do not need in this situation is a chance for more bugs!
Hook::LexWrap allows you to attach subroutines to existing ones that will be run before or after (pre or post) the original subroutine being run.
wrap "MyCode::get_conf", pre => sub { print STDERR "get_conf called from ".caller.", var='$_[1]'\n"; }, post => sub { print STDERR "returning $_[0]\n"; };
(Note that in this example get_conf was declared in the MyCode package). This code completely replaces the modifications we made to the previous code, and can be declared in the script that calls MyCode::get_conf rather than in the MyCode module itself.
It's a lot easier to write this code - which will be run after we return
- than the debug statements in the modified get_conf
routine; we
don't have to worry about altering the flow of control and manually
printing whatever we were just about to return - we simply let the
original subroutine return normally and then print whatever is in the
@_
array passed to the post
subroutine.
One of the major advantages of the Hook::LexWrap module is that it can be used to wrap the code lexically if needed. That is to say, when the current wrapping statement falls out of scope - you reach the next closing curly bracket or reach the end of the file - the wrapping stops, and the original subroutine is returned.
In the altered version of get_conf
, before we implemented the
Hook::LexWrap wrappers, each and every single time get_conf
was
called it would print all the information. If we have a hundred or
so calls to this method then we'll have to dig all the way though
these each time we run the program until we find the one we want.
However, with the Hook::LexWrap mode we can turn on debugging for
just sections of our code.
Let's look at an example where we use our config module to get paths.
We use the special lexical form of calling wrap
where we assign it to a
variable. When this variable falls out of scope the wrapping is
undone.
sub get_system_paths { my $self = shift;
my $wrapping = wrap "get_conf", pre => sub { print STDERR "get_conf called from ".caller.", var='$_[1]'\n"; }, post => sub { print STDERR "returning $_[0]\n"; };
return $self->get_paths("system"); }
sub get_perl_paths { return $self->get_paths("perl"); }
sub get_paths { my $self = shift; my $name = shift; return $self->{obj}->get_conf($name . "_dir"), $self->{obj}->get_conf($name . "_filename"), $self->{obj}->get_conf($name . "_ext"); }
In this example, only when get_conf
is called via
get_system_paths
will it print any debug info out at all - because
when get_perl_paths
is called after get_system_paths
$wrapper
will have fallen out of scope and the wrapping will be undone.