File Coverage

inc/Test/Base.pm
Criterion Covered Total %
statement 214 364 58.7
branch 48 178 26.9
condition 15 74 20.2
subroutine 37 60 61.6
pod 0 26 0.0
total 314 702 44.7


line stmt bran cond sub pod time code
1             #line 1
2             # TODO:
3             #
4 1     1   669 package Test::Base;
  1         3  
  1         51  
5 1     1   666 use 5.006001;
  1         63  
  1         9  
6 1     1   16 use Spiffy 0.30 -Base;
  1     1   3  
  1     1   53  
  1         6  
  1         2  
  1         43  
  1         5  
  1         3  
  1         8  
7             use Spiffy ':XXX';
8             our $VERSION = '0.55';
9              
10             my @test_more_exports;
11 1     1   27 BEGIN {
12             @test_more_exports = qw(
13             ok isnt like unlike is_deeply cmp_ok
14             skip todo_skip pass fail
15             eq_array eq_hash eq_set
16             plan can_ok isa_ok diag
17             use_ok
18             $TODO
19             );
20             }
21 1     1   991  
  1         4  
  1         24  
22 1     1   9 use Test::More import => \@test_more_exports;
  1         3  
  1         644  
23             use Carp;
24              
25             our @EXPORT = (@test_more_exports, qw(
26             is no_diff
27              
28             blocks next_block first_block
29             delimiters spec_file spec_string
30             filters filters_delay filter_arguments
31             run run_compare run_is run_is_deeply run_like run_unlike
32             WWW XXX YYY ZZZ
33             tie_output no_diag_on_only
34              
35             find_my_self default_object
36              
37             croak carp cluck confess
38             ));
39              
40             field '_spec_file';
41             field '_spec_string';
42             field _filters => [qw(norm trim)];
43             field _filters_map => {};
44             field spec =>
45             -init => '$self->_spec_init';
46             field block_list =>
47             -init => '$self->_block_list_init';
48             field _next_list => [];
49             field block_delim =>
50             -init => '$self->block_delim_default';
51             field data_delim =>
52             -init => '$self->data_delim_default';
53             field _filters_delay => 0;
54             field _no_diag_on_only => 0;
55              
56             field block_delim_default => '===';
57             field data_delim_default => '---';
58              
59             my $default_class;
60             my $default_object;
61             my $reserved_section_names = {};
62 1002     1002 0 1617  
63 1002   66     2956 sub default_object {
64 1002         3175 $default_object ||= $default_class->new;
65             return $default_object;
66             }
67              
68             my $import_called = 0;
69 1     1   10 sub import() {
70 1 50       10 $import_called = 1;
71             my $class = (grep /^-base$/i, @_)
72             ? scalar(caller)
73 1 50       4 : $_[0];
74 1         2 if (not defined $default_class) {
75             $default_class = $class;
76             }
77             # else {
78             # croak "Can't use $class after using $default_class"
79             # unless $default_class->isa($class);
80             # }
81 1 50       5  
82 1         2 unless (grep /^-base$/i, @_) {
83 1         6 my @args;
84 0 0       0 for (my $ii = 1; $ii <= $#_; ++$ii) {
85 0         0 if ($_[$ii] eq '-package') {
86             ++$ii;
87 0         0 } else {
88             push @args, $_[$ii];
89             }
90 1 50       4 }
91             Test::More->import(import => \@test_more_exports, @args)
92             if @args;
93             }
94 1         4
95 1         24 _strict_warnings();
96             goto &Spiffy::import;
97             }
98              
99             # Wrap Test::Builder::plan
100             my $plan_code = \&Test::Builder::plan;
101             my $Have_Plan = 0;
102 1     1   7 {
  1         2  
  1         4142  
103             no warnings 'redefine';
104 1     1   3 *Test::Builder::plan = sub {
105 1         10 $Have_Plan = 1;
106             goto &$plan_code;
107             };
108             }
109              
110             my $DIED = 0;
111             $SIG{__DIE__} = sub { $DIED = 1; die @_ };
112 500     500 0 617  
  500         8483  
113 2000     2000 0 3554 sub block_class { $self->find_class('Block') }
  2000         5905  
