commit 044aff3ba09e87f8fa574c2b422cd843c092971c Author: David Dick <ddick@xxxxxxxx> Date: Wed May 7 19:43:21 2014 +1000 Initial import (#1093640). .gitignore | 1 + perl-Protocol-WebSocket.spec | 84 + ...col_websocket_remove_cpan_meta_references.patch | 33 + sources | 1 + test_simple_include.patch | 6114 ++++++++++++++++++++ 5 files changed, 6233 insertions(+), 0 deletions(-) --- diff --git a/.gitignore b/.gitignore index e69de29..99591c1 100644 --- a/.gitignore +++ b/.gitignore @@ -0,0 +1 @@ +/Protocol-WebSocket-0.17.tar.gz diff --git a/perl-Protocol-WebSocket.spec b/perl-Protocol-WebSocket.spec new file mode 100644 index 0000000..713eade --- /dev/null +++ b/perl-Protocol-WebSocket.spec @@ -0,0 +1,84 @@ +Name: perl-Protocol-WebSocket +Version: 0.17 +Release: 1%{?dist} +Summary: WebSocket protocol +License: GPL+ or Artistic +Group: Development/Libraries +URL: http://search.cpan.org/dist/Protocol-WebSocket/ +Source0: http://www.cpan.org/modules/by-module/Protocol/Protocol-WebSocket-%{version}.tar.gz +# includes Test::More with a higher version than available for epel6 +Patch1: test_simple_include.patch +# Remove the CPAN::Meta references from Build.PL +Patch2: protocol_websocket_remove_cpan_meta_references.patch +BuildArch: noarch +BuildRequires: perl +BuildRequires: perl(base) +BuildRequires: perl(Carp) +BuildRequires: perl(Config) +BuildRequires: perl(constant) +%if 0%{?el6} +%else +BuildRequires: perl(CPAN::Meta) +BuildRequires: perl(CPAN::Meta::Prereqs) +%endif +BuildRequires: perl(Digest::MD5) +BuildRequires: perl(Digest::SHA1) +BuildRequires: perl(Encode) +%if 0%{?el6} +BuildRequires: perl(Exporter) +%endif +BuildRequires: perl(File::Basename) +BuildRequires: perl(File::Spec) +BuildRequires: perl(IO::Handle) +BuildRequires: perl(MIME::Base64) +BuildRequires: perl(Module::Build) +BuildRequires: perl(Scalar::Util) +BuildRequires: perl(strict) +BuildRequires: perl(Test::More) +BuildRequires: perl(utf8) +BuildRequires: perl(warnings) +%if 0%{?el6} +BuildRequires: perl(vars) +%endif +Requires: perl(:MODULE_COMPAT_%(eval "`%{__perl} -V:version`"; echo $version)) + +%description +Client/server WebSocket message and frame parser/constructor. This module +does not provide a WebSocket server or client, but is made for using in +http servers or clients to provide WebSocket support. + +%prep +%setup -q -n Protocol-WebSocket-%{version} +%{__sed} -i 's|\r||' ./examples/reflex.pl +%if 0%{?el6} +%patch1 -p1 +%patch2 -p1 +%endif +# Upstream is okay with wsconsole being made available as a binary for Fedora/EPEL +%{__mv} util bin + +%build +%{__perl} Build.PL installdirs=vendor +./Build + +%install +./Build install destdir=$RPM_BUILD_ROOT create_packlist=0 + +%{_fixperms} $RPM_BUILD_ROOT/* + +%check +%if 0%{?el6} +PERL5LIB=test_simple_patch/lib ./Build test +%else +./Build test +%endif + +%files +%doc Changes LICENSE examples +%{perl_vendorlib}/* +%{_mandir}/man3/* +%{_bindir}/* + +%changelog +* Sat Apr 12 2014 David Dick <ddick@xxxxxxxx> - 0.17-1 +- Initial release diff --git a/protocol_websocket_remove_cpan_meta_references.patch b/protocol_websocket_remove_cpan_meta_references.patch new file mode 100644 index 0000000..37725eb --- /dev/null +++ b/protocol_websocket_remove_cpan_meta_references.patch @@ -0,0 +1,33 @@ +diff -Naur old/Build.PL new/Build.PL +--- old/Build.PL 2014-04-09 18:12:18.000000000 +1000 ++++ new/Build.PL 2014-04-13 22:01:57.890616362 +1000 +@@ -12,8 +12,6 @@ + use Module::Build; + use File::Basename; + use File::Spec; +-use CPAN::Meta; +-use CPAN::Meta::Prereqs; + + my %args = ( + license => 'perl', +@@ -53,20 +51,3 @@ + )->new(%args); + $builder->create_build_script(); + +-my $mbmeta = CPAN::Meta->load_file('MYMETA.json'); +-my $meta = CPAN::Meta->load_file('META.json'); +-my $prereqs_hash = CPAN::Meta::Prereqs->new( +- $meta->prereqs +-)->with_merged_prereqs( +- CPAN::Meta::Prereqs->new($mbmeta->prereqs) +-)->as_string_hash; +-my $mymeta = CPAN::Meta->new( +- { +- %{$meta->as_struct}, +- prereqs => $prereqs_hash +- } +-); +-print "Merging cpanfile prereqs to MYMETA.yml\n"; +-$mymeta->save('MYMETA.yml', { version => 1.4 }); +-print "Merging cpanfile prereqs to MYMETA.json\n"; +-$mymeta->save('MYMETA.json', { version => 2 }); diff --git a/sources b/sources index e69de29..4834a61 100644 --- a/sources +++ b/sources @@ -0,0 +1 @@ +522b7d591b0e9206385352a69b5fd85f Protocol-WebSocket-0.17.tar.gz diff --git a/test_simple_include.patch b/test_simple_include.patch new file mode 100644 index 0000000..7158eb6 --- /dev/null +++ b/test_simple_include.patch @@ -0,0 +1,6114 @@ +diff -Naur old/test_simple_patch/lib/Test/Builder/IO/Scalar.pm new/test_simple_patch/lib/Test/Builder/IO/Scalar.pm +--- old/test_simple_patch/lib/Test/Builder/IO/Scalar.pm 1970-01-01 10:00:00.000000000 +1000 ++++ new/test_simple_patch/lib/Test/Builder/IO/Scalar.pm 2014-03-26 21:48:11.510257612 +1100 +@@ -0,0 +1,658 @@ ++package Test::Builder::IO::Scalar; ++ ++ ++=head1 NAME ++ ++Test::Builder::IO::Scalar - A copy of IO::Scalar for Test::Builder ++ ++=head1 DESCRIPTION ++ ++This is a copy of IO::Scalar which ships with Test::Builder to ++support scalar references as filehandles on Perl 5.6. Newer ++versions of Perl simply use C<<open()>>'s built in support. ++ ++Test::Builder can not have dependencies on other modules without ++careful consideration, so its simply been copied into the distribution. ++ ++=head1 COPYRIGHT and LICENSE ++ ++This file came from the "IO-stringy" Perl5 toolkit. ++ ++Copyright (c) 1996 by Eryq. All rights reserved. ++Copyright (c) 1999,2001 by ZeeGee Software Inc. All rights reserved. ++ ++This program is free software; you can redistribute it and/or ++modify it under the same terms as Perl itself. ++ ++ ++=cut ++ ++# This is copied code, I don't care. ++##no critic ++ ++use Carp; ++use strict; ++use vars qw($VERSION @ISA); ++use IO::Handle; ++ ++use 5.005; ++ ++### The package version, both in 1.23 style *and* usable by MakeMaker: ++$VERSION = "2.110"; ++ ++### Inheritance: ++@ISA = qw(IO::Handle); ++ ++#============================== ++ ++=head2 Construction ++ ++=over 4 ++ ++=cut ++ ++#------------------------------ ++ ++=item new [ARGS...] ++ ++I<Class method.> ++Return a new, unattached scalar handle. ++If any arguments are given, they're sent to open(). ++ ++=cut ++ ++sub new { ++ my $proto = shift; ++ my $class = ref($proto) || $proto; ++ my $self = bless \do { local *FH }, $class; ++ tie *$self, $class, $self; ++ $self->open(@_); ### open on anonymous by default ++ $self; ++} ++sub DESTROY { ++ shift->close; ++} ++ ++#------------------------------ ++ ++=item open [SCALARREF] ++ ++I<Instance method.> ++Open the scalar handle on a new scalar, pointed to by SCALARREF. ++If no SCALARREF is given, a "private" scalar is created to hold ++the file data. ++ ++Returns the self object on success, undefined on error. ++ ++=cut ++ ++sub open { ++ my ($self, $sref) = @_; ++ ++ ### Sanity: ++ defined($sref) or do {my $s = ''; $sref = \$s}; ++ (ref($sref) eq "SCALAR") or croak "open() needs a ref to a scalar"; ++ ++ ### Setup: ++ *$self->{Pos} = 0; ### seek position ++ *$self->{SR} = $sref; ### scalar reference ++ $self; ++} ++ ++#------------------------------ ++ ++=item opened ++ ++I<Instance method.> ++Is the scalar handle opened on something? ++ ++=cut ++ ++sub opened { ++ *{shift()}->{SR}; ++} ++ ++#------------------------------ ++ ++=item close ++ ++I<Instance method.> ++Disassociate the scalar handle from its underlying scalar. ++Done automatically on destroy. ++ ++=cut ++ ++sub close { ++ my $self = shift; ++ %{*$self} = (); ++ 1; ++} ++ ++=back ++ ++=cut ++ ++ ++ ++#============================== ++ ++=head2 Input and output ++ ++=over 4 ++ ++=cut ++ ++ ++#------------------------------ ++ ++=item flush ++ ++I<Instance method.> ++No-op, provided for OO compatibility. ++ ++=cut ++ ++sub flush { "0 but true" } ++ ++#------------------------------ ++ ++=item getc ++ ++I<Instance method.> ++Return the next character, or undef if none remain. ++ ++=cut ++ ++sub getc { ++ my $self = shift; ++ ++ ### Return undef right away if at EOF; else, move pos forward: ++ return undef if $self->eof; ++ substr(${*$self->{SR}}, *$self->{Pos}++, 1); ++} ++ ++#------------------------------ ++ ++=item getline ++ ++I<Instance method.> ++Return the next line, or undef on end of string. ++Can safely be called in an array context. ++Currently, lines are delimited by "\n". ++ ++=cut ++ ++sub getline { ++ my $self = shift; ++ ++ ### Return undef right away if at EOF: ++ return undef if $self->eof; ++ ++ ### Get next line: ++ my $sr = *$self->{SR}; ++ my $i = *$self->{Pos}; ### Start matching at this point. ++ ++ ### Minimal impact implementation! ++ ### We do the fast fast thing (no regexps) if using the ++ ### classic input record separator. ++ ++ ### Case 1: $/ is undef: slurp all... ++ if (!defined($/)) { ++ *$self->{Pos} = length $$sr; ++ return substr($$sr, $i); ++ } ++ ++ ### Case 2: $/ is "\n": zoom zoom zoom... ++ elsif ($/ eq "\012") { ++ ++ ### Seek ahead for "\n"... yes, this really is faster than regexps. ++ my $len = length($$sr); ++ for (; $i < $len; ++$i) { ++ last if ord (substr ($$sr, $i, 1)) == 10; ++ } ++ ++ ### Extract the line: ++ my $line; ++ if ($i < $len) { ### We found a "\n": ++ $line = substr ($$sr, *$self->{Pos}, $i - *$self->{Pos} + 1); ++ *$self->{Pos} = $i+1; ### Remember where we finished up. ++ } ++ else { ### No "\n"; slurp the remainder: ++ $line = substr ($$sr, *$self->{Pos}, $i - *$self->{Pos}); ++ *$self->{Pos} = $len; ++ } ++ return $line; ++ } ++ ++ ### Case 3: $/ is ref to int. Do fixed-size records. ++ ### (Thanks to Dominique Quatravaux.) ++ elsif (ref($/)) { ++ my $len = length($$sr); ++ my $i = ${$/} + 0; ++ my $line = substr ($$sr, *$self->{Pos}, $i); ++ *$self->{Pos} += $i; ++ *$self->{Pos} = $len if (*$self->{Pos} > $len); ++ return $line; ++ } ++ ++ ### Case 4: $/ is either "" (paragraphs) or something weird... ++ ### This is Graham's general-purpose stuff, which might be ++ ### a tad slower than Case 2 for typical data, because ++ ### of the regexps. ++ else { ++ pos($$sr) = $i; ++ ++ ### If in paragraph mode, skip leading lines (and update i!): ++ length($/) or ++ (($$sr =~ m/\G\n*/g) and ($i = pos($$sr))); ++ ++ ### If we see the separator in the buffer ahead... ++ if (length($/) ++ ? $$sr =~ m,\Q$/\E,g ### (ordinary sep) TBD: precomp! ++ : $$sr =~ m,\n\n,g ### (a paragraph) ++ ) { ++ *$self->{Pos} = pos $$sr; ++ return substr($$sr, $i, *$self->{Pos}-$i); ++ } ++ ### Else if no separator remains, just slurp the rest: ++ else { ++ *$self->{Pos} = length $$sr; ++ return substr($$sr, $i); ++ } ++ } ++} ++ ++#------------------------------ ++ ++=item getlines ++ ++I<Instance method.> ++Get all remaining lines. ++It will croak() if accidentally called in a scalar context. ++ ++=cut ++ ++sub getlines { ++ my $self = shift; ++ wantarray or croak("can't call getlines in scalar context!"); ++ my ($line, @lines); ++ push @lines, $line while (defined($line = $self->getline)); ++ @lines; ++} ++ ++#------------------------------ ++ ++=item print ARGS... ++ ++I<Instance method.> ++Print ARGS to the underlying scalar. ++ ++B<Warning:> this continues to always cause a seek to the end ++of the string, but if you perform seek()s and tell()s, it is ++still safer to explicitly seek-to-end before subsequent print()s. ++ ++=cut ++ ++sub print { ++ my $self = shift; ++ *$self->{Pos} = length(${*$self->{SR}} .= join('', @_) . (defined($\) ? $\ : "")); ++ 1; ++} ++sub _unsafe_print { ++ my $self = shift; ++ my $append = join('', @_) . $\; ++ ${*$self->{SR}} .= $append; ++ *$self->{Pos} += length($append); ++ 1; ++} ++sub _old_print { ++ my $self = shift; ++ ${*$self->{SR}} .= join('', @_) . $\; ++ *$self->{Pos} = length(${*$self->{SR}}); ++ 1; ++} ++ ++ ++#------------------------------ ++ ++=item read BUF, NBYTES, [OFFSET] ++ ++I<Instance method.> ++Read some bytes from the scalar. ++Returns the number of bytes actually read, 0 on end-of-file, undef on error. ++ ++=cut ++ ++sub read { ++ my $self = $_[0]; ++ my $n = $_[2]; ++ my $off = $_[3] || 0; ++ ++ my $read = substr(${*$self->{SR}}, *$self->{Pos}, $n); ++ $n = length($read); ++ *$self->{Pos} += $n; ++ ($off ? substr($_[1], $off) : $_[1]) = $read; ++ return $n; ++} ++ ++#------------------------------ ++ ++=item write BUF, NBYTES, [OFFSET] ++ ++I<Instance method.> ++Write some bytes to the scalar. ++ ++=cut ++ ++sub write { ++ my $self = $_[0]; ++ my $n = $_[2]; ++ my $off = $_[3] || 0; ++ ++ my $data = substr($_[1], $off, $n); ++ $n = length($data); ++ $self->print($data); ++ return $n; ++} ++ ++#------------------------------ ++ ++=item sysread BUF, LEN, [OFFSET] ++ ++I<Instance method.> ++Read some bytes from the scalar. ++Returns the number of bytes actually read, 0 on end-of-file, undef on error. ++ ++=cut ++ ++sub sysread { ++ my $self = shift; ++ $self->read(@_); ++} ++ ++#------------------------------ ++ ++=item syswrite BUF, NBYTES, [OFFSET] ++ ++I<Instance method.> ++Write some bytes to the scalar. ++ ++=cut ++ ++sub syswrite { ++ my $self = shift; ++ $self->write(@_); ++} ++ ++=back ++ ++=cut ++ ++ ++#============================== ++ ++=head2 Seeking/telling and other attributes ++ ++=over 4 ++ ++=cut ++ ++ ++#------------------------------ ++ ++=item autoflush ++ ++I<Instance method.> ++No-op, provided for OO compatibility. ++ ++=cut ++ ++sub autoflush {} ++ ++#------------------------------ ++ ++=item binmode ++ ++I<Instance method.> ++No-op, provided for OO compatibility. ++ ++=cut ++ ++sub binmode {} ++ ++#------------------------------ ++ ++=item clearerr ++ ++I<Instance method.> Clear the error and EOF flags. A no-op. ++ ++=cut ++ ++sub clearerr { 1 } ++ ++#------------------------------ ++ ++=item eof ++ ++I<Instance method.> Are we at end of file? ++ ++=cut ++ ++sub eof { ++ my $self = shift; ++ (*$self->{Pos} >= length(${*$self->{SR}})); ++} ++ ++#------------------------------ ++ ++=item seek OFFSET, WHENCE ++ ++I<Instance method.> Seek to a given position in the stream. ++ ++=cut ++ ++sub seek { ++ my ($self, $pos, $whence) = @_; ++ my $eofpos = length(${*$self->{SR}}); ++ ++ ### Seek: ++ if ($whence == 0) { *$self->{Pos} = $pos } ### SEEK_SET ++ elsif ($whence == 1) { *$self->{Pos} += $pos } ### SEEK_CUR ++ elsif ($whence == 2) { *$self->{Pos} = $eofpos + $pos} ### SEEK_END ++ else { croak "bad seek whence ($whence)" } ++ ++ ### Fixup: ++ if (*$self->{Pos} < 0) { *$self->{Pos} = 0 } ++ if (*$self->{Pos} > $eofpos) { *$self->{Pos} = $eofpos } ++ return 1; ++} ++ ++#------------------------------ ++ ++=item sysseek OFFSET, WHENCE ++ ++I<Instance method.> Identical to C<seek OFFSET, WHENCE>, I<q.v.> ++ ++=cut ++ ++sub sysseek { ++ my $self = shift; ++ $self->seek (@_); ++} ++ ++#------------------------------ ++ ++=item tell ++ ++I<Instance method.> ++Return the current position in the stream, as a numeric offset. ++ ++=cut ++ ++sub tell { *{shift()}->{Pos} } ++ ++#------------------------------ ++ ++=item use_RS [YESNO] ++ ++I<Instance method.> ++B<Deprecated and ignored.> ++Obey the current setting of $/, like IO::Handle does? ++Default is false in 1.x, but cold-welded true in 2.x and later. ++ ++=cut ++ ++sub use_RS { ++ my ($self, $yesno) = @_; ++ carp "use_RS is deprecated and ignored; \$/ is always consulted\n"; ++ } ++ ++#------------------------------ ++ ++=item setpos POS ++ ++I<Instance method.> ++Set the current position, using the opaque value returned by C<getpos()>. ++ ++=cut ++ ++sub setpos { shift->seek($_[0],0) } ++ ++#------------------------------ ++ ++=item getpos ++ ++I<Instance method.> ++Return the current position in the string, as an opaque object. ++ ++=cut ++ ++*getpos = \&tell; ++ ++ ++#------------------------------ ++ ++=item sref ++ ++I<Instance method.> ++Return a reference to the underlying scalar. ++ ++=cut ++ ++sub sref { *{shift()}->{SR} } ++ ++ ++#------------------------------ ++# Tied handle methods... ++#------------------------------ ++ ++# Conventional tiehandle interface: ++sub TIEHANDLE { ++ ((defined($_[1]) && UNIVERSAL::isa($_[1], __PACKAGE__)) ++ ? $_[1] ++ : shift->new(@_)); ++} ++sub GETC { shift->getc(@_) } ++sub PRINT { shift->print(@_) } ++sub PRINTF { shift->print(sprintf(shift, @_)) } ++sub READ { shift->read(@_) } ++sub READLINE { wantarray ? shift->getlines(@_) : shift->getline(@_) } ++sub WRITE { shift->write(@_); } ++sub CLOSE { shift->close(@_); } ++sub SEEK { shift->seek(@_); } ++sub TELL { shift->tell(@_); } ++sub EOF { shift->eof(@_); } ++ ++#------------------------------------------------------------ ++ ++1; ++ ++__END__ ++ ++ ++ ++=back ++ ++=cut ++ ++ ++=head1 WARNINGS ++ ++Perl's TIEHANDLE spec was incomplete prior to 5.005_57; ++it was missing support for C<seek()>, C<tell()>, and C<eof()>. ++Attempting to use these functions with an IO::Scalar will not work ++prior to 5.005_57. IO::Scalar will not have the relevant methods ++invoked; and even worse, this kind of bug can lie dormant for a while. ++If you turn warnings on (via C<$^W> or C<perl -w>), ++and you see something like this... ++ ++ attempt to seek on unopened filehandle ++ ++...then you are probably trying to use one of these functions ++on an IO::Scalar with an old Perl. The remedy is to simply ++use the OO version; e.g.: ++ ++ $SH->seek(0,0); ### GOOD: will work on any 5.005 ++ seek($SH,0,0); ### WARNING: will only work on 5.005_57 and beyond ++ ++ ++=head1 VERSION ++ ++$Id: Scalar.pm,v 1.6 2005/02/10 21:21:53 dfs Exp $ ++ ++ ++=head1 AUTHORS ++ ++=head2 Primary Maintainer ++ ++David F. Skoll (F<dfs@xxxxxxxxxxxxxxxxxx>). ++ ++=head2 Principal author ++ ++Eryq (F<eryq@xxxxxxxxxx>). ++President, ZeeGee Software Inc (F<http://www.zeegee.com>). ++ ++ ++=head2 Other contributors ++ ++The full set of contributors always includes the folks mentioned ++in L<IO::Stringy/"CHANGE LOG">. But just the same, special ++thanks to the following individuals for their invaluable contributions ++(if I've forgotten or misspelled your name, please email me!): ++ ++I<Andy Glew,> ++for contributing C<getc()>. ++ ++I<Brandon Browning,> ++for suggesting C<opened()>. ++ ++I<David Richter,> ++for finding and fixing the bug in C<PRINTF()>. ++ ++I<Eric L. Brine,> ++for his offset-using read() and write() implementations. ++ ++I<Richard Jones,> ++for his patches to massively improve the performance of C<getline()> ++and add C<sysread> and C<syswrite>. ++ ++I<B. K. Oxley (binkley),> ++for stringification and inheritance improvements, ++and sundry good ideas. ++ ++I<Doug Wilson,> ++for the IO::Handle inheritance and automatic tie-ing. ++ ++ ++=head1 SEE ALSO ++ ++L<IO::String>, which is quite similar but which was designed ++more-recently and with an IO::Handle-like interface in mind, ++so you could mix OO- and native-filehandle usage without using tied(). ++ ++I<Note:> as of version 2.x, these classes all work like ++their IO::Handle counterparts, so we have comparable ++functionality to IO::String. ++ ++=cut ++ +diff -Naur old/test_simple_patch/lib/Test/Builder/Module.pm new/test_simple_patch/lib/Test/Builder/Module.pm +--- old/test_simple_patch/lib/Test/Builder/Module.pm 1970-01-01 10:00:00.000000000 +1000 ++++ new/test_simple_patch/lib/Test/Builder/Module.pm 2014-03-26 21:48:11.510257612 +1100 +@@ -0,0 +1,173 @@ ++package Test::Builder::Module; ++ ++use strict; ++ ++use Test::Builder 0.99; ++ ++require Exporter; ++our @ISA = qw(Exporter); ++ ++our $VERSION = '1.001003'; ++$VERSION = eval $VERSION; ## no critic (BuiltinFunctions::ProhibitStringyEval) ++ ++ ++=head1 NAME ++ ++Test::Builder::Module - Base class for test modules ++ ++=head1 SYNOPSIS ++ ++ # Emulates Test::Simple ++ package Your::Module; ++ ++ my $CLASS = __PACKAGE__; ++ ++ use base 'Test::Builder::Module'; ++ @EXPORT = qw(ok); ++ ++ sub ok ($;$) { ++ my $tb = $CLASS->builder; ++ return $tb->ok(@_); ++ } ++ ++ 1; ++ ++ ++=head1 DESCRIPTION ++ ++This is a superclass for Test::Builder-based modules. It provides a ++handful of common functionality and a method of getting at the underlying ++Test::Builder object. ++ ++ ++=head2 Importing ++ ++Test::Builder::Module is a subclass of Exporter which means your ++module is also a subclass of Exporter. @EXPORT, @EXPORT_OK, etc... ++all act normally. ++ ++A few methods are provided to do the C<use Your::Module tests => 23> part ++for you. ++ ++=head3 import ++ ++Test::Builder::Module provides an import() method which acts in the ++same basic way as Test::More's, setting the plan and controlling ++exporting of functions and variables. This allows your module to set ++the plan independent of Test::More. ++ ++All arguments passed to import() are passed onto ++C<< Your::Module->builder->plan() >> with the exception of ++C<< import =>[qw(things to import)] >>. ++ ++ use Your::Module import => [qw(this that)], tests => 23; ++ ++says to import the functions this() and that() as well as set the plan ++to be 23 tests. ++ ++import() also sets the exported_to() attribute of your builder to be ++the caller of the import() function. ++ ++Additional behaviors can be added to your import() method by overriding ++import_extra(). ++ ++=cut ++ ++sub import { ++ my($class) = shift; ++ ++ # Don't run all this when loading ourself. ++ return 1 if $class eq 'Test::Builder::Module'; ++ ++ my $test = $class->builder; ++ ++ my $caller = caller; ++ ++ $test->exported_to($caller); ++ ++ $class->import_extra( \@_ ); ++ my(@imports) = $class->_strip_imports( \@_ ); ++ ++ $test->plan(@_); ++ ++ $class->export_to_level( 1, $class, @imports ); ++} ++ ++sub _strip_imports { ++ my $class = shift; ++ my $list = shift; ++ ++ my @imports = (); ++ my @other = (); ++ my $idx = 0; ++ while( $idx <= $#{$list} ) { ++ my $item = $list->[$idx]; ++ ++ if( defined $item and $item eq 'import' ) { ++ push @imports, @{ $list->[ $idx + 1 ] }; ++ $idx++; ++ } ++ else { ++ push @other, $item; ++ } ++ ++ $idx++; ++ } ++ ++ @$list = @other; ++ ++ return @imports; ++} ++ ++=head3 import_extra ++ ++ Your::Module->import_extra(\@import_args); ++ ++import_extra() is called by import(). It provides an opportunity for you ++to add behaviors to your module based on its import list. ++ ++Any extra arguments which shouldn't be passed on to plan() should be ++stripped off by this method. ++ ++See Test::More for an example of its use. ++ ++B<NOTE> This mechanism is I<VERY ALPHA AND LIKELY TO CHANGE> as it ++feels like a bit of an ugly hack in its current form. ++ ++=cut ++ ++sub import_extra { } ++ ++=head2 Builder ++ ++Test::Builder::Module provides some methods of getting at the underlying ++Test::Builder object. ++ ++=head3 builder ++ ++ my $builder = Your::Class->builder; ++ ++This method returns the Test::Builder object associated with Your::Class. ++It is not a constructor so you can call it as often as you like. ++ ++This is the preferred way to get the Test::Builder object. You should ++I<not> get it via C<< Test::Builder->new >> as was previously ++recommended. ++ ++The object returned by builder() may change at runtime so you should ++call builder() inside each function rather than store it in a global. ++ ++ sub ok { ++ my $builder = Your::Class->builder; ++ ++ return $builder->ok(@_); ++ } ++ ++ ++=cut ++ ++sub builder { ++ return Test::Builder->new; ++} ++ ++1; +diff -Naur old/test_simple_patch/lib/Test/Builder/Tester/Color.pm new/test_simple_patch/lib/Test/Builder/Tester/Color.pm +--- old/test_simple_patch/lib/Test/Builder/Tester/Color.pm 1970-01-01 10:00:00.000000000 +1000 ++++ new/test_simple_patch/lib/Test/Builder/Tester/Color.pm 2014-03-26 21:48:11.510257612 +1100 +@@ -0,0 +1,51 @@ ++package Test::Builder::Tester::Color; ++ ++use strict; ++our $VERSION = "1.23_002"; ++ ++require Test::Builder::Tester; ++ ++ ++=head1 NAME ++ ++Test::Builder::Tester::Color - turn on colour in Test::Builder::Tester ++ ++=head1 SYNOPSIS ++ ++ When running a test script ++ ++ perl -MTest::Builder::Tester::Color test.t ++ ++=head1 DESCRIPTION ++ ++Importing this module causes the subroutine color in Test::Builder::Tester ++to be called with a true value causing colour highlighting to be turned ++on in debug output. ++ ++The sole purpose of this module is to enable colour highlighting ++from the command line. ++ ++=cut ++ ++sub import { ++ Test::Builder::Tester::color(1); ++} ++ ++=head1 AUTHOR ++ ++Copyright Mark Fowler E<lt>mark@xxxxxxxxxxxxxxxxxxx<gt> 2002. ++ ++This program is free software; you can redistribute it ++and/or modify it under the same terms as Perl itself. ++ ++=head1 BUGS ++ ++This module will have no effect unless Term::ANSIColor is installed. ++ ++=head1 SEE ALSO ++ ++L<Test::Builder::Tester>, L<Term::ANSIColor> ++ ++=cut ++ ++1; +diff -Naur old/test_simple_patch/lib/Test/Builder/Tester.pm new/test_simple_patch/lib/Test/Builder/Tester.pm +--- old/test_simple_patch/lib/Test/Builder/Tester.pm 1970-01-01 10:00:00.000000000 +1000 ++++ new/test_simple_patch/lib/Test/Builder/Tester.pm 2014-03-26 21:48:11.511257623 +1100 +@@ -0,0 +1,620 @@ ++package Test::Builder::Tester; ++ ++use strict; ++our $VERSION = "1.23_003"; ++ ++use Test::Builder 0.98; ++use Symbol; ++use Carp; ++ ++=head1 NAME ++ ++Test::Builder::Tester - test testsuites that have been built with ++Test::Builder ++ ++=head1 SYNOPSIS ++ ++ use Test::Builder::Tester tests => 1; ++ use Test::More; ++ ++ test_out("not ok 1 - foo"); ++ test_fail(+1); ++ fail("foo"); ++ test_test("fail works"); ++ ++=head1 DESCRIPTION ++ ++A module that helps you test testing modules that are built with ++B<Test::Builder>. ++ ++The testing system is designed to be used by performing a three step ++process for each test you wish to test. This process starts with using ++C<test_out> and C<test_err> in advance to declare what the testsuite you ++are testing will output with B<Test::Builder> to stdout and stderr. ++ ++You then can run the test(s) from your test suite that call ++B<Test::Builder>. At this point the output of B<Test::Builder> is ++safely captured by B<Test::Builder::Tester> rather than being ++interpreted as real test output. ++ ++The final stage is to call C<test_test> that will simply compare what you ++predeclared to what B<Test::Builder> actually outputted, and report the ++results back with a "ok" or "not ok" (with debugging) to the normal ++output. ++ ++=cut ++ ++#### ++# set up testing ++#### ++ ++my $t = Test::Builder->new; ++ ++### ++# make us an exporter ++### ++ ++use Exporter; ++our @ISA = qw(Exporter); ++ ++our @EXPORT = qw(test_out test_err test_fail test_diag test_test line_num); ++ ++sub import { ++ my $class = shift; ++ my(@plan) = @_; ++ ++ my $caller = caller; ++ ++ $t->exported_to($caller); ++ $t->plan(@plan); ++ ++ my @imports = (); ++ foreach my $idx ( 0 .. $#plan ) { ++ if( $plan[$idx] eq 'import' ) { ++ @imports = @{ $plan[ $idx + 1 ] }; ++ last; ++ } ++ } ++ ++ __PACKAGE__->export_to_level( 1, __PACKAGE__, @imports ); ++} ++ ++### ++# set up file handles ++### ++ ++# create some private file handles ++my $output_handle = gensym; ++my $error_handle = gensym; ++ ++# and tie them to this package ++my $out = tie *$output_handle, "Test::Builder::Tester::Tie", "STDOUT"; ++my $err = tie *$error_handle, "Test::Builder::Tester::Tie", "STDERR"; ++ ++#### ++# exported functions ++#### ++ ++# for remembering that we're testing and where we're testing at ++my $testing = 0; ++my $testing_num; ++my $original_is_passing; ++ ++# remembering where the file handles were originally connected ++my $original_output_handle; ++my $original_failure_handle; ++my $original_todo_handle; ++ ++my $original_harness_env; ++ ++# function that starts testing and redirects the filehandles for now ++sub _start_testing { ++ # even if we're running under Test::Harness pretend we're not ++ # for now. This needed so Test::Builder doesn't add extra spaces ++ $original_harness_env = $ENV{HARNESS_ACTIVE} || 0; ++ $ENV{HARNESS_ACTIVE} = 0; ++ ++ # remember what the handles were set to ++ $original_output_handle = $t->output(); ++ $original_failure_handle = $t->failure_output(); ++ $original_todo_handle = $t->todo_output(); ++ ++ # switch out to our own handles ++ $t->output($output_handle); ++ $t->failure_output($error_handle); ++ $t->todo_output($output_handle); ++ ++ # clear the expected list ++ $out->reset(); ++ $err->reset(); ++ ++ # remember that we're testing ++ $testing = 1; ++ $testing_num = $t->current_test; ++ $t->current_test(0); ++ $original_is_passing = $t->is_passing; ++ $t->is_passing(1); ++ ++ # look, we shouldn't do the ending stuff ++ $t->no_ending(1); ++} ++ ++=head2 Functions ++ ++These are the six methods that are exported as default. ++ ++=over 4 ++ ++=item test_out ++ ++=item test_err ++ ++Procedures for predeclaring the output that your test suite is ++expected to produce until C<test_test> is called. These procedures ++automatically assume that each line terminates with "\n". So ++ ++ test_out("ok 1","ok 2"); ++ ++is the same as ++ ++ test_out("ok 1\nok 2"); ++ ++which is even the same as ++ ++ test_out("ok 1"); ++ test_out("ok 2"); ++ ++Once C<test_out> or C<test_err> (or C<test_fail> or C<test_diag>) have ++been called, all further output from B<Test::Builder> will be ++captured by B<Test::Builder::Tester>. This means that you will not ++be able perform further tests to the normal output in the normal way ++until you call C<test_test> (well, unless you manually meddle with the ++output filehandles) ++ ++=cut ++ ++sub test_out { ++ # do we need to do any setup? ++ _start_testing() unless $testing; ++ ++ $out->expect(@_); ++} ++ ++sub test_err { ++ # do we need to do any setup? ++ _start_testing() unless $testing; ++ ++ $err->expect(@_); ++} ++ ++=item test_fail ++ ++Because the standard failure message that B<Test::Builder> produces ++whenever a test fails will be a common occurrence in your test error ++output, and because it has changed between Test::Builder versions, rather ++than forcing you to call C<test_err> with the string all the time like ++so ++ ++ test_err("# Failed test ($0 at line ".line_num(+1).")"); ++ ++C<test_fail> exists as a convenience function that can be called ++instead. It takes one argument, the offset from the current line that ++the line that causes the fail is on. ++ ++ test_fail(+1); ++ ++This means that the example in the synopsis could be rewritten ++more simply as: ++ ++ test_out("not ok 1 - foo"); ++ test_fail(+1); ++ fail("foo"); ++ test_test("fail works"); ++ ++=cut ++ ++sub test_fail { ++ # do we need to do any setup? ++ _start_testing() unless $testing; ++ ++ # work out what line we should be on ++ my( $package, $filename, $line ) = caller; ++ $line = $line + ( shift() || 0 ); # prevent warnings ++ ++ # expect that on stderr ++ $err->expect("# Failed test ($filename at line $line)"); ++} ++ ++=item test_diag ++ ++As most of the remaining expected output to the error stream will be ++created by Test::Builder's C<diag> function, B<Test::Builder::Tester> ++provides a convenience function C<test_diag> that you can use instead of ++C<test_err>. ++ ++The C<test_diag> function prepends comment hashes and spacing to the ++start and newlines to the end of the expected output passed to it and ++adds it to the list of expected error output. So, instead of writing ++ ++ test_err("# Couldn't open file"); ++ ++you can write ++ ++ test_diag("Couldn't open file"); ++ ++Remember that B<Test::Builder>'s diag function will not add newlines to ++the end of output and test_diag will. So to check ++ ++ Test::Builder->new->diag("foo\n","bar\n"); ++ ++You would do ++ ++ test_diag("foo","bar") ++ ++without the newlines. ++ ++=cut ++ ++sub test_diag { ++ # do we need to do any setup? ++ _start_testing() unless $testing; ++ ++ # expect the same thing, but prepended with "# " ++ local $_; ++ $err->expect( map { "# $_" } @_ ); ++} ++ ++=item test_test ++ ++Actually performs the output check testing the tests, comparing the ++data (with C<eq>) that we have captured from B<Test::Builder> against ++what was declared with C<test_out> and C<test_err>. ++ ++This takes name/value pairs that effect how the test is run. ++ ++=over ++ ++=item title (synonym 'name', 'label') ++ ++The name of the test that will be displayed after the C<ok> or C<not ++ok>. ++ ++=item skip_out ++ ++Setting this to a true value will cause the test to ignore if the ++output sent by the test to the output stream does not match that ++declared with C<test_out>. ++ ++=item skip_err ++ ++Setting this to a true value will cause the test to ignore if the ++output sent by the test to the error stream does not match that ++declared with C<test_err>. ++ ++=back ++ ++As a convenience, if only one argument is passed then this argument ++is assumed to be the name of the test (as in the above examples.) ++ ++Once C<test_test> has been run test output will be redirected back to ++the original filehandles that B<Test::Builder> was connected to ++(probably STDOUT and STDERR,) meaning any further tests you run ++will function normally and cause success/errors for B<Test::Harness>. ++ ++=cut ++ ++sub test_test { ++ # decode the arguments as described in the pod ++ my $mess; ++ my %args; ++ if( @_ == 1 ) { ++ $mess = shift ++ } ++ else { ++ %args = @_; ++ $mess = $args{name} if exists( $args{name} ); ++ $mess = $args{title} if exists( $args{title} ); ++ $mess = $args{label} if exists( $args{label} ); ++ } ++ ++ # er, are we testing? ++ croak "Not testing. You must declare output with a test function first." ++ unless $testing; ++ ++ # okay, reconnect the test suite back to the saved handles ++ $t->output($original_output_handle); ++ $t->failure_output($original_failure_handle); ++ $t->todo_output($original_todo_handle); ++ ++ # restore the test no, etc, back to the original point ++ $t->current_test($testing_num); ++ $testing = 0; ++ $t->is_passing($original_is_passing); ++ ++ # re-enable the original setting of the harness ++ $ENV{HARNESS_ACTIVE} = $original_harness_env; ++ ++ # check the output we've stashed ++ unless( $t->ok( ( $args{skip_out} || $out->check ) && ++ ( $args{skip_err} || $err->check ), $mess ) ++ ) ++ { ++ # print out the diagnostic information about why this ++ # test failed ++ ++ local $_; ++ ++ $t->diag( map { "$_\n" } $out->complaint ) ++ unless $args{skip_out} || $out->check; ++ ++ $t->diag( map { "$_\n" } $err->complaint ) ++ unless $args{skip_err} || $err->check; ++ } ++} ++ ++=item line_num ++ ++A utility function that returns the line number that the function was ++called on. You can pass it an offset which will be added to the ++result. This is very useful for working out the correct text of ++diagnostic functions that contain line numbers. ++ ++Essentially this is the same as the C<__LINE__> macro, but the ++C<line_num(+3)> idiom is arguably nicer. ++ ++=cut ++ ++sub line_num { ++ my( $package, $filename, $line ) = caller; ++ return $line + ( shift() || 0 ); # prevent warnings ++} ++ ++=back ++ ++In addition to the six exported functions there exists one ++function that can only be accessed with a fully qualified function ++call. ++ ++=over 4 ++ ++=item color ++ ++When C<test_test> is called and the output that your tests generate ++does not match that which you declared, C<test_test> will print out ++debug information showing the two conflicting versions. As this ++output itself is debug information it can be confusing which part of ++the output is from C<test_test> and which was the original output from ++your original tests. Also, it may be hard to spot things like ++extraneous whitespace at the end of lines that may cause your test to ++fail even though the output looks similar. ++ ++To assist you C<test_test> can colour the background of the debug ++information to disambiguate the different types of output. The debug ++output will have its background coloured green and red. The green ++part represents the text which is the same between the executed and ++actual output, the red shows which part differs. ++ ++The C<color> function determines if colouring should occur or not. ++Passing it a true or false value will enable or disable colouring ++respectively, and the function called with no argument will return the ++current setting. ++ ++To enable colouring from the command line, you can use the ++B<Text::Builder::Tester::Color> module like so: ++ ++ perl -Mlib=Text::Builder::Tester::Color test.t ++ ++Or by including the B<Test::Builder::Tester::Color> module directly in ++the PERL5LIB. ++ ++=cut ++ ++my $color; ++ ++sub color { ++ $color = shift if @_; ++ $color; ++} ++ ++=back ++ ++=head1 BUGS ++ ++Calls C<<Test::Builder->no_ending>> turning off the ending tests. ++This is needed as otherwise it will trip out because we've run more ++tests than we strictly should have and it'll register any failures we ++had that we were testing for as real failures. ++ ++The color function doesn't work unless B<Term::ANSIColor> is ++compatible with your terminal. ++ ++Bugs (and requests for new features) can be reported to the author ++though the CPAN RT system: ++L<http://rt.cpan.org/NoAuth/ReportBug.html?Queue=Test-Builder-Tester> ++ ++=head1 AUTHOR ++ ++Copyright Mark Fowler E<lt>mark@xxxxxxxxxxxxxxxxxxx<gt> 2002, 2004. ++ ++Some code taken from B<Test::More> and B<Test::Catch>, written by ++Michael G Schwern E<lt>schwern@xxxxxxxxxx<gt>. Hence, those parts ++Copyright Micheal G Schwern 2001. Used and distributed with ++permission. ++ ++This program is free software; you can redistribute it ++and/or modify it under the same terms as Perl itself. ++ ++=head1 MAINTAINERS ++ ++=over 4 ++ ++=item Chad Granum E<lt>exodist@xxxxxxxxx<gt> ++ ++=back ++ ++=head1 NOTES ++ ++Thanks to Richard Clamp E<lt>richardc@xxxxxxxxxxxxxx<gt> for letting ++me use his testing system to try this module out on. ++ ++=head1 SEE ALSO ++ ++L<Test::Builder>, L<Test::Builder::Tester::Color>, L<Test::More>. ++ ++=cut ++ ++1; ++ ++#################################################################### ++# Helper class that is used to remember expected and received data ++ ++package Test::Builder::Tester::Tie; ++ ++## ++# add line(s) to be expected ++ ++sub expect { ++ my $self = shift; ++ ++ my @checks = @_; ++ foreach my $check (@checks) { ++ $check = $self->_account_for_subtest($check); ++ $check = $self->_translate_Failed_check($check); ++ push @{ $self->{wanted} }, ref $check ? $check : "$check\n"; ++ } ++} ++ ++sub _account_for_subtest { ++ my( $self, $check ) = @_; ++ ++ # Since we ship with Test::Builder, calling a private method is safe...ish. ++ return ref($check) ? $check : $t->_indent . $check; ++} ++ ++sub _translate_Failed_check { ++ my( $self, $check ) = @_; ++ ++ if( $check =~ /\A(.*)# (Failed .*test) \((.*?) at line (\d+)\)\Z(?!\n)/ ) { ++ $check = "/\Q$1\E#\\s+\Q$2\E.*?\\n?.*?\Qat $3\E line \Q$4\E.*\\n?/"; ++ } ++ ++ return $check; ++} ++ ++## ++# return true iff the expected data matches the got data ++ ++sub check { ++ my $self = shift; ++ ++ # turn off warnings as these might be undef ++ local $^W = 0; ++ ++ my @checks = @{ $self->{wanted} }; ++ my $got = $self->{got}; ++ foreach my $check (@checks) { ++ $check = "\Q$check\E" unless( $check =~ s,^/(.*)/$,$1, or ref $check ); ++ return 0 unless $got =~ s/^$check//; ++ } ++ ++ return length $got == 0; ++} ++ ++## ++# a complaint message about the inputs not matching (to be ++# used for debugging messages) ++ ++sub complaint { ++ my $self = shift; ++ my $type = $self->type; ++ my $got = $self->got; ++ my $wanted = join '', @{ $self->wanted }; ++ ++ # are we running in colour mode? ++ if(Test::Builder::Tester::color) { ++ # get color ++ eval { require Term::ANSIColor }; ++ unless($@) { ++ # colours ++ ++ my $green = Term::ANSIColor::color("black") . Term::ANSIColor::color("on_green"); ++ my $red = Term::ANSIColor::color("black") . Term::ANSIColor::color("on_red"); ++ my $reset = Term::ANSIColor::color("reset"); ++ ++ # work out where the two strings start to differ ++ my $char = 0; ++ $char++ while substr( $got, $char, 1 ) eq substr( $wanted, $char, 1 ); ++ ++ # get the start string and the two end strings ++ my $start = $green . substr( $wanted, 0, $char ); ++ my $gotend = $red . substr( $got, $char ) . $reset; ++ my $wantedend = $red . substr( $wanted, $char ) . $reset; ++ ++ # make the start turn green on and off ++ $start =~ s/\n/$reset\n$green/g; ++ ++ # make the ends turn red on and off ++ $gotend =~ s/\n/$reset\n$red/g; ++ $wantedend =~ s/\n/$reset\n$red/g; ++ ++ # rebuild the strings ++ $got = $start . $gotend; ++ $wanted = $start . $wantedend; ++ } ++ } ++ ++ return "$type is:\n" . "$got\nnot:\n$wanted\nas expected"; ++} ++ ++## ++# forget all expected and got data ++ ++sub reset { ++ my $self = shift; ++ %$self = ( ++ type => $self->{type}, ++ got => '', ++ wanted => [], ++ ); ++} ++ ++sub got { ++ my $self = shift; ++ return $self->{got}; ++} ++ ++sub wanted { ++ my $self = shift; ++ return $self->{wanted}; ++} ++ ++sub type { ++ my $self = shift; ++ return $self->{type}; ++} ++ ++### ++# tie interface ++### ++ ++sub PRINT { ++ my $self = shift; ++ $self->{got} .= join '', @_; ++} ++ ++sub TIEHANDLE { ++ my( $class, $type ) = @_; ++ ++ my $self = bless { type => $type }, $class; ++ ++ $self->reset; ++ ++ return $self; ++} ++ ++sub READ { } ++sub READLINE { } ++sub GETC { } ++sub FILENO { } ++ ++1; +diff -Naur old/test_simple_patch/lib/Test/Builder.pm new/test_simple_patch/lib/Test/Builder.pm +--- old/test_simple_patch/lib/Test/Builder.pm 1970-01-01 10:00:00.000000000 +1000 ++++ new/test_simple_patch/lib/Test/Builder.pm 2014-03-26 21:48:11.513257645 +1100 +@@ -0,0 +1,2667 @@ ++package Test::Builder; ++ ++use 5.006; ++use strict; ++use warnings; ++ ++our $VERSION = '1.001003'; ++$VERSION = eval $VERSION; ## no critic (BuiltinFunctions::ProhibitStringyEval) ++ ++BEGIN { ++ if( $] < 5.008 ) { ++ require Test::Builder::IO::Scalar; ++ } ++} ++ ++ ++# Make Test::Builder thread-safe for ithreads. ++BEGIN { ++ use Config; ++ # Load threads::shared when threads are turned on. ++ # 5.8.0's threads are so busted we no longer support them. ++ if( $] >= 5.008001 && $Config{useithreads} && $INC{'threads.pm'} ) { ++ require threads::shared; ++ ++ # Hack around YET ANOTHER threads::shared bug. It would ++ # occasionally forget the contents of the variable when sharing it. ++ # So we first copy the data, then share, then put our copy back. ++ *share = sub (\[$@%]) { ++ my $type = ref $_[0]; ++ my $data; ++ ++ if( $type eq 'HASH' ) { ++ %$data = %{ $_[0] }; ++ } ++ elsif( $type eq 'ARRAY' ) { ++ @$data = @{ $_[0] }; ++ } ++ elsif( $type eq 'SCALAR' ) { ++ $$data = ${ $_[0] }; ++ } ++ else { ++ die( "Unknown type: " . $type ); ++ } ++ ++ $_[0] = &threads::shared::share( $_[0] ); ++ ++ if( $type eq 'HASH' ) { ++ %{ $_[0] } = %$data; ++ } ++ elsif( $type eq 'ARRAY' ) { ++ @{ $_[0] } = @$data; ++ } ++ elsif( $type eq 'SCALAR' ) { ++ ${ $_[0] } = $$data; ++ } ++ else { ++ die( "Unknown type: " . $type ); ++ } ++ ++ return $_[0]; ++ }; ++ } ++ # 5.8.0's threads::shared is busted when threads are off ++ # and earlier Perls just don't have that module at all. ++ else { ++ *share = sub { return $_[0] }; ++ *lock = sub { 0 }; ++ } ++} ++ ++=head1 NAME ++ ++Test::Builder - Backend for building test libraries ++ ++=head1 SYNOPSIS ++ ++ package My::Test::Module; ++ use base 'Test::Builder::Module'; ++ ++ my $CLASS = __PACKAGE__; ++ ++ sub ok { ++ my($test, $name) = @_; ++ my $tb = $CLASS->builder; ++ ++ $tb->ok($test, $name); ++ } ++ ++ ++=head1 DESCRIPTION ++ ++Test::Simple and Test::More have proven to be popular testing modules, ++but they're not always flexible enough. Test::Builder provides a ++building block upon which to write your own test libraries I<which can ++work together>. ++ ++=head2 Construction ++ ++=over 4 ++ ++=item B<new> ++ ++ my $Test = Test::Builder->new; ++ ++Returns a Test::Builder object representing the current state of the ++test. ++ ++Since you only run one test per program C<new> always returns the same ++Test::Builder object. No matter how many times you call C<new()>, you're ++getting the same object. This is called a singleton. This is done so that ++multiple modules share such global information as the test counter and ++where test output is going. ++ ++If you want a completely new Test::Builder object different from the ++singleton, use C<create>. ++ ++=cut ++ ++our $Test = Test::Builder->new; ++ ++sub new { ++ my($class) = shift; ++ $Test ||= $class->create; ++ return $Test; ++} ++ ++=item B<create> ++ ++ my $Test = Test::Builder->create; ++ ++Ok, so there can be more than one Test::Builder object and this is how ++you get it. You might use this instead of C<new()> if you're testing ++a Test::Builder based module, but otherwise you probably want C<new>. ++ ++B<NOTE>: the implementation is not complete. C<level>, for example, is ++still shared amongst B<all> Test::Builder objects, even ones created using ++this method. Also, the method name may change in the future. ++ ++=cut ++ ++sub create { ++ my $class = shift; ++ ++ my $self = bless {}, $class; ++ $self->reset; ++ ++ return $self; ++} ++ ++ ++# Copy an object, currently a shallow. ++# This does *not* bless the destination. This keeps the destructor from ++# firing when we're just storing a copy of the object to restore later. ++sub _copy { ++ my($src, $dest) = @_; ++ ++ %$dest = %$src; ++ _share_keys($dest); ++ ++ return; ++} ++ ++ ++=item B<child> ++ ++ my $child = $builder->child($name_of_child); ++ $child->plan( tests => 4 ); ++ $child->ok(some_code()); ++ ... ++ $child->finalize; ++ ++Returns a new instance of C<Test::Builder>. Any output from this child will ++be indented four spaces more than the parent's indentation. When done, the ++C<finalize> method I<must> be called explicitly. ++ ++Trying to create a new child with a previous child still active (i.e., ++C<finalize> not called) will C<croak>. ++ ++Trying to run a test when you have an open child will also C<croak> and cause ++the test suite to fail. ++ ++=cut ++ ++sub child { ++ my( $self, $name ) = @_; ++ ++ if( $self->{Child_Name} ) { ++ $self->croak("You already have a child named ($self->{Child_Name}) running"); ++ } ++ ++ my $parent_in_todo = $self->in_todo; ++ ++ # Clear $TODO for the child. ++ my $orig_TODO = $self->find_TODO(undef, 1, undef); ++ ++ my $class = ref $self; ++ my $child = $class->create; ++ ++ # Add to our indentation ++ $child->_indent( $self->_indent . ' ' ); ++ ++ # Make the child use the same outputs as the parent ++ for my $method (qw(output failure_output todo_output)) { ++ $child->$method( $self->$method ); ++ } ++ ++ # Ensure the child understands if they're inside a TODO ++ if( $parent_in_todo ) { ++ $child->failure_output( $self->todo_output ); ++ } ++ ++ # This will be reset in finalize. We do this here lest one child failure ++ # cause all children to fail. ++ $child->{Child_Error} = $?; ++ $? = 0; ++ $child->{Parent} = $self; ++ $child->{Parent_TODO} = $orig_TODO; ++ $child->{Name} = $name || "Child of " . $self->name; ++ $self->{Child_Name} = $child->name; ++ return $child; ++} ++ ++ ++=item B<subtest> ++ ++ $builder->subtest($name, \&subtests); ++ ++See documentation of C<subtest> in Test::More. ++ ++=cut ++ ++sub subtest { ++ my $self = shift; ++ my($name, $subtests) = @_; ++ ++ if ('CODE' ne ref $subtests) { ++ $self->croak("subtest()'s second argument must be a code ref"); ++ } ++ ++ # Turn the child into the parent so anyone who has stored a copy of ++ # the Test::Builder singleton will get the child. ++ my $error; ++ my $child; ++ my $parent = {}; ++ { ++ # child() calls reset() which sets $Level to 1, so we localize ++ # $Level first to limit the scope of the reset to the subtest. ++ local $Test::Builder::Level = $Test::Builder::Level + 1; ++ ++ # Store the guts of $self as $parent and turn $child into $self. ++ $child = $self->child($name); ++ _copy($self, $parent); ++ _copy($child, $self); ++ ++ my $run_the_subtests = sub { ++ # Add subtest name for clarification of starting point ++ $self->note("Subtest: $name"); ++ $subtests->(); ++ $self->done_testing unless $self->_plan_handled; ++ 1; ++ }; ++ ++ if( !eval { $run_the_subtests->() } ) { ++ $error = $@; ++ } ++ } ++ ++ # Restore the parent and the copied child. ++ _copy($self, $child); ++ _copy($parent, $self); ++ ++ # Restore the parent's $TODO ++ $self->find_TODO(undef, 1, $child->{Parent_TODO}); ++ ++ # Die *after* we restore the parent. ++ die $error if $error and !eval { $error->isa('Test::Builder::Exception') }; ++ ++ local $Test::Builder::Level = $Test::Builder::Level + 1; ++ my $finalize = $child->finalize; ++ ++ $self->BAIL_OUT($child->{Bailed_Out_Reason}) if $child->{Bailed_Out}; ++ ++ return $finalize; ++} ++ ++=begin _private ++ ++=item B<_plan_handled> ++ ++ if ( $Test->_plan_handled ) { ... } ++ ++Returns true if the developer has explicitly handled the plan via: ++ ++=over 4 ++ ++=item * Explicitly setting the number of tests ++ ++=item * Setting 'no_plan' ++ ++=item * Set 'skip_all'. ++ ++=back ++ ++This is currently used in subtests when we implicitly call C<< $Test->done_testing >> ++if the developer has not set a plan. ++ ++=end _private ++ ++=cut ++ ++sub _plan_handled { ++ my $self = shift; ++ return $self->{Have_Plan} || $self->{No_Plan} || $self->{Skip_All}; ++} ++ ++ ++=item B<finalize> ++ ++ my $ok = $child->finalize; ++ ++When your child is done running tests, you must call C<finalize> to clean up ++and tell the parent your pass/fail status. ++ ++Calling finalize on a child with open children will C<croak>. ++ ++If the child falls out of scope before C<finalize> is called, a failure ++diagnostic will be issued and the child is considered to have failed. ++ ++No attempt to call methods on a child after C<finalize> is called is ++guaranteed to succeed. ++ ++Calling this on the root builder is a no-op. ++ ++=cut ++ ++sub finalize { ++ my $self = shift; ++ ++ return unless $self->parent; ++ if( $self->{Child_Name} ) { ++ $self->croak("Can't call finalize() with child ($self->{Child_Name}) active"); ++ } ++ ++ local $? = 0; # don't fail if $subtests happened to set $? nonzero ++ $self->_ending; ++ ++ # XXX This will only be necessary for TAP envelopes (we think) ++ #$self->_print( $self->is_passing ? "PASS\n" : "FAIL\n" ); ++ ++ local $Test::Builder::Level = $Test::Builder::Level + 1; ++ my $ok = 1; ++ $self->parent->{Child_Name} = undef; ++ unless ($self->{Bailed_Out}) { ++ if ( $self->{Skip_All} ) { ++ $self->parent->skip($self->{Skip_All}); ++ } ++ elsif ( not @{ $self->{Test_Results} } ) { ++ $self->parent->ok( 0, sprintf q[No tests run for subtest "%s"], $self->name ); ++ } ++ else { ++ $self->parent->ok( $self->is_passing, $self->name ); ++ } ++ } ++ $? = $self->{Child_Error}; ++ delete $self->{Parent}; ++ ++ return $self->is_passing; ++} ++ ++sub _indent { ++ my $self = shift; ++ ++ if( @_ ) { ++ $self->{Indent} = shift; ++ } ++ ++ return $self->{Indent}; ++} ++ ++=item B<parent> ++ ++ if ( my $parent = $builder->parent ) { ++ ... ++ } ++ ++Returns the parent C<Test::Builder> instance, if any. Only used with child ++builders for nested TAP. ++ ++=cut ++ ++sub parent { shift->{Parent} } ++ ++=item B<name> ++ ++ diag $builder->name; ++ ++Returns the name of the current builder. Top level builders default to C<$0> ++(the name of the executable). Child builders are named via the C<child> ++method. If no name is supplied, will be named "Child of $parent->name". ++ ++=cut ++ ++sub name { shift->{Name} } ++ ++sub DESTROY { ++ my $self = shift; ++ if ( $self->parent and $$ == $self->{Original_Pid} ) { ++ my $name = $self->name; ++ $self->diag(<<"FAIL"); ++Child ($name) exited without calling finalize() ++FAIL ++ $self->parent->{In_Destroy} = 1; ++ $self->parent->ok(0, $name); ++ } ++} ++ ++=item B<reset> ++ ++ $Test->reset; ++ ++Reinitializes the Test::Builder singleton to its original state. ++Mostly useful for tests run in persistent environments where the same ++test might be run multiple times in the same process. ++ ++=cut ++ ++our $Level; ++ ++sub reset { ## no critic (Subroutines::ProhibitBuiltinHomonyms) ++ my($self) = @_; ++ ++ # We leave this a global because it has to be localized and localizing ++ # hash keys is just asking for pain. Also, it was documented. ++ $Level = 1; ++ ++ $self->{Name} = $0; ++ $self->is_passing(1); ++ $self->{Ending} = 0; ++ $self->{Have_Plan} = 0; ++ $self->{No_Plan} = 0; ++ $self->{Have_Output_Plan} = 0; ++ $self->{Done_Testing} = 0; ++ ++ $self->{Original_Pid} = $$; ++ $self->{Child_Name} = undef; ++ $self->{Indent} ||= ''; ++ ++ $self->{Curr_Test} = 0; ++ $self->{Test_Results} = &share( [] ); ++ ++ $self->{Exported_To} = undef; ++ $self->{Expected_Tests} = 0; ++ ++ $self->{Skip_All} = 0; ++ ++ $self->{Use_Nums} = 1; ++ ++ $self->{No_Header} = 0; ++ $self->{No_Ending} = 0; ++ ++ $self->{Todo} = undef; ++ $self->{Todo_Stack} = []; ++ $self->{Start_Todo} = 0; ++ $self->{Opened_Testhandles} = 0; ++ ++ $self->_share_keys; ++ $self->_dup_stdhandles; ++ ++ return; ++} ++ ++ ++# Shared scalar values are lost when a hash is copied, so we have ++# a separate method to restore them. ++# Shared references are retained across copies. ++sub _share_keys { ++ my $self = shift; ++ ++ share( $self->{Curr_Test} ); ++ ++ return; ++} ++ ++ ++=back ++ ++=head2 Setting up tests ++ ++These methods are for setting up tests and declaring how many there ++are. You usually only want to call one of these methods. ++ ++=over 4 ++ ++=item B<plan> ++ ++ $Test->plan('no_plan'); ++ $Test->plan( skip_all => $reason ); ++ $Test->plan( tests => $num_tests ); ++ ++A convenient way to set up your tests. Call this and Test::Builder ++will print the appropriate headers and take the appropriate actions. ++ ++If you call C<plan()>, don't call any of the other methods below. ++ ++If a child calls "skip_all" in the plan, a C<Test::Builder::Exception> is ++thrown. Trap this error, call C<finalize()> and don't run any more tests on ++the child. ++ ++ my $child = $Test->child('some child'); ++ eval { $child->plan( $condition ? ( skip_all => $reason ) : ( tests => 3 ) ) }; ++ if ( eval { $@->isa('Test::Builder::Exception') } ) { ++ $child->finalize; ++ return; ++ } ++ # run your tests ++ ++=cut ++ ++my %plan_cmds = ( ++ no_plan => \&no_plan, ++ skip_all => \&skip_all, ++ tests => \&_plan_tests, ++); ++ ++sub plan { ++ my( $self, $cmd, $arg ) = @_; ++ ++ return unless $cmd; ++ ++ local $Level = $Level + 1; ++ ++ $self->croak("You tried to plan twice") if $self->{Have_Plan}; ++ ++ if( my $method = $plan_cmds{$cmd} ) { ++ local $Level = $Level + 1; ++ $self->$method($arg); ++ } ++ else { ++ my @args = grep { defined } ( $cmd, $arg ); ++ $self->croak("plan() doesn't understand @args"); ++ } ++ ++ return 1; ++} ++ ++ ++sub _plan_tests { ++ my($self, $arg) = @_; ++ ++ if($arg) { ++ local $Level = $Level + 1; ++ return $self->expected_tests($arg); ++ } ++ elsif( !defined $arg ) { ++ $self->croak("Got an undefined number of tests"); ++ } ++ else { ++ $self->croak("You said to run 0 tests"); ++ } ++ ++ return; ++} ++ ++=item B<expected_tests> ++ ++ my $max = $Test->expected_tests; ++ $Test->expected_tests($max); ++ ++Gets/sets the number of tests we expect this test to run and prints out ++the appropriate headers. ++ ++=cut ++ ++sub expected_tests { ++ my $self = shift; ++ my($max) = @_; ++ ++ if(@_) { ++ $self->croak("Number of tests must be a positive integer. You gave it '$max'") ++ unless $max =~ /^\+?\d+$/; ++ ++ $self->{Expected_Tests} = $max; ++ $self->{Have_Plan} = 1; ++ ++ $self->_output_plan($max) unless $self->no_header; ++ } ++ return $self->{Expected_Tests}; ++} ++ ++=item B<no_plan> ++ ++ $Test->no_plan; ++ ++Declares that this test will run an indeterminate number of tests. ++ ++=cut ++ ++sub no_plan { ++ my($self, $arg) = @_; ++ ++ $self->carp("no_plan takes no arguments") if $arg; ++ ++ $self->{No_Plan} = 1; ++ $self->{Have_Plan} = 1; ++ ++ return 1; ++} ++ ++=begin private ++ ++=item B<_output_plan> ++ ++ $tb->_output_plan($max); ++ $tb->_output_plan($max, $directive); ++ $tb->_output_plan($max, $directive => $reason); ++ ++Handles displaying the test plan. ++ ++If a C<$directive> and/or C<$reason> are given they will be output with the ++plan. So here's what skipping all tests looks like: ++ ++ $tb->_output_plan(0, "SKIP", "Because I said so"); ++ ++It sets C<< $tb->{Have_Output_Plan} >> and will croak if the plan was already ++output. ++ ++=end private ++ ++=cut ++ ++sub _output_plan { ++ my($self, $max, $directive, $reason) = @_; ++ ++ $self->carp("The plan was already output") if $self->{Have_Output_Plan}; ++ ++ my $plan = "1..$max"; ++ $plan .= " # $directive" if defined $directive; ++ $plan .= " $reason" if defined $reason; ++ ++ $self->_print("$plan\n"); ++ ++ $self->{Have_Output_Plan} = 1; ++ ++ return; ++} ++ ++ ++=item B<done_testing> ++ ++ $Test->done_testing(); ++ $Test->done_testing($num_tests); ++ ++Declares that you are done testing, no more tests will be run after this point. ++ ++If a plan has not yet been output, it will do so. ++ ++$num_tests is the number of tests you planned to run. If a numbered ++plan was already declared, and if this contradicts, a failing test ++will be run to reflect the planning mistake. If C<no_plan> was declared, ++this will override. ++ ++If C<done_testing()> is called twice, the second call will issue a ++failing test. ++ ++If C<$num_tests> is omitted, the number of tests run will be used, like ++no_plan. ++ ++C<done_testing()> is, in effect, used when you'd want to use C<no_plan>, but ++safer. You'd use it like so: ++ ++ $Test->ok($a == $b); ++ $Test->done_testing(); ++ ++Or to plan a variable number of tests: ++ ++ for my $test (@tests) { ++ $Test->ok($test); ++ } ++ $Test->done_testing(scalar @tests); ++ ++=cut ++ ++sub done_testing { ++ my($self, $num_tests) = @_; ++ ++ # If done_testing() specified the number of tests, shut off no_plan. ++ if( defined $num_tests ) { ++ $self->{No_Plan} = 0; ++ } ++ else { ++ $num_tests = $self->current_test; ++ } ++ ++ if( $self->{Done_Testing} ) { ++ my($file, $line) = @{$self->{Done_Testing}}[1,2]; ++ $self->ok(0, "done_testing() was already called at $file line $line"); ++ return; ++ } ++ ++ $self->{Done_Testing} = [caller]; ++ ++ if( $self->expected_tests && $num_tests != $self->expected_tests ) { ++ $self->ok(0, "planned to run @{[ $self->expected_tests ]} ". ++ "but done_testing() expects $num_tests"); ++ } ++ else { ++ $self->{Expected_Tests} = $num_tests; ++ } ++ ++ $self->_output_plan($num_tests) unless $self->{Have_Output_Plan}; ++ ++ $self->{Have_Plan} = 1; ++ ++ # The wrong number of tests were run ++ $self->is_passing(0) if $self->{Expected_Tests} != $self->{Curr_Test}; ++ ++ # No tests were run ++ $self->is_passing(0) if $self->{Curr_Test} == 0; ++ ++ return 1; ++} ++ ++ ++=item B<has_plan> ++ ++ $plan = $Test->has_plan ++ ++Find out whether a plan has been defined. C<$plan> is either C<undef> (no plan ++has been set), C<no_plan> (indeterminate # of tests) or an integer (the number ++of expected tests). ++ ++=cut ++ ++sub has_plan { ++ my $self = shift; ++ ++ return( $self->{Expected_Tests} ) if $self->{Expected_Tests}; ++ return('no_plan') if $self->{No_Plan}; ++ return(undef); ++} ++ ++=item B<skip_all> ++ ++ $Test->skip_all; ++ $Test->skip_all($reason); ++ ++Skips all the tests, using the given C<$reason>. Exits immediately with 0. ++ ++=cut ++ ++sub skip_all { ++ my( $self, $reason ) = @_; ++ ++ $self->{Skip_All} = $self->parent ? $reason : 1; ++ ++ $self->_output_plan(0, "SKIP", $reason) unless $self->no_header; ++ if ( $self->parent ) { ++ die bless {} => 'Test::Builder::Exception'; ++ } ++ exit(0); ++} ++ ++=item B<exported_to> ++ ++ my $pack = $Test->exported_to; ++ $Test->exported_to($pack); ++ ++Tells Test::Builder what package you exported your functions to. ++ ++This method isn't terribly useful since modules which share the same ++Test::Builder object might get exported to different packages and only ++the last one will be honored. ++ ++=cut ++ ++sub exported_to { ++ my( $self, $pack ) = @_; ++ ++ if( defined $pack ) { ++ $self->{Exported_To} = $pack; ++ } ++ return $self->{Exported_To}; ++} ++ ++=back ++ ++=head2 Running tests ++ ++These actually run the tests, analogous to the functions in Test::More. ++ ++They all return true if the test passed, false if the test failed. ++ ++C<$name> is always optional. ++ ++=over 4 ++ ++=item B<ok> ++ ++ $Test->ok($test, $name); ++ ++Your basic test. Pass if C<$test> is true, fail if $test is false. Just ++like Test::Simple's C<ok()>. ++ ++=cut ++ ++sub ok { ++ my( $self, $test, $name ) = @_; ++ ++ if ( $self->{Child_Name} and not $self->{In_Destroy} ) { ++ $name = 'unnamed test' unless defined $name; ++ $self->is_passing(0); ++ $self->croak("Cannot run test ($name) with active children"); ++ } ++ # $test might contain an object which we don't want to accidentally ++ # store, so we turn it into a boolean. ++ $test = $test ? 1 : 0; ++ ++ lock $self->{Curr_Test}; ++ $self->{Curr_Test}++; ++ ++ # In case $name is a string overloaded object, force it to stringify. ++ $self->_unoverload_str( \$name ); ++ ++ $self->diag(<<"ERR") if defined $name and $name =~ /^[\d\s]+$/; ++ You named your test '$name'. You shouldn't use numbers for your test names. ++ Very confusing. ++ERR ++ ++ # Capture the value of $TODO for the rest of this ok() call ++ # so it can more easily be found by other routines. ++ my $todo = $self->todo(); ++ my $in_todo = $self->in_todo; ++ local $self->{Todo} = $todo if $in_todo; ++ ++ $self->_unoverload_str( \$todo ); ++ ++ my $out; ++ my $result = &share( {} ); ++ ++ unless($test) { ++ $out .= "not "; ++ @$result{ 'ok', 'actual_ok' } = ( ( $self->in_todo ? 1 : 0 ), 0 ); ++ } ++ else { ++ @$result{ 'ok', 'actual_ok' } = ( 1, $test ); ++ } ++ ++ $out .= "ok"; ++ $out .= " $self->{Curr_Test}" if $self->use_numbers; ++ ++ if( defined $name ) { ++ $name =~ s|#|\\#|g; # # in a name can confuse Test::Harness. ++ $out .= " - $name"; ++ $result->{name} = $name; ++ } ++ else { ++ $result->{name} = ''; ++ } ++ ++ if( $self->in_todo ) { ++ $out .= " # TODO $todo"; ++ $result->{reason} = $todo; ++ $result->{type} = 'todo'; ++ } ++ else { ++ $result->{reason} = ''; ++ $result->{type} = ''; ++ } ++ ++ $self->{Test_Results}[ $self->{Curr_Test} - 1 ] = $result; ++ $out .= "\n"; ++ ++ $self->_print($out); ++ ++ unless($test) { ++ my $msg = $self->in_todo ? "Failed (TODO)" : "Failed"; ++ $self->_print_to_fh( $self->_diag_fh, "\n" ) if $ENV{HARNESS_ACTIVE}; ++ ++ my( undef, $file, $line ) = $self->caller; ++ if( defined $name ) { ++ $self->diag(qq[ $msg test '$name'\n]); ++ $self->diag(qq[ at $file line $line.\n]); ++ } ++ else { ++ $self->diag(qq[ $msg test at $file line $line.\n]); ++ } ++ } ++ ++ $self->is_passing(0) unless $test || $self->in_todo; ++ ++ # Check that we haven't violated the plan ++ $self->_check_is_passing_plan(); ++ ++ return $test ? 1 : 0; ++} ++ ++ ++# Check that we haven't yet violated the plan and set ++# is_passing() accordingly ++sub _check_is_passing_plan { ++ my $self = shift; ++ ++ my $plan = $self->has_plan; ++ return unless defined $plan; # no plan yet defined ++ return unless $plan !~ /\D/; # no numeric plan ++ $self->is_passing(0) if $plan < $self->{Curr_Test}; ++} ++ ++ ++sub _unoverload { ++ my $self = shift; ++ my $type = shift; ++ ++ $self->_try(sub { require overload; }, die_on_fail => 1); ++ ++ foreach my $thing (@_) { ++ if( $self->_is_object($$thing) ) { ++ if( my $string_meth = overload::Method( $$thing, $type ) ) { ++ $$thing = $$thing->$string_meth(); ++ } ++ } ++ } ++ ++ return; ++} ++ ++sub _is_object { ++ my( $self, $thing ) = @_; ++ ++ return $self->_try( sub { ref $thing && $thing->isa('UNIVERSAL') } ) ? 1 : 0; ++} ++ ++sub _unoverload_str { ++ my $self = shift; ++ ++ return $self->_unoverload( q[""], @_ ); ++} ++ ++sub _unoverload_num { ++ my $self = shift; ++ ++ $self->_unoverload( '0+', @_ ); ++ ++ for my $val (@_) { ++ next unless $self->_is_dualvar($$val); ++ $$val = $$val + 0; ++ } ++ ++ return; ++} ++ ++# This is a hack to detect a dualvar such as $! ++sub _is_dualvar { ++ my( $self, $val ) = @_; ++ ++ # Objects are not dualvars. ++ return 0 if ref $val; ++ ++ no warnings 'numeric'; ++ my $numval = $val + 0; ++ return ($numval != 0 and $numval ne $val ? 1 : 0); ++} ++ ++=item B<is_eq> ++ ++ $Test->is_eq($got, $expected, $name); ++ ++Like Test::More's C<is()>. Checks if C<$got eq $expected>. This is the ++string version. ++ ++C<undef> only ever matches another C<undef>. ++ ++=item B<is_num> ++ ++ $Test->is_num($got, $expected, $name); ++ ++Like Test::More's C<is()>. Checks if C<$got == $expected>. This is the ++numeric version. ++ ++C<undef> only ever matches another C<undef>. ++ ++=cut ++ ++sub is_eq { ++ my( $self, $got, $expect, $name ) = @_; ++ local $Level = $Level + 1; ++ ++ if( !defined $got || !defined $expect ) { ++ # undef only matches undef and nothing else ++ my $test = !defined $got && !defined $expect; ++ ++ $self->ok( $test, $name ); ++ $self->_is_diag( $got, 'eq', $expect ) unless $test; ++ return $test; ++ } ++ ++ return $self->cmp_ok( $got, 'eq', $expect, $name ); ++} ++ ++sub is_num { ++ my( $self, $got, $expect, $name ) = @_; ++ local $Level = $Level + 1; ++ ++ if( !defined $got || !defined $expect ) { ++ # undef only matches undef and nothing else ++ my $test = !defined $got && !defined $expect; ++ ++ $self->ok( $test, $name ); ++ $self->_is_diag( $got, '==', $expect ) unless $test; ++ return $test; ++ } ++ ++ return $self->cmp_ok( $got, '==', $expect, $name ); ++} ++ ++sub _diag_fmt { ++ my( $self, $type, $val ) = @_; ++ ++ if( defined $$val ) { ++ if( $type eq 'eq' or $type eq 'ne' ) { ++ # quote and force string context ++ $$val = "'$$val'"; ++ } ++ else { ++ # force numeric context ++ $self->_unoverload_num($val); ++ } ++ } ++ else { ++ $$val = 'undef'; ++ } ++ ++ return; ++} ++ ++sub _is_diag { ++ my( $self, $got, $type, $expect ) = @_; ++ ++ $self->_diag_fmt( $type, $_ ) for \$got, \$expect; ++ ++ local $Level = $Level + 1; ++ return $self->diag(<<"DIAGNOSTIC"); ++ got: $got ++ expected: $expect ++DIAGNOSTIC ++ ++} ++ ++sub _isnt_diag { ++ my( $self, $got, $type ) = @_; ++ ++ $self->_diag_fmt( $type, \$got ); ++ ++ local $Level = $Level + 1; ++ return $self->diag(<<"DIAGNOSTIC"); ++ got: $got ++ expected: anything else ++DIAGNOSTIC ++} ++ ++=item B<isnt_eq> ++ ++ $Test->isnt_eq($got, $dont_expect, $name); ++ ++Like Test::More's C<isnt()>. Checks if C<$got ne $dont_expect>. This is ++the string version. ++ ++=item B<isnt_num> ++ ++ $Test->isnt_num($got, $dont_expect, $name); ++ ++Like Test::More's C<isnt()>. Checks if C<$got ne $dont_expect>. This is ++the numeric version. ++ ++=cut ++ ++sub isnt_eq { ++ my( $self, $got, $dont_expect, $name ) = @_; ++ local $Level = $Level + 1; ++ ++ if( !defined $got || !defined $dont_expect ) { ++ # undef only matches undef and nothing else ++ my $test = defined $got || defined $dont_expect; ++ ++ $self->ok( $test, $name ); ++ $self->_isnt_diag( $got, 'ne' ) unless $test; ++ return $test; ++ } ++ ++ return $self->cmp_ok( $got, 'ne', $dont_expect, $name ); ++} ++ ++sub isnt_num { ++ my( $self, $got, $dont_expect, $name ) = @_; ++ local $Level = $Level + 1; ++ ++ if( !defined $got || !defined $dont_expect ) { ++ # undef only matches undef and nothing else ++ my $test = defined $got || defined $dont_expect; ++ ++ $self->ok( $test, $name ); ++ $self->_isnt_diag( $got, '!=' ) unless $test; ++ return $test; ++ } ++ ++ return $self->cmp_ok( $got, '!=', $dont_expect, $name ); ++} ++ ++=item B<like> ++ ++ $Test->like($thing, qr/$regex/, $name); ++ $Test->like($thing, '/$regex/', $name); ++ ++Like Test::More's C<like()>. Checks if $thing matches the given C<$regex>. ++ ++=item B<unlike> ++ ++ $Test->unlike($thing, qr/$regex/, $name); ++ $Test->unlike($thing, '/$regex/', $name); ++ ++Like Test::More's C<unlike()>. Checks if $thing B<does not match> the ++given C<$regex>. ++ ++=cut ++ ++sub like { ++ my( $self, $thing, $regex, $name ) = @_; ++ ++ local $Level = $Level + 1; ++ return $self->_regex_ok( $thing, $regex, '=~', $name ); ++} ++ ++sub unlike { ++ my( $self, $thing, $regex, $name ) = @_; ++ ++ local $Level = $Level + 1; ++ return $self->_regex_ok( $thing, $regex, '!~', $name ); ++} ++ ++=item B<cmp_ok> ++ ++ $Test->cmp_ok($thing, $type, $that, $name); ++ ++Works just like Test::More's C<cmp_ok()>. ++ ++ $Test->cmp_ok($big_num, '!=', $other_big_num); ++ ++=cut ++ ++my %numeric_cmps = map { ( $_, 1 ) } ( "<", "<=", ">", ">=", "==", "!=", "<=>" ); ++ ++# Bad, these are not comparison operators. Should we include more? ++my %cmp_ok_bl = map { ( $_, 1 ) } ( "=", "+=", ".=", "x=", "^=", "|=", "||=", "&&=", "..."); ++ ++sub cmp_ok { ++ my( $self, $got, $type, $expect, $name ) = @_; ++ ++ if ($cmp_ok_bl{$type}) { ++ $self->croak("$type is not a valid comparison operator in cmp_ok()"); ++ } ++ ++ my $test; ++ my $error; ++ { ++ ## no critic (BuiltinFunctions::ProhibitStringyEval) ++ ++ local( $@, $!, $SIG{__DIE__} ); # isolate eval ++ ++ my($pack, $file, $line) = $self->caller(); ++ ++ # This is so that warnings come out at the caller's level ++ $test = eval qq[ ++#line $line "(eval in cmp_ok) $file" ++\$got $type \$expect; ++]; ++ $error = $@; ++ } ++ local $Level = $Level + 1; ++ my $ok = $self->ok( $test, $name ); ++ ++ # Treat overloaded objects as numbers if we're asked to do a ++ # numeric comparison. ++ my $unoverload ++ = $numeric_cmps{$type} ++ ? '_unoverload_num' ++ : '_unoverload_str'; ++ ++ $self->diag(<<"END") if $error; ++An error occurred while using $type: ++------------------------------------ ++$error ++------------------------------------ ++END ++ ++ unless($ok) { ++ $self->$unoverload( \$got, \$expect ); ++ ++ if( $type =~ /^(eq|==)$/ ) { ++ $self->_is_diag( $got, $type, $expect ); ++ } ++ elsif( $type =~ /^(ne|!=)$/ ) { ++ $self->_isnt_diag( $got, $type ); ++ } ++ else { ++ $self->_cmp_diag( $got, $type, $expect ); ++ } ++ } ++ return $ok; ++} ++ ++sub _cmp_diag { ++ my( $self, $got, $type, $expect ) = @_; ++ ++ $got = defined $got ? "'$got'" : 'undef'; ++ $expect = defined $expect ? "'$expect'" : 'undef'; ++ ++ local $Level = $Level + 1; ++ return $self->diag(<<"DIAGNOSTIC"); ++ $got ++ $type ++ $expect ++DIAGNOSTIC ++} ++ ++sub _caller_context { ++ my $self = shift; ++ ++ my( $pack, $file, $line ) = $self->caller(1); ++ ++ my $code = ''; ++ $code .= "#line $line $file\n" if defined $file and defined $line; ++ ++ return $code; ++} ++ ++=back ++ ++ ++=head2 Other Testing Methods ++ ++These are methods which are used in the course of writing a test but are not themselves tests. ++ ++=over 4 ++ ++=item B<BAIL_OUT> ++ ++ $Test->BAIL_OUT($reason); ++ ++Indicates to the Test::Harness that things are going so badly all ++testing should terminate. This includes running any additional test ++scripts. ++ ++It will exit with 255. ++ ++=cut ++ ++sub BAIL_OUT { ++ my( $self, $reason ) = @_; ++ ++ $self->{Bailed_Out} = 1; ++ ++ if ($self->parent) { ++ $self->{Bailed_Out_Reason} = $reason; ++ $self->no_ending(1); ++ die bless {} => 'Test::Builder::Exception'; ++ } ++ ++ $self->_print("Bail out! $reason"); ++ exit 255; ++} ++ ++=for deprecated ++BAIL_OUT() used to be BAILOUT() ++ ++=cut ++ ++{ ++ no warnings 'once'; ++ *BAILOUT = \&BAIL_OUT; ++} ++ ++=item B<skip> ++ ++ $Test->skip; ++ $Test->skip($why); ++ ++Skips the current test, reporting C<$why>. ++ ++=cut ++ ++sub skip { ++ my( $self, $why ) = @_; ++ $why ||= ''; ++ $self->_unoverload_str( \$why ); ++ ++ lock( $self->{Curr_Test} ); ++ $self->{Curr_Test}++; ++ ++ $self->{Test_Results}[ $self->{Curr_Test} - 1 ] = &share( ++ { ++ 'ok' => 1, ++ actual_ok => 1, ++ name => '', ++ type => 'skip', ++ reason => $why, ++ } ++ ); ++ ++ my $out = "ok"; ++ $out .= " $self->{Curr_Test}" if $self->use_numbers; ++ $out .= " # skip"; ++ $out .= " $why" if length $why; ++ $out .= "\n"; ++ ++ $self->_print($out); ++ ++ return 1; ++} ++ ++=item B<todo_skip> ++ ++ $Test->todo_skip; ++ $Test->todo_skip($why); ++ ++Like C<skip()>, only it will declare the test as failing and TODO. Similar ++to ++ ++ print "not ok $tnum # TODO $why\n"; ++ ++=cut ++ ++sub todo_skip { ++ my( $self, $why ) = @_; ++ $why ||= ''; ++ ++ lock( $self->{Curr_Test} ); ++ $self->{Curr_Test}++; ++ ++ $self->{Test_Results}[ $self->{Curr_Test} - 1 ] = &share( ++ { ++ 'ok' => 1, ++ actual_ok => 0, ++ name => '', ++ type => 'todo_skip', ++ reason => $why, ++ } ++ ); ++ ++ my $out = "not ok"; ++ $out .= " $self->{Curr_Test}" if $self->use_numbers; ++ $out .= " # TODO & SKIP $why\n"; ++ ++ $self->_print($out); ++ ++ return 1; ++} ++ ++=begin _unimplemented ++ ++=item B<skip_rest> ++ ++ $Test->skip_rest; ++ $Test->skip_rest($reason); ++ ++Like C<skip()>, only it skips all the rest of the tests you plan to run ++and terminates the test. ++ ++If you're running under C<no_plan>, it skips once and terminates the ++test. ++ ++=end _unimplemented ++ ++=back ++ ++ ++=head2 Test building utility methods ++ ++These methods are useful when writing your own test methods. ++ ++=over 4 ++ ++=item B<maybe_regex> ++ ++ $Test->maybe_regex(qr/$regex/); ++ $Test->maybe_regex('/$regex/'); ++ ++This method used to be useful back when Test::Builder worked on Perls ++before 5.6 which didn't have qr//. Now its pretty useless. ++ ++Convenience method for building testing functions that take regular ++expressions as arguments. ++ ++Takes a quoted regular expression produced by C<qr//>, or a string ++representing a regular expression. ++ ++Returns a Perl value which may be used instead of the corresponding ++regular expression, or C<undef> if its argument is not recognised. ++ ++For example, a version of C<like()>, sans the useful diagnostic messages, ++could be written as: ++ ++ sub laconic_like { ++ my ($self, $thing, $regex, $name) = @_; ++ my $usable_regex = $self->maybe_regex($regex); ++ die "expecting regex, found '$regex'\n" ++ unless $usable_regex; ++ $self->ok($thing =~ m/$usable_regex/, $name); ++ } ++ ++=cut ++ ++sub maybe_regex { ++ my( $self, $regex ) = @_; ++ my $usable_regex = undef; ++ ++ return $usable_regex unless defined $regex; ++ ++ my( $re, $opts ); ++ ++ # Check for qr/foo/ ++ if( _is_qr($regex) ) { ++ $usable_regex = $regex; ++ } ++ # Check for '/foo/' or 'm,foo,' ++ elsif(( $re, $opts ) = $regex =~ m{^ /(.*)/ (\w*) $ }sx or ++ ( undef, $re, $opts ) = $regex =~ m,^ m([^\w\s]) (.+) \1 (\w*) $,sx ++ ) ++ { ++ $usable_regex = length $opts ? "(?$opts)$re" : $re; ++ } ++ ++ return $usable_regex; ++} ++ ++sub _is_qr { ++ my $regex = shift; ++ ++ # is_regexp() checks for regexes in a robust manner, say if they're ++ # blessed. ++ return re::is_regexp($regex) if defined &re::is_regexp; ++ return ref $regex eq 'Regexp'; ++} ++ ++sub _regex_ok { ++ my( $self, $thing, $regex, $cmp, $name ) = @_; ++ ++ my $ok = 0; ++ my $usable_regex = $self->maybe_regex($regex); ++ unless( defined $usable_regex ) { ++ local $Level = $Level + 1; ++ $ok = $self->ok( 0, $name ); ++ $self->diag(" '$regex' doesn't look much like a regex to me."); ++ return $ok; ++ } ++ ++ { ++ my $test; ++ my $context = $self->_caller_context; ++ ++ { ++ ## no critic (BuiltinFunctions::ProhibitStringyEval) ++ ++ local( $@, $!, $SIG{__DIE__} ); # isolate eval ++ ++ # No point in issuing an uninit warning, they'll see it in the diagnostics ++ no warnings 'uninitialized'; ++ ++ $test = eval $context . q{$test = $thing =~ /$usable_regex/ ? 1 : 0}; ++ } ++ ++ $test = !$test if $cmp eq '!~'; ++ ++ local $Level = $Level + 1; ++ $ok = $self->ok( $test, $name ); ++ } ++ ++ unless($ok) { ++ $thing = defined $thing ? "'$thing'" : 'undef'; ++ my $match = $cmp eq '=~' ? "doesn't match" : "matches"; ++ ++ local $Level = $Level + 1; ++ $self->diag( sprintf <<'DIAGNOSTIC', $thing, $match, $regex ); ++ %s ++ %13s '%s' ++DIAGNOSTIC ++ ++ } ++ ++ return $ok; ++} ++ ++# I'm not ready to publish this. It doesn't deal with array return ++# values from the code or context. ++ ++=begin private ++ ++=item B<_try> ++ ++ my $return_from_code = $Test->try(sub { code }); ++ my($return_from_code, $error) = $Test->try(sub { code }); ++ ++Works like eval BLOCK except it ensures it has no effect on the rest ++of the test (ie. C<$@> is not set) nor is effected by outside ++interference (ie. C<$SIG{__DIE__}>) and works around some quirks in older ++Perls. ++ ++C<$error> is what would normally be in C<$@>. ++ ++It is suggested you use this in place of eval BLOCK. ++ ++=cut ++ ++sub _try { ++ my( $self, $code, %opts ) = @_; ++ ++ my $error; ++ my $return; ++ { ++ local $!; # eval can mess up $! ++ local $@; # don't set $@ in the test ++ local $SIG{__DIE__}; # don't trip an outside DIE handler. ++ $return = eval { $code->() }; ++ $error = $@; ++ } ++ ++ die $error if $error and $opts{die_on_fail}; ++ ++ return wantarray ? ( $return, $error ) : $return; ++} ++ ++=end private ++ ++ ++=item B<is_fh> ++ ++ my $is_fh = $Test->is_fh($thing); ++ ++Determines if the given C<$thing> can be used as a filehandle. ++ ++=cut ++ ++sub is_fh { ++ my $self = shift; ++ my $maybe_fh = shift; ++ return 0 unless defined $maybe_fh; ++ ++ return 1 if ref $maybe_fh eq 'GLOB'; # its a glob ref ++ return 1 if ref \$maybe_fh eq 'GLOB'; # its a glob ++ ++ return eval { $maybe_fh->isa("IO::Handle") } || ++ eval { tied($maybe_fh)->can('TIEHANDLE') }; ++} ++ ++=back ++ ++ ++=head2 Test style ++ ++ ++=over 4 ++ ++=item B<level> ++ ++ $Test->level($how_high); ++ ++How far up the call stack should C<$Test> look when reporting where the ++test failed. ++ ++Defaults to 1. ++ ++Setting L<$Test::Builder::Level> overrides. This is typically useful ++localized: ++ ++ sub my_ok { ++ my $test = shift; ++ ++ local $Test::Builder::Level = $Test::Builder::Level + 1; ++ $TB->ok($test); ++ } ++ ++To be polite to other functions wrapping your own you usually want to increment C<$Level> rather than set it to a constant. ++ ++=cut ++ ++sub level { ++ my( $self, $level ) = @_; ++ ++ if( defined $level ) { ++ $Level = $level; ++ } ++ return $Level; ++} ++ ++=item B<use_numbers> ++ ++ $Test->use_numbers($on_or_off); ++ ++Whether or not the test should output numbers. That is, this if true: ++ ++ ok 1 ++ ok 2 ++ ok 3 ++ ++or this if false ++ ++ ok ++ ok ++ ok ++ ++Most useful when you can't depend on the test output order, such as ++when threads or forking is involved. ++ ++Defaults to on. ++ ++=cut ++ ++sub use_numbers { ++ my( $self, $use_nums ) = @_; ++ ++ if( defined $use_nums ) { ++ $self->{Use_Nums} = $use_nums; ++ } ++ return $self->{Use_Nums}; ++} ++ ++=item B<no_diag> ++ ++ $Test->no_diag($no_diag); ++ ++If set true no diagnostics will be printed. This includes calls to ++C<diag()>. ++ ++=item B<no_ending> ++ ++ $Test->no_ending($no_ending); ++ ++Normally, Test::Builder does some extra diagnostics when the test ++ends. It also changes the exit code as described below. ++ ++If this is true, none of that will be done. ++ ++=item B<no_header> ++ ++ $Test->no_header($no_header); ++ ++If set to true, no "1..N" header will be printed. ++ ++=cut ++ ++foreach my $attribute (qw(No_Header No_Ending No_Diag)) { ++ my $method = lc $attribute; ++ ++ my $code = sub { ++ my( $self, $no ) = @_; ++ ++ if( defined $no ) { ++ $self->{$attribute} = $no; ++ } ++ return $self->{$attribute}; ++ }; ++ ++ no strict 'refs'; ## no critic ++ *{ __PACKAGE__ . '::' . $method } = $code; ++} ++ ++=back ++ ++=head2 Output ++ ++Controlling where the test output goes. ++ ++It's ok for your test to change where STDOUT and STDERR point to, ++Test::Builder's default output settings will not be affected. ++ ++=over 4 ++ ++=item B<diag> ++ ++ $Test->diag(@msgs); ++ ++Prints out the given C<@msgs>. Like C<print>, arguments are simply ++appended together. ++ ++Normally, it uses the C<failure_output()> handle, but if this is for a ++TODO test, the C<todo_output()> handle is used. ++ ++Output will be indented and marked with a # so as not to interfere ++with test output. A newline will be put on the end if there isn't one ++already. ++ ++We encourage using this rather than calling print directly. ++ ++Returns false. Why? Because C<diag()> is often used in conjunction with ++a failing test (C<ok() || diag()>) it "passes through" the failure. ++ ++ return ok(...) || diag(...); ++ ++=for blame transfer ++Mark Fowler <mark@xxxxxxxxxxxxxxxxxx> ++ ++=cut ++ ++sub diag { ++ my $self = shift; ++ ++ $self->_print_comment( $self->_diag_fh, @_ ); ++} ++ ++=item B<note> ++ ++ $Test->note(@msgs); ++ ++Like C<diag()>, but it prints to the C<output()> handle so it will not ++normally be seen by the user except in verbose mode. ++ ++=cut ++ ++sub note { ++ my $self = shift; ++ ++ $self->_print_comment( $self->output, @_ ); ++} ++ ++sub _diag_fh { ++ my $self = shift; ++ ++ local $Level = $Level + 1; ++ return $self->in_todo ? $self->todo_output : $self->failure_output; ++} ++ ++sub _print_comment { ++ my( $self, $fh, @msgs ) = @_; ++ ++ return if $self->no_diag; ++ return unless @msgs; ++ ++ # Prevent printing headers when compiling (i.e. -c) ++ return if $^C; ++ ++ # Smash args together like print does. ++ # Convert undef to 'undef' so its readable. ++ my $msg = join '', map { defined($_) ? $_ : 'undef' } @msgs; ++ ++ # Escape the beginning, _print will take care of the rest. ++ $msg =~ s/^/# /; ++ ++ local $Level = $Level + 1; ++ $self->_print_to_fh( $fh, $msg ); ++ ++ return 0; ++} ++ ++=item B<explain> ++ ++ my @dump = $Test->explain(@msgs); ++ ++Will dump the contents of any references in a human readable format. ++Handy for things like... ++ ++ is_deeply($have, $want) || diag explain $have; ++ ++or ++ ++ is_deeply($have, $want) || note explain $have; ++ ++=cut ++ ++sub explain { ++ my $self = shift; ++ ++ return map { ++ ref $_ ++ ? do { ++ $self->_try(sub { require Data::Dumper }, die_on_fail => 1); ++ ++ my $dumper = Data::Dumper->new( [$_] ); ++ $dumper->Indent(1)->Terse(1); ++ $dumper->Sortkeys(1) if $dumper->can("Sortkeys"); ++ $dumper->Dump; ++ } ++ : $_ ++ } @_; ++} ++ ++=begin _private ++ ++=item B<_print> ++ ++ $Test->_print(@msgs); ++ ++Prints to the C<output()> filehandle. ++ ++=end _private ++ ++=cut ++ ++sub _print { ++ my $self = shift; ++ return $self->_print_to_fh( $self->output, @_ ); ++} ++ ++sub _print_to_fh { ++ my( $self, $fh, @msgs ) = @_; ++ ++ # Prevent printing headers when only compiling. Mostly for when ++ # tests are deparsed with B::Deparse ++ return if $^C; ++ ++ my $msg = join '', @msgs; ++ my $indent = $self->_indent; ++ ++ local( $\, $", $, ) = ( undef, ' ', '' ); ++ ++ # Escape each line after the first with a # so we don't ++ # confuse Test::Harness. ++ $msg =~ s{\n(?!\z)}{\n$indent# }sg; ++ ++ # Stick a newline on the end if it needs it. ++ $msg .= "\n" unless $msg =~ /\n\z/; ++ ++ return print $fh $indent, $msg; ++} ++ ++=item B<output> ++ ++=item B<failure_output> ++ ++=item B<todo_output> ++ ++ my $filehandle = $Test->output; ++ $Test->output($filehandle); ++ $Test->output($filename); ++ $Test->output(\$scalar); ++ ++These methods control where Test::Builder will print its output. ++They take either an open C<$filehandle>, a C<$filename> to open and write to ++or a C<$scalar> reference to append to. It will always return a C<$filehandle>. ++ ++B<output> is where normal "ok/not ok" test output goes. ++ ++Defaults to STDOUT. ++ ++B<failure_output> is where diagnostic output on test failures and ++C<diag()> goes. It is normally not read by Test::Harness and instead is ++displayed to the user. ++ ++Defaults to STDERR. ++ ++C<todo_output> is used instead of C<failure_output()> for the ++diagnostics of a failing TODO test. These will not be seen by the ++user. ++ ++Defaults to STDOUT. ++ ++=cut ++ ++sub output { ++ my( $self, $fh ) = @_; ++ ++ if( defined $fh ) { ++ $self->{Out_FH} = $self->_new_fh($fh); ++ } ++ return $self->{Out_FH}; ++} ++ ++sub failure_output { ++ my( $self, $fh ) = @_; ++ ++ if( defined $fh ) { ++ $self->{Fail_FH} = $self->_new_fh($fh); ++ } ++ return $self->{Fail_FH}; ++} ++ ++sub todo_output { ++ my( $self, $fh ) = @_; ++ ++ if( defined $fh ) { ++ $self->{Todo_FH} = $self->_new_fh($fh); ++ } ++ return $self->{Todo_FH}; ++} ++ ++sub _new_fh { ++ my $self = shift; ++ my($file_or_fh) = shift; ++ ++ my $fh; ++ if( $self->is_fh($file_or_fh) ) { ++ $fh = $file_or_fh; ++ } ++ elsif( ref $file_or_fh eq 'SCALAR' ) { ++ # Scalar refs as filehandles was added in 5.8. ++ if( $] >= 5.008 ) { ++ open $fh, ">>", $file_or_fh ++ or $self->croak("Can't open scalar ref $file_or_fh: $!"); ++ } ++ # Emulate scalar ref filehandles with a tie. ++ else { ++ $fh = Test::Builder::IO::Scalar->new($file_or_fh) ++ or $self->croak("Can't tie scalar ref $file_or_fh"); ++ } ++ } ++ else { ++ open $fh, ">", $file_or_fh ++ or $self->croak("Can't open test output log $file_or_fh: $!"); ++ _autoflush($fh); ++ } ++ ++ return $fh; ++} ++ ++sub _autoflush { ++ my($fh) = shift; ++ my $old_fh = select $fh; ++ $| = 1; ++ select $old_fh; ++ ++ return; ++} ++ ++my( $Testout, $Testerr ); ++ ++sub _dup_stdhandles { ++ my $self = shift; ++ ++ $self->_open_testhandles; ++ ++ # Set everything to unbuffered else plain prints to STDOUT will ++ # come out in the wrong order from our own prints. ++ _autoflush($Testout); ++ _autoflush( \*STDOUT ); ++ _autoflush($Testerr); ++ _autoflush( \*STDERR ); ++ ++ $self->reset_outputs; ++ ++ return; ++} ++ ++sub _open_testhandles { ++ my $self = shift; ++ ++ return if $self->{Opened_Testhandles}; ++ ++ # We dup STDOUT and STDERR so people can change them in their ++ # test suites while still getting normal test output. ++ open( $Testout, ">&STDOUT" ) or die "Can't dup STDOUT: $!"; ++ open( $Testerr, ">&STDERR" ) or die "Can't dup STDERR: $!"; ++ ++ $self->_copy_io_layers( \*STDOUT, $Testout ); ++ $self->_copy_io_layers( \*STDERR, $Testerr ); ++ ++ $self->{Opened_Testhandles} = 1; ++ ++ return; ++} ++ ++sub _copy_io_layers { ++ my( $self, $src, $dst ) = @_; ++ ++ $self->_try( ++ sub { ++ require PerlIO; ++ my @src_layers = PerlIO::get_layers($src); ++ ++ _apply_layers($dst, @src_layers) if @src_layers; ++ } ++ ); ++ ++ return; ++} ++ ++sub _apply_layers { ++ my ($fh, @layers) = @_; ++ my %seen; ++ my @unique = grep { $_ ne 'unix' and !$seen{$_}++ } @layers; ++ binmode($fh, join(":", "", "raw", @unique)); ++} ++ ++ ++=item reset_outputs ++ ++ $tb->reset_outputs; ++ ++Resets all the output filehandles back to their defaults. ++ ++=cut ++ ++sub reset_outputs { ++ my $self = shift; ++ ++ $self->output ($Testout); ++ $self->failure_output($Testerr); ++ $self->todo_output ($Testout); ++ ++ return; ++} ++ ++=item carp ++ ++ $tb->carp(@message); ++ ++Warns with C<@message> but the message will appear to come from the ++point where the original test function was called (C<< $tb->caller >>). ++ ++=item croak ++ ++ $tb->croak(@message); ++ ++Dies with C<@message> but the message will appear to come from the ++point where the original test function was called (C<< $tb->caller >>). ++ ++=cut ++ ++sub _message_at_caller { ++ my $self = shift; ++ ++ local $Level = $Level + 1; ++ my( $pack, $file, $line ) = $self->caller; ++ return join( "", @_ ) . " at $file line $line.\n"; ++} ++ ++sub carp { ++ my $self = shift; ++ return warn $self->_message_at_caller(@_); ++} ++ ++sub croak { ++ my $self = shift; ++ return die $self->_message_at_caller(@_); ++} ++ ++ ++=back ++ ++ ++=head2 Test Status and Info ++ ++=over 4 ++ ++=item B<current_test> ++ ++ my $curr_test = $Test->current_test; ++ $Test->current_test($num); ++ ++Gets/sets the current test number we're on. You usually shouldn't ++have to set this. ++ ++If set forward, the details of the missing tests are filled in as 'unknown'. ++if set backward, the details of the intervening tests are deleted. You ++can erase history if you really want to. ++ ++=cut ++ ++sub current_test { ++ my( $self, $num ) = @_; ++ ++ lock( $self->{Curr_Test} ); ++ if( defined $num ) { ++ $self->{Curr_Test} = $num; ++ ++ # If the test counter is being pushed forward fill in the details. ++ my $test_results = $self->{Test_Results}; ++ if( $num > @$test_results ) { ++ my $start = @$test_results ? @$test_results : 0; ++ for( $start .. $num - 1 ) { ++ $test_results->[$_] = &share( ++ { ++ 'ok' => 1, ++ actual_ok => undef, ++ reason => 'incrementing test number', ++ type => 'unknown', ++ name => undef ++ } ++ ); ++ } ++ } ++ # If backward, wipe history. Its their funeral. ++ elsif( $num < @$test_results ) { ++ $#{$test_results} = $num - 1; ++ } ++ } ++ return $self->{Curr_Test}; ++} ++ ++=item B<is_passing> ++ ++ my $ok = $builder->is_passing; ++ ++Indicates if the test suite is currently passing. ++ ++More formally, it will be false if anything has happened which makes ++it impossible for the test suite to pass. True otherwise. ++ ++For example, if no tests have run C<is_passing()> will be true because ++even though a suite with no tests is a failure you can add a passing ++test to it and start passing. ++ ++Don't think about it too much. ++ ++=cut ++ ++sub is_passing { ++ my $self = shift; ++ ++ if( @_ ) { ++ $self->{Is_Passing} = shift; ++ } ++ ++ return $self->{Is_Passing}; ++} ++ ++ ++=item B<summary> ++ ++ my @tests = $Test->summary; ++ ++A simple summary of the tests so far. True for pass, false for fail. ++This is a logical pass/fail, so todos are passes. ++ ++Of course, test #1 is $tests[0], etc... ++ ++=cut ++ ++sub summary { ++ my($self) = shift; ++ ++ return map { $_->{'ok'} } @{ $self->{Test_Results} }; ++} ++ ++=item B<details> ++ ++ my @tests = $Test->details; ++ ++Like C<summary()>, but with a lot more detail. ++ ++ $tests[$test_num - 1] = ++ { 'ok' => is the test considered a pass? ++ actual_ok => did it literally say 'ok'? ++ name => name of the test (if any) ++ type => type of test (if any, see below). ++ reason => reason for the above (if any) ++ }; ++ ++'ok' is true if Test::Harness will consider the test to be a pass. ++ ++'actual_ok' is a reflection of whether or not the test literally ++printed 'ok' or 'not ok'. This is for examining the result of 'todo' ++tests. ++ ++'name' is the name of the test. ++ ++'type' indicates if it was a special test. Normal tests have a type ++of ''. Type can be one of the following: ++ ++ skip see skip() ++ todo see todo() ++ todo_skip see todo_skip() ++ unknown see below ++ ++Sometimes the Test::Builder test counter is incremented without it ++printing any test output, for example, when C<current_test()> is changed. ++In these cases, Test::Builder doesn't know the result of the test, so ++its type is 'unknown'. These details for these tests are filled in. ++They are considered ok, but the name and actual_ok is left C<undef>. ++ ++For example "not ok 23 - hole count # TODO insufficient donuts" would ++result in this structure: ++ ++ $tests[22] = # 23 - 1, since arrays start from 0. ++ { ok => 1, # logically, the test passed since its todo ++ actual_ok => 0, # in absolute terms, it failed ++ name => 'hole count', ++ type => 'todo', ++ reason => 'insufficient donuts' ++ }; ++ ++=cut ++ ++sub details { ++ my $self = shift; ++ return @{ $self->{Test_Results} }; ++} ++ ++=item B<todo> ++ ++ my $todo_reason = $Test->todo; ++ my $todo_reason = $Test->todo($pack); ++ ++If the current tests are considered "TODO" it will return the reason, ++if any. This reason can come from a C<$TODO> variable or the last call ++to C<todo_start()>. ++ ++Since a TODO test does not need a reason, this function can return an ++empty string even when inside a TODO block. Use C<< $Test->in_todo >> ++to determine if you are currently inside a TODO block. ++ ++C<todo()> is about finding the right package to look for C<$TODO> in. It's ++pretty good at guessing the right package to look at. It first looks for ++the caller based on C<$Level + 1>, since C<todo()> is usually called inside ++a test function. As a last resort it will use C<exported_to()>. ++ ++Sometimes there is some confusion about where todo() should be looking ++for the C<$TODO> variable. If you want to be sure, tell it explicitly ++what $pack to use. ++ ++=cut ++ ++sub todo { ++ my( $self, $pack ) = @_; ++ ++ return $self->{Todo} if defined $self->{Todo}; ++ ++ local $Level = $Level + 1; ++ my $todo = $self->find_TODO($pack); ++ return $todo if defined $todo; ++ ++ return ''; ++} ++ ++=item B<find_TODO> ++ ++ my $todo_reason = $Test->find_TODO(); ++ my $todo_reason = $Test->find_TODO($pack); ++ ++Like C<todo()> but only returns the value of C<$TODO> ignoring ++C<todo_start()>. ++ ++Can also be used to set C<$TODO> to a new value while returning the ++old value: ++ ++ my $old_reason = $Test->find_TODO($pack, 1, $new_reason); ++ ++=cut ++ ++sub find_TODO { ++ my( $self, $pack, $set, $new_value ) = @_; ++ ++ $pack = $pack || $self->caller(1) || $self->exported_to; ++ return unless $pack; ++ ++ no strict 'refs'; ## no critic ++ my $old_value = ${ $pack . '::TODO' }; ++ $set and ${ $pack . '::TODO' } = $new_value; ++ return $old_value; ++} ++ ++=item B<in_todo> ++ ++ my $in_todo = $Test->in_todo; ++ ++Returns true if the test is currently inside a TODO block. ++ ++=cut ++ ++sub in_todo { ++ my $self = shift; ++ ++ local $Level = $Level + 1; ++ return( defined $self->{Todo} || $self->find_TODO ) ? 1 : 0; ++} ++ ++=item B<todo_start> ++ ++ $Test->todo_start(); ++ $Test->todo_start($message); ++ ++This method allows you declare all subsequent tests as TODO tests, up until ++the C<todo_end> method has been called. ++ ++The C<TODO:> and C<$TODO> syntax is generally pretty good about figuring out ++whether or not we're in a TODO test. However, often we find that this is not ++possible to determine (such as when we want to use C<$TODO> but ++the tests are being executed in other packages which can't be inferred ++beforehand). ++ ++Note that you can use this to nest "todo" tests ++ ++ $Test->todo_start('working on this'); ++ # lots of code ++ $Test->todo_start('working on that'); ++ # more code ++ $Test->todo_end; ++ $Test->todo_end; ++ ++This is generally not recommended, but large testing systems often have weird ++internal needs. ++ ++We've tried to make this also work with the TODO: syntax, but it's not ++guaranteed and its use is also discouraged: ++ ++ TODO: { ++ local $TODO = 'We have work to do!'; ++ $Test->todo_start('working on this'); ++ # lots of code ++ $Test->todo_start('working on that'); ++ # more code ++ $Test->todo_end; ++ $Test->todo_end; ++ } ++ ++Pick one style or another of "TODO" to be on the safe side. ++ ++=cut ++ ++sub todo_start { ++ my $self = shift; ++ my $message = @_ ? shift : ''; ++ ++ $self->{Start_Todo}++; ++ if( $self->in_todo ) { ++ push @{ $self->{Todo_Stack} } => $self->todo; ++ } ++ $self->{Todo} = $message; ++ ++ return; ++} ++ ++=item C<todo_end> ++ ++ $Test->todo_end; ++ ++Stops running tests as "TODO" tests. This method is fatal if called without a ++preceding C<todo_start> method call. ++ ++=cut ++ ++sub todo_end { ++ my $self = shift; ++ ++ if( !$self->{Start_Todo} ) { ++ $self->croak('todo_end() called without todo_start()'); ++ } ++ ++ $self->{Start_Todo}--; ++ ++ if( $self->{Start_Todo} && @{ $self->{Todo_Stack} } ) { ++ $self->{Todo} = pop @{ $self->{Todo_Stack} }; ++ } ++ else { ++ delete $self->{Todo}; ++ } ++ ++ return; ++} ++ ++=item B<caller> ++ ++ my $package = $Test->caller; ++ my($pack, $file, $line) = $Test->caller; ++ my($pack, $file, $line) = $Test->caller($height); ++ ++Like the normal C<caller()>, except it reports according to your C<level()>. ++ ++C<$height> will be added to the C<level()>. ++ ++If C<caller()> winds up off the top of the stack it report the highest context. ++ ++=cut ++ ++sub caller { ## no critic (Subroutines::ProhibitBuiltinHomonyms) ++ my( $self, $height ) = @_; ++ $height ||= 0; ++ ++ my $level = $self->level + $height + 1; ++ my @caller; ++ do { ++ @caller = CORE::caller( $level ); ++ $level--; ++ } until @caller; ++ return wantarray ? @caller : $caller[0]; ++} ++ ++=back ++ ++=cut ++ ++=begin _private ++ ++=over 4 ++ ++=item B<_sanity_check> ++ ++ $self->_sanity_check(); ++ ++Runs a bunch of end of test sanity checks to make sure reality came ++through ok. If anything is wrong it will die with a fairly friendly ++error message. ++ ++=cut ++ ++#'# ++sub _sanity_check { ++ my $self = shift; ++ ++ $self->_whoa( $self->{Curr_Test} < 0, 'Says here you ran a negative number of tests!' ); ++ $self->_whoa( $self->{Curr_Test} != @{ $self->{Test_Results} }, ++ 'Somehow you got a different number of results than tests ran!' ); ++ ++ return; ++} ++ ++=item B<_whoa> ++ ++ $self->_whoa($check, $description); ++ ++A sanity check, similar to C<assert()>. If the C<$check> is true, something ++has gone horribly wrong. It will die with the given C<$description> and ++a note to contact the author. ++ ++=cut ++ ++sub _whoa { ++ my( $self, $check, $desc ) = @_; ++ if($check) { ++ local $Level = $Level + 1; ++ $self->croak(<<"WHOA"); ++WHOA! $desc ++This should never happen! Please contact the author immediately! ++WHOA ++ } ++ ++ return; ++} ++ ++=item B<_my_exit> ++ ++ _my_exit($exit_num); ++ ++Perl seems to have some trouble with exiting inside an C<END> block. ++5.6.1 does some odd things. Instead, this function edits C<$?> ++directly. It should B<only> be called from inside an C<END> block. ++It doesn't actually exit, that's your job. ++ ++=cut ++ ++sub _my_exit { ++ $? = $_[0]; ## no critic (Variables::RequireLocalizedPunctuationVars) ++ ++ return 1; ++} ++ ++=back ++ ++=end _private ++ ++=cut ++ ++sub _ending { ++ my $self = shift; ++ return if $self->no_ending; ++ return if $self->{Ending}++; ++ ++ my $real_exit_code = $?; ++ ++ # Don't bother with an ending if this is a forked copy. Only the parent ++ # should do the ending. ++ if( $self->{Original_Pid} != $$ ) { ++ return; ++ } ++ ++ # Ran tests but never declared a plan or hit done_testing ++ if( !$self->{Have_Plan} and $self->{Curr_Test} ) { ++ $self->is_passing(0); ++ $self->diag("Tests were run but no plan was declared and done_testing() was not seen."); ++ ++ if($real_exit_code) { ++ $self->diag(<<"FAIL"); ++Looks like your test exited with $real_exit_code just after $self->{Curr_Test}. ++FAIL ++ $self->is_passing(0); ++ _my_exit($real_exit_code) && return; ++ } ++ ++ # But if the tests ran, handle exit code. ++ my $test_results = $self->{Test_Results}; ++ if(@$test_results) { ++ my $num_failed = grep !$_->{'ok'}, @{$test_results}[ 0 .. $self->{Curr_Test} - 1 ]; ++ if ($num_failed > 0) { ++ ++ my $exit_code = $num_failed <= 254 ? $num_failed : 254; ++ _my_exit($exit_code) && return; ++ } ++ } ++ _my_exit(254) && return; ++ } ++ ++ # Exit if plan() was never called. This is so "require Test::Simple" ++ # doesn't puke. ++ if( !$self->{Have_Plan} ) { ++ return; ++ } ++ ++ # Don't do an ending if we bailed out. ++ if( $self->{Bailed_Out} ) { ++ $self->is_passing(0); ++ return; ++ } ++ # Figure out if we passed or failed and print helpful messages. ++ my $test_results = $self->{Test_Results}; ++ if(@$test_results) { ++ # The plan? We have no plan. ++ if( $self->{No_Plan} ) { ++ $self->_output_plan($self->{Curr_Test}) unless $self->no_header; ++ $self->{Expected_Tests} = $self->{Curr_Test}; ++ } ++ ++ # Auto-extended arrays and elements which aren't explicitly ++ # filled in with a shared reference will puke under 5.8.0 ++ # ithreads. So we have to fill them in by hand. :( ++ my $empty_result = &share( {} ); ++ for my $idx ( 0 .. $self->{Expected_Tests} - 1 ) { ++ $test_results->[$idx] = $empty_result ++ unless defined $test_results->[$idx]; ++ } ++ ++ my $num_failed = grep !$_->{'ok'}, @{$test_results}[ 0 .. $self->{Curr_Test} - 1 ]; ++ ++ my $num_extra = $self->{Curr_Test} - $self->{Expected_Tests}; ++ ++ if( $num_extra != 0 ) { ++ my $s = $self->{Expected_Tests} == 1 ? '' : 's'; ++ $self->diag(<<"FAIL"); ++Looks like you planned $self->{Expected_Tests} test$s but ran $self->{Curr_Test}. ++FAIL ++ $self->is_passing(0); ++ } ++ ++ if($num_failed) { ++ my $num_tests = $self->{Curr_Test}; ++ my $s = $num_failed == 1 ? '' : 's'; ++ ++ my $qualifier = $num_extra == 0 ? '' : ' run'; ++ ++ $self->diag(<<"FAIL"); ++Looks like you failed $num_failed test$s of $num_tests$qualifier. ++FAIL ++ $self->is_passing(0); ++ } ++ ++ if($real_exit_code) { ++ $self->diag(<<"FAIL"); ++Looks like your test exited with $real_exit_code just after $self->{Curr_Test}. ++FAIL ++ $self->is_passing(0); ++ _my_exit($real_exit_code) && return; ++ } ++ ++ my $exit_code; ++ if($num_failed) { ++ $exit_code = $num_failed <= 254 ? $num_failed : 254; ++ } ++ elsif( $num_extra != 0 ) { ++ $exit_code = 255; ++ } ++ else { ++ $exit_code = 0; ++ } ++ ++ _my_exit($exit_code) && return; ++ } ++ elsif( $self->{Skip_All} ) { ++ _my_exit(0) && return; ++ } ++ elsif($real_exit_code) { ++ $self->diag(<<"FAIL"); ++Looks like your test exited with $real_exit_code before it could output anything. ++FAIL ++ $self->is_passing(0); ++ _my_exit($real_exit_code) && return; ++ } ++ else { ++ $self->diag("No tests run!\n"); ++ $self->is_passing(0); ++ _my_exit(255) && return; ++ } ++ ++ $self->is_passing(0); ++ $self->_whoa( 1, "We fell off the end of _ending()" ); ++} ++ ++END { ++ $Test->_ending if defined $Test; ++} ++ ++=head1 EXIT CODES ++ ++If all your tests passed, Test::Builder will exit with zero (which is ++normal). If anything failed it will exit with how many failed. If ++you run less (or more) tests than you planned, the missing (or extras) ++will be considered failures. If no tests were ever run Test::Builder ++will throw a warning and exit with 255. If the test died, even after ++having successfully completed all its tests, it will still be ++considered a failure and will exit with 255. ++ ++So the exit codes are... ++ ++ 0 all tests successful ++ 255 test died or all passed but wrong # of tests run ++ any other number how many failed (including missing or extras) ++ ++If you fail more than 254 tests, it will be reported as 254. ++ ++=head1 THREADS ++ ++In perl 5.8.1 and later, Test::Builder is thread-safe. The test ++number is shared amongst all threads. This means if one thread sets ++the test number using C<current_test()> they will all be effected. ++ ++While versions earlier than 5.8.1 had threads they contain too many ++bugs to support. ++ ++Test::Builder is only thread-aware if threads.pm is loaded I<before> ++Test::Builder. ++ ++=head1 MEMORY ++ ++An informative hash, accessible via C<<details()>>, is stored for each ++test you perform. So memory usage will scale linearly with each test ++run. Although this is not a problem for most test suites, it can ++become an issue if you do large (hundred thousands to million) ++combinatorics tests in the same run. ++ ++In such cases, you are advised to either split the test file into smaller ++ones, or use a reverse approach, doing "normal" (code) compares and ++triggering fail() should anything go unexpected. ++ ++Future versions of Test::Builder will have a way to turn history off. ++ ++ ++=head1 EXAMPLES ++ ++CPAN can provide the best examples. Test::Simple, Test::More, ++Test::Exception and Test::Differences all use Test::Builder. ++ ++=head1 SEE ALSO ++ ++Test::Simple, Test::More, Test::Harness ++ ++=head1 AUTHORS ++ ++Original code by chromatic, maintained by Michael G Schwern ++E<lt>schwern@xxxxxxxxxx<gt> ++ ++=head1 MAINTAINERS ++ ++=over 4 ++ ++=item Chad Granum E<lt>exodist@xxxxxxxxx<gt> ++ ++=back ++ ++=head1 COPYRIGHT ++ ++Copyright 2002-2008 by chromatic E<lt>chromatic@xxxxxxxx<gt> and ++ Michael G Schwern E<lt>schwern@xxxxxxxxxx<gt>. ++ ++This program is free software; you can redistribute it and/or ++modify it under the same terms as Perl itself. ++ ++See F<http://www.perl.com/perl/misc/Artistic.html> ++ ++=cut ++ ++1; ++ +diff -Naur old/test_simple_patch/lib/Test/More.pm new/test_simple_patch/lib/Test/More.pm +--- old/test_simple_patch/lib/Test/More.pm 1970-01-01 10:00:00.000000000 +1000 ++++ new/test_simple_patch/lib/Test/More.pm 2014-03-26 21:48:11.514257656 +1100 +@@ -0,0 +1,1921 @@ ++package Test::More; ++ ++use 5.006; ++use strict; ++use warnings; ++ ++#---- perlcritic exemptions. ----# ++ ++# We use a lot of subroutine prototypes ++## no critic (Subroutines::ProhibitSubroutinePrototypes) ++ ++# Can't use Carp because it might cause use_ok() to accidentally succeed ++# even though the module being used forgot to use Carp. Yes, this ++# actually happened. ++sub _carp { ++ my( $file, $line ) = ( caller(1) )[ 1, 2 ]; ++ return warn @_, " at $file line $line\n"; ++} ++ ++our $VERSION = '1.001003'; ++$VERSION = eval $VERSION; ## no critic (BuiltinFunctions::ProhibitStringyEval) ++ ++use Test::Builder::Module 0.99; ++our @ISA = qw(Test::Builder::Module); ++our @EXPORT = qw(ok use_ok require_ok ++ is isnt like unlike is_deeply ++ cmp_ok ++ skip todo todo_skip ++ pass fail ++ eq_array eq_hash eq_set ++ $TODO ++ plan ++ done_testing ++ can_ok isa_ok new_ok ++ diag note explain ++ subtest ++ BAIL_OUT ++); ++ ++=head1 NAME ++ ++Test::More - yet another framework for writing test scripts ++ ++=head1 SYNOPSIS ++ ++ use Test::More tests => 23; ++ # or ++ use Test::More skip_all => $reason; ++ # or ++ use Test::More; # see done_testing() ++ ++ require_ok( 'Some::Module' ); ++ ++ # Various ways to say "ok" ++ ok($got eq $expected, $test_name); ++ ++ is ($got, $expected, $test_name); ++ isnt($got, $expected, $test_name); ++ ++ # Rather than print STDERR "# here's what went wrong\n" ++ diag("here's what went wrong"); ++ ++ like ($got, qr/expected/, $test_name); ++ unlike($got, qr/expected/, $test_name); ++ ++ cmp_ok($got, '==', $expected, $test_name); ++ ++ is_deeply($got_complex_structure, $expected_complex_structure, $test_name); ++ ++ SKIP: { ++ skip $why, $how_many unless $have_some_feature; ++ ++ ok( foo(), $test_name ); ++ is( foo(42), 23, $test_name ); ++ }; ++ ++ TODO: { ++ local $TODO = $why; ++ ++ ok( foo(), $test_name ); ++ is( foo(42), 23, $test_name ); ++ }; ++ ++ can_ok($module, @methods); ++ isa_ok($object, $class); ++ ++ pass($test_name); ++ fail($test_name); ++ ++ BAIL_OUT($why); ++ ++ # UNIMPLEMENTED!!! ++ my @status = Test::More::status; ++ ++ ++=head1 DESCRIPTION ++ ++B<STOP!> If you're just getting started writing tests, have a look at ++L<Test::Simple> first. This is a drop in replacement for Test::Simple ++which you can switch to once you get the hang of basic testing. ++ ++The purpose of this module is to provide a wide range of testing ++utilities. Various ways to say "ok" with better diagnostics, ++facilities to skip tests, test future features and compare complicated ++data structures. While you can do almost anything with a simple ++C<ok()> function, it doesn't provide good diagnostic output. ++ ++ ++=head2 I love it when a plan comes together ++ ++Before anything else, you need a testing plan. This basically declares ++how many tests your script is going to run to protect against premature ++failure. ++ ++The preferred way to do this is to declare a plan when you C<use Test::More>. ++ ++ use Test::More tests => 23; ++ ++There are cases when you will not know beforehand how many tests your ++script is going to run. In this case, you can declare your tests at ++the end. ++ ++ use Test::More; ++ ++ ... run your tests ... ++ ++ done_testing( $number_of_tests_run ); ++ ++Sometimes you really don't know how many tests were run, or it's too ++difficult to calculate. In which case you can leave off ++$number_of_tests_run. ++ ++In some cases, you'll want to completely skip an entire testing script. ++ ++ use Test::More skip_all => $skip_reason; ++ ++Your script will declare a skip with the reason why you skipped and ++exit immediately with a zero (success). See L<Test::Harness> for ++details. ++ ++If you want to control what functions Test::More will export, you ++have to use the 'import' option. For example, to import everything ++but 'fail', you'd do: ++ ++ use Test::More tests => 23, import => ['!fail']; ++ ++Alternatively, you can use the plan() function. Useful for when you ++have to calculate the number of tests. ++ ++ use Test::More; ++ plan tests => keys %Stuff * 3; ++ ++or for deciding between running the tests at all: ++ ++ use Test::More; ++ if( $^O eq 'MacOS' ) { ++ plan skip_all => 'Test irrelevant on MacOS'; ++ } ++ else { ++ plan tests => 42; ++ } ++ ++=cut ++ ++sub plan { ++ my $tb = Test::More->builder; ++ ++ return $tb->plan(@_); ++} ++ ++# This implements "use Test::More 'no_diag'" but the behavior is ++# deprecated. ++sub import_extra { ++ my $class = shift; ++ my $list = shift; ++ ++ my @other = (); ++ my $idx = 0; ++ while( $idx <= $#{$list} ) { ++ my $item = $list->[$idx]; ++ ++ if( defined $item and $item eq 'no_diag' ) { ++ $class->builder->no_diag(1); ++ } ++ else { ++ push @other, $item; ++ } ++ ++ $idx++; ++ } ++ ++ @$list = @other; ++ ++ return; ++} ++ ++=over 4 ++ ++=item B<done_testing> ++ ++ done_testing(); ++ done_testing($number_of_tests); ++ ++If you don't know how many tests you're going to run, you can issue ++the plan when you're done running tests. ++ ++$number_of_tests is the same as plan(), it's the number of tests you ++expected to run. You can omit this, in which case the number of tests ++you ran doesn't matter, just the fact that your tests ran to ++conclusion. ++ ++This is safer than and replaces the "no_plan" plan. ++ ++=back ++ ++=cut ++ ++sub done_testing { ++ my $tb = Test::More->builder; ++ $tb->done_testing(@_); ++} ++ ++=head2 Test names ++ ++By convention, each test is assigned a number in order. This is ++largely done automatically for you. However, it's often very useful to ++assign a name to each test. Which would you rather see: ++ ++ ok 4 ++ not ok 5 ++ ok 6 ++ ++or ++ ++ ok 4 - basic multi-variable ++ not ok 5 - simple exponential ++ ok 6 - force == mass * acceleration ++ ++The later gives you some idea of what failed. It also makes it easier ++to find the test in your script, simply search for "simple ++exponential". ++ ++All test functions take a name argument. It's optional, but highly ++suggested that you use it. ++ ++=head2 I'm ok, you're not ok. ++ ++The basic purpose of this module is to print out either "ok #" or "not ++ok #" depending on if a given test succeeded or failed. Everything ++else is just gravy. ++ ++All of the following print "ok" or "not ok" depending on if the test ++succeeded or failed. They all also return true or false, ++respectively. ++ ++=over 4 ++ ++=item B<ok> ++ ++ ok($got eq $expected, $test_name); ++ ++This simply evaluates any expression (C<$got eq $expected> is just a ++simple example) and uses that to determine if the test succeeded or ++failed. A true expression passes, a false one fails. Very simple. ++ ++For example: ++ ++ ok( $exp{9} == 81, 'simple exponential' ); ++ ok( Film->can('db_Main'), 'set_db()' ); ++ ok( $p->tests == 4, 'saw tests' ); ++ ok( !grep(!defined $_, @items), 'all items defined' ); ++ ++(Mnemonic: "This is ok.") ++ ++$test_name is a very short description of the test that will be printed ++out. It makes it very easy to find a test in your script when it fails ++and gives others an idea of your intentions. $test_name is optional, ++but we B<very> strongly encourage its use. ++ ++Should an ok() fail, it will produce some diagnostics: ++ ++ not ok 18 - sufficient mucus ++ # Failed test 'sufficient mucus' ++ # in foo.t at line 42. ++ ++This is the same as Test::Simple's ok() routine. ++ ++=cut ++ ++sub ok ($;$) { ++ my( $test, $name ) = @_; ++ my $tb = Test::More->builder; ++ ++ return $tb->ok( $test, $name ); ++} ++ ++=item B<is> ++ ++=item B<isnt> ++ ++ is ( $got, $expected, $test_name ); ++ isnt( $got, $expected, $test_name ); ++ ++Similar to ok(), is() and isnt() compare their two arguments ++with C<eq> and C<ne> respectively and use the result of that to ++determine if the test succeeded or failed. So these: ++ ++ # Is the ultimate answer 42? ++ is( ultimate_answer(), 42, "Meaning of Life" ); ++ ++ # $foo isn't empty ++ isnt( $foo, '', "Got some foo" ); ++ ++are similar to these: ++ ++ ok( ultimate_answer() eq 42, "Meaning of Life" ); ++ ok( $foo ne '', "Got some foo" ); ++ ++C<undef> will only ever match C<undef>. So you can test a value ++against C<undef> like this: ++ ++ is($not_defined, undef, "undefined as expected"); ++ ++(Mnemonic: "This is that." "This isn't that.") ++ ++So why use these? They produce better diagnostics on failure. ok() ++cannot know what you are testing for (beyond the name), but is() and ++isnt() know what the test was and why it failed. For example this ++test: ++ ++ my $foo = 'waffle'; my $bar = 'yarblokos'; ++ is( $foo, $bar, 'Is foo the same as bar?' ); ++ ++Will produce something like this: ++ ++ not ok 17 - Is foo the same as bar? ++ # Failed test 'Is foo the same as bar?' ++ # in foo.t at line 139. ++ # got: 'waffle' ++ # expected: 'yarblokos' ++ ++So you can figure out what went wrong without rerunning the test. ++ ++You are encouraged to use is() and isnt() over ok() where possible, ++however do not be tempted to use them to find out if something is ++true or false! ++ ++ # XXX BAD! ++ is( exists $brooklyn{tree}, 1, 'A tree grows in Brooklyn' ); ++ ++This does not check if C<exists $brooklyn{tree}> is true, it checks if ++it returns 1. Very different. Similar caveats exist for false and 0. ++In these cases, use ok(). ++ ++ ok( exists $brooklyn{tree}, 'A tree grows in Brooklyn' ); ++ ++A simple call to isnt() usually does not provide a strong test but there ++are cases when you cannot say much more about a value than that it is ++different from some other value: ++ ++ new_ok $obj, "Foo"; ++ ++ my $clone = $obj->clone; ++ isa_ok $obj, "Foo", "Foo->clone"; ++ ++ isnt $obj, $clone, "clone() produces a different object"; ++ ++For those grammatical pedants out there, there's an C<isn't()> ++function which is an alias of isnt(). ++ ++=cut ++ ++sub is ($$;$) { ++ my $tb = Test::More->builder; ++ ++ return $tb->is_eq(@_); ++} ++ ++sub isnt ($$;$) { ++ my $tb = Test::More->builder; ++ ++ return $tb->isnt_eq(@_); ++} ++ ++*isn't = \&isnt; ++ ++=item B<like> ++ ++ like( $got, qr/expected/, $test_name ); ++ ++Similar to ok(), like() matches $got against the regex C<qr/expected/>. ++ ++So this: ++ ++ like($got, qr/expected/, 'this is like that'); ++ ++is similar to: ++ ++ ok( $got =~ m/expected/, 'this is like that'); ++ ++(Mnemonic "This is like that".) ++ ++The second argument is a regular expression. It may be given as a ++regex reference (i.e. C<qr//>) or (for better compatibility with older ++perls) as a string that looks like a regex (alternative delimiters are ++currently not supported): ++ ++ like( $got, '/expected/', 'this is like that' ); ++ ++Regex options may be placed on the end (C<'/expected/i'>). ++ ++Its advantages over ok() are similar to that of is() and isnt(). Better ++diagnostics on failure. ++ ++=cut ++ ++sub like ($$;$) { ++ my $tb = Test::More->builder; ++ ++ return $tb->like(@_); ++} ++ ++=item B<unlike> ++ ++ unlike( $got, qr/expected/, $test_name ); ++ ++Works exactly as like(), only it checks if $got B<does not> match the ++given pattern. ++ ++=cut ++ ++sub unlike ($$;$) { ++ my $tb = Test::More->builder; ++ ++ return $tb->unlike(@_); ++} ++ ++=item B<cmp_ok> ++ ++ cmp_ok( $got, $op, $expected, $test_name ); ++ ++Halfway between C<ok()> and C<is()> lies C<cmp_ok()>. This allows you ++to compare two arguments using any binary perl operator. The test ++passes if the comparison is true and fails otherwise. ++ ++ # ok( $got eq $expected ); ++ cmp_ok( $got, 'eq', $expected, 'this eq that' ); ++ ++ # ok( $got == $expected ); ++ cmp_ok( $got, '==', $expected, 'this == that' ); ++ ++ # ok( $got && $expected ); ++ cmp_ok( $got, '&&', $expected, 'this && that' ); ++ ...etc... ++ ++Its advantage over ok() is when the test fails you'll know what $got ++and $expected were: ++ ++ not ok 1 ++ # Failed test in foo.t at line 12. ++ # '23' ++ # && ++ # undef ++ ++It's also useful in those cases where you are comparing numbers and ++is()'s use of C<eq> will interfere: ++ ++ cmp_ok( $big_hairy_number, '==', $another_big_hairy_number ); ++ ++It's especially useful when comparing greater-than or smaller-than ++relation between values: ++ ++ cmp_ok( $some_value, '<=', $upper_limit ); ++ ++ ++=cut ++ ++sub cmp_ok($$$;$) { ++ my $tb = Test::More->builder; ++ ++ return $tb->cmp_ok(@_); ++} ++ ++=item B<can_ok> ++ ++ can_ok($module, @methods); ++ can_ok($object, @methods); ++ ++Checks to make sure the $module or $object can do these @methods ++(works with functions, too). ++ ++ can_ok('Foo', qw(this that whatever)); ++ ++is almost exactly like saying: ++ ++ ok( Foo->can('this') && ++ Foo->can('that') && ++ Foo->can('whatever') ++ ); ++ ++only without all the typing and with a better interface. Handy for ++quickly testing an interface. ++ ++No matter how many @methods you check, a single can_ok() call counts ++as one test. If you desire otherwise, use: ++ ++ foreach my $meth (@methods) { ++ can_ok('Foo', $meth); ++ } ++ ++=cut ++ ++sub can_ok ($@) { ++ my( $proto, @methods ) = @_; ++ my $class = ref $proto || $proto; ++ my $tb = Test::More->builder; ++ ++ unless($class) { ++ my $ok = $tb->ok( 0, "->can(...)" ); ++ $tb->diag(' can_ok() called with empty class or reference'); ++ return $ok; ++ } ++ ++ unless(@methods) { ++ my $ok = $tb->ok( 0, "$class->can(...)" ); ++ $tb->diag(' can_ok() called with no methods'); ++ return $ok; ++ } ++ ++ my @nok = (); ++ foreach my $method (@methods) { ++ $tb->_try( sub { $proto->can($method) } ) or push @nok, $method; ++ } ++ ++ my $name = (@methods == 1) ? "$class->can('$methods[0]')" : ++ "$class->can(...)" ; ++ ++ my $ok = $tb->ok( !@nok, $name ); ++ ++ $tb->diag( map " $class->can('$_') failed\n", @nok ); ++ ++ return $ok; ++} ++ ++=item B<isa_ok> ++ ++ isa_ok($object, $class, $object_name); ++ isa_ok($subclass, $class, $object_name); ++ isa_ok($ref, $type, $ref_name); ++ ++Checks to see if the given C<< $object->isa($class) >>. Also checks to make ++sure the object was defined in the first place. Handy for this sort ++of thing: ++ ++ my $obj = Some::Module->new; ++ isa_ok( $obj, 'Some::Module' ); ++ ++where you'd otherwise have to write ++ ++ my $obj = Some::Module->new; ++ ok( defined $obj && $obj->isa('Some::Module') ); ++ ++to safeguard against your test script blowing up. ++ ++You can also test a class, to make sure that it has the right ancestor: ++ ++ isa_ok( 'Vole', 'Rodent' ); ++ ++It works on references, too: ++ ++ isa_ok( $array_ref, 'ARRAY' ); ++ ++The diagnostics of this test normally just refer to 'the object'. If ++you'd like them to be more specific, you can supply an $object_name ++(for example 'Test customer'). ++ ++=cut ++ ++sub isa_ok ($$;$) { ++ my( $thing, $class, $thing_name ) = @_; ++ my $tb = Test::More->builder; ++ ++ my $whatami; ++ if( !defined $thing ) { ++ $whatami = 'undef'; ++ } ++ elsif( ref $thing ) { ++ $whatami = 'reference'; ++ ++ local($@,$!); ++ require Scalar::Util; ++ if( Scalar::Util::blessed($thing) ) { ++ $whatami = 'object'; ++ } ++ } ++ else { ++ $whatami = 'class'; ++ } ++ ++ # We can't use UNIVERSAL::isa because we want to honor isa() overrides ++ my( $rslt, $error ) = $tb->_try( sub { $thing->isa($class) } ); ++ ++ if($error) { ++ die <<WHOA unless $error =~ /^Can't (locate|call) method "isa"/; ++WHOA! I tried to call ->isa on your $whatami and got some weird error. ++Here's the error. ++$error ++WHOA ++ } ++ ++ # Special case for isa_ok( [], "ARRAY" ) and like ++ if( $whatami eq 'reference' ) { ++ $rslt = UNIVERSAL::isa($thing, $class); ++ } ++ ++ my($diag, $name); ++ if( defined $thing_name ) { ++ $name = "'$thing_name' isa '$class'"; ++ $diag = defined $thing ? "'$thing_name' isn't a '$class'" : "'$thing_name' isn't defined"; ++ } ++ elsif( $whatami eq 'object' ) { ++ my $my_class = ref $thing; ++ $thing_name = qq[An object of class '$my_class']; ++ $name = "$thing_name isa '$class'"; ++ $diag = "The object of class '$my_class' isn't a '$class'"; ++ } ++ elsif( $whatami eq 'reference' ) { ++ my $type = ref $thing; ++ $thing_name = qq[A reference of type '$type']; ++ $name = "$thing_name isa '$class'"; ++ $diag = "The reference of type '$type' isn't a '$class'"; ++ } ++ elsif( $whatami eq 'undef' ) { ++ $thing_name = 'undef'; ++ $name = "$thing_name isa '$class'"; ++ $diag = "$thing_name isn't defined"; ++ } ++ elsif( $whatami eq 'class' ) { ++ $thing_name = qq[The class (or class-like) '$thing']; ++ $name = "$thing_name isa '$class'"; ++ $diag = "$thing_name isn't a '$class'"; ++ } ++ else { ++ die; ++ } ++ ++ my $ok; ++ if($rslt) { ++ $ok = $tb->ok( 1, $name ); ++ } ++ else { ++ $ok = $tb->ok( 0, $name ); ++ $tb->diag(" $diag\n"); ++ } ++ ++ return $ok; ++} ++ ++=item B<new_ok> ++ ++ my $obj = new_ok( $class ); ++ my $obj = new_ok( $class => \@args ); ++ my $obj = new_ok( $class => \@args, $object_name ); ++ ++A convenience function which combines creating an object and calling ++isa_ok() on that object. ++ ++It is basically equivalent to: ++ ++ my $obj = $class->new(@args); ++ isa_ok $obj, $class, $object_name; ++ ++If @args is not given, an empty list will be used. ++ ++This function only works on new() and it assumes new() will return ++just a single object which isa C<$class>. ++ ++=cut ++ ++sub new_ok { ++ my $tb = Test::More->builder; ++ $tb->croak("new_ok() must be given at least a class") unless @_; ++ ++ my( $class, $args, $object_name ) = @_; ++ ++ $args ||= []; ++ ++ my $obj; ++ my( $success, $error ) = $tb->_try( sub { $obj = $class->new(@$args); 1 } ); ++ if($success) { ++ local $Test::Builder::Level = $Test::Builder::Level + 1; ++ isa_ok $obj, $class, $object_name; ++ } ++ else { ++ $class = 'undef' if !defined $class; ++ $tb->ok( 0, "$class->new() died" ); ++ $tb->diag(" Error was: $error"); ++ } ++ ++ return $obj; ++} ++ ++=item B<subtest> ++ ++ subtest $name => \&code; ++ ++subtest() runs the &code as its own little test with its own plan and ++its own result. The main test counts this as a single test using the ++result of the whole subtest to determine if its ok or not ok. ++ ++For example... ++ ++ use Test::More tests => 3; ++ ++ pass("First test"); ++ ++ subtest 'An example subtest' => sub { ++ plan tests => 2; ++ ++ pass("This is a subtest"); ++ pass("So is this"); ++ }; ++ ++ pass("Third test"); ++ ++This would produce. ++ ++ 1..3 ++ ok 1 - First test ++ # Subtest: An example subtest ++ 1..2 ++ ok 1 - This is a subtest ++ ok 2 - So is this ++ ok 2 - An example subtest ++ ok 3 - Third test ++ ++A subtest may call "skip_all". No tests will be run, but the subtest is ++considered a skip. ++ ++ subtest 'skippy' => sub { ++ plan skip_all => 'cuz I said so'; ++ pass('this test will never be run'); ++ }; ++ ++Returns true if the subtest passed, false otherwise. ++ ++Due to how subtests work, you may omit a plan if you desire. This adds an ++implicit C<done_testing()> to the end of your subtest. The following two ++subtests are equivalent: ++ ++ subtest 'subtest with implicit done_testing()', sub { ++ ok 1, 'subtests with an implicit done testing should work'; ++ ok 1, '... and support more than one test'; ++ ok 1, '... no matter how many tests are run'; ++ }; ++ ++ subtest 'subtest with explicit done_testing()', sub { ++ ok 1, 'subtests with an explicit done testing should work'; ++ ok 1, '... and support more than one test'; ++ ok 1, '... no matter how many tests are run'; ++ done_testing(); ++ }; ++ ++=cut ++ ++sub subtest { ++ my ($name, $subtests) = @_; ++ ++ my $tb = Test::More->builder; ++ return $tb->subtest(@_); ++} ++ ++=item B<pass> ++ ++=item B<fail> ++ ++ pass($test_name); ++ fail($test_name); ++ ++Sometimes you just want to say that the tests have passed. Usually ++the case is you've got some complicated condition that is difficult to ++wedge into an ok(). In this case, you can simply use pass() (to ++declare the test ok) or fail (for not ok). They are synonyms for ++ok(1) and ok(0). ++ ++Use these very, very, very sparingly. ++ ++=cut ++ ++sub pass (;$) { ++ my $tb = Test::More->builder; ++ ++ return $tb->ok( 1, @_ ); ++} ++ ++sub fail (;$) { ++ my $tb = Test::More->builder; ++ ++ return $tb->ok( 0, @_ ); ++} ++ ++=back ++ ++ ++=head2 Module tests ++ ++Sometimes you want to test if a module, or a list of modules, can ++successfully load. For example, you'll often want a first test which ++simply loads all the modules in the distribution to make sure they ++work before going on to do more complicated testing. ++ ++For such purposes we have C<use_ok> and C<require_ok>. ++ ++=over 4 ++ ++=item B<require_ok> ++ ++ require_ok($module); ++ require_ok($file); ++ ++Tries to C<require> the given $module or $file. If it loads ++successfully, the test will pass. Otherwise it fails and displays the ++load error. ++ ++C<require_ok> will guess whether the input is a module name or a ++filename. ++ ++No exception will be thrown if the load fails. ++ ++ # require Some::Module ++ require_ok "Some::Module"; ++ ++ # require "Some/File.pl"; ++ require_ok "Some/File.pl"; ++ ++ # stop testing if any of your modules will not load ++ for my $module (@module) { ++ require_ok $module or BAIL_OUT "Can't load $module"; ++ } ++ ++=cut ++ ++sub require_ok ($) { ++ my($module) = shift; ++ my $tb = Test::More->builder; ++ ++ my $pack = caller; ++ ++ # Try to determine if we've been given a module name or file. ++ # Module names must be barewords, files not. ++ $module = qq['$module'] unless _is_module_name($module); ++ ++ my $code = <<REQUIRE; ++package $pack; ++require $module; ++1; ++REQUIRE ++ ++ my( $eval_result, $eval_error ) = _eval($code); ++ my $ok = $tb->ok( $eval_result, "require $module;" ); ++ ++ unless($ok) { ++ chomp $eval_error; ++ $tb->diag(<<DIAGNOSTIC); ++ Tried to require '$module'. ++ Error: $eval_error ++DIAGNOSTIC ++ ++ } ++ ++ return $ok; ++} ++ ++sub _is_module_name { ++ my $module = shift; ++ ++ # Module names start with a letter. ++ # End with an alphanumeric. ++ # The rest is an alphanumeric or :: ++ $module =~ s/\b::\b//g; ++ ++ return $module =~ /^[a-zA-Z]\w*$/ ? 1 : 0; ++} ++ ++ ++=item B<use_ok> ++ ++ BEGIN { use_ok($module); } ++ BEGIN { use_ok($module, @imports); } ++ ++Like C<require_ok>, but it will C<use> the $module in question and ++only loads modules, not files. ++ ++If you just want to test a module can be loaded, use C<require_ok>. ++ ++If you just want to load a module in a test, we recommend simply using ++C<use> directly. It will cause the test to stop. ++ ++It's recommended that you run use_ok() inside a BEGIN block so its ++functions are exported at compile-time and prototypes are properly ++honored. ++ ++If @imports are given, they are passed through to the use. So this: ++ ++ BEGIN { use_ok('Some::Module', qw(foo bar)) } ++ ++is like doing this: ++ ++ use Some::Module qw(foo bar); ++ ++Version numbers can be checked like so: ++ ++ # Just like "use Some::Module 1.02" ++ BEGIN { use_ok('Some::Module', 1.02) } ++ ++Don't try to do this: ++ ++ BEGIN { ++ use_ok('Some::Module'); ++ ++ ...some code that depends on the use... ++ ...happening at compile time... ++ } ++ ++because the notion of "compile-time" is relative. Instead, you want: ++ ++ BEGIN { use_ok('Some::Module') } ++ BEGIN { ...some code that depends on the use... } ++ ++If you want the equivalent of C<use Foo ()>, use a module but not ++import anything, use C<require_ok>. ++ ++ BEGIN { require_ok "Foo" } ++ ++=cut ++ ++sub use_ok ($;@) { ++ my( $module, @imports ) = @_; ++ @imports = () unless @imports; ++ my $tb = Test::More->builder; ++ ++ my( $pack, $filename, $line ) = caller; ++ $filename =~ y/\n\r/_/; # so it doesn't run off the "#line $line $f" line ++ ++ my $code; ++ if( @imports == 1 and $imports[0] =~ /^\d+(?:\.\d+)?$/ ) { ++ # probably a version check. Perl needs to see the bare number ++ # for it to work with non-Exporter based modules. ++ $code = <<USE; ++package $pack; ++ ++#line $line $filename ++use $module $imports[0]; ++1; ++USE ++ } ++ else { ++ $code = <<USE; ++package $pack; ++ ++#line $line $filename ++use $module \@{\$args[0]}; ++1; ++USE ++ } ++ ++ my( $eval_result, $eval_error ) = _eval( $code, \@imports ); ++ my $ok = $tb->ok( $eval_result, "use $module;" ); ++ ++ unless($ok) { ++ chomp $eval_error; ++ $@ =~ s{^BEGIN failed--compilation aborted at .*$} ++ {BEGIN failed--compilation aborted at $filename line $line.}m; ++ $tb->diag(<<DIAGNOSTIC); ++ Tried to use '$module'. ++ Error: $eval_error ++DIAGNOSTIC ++ ++ } ++ ++ return $ok; ++} ++ ++sub _eval { ++ my( $code, @args ) = @_; ++ ++ # Work around oddities surrounding resetting of $@ by immediately ++ # storing it. ++ my( $sigdie, $eval_result, $eval_error ); ++ { ++ local( $@, $!, $SIG{__DIE__} ); # isolate eval ++ $eval_result = eval $code; ## no critic (BuiltinFunctions::ProhibitStringyEval) ++ $eval_error = $@; ++ $sigdie = $SIG{__DIE__} || undef; ++ } ++ # make sure that $code got a chance to set $SIG{__DIE__} ++ $SIG{__DIE__} = $sigdie if defined $sigdie; ++ ++ return( $eval_result, $eval_error ); ++} ++ ++ ++=back ++ ++ ++=head2 Complex data structures ++ ++Not everything is a simple eq check or regex. There are times you ++need to see if two data structures are equivalent. For these ++instances Test::More provides a handful of useful functions. ++ ++B<NOTE> I'm not quite sure what will happen with filehandles. ++ ++=over 4 ++ ++=item B<is_deeply> ++ ++ is_deeply( $got, $expected, $test_name ); ++ ++Similar to is(), except that if $got and $expected are references, it ++does a deep comparison walking each data structure to see if they are ++equivalent. If the two structures are different, it will display the ++place where they start differing. ++ ++is_deeply() compares the dereferenced values of references, the ++references themselves (except for their type) are ignored. This means ++aspects such as blessing and ties are not considered "different". ++ ++is_deeply() currently has very limited handling of function reference ++and globs. It merely checks if they have the same referent. This may ++improve in the future. ++ ++L<Test::Differences> and L<Test::Deep> provide more in-depth functionality ++along these lines. ++ ++=cut ++ ++our( @Data_Stack, %Refs_Seen ); ++my $DNE = bless [], 'Does::Not::Exist'; ++ ++sub _dne { ++ return ref $_[0] eq ref $DNE; ++} ++ ++## no critic (Subroutines::RequireArgUnpacking) ++sub is_deeply { ++ my $tb = Test::More->builder; ++ ++ unless( @_ == 2 or @_ == 3 ) { ++ my $msg = <<'WARNING'; ++is_deeply() takes two or three args, you gave %d. ++This usually means you passed an array or hash instead ++of a reference to it ++WARNING ++ chop $msg; # clip off newline so carp() will put in line/file ++ ++ _carp sprintf $msg, scalar @_; ++ ++ return $tb->ok(0); ++ } ++ ++ my( $got, $expected, $name ) = @_; ++ ++ $tb->_unoverload_str( \$expected, \$got ); ++ ++ my $ok; ++ if( !ref $got and !ref $expected ) { # neither is a reference ++ $ok = $tb->is_eq( $got, $expected, $name ); ++ } ++ elsif( !ref $got xor !ref $expected ) { # one's a reference, one isn't ++ $ok = $tb->ok( 0, $name ); ++ $tb->diag( _format_stack({ vals => [ $got, $expected ] }) ); ++ } ++ else { # both references ++ local @Data_Stack = (); ++ if( _deep_check( $got, $expected ) ) { ++ $ok = $tb->ok( 1, $name ); ++ } ++ else { ++ $ok = $tb->ok( 0, $name ); ++ $tb->diag( _format_stack(@Data_Stack) ); ++ } ++ } ++ ++ return $ok; ++} ++ ++sub _format_stack { ++ my(@Stack) = @_; ++ ++ my $var = '$FOO'; ++ my $did_arrow = 0; ++ foreach my $entry (@Stack) { ++ my $type = $entry->{type} || ''; ++ my $idx = $entry->{'idx'}; ++ if( $type eq 'HASH' ) { ++ $var .= "->" unless $did_arrow++; ++ $var .= "{$idx}"; ++ } ++ elsif( $type eq 'ARRAY' ) { ++ $var .= "->" unless $did_arrow++; ++ $var .= "[$idx]"; ++ } ++ elsif( $type eq 'REF' ) { ++ $var = "\${$var}"; ++ } ++ } ++ ++ my @vals = @{ $Stack[-1]{vals} }[ 0, 1 ]; ++ my @vars = (); ++ ( $vars[0] = $var ) =~ s/\$FOO/ \$got/; ++ ( $vars[1] = $var ) =~ s/\$FOO/\$expected/; ++ ++ my $out = "Structures begin differing at:\n"; ++ foreach my $idx ( 0 .. $#vals ) { ++ my $val = $vals[$idx]; ++ $vals[$idx] ++ = !defined $val ? 'undef' ++ : _dne($val) ? "Does not exist" ++ : ref $val ? "$val" ++ : "'$val'"; ++ } ++ ++ $out .= "$vars[0] = $vals[0]\n"; ++ $out .= "$vars[1] = $vals[1]\n"; ++ ++ $out =~ s/^/ /msg; ++ return $out; ++} ++ ++sub _type { ++ my $thing = shift; ++ ++ return '' if !ref $thing; ++ ++ for my $type (qw(Regexp ARRAY HASH REF SCALAR GLOB CODE)) { ++ return $type if UNIVERSAL::isa( $thing, $type ); ++ } ++ ++ return ''; ++} ++ ++=back ++ ++ ++=head2 Diagnostics ++ ++If you pick the right test function, you'll usually get a good idea of ++what went wrong when it failed. But sometimes it doesn't work out ++that way. So here we have ways for you to write your own diagnostic ++messages which are safer than just C<print STDERR>. ++ ++=over 4 ++ ++=item B<diag> ++ ++ diag(@diagnostic_message); ++ ++Prints a diagnostic message which is guaranteed not to interfere with ++test output. Like C<print> @diagnostic_message is simply concatenated ++together. ++ ++Returns false, so as to preserve failure. ++ ++Handy for this sort of thing: ++ ++ ok( grep(/foo/, @users), "There's a foo user" ) or ++ diag("Since there's no foo, check that /etc/bar is set up right"); ++ ++which would produce: ++ ++ not ok 42 - There's a foo user ++ # Failed test 'There's a foo user' ++ # in foo.t at line 52. ++ # Since there's no foo, check that /etc/bar is set up right. ++ ++You might remember C<ok() or diag()> with the mnemonic C<open() or ++die()>. ++ ++B<NOTE> The exact formatting of the diagnostic output is still ++changing, but it is guaranteed that whatever you throw at it won't ++interfere with the test. ++ ++=item B<note> ++ ++ note(@diagnostic_message); ++ ++Like diag(), except the message will not be seen when the test is run ++in a harness. It will only be visible in the verbose TAP stream. ++ ++Handy for putting in notes which might be useful for debugging, but ++don't indicate a problem. ++ ++ note("Tempfile is $tempfile"); ++ ++=cut ++ ++sub diag { ++ return Test::More->builder->diag(@_); ++} ++ ++sub note { ++ return Test::More->builder->note(@_); ++} ++ ++=item B<explain> ++ ++ my @dump = explain @diagnostic_message; ++ ++Will dump the contents of any references in a human readable format. ++Usually you want to pass this into C<note> or C<diag>. ++ ++Handy for things like... ++ ++ is_deeply($have, $want) || diag explain $have; ++ ++or ++ ++ note explain \%args; ++ Some::Class->method(%args); ++ ++=cut ++ ++sub explain { ++ return Test::More->builder->explain(@_); ++} ++ ++=back ++ ++ ++=head2 Conditional tests ++ ++Sometimes running a test under certain conditions will cause the ++test script to die. A certain function or method isn't implemented ++(such as fork() on MacOS), some resource isn't available (like a ++net connection) or a module isn't available. In these cases it's ++necessary to skip tests, or declare that they are supposed to fail ++but will work in the future (a todo test). ++ ++For more details on the mechanics of skip and todo tests see ++L<Test::Harness>. ++ ++The way Test::More handles this is with a named block. Basically, a ++block of tests which can be skipped over or made todo. It's best if I ++just show you... ++ ++=over 4 ++ ++=item B<SKIP: BLOCK> ++ ++ SKIP: { ++ skip $why, $how_many if $condition; ++ ++ ...normal testing code goes here... ++ } ++ ++This declares a block of tests that might be skipped, $how_many tests ++there are, $why and under what $condition to skip them. An example is ++the easiest way to illustrate: ++ ++ SKIP: { ++ eval { require HTML::Lint }; ++ ++ skip "HTML::Lint not installed", 2 if $@; ++ ++ my $lint = new HTML::Lint; ++ isa_ok( $lint, "HTML::Lint" ); ++ ++ $lint->parse( $html ); ++ is( $lint->errors, 0, "No errors found in HTML" ); ++ } ++ ++If the user does not have HTML::Lint installed, the whole block of ++code I<won't be run at all>. Test::More will output special ok's ++which Test::Harness interprets as skipped, but passing, tests. ++ ++It's important that $how_many accurately reflects the number of tests ++in the SKIP block so the # of tests run will match up with your plan. ++If your plan is C<no_plan> $how_many is optional and will default to 1. ++ ++It's perfectly safe to nest SKIP blocks. Each SKIP block must have ++the label C<SKIP>, or Test::More can't work its magic. ++ ++You don't skip tests which are failing because there's a bug in your ++program, or for which you don't yet have code written. For that you ++use TODO. Read on. ++ ++=cut ++ ++## no critic (Subroutines::RequireFinalReturn) ++sub skip { ++ my( $why, $how_many ) = @_; ++ my $tb = Test::More->builder; ++ ++ unless( defined $how_many ) { ++ # $how_many can only be avoided when no_plan is in use. ++ _carp "skip() needs to know \$how_many tests are in the block" ++ unless $tb->has_plan eq 'no_plan'; ++ $how_many = 1; ++ } ++ ++ if( defined $how_many and $how_many =~ /\D/ ) { ++ _carp ++ "skip() was passed a non-numeric number of tests. Did you get the arguments backwards?"; ++ $how_many = 1; ++ } ++ ++ for( 1 .. $how_many ) { ++ $tb->skip($why); ++ } ++ ++ no warnings 'exiting'; ++ last SKIP; ++} ++ ++=item B<TODO: BLOCK> ++ ++ TODO: { ++ local $TODO = $why if $condition; ++ ++ ...normal testing code goes here... ++ } ++ ++Declares a block of tests you expect to fail and $why. Perhaps it's ++because you haven't fixed a bug or haven't finished a new feature: ++ ++ TODO: { ++ local $TODO = "URI::Geller not finished"; ++ ++ my $card = "Eight of clubs"; ++ is( URI::Geller->your_card, $card, 'Is THIS your card?' ); ++ ++ my $spoon; ++ URI::Geller->bend_spoon; ++ is( $spoon, 'bent', "Spoon bending, that's original" ); ++ } ++ ++With a todo block, the tests inside are expected to fail. Test::More ++will run the tests normally, but print out special flags indicating ++they are "todo". Test::Harness will interpret failures as being ok. ++Should anything succeed, it will report it as an unexpected success. ++You then know the thing you had todo is done and can remove the ++TODO flag. ++ ++The nice part about todo tests, as opposed to simply commenting out a ++block of tests, is it's like having a programmatic todo list. You know ++how much work is left to be done, you're aware of what bugs there are, ++and you'll know immediately when they're fixed. ++ ++Once a todo test starts succeeding, simply move it outside the block. ++When the block is empty, delete it. ++ ++ ++=item B<todo_skip> ++ ++ TODO: { ++ todo_skip $why, $how_many if $condition; ++ ++ ...normal testing code... ++ } ++ ++With todo tests, it's best to have the tests actually run. That way ++you'll know when they start passing. Sometimes this isn't possible. ++Often a failing test will cause the whole program to die or hang, even ++inside an C<eval BLOCK> with and using C<alarm>. In these extreme ++cases you have no choice but to skip over the broken tests entirely. ++ ++The syntax and behavior is similar to a C<SKIP: BLOCK> except the ++tests will be marked as failing but todo. Test::Harness will ++interpret them as passing. ++ ++=cut ++ ++sub todo_skip { ++ my( $why, $how_many ) = @_; ++ my $tb = Test::More->builder; ++ ++ unless( defined $how_many ) { ++ # $how_many can only be avoided when no_plan is in use. ++ _carp "todo_skip() needs to know \$how_many tests are in the block" ++ unless $tb->has_plan eq 'no_plan'; ++ $how_many = 1; ++ } ++ ++ for( 1 .. $how_many ) { ++ $tb->todo_skip($why); ++ } ++ ++ no warnings 'exiting'; ++ last TODO; ++} ++ ++=item When do I use SKIP vs. TODO? ++ ++B<If it's something the user might not be able to do>, use SKIP. ++This includes optional modules that aren't installed, running under ++an OS that doesn't have some feature (like fork() or symlinks), or maybe ++you need an Internet connection and one isn't available. ++ ++B<If it's something the programmer hasn't done yet>, use TODO. This ++is for any code you haven't written yet, or bugs you have yet to fix, ++but want to put tests in your testing script (always a good idea). ++ ++ ++=back ++ ++ ++=head2 Test control ++ ++=over 4 ++ ++=item B<BAIL_OUT> ++ ++ BAIL_OUT($reason); ++ ++Indicates to the harness that things are going so badly all testing ++should terminate. This includes the running of any additional test scripts. ++ ++This is typically used when testing cannot continue such as a critical ++module failing to compile or a necessary external utility not being ++available such as a database connection failing. ++ ++The test will exit with 255. ++ ++For even better control look at L<Test::Most>. ++ ++=cut ++ ++sub BAIL_OUT { ++ my $reason = shift; ++ my $tb = Test::More->builder; ++ ++ $tb->BAIL_OUT($reason); ++} ++ ++=back ++ ++ ++=head2 Discouraged comparison functions ++ ++The use of the following functions is discouraged as they are not ++actually testing functions and produce no diagnostics to help figure ++out what went wrong. They were written before is_deeply() existed ++because I couldn't figure out how to display a useful diff of two ++arbitrary data structures. ++ ++These functions are usually used inside an ok(). ++ ++ ok( eq_array(\@got, \@expected) ); ++ ++C<is_deeply()> can do that better and with diagnostics. ++ ++ is_deeply( \@got, \@expected ); ++ ++They may be deprecated in future versions. ++ ++=over 4 ++ ++=item B<eq_array> ++ ++ my $is_eq = eq_array(\@got, \@expected); ++ ++Checks if two arrays are equivalent. This is a deep check, so ++multi-level structures are handled correctly. ++ ++=cut ++ ++#'# ++sub eq_array { ++ local @Data_Stack = (); ++ _deep_check(@_); ++} ++ ++sub _eq_array { ++ my( $a1, $a2 ) = @_; ++ ++ if( grep _type($_) ne 'ARRAY', $a1, $a2 ) { ++ warn "eq_array passed a non-array ref"; ++ return 0; ++ } ++ ++ return 1 if $a1 eq $a2; ++ ++ my $ok = 1; ++ my $max = $#$a1 > $#$a2 ? $#$a1 : $#$a2; ++ for( 0 .. $max ) { ++ my $e1 = $_ > $#$a1 ? $DNE : $a1->[$_]; ++ my $e2 = $_ > $#$a2 ? $DNE : $a2->[$_]; ++ ++ next if _equal_nonrefs($e1, $e2); ++ ++ push @Data_Stack, { type => 'ARRAY', idx => $_, vals => [ $e1, $e2 ] }; ++ $ok = _deep_check( $e1, $e2 ); ++ pop @Data_Stack if $ok; ++ ++ last unless $ok; ++ } ++ ++ return $ok; ++} ++ ++sub _equal_nonrefs { ++ my( $e1, $e2 ) = @_; ++ ++ return if ref $e1 or ref $e2; ++ ++ if ( defined $e1 ) { ++ return 1 if defined $e2 and $e1 eq $e2; ++ } ++ else { ++ return 1 if !defined $e2; ++ } ++ ++ return; ++} ++ ++sub _deep_check { ++ my( $e1, $e2 ) = @_; ++ my $tb = Test::More->builder; ++ ++ my $ok = 0; ++ ++ # Effectively turn %Refs_Seen into a stack. This avoids picking up ++ # the same referenced used twice (such as [\$a, \$a]) to be considered ++ # circular. ++ local %Refs_Seen = %Refs_Seen; ++ ++ { ++ $tb->_unoverload_str( \$e1, \$e2 ); ++ ++ # Either they're both references or both not. ++ my $same_ref = !( !ref $e1 xor !ref $e2 ); ++ my $not_ref = ( !ref $e1 and !ref $e2 ); ++ ++ if( defined $e1 xor defined $e2 ) { ++ $ok = 0; ++ } ++ elsif( !defined $e1 and !defined $e2 ) { ++ # Shortcut if they're both undefined. ++ $ok = 1; ++ } ++ elsif( _dne($e1) xor _dne($e2) ) { ++ $ok = 0; ++ } ++ elsif( $same_ref and( $e1 eq $e2 ) ) { ++ $ok = 1; ++ } ++ elsif($not_ref) { ++ push @Data_Stack, { type => '', vals => [ $e1, $e2 ] }; ++ $ok = 0; ++ } ++ else { ++ if( $Refs_Seen{$e1} ) { ++ return $Refs_Seen{$e1} eq $e2; ++ } ++ else { ++ $Refs_Seen{$e1} = "$e2"; ++ } ++ ++ my $type = _type($e1); ++ $type = 'DIFFERENT' unless _type($e2) eq $type; ++ ++ if( $type eq 'DIFFERENT' ) { ++ push @Data_Stack, { type => $type, vals => [ $e1, $e2 ] }; ++ $ok = 0; ++ } ++ elsif( $type eq 'ARRAY' ) { ++ $ok = _eq_array( $e1, $e2 ); ++ } ++ elsif( $type eq 'HASH' ) { ++ $ok = _eq_hash( $e1, $e2 ); ++ } ++ elsif( $type eq 'REF' ) { ++ push @Data_Stack, { type => $type, vals => [ $e1, $e2 ] }; ++ $ok = _deep_check( $$e1, $$e2 ); ++ pop @Data_Stack if $ok; ++ } ++ elsif( $type eq 'SCALAR' ) { ++ push @Data_Stack, { type => 'REF', vals => [ $e1, $e2 ] }; ++ $ok = _deep_check( $$e1, $$e2 ); ++ pop @Data_Stack if $ok; ++ } ++ elsif($type) { ++ push @Data_Stack, { type => $type, vals => [ $e1, $e2 ] }; ++ $ok = 0; ++ } ++ else { ++ _whoa( 1, "No type in _deep_check" ); ++ } ++ } ++ } ++ ++ return $ok; ++} ++ ++sub _whoa { ++ my( $check, $desc ) = @_; ++ if($check) { ++ die <<"WHOA"; ++WHOA! $desc ++This should never happen! Please contact the author immediately! ++WHOA ++ } ++} ++ ++=item B<eq_hash> ++ ++ my $is_eq = eq_hash(\%got, \%expected); ++ ++Determines if the two hashes contain the same keys and values. This ++is a deep check. ++ ++=cut ++ ++sub eq_hash { ++ local @Data_Stack = (); ++ return _deep_check(@_); ++} ++ ++sub _eq_hash { ++ my( $a1, $a2 ) = @_; ++ ++ if( grep _type($_) ne 'HASH', $a1, $a2 ) { ++ warn "eq_hash passed a non-hash ref"; ++ return 0; ++ } ++ ++ return 1 if $a1 eq $a2; ++ ++ my $ok = 1; ++ my $bigger = keys %$a1 > keys %$a2 ? $a1 : $a2; ++ foreach my $k ( keys %$bigger ) { ++ my $e1 = exists $a1->{$k} ? $a1->{$k} : $DNE; ++ my $e2 = exists $a2->{$k} ? $a2->{$k} : $DNE; ++ ++ next if _equal_nonrefs($e1, $e2); ++ ++ push @Data_Stack, { type => 'HASH', idx => $k, vals => [ $e1, $e2 ] }; ++ $ok = _deep_check( $e1, $e2 ); ++ pop @Data_Stack if $ok; ++ ++ last unless $ok; ++ } ++ ++ return $ok; ++} ++ ++=item B<eq_set> ++ ++ my $is_eq = eq_set(\@got, \@expected); ++ ++Similar to eq_array(), except the order of the elements is B<not> ++important. This is a deep check, but the irrelevancy of order only ++applies to the top level. ++ ++ ok( eq_set(\@got, \@expected) ); ++ ++Is better written: ++ ++ is_deeply( [sort @got], [sort @expected] ); ++ ++B<NOTE> By historical accident, this is not a true set comparison. ++While the order of elements does not matter, duplicate elements do. ++ ++B<NOTE> eq_set() does not know how to deal with references at the top ++level. The following is an example of a comparison which might not work: ++ ++ eq_set([\1, \2], [\2, \1]); ++ ++L<Test::Deep> contains much better set comparison functions. ++ ++=cut ++ ++sub eq_set { ++ my( $a1, $a2 ) = @_; ++ return 0 unless @$a1 == @$a2; ++ ++ no warnings 'uninitialized'; ++ ++ # It really doesn't matter how we sort them, as long as both arrays are ++ # sorted with the same algorithm. ++ # ++ # Ensure that references are not accidentally treated the same as a ++ # string containing the reference. ++ # ++ # Have to inline the sort routine due to a threading/sort bug. ++ # See [rt.cpan.org 6782] ++ # ++ # I don't know how references would be sorted so we just don't sort ++ # them. This means eq_set doesn't really work with refs. ++ return eq_array( ++ [ grep( ref, @$a1 ), sort( grep( !ref, @$a1 ) ) ], ++ [ grep( ref, @$a2 ), sort( grep( !ref, @$a2 ) ) ], ++ ); ++} ++ ++=back ++ ++ ++=head2 Extending and Embedding Test::More ++ ++Sometimes the Test::More interface isn't quite enough. Fortunately, ++Test::More is built on top of Test::Builder which provides a single, ++unified backend for any test library to use. This means two test ++libraries which both use Test::Builder B<can be used together in the ++same program>. ++ ++If you simply want to do a little tweaking of how the tests behave, ++you can access the underlying Test::Builder object like so: ++ ++=over 4 ++ ++=item B<builder> ++ ++ my $test_builder = Test::More->builder; ++ ++Returns the Test::Builder object underlying Test::More for you to play ++with. ++ ++ ++=back ++ ++ ++=head1 EXIT CODES ++ ++If all your tests passed, Test::Builder will exit with zero (which is ++normal). If anything failed it will exit with how many failed. If ++you run less (or more) tests than you planned, the missing (or extras) ++will be considered failures. If no tests were ever run Test::Builder ++will throw a warning and exit with 255. If the test died, even after ++having successfully completed all its tests, it will still be ++considered a failure and will exit with 255. ++ ++So the exit codes are... ++ ++ 0 all tests successful ++ 255 test died or all passed but wrong # of tests run ++ any other number how many failed (including missing or extras) ++ ++If you fail more than 254 tests, it will be reported as 254. ++ ++B<NOTE> This behavior may go away in future versions. ++ ++ ++=head1 COMPATIBILITY ++ ++Test::More works with Perls as old as 5.8.1. ++ ++Thread support is not very reliable before 5.10.1, but that's ++because threads are not very reliable before 5.10.1. ++ ++Although Test::More has been a core module in versions of Perl since 5.6.2, Test::More has evolved since then, and not all of the features you're used to will be present in the shipped version of Test::More. If you are writing a module, don't forget to indicate in your package metadata the minimum version of Test::More that you require. For instance, if you want to use C<done_testing()> but want your test script to run on Perl 5.10.0, you will need to explicitly require Test::More > 0.88. ++ ++Key feature milestones include: ++ ++=over 4 ++ ++=item subtests ++ ++Subtests were released in Test::More 0.94, which came with Perl 5.12.0. Subtests did not implicitly call C<done_testing()> until 0.96; the first Perl with that fix was Perl 5.14.0 with 0.98. ++ ++=item C<done_testing()> ++ ++This was released in Test::More 0.88 and first shipped with Perl in 5.10.1 as part of Test::More 0.92. ++ ++=item C<cmp_ok()> ++ ++Although C<cmp_ok()> was introduced in 0.40, 0.86 fixed an important bug to make it safe for overloaded objects; the fixed first shipped with Perl in 5.10.1 as part of Test::More 0.92. ++ ++=item C<new_ok()> C<note()> and C<explain()> ++ ++These were was released in Test::More 0.82, and first shipped with Perl in 5.10.1 as part of Test::More 0.92. ++ ++=back ++ ++There is a full version history in the Changes file, and the Test::More versions included as core can be found using L<Module::CoreList>: ++ ++ $ corelist -a Test::More ++ ++ ++=head1 CAVEATS and NOTES ++ ++=over 4 ++ ++=item utf8 / "Wide character in print" ++ ++If you use utf8 or other non-ASCII characters with Test::More you ++might get a "Wide character in print" warning. Using C<binmode ++STDOUT, ":utf8"> will not fix it. Test::Builder (which powers ++Test::More) duplicates STDOUT and STDERR. So any changes to them, ++including changing their output disciplines, will not be seem by ++Test::More. ++ ++One work around is to apply encodings to STDOUT and STDERR as early ++as possible and before Test::More (or any other Test module) loads. ++ ++ use open ':std', ':encoding(utf8)'; ++ use Test::More; ++ ++A more direct work around is to change the filehandles used by ++Test::Builder. ++ ++ my $builder = Test::More->builder; ++ binmode $builder->output, ":encoding(utf8)"; ++ binmode $builder->failure_output, ":encoding(utf8)"; ++ binmode $builder->todo_output, ":encoding(utf8)"; ++ ++ ++=item Overloaded objects ++ ++String overloaded objects are compared B<as strings> (or in cmp_ok()'s ++case, strings or numbers as appropriate to the comparison op). This ++prevents Test::More from piercing an object's interface allowing ++better blackbox testing. So if a function starts returning overloaded ++objects instead of bare strings your tests won't notice the ++difference. This is good. ++ ++However, it does mean that functions like is_deeply() cannot be used to ++test the internals of string overloaded objects. In this case I would ++suggest L<Test::Deep> which contains more flexible testing functions for ++complex data structures. ++ ++ ++=item Threads ++ ++Test::More will only be aware of threads if "use threads" has been done ++I<before> Test::More is loaded. This is ok: ++ ++ use threads; ++ use Test::More; ++ ++This may cause problems: ++ ++ use Test::More ++ use threads; ++ ++5.8.1 and above are supported. Anything below that has too many bugs. ++ ++=back ++ ++ ++=head1 HISTORY ++ ++This is a case of convergent evolution with Joshua Pritikin's Test ++module. I was largely unaware of its existence when I'd first ++written my own ok() routines. This module exists because I can't ++figure out how to easily wedge test names into Test's interface (along ++with a few other problems). ++ ++The goal here is to have a testing utility that's simple to learn, ++quick to use and difficult to trip yourself up with while still ++providing more flexibility than the existing Test.pm. As such, the ++names of the most common routines are kept tiny, special cases and ++magic side-effects are kept to a minimum. WYSIWYG. ++ ++ ++=head1 SEE ALSO ++ ++L<Test::Simple> if all this confuses you and you just want to write ++some tests. You can upgrade to Test::More later (it's forward ++compatible). ++ ++L<Test::Harness> is the test runner and output interpreter for Perl. ++It's the thing that powers C<make test> and where the C<prove> utility ++comes from. ++ ++L<Test::Legacy> tests written with Test.pm, the original testing ++module, do not play well with other testing libraries. Test::Legacy ++emulates the Test.pm interface and does play well with others. ++ ++L<Test::Differences> for more ways to test complex data structures. ++And it plays well with Test::More. ++ ++L<Test::Class> is like xUnit but more perlish. ++ ++L<Test::Deep> gives you more powerful complex data structure testing. ++ ++L<Test::Inline> shows the idea of embedded testing. ++ ++L<Bundle::Test> installs a whole bunch of useful test modules. ++ ++ ++=head1 AUTHORS ++ ++Michael G Schwern E<lt>schwern@xxxxxxxxxx<gt> with much inspiration ++from Joshua Pritikin's Test module and lots of help from Barrie ++Slaymaker, Tony Bowden, blackstar.co.uk, chromatic, Fergal Daly and ++the perl-qa gang. ++ ++=head1 MAINTAINERS ++ ++=over 4 ++ ++=item Chad Granum E<lt>exodist@xxxxxxxxx<gt> ++ ++=back ++ ++ ++=head1 BUGS ++ ++See F<http://rt.cpan.org> to report and view bugs. ++ ++ ++=head1 SOURCE ++ ++The source code repository for Test::More can be found at ++F<http://github.com/Test-More/test-more/>. ++ ++ ++=head1 COPYRIGHT ++ ++Copyright 2001-2008 by Michael G Schwern E<lt>schwern@xxxxxxxxxx<gt>. ++ ++This program is free software; you can redistribute it and/or ++modify it under the same terms as Perl itself. ++ ++See F<http://www.perl.com/perl/misc/Artistic.html> ++ ++=cut ++ ++1; -- Fedora Extras Perl SIG http://www.fedoraproject.org/wiki/Extras/SIGs/Perl perl-devel mailing list perl-devel@xxxxxxxxxxxxxxxxxxxxxxx https://admin.fedoraproject.org/mailman/listinfo/perl-devel