line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
1
|
4
|
|
|
4
|
|
16
|
use strict; |
|
4
|
|
|
|
|
6
|
|
|
4
|
|
|
|
|
194
|
|
2
|
|
|
|
|
|
|
package CPAN::Testers::Common::Client::PrereqCheck; |
3
|
|
|
|
|
|
|
|
4
|
|
|
|
|
|
|
# forked from CPAN::Reporter::PrereqCheck 1.2018 |
5
|
|
|
|
|
|
|
|
6
|
4
|
|
|
4
|
|
3081
|
use ExtUtils::MakeMaker 6.36; |
|
4
|
|
|
|
|
357585
|
|
|
4
|
|
|
|
|
483
|
|
7
|
4
|
|
|
4
|
|
27
|
use File::Spec; |
|
4
|
|
|
|
|
6
|
|
|
4
|
|
|
|
|
70
|
|
8
|
4
|
|
|
4
|
|
1759
|
use CPAN::Version; |
|
4
|
|
|
|
|
5367
|
|
|
4
|
|
|
|
|
1129
|
|
9
|
|
|
|
|
|
|
|
10
|
|
|
|
|
|
|
_run() if ! caller(); |
11
|
|
|
|
|
|
|
|
12
|
|
|
|
|
|
|
sub _run { |
13
|
0
|
|
|
0
|
|
|
my %saw_mod; |
14
|
|
|
|
|
|
|
# read module and prereq string from STDIN |
15
|
|
|
|
|
|
|
# do this as early as possible: https://github.com/cpan-testers/CPAN-Reporter/issues/20 |
16
|
|
|
|
|
|
|
my @modules; |
17
|
0
|
|
|
|
|
|
while ( <> ) { |
18
|
0
|
|
|
|
|
|
push @modules, $_; |
19
|
|
|
|
|
|
|
} |
20
|
0
|
|
|
|
|
|
local *DEVNULL; |
21
|
0
|
|
|
|
|
|
open DEVNULL, '>' . File::Spec->devnull; ## no critic |
22
|
|
|
|
|
|
|
# ensure actually installed, not ./inc/... or ./t/..., etc. |
23
|
0
|
|
|
|
|
|
local @INC = grep { $_ ne '.' } @INC; |
|
0
|
|
|
|
|
|
|
24
|
0
|
|
|
|
|
|
for (@modules) { |
25
|
0
|
|
|
|
|
|
m/^(\S+)\s+([^\n]*)/; |
26
|
0
|
|
|
|
|
|
my ($mod, $need) = ($1, $2); |
27
|
0
|
0
|
|
|
|
|
die "Couldn't read module for '$_'" unless $mod; |
28
|
0
|
0
|
|
|
|
|
$need = 0 if not defined $need; |
29
|
|
|
|
|
|
|
|
30
|
|
|
|
|
|
|
# only evaluate a module once |
31
|
0
|
0
|
|
|
|
|
next if $saw_mod{$mod}++; |
32
|
|
|
|
|
|
|
|
33
|
|
|
|
|
|
|
# get installed version from file with EU::MM |
34
|
0
|
|
|
|
|
|
my($have, $inst_file, $dir, @packpath); |
35
|
0
|
0
|
|
|
|
|
if ( $mod eq "perl" ) { |
36
|
0
|
|
|
|
|
|
$have = $]; |
37
|
|
|
|
|
|
|
} |
38
|
|
|
|
|
|
|
else { |
39
|
0
|
|
|
|
|
|
@packpath = split( /::/, $mod ); |
40
|
0
|
|
|
|
|
|
$packpath[-1] .= '.pm'; |
41
|
0
|
0
|
0
|
|
|
|
if (@packpath == 1 && $packpath[0] eq 'readline.pm') { |
42
|
0
|
|
|
|
|
|
unshift @packpath, 'Term', 'ReadLine'; # historical reasons |
43
|
|
|
|
|
|
|
} |
44
|
|
|
|
|
|
|
INCDIR: |
45
|
0
|
|
|
|
|
|
foreach my $dir (@INC) { |
46
|
0
|
|
|
|
|
|
my $pmfile = File::Spec->catfile($dir,@packpath); |
47
|
0
|
0
|
|
|
|
|
if (-f $pmfile){ |
48
|
0
|
|
|
|
|
|
$inst_file = $pmfile; |
49
|
0
|
|
|
|
|
|
last INCDIR; |
50
|
|
|
|
|
|
|
} |
51
|
|
|
|
|
|
|
} |
52
|
|
|
|
|
|
|
|
53
|
|
|
|
|
|
|
# get version from file or else report missing |
54
|
0
|
0
|
|
|
|
|
if ( defined $inst_file ) { |
55
|
0
|
|
|
|
|
|
$have = my $preliminary_version = MM->parse_version($inst_file); |
56
|
0
|
0
|
0
|
|
|
|
$preliminary_version = '0' if ! defined $preliminary_version || $preliminary_version eq 'undef'; |
57
|
|
|
|
|
|
|
# report broken if it can't be loaded |
58
|
|
|
|
|
|
|
# "select" to try to suppress spurious newlines |
59
|
0
|
|
|
|
|
|
select DEVNULL; ## no critic |
60
|
0
|
0
|
|
|
|
|
if ( ! _try_load( $mod, $preliminary_version ) ) { |
61
|
0
|
|
|
|
|
|
select STDOUT; ## no critic |
62
|
0
|
|
|
|
|
|
print "$mod 0 broken\n"; |
63
|
0
|
|
|
|
|
|
next; |
64
|
|
|
|
|
|
|
} |
65
|
|
|
|
|
|
|
# Now the module is loaded: if MM->parse_version previously failed to |
66
|
|
|
|
|
|
|
# get the version, then we can now look at the value of the $VERSION |
67
|
|
|
|
|
|
|
# variable. |
68
|
0
|
0
|
0
|
|
|
|
if (! defined $have || $have eq 'undef') { |
69
|
4
|
|
|
4
|
|
28
|
no strict 'refs'; |
|
4
|
|
|
|
|
6
|
|
|
4
|
|
|
|
|
2220
|
|
70
|
0
|
|
|
|
|
|
my $mod_version = ${$mod.'::VERSION'}; |
|
0
|
|
|
|
|
|
|
71
|
0
|
0
|
|
|
|
|
if (defined $mod_version) { |
72
|
0
|
|
|
|
|
|
$have = $mod_version; |
73
|
|
|
|
|
|
|
} else { |
74
|
0
|
|
|
|
|
|
$have = 0; # fallback |
75
|
|
|
|
|
|
|
} |
76
|
|
|
|
|
|
|
} |
77
|
0
|
|
|
|
|
|
select STDOUT; ## no critic |
78
|
|
|
|
|
|
|
} |
79
|
|
|
|
|
|
|
else { |
80
|
0
|
|
|
|
|
|
print "$mod 0 n/a\n"; |
81
|
0
|
|
|
|
|
|
next; |
82
|
|
|
|
|
|
|
} |
83
|
|
|
|
|
|
|
} |
84
|
|
|
|
|
|
|
|
85
|
|
|
|
|
|
|
# complex requirements are comma separated |
86
|
0
|
|
|
|
|
|
my ( @requirements ) = split /\s*,\s*/, $need; |
87
|
|
|
|
|
|
|
|
88
|
0
|
|
|
|
|
|
my $passes = 0; |
89
|
|
|
|
|
|
|
RQ: |
90
|
0
|
|
|
|
|
|
for my $rq (@requirements) { |
91
|
0
|
0
|
|
|
|
|
if ($rq =~ s|>=\s*||) { |
|
|
0
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
92
|
|
|
|
|
|
|
# no-op -- just trimmed string |
93
|
|
|
|
|
|
|
} elsif ($rq =~ s|>\s*||) { |
94
|
0
|
0
|
|
|
|
|
if (CPAN::Version->vgt($have,$rq)){ |
95
|
0
|
|
|
|
|
|
$passes++; |
96
|
|
|
|
|
|
|
} |
97
|
0
|
|
|
|
|
|
next RQ; |
98
|
|
|
|
|
|
|
} elsif ($rq =~ s|!=\s*||) { |
99
|
0
|
0
|
|
|
|
|
if (CPAN::Version->vcmp($have,$rq)) { |
100
|
0
|
|
|
|
|
|
$passes++; # didn't match |
101
|
|
|
|
|
|
|
} |
102
|
0
|
|
|
|
|
|
next RQ; |
103
|
|
|
|
|
|
|
} elsif ($rq =~ s|<=\s*||) { |
104
|
0
|
0
|
|
|
|
|
if (! CPAN::Version->vgt($have,$rq)){ |
105
|
0
|
|
|
|
|
|
$passes++; |
106
|
|
|
|
|
|
|
} |
107
|
0
|
|
|
|
|
|
next RQ; |
108
|
|
|
|
|
|
|
} elsif ($rq =~ s|<\s*||) { |
109
|
0
|
0
|
|
|
|
|
if (CPAN::Version->vlt($have,$rq)){ |
110
|
0
|
|
|
|
|
|
$passes++; |
111
|
|
|
|
|
|
|
} |
112
|
0
|
|
|
|
|
|
next RQ; |
113
|
|
|
|
|
|
|
} |
114
|
|
|
|
|
|
|
# if made it here, then it's a normal >= comparison |
115
|
0
|
0
|
|
|
|
|
if (! CPAN::Version->vlt($have, $rq)){ |
116
|
0
|
|
|
|
|
|
$passes++; |
117
|
|
|
|
|
|
|
} |
118
|
|
|
|
|
|
|
} |
119
|
0
|
0
|
|
|
|
|
my $ok = $passes == @requirements ? 1 : 0; |
120
|
0
|
|
|
|
|
|
print "$mod $ok $have\n" |
121
|
|
|
|
|
|
|
} |
122
|
0
|
|
|
|
|
|
return; |
123
|
|
|
|
|
|
|
} |
124
|
|
|
|
|
|
|
|
125
|
|
|
|
|
|
|
sub _try_load { |
126
|
0
|
|
|
0
|
|
|
my ($module, $have) = @_; |
127
|
|
|
|
|
|
|
|
128
|
0
|
|
|
|
|
|
my @do_not_load = ( |
129
|
|
|
|
|
|
|
# should not be loaded directly |
130
|
|
|
|
|
|
|
qw/Term::ReadLine::Perl Term::ReadLine::Gnu MooseX::HasDefaults Readonly::XS |
131
|
|
|
|
|
|
|
POE::Loop::Event SOAP::Constants |
132
|
|
|
|
|
|
|
Moose::Meta::TypeConstraint::Parameterizable Moose::Meta::TypeConstraint::Parameterized/, |
133
|
|
|
|
|
|
|
'Devel::Trepan', #"require Enbugger; require Devel::Trepan;" starts debugging session |
134
|
|
|
|
|
|
|
|
135
|
|
|
|
|
|
|
#removed modules |
136
|
|
|
|
|
|
|
qw/Pegex::Mo YAML::LibYAML/, |
137
|
|
|
|
|
|
|
|
138
|
|
|
|
|
|
|
#have additional prereqs |
139
|
|
|
|
|
|
|
qw/Log::Dispatch::Email::MailSender RDF::NS::Trine Plack::Handler::FCGI Web::Scraper::LibXML/, |
140
|
|
|
|
|
|
|
|
141
|
|
|
|
|
|
|
#modify @INC. 'lib' appearing in @INC will prevent correct |
142
|
|
|
|
|
|
|
#checking of modules with XS part, for ex. List::Util |
143
|
|
|
|
|
|
|
qw/ExtUtils::ParseXS ExtUtils::ParseXS::Utilities/, |
144
|
|
|
|
|
|
|
|
145
|
|
|
|
|
|
|
#require special conditions to run |
146
|
|
|
|
|
|
|
qw/mylib/, |
147
|
|
|
|
|
|
|
|
148
|
|
|
|
|
|
|
#do not return true value |
149
|
|
|
|
|
|
|
qw/perlsecret Alt::Crypt::RSA::BigInt/, |
150
|
|
|
|
|
|
|
); |
151
|
|
|
|
|
|
|
|
152
|
0
|
|
|
|
|
|
my %loading_conflicts = ( |
153
|
|
|
|
|
|
|
'signatures' => ['Catalyst'], |
154
|
|
|
|
|
|
|
'Dancer::Plugin::FlashMessage' => ['Dancer::Plugin::FlashNote'], |
155
|
|
|
|
|
|
|
'Dancer::Plugin::FlashNote' => ['Dancer::Plugin::FlashMessage'], |
156
|
|
|
|
|
|
|
'Dancer::Plugin::Mongoose' => ['Dancer::Plugin::DBIC'], |
157
|
|
|
|
|
|
|
'Dancer::Plugin::DBIC' => ['Dancer::Plugin::Mongoose'], |
158
|
|
|
|
|
|
|
'Test::BDD::Cucumber::Loader' => ['Test::Exception', 'Test::MockObject'], #works in different order |
159
|
|
|
|
|
|
|
'Test::Mock::LWP::UserAgent' => ['HTTP::Response'], |
160
|
|
|
|
|
|
|
'Test::SharedFork' => ['threads'], #dies if $INC{'threads.pm'} |
161
|
|
|
|
|
|
|
'Test::TCP' => ['threads'], #loads Test::SharedFork |
162
|
|
|
|
|
|
|
'Test::Fake::HTTPD' => ['threads'], #loads Test::SharedFork |
163
|
|
|
|
|
|
|
#Note: Test::Perl::Critic and other modules load threads, so reordering will not help |
164
|
|
|
|
|
|
|
); #modules that conflict with each other |
165
|
|
|
|
|
|
|
|
166
|
0
|
|
|
|
|
|
my %load_before = ( |
167
|
|
|
|
|
|
|
'Tk::Font' => 'Tk', |
168
|
|
|
|
|
|
|
'Tk::Widget' => 'Tk', |
169
|
|
|
|
|
|
|
'Tk::Label' => 'Tk', |
170
|
|
|
|
|
|
|
'Tk::Menubutton' => 'Tk', |
171
|
|
|
|
|
|
|
'Tk::Entry' => 'Tk', |
172
|
|
|
|
|
|
|
'Class::MOP::Class' => 'Class::MOP', |
173
|
|
|
|
|
|
|
'Moose::Meta::TypeConstraint::Role' => 'Moose', |
174
|
|
|
|
|
|
|
'Moose::Meta::TypeConstraint::Union' => 'Moose', |
175
|
|
|
|
|
|
|
'Moose::Meta::Attribute::Native' => 'Class::MOP', |
176
|
|
|
|
|
|
|
'Moose::Meta::Role::Attribute' => 'Class::MOP', |
177
|
|
|
|
|
|
|
'Test::More::Hooks' => 'Test::More', |
178
|
|
|
|
|
|
|
'Net::HTTP::Spore::Middleware::DefaultParams' => 'Net::HTTP::Spore::Meta::Method', |
179
|
|
|
|
|
|
|
'Log::Log4perl::Filter' => 'Log::Log4perl', |
180
|
|
|
|
|
|
|
'RDF::DOAP::Project' => 'RDF::Trine', #or other modules that use RDF::Trine will fail |
181
|
|
|
|
|
|
|
); |
182
|
|
|
|
|
|
|
|
183
|
|
|
|
|
|
|
# M::I < 0.95 dies in require, so we can't check if it loads |
184
|
|
|
|
|
|
|
# Instead we just pretend that it works |
185
|
0
|
0
|
0
|
|
|
|
if ( $module eq 'Module::Install' && $have < 0.95 ) { |
|
|
0
|
0
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
186
|
0
|
|
|
|
|
|
return 1; |
187
|
|
|
|
|
|
|
} |
188
|
|
|
|
|
|
|
# circular dependency with Catalyst::Runtime, so this module |
189
|
|
|
|
|
|
|
# does not depends on it, but still does not work without it. |
190
|
|
|
|
|
|
|
elsif ( $module eq 'Catalyst::DispatchType::Regex' && $have <= 5.90032 ) { |
191
|
0
|
|
|
|
|
|
return 1; |
192
|
|
|
|
|
|
|
} |
193
|
0
|
|
|
|
|
|
elsif ( grep { $_ eq $module } @do_not_load ) { |
194
|
0
|
|
|
|
|
|
return 1; |
195
|
|
|
|
|
|
|
} |
196
|
|
|
|
|
|
|
# loading Acme modules like Acme::Bleach can do bad things, |
197
|
|
|
|
|
|
|
# so never try to load them; just pretend that they work |
198
|
|
|
|
|
|
|
elsif( $module =~ /^Acme::/ ) { |
199
|
0
|
|
|
|
|
|
return 1; |
200
|
|
|
|
|
|
|
} |
201
|
|
|
|
|
|
|
|
202
|
0
|
0
|
|
|
|
|
if ( exists $loading_conflicts{$module} ) { |
203
|
0
|
|
|
|
|
|
foreach my $mod1 ( @{ $loading_conflicts{$module} } ) { |
|
0
|
|
|
|
|
|
|
204
|
0
|
|
|
|
|
|
my $file = "$mod1.pm"; |
205
|
0
|
|
|
|
|
|
$file =~ s{::}{/}g; |
206
|
0
|
0
|
|
|
|
|
if (exists $INC{$file}) { |
207
|
0
|
|
|
|
|
|
return 1; |
208
|
|
|
|
|
|
|
} |
209
|
|
|
|
|
|
|
} |
210
|
|
|
|
|
|
|
} |
211
|
|
|
|
|
|
|
|
212
|
0
|
0
|
|
|
|
|
if (exists $load_before{$module}) { |
213
|
0
|
|
|
|
|
|
eval "require $load_before{$module};1;"; |
214
|
|
|
|
|
|
|
} |
215
|
0
|
|
|
|
|
|
my $file = "$module.pm"; |
216
|
0
|
|
|
|
|
|
$file =~ s{::}{/}g; |
217
|
|
|
|
|
|
|
|
218
|
0
|
|
|
|
|
|
return eval {require $file; 1}; ## no critic |
|
0
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
219
|
|
|
|
|
|
|
} |
220
|
|
|
|
|
|
|
|
221
|
|
|
|
|
|
|
1; |
222
|
|
|
|
|
|
|
__END__ |