114             sub filter_class { $self->find_class('Filter') }
115 2500     2500 0 3272  
116 2500         19168 sub find_class {
117 2500         19167 my $suffix = shift;
118 2500 100       13446 my $class = ref($self) . "::$suffix";
119 1         3 return $class if $class->can('new');
120 1 50       8 $class = __PACKAGE__ . "::$suffix";
121 1         97 return $class if $class->can('new');
122 1 50       17 eval "require $class";
123 0         0 return $class if $class->can('new');
124             die "Can't find a class for $suffix";
125             }
126 0     0 0 0  
127 0 0       0 sub check_late {
128 0         0 if ($self->{block_list}) {
129 0         0 my $caller = (caller(1))[3];
130 0         0 $caller =~ s/.*:://;
131             croak "Too late to call $caller()"
132             }
133             }
134              
135 1002 50   1002 0 5327 sub find_my_self() {
136             my $self = ref($_[0]) eq $default_class
137             ? splice(@_, 0, 1)
138 1002         4389 : default_object();
139             return $self, @_;
140             }
141              
142 1     1 0 10 sub blocks() {
143             (my ($self), @_) = find_my_self(@_);
144 1 50       7  
145             croak "Invalid arguments passed to 'blocks'"
146 1 50 33     7 if @_ > 1;
147             croak sprintf("'%s' is invalid argument to blocks()", shift(@_))
148             if @_ && $_[0] !~ /^[a-zA-Z]\w*$/;
149 1         38  
150             my $blocks = $self->block_list;
151 1   50     13
152 0         0 my $section_name = shift || '';
153 1 50       106 my @blocks = $section_name
154             ? (grep { exists $_->{$section_name} } @$blocks)
155             : (@$blocks);
156 1 50       35  
157             return scalar(@blocks) unless wantarray;
158 0 0       0
159             return (@blocks) if $self->_filters_delay;
160 0         0  
161 0 0       0 for my $block (@blocks) {
162             $block->run_filters
163             unless $block->is_filtered;
164             }
165 0         0  
166             return (@blocks);
167             }
168              
169 0     0 0 0 sub next_block() {
170 0         0 (my ($self), @_) = find_my_self(@_);
171 0 0       0 my $list = $self->_next_list;
172 0         0 if (@$list == 0) {
  0         0  
173 0         0 $list = [@{$self->block_list}, undef];
174             $self->_next_list($list);
175 0         0 }
176 0 0 0     0 my $block = shift @$list;
177 0         0 if (defined $block and not $block->is_filtered) {
178             $block->run_filters;
179 0         0 }
180             return $block;
181             }
182              
183 0     0 0 0 sub first_block() {
184 0         0 (my ($self), @_) = find_my_self(@_);
185 0         0 $self->_next_list([]);
186             $self->next_block;
187             }
188              
189 0     0 0 0 sub filters_delay() {
190 0 0       0 (my ($self), @_) = find_my_self(@_);
191             $self->_filters_delay(defined $_[0] ? shift : 1);
192             }
193              
194 0     0 0 0 sub no_diag_on_only() {
195 0 0       0 (my ($self), @_) = find_my_self(@_);
196             $self->_no_diag_on_only(defined $_[0] ? shift : 1);
197             }
198              
199 0     0 0 0 sub delimiters() {
200 0         0 (my ($self), @_) = find_my_self(@_);
201 0         0 $self->check_late;
202 0   0     0 my ($block_delimiter, $data_delimiter) = @_;
203 0   0     0 $block_delimiter ||= $self->block_delim_default;
204 0         0 $data_delimiter ||= $self->data_delim_default;
205 0         0 $self->block_delim($block_delimiter);
206 0         0 $self->data_delim($data_delimiter);
207             return $self;
208             }
209              
210 0     0 0 0 sub spec_file() {
211 0         0 (my ($self), @_) = find_my_self(@_);
212 0         0 $self->check_late;
213 0         0 $self->_spec_file(shift);
214             return $self;
215             }
216              
217 0     0 0 0 sub spec_string() {
218 0         0 (my ($self), @_) = find_my_self(@_);
219 0         0 $self->check_late;
220 0         0 $self->_spec_string(shift);
221             return $self;
222             }
223              
224 0     0 0 0 sub filters() {
225 0 0       0 (my ($self), @_) = find_my_self(@_);
226 0         0 if (ref($_[0]) eq 'HASH') {
227             $self->_filters_map(shift);
228             }
229 0         0 else {
230 0         0 my $filters = $self->_filters;
231             push @$filters, @_;
232 0         0 }
233             return $self;
234             }
235              
236 0     0 0 0 sub filter_arguments() {
237             $Test::Base::Filter::arguments;
238             }
239 0     0 0 0  
240 0 0 0     0 sub have_text_diff {
  0         0  
  0         0  
241             eval { require Text::Diff; 1 } &&
242             $Text::Diff::VERSION >= 0.35 &&
243             $Algorithm::Diff::VERSION >= 1.15;
244             }
245              
246 1000     1000 0 3072 sub is($$;$) {
247 1000         2069 (my ($self), @_) = find_my_self(@_);
248 1000         1788 my ($actual, $expected, $name) = @_;
249 1000 50 33     13563 local $Test::Builder::Level = $Test::Builder::Level + 1;
      33        
      33        
      33        
      33        
250             if ($ENV{TEST_SHOW_NO_DIFFS} or
251             not defined $actual or
252             not defined $expected or
253             $actual eq $expected or
254             not($self->have_text_diff) or
255             $expected !~ /\n./s
256 1000         5863 ) {
257             Test::More::is($actual, $expected, $name);
258             }
259 0 0       0 else {
260 0         0 $name = '' unless defined $name;
261             ok $actual eq $expected,
262             $name . "\n" . Text::Diff::diff(\$expected, \$actual);
263             }
264             }
265              
266 1     1 0 6 sub run(&;$) {
267 1         2 (my ($self), @_) = find_my_self(@_);
268 1         3 my $callback = shift;
  1         36  
269 500 50       18831 for my $block (@{$self->block_list}) {
270 500         804 $block->run_filters unless $block->is_filtered;
  500         1762  
271             &{$callback}($block);
272             }
273             }
274              
275 0     0   0 my $name_error = "Can't determine section names";
276 0 0       0 sub _section_names {
277 0 0       0 return @_ if @_ == 2;
278             my $block = $self->first_block
279 0         0 or croak $name_error;
280 0 0       0 my @names = grep {
281 0         0 $_ !~ /^(ONLY|LAST|SKIP)$/;
282 0 0       0 } @{$block->{_section_order}[0] || []};
283             croak "$name_error. Need two sections in first block"
284 0         0 unless @names == 2;
285             return @names;
286             }
287 0     0   0  
288 0 0       0 sub _assert_plan {
289             plan('no_plan') unless $Have_Plan;
290             }
291 1     1   3  
292 1 0 33     8 sub END {
      33        
293             run_compare() unless $Have_Plan or $DIED or not $import_called;
294             }
295              
296 0     0 0 0 sub run_compare() {
297 0         0 (my ($self), @_) = find_my_self(@_);
298 0         0 $self->_assert_plan;
299 0         0 my ($x, $y) = $self->_section_names(@_);
300 0         0 local $Test::Builder::Level = $Test::Builder::Level + 1;
  0         0  
301 0 0 0     0 for my $block (@{$self->block_list}) {
302 0 0       0 next unless exists($block->{$x}) and exists($block->{$y});
303 0 0       0 $block->run_filters unless $block->is_filtered;
    0          
304 0 0       0 if (ref $block->$x) {
305             is_deeply($block->$x, $block->$y,
306             $block->name ? $block->name : ());
307             }
308 0 0       0 elsif (ref $block->$y eq 'Regexp') {
309 0 0       0 my $regexp = ref $y ? $y : $block->$y;
310             like($block->$x, $regexp, $block->name ? $block->name : ());
311             }
312 0 0       0 else {
313             is($block->$x, $block->$y, $block->name ? $block->name : ());
314             }
315             }
316             }
317              
318 0     0 0 0 sub run_is() {
319 0         0 (my ($self), @_) = find_my_self(@_);
320 0         0 $self->_assert_plan;
321 0         0 my ($x, $y) = $self->_section_names(@_);
322 0         0 local $Test::Builder::Level = $Test::Builder::Level + 1;
  0         0  
323 0 0 0     0 for my $block (@{$self->block_list}) {
324 0 0       0 next unless exists($block->{$x}) and exists($block->{$y});
325 0 0       0 $block->run_filters unless $block->is_filtered;
326             is($block->$x, $block->$y,
327             $block->name ? $block->name : ()
328             );
329             }
330             }
331              
332 0     0 0 0 sub run_is_deeply() {
333 0         0 (my ($self), @_) = find_my_self(@_);
334 0         0 $self->_assert_plan;
335 0         0 my ($x, $y) = $self->_section_names(@_);
  0         0  
336 0 0 0     0 for my $block (@{$self->block_list}) {
337 0 0       0 next unless exists($block->{$x}) and exists($block->{$y});
338 0 0       0 $block->run_filters unless $block->is_filtered;
339             is_deeply($block->$x, $block->$y,
340             $block->name ? $block->name : ()
341             );
342             }
343             }
344              
345 0     0 0 0 sub run_like() {
346 0         0 (my ($self), @_) = find_my_self(@_);
347 0         0 $self->_assert_plan;
348 0         0 my ($x, $y) = $self->_section_names(@_);
  0         0  
349 0 0 0     0 for my $block (@{$self->block_list}) {
350 0 0       0 next unless exists($block->{$x}) and defined($y);
351 0 0       0 $block->run_filters unless $block->is_filtered;
352 0 0       0 my $regexp = ref $y ? $y : $block->$y;
353             like($block->$x, $regexp,
354             $block->name ? $block->name : ()
355             );
356             }
357             }
358              
359 0     0 0 0 sub run_unlike() {
360 0         0 (my ($self), @_) = find_my_self(@_);
361 0         0 $self->_assert_plan;
362 0         0 my ($x, $y) = $self->_section_names(@_);
  0         0  
363 0 0 0     0 for my $block (@{$self->block_list}) {
364 0 0       0 next unless exists($block->{$x}) and defined($y);
365 0 0       0 $block->run_filters unless $block->is_filtered;
366 0 0       0 my $regexp = ref $y ? $y : $block->$y;
367             unlike($block->$x, $regexp,
368             $block->name ? $block->name : ()
369             );
370             }
371             }
372 1     1   2  
373 1         3 sub _pre_eval {
374 1 50       40 my $spec = shift;
375             return $spec unless $spec =~
376 0         0 s/\A\s*<<<(.*?)>>>\s*$//sm;
377 0         0 my $eval_code = $1;
378 0 0       0 eval "package main; $eval_code";
379 0         0 croak $@ if $@;
380             return $spec;
381             }
382 1     1   3  
383 1         29 sub _block_list_init {
384 1         6 my $spec = $self->spec;
385 1         30 $spec = $self->_pre_eval($spec);
386 1         3767 my $cd = $self->block_delim;
387 1         31 my @hunks = ($spec =~ /^(\Q${cd}\E.*?(?=^\Q${cd}\E|\z))/msg);
388 1         38 my $blocks = $self->_choose_blocks(@hunks);
389 1         2 $self->block_list($blocks); # Need to set early for possible filter use
390 1         3 my $seq = 1;
391 500         12404 for my $block (@$blocks) {
392 500         12860 $block->blocks_object($self);
393             $block->seq_num($seq++);
394 1         148 }
395             return $blocks;
396             }
397 1     1   8  
398 1         2 sub _choose_blocks {
399 1         5 my $blocks = [];
400 500         1103 for my $hunk (@_) {
401 500 50       1167 my $block = $self->_make_block($hunk);
402 0 0       0 if (exists $block->{ONLY}) {
403             diag "I found ONLY: maybe you're debugging?"
404 0         0 unless $self->_no_diag_on_only;
405             return [$block];
406 500 50       1163 }
407 500         896 next if exists $block->{SKIP};
408 500 50       1552 push @$blocks, $block;
409 0         0 if (exists $block->{LAST}) {
410             return $blocks;
411             }
412 1         5 }
413             return $blocks;
414             }
415 1000     1000   1177  
416 1000         1274 sub _check_reserved {
417 1000 50 33     5860 my $id = shift;
418             croak "'$id' is a reserved name. Use something else.\n"
419             if $reserved_section_names->{$id} or
420             $id =~ /^_/;
421             }
422 500     500   647  
423 500         748 sub _make_block {
424 500         16917 my $hunk = shift;
425 500         13384 my $cd = $self->block_delim;
426 500         1252 my $dd = $self->data_delim;
427 500 50       4271 my $block = $self->block_class->new;
428 500         1085 $hunk =~ s/\A\Q${cd}\E[ \t]*(.*)\s+// or die;
429 500         7117 my $name = $1;
430 500         1008 my @parts = split /^\Q${dd}\E +\(?(\w+)\)? *(.*)?\n/m, $hunk;
431 500   50     2105 my $description = shift @parts;
432 500 50       1147 $description ||= '';
433 500         705 unless ($description =~ /\S/) {
434             $description = $name;
435 500         2083 }
436 500         10342 $description =~ s/\s*\z//;
437             $block->set_value(description => $description);
438 500         1012
439 500         768 my $section_map = {};
440 500         1041 my $section_order = [];
441 1000         2347 while (@parts) {
442 1000         2412 my ($type, $filters, $value) = splice(@parts, 0, 3);
443 1000 50       1836 $self->_check_reserved($type);
444 1000 50       1646 $value = '' unless defined $value;
445 1000 50       4789 $filters = '' unless defined $filters;
446 1000 50       2306 if ($filters =~ /:(\s|\z)/) {
447             croak "Extra lines not allowed in '$type' section"
448 1000         4895 if $value =~ /\S/;
449 1000 50       2335 ($filters, $value) = split /\s*:(?:\s+|\z)/, $filters, 2;
450 1000         10702 $value = '' unless defined $value;
451             $value =~ s/^\s*(.*?)\s*$/$1/;
452 1000         8610 }
453             $section_map->{$type} = {
454             filters => $filters,
455 1000         2255 };
456 1000         2313 push @$section_order, $type;
457             $block->set_value($type, $value);
458 500         1166 }
459 500         1003 $block->set_value(name => $name);
460 500         1409 $block->set_value(_section_map => $section_map);
461 500         2123 $block->set_value(_section_order => $section_order);
462             return $block;
463             }
464 1     1   3  
465 1 50       29 sub _spec_init {
466             return $self->_spec_string
467 1         4 if $self->_spec_string;
468 1         3 local $/;
469 1 50       27 my $spec;
470 0 0       0 if (my $spec_file = $self->_spec_file) {
471 0         0 open FILE, $spec_file or die $!;
472 0         0 $spec = ;
473             close FILE;
474             }
475 1         3 else {
476             $spec = do {
477 1     1   12 package main;
  1         2  
  1         526  
478 1         68 no warnings 'once';
479             ;
480             };
481 1         8 }
482             return $spec;
483             }
484              
485 1     1   6 sub _strict_warnings() {
486 1         3 require Filter::Util::Call;
487             my $done = 0;
488             Filter::Util::Call::filter_add(
489 1 50   1   6 sub {
490 1         4 return 0 if $done;
491 1         117 my ($data, $end) = ('', '');
492 11 50       22 while (my $status = Filter::Util::Call::filter_read()) {
493 11 100       28 return $status if $status < 0;
494 1         3 if (/^__(?:END|DATA)__\r?$/) {
495 1         3 $end = $_;
496             last;
497 10         16 }
498 10         41 $data .= $_;
499             $_ = '';
500 1         4 }
501 1         112 $_ = "use strict;use warnings;$data$end";
502             $done = 1;
503 1         9 }
504             );
505             }
506              
507 0     0 0 0 sub tie_output() {
508 0 0       0 my $handle = shift;
509 0         0 die "No buffer to tie" unless @_;
510             tie $handle, 'Test::Base::Handle', $_[0];
511             }
512 0     0 0 0  
513 0         0 sub no_diff {
514             $ENV{TEST_SHOW_NO_DIFFS} = 1;
515             }
516              
517             package Test::Base::Handle;
518              
519 0     0   0 sub TIEHANDLE() {
520 0         0 my $class = shift;
521             bless \ $_[0], $class;
522             }
523 0     0   0  
524 0         0 sub PRINT {
525             $$self .= $_ for @_;
526             }
527              
528             #===============================================================================
529             # Test::Base::Block
530             #
531             # This is the default class for accessing a Test::Base block object.
532             #===============================================================================
533             package Test::Base::Block;
534             our @ISA = qw(Spiffy);
535              
536             our @EXPORT = qw(block_accessor);
537 0     0   0  
538 0         0 sub AUTOLOAD {
539             return;
540             }
541              
542 6     6   11 sub block_accessor() {
543 1     1   7 my $accessor = shift;
  1         2  
  1         259  
544 6 50       29 no strict 'refs';
545             return if defined &$accessor;
546 4000     4000   7162 *$accessor = sub {
547 4000 50       8831 my $self = shift;
548 0         0 if (@_) {
549             Carp::croak "Not allowed to set values for '$accessor'";
550 4000 50       4311 }
  4000         16998  
551             my @list = @{$self->{$accessor} || []};
552 4000 100       23118 return wantarray
553             ? (@list)
554 6         43 : $list[0];
555             };
556             }
557              
558             block_accessor 'name';
559             block_accessor 'description';
560             Spiffy::field 'seq_num';
561             Spiffy::field 'is_filtered';
562             Spiffy::field 'blocks_object';
563             Spiffy::field 'original_values' => {};
564 5000     5000   15060  
565 1     1   6 sub set_value {
  1         2  
  1         237  
566 5000         9356 no strict 'refs';
567 5000 100       14611 my $accessor = shift;
568             block_accessor $accessor
569 5000         38096 unless defined &$accessor;
570             $self->{$accessor} = [@_];
571             }
572 500     500   6296  
573 500         1487 sub run_filters {
574 500         1421 my $map = $self->_section_map;
575 500 50       24626 my $order = $self->_section_order;
576             Carp::croak "Attempt to filter a block twice"
577 500         1122 if $self->is_filtered;
578 1000         2671 for my $type (@$order) {
579 1000         3076 my $filters = $map->{$type}{filters};
580 1000         39142 my @value = $self->$type;
581 1000         3348 $self->original_values->{$type} = $value[0];
582 2000 50       5639 for my $filter ($self->_get_filters($type, $filters)) {
583             $Test::Base::Filter::arguments =
584 2000         4010 $filter =~ s/=(.*)$// ? $1 : undef;
585 1     1   7 my $function = "main::$filter";
  1         2  
  1         713  
586 2000 50       17344 no strict 'refs';
587 0         0 if (defined &$function) {
588 0         0 local $_ = join '', @value;
589 0         0 my $old = $_;
590 0 0 0     0 @value = &$function(@value);
      0        
591             if (not(@value) or
592             @value == 1 and $value[0] =~ /\A(\d+|)\z/
593 0 0 0     0 ) {
594 0         0 if ($value[0] && $_ eq $old) {
595             Test::Base::diag("Filters returning numbers are supposed to do munging \$_: your filter '$function' apparently doesn't.");
596 0         0 }
597             @value = ($_);
598             }
599             }
600 2000         57449 else {
601 2000 50       14363 my $filter_object = $self->blocks_object->filter_class->new;
602             die "Can't find a function or method for '$filter' filter\n"
603 2000         58944 unless $filter_object->can($filter);
604 2000         8167 $filter_object->current_block($self);
605             @value = $filter_object->$filter(@value);
606             }
607             # Set the value after each filter since other filters may be
608 2000         6946 # introspecting.
609             $self->set_value($type, @value);
610             }
611 500         15431 }
612             $self->is_filtered(1);
613             }
614 1000     1000   1642  
615 1000         2520 sub _get_filters {
616 1000   50     4311 my $type = shift;
617 1000         7327 my $string = shift || '';
618 1000         1867 $string =~ s/\s*(.*?)\s*/$1/;
619 1000   50     52649 my @filters = ();
620 1000 50       2599 my $map_filters = $self->blocks_object->_filters_map->{$type} || [];
621 1000         2194 $map_filters = [ $map_filters ] unless ref $map_filters;
622 1000         1103 my @append = ();
623 1000         43494 for (
624             @{$self->blocks_object->_filters},
625             @$map_filters,
626             split(/\s+/, $string),
627 2000         2620 ) {
628 2000 50       4376 my $filter = $_;
629 2000 50       5868 last unless length $filter;
    50          
630 0         0 if ($filter =~ s/^-//) {
  0         0  
631             @filters = grep { $_ ne $filter } @filters;
632             }
633 0         0 elsif ($filter =~ s/^\+//) {
634             push @append, $filter;
635             }
636 2000         15841 else {
637             push @filters, $filter;
638             }
639 1000         15753 }
640             return @filters, @append;
641             }
642              
643             {
644             %$reserved_section_names = map {
645             ($_, 1);
646             } keys(%Test::Base::Block::), qw( new DESTROY );
647             }
648              
649             __DATA__