BEGIN { chdir 't' if -d 't'; @INC = '../lib'; require './test.pl'; } use warnings; no warnings "once"; my $Is_VMS = $^O eq 'VMS'; use Carp qw(carp cluck croak confess); BEGIN { plan tests => 56; # This test must be run at BEGIN time, because code later in this file # sets CORE::GLOBAL::caller ok !exists $CORE::GLOBAL::{caller}, "Loading doesn't create CORE::GLOBAL::caller"; } { local $SIG{__WARN__} = sub { like $_[0], qr/ok (\d+)\n at.+\b(?i:carp\.t) line \d+$/, 'ok 2\n'; }; carp "ok 2\n"; } { local $SIG{__WARN__} = sub { like $_[0], qr/(\d+) at.+\b(?i:carp\.t) line \d+$/, 'carp 3'; }; carp 3; } sub sub_4 { local $SIG{__WARN__} = sub { like $_[0], qr/^(\d+) at.+\b(?i:carp\.t) line \d+\n\tmain::sub_4\(\) called at.+\b(?i:carp\.t) line \d+$/, 'cluck 4'; }; cluck 4; } sub_4; { local $SIG{__DIE__} = sub { like $_[0], qr/^(\d+) at.+\b(?i:carp\.t) line \d+\n\teval \Q{...}\E called at.+\b(?i:carp\.t) line \d+$/, 'croak 5'; }; eval { croak 5 }; } sub sub_6 { local $SIG{__DIE__} = sub { like $_[0], qr/^(\d+) at.+\b(?i:carp\.t) line \d+\n\teval \Q{...}\E called at.+\b(?i:carp\.t) line \d+\n\tmain::sub_6\(\) called at.+\b(?i:carp\.t) line \d+$/, 'confess 6'; }; eval { confess 6 }; } sub_6; ok(1); # test for caller_info API my $eval = "use Carp; return Carp::caller_info(0);"; my %info = eval($eval); is( $info{sub_name}, "eval '$eval'", 'caller_info API' ); # test for '...::CARP_NOT used only once' warning from Carp my $warning; eval { BEGIN { local $SIG{__WARN__} = sub { if ( defined $^S ) { warn $_[0] } else { $warning = $_[0] } } } package Z; BEGIN { eval { Carp::croak() }; } }; ok !$warning, q/'...::CARP_NOT used only once' warning from Carp/; # Test the location of error messages. like( A::short(), qr/^Error at C/, "Short messages skip carped package" ); { local @C::ISA = "D"; like( A::short(), qr/^Error at B/, "Short messages skip inheritance" ); } { local @D::ISA = "C"; like( A::short(), qr/^Error at B/, "Short messages skip inheritance" ); } { local @D::ISA = "B"; local @B::ISA = "C"; like( A::short(), qr/^Error at A/, "Inheritance is transitive" ); } { local @B::ISA = "D"; local @C::ISA = "B"; like( A::short(), qr/^Error at A/, "Inheritance is transitive" ); } { local @C::CARP_NOT = "D"; like( A::short(), qr/^Error at B/, "Short messages see \@CARP_NOT" ); } { local @D::CARP_NOT = "C"; like( A::short(), qr/^Error at B/, "Short messages see \@CARP_NOT" ); } { local @D::CARP_NOT = "B"; local @B::CARP_NOT = "C"; like( A::short(), qr/^Error at A/, "\@CARP_NOT is transitive" ); } { local @B::CARP_NOT = "D"; local @C::CARP_NOT = "B"; like( A::short(), qr/^Error at A/, "\@CARP_NOT is transitive" ); } { local @D::ISA = "C"; local @D::CARP_NOT = "B"; like( A::short(), qr/^Error at C/, "\@CARP_NOT overrides inheritance" ); } { local @D::ISA = "B"; local @D::CARP_NOT = "C"; like( A::short(), qr/^Error at B/, "\@CARP_NOT overrides inheritance" ); } # %Carp::Internal { local $Carp::Internal{C} = 1; like( A::short(), qr/^Error at B/, "Short doesn't report Internal" ); } { local $Carp::Internal{D} = 1; like( A::long(), qr/^Error at C/, "Long doesn't report Internal" ); } # %Carp::CarpInternal { local $Carp::CarpInternal{D} = 1; like( A::short(), qr/^Error at B/, "Short doesn't report calls to CarpInternal" ); } { local $Carp::CarpInternal{D} = 1; like( A::long(), qr/^Error at C/, "Long doesn't report CarpInternal" ); } # tests for global variables sub x { carp @_ } sub w { cluck @_ } # $Carp::Verbose; { my $aref = [ qr/t at \S*(?i:carp.t) line \d+/, qr/t at \S*(?i:carp.t) line \d+\n\s*main::x\('t'\) called at \S*(?i:carp.t) line \d+/ ]; my $i = 0; for my $re (@$aref) { local $Carp::Verbose = $i++; local $SIG{__WARN__} = sub { like $_[0], $re, 'Verbose'; }; package Z; main::x('t'); } } # $Carp::MaxEvalLen { my $test_num = 1; for ( 0, 4 ) { my $txt = "Carp::cluck($test_num)"; local $Carp::MaxEvalLen = $_; local $SIG{__WARN__} = sub { "@_" =~ /'(.+?)(?:\n|')/s; is length($1), length( $_ ? substr( $txt, 0, $_ ) : substr( $txt, 0 ) ), 'MaxEvalLen'; }; eval "$txt"; $test_num++; } } # $Carp::MaxArgLen { for ( 0, 4 ) { my $arg = 'testtest'; local $Carp::MaxArgLen = $_; local $SIG{__WARN__} = sub { "@_" =~ /'(.+?)'/; is length($1), length( $_ ? substr( $arg, 0, $_ ) : substr( $arg, 0 ) ), 'MaxArgLen'; }; package Z; main::w($arg); } } # $Carp::MaxArgNums { my $i = 0; my $aref = [ qr/1234 at \S*(?i:carp.t) line \d+\n\s*main::w\(1, 2, 3, 4\) called at \S*(?i:carp.t) line \d+/, qr/1234 at \S*(?i:carp.t) line \d+\n\s*main::w\(1, 2, \.\.\.\) called at \S*(?i:carp.t) line \d+/, ]; for (@$aref) { local $Carp::MaxArgNums = $i++; local $SIG{__WARN__} = sub { like "@_", $_, 'MaxArgNums'; }; package Z; main::w( 1 .. 4 ); } } # $Carp::CarpLevel { my $i = 0; my $aref = [ qr/1 at \S*(?i:carp.t) line \d+\n\s*main::w\(1\) called at \S*(?i:carp.t) line \d+/, qr/1 at \S*(?i:carp.t) line \d+$/, ]; for (@$aref) { local $Carp::CarpLevel = $i++; local $SIG{__WARN__} = sub { like "@_", $_, 'CarpLevel'; }; package Z; main::w(1); } } { local $TODO = "VMS exit status semantics don't work this way" if $Is_VMS; # Check that croak() and confess() don't clobber $! runperl( prog => 'use Carp; $@=q{Phooey}; $!=42; croak(q{Dead})', stderr => 1 ); is( $? >> 8, 42, 'croak() doesn\'t clobber $!' ); runperl( prog => 'use Carp; $@=q{Phooey}; $!=42; confess(q{Dead})', stderr => 1 ); is( $? >> 8, 42, 'confess() doesn\'t clobber $!' ); } # undef used to be incorrectly reported as the string "undef" sub cluck_undef { local $SIG{__WARN__} = sub { like $_[0], qr/^Bang! at.+\b(?i:carp\.t) line \d+\n\tmain::cluck_undef\(0, 'undef', 2, undef, 4\) called at.+\b(?i:carp\.t) line \d+$/, "cluck doesn't quote undef"; }; cluck "Bang!" } cluck_undef( 0, "undef", 2, undef, 4 ); # check that Carp respects CORE::GLOBAL::caller override after Carp # has been compiled for my $bodge_job ( 2, 1, 0 ) { print '# ', ( $bodge_job ? 'Not ' : '' ), "setting \@DB::args in caller override\n"; if ( $bodge_job == 1 ) { require B; print "# required B\n"; } my $accum = ''; local *CORE::GLOBAL::caller = sub { local *__ANON__ = "fakecaller"; my @c = CORE::caller(@_); $c[0] ||= 'undef'; $accum .= "@c[0..3]\n"; if ( !$bodge_job && CORE::caller() eq 'DB' ) { package DB; return CORE::caller( ( $_[0] || 0 ) + 1 ); } else { return CORE::caller( ( $_[0] || 0 ) + 1 ); } }; eval "scalar caller()"; like( $accum, qr/main::fakecaller/, "test CORE::GLOBAL::caller override in eval" ); $accum = ''; my $got = A::long(42); like( $accum, qr/main::fakecaller/, "test CORE::GLOBAL::caller override in Carp" ); my $package = 'A'; my $where = $bodge_job == 1 ? ' in &main::__ANON__' : ''; my $warning = $bodge_job ? "\Q** Incomplete caller override detected$where; \@DB::args were not set **\E" : ''; for ( 0 .. 2 ) { my $previous_package = $package; ++$package; like( $got, qr/${package}::long\($warning\) called at $previous_package line \d+/, "Correct arguments for $package" ); } my $arg = $bodge_job ? $warning : 42; like( $got, qr!A::long\($arg\) called at.+\b(?i:carp\.t) line \d+!, 'Correct arguments for A' ); } eval <<'EOT'; no warnings 'redefine'; sub CORE::GLOBAL::caller { my $height = $_[0]; $height++; return CORE::caller($height); } EOT my $got = A::long(42); like( $got, qr!A::long\(\Q** Incomplete caller override detected; \E\@DB::args\Q were not set **\E\) called at.+\b(?i:carp\.t) line \d+!, 'Correct arguments for A' ); # New tests go here # line 1 "A" package A; sub short { B::short(); } sub long { B::long(); } # line 1 "B" package B; sub short { C::short(); } sub long { C::long(); } # line 1 "C" package C; sub short { D::short(); } sub long { D::long(); } # line 1 "D" package D; sub short { eval { Carp::croak("Error") }; return $@; } sub long { eval { Carp::confess("Error") }; return $@; } # Put new tests at "new tests go here" __END__