—package
Test::Stream::Util;
use
strict;
use
warnings;
use
B();
exports
qw{
try protect
get_tid USE_THREADS
pkg_to_file
get_stash
sig_to_slot slot_to_sig
parse_symbol
term_size
rtype render_ref
set_sub_name
CAN_SET_SUB_NAME
sub_info
sub_name
rename_anon_sub
update_mask
}
;
no
Test::Stream::Exporter;
BEGIN {
local
($@, $!,
$SIG
{__DIE__});
my
$set_subname
=
$have_sub_util
? Sub::Util->can(
'set_subname'
) :
undef
;
my
$subname
=
$have_sub_name
? Sub::Name->can(
'subname'
) :
undef
;
*set_sub_name
=
$set_subname
||
$subname
||
sub
{ croak
"Cannot set sub name"
};
if
(
$set_subname
||
$subname
) {
*CAN_SET_SUB_NAME
=
sub
() { 1 };
}
else
{
*CAN_SET_SUB_NAME
=
sub
() { 0 };
}
}
sub
sub_name {
my
(
$sub
) =
@_
;
croak
"sub_name requires a coderef as its only argument"
unless
$sub
&&
ref
(
$sub
) && reftype(
$sub
) eq
'CODE'
;
my
$cobj
= B::svref_2object(
$sub
);
my
$name
=
$cobj
->GV->NAME;
return
$name
;
}
sub
rename_anon_sub {
my
(
$name
,
$sub
,
$caller
) =
@_
;
$caller
||=
caller
();
croak
"sub_name requires a coderef as its second argument"
unless
$sub
&&
ref
(
$sub
) && reftype(
$sub
) eq
'CODE'
;
my
$cobj
= B::svref_2object(
$sub
);
my
$orig
=
$cobj
->GV->NAME;
return
unless
$orig
=~ m/__ANON__$/;
set_sub_name(
"${caller}::${name}"
,
$sub
);
}
sub
update_mask {
my
(
$file
,
$line
,
$name
,
$mask
) =
@_
;
no
warnings
'once'
;
my
$masks
= \
%Trace::Mask::MASKS
;
# Get existing ref, if any
my
$ref
=
$masks
->{
$file
}->{
$line
}->{
$name
};
# No ref, easy!
return
$masks
->{
$file
}->{
$line
}->{
$name
} = {
%$mask
}
unless
$ref
;
# Merge new mask into old
%$ref
= (
%$ref
,
%$mask
);
return
;
}
sub
_manual_protect(&) {
my
$code
=
shift
;
rename_anon_sub(
'protect'
,
$code
,
caller
)
if
CAN_SET_SUB_NAME;
my
(
$ok
,
$error
);
{
my
(
$msg
,
$no
) = ($@, $!);
$ok
=
eval
{
BEGIN { update_mask(__FILE__, __LINE__ + 1,
'*'
, {
hide
=> 3}) }
$code
->();
1
} || 0;
$error
= $@ ||
"Error was squashed!\n"
;
($@, $!) = (
$msg
,
$no
);
}
die
$error
unless
$ok
;
return
$ok
;
}
sub
_local_protect(&) {
my
$code
=
shift
;
rename_anon_sub(
'protect'
,
$code
,
caller
)
if
CAN_SET_SUB_NAME;
my
(
$ok
,
$error
);
{
local
($@, $!);
$ok
=
eval
{
BEGIN { update_mask(__FILE__, __LINE__ + 1,
'*'
, {
hide
=> 3}) }
$code
->();
1
} || 0;
$error
= $@ ||
"Error was squashed!\n"
;
}
die
$error
unless
$ok
;
return
$ok
;
}
sub
_manual_try(&;@) {
my
$code
=
shift
;
my
$args
= \
@_
;
my
$error
;
my
$ok
;
rename_anon_sub(
'try'
,
$code
,
caller
)
if
CAN_SET_SUB_NAME;
{
my
(
$msg
,
$no
) = ($@, $!);
my
$die
=
delete
$SIG
{__DIE__};
$ok
=
eval
{
BEGIN { update_mask(__FILE__, __LINE__ + 1,
'*'
, {
hide
=> 3}) }
$code
->(
@$args
);
1
} || 0;
unless
(
$ok
) {
$error
= $@ ||
"Error was squashed!\n"
;
}
($@, $!) = (
$msg
,
$no
);
if
(
$die
) {
$SIG
{__DIE__} =
$die
;
}
else
{
delete
$SIG
{__DIE__};
}
}
return
(
$ok
,
$error
);
}
sub
_local_try(&;@) {
my
$code
=
shift
;
my
$args
= \
@_
;
my
$error
;
my
$ok
;
rename_anon_sub(
'try'
,
$code
,
caller
)
if
CAN_SET_SUB_NAME;
{
local
($@, $!,
$SIG
{__DIE__});
$ok
=
eval
{
BEGIN { update_mask(__FILE__, __LINE__ + 1,
'*'
, {
hide
=> 3}) }
$code
->(
@$args
);
1
} || 0;
unless
(
$ok
) {
$error
= $@ ||
"Error was squashed!\n"
;
}
}
return
(
$ok
,
$error
);
}
# Older versions of perl have a nasty bug on win32 when localizing a variable
# before forking or starting a new thread. So for those systems we use the
# non-local form. When possible though we use the faster 'local' form.
BEGIN {
if
($^O eq
'MSWin32'
&& $] < 5.020002) {
*protect
= \
&_manual_protect
;
*try
= \
&_manual_try
;
}
else
{
*protect
= \
&_local_protect
;
*try
= \
&_local_try
;
}
}
BEGIN {
if
(CAN_THREAD) {
if
(
$INC
{
'threads.pm'
}) {
# Threads are already loaded, so we do not need to check if they
# are loaded each time
*USE_THREADS
=
sub
() { 1 };
*get_tid
=
sub
{ threads->tid() };
}
else
{
# :-( Need to check each time to see if they have been loaded.
*USE_THREADS
=
sub
{
$INC
{
'threads.pm'
} ? 1 : 0 };
*get_tid
=
sub
{
$INC
{
'threads.pm'
} ? threads->tid() : 0 };
}
}
else
{
# No threads, not now, not ever!
*USE_THREADS
=
sub
() { 0 };
*get_tid
=
sub
() { 0 };
}
}
sub
pkg_to_file {
my
$pkg
=
shift
;
my
$file
=
$pkg
;
$file
=~ s{(::|')}{/}g;
$file
.=
'.pm'
;
return
$file
;
}
sub
get_stash {
my
$pkg
=
shift
;
no
strict
'refs'
;
return
\%{
"$pkg\::"
};
}
my
%SIG_TABLE
= (
'&'
=>
'CODE'
,
'%'
=>
'HASH'
,
'@'
=>
'ARRAY'
,
'$'
=>
'SCALAR'
,
'*'
=>
'GLOB'
,
);
my
%SLOT_TABLE
=
reverse
%SIG_TABLE
;
sub
sig_to_slot {
$SIG_TABLE
{
$_
[0]} }
sub
slot_to_sig {
$SLOT_TABLE
{
$_
[0]} }
sub
parse_symbol {
my
(
$sym
) =
@_
;
my
$sig
=
substr
(
$sym
, 0, 1);
return
(
$sym
,
'CODE'
)
unless
$SIG_TABLE
{
$sig
};
my
$name
=
substr
(
$sym
, 1);
my
$slot
=
$SIG_TABLE
{
$sig
} or croak
"'$sig' is not a supported sigil"
;
return
(
$name
,
$slot
);
}
BEGIN {
$ok
&&= Term::ReadKey->can(
'GetTerminalSize'
);
*USE_TERM_READKEY
=
$ok
?
sub
() { 1 } :
sub
() { 0 };
};
sub
term_size {
return
$ENV
{TS_TERM_SIZE}
if
$ENV
{TS_TERM_SIZE};
return
80
unless
USE_TERM_READKEY;
my
$total
;
try
{
my
@warnings
;
local
$SIG
{__WARN__} =
sub
{
push
@warnings
=>
@_
};
(
$total
) = Term::ReadKey::GetTerminalSize(
*STDOUT
);
@warnings
=
grep
{
$_
!~ m/Unable to get Terminal Size/ }
@warnings
;
warn
@warnings
;
};
return
80
if
!
$total
;
return
80
if
$total
< 80;
return
$total
;
}
sub
rtype {
my
(
$thing
) =
@_
;
return
''
unless
defined
$thing
;
my
$rf
=
ref
$thing
;
my
$rt
= reftype
$thing
;
return
''
unless
$rf
||
$rt
;
return
'REGEXP'
if
$rf
=~ m/Regex/i;
return
'REGEXP'
if
$rt
=~ m/Regex/i;
return
$rt
||
''
;
}
sub
render_ref {
my
(
$in
) =
@_
;
return
'undef'
unless
defined
(
$in
);
my
$type
= rtype(
$in
);
return
"$in"
unless
$type
;
# Look past overloading
my
$class
= blessed(
$in
) ||
''
;
my
$it
=
sprintf
(
'0x%x'
, refaddr(
$in
));
my
$ref
=
"$type($it)"
;
return
$ref
unless
$class
;
return
"$class=$ref"
;
}
sub
sub_info {
my
(
$sub
,
@all_lines
) =
@_
;
my
%in
=
map
{
$_
=> 1}
@all_lines
;
croak
"sub_info requires a coderef as its first argument"
unless
$sub
&&
ref
(
$sub
) && reftype(
$sub
) eq
'CODE'
;
my
$cobj
= B::svref_2object(
$sub
);
my
$name
=
$cobj
->GV->NAME;
my
$file
=
$cobj
->FILE;
my
$package
=
$cobj
->GV->STASH->NAME;
my
$op
=
$cobj
->START;
while
(
$op
) {
push
@all_lines
=>
$op
->line
if
$op
->can(
'line'
);
last
unless
$op
->can(
'next'
);
$op
=
$op
->
next
;
}
my
(
$start
,
$end
,
@lines
);
if
(
@all_lines
) {
@all_lines
=
sort
{
$a
<=>
$b
}
@all_lines
;
(
$start
,
$end
) = (
$all_lines
[0],
$all_lines
[-1]);
# Adjust start and end for the most common case of a multi-line block with
# parens on the lines before and after.
if
(
$start
<
$end
) {
$start
--
unless
$start
<= 1 ||
$in
{
$start
};
$end
++
unless
$in
{
$end
};
}
@lines
= (
$start
,
$end
);
}
return
{
ref
=>
$sub
,
cobj
=>
$cobj
,
name
=>
$name
,
file
=>
$file
,
package
=>
$package
,
start_line
=>
$start
,
end_line
=>
$end
,
all_lines
=> \
@all_lines
,
lines
=> \
@lines
,
};
}
1;
__END__
=pod
=encoding UTF-8
=head1 NAME
Test::Stream::Util - Tools used by Test::Stream and friends.
=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 DESCRIPTION
Collection of tools used by L<Test::Stream> and friends.
=head1 EXPORTS
All exports are optional, you must specify subs to import. If you want to
import everything use '-all'.
use Test::Stream::Util '-all';
=over 4
=item ($success, $error) = try { ... }
Eval the codeblock, return success or failure, and the error message. This code
protects $@ and $!, they will be restored by the end of the run. This code also
temporarily blocks $SIG{DIE} handlers.
=item protect { ... }
Similar to try, except that it does not catch exceptions. The idea here is to
protect $@ and $! from changes. $@ and $! will be restored to whatever they
were before the run so long as it is successful. If the run fails $! will still
be restored, but $@ will contain the exception being thrown.
=item USE_THREADS
Returns true if threads are enabled, false if they are not.
=item get_tid
This will return the id of the current thread when threads are enabled,
otherwise it returns 0.
=item my $file = pkg_to_file($package)
Convert a package name to a filename.
=item $stash = get_stash($package)
Returns the stash reference for the given package. The stash reference can be
treated like a hashref, you can get keys and values from it.
=item $slot = sig_to_slot($sigil)
Given a sigil such as C<$>, C<@>, C<%>, C<&>, C<*>, this will return the GLOB
slot for that sigil such as C<SCALAR>, C<ARRAY>, C<HASH>, C<CODE>, C<GLOB>.
=item $sigil = slot_to_sig($slot)
Given a a glob slot such as C<SCALAR>, C<ARRAY>, C<HASH>, C<CODE>, C<GLOB>,
this will return the typical sigil for that slot such as C<$>, C<@>, C<%>,
C<&>, C<*>.
=item ($name, $type) = parse_symbol($symbol)
When given a symbol name such as C<$foo> or C<@bar> this will return the symbol
name, and the type name. If no sigil is present in the variable name it will
assume it is a subroutine and return the C<CODE> type. C<$symbol> should be a
string containing the name of the symbol with optional sigil.
=item my $cols = term_size()
Attempts to find the width in columns (characters) of the current terminal.
Returns 80 as a safe bet if it cannot find it another way. This is most
accurate if L<Term::ReadKey> is installed.
=item $type = rtype($ref)
A normalization between C<Scalar::Util::reftype()> and C<ref()>.
Always returns a string.
Returns C<'REGEXP'> for regex types
Returns C<''> for non-refs
Otherwise returns what C<Scalar::Util::reftype()> returns.
=item $addr_str = render_ref($ref)
Always returns a string. For unblessed references this returns something like
C<"SCALAR(0x...)">. For blessed references it returns
C<"My::Thing=SCALAR(0x...)">. The only difference between this and C<$add_str =
"$thing"> is that it ignores any overloading to ensure it is always the ref
address.
=item $bool = CAN_SET_SUB_NAME()
A constant, it returns true if either L<Sub::Name> or L<Sub::Util> are
installed and have the code necessary to set a sub name.
=item set_sub_name($name, $coderef)
When L<Sub::Name> or L<Sub::Util> are installed, this will be an alias to the
sub name setting function from one or the other. If neither are installed then
this will be a sub that throws an exception.
If setting the sub name is something nice, but not strictly necessary, you can
use this conditionally with C<CAN_SET_SUB_NAME()>.
use Test::Stream::Util qw/CAN_SET_SUB_NAME set_sub_name/;
set_sub_name('foo', \&sub) if CAN_SET_SUB_NAME();
=item my $hr = sub_info(\&code)
This returns a hashref with information about the sub:
{
ref => \&code,
cobj => $cobj,
name => "Some::Mod::code",
file => "Some/Mod.pm",
package => "Some::Mod",
# Note: These have been adjusted based on guesswork.
start_line => 22,
end_line => 42,
lines => [22, 42],
# Not a bug, these lines are different!
all_lines => [23, 25, ..., 39, 41],
};
=over 4
=item $info->{ref} => \&code
This is the original sub passed to C<sub_info()>.
=item $info->{cobj} => $cobj
This is the c-object representation of the coderef.
=item $info->{name} => "Some::Mod::code"
This is the name of the coderef, for anonymous coderefs this may end with
C<'__ANON__'>. Also note that the package 'main' is special, and 'main::' may
be omitted.
=item $info->{file} => "Some/Mod.pm"
The file in which the sub was defined.
=item $info->{package} => "Some::Mod"
The package in which the sub was defined.
=item $info->{start_line} => 22
=item $info->{end_line} => 42
=item $info->{lines} => [22, 42]
These 3 fields are the I<adjusted> start line, end line, and array with both.
It is important to note that these lines have been adjusted and may not be
accurate.
The lines are obtained by walking the ops, as such the first line is the line
of the first statement, and the last line is the line of the last statement.
This means that in multi-line subs the lines are usually off by 1. The lines
in these keys will be adjusted for you if it detects a multi-line sub.
=item $info->{all_lines} => [23, 25, ..., 39, 41]
This is an array with the lines of every statement in the sub. unlike the other
line fields, these have not been adjusted for you.
=back
=item update_mask($file, $line, $sub, {...})
This sets masking behavior in accordance with L<Trace::Mask>. This will have no
effect on anything that does not honor L<Trace::Mask>.
=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>
=item Kent Fredric E<lt>kentnl@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