File Coverage

blib/lib/Test/Spec/Example.pm
Criterion Covered Total %
statement 50 79 63.2
branch 4 18 22.2
condition 4 11 36.3
subroutine 15 16 93.7
pod 0 8 0.0
total 73 132 55.3


line stmt bran cond sub pod time code
1             package Test::Spec::Example;
2              
3             # Purpose: represents an `it` block
4              
5 14     14   87 use strict;
  14         39  
  14         378  
6 14     14   74 use warnings;
  14         29  
  14         337  
7              
8             ########################################################################
9             # NO USER-SERVICEABLE PARTS INSIDE.
10             ########################################################################
11              
12 14     14   68 use Carp ();
  14         25  
  14         196  
13 14     14   68 use Scalar::Util ();
  14         30  
  14         4131  
14              
15             sub new {
16 122     122 0 1114 my ($class, $args) = @_;
17              
18 122 50 33     517 if (!$args || ref($args) ne 'HASH') {
19 0         0 Carp::croak "usage: $class->new(\\%args)";
20             }
21              
22 122         237 my $self = bless {}, $class;
23 122         224 foreach my $attr ( qw/name description code builder context/ ) {
24 610   33     1330 $self->{$attr} = $args->{$attr} || Carp::croak "$attr missing";
25             }
26              
27 122         331 Scalar::Util::weaken($self->{context});
28              
29 122         265 return $self;
30             }
31              
32 122     122 0 600 sub name { shift->{name} }
33 119     119 0 463 sub description { shift->{description} }
34 122     122 0 368 sub code { shift->{code} }
35 0     0 0 0 sub builder { shift->{builder} }
36 122     122 0 296 sub context { shift->{context} }
37              
38             # Build a stack from the starting context
39             # down to the current context
40             sub stack {
41 122     122 0 217 my ($self) = @_;
42              
43 122         258 my $ctx = $self->context;
44              
45 122         258 my @ancestors = $ctx;
46 122         299 while ( $ctx = $ctx->parent ) {
47 222         406 push @ancestors, $ctx;
48             }
49              
50 122         401 return reverse(@ancestors);
51             }
52              
53             sub run {
54 122     122 0 244 my ($self) = @_;
55              
56             # clobber Test::Builder's ok() method just like Test::Class does,
57             # but without screwing up underscores.
58 14     14   89 no warnings 'redefine';
  14         30  
  14         7189  
59 122         241 my $orig_builder_ok = \&Test::Builder::ok;
60             local *Test::Builder::ok = sub {
61 121     121   178593 my ($builder,$test,$desc) = splice(@_,0,3);
62 121   66     551 $desc ||= $self->description;
63 121         217 local $Test::Builder::Level = $Test::Builder::Level+1;
64 121         399 $orig_builder_ok->($builder, $test, $desc, @_);
65 122         676 };
66              
67             # Run the test
68 122         331 eval { $self->_runner($self->stack) };
  122         411  
69              
70             # And trap any errors
71 122 50       1142 if (my $err = $@) {
72 0         0 my $builder = $self->builder;
73 0         0 my $description = $self->description;
74              
75             # eval in case stringification overload croaks
76 0   0     0 chomp($err = eval { $err . '' } || 'unknown error');
77 0         0 my ($file,$line);
78 0 0       0 ($file,$line) = ($1,$2) if ($err =~ s/ at (.+?) line (\d+)\.\Z//);
79              
80             # disable ok()'s diagnostics so we can generate a custom TAP message
81 0         0 my $old_diag = $builder->no_diag;
82 0         0 $builder->no_diag(1);
83             # make sure we can restore no_diag
84 0         0 eval { $builder->ok(0, $description) };
  0         0  
85 0         0 my $secondary_err = $@;
86             # no_diag needs a defined value, so double-negate it to get either '' or 1
87 0         0 $builder->no_diag(!!$old_diag);
88              
89 0 0       0 unless ($builder->no_diag) {
90             # emulate Test::Builder::ok's diagnostics, but with more details
91 0         0 my ($msg,$diag_fh);
92 0 0       0 if ($builder->in_todo) {
93 0         0 $msg = "Failed (TODO)";
94 0         0 $diag_fh = $builder->todo_output;
95             }
96             else {
97 0         0 $msg = "Failed";
98 0         0 $diag_fh = $builder->failure_output;
99             }
100 0 0       0 print {$diag_fh} "\n" if $ENV{HARNESS_ACTIVE};
  0         0  
101 0         0 print {$builder->failure_output} qq[# $msg test '$description' by dying:\n];
  0         0  
102 0         0 print {$builder->failure_output} qq[# $err\n];
  0         0  
103 0 0       0 print {$builder->failure_output} qq[# at $file line $line.\n] if defined($file);
  0         0  
104             }
105 0 0       0 die $secondary_err if $secondary_err;
106             }
107             }
108              
109             sub _runner {
110 344     344   690 my ($self, $ctx, @remainder) = @_;
111              
112             # This recursive closure essentially does this
113             # $outer->contextualize {
114             # $outer->before_each
115             # $inner->contextualize {
116             # $inner->before_each
117             # $anon->contextualize {
118             # $anon->before_each (no-op)
119             # execute test
120             # $anon->after_each (no-op)
121             # }
122             # $inner->after_each
123             # }
124             # $outer->after_each
125             # }
126             #
127             return $ctx->contextualize(sub {
128 344     344   765 $ctx->_run_before_all_once;
129 344         688414 $ctx->_run_before('each');
130 344 100       1215 if ( @remainder ) {
131 222         431 $self->_runner(@remainder);
132             }
133             else {
134 122         306 $ctx->_in_anonymous_context($self->code, $self);
135             }
136 344         1219 $ctx->_run_after('each');
137             # "after 'all'" only happens during context destruction (DEMOLISH).
138             # This is the only way I can think to make this work right
139             # in the case that only specific test methods are run.
140             # Otherwise, the global teardown would only happen when you
141             # happen to run the last test of the context.
142 344         1393 }, $self);
143             }
144              
145             1;