The Perl and Raku Conference 2025: Greenville, South Carolina - June 27-29 Learn more

use strict;
use Test::Stream::Context qw/context/;
use Test::Stream::Util qw/try/;
use Test::Stream::Plugin qw/import/;
sub load_ts_plugin {
my $class = shift;
my $caller = shift;
if (!@_ || (@_ == 1 && $_[0] =~ m/^(streamed|buffered)$/)) {
my $name = $1 || $_[0] || 'buffered';
no strict 'refs';
*{"$caller->[0]\::subtest"} = $class->can("subtest_$name");
return;
}
for my $arg (@_) {
my $ok = $arg =~ m/^subtest_(streamed|buffered)$/;
my $sub = $ok ? $class->can($arg) : undef;
die "$class does not export '$arg' at $caller->[1] line $caller->[2].\n"
unless $sub;
no strict 'refs';
*{"$caller->[0]\::$arg"} = $sub;
}
}
sub subtest_streamed {
my ($name, $code, @args) = @_;
my $ctx = context();
my $pass = _subtest("Subtest: $name", $code, 0, @args);
$ctx->release;
return $pass;
}
sub subtest_buffered {
my ($name, $code, @args) = @_;
my $ctx = context();
my $pass = _subtest($name, $code, 1, @args);
$ctx->release;
return $pass;
}
sub _subtest {
my ($name, $code, $buffered, @args) = @_;
my $ctx = context();
$ctx->note($name) unless $buffered;
my $parent = $ctx->hub;
my $stack = $ctx->stack || Test::Stream::Sync->stack;
my $hub = $stack->new_hub(
class => 'Test::Stream::Hub::Subtest',
);
my @events;
$hub->set_nested( $parent->isa('Test::Stream::Hub::Subtest') ? $parent->nested + 1 : 1 );
$hub->listen(sub { push @events => $_[1] });
$hub->format(undef) if $buffered;
my $no_diag = $ctx->debug->no_diag;
$hub->set_parent_todo($no_diag) if $no_diag;
my ($ok, $err, $finished);
TS_SUBTEST_WRAPPER: {
($ok, $err) = try { $code->(@args) };
# They might have done 'BEGIN { skip_all => "whatever" }'
if (!$ok && $err =~ m/Label not found for "last TS_SUBTEST_WRAPPER"/) {
$ok = undef;
$err = undef;
}
else {
$finished = 1;
}
}
$stack->pop($hub);
my $dbg = $ctx->debug;
if (!$finished) {
if(my $bailed = $hub->bailed_out) {
$ctx->bail($bailed->reason);
}
my $code = $hub->exit_code;
$ok = !$code;
$err = "Subtest ended with exit code $code" if $code;
}
$hub->finalize($dbg, 1)
if $ok
&& !$hub->no_ending
&& !$hub->state->ended;
my $pass = $ok && $hub->state->is_passing;
my $e = $ctx->build_event(
'Subtest',
pass => $pass,
name => $name,
buffered => $buffered,
subevents => \@events,
);
$e->set_diag([
$e->default_diag,
$ok ? () : ("Caught exception in subtest: $err"),
]) unless $pass;
$ctx->hub->send($e);
$ctx->release;
return $hub->state->is_passing;
}
1;
__END__
=pod
=encoding UTF-8
=head1 NAME
Test::Stream::Plugin::Subtest - Tools for writing subtests
=head1 DESCRIPTION
This package exports subs that let you write subtests.
There are 2 types of subtests, buffered and streamed. Streamed subtests mimick
subtest from L<Test::More> in that they render all events as soon as they are
produced. Buffered subtests wait until the subtest completes before rendering
any results.
The main difference is that streamed subtests are unreadable when combined with
concurrency. Buffered subtests look fine with any number of concurrent threads
and processes.
=head1 DEPRECATED
B<This distribution is deprecated> in favor of L<Test2>, L<Test2::Suite>, and
L<Test2::Workflow>.
See L<Test::Stream::Manual::ToTest2> for a conversion guide.
=head1 SYNOPSIS
use Test::Stream qw/Subtest/;
subtest foo => sub {
...
};
=head2 STREAMED
The default option is 'buffered', use this if you want streamed, the way
L<Test::Builder> does it.
# You can use either of the next 2 lines, they are both equivilent
use Test::Stream Subtest => ['streamed'];
use Test::Stream::Plugin::Subtest qw/streamed/;
subtest my_test => sub {
ok(1, "subtest event A");
ok(1, "subtest event B");
};
This will produce output like this:
# Subtest: my_test
ok 1 - subtest event A
ok 2 - subtest event B
1..2
ok 1 - Subtest: my_test
=head2 BUFFERED
# You can use either of the next 2 lines, they are both equivilent
use Test::Stream Subtest => ['buffered'];
use Test::Stream::Plugin::Subtest qw/buffered/;
subtest my_test => sub {
ok(1, "subtest event A");
ok(1, "subtest event B");
};
This will produce output like this:
ok 1 - my_test {
ok 1 - subtest event A
ok 2 - subtest event B
1..2
}
=head2 BOTH
use Test::Stream::Plugin::Subtest qw/subtest_streamed subtest_buffered/;
subtest_streamed my_streamed_test => sub {
ok(1, "subtest event A");
ok(1, "subtest event B");
};
subtest_buffered my_buffered_test => sub {
ok(1, "subtest event A");
ok(1, "subtest event B");
};
This will produce the following output:
# Subtest: my_test
ok 1 - subtest event A
ok 2 - subtest event B
1..2
ok 1 - Subtest: my_test
ok 2 - my_test {
ok 1 - subtest event A
ok 2 - subtest event B
1..2
}
=head1 IMPORTANT NOTE
You can use C<bail_out> or C<skip_all> in a subtest, but not in a BEGIN block
or use statement. This is due to the way flow control works within a begin
block. This is not normally an issue, but can happen in rare conditions using
eval, or script files as subtests.
=head1 EXPORTS
=over 4
=item subtest_streamed $name => $sub
=item subtest_streamed($name, $sub, @args)
Run subtest coderef, stream events as they happen.
=item subtest_buffered $name => $sub
=item subtest_buffered($name, $sub, @args)
Run subtest coderef, render events all at once when subtest is complete.
=back
=head1 SOURCE
The source code repository for Test::Stream can be found at
=head1 MAINTAINERS
=over 4
=item Chad Granum E<lt>exodist@cpan.orgE<gt>
=back
=head1 AUTHORS
=over 4
=item Chad Granum E<lt>exodist@cpan.orgE<gt>
=back
=head1 COPYRIGHT
Copyright 2015 Chad Granum E<lt>exodist7@gmail.comE<gt>.
This program is free software; you can redistribute it and/or
modify it under the same terms as Perl itself.
=cut