Module-Runtime

 view release on metacpan or  search on metacpan

t/upo.t  view on Meta::CPAN

use warnings;
use strict;

use Test::More tests => 42;

BEGIN { use_ok "Module::Runtime", qw(use_package_optimistically); }

unshift @INC, "./t/lib";
my $result;

# a module that doesn't exist
$result = eval { use_package_optimistically("t::NotExist") };
is $@, "";
is $result, "t::NotExist";

# a module that's already loaded
$result = eval { use_package_optimistically("Test::More") };
is $@, "";
is $result, "Test::More";

# a module that we'll load now
$result = eval { use_package_optimistically("t::Simple") };
is $@, "";
is $result, "t::Simple";
no strict "refs";
ok defined(${"t::Simple::VERSION"});

# lexical hints don't leak through
my $have_runtime_hint_hash = "$]" >= 5.009004;
sub test_runtime_hint_hash($$) {
    SKIP: {
        skip "no runtime hint hash", 1 unless $have_runtime_hint_hash;
        is +((caller(0))[10] || {})->{$_[0]}, $_[1];
    }
}
SKIP: {
    skip "core bug makes this test crash", 13
        if "$]" >= 5.008 && "$]" < 5.008004;
    skip "can't work around hint leakage in pure Perl", 13
        if "$]" >= 5.009004 && "$]" < 5.010001;
    $^H |= 0x20000 if "$]" < 5.009004;
    $^H{"Module::Runtime/test_a"} = 1;
    is $^H{"Module::Runtime/test_a"}, 1;
    is $^H{"Module::Runtime/test_b"}, undef;
    use_package_optimistically("t::Hints");
    is $^H{"Module::Runtime/test_a"}, 1;
    is $^H{"Module::Runtime/test_b"}, undef;
    t::Hints->import;
    is $^H{"Module::Runtime/test_a"}, 1;
    is $^H{"Module::Runtime/test_b"}, 1;
    eval q{
        BEGIN { $^H |= 0x20000; $^H{foo} = 1; }
        BEGIN { is $^H{foo}, 1; }
        main::test_runtime_hint_hash("foo", 1);
        BEGIN { use_package_optimistically("Math::BigInt"); }
        BEGIN { is $^H{foo}, 1; }
        main::test_runtime_hint_hash("foo", 1);
        1;
    }; die $@ unless $@ eq "";
}

# broken module is visibly broken when re-required
eval { use_package_optimistically("t::Break") };
like $@, qr/\A(?:broken |Attempt to reload )/;
eval { use_package_optimistically("t::Break") };
like $@, qr/\A(?:broken |Attempt to reload )/;

# module broken by virtue of trying to non-optimistically load a
# non-existent module via "use"
eval { use_package_optimistically("t::Nest0") };
like $@, qr/\ACan't locate /;
eval { use_package_optimistically("t::Nest0") };
like $@, qr/\A(?:Can't locate |Attempt to reload )/;

# module broken by virtue of trying to non-optimistically load a
# non-existent module via require_module()
eval { use_package_optimistically("t::Nest1") };
like $@, qr/\ACan't locate /;
eval { use_package_optimistically("t::Nest1") };
like $@, qr/\A(?:Can't locate |Attempt to reload )/;

# successful version check
$result = eval { use_package_optimistically("Module::Runtime", 0.001) };
is $@, "";
is $result, "Module::Runtime";

# failing version check
$result = eval { use_package_optimistically("Module::Runtime", 999) };
like $@, qr/^Module::Runtime version /;

# even load module if $VERSION already set, unlike older behaviour
$t::Context::VERSION = undef;
$result = eval { use_package_optimistically("t::Context") };
is $@, "";
is $result, "t::Context";
ok defined($t::Context::VERSION);
ok $INC{"t/Context.pm"};

# make sure any version argument gets passed through
my @version_calls;
sub t::HasVersion::VERSION {
    push @version_calls, [@_];
}
$INC{"t/HasVersion.pm"} = 1;
eval { use_package_optimistically("t::HasVersion") };
is $@, "";
is_deeply \@version_calls, [];
@version_calls = ();
eval { use_package_optimistically("t::HasVersion", 2) };
is $@, "";
is_deeply \@version_calls, [["t::HasVersion",2]];
@version_calls = ();



( run in 0.537 second using v1.01-cache-2.11-cpan-0bb4e1dffa6 )