NAME
Capture::Attribute - s/return/print/g
SYNOPSIS
use Capture::Attribute;
sub foobar :Capture {
print "Hello World\n";
}
my $result = foobar();
$result =~ s/World/Planet/;
print "$result"; # says "Hello Planet"
DESCRIPTION
Sometimes you write a function that needs to build a long string via a convoluted series of conditional statements, loops and so on. I tend to end up defining a variable $return
at the top of the code, concatenating bits to it as required, and then return it at the end. For example:
sub count_to_10 {
my $return = "Listen to me count!\n";
foreach (1..10) {
$return .= "$_\n";
$return .= "Half-way there!\n" if $_==5;
}
$return .= "All done!\n";
return $return;
}
Mail::Message->new(
To => 'teacher@example.com',
From => 'student@example.com',
Subject => 'I can count!',
data => count_to_ten(),
)->send;
Capture::Attribute simplifies this pattern by capturing all output to STDOUT, so you can use STDOUT as a place to capture each part of the string.
sub count_to_10 :Capture {
say "Listen to me count!";
foreach (1..10) {
say $_;
say "Half-way there!" if $_==5;
}
say "All done!";
}
Mail::Message->new(
To => 'teacher@example.com',
From => 'student@example.com',
Subject => 'I can count!',
data => count_to_ten(),
)->send;
Doesn't that look nicer?
Within a sub marked with the ":Capture" attribute, all data that would be printed is captured instead. When the sub is finished, the return value is ignored and the captured text is returned instead.
The return
keyword still works just fine for its control flow purpose inside a captured sub. The return value just doesn't get returned.
How does it work?
When you use Capture::Attributes
, then at BEGIN time (see perlmod) your package will be automatically made into an subclass of Capture::Attributes.
At CHECK time (again perlmod), Capture::Attributes will then use Attribute::Handlers to wrap each sub marked with the ":Capture" attribute with a sub that captures its output via Capture::Tiny, and returns the output.
At INIT time (again perlmod), Capture::Attributes then removes itself from your package's @ISA
, thus your package is no longer a subclass of Capture::Attributes. (It would be nice if the subclassing could be avoided altogether, but alas this seems to be the way Attribute::Handlers works.)
The ":Capture" Attribute
There are actually various options you can use on the ":Capture" attribute. They are mostly useless.
:Capture(STDOUT)
This is the default. Captures STDOUT.
:Capture(STDERR)
Captures STDERR instead of STDOUT.
:Capture(MERGED)
Captures both STDOUT and STDERR, merged into one. Because of buffering, lines from different handles may interleave differently than expected.
:Capture(STDOUT,STDERR)
Capture both STDOUT and STDERR. In scalar context, returns STDOUT. In List context returns both.
sub foo :Capture(STDOUT,STDERR) {
print "World\n";
warn "Hello\n";
}
my ($hello, $world) = map { chomp;$_ } foo();
:Capture(STDERR,STDOUT)
Capture both STDOUT and STDERR. In scalar context, returns STDERR. In List context returns both.
CAVEATS
Subclassing
As mentioned above, Capture::Attributes temporarily installs itself as a superclass of your class. If your class has subs named any of the following, they may override the Capture::Attributes versions, and bad stuff may happen.
ATTR
Capture
MODIFY_CODE_ATTRIBUTES
any sub matching the expresssion
/^_ATTR_CODE_/
Accessing the real return value
sub quux :Capture
{
print "foo";
return "bar";
}
say quux(); # says "foo"
say Capture::Attributes->return->value; # says "bar"
The Capture::Attributes->return
class method gives you the real return value from the most recently captured sub. This is a Capture::Attribute::Return object.
However, this section is listed under CAVEATS for a good reason. The fact that a sub happens to use the ":Capture" attribute should be considered private to it. The caller shouldn't consider there to be any difference between:
sub foo :Capture { print "foo" }
and
sub foo { return "foo" }
If the caller of the captured sub goes on to inspect Capture::Attributes->return
, then this assumes an implementation detail of the captured sub, which breaks encapsulation.
Adding a ":Capture" attribute to somebody else's function
So you want to do something like:
add_attribute(\&Some::Module::function, ':Capture(STDOUT)');
Here's how:
# Declare a generic wrapper
sub CAP :Capture { (shift)->(@_) }
# Wrap Some::Module::function in our wrapper.
my $orig = \&Some::Module::function;
local *Some::Module::function = sub { CAP($orig, @_) };
Though you are probably better off investigating Capture::Tiny.
Call stack
Capture::Attribute adds two extra frames to the call stack, and Capture::Tiny adds (it seems) two more again. So any code that you capture will see them quite clearly in the call stack if they decide to look. They'll show up in stack traces, etc.
BUGS
Please report any bugs to http://rt.cpan.org/Dist/Display.html?Queue=Capture-Attribute.
SEE ALSO
AUTHOR
Toby Inkster <tobyink@cpan.org>.
COPYRIGHT AND LICENCE
This software is copyright (c) 2012 by Toby Inkster.
This is free software; you can redistribute it and/or modify it under the same terms as the Perl 5 programming language system itself.
DISCLAIMER OF WARRANTIES
THIS PACKAGE IS PROVIDED "AS IS" AND WITHOUT ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR A PARTICULAR PURPOSE.