I often find myself doing things like this when testing exception-based code.
use Test::More 'no_plan';
use Test::Exception;
my $result = lives_ok {$o->answer} 'answer worked';
is $result, 42, 'answer returned 42';
Update:
my $result;
lives_ok {$result = $o->answer} 'answer worked';
is $result, 42, 'answer returned 42';
This is a bit clumsy since to me it is conceptually one test ($o->answer does not throw an exception and returns 42).
Of course, you can write a custom test subroutine lives_and_is, but you can very quickly end up writing lives_and_isnt, lives_and_is_deeply, etc.
Is there a more generic way?
I'm considering adding a live subroutine to Test::Exception that you can use like this:
is live{$o->answer}, 42, 'answer worked';
Implemented by, when an exception is thrown, returning an object that overloads all the comparison operations to return false.
Documentation:
live
Returns the result of the given expression if no
exception is thrown.
If an exception is thrown an object is returned that
evaluates to false when used in any standard boolean
test or comparison operation ("<", "<=", ">", ">=",
"==", "!=", "lt", "le", "gt", "ge", "eq" and "ne").
This can be used to simplify testing for exceptions.
For example:
foreach my $n (3,2,1,0) {
is live{$n/$n}, 1, "$n/$n == 1";
};
will produce
ok 1 - 3/3 == 1
ok 2 - 2/2 == 1
ok 3 - 1/1 == 1
not ok 4 - 0/0 == 1
# Failed test (test.pl at line 37)
# got: 'DIED: Illegal division by zero at test.pl line 37
+.'
# expected: '1'
NOTE: that it is an object not a false value that is
returned when an exception occurred.
# this will pass even if foo dies
ok(refaddr live {$o->foo}, 'foo returns object';
# this will pass even if foo dies
unlike live {$o->foo}, '/fribble/', 'no fribble';
Appropriate care must be taken.
Implementation:
use Sub::Uplevel;
sub _exception_as_string {
my $exception = shift;
my $class = ref($exception);
$exception = "$class ($exception)"
if $class && "$exception" !~ m/^\Q$class/;
chomp($exception);
return($exception);
};
{
package Test::Exception::_NULL;
use overload
q{""} => sub { "${$_[0]}" },
map { $_ => sub { return } }
( qw( < <= > >= == != lt le gt ge eq ne bool ) ),
fallback => 1;
sub new { bless \$_[1], $_[0] };
};
sub live (&) {
my $coderef = shift;
my $value = eval { uplevel 2, $coderef };
$value = Test::Exception::_NULL->new(
"DIED: ". _exception_as_string($@)
) if $@;
$value;
};
Sound sane? Or is Test::Exception::_NULL too evil to be allowed out in public?
Alternate suggestions?
Update per ER request - dvergin 2003-01-23
-
Are you posting in the right place? Check out Where do I post X? to know for sure.
-
Posts may use any of the Perl Monks Approved HTML tags. Currently these include the following:
<code> <a> <b> <big>
<blockquote> <br /> <dd>
<dl> <dt> <em> <font>
<h1> <h2> <h3> <h4>
<h5> <h6> <hr /> <i>
<li> <nbsp> <ol> <p>
<small> <strike> <strong>
<sub> <sup> <table>
<td> <th> <tr> <tt>
<u> <ul>
-
Snippets of code should be wrapped in
<code> tags not
<pre> tags. In fact, <pre>
tags should generally be avoided. If they must
be used, extreme care should be
taken to ensure that their contents do not
have long lines (<70 chars), in order to prevent
horizontal scrolling (and possible janitor
intervention).
-
Want more info? How to link
or How to display code and escape characters
are good places to start.