line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
1
|
|
|
|
|
|
|
package Perl::Lint::Policy::Variables::ProhibitPackageVars; |
2
|
133
|
|
|
133
|
|
73284
|
use strict; |
|
133
|
|
|
|
|
190
|
|
|
133
|
|
|
|
|
3199
|
|
3
|
133
|
|
|
133
|
|
428
|
use warnings; |
|
133
|
|
|
|
|
171
|
|
|
133
|
|
|
|
|
2807
|
|
4
|
133
|
|
|
133
|
|
403
|
use List::Util qw/any/; |
|
133
|
|
|
|
|
142
|
|
|
133
|
|
|
|
|
6421
|
|
5
|
133
|
|
|
133
|
|
844
|
use Perl::Lint::Constants::Type; |
|
133
|
|
|
|
|
145
|
|
|
133
|
|
|
|
|
57915
|
|
6
|
133
|
|
|
133
|
|
548
|
use parent "Perl::Lint::Policy"; |
|
133
|
|
|
|
|
146
|
|
|
133
|
|
|
|
|
553
|
|
7
|
|
|
|
|
|
|
|
8
|
|
|
|
|
|
|
use constant { |
9
|
133
|
|
|
|
|
145419
|
DESC => 'Package variable declared or used', |
10
|
|
|
|
|
|
|
EXPL => [73, 75], |
11
|
133
|
|
|
133
|
|
7091
|
}; |
|
133
|
|
|
|
|
152
|
|
12
|
|
|
|
|
|
|
|
13
|
|
|
|
|
|
|
sub evaluate { |
14
|
10
|
|
|
10
|
0
|
21
|
my ($class, $file, $tokens, $src, $args) = @_; |
15
|
|
|
|
|
|
|
|
16
|
10
|
|
|
|
|
23
|
my @allowed_packages = qw/Data::Dumper File::Find FindBin Log::Log4perl/; |
17
|
10
|
100
|
|
|
|
26
|
if (my $this_policies_arg = $args->{prohibit_package_vars}) { |
18
|
3
|
50
|
|
|
|
13
|
if (my $add_packages = $this_policies_arg->{add_packages}) { |
19
|
3
|
|
|
|
|
11
|
push @allowed_packages, split /\s+/, $add_packages; |
20
|
|
|
|
|
|
|
} |
21
|
|
|
|
|
|
|
} |
22
|
|
|
|
|
|
|
|
23
|
10
|
|
|
|
|
12
|
my @violations; |
24
|
10
|
|
|
|
|
33
|
for (my $i = 0, my $token_type, my $token_data; my $token = $tokens->[$i]; $i++) { |
25
|
226
|
|
|
|
|
160
|
$token_type = $token->{type}; |
26
|
226
|
|
|
|
|
170
|
$token_data = $token->{data}; |
27
|
|
|
|
|
|
|
|
28
|
226
|
100
|
100
|
|
|
2002
|
if ($token_type == OUR_DECL) { |
|
|
100
|
100
|
|
|
|
|
|
|
100
|
100
|
|
|
|
|
|
|
100
|
66
|
|
|
|
|
|
|
100
|
66
|
|
|
|
|
|
|
|
66
|
|
|
|
|
|
|
|
100
|
|
|
|
|
29
|
10
|
|
|
|
|
9
|
$token = $tokens->[++$i]; |
30
|
10
|
|
|
|
|
12
|
$token_type = $token->{type}; |
31
|
10
|
100
|
66
|
|
|
26
|
if ($token_type == LEFT_PAREN) { |
|
|
50
|
33
|
|
|
|
|
|
|
|
33
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
0
|
|
|
|
|
32
|
5
|
|
|
|
|
6
|
my $violation; |
33
|
5
|
|
|
|
|
5
|
my $left_paren_num = 1; |
34
|
5
|
|
|
|
|
10
|
for ($i++; $token = $tokens->[$i]; $i++) { |
35
|
20
|
|
|
|
|
13
|
$token_type = $token->{type}; |
36
|
20
|
50
|
66
|
|
|
111
|
if ($token_type == LEFT_PAREN) { |
|
|
100
|
100
|
|
|
|
|
|
|
100
|
66
|
|
|
|
|
|
|
|
66
|
|
|
|
|
|
|
|
33
|
|
|
|
|
37
|
0
|
|
|
|
|
0
|
$left_paren_num++; |
38
|
|
|
|
|
|
|
} |
39
|
|
|
|
|
|
|
elsif ($token_type == RIGHT_PAREN) { |
40
|
5
|
50
|
|
|
|
9
|
if (--$left_paren_num <= 0) { |
41
|
5
|
100
|
|
|
|
7
|
if ($violation) { |
42
|
4
|
|
|
|
|
4
|
push @violations, $violation; |
43
|
4
|
|
|
|
|
4
|
undef $violation; |
44
|
|
|
|
|
|
|
} |
45
|
5
|
|
|
|
|
12
|
last; |
46
|
|
|
|
|
|
|
} |
47
|
|
|
|
|
|
|
} |
48
|
|
|
|
|
|
|
elsif ( |
49
|
|
|
|
|
|
|
$token_type == GLOBAL_VAR || |
50
|
|
|
|
|
|
|
$token_type == GLOBAL_ARRAY_VAR || |
51
|
|
|
|
|
|
|
$token_type == GLOBAL_HASH_VAR || |
52
|
|
|
|
|
|
|
$token_type == VAR || |
53
|
|
|
|
|
|
|
$token_type == ARRAY_VAR || |
54
|
|
|
|
|
|
|
$token_type == HASH_VAR |
55
|
|
|
|
|
|
|
) { |
56
|
10
|
100
|
|
|
|
28
|
if ($token->{data} !~ /\A.[A-Z0-9_]+\Z/) { |
57
|
|
|
|
|
|
|
$violation ||= +{ |
58
|
|
|
|
|
|
|
filename => $file, |
59
|
|
|
|
|
|
|
line => $token->{line}, |
60
|
6
|
|
100
|
|
|
30
|
description => DESC, |
61
|
|
|
|
|
|
|
explanation => EXPL, |
62
|
|
|
|
|
|
|
policy => __PACKAGE__, |
63
|
|
|
|
|
|
|
}; |
64
|
|
|
|
|
|
|
} |
65
|
|
|
|
|
|
|
} |
66
|
|
|
|
|
|
|
} |
67
|
|
|
|
|
|
|
} |
68
|
|
|
|
|
|
|
elsif ( |
69
|
|
|
|
|
|
|
$token_type == GLOBAL_VAR || |
70
|
|
|
|
|
|
|
$token_type == GLOBAL_ARRAY_VAR || |
71
|
|
|
|
|
|
|
$token_type == GLOBAL_HASH_VAR || |
72
|
|
|
|
|
|
|
$token_type == VAR || |
73
|
|
|
|
|
|
|
$token_type == ARRAY_VAR || |
74
|
|
|
|
|
|
|
$token_type == HASH_VAR |
75
|
|
|
|
|
|
|
) { |
76
|
5
|
100
|
|
|
|
19
|
if ($token->{data} !~ /\A.[A-Z0-9_]+\Z/) { |
77
|
|
|
|
|
|
|
push @violations, { |
78
|
|
|
|
|
|
|
filename => $file, |
79
|
|
|
|
|
|
|
line => $token->{line}, |
80
|
2
|
|
|
|
|
17
|
description => DESC, |
81
|
|
|
|
|
|
|
explanation => EXPL, |
82
|
|
|
|
|
|
|
policy => __PACKAGE__, |
83
|
|
|
|
|
|
|
}; |
84
|
|
|
|
|
|
|
} |
85
|
|
|
|
|
|
|
} |
86
|
|
|
|
|
|
|
} |
87
|
|
|
|
|
|
|
elsif ($token_type == LOCAL_DECL) { |
88
|
8
|
|
|
|
|
8
|
$token = $tokens->[++$i]; |
89
|
8
|
|
|
|
|
7
|
$token_type = $token->{type}; |
90
|
8
|
100
|
|
|
|
11
|
if ($token_type == LEFT_PAREN) { |
91
|
2
|
|
|
|
|
2
|
my $violation; |
92
|
2
|
|
|
|
|
1
|
my $left_paren_num = 1; |
93
|
2
|
|
|
|
|
2
|
my $does_exist_namespace_resolver = 0; |
94
|
|
|
|
|
|
|
|
95
|
2
|
|
|
|
|
3
|
my @namespaces; |
96
|
|
|
|
|
|
|
|
97
|
|
|
|
|
|
|
my @packages; |
98
|
0
|
|
|
|
|
0
|
my @var_names; |
99
|
2
|
|
|
|
|
5
|
for ($i++; $token = $tokens->[$i]; $i++) { |
100
|
20
|
|
|
|
|
11
|
$token_type = $token->{type}; |
101
|
20
|
50
|
|
|
|
38
|
if ($token_type == LEFT_PAREN) { |
|
|
100
|
|
|
|
|
|
|
|
100
|
|
|
|
|
|
|
|
100
|
|
|
|
|
|
102
|
0
|
|
|
|
|
0
|
$left_paren_num++; |
103
|
|
|
|
|
|
|
} |
104
|
|
|
|
|
|
|
elsif ($token_type == RIGHT_PAREN) { |
105
|
2
|
|
|
|
|
2
|
push @var_names, pop @namespaces; |
106
|
2
|
|
|
|
|
3
|
push @packages, join '::', @namespaces; |
107
|
2
|
50
|
|
|
|
6
|
if (--$left_paren_num <= 0) { |
108
|
2
|
|
|
|
|
2
|
last; |
109
|
|
|
|
|
|
|
} |
110
|
0
|
|
|
|
|
0
|
@namespaces = (); |
111
|
|
|
|
|
|
|
} |
112
|
|
|
|
|
|
|
elsif ($token_type == COMMA) { |
113
|
2
|
|
|
|
|
3
|
push @var_names, pop @namespaces; |
114
|
2
|
|
|
|
|
4
|
push @packages, join '::', @namespaces; |
115
|
2
|
|
|
|
|
4
|
@namespaces = (); |
116
|
|
|
|
|
|
|
} |
117
|
|
|
|
|
|
|
elsif ($token_type == NAMESPACE_RESOLVER) { |
118
|
6
|
|
|
|
|
8
|
$does_exist_namespace_resolver = 1; |
119
|
|
|
|
|
|
|
} |
120
|
|
|
|
|
|
|
else { |
121
|
10
|
|
|
|
|
19
|
push @namespaces, $token->{data}; |
122
|
|
|
|
|
|
|
} |
123
|
|
|
|
|
|
|
} |
124
|
|
|
|
|
|
|
|
125
|
2
|
50
|
|
|
|
6
|
if ($does_exist_namespace_resolver) { |
126
|
2
|
|
|
|
|
2
|
$token = $tokens->[++$i]; |
127
|
2
|
100
|
|
|
|
5
|
if ($token->{type} == ASSIGN) { |
128
|
1
|
|
|
|
|
2
|
my $is_violated = 0; |
129
|
1
|
|
|
|
|
2
|
for my $package (@packages) { |
130
|
2
|
50
|
|
8
|
|
7
|
if (!any {$package =~ /\A[\$\@\%]$_/} @allowed_packages) { |
|
8
|
|
|
|
|
50
|
|
131
|
0
|
|
|
|
|
0
|
$is_violated = 1; |
132
|
|
|
|
|
|
|
} |
133
|
|
|
|
|
|
|
} |
134
|
|
|
|
|
|
|
|
135
|
|
|
|
|
|
|
# TODO check @var_names ? |
136
|
|
|
|
|
|
|
|
137
|
1
|
50
|
|
|
|
5
|
if ($is_violated) { |
138
|
|
|
|
|
|
|
push @violations, { |
139
|
|
|
|
|
|
|
filename => $file, |
140
|
|
|
|
|
|
|
line => $token->{line}, |
141
|
0
|
|
|
|
|
0
|
description => DESC, |
142
|
|
|
|
|
|
|
explanation => EXPL, |
143
|
|
|
|
|
|
|
policy => __PACKAGE__, |
144
|
|
|
|
|
|
|
}; |
145
|
|
|
|
|
|
|
} |
146
|
|
|
|
|
|
|
} |
147
|
|
|
|
|
|
|
} |
148
|
|
|
|
|
|
|
} |
149
|
|
|
|
|
|
|
else { |
150
|
6
|
|
|
|
|
6
|
my $does_exist_namespace_resolver = 0; |
151
|
6
|
|
|
|
|
3
|
my $is_assigned = 0; |
152
|
6
|
|
|
|
|
9
|
my @namespaces = ($token->{data}); |
153
|
6
|
|
|
|
|
12
|
for ($i++; $token = $tokens->[$i]; $i++) { |
154
|
26
|
|
|
|
|
19
|
$token_type = $token->{type}; |
155
|
26
|
|
|
|
|
21
|
$token_data = $token->{data}; |
156
|
26
|
100
|
|
|
|
51
|
if ($token_type == NAMESPACE) { |
|
|
100
|
|
|
|
|
|
|
|
100
|
|
|
|
|
|
|
|
100
|
|
|
|
|
|
157
|
7
|
|
|
|
|
11
|
push @namespaces, $token_data; |
158
|
|
|
|
|
|
|
} |
159
|
|
|
|
|
|
|
elsif ($token_type == NAMESPACE_RESOLVER) { |
160
|
7
|
|
|
|
|
9
|
$does_exist_namespace_resolver = 1; |
161
|
|
|
|
|
|
|
} |
162
|
|
|
|
|
|
|
elsif ($token_type == ASSIGN) { |
163
|
1
|
|
|
|
|
2
|
$is_assigned = 1; |
164
|
1
|
|
|
|
|
1
|
last; |
165
|
|
|
|
|
|
|
} |
166
|
|
|
|
|
|
|
elsif ($token_type == SEMI_COLON) { |
167
|
5
|
|
|
|
|
5
|
last; |
168
|
|
|
|
|
|
|
} |
169
|
|
|
|
|
|
|
} |
170
|
|
|
|
|
|
|
|
171
|
6
|
100
|
66
|
|
|
25
|
if ($does_exist_namespace_resolver && $is_assigned) { |
172
|
1
|
|
|
|
|
2
|
pop @namespaces; # throw variable name away |
173
|
1
|
|
|
|
|
4
|
my $package_name = join '::', @namespaces; |
174
|
1
|
50
|
|
4
|
|
6
|
if (any {$package_name =~ /\A[\$\@\%]$_/} @allowed_packages) { |
|
4
|
|
|
|
|
46
|
|
175
|
1
|
|
|
|
|
6
|
next; |
176
|
|
|
|
|
|
|
} |
177
|
|
|
|
|
|
|
|
178
|
|
|
|
|
|
|
# TODO check the var name |
179
|
|
|
|
|
|
|
push @violations, { |
180
|
|
|
|
|
|
|
filename => $file, |
181
|
|
|
|
|
|
|
line => $token->{line}, |
182
|
0
|
|
|
|
|
0
|
description => DESC, |
183
|
|
|
|
|
|
|
explanation => EXPL, |
184
|
|
|
|
|
|
|
policy => __PACKAGE__, |
185
|
|
|
|
|
|
|
}; |
186
|
|
|
|
|
|
|
} |
187
|
|
|
|
|
|
|
} |
188
|
|
|
|
|
|
|
} |
189
|
|
|
|
|
|
|
elsif ( |
190
|
|
|
|
|
|
|
$token_type == GLOBAL_VAR || |
191
|
|
|
|
|
|
|
$token_type == GLOBAL_ARRAY_VAR || |
192
|
|
|
|
|
|
|
$token_type == GLOBAL_HASH_VAR || |
193
|
|
|
|
|
|
|
$token_type == VAR || |
194
|
|
|
|
|
|
|
$token_type == ARRAY_VAR || |
195
|
|
|
|
|
|
|
$token_type == HASH_VAR |
196
|
|
|
|
|
|
|
) { |
197
|
28
|
|
|
|
|
38
|
my @namespaces = ($token->{data}); |
198
|
|
|
|
|
|
|
|
199
|
28
|
100
|
|
|
|
43
|
my $does_exist_namespace_resolver = $tokens->[$i+1]->{type} == NAMESPACE_RESOLVER ? 1 : 0; |
200
|
|
|
|
|
|
|
|
201
|
28
|
|
|
|
|
55
|
for ($i++; $token = $tokens->[$i]; $i++) { |
202
|
110
|
|
|
|
|
76
|
$token_type = $token->{type}; |
203
|
110
|
100
|
100
|
|
|
316
|
if ($token_type == ASSIGN || $token_type == SEMI_COLON) { |
|
|
100
|
|
|
|
|
|
204
|
28
|
|
|
|
|
27
|
last; |
205
|
|
|
|
|
|
|
} |
206
|
|
|
|
|
|
|
elsif ($token_type == NAMESPACE) { |
207
|
35
|
|
|
|
|
56
|
push @namespaces, $token->{data}; |
208
|
|
|
|
|
|
|
} |
209
|
|
|
|
|
|
|
} |
210
|
|
|
|
|
|
|
|
211
|
28
|
100
|
|
|
|
43
|
if ($does_exist_namespace_resolver) { |
212
|
24
|
|
|
|
|
23
|
my $var_name = pop @namespaces; |
213
|
|
|
|
|
|
|
|
214
|
24
|
|
|
|
|
38
|
my $package_name = join '::', @namespaces; |
215
|
24
|
100
|
|
92
|
|
111
|
if (any {$package_name =~ /\A[\$\@\%]$_/} @allowed_packages) { |
|
92
|
|
|
|
|
671
|
|
216
|
7
|
|
|
|
|
29
|
next; |
217
|
|
|
|
|
|
|
} |
218
|
|
|
|
|
|
|
|
219
|
17
|
100
|
|
|
|
71
|
if ($var_name !~ /\A.[A-Z0-9_]+\Z/) { |
220
|
|
|
|
|
|
|
push @violations, { |
221
|
|
|
|
|
|
|
filename => $file, |
222
|
|
|
|
|
|
|
line => $token->{line}, |
223
|
12
|
|
|
|
|
57
|
description => DESC, |
224
|
|
|
|
|
|
|
explanation => EXPL, |
225
|
|
|
|
|
|
|
policy => __PACKAGE__, |
226
|
|
|
|
|
|
|
}; |
227
|
|
|
|
|
|
|
} |
228
|
|
|
|
|
|
|
} |
229
|
|
|
|
|
|
|
} |
230
|
|
|
|
|
|
|
elsif ($token_type == SPECIFIC_VALUE && $token_data eq '$:') { |
231
|
3
|
|
|
|
|
6
|
$token = $tokens->[++$i]; |
232
|
3
|
50
|
|
|
|
7
|
my $does_exist_namespace_resolver = $token->{type} == COLON ? 1 : 0; |
233
|
|
|
|
|
|
|
|
234
|
3
|
|
|
|
|
5
|
my $var_token; |
235
|
3
|
|
|
|
|
7
|
for ($i++; $token = $tokens->[$i]; $i++) { |
236
|
12
|
|
|
|
|
9
|
$token_type = $token->{type}; |
237
|
12
|
100
|
|
|
|
27
|
if ($token_type == ASSIGN) { |
|
|
100
|
|
|
|
|
|
238
|
3
|
|
|
|
|
8
|
$var_token = $tokens->[$i-1]; |
239
|
|
|
|
|
|
|
} |
240
|
|
|
|
|
|
|
elsif ($token_type == SEMI_COLON) { # XXX skip to the edge |
241
|
3
|
|
|
|
|
2
|
last; |
242
|
|
|
|
|
|
|
} |
243
|
|
|
|
|
|
|
} |
244
|
|
|
|
|
|
|
|
245
|
3
|
100
|
66
|
|
|
21
|
if ($does_exist_namespace_resolver && $var_token->{data} !~ /\A.[A-Z0-9_]+\Z/) { |
246
|
|
|
|
|
|
|
push @violations, { |
247
|
|
|
|
|
|
|
filename => $file, |
248
|
|
|
|
|
|
|
line => $token->{line}, |
249
|
2
|
|
|
|
|
12
|
description => DESC, |
250
|
|
|
|
|
|
|
explanation => EXPL, |
251
|
|
|
|
|
|
|
policy => __PACKAGE__, |
252
|
|
|
|
|
|
|
}; |
253
|
|
|
|
|
|
|
} |
254
|
|
|
|
|
|
|
} |
255
|
|
|
|
|
|
|
elsif ($token_type == USED_NAME && $token_data eq 'vars') { |
256
|
10
|
|
|
|
|
8
|
my $is_used_package_var = 0; |
257
|
10
|
|
|
|
|
15
|
for ($i++; $token = $tokens->[$i]; $i++) { |
258
|
47
|
|
|
|
|
34
|
$token_type = $token->{type}; |
259
|
47
|
|
|
|
|
38
|
$token_data = $token->{data}; |
260
|
|
|
|
|
|
|
|
261
|
47
|
100
|
100
|
|
|
141
|
if ($token_type == REG_EXP) { |
|
|
100
|
|
|
|
|
|
|
|
100
|
|
|
|
|
|
262
|
3
|
|
|
|
|
11
|
for my $elem (split /\s+/, $token_data) { |
263
|
6
|
50
|
|
|
|
14
|
if ($elem =~ /\A[\$\@\%](.*)\Z/) { |
264
|
6
|
100
|
|
|
|
18
|
if ($1 !~ /\A[A-Z0-9_]+\Z/) { |
265
|
3
|
|
|
|
|
5
|
$is_used_package_var = 1; |
266
|
|
|
|
|
|
|
} |
267
|
|
|
|
|
|
|
} |
268
|
|
|
|
|
|
|
} |
269
|
|
|
|
|
|
|
} |
270
|
|
|
|
|
|
|
elsif ($token_type == STRING || $token_type == RAW_STRING) { |
271
|
13
|
100
|
|
|
|
28
|
if ($token_data =~ /\A[\$\@\%](.*)\Z/) { |
272
|
12
|
100
|
|
|
|
29
|
if ($1 !~ /\A[A-Z0-9_]+\Z/) { |
273
|
6
|
|
|
|
|
9
|
$is_used_package_var = 1; |
274
|
|
|
|
|
|
|
} |
275
|
|
|
|
|
|
|
} |
276
|
|
|
|
|
|
|
} |
277
|
|
|
|
|
|
|
elsif ($token_type == SEMI_COLON) { |
278
|
10
|
|
|
|
|
8
|
last; |
279
|
|
|
|
|
|
|
} |
280
|
|
|
|
|
|
|
} |
281
|
10
|
100
|
|
|
|
16
|
if ($is_used_package_var) { |
282
|
|
|
|
|
|
|
push @violations, { |
283
|
|
|
|
|
|
|
filename => $file, |
284
|
|
|
|
|
|
|
line => $token->{line}, |
285
|
6
|
|
|
|
|
21
|
description => DESC, |
286
|
|
|
|
|
|
|
explanation => EXPL, |
287
|
|
|
|
|
|
|
policy => __PACKAGE__, |
288
|
|
|
|
|
|
|
}; |
289
|
|
|
|
|
|
|
} |
290
|
|
|
|
|
|
|
} |
291
|
|
|
|
|
|
|
} |
292
|
|
|
|
|
|
|
|
293
|
10
|
|
|
|
|
43
|
return \@violations; |
294
|
|
|
|
|
|
|
} |
295
|
|
|
|
|
|
|
|
296
|
|
|
|
|
|
|
1; |
297
|
|
|
|
|
|
|
|