File Coverage

inc/Test/Exception.pm
Criterion Covered Total %
statement 62 99 62.6
branch 11 32 34.3
condition 2 9 22.2
subroutine 14 18 77.7
pod 4 4 100.0
total 93 162 57.4


line stmt bran cond sub pod time code
1 4     4   8416 #line 1
  4         8  
  4         175  
2 4     4   24 use strict;
  4         7  
  4         203  
3             use warnings;
4              
5 4     4   23 package Test::Exception;
  4         10  
  4         10382  
6 4     4   4224 use Test::Builder;
  4         6151  
  4         25  
7 4     4   519 use Sub::Uplevel qw( uplevel );
  4         8  
  4         2573  
8             use base qw( Exporter );
9              
10             our $VERSION = '0.31';
11             our @EXPORT = qw(dies_ok lives_ok throws_ok lives_and);
12              
13             my $Tester = Test::Builder->new;
14              
15 4     4   30 sub import {
16 4 50       28 my $self = shift;
17 0         0 if ( @_ ) {
18 0         0 my $package = caller;
19 0         0 $Tester->exported_to( $package );
20             $Tester->plan( @_ );
21 4         1659 };
22             $self->export_to_level( 1, $self, $_ ) foreach @EXPORT;
23             }
24              
25             #line 83
26              
27             sub _quiet_caller (;$) { ## no critic Prototypes
28             my $height = $_[0];
29             $height++;
30              
31             if ( CORE::caller() eq 'DB' ) {
32             # passthrough the @DB::args trick
33             package DB;
34             if( wantarray ) {
35             if ( !@_ ) {
36             return (CORE::caller($height))[0..2];
37             }
38             else {
39             # If we got here, we are within a Test::Exception test, and
40             # something is producing a stacktrace. In case this is a full
41             # trace (i.e. confess() ), we have to make sure that the sub
42             # args are not visible. If we do not do this, and the test in
43             # question is throws_ok() with a regex, it will end up matching
44             # against itself in the args to throws_ok().
45             #
46             # While it is possible (and maybe wise), to test if we are
47             # indeed running under throws_ok (by crawling the stack right
48             # up from here), the old behavior of Test::Exception was to
49             # simply obliterate @DB::args altogether in _quiet_caller, so
50             # we are just preserving the behavior to avoid surprises
51             #
52             my @frame_info = CORE::caller($height);
53             @DB::args = ();
54             return @frame_info;
55             }
56             }
57              
58             # fallback if nothing above returns
59             return CORE::caller($height);
60             }
61             else {
62             if( wantarray and !@_ ) {
63             return (CORE::caller($height))[0..2];
64             }
65             else {
66             return CORE::caller($height);
67             }
68             }
69             }
70              
71             sub _try_as_caller {
72             my $coderef = shift;
73              
74             # local works here because Sub::Uplevel has already overridden caller
75             local *CORE::GLOBAL::caller;
76             { no warnings 'redefine'; *CORE::GLOBAL::caller = \&_quiet_caller; }
77              
78             eval { uplevel 3, $coderef };
79             return $@;
80             };
81              
82              
83             sub _is_exception {
84             my $exception = shift;
85 81     81   7663 return ref $exception || $exception ne '';
86 81         96 };
87              
88 81 100       194  
89             sub _exception_as_string {
90             my ( $prefix, $exception ) = @_;
91 25 50       75 return "$prefix normal exit" unless _is_exception( $exception );
92 25 50       50 my $class = ref $exception;
93 0         0 $exception = "$class ($exception)"
94             if $class && "$exception" !~ m/^\Q$class/;
95             chomp $exception;
96             return "$prefix $exception";
97             };
98              
99              
100             #line 206
101              
102              
103             sub throws_ok (&$;$) {
104             my ( $coderef, $expecting, $description ) = @_;
105             unless (defined $expecting) {
106             require Carp;
107             Carp::croak( "throws_ok: must pass exception class/object or regex" );
108             }
109 25         144 $description = _exception_as_string( "threw", $expecting )
110 25         47 unless defined $description;
111 25         208 my $exception = _try_as_caller( $coderef );
112             my $regex = $Tester->maybe_regex( $expecting );
113             my $ok = $regex
114             ? ( $exception =~ m/$regex/ )
115             : eval {
116 0         0 $exception->isa( ref $expecting ? ref $expecting : $expecting )
117             };
118             $Tester->ok( $ok, $description );
119 56 50 33     296 unless ( $ok ) {
120 0         0 $Tester->diag( _exception_as_string( "expecting:", $expecting ) );
121             $Tester->diag( _exception_as_string( "found:", $exception ) );
122             };
123 56         398 $@ = $exception;
124             return $ok;
125             };
126              
127              
128             #line 254
129 9     9   22  
130             sub dies_ok (&;$) {
131             my ( $coderef, $description ) = @_;
132 9         38 my $exception = _try_as_caller( $coderef );
133 4     4   29 my $ok = $Tester->ok( _is_exception($exception), $description );
  4         32  
  4         3274  
  9         18  
  9         31  
134             $@ = $exception;
135 9         17 return $ok;
  9         41  
136 9         10236 }
137              
138              
139             #line 293
140              
141 5     5   11 sub lives_ok (&;$) {
142 5   33     106 my ( $coderef, $description ) = @_;
143             my $exception = _try_as_caller( $coderef );
144             my $ok = $Tester->ok( ! _is_exception( $exception ), $description );
145             $Tester->diag( _exception_as_string( "died:", $exception ) ) unless $ok;
146             $@ = $exception;
147 0     0   0 return $ok;
148 0 0       0 }
149 0         0  
150 0 0 0     0  
151             #line 333
152 0         0  
153 0         0 sub lives_and (&;$) {
154             my ( $test, $description ) = @_;
155             {
156             local $Test::Builder::Level = $Test::Builder::Level + 1;
157             my $ok = \&Test::Builder::ok;
158             no warnings;
159             local *Test::Builder::ok = sub {
160             $_[2] = $description unless defined $_[2];
161             $ok->(@_);
162             };
163             use warnings;
164             eval { $test->() } and return 1;
165             };
166             my $exception = $@;
167             if ( _is_exception( $exception ) ) {
168             $Tester->ok( 0, $description );
169             $Tester->diag( _exception_as_string( "died:", $exception ) );
170             };
171             $@ = $exception;
172             return;
173             }
174              
175             #line 502
176              
177             1;