| line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
|
1
|
|
|
|
|
|
|
# cmdline.yp: Parse::Yapp file for the command-line parser for App::GitFind |
|
2
|
|
|
|
|
|
|
# Copyright (c) 2019 Christopher White. |
|
3
|
|
|
|
|
|
|
# Copyright (c) 2019 D3 Engineering, LLC. |
|
4
|
|
|
|
|
|
|
# Licensed MIT. |
|
5
|
|
|
|
|
|
|
|
|
6
|
|
|
|
|
|
|
############################################################################# |
|
7
|
|
|
|
|
|
|
# Header |
|
8
|
|
|
|
|
|
|
|
|
9
|
|
|
|
|
|
|
%{ |
|
10
|
|
|
|
|
|
|
|
|
11
|
|
|
|
|
|
|
# Imports {{{1 |
|
12
|
|
|
|
|
|
|
|
|
13
|
4
|
|
|
4
|
|
60
|
use 5.010; |
|
|
4
|
|
|
|
|
10
|
|
|
14
|
4
|
|
|
4
|
|
18
|
use strict; |
|
|
4
|
|
|
|
|
5
|
|
|
|
4
|
|
|
|
|
79
|
|
|
15
|
4
|
|
|
4
|
|
16
|
use warnings; |
|
|
4
|
|
|
|
|
5
|
|
|
|
4
|
|
|
|
|
145
|
|
|
16
|
|
|
|
|
|
|
|
|
17
|
4
|
|
|
4
|
|
752
|
use App::GitFind::Base; |
|
|
4
|
|
|
|
|
5
|
|
|
|
4
|
|
|
|
|
624
|
|
|
18
|
4
|
|
|
4
|
|
1104
|
use App::GitFind::Actions; |
|
|
4
|
|
|
|
|
9
|
|
|
|
4
|
|
|
|
|
19
|
|
|
19
|
4
|
|
|
4
|
|
1911
|
use Hash::Merge; |
|
|
4
|
|
|
|
|
31891
|
|
|
|
4
|
|
|
|
|
462
|
|
|
20
|
|
|
|
|
|
|
|
|
21
|
|
|
|
|
|
|
# Debugging support |
|
22
|
|
|
|
|
|
|
BEGIN { |
|
23
|
4
|
50
|
50
|
4
|
|
27
|
if($App::GitFind::cmdline::SHOW_AST // 0) { |
|
24
|
0
|
|
|
|
|
0
|
require XXX; |
|
25
|
0
|
|
|
|
|
0
|
XXX->import; |
|
26
|
|
|
|
|
|
|
} else { # !SHOW_AST - make YYY a passthrough |
|
27
|
4
|
|
|
4
|
|
25
|
no strict 'refs'; |
|
|
4
|
|
|
|
|
6
|
|
|
|
4
|
|
|
|
|
175
|
|
|
28
|
4
|
|
|
|
|
80
|
*{'App::GitFind::cmdline::YYY'} = sub { |
|
29
|
78
|
50
|
|
78
|
|
23629
|
return wantarray ? @_ : $_[0]; |
|
30
|
4
|
|
|
|
|
13
|
}; |
|
31
|
|
|
|
|
|
|
} |
|
32
|
|
|
|
|
|
|
} #BEGIN |
|
33
|
|
|
|
|
|
|
|
|
34
|
4
|
|
|
4
|
|
19
|
BEGIN { YYY +{ 'YYY loaded' => 1 } } |
|
35
|
|
|
|
|
|
|
|
|
36
|
|
|
|
|
|
|
# }}}1 |
|
37
|
|
|
|
|
|
|
# Documentation {{{1 |
|
38
|
|
|
|
|
|
|
|
|
39
|
|
|
|
|
|
|
=head1 NAME |
|
40
|
|
|
|
|
|
|
|
|
41
|
|
|
|
|
|
|
App::GitFind::cmdline - Command-line parser for git-find |
|
42
|
|
|
|
|
|
|
|
|
43
|
|
|
|
|
|
|
=head1 SYNOPSIS |
|
44
|
|
|
|
|
|
|
|
|
45
|
|
|
|
|
|
|
Generate the .pm file: |
|
46
|
|
|
|
|
|
|
|
|
47
|
|
|
|
|
|
|
yapp -m App::GitFind::cmdline -o lib/App/GitFind/cmdline.pm support/cmdline.yp |
|
48
|
|
|
|
|
|
|
|
|
49
|
|
|
|
|
|
|
And then: |
|
50
|
|
|
|
|
|
|
|
|
51
|
|
|
|
|
|
|
use App::GitFind::cmdline; |
|
52
|
|
|
|
|
|
|
App::GitFind::cmdline::Parse(\@ARGV); |
|
53
|
|
|
|
|
|
|
|
|
54
|
|
|
|
|
|
|
For debugging output, define C<$SHOW_AST> before the C<use> statement: |
|
55
|
|
|
|
|
|
|
|
|
56
|
|
|
|
|
|
|
BEGIN { $App::GitFind::cmdline::SHOW_AST = 1; } |
|
57
|
|
|
|
|
|
|
|
|
58
|
|
|
|
|
|
|
=head1 FUNCTIONS |
|
59
|
|
|
|
|
|
|
|
|
60
|
|
|
|
|
|
|
=cut |
|
61
|
|
|
|
|
|
|
|
|
62
|
|
|
|
|
|
|
# }}}1 |
|
63
|
|
|
|
|
|
|
# Helpers for the parser {{{1 |
|
64
|
|
|
|
|
|
|
|
|
65
|
|
|
|
|
|
|
# Merge any number of hashrefs together and return a hashref |
|
66
|
|
|
|
|
|
|
sub _merge { |
|
67
|
21
|
|
|
21
|
|
34
|
state $merger = Hash::Merge->new('RETAINMENT_PRECEDENT'); |
|
68
|
21
|
|
|
|
|
274
|
$merger->set_clone_behavior(false); # No cloning |
|
69
|
21
|
|
|
|
|
179
|
my $retval = {}; |
|
70
|
21
|
|
|
|
|
38
|
for(@_) { |
|
71
|
42
|
50
|
|
|
|
730
|
next unless ref eq 'HASH'; |
|
72
|
42
|
|
|
|
|
68
|
$retval = $merger->merge($retval, $_); |
|
73
|
|
|
|
|
|
|
} |
|
74
|
21
|
|
|
|
|
603
|
return $retval; |
|
75
|
|
|
|
|
|
|
} |
|
76
|
|
|
|
|
|
|
|
|
77
|
|
|
|
|
|
|
# Check for { expr => FOO } |
|
78
|
|
|
|
|
|
|
sub _is_single_expr { |
|
79
|
0
|
0
|
|
0
|
|
0
|
return false unless @_ eq 1; |
|
80
|
0
|
|
|
|
|
0
|
my $h = $_[0]; |
|
81
|
0
|
0
|
|
|
|
0
|
return false unless ref $h eq 'HASH'; |
|
82
|
0
|
0
|
|
|
|
0
|
return false unless keys(%$h) eq 1; |
|
83
|
0
|
0
|
|
|
|
0
|
return false unless (keys %$h)[0] eq 'expr'; |
|
84
|
0
|
|
|
|
|
0
|
return true; |
|
85
|
|
|
|
|
|
|
} |
|
86
|
|
|
|
|
|
|
|
|
87
|
|
|
|
|
|
|
# }}}1 |
|
88
|
|
|
|
|
|
|
|
|
89
|
|
|
|
|
|
|
%} |
|
90
|
|
|
|
|
|
|
|
|
91
|
|
|
|
|
|
|
############################################################################# |
|
92
|
20
|
|
|
20
|
0
|
91
|
# Token and precedence definitions |
|
93
|
20
|
50
|
|
|
|
39
|
|
|
94
|
|
|
|
|
|
|
%token SWITCH # flags controlling the operation of git-find |
|
95
|
|
|
|
|
|
|
%token REV # a git ref or rev |
|
96
|
|
|
|
|
|
|
|
|
97
|
|
|
|
|
|
|
# elements of expressions |
|
98
|
|
|
|
|
|
|
%token COMMA |
|
99
|
|
|
|
|
|
|
%token OR |
|
100
|
|
|
|
|
|
|
%token AND |
|
101
|
|
|
|
|
|
|
%token NOT |
|
102
|
|
|
|
|
|
|
%token LPAREN |
|
103
|
|
|
|
|
|
|
%token RPAREN |
|
104
|
|
|
|
|
|
|
|
|
105
|
|
|
|
|
|
|
%token TEST |
|
106
|
|
|
|
|
|
|
%token ACTION |
|
107
|
|
|
|
|
|
|
|
|
108
|
|
|
|
|
|
|
%left COMMA |
|
109
|
|
|
|
|
|
|
%left OR |
|
110
|
|
|
|
|
|
|
%left AND |
|
111
|
|
|
|
|
|
|
%left NOT |
|
112
|
|
|
|
|
|
|
|
|
113
|
|
|
|
|
|
|
%% |
|
114
|
|
|
|
|
|
|
|
|
115
|
|
|
|
|
|
|
############################################################################# |
|
116
|
|
|
|
|
|
|
# Rules |
|
117
|
|
|
|
|
|
|
|
|
118
|
|
|
|
|
|
|
cmdline: |
|
119
|
0
|
|
|
0
|
|
0
|
{ YYY +{} } # empty |
|
120
|
7
|
|
|
7
|
|
179
|
| maybeexprplus { YYY $_[1] } |
|
121
|
|
|
|
|
|
|
| switches_and_revs maybeexprplus |
|
122
|
13
|
|
|
13
|
|
362
|
{ YYY _merge($_[1], $_[2]) } |
|
123
|
|
|
|
|
|
|
; |
|
124
|
|
|
|
|
|
|
|
|
125
|
|
|
|
|
|
|
# Switches |
|
126
|
|
|
|
|
|
|
|
|
127
|
11
|
|
|
11
|
|
463
|
switches_and_revs: SWITCH { YYY +{ switches => {$_[1]=>[true]} } } |
|
128
|
|
|
|
|
|
|
# arrayref so @{{switches}->{foo}} will |
|
129
|
|
|
|
|
|
|
# always work. That way we can test for |
|
130
|
|
|
|
|
|
|
# switches specified multiple times. |
|
131
|
|
|
|
|
|
|
|
|
132
|
2
|
|
|
2
|
|
74
|
| REV { YYY +{ revs => [$_[1]] } } |
|
133
|
|
|
|
|
|
|
| switches_and_revs SWITCH |
|
134
|
0
|
|
|
0
|
|
0
|
{ YYY _merge($_[1], +{ switches => {$_[2]=>[true]} }) } |
|
135
|
|
|
|
|
|
|
| switches_and_revs REV |
|
136
|
8
|
|
|
8
|
|
262
|
{ YYY _merge($_[1], +{ revs => [$_[2]] }) } |
|
137
|
|
|
|
|
|
|
; |
|
138
|
|
|
|
|
|
|
|
|
139
|
|
|
|
|
|
|
# Expressions. Since "and", "or", and "not" are keywords in Perl, the |
|
140
|
|
|
|
|
|
|
# hash keys for our operators are all upper-case. |
|
141
|
|
|
|
|
|
|
maybeexprplus: |
|
142
|
7
|
|
|
7
|
|
106
|
{ YYY +{} } # empty |
|
143
|
13
|
|
|
13
|
|
254
|
| expr { YYY +{ expr => $_[1] } } |
|
144
|
|
|
|
|
|
|
#{ _is_single_expr($_[1]) ? $_[1] : +{ expr => $_[1] } } |
|
145
|
0
|
|
|
0
|
|
0
|
| expr switches_and_revs { YYY +{ expr => $_[1], %{$_[2]} } } |
|
|
0
|
|
|
|
|
0
|
|
|
146
|
|
|
|
|
|
|
# %$_[2]: inline the switches and revs |
|
147
|
|
|
|
|
|
|
; |
|
148
|
|
|
|
|
|
|
|
|
149
|
|
|
|
|
|
|
# TODO once the parser is solid and the tests are written: |
|
150
|
|
|
|
|
|
|
# collapse consecutive elements of the same type into flat lists. |
|
151
|
|
|
|
|
|
|
expr: element |
|
152
|
2
|
|
|
2
|
|
41
|
| expr COMMA expr { YYY +{ SEQ => [@_[1,3]] } } |
|
153
|
1
|
|
|
1
|
|
20
|
| expr OR expr { YYY +{ OR => [@_[1,3]] } } |
|
154
|
|
|
|
|
|
|
# "OR" because "or" is a Perl keyword |
|
155
|
|
|
|
|
|
|
|
|
156
|
|
|
|
|
|
|
# subsequent_expr is used so the parser will know when to reduce. |
|
157
|
|
|
|
|
|
|
# The rule "expr expr %prec AND" was right-associative |
|
158
|
|
|
|
|
|
|
# even though explicit-AND was left-associative because |
|
159
|
|
|
|
|
|
|
# the parser did not start reducing until after anything that |
|
160
|
|
|
|
|
|
|
# could be an expr. A subsequent_expr cannot start with |
|
161
|
|
|
|
|
|
|
# a plain expr, so the parser has a reason to reduce the expr. |
|
162
|
|
|
|
|
|
|
| expr subsequent_expr %prec AND |
|
163
|
4
|
|
|
4
|
|
260
|
{ YYY +{ AND => [@_[1,2]] } } |
|
164
|
0
|
|
|
0
|
|
0
|
| expr AND expr { YYY +{ AND => [@_[1,3]] } } |
|
165
|
0
|
|
|
0
|
|
0
|
| NOT expr4 { YYY +{ NOT => $_[2] } } |
|
166
|
0
|
|
|
0
|
|
0
|
| LPAREN expr RPAREN { YYY $_[2] } |
|
167
|
|
|
|
|
|
|
; |
|
168
|
|
|
|
|
|
|
|
|
169
|
|
|
|
|
|
|
subsequent_expr: # Everything higher-precedence than AND |
|
170
|
|
|
|
|
|
|
element |
|
171
|
0
|
|
|
0
|
|
0
|
| NOT expr4 { YYY +{ NOT => $_[2] } } |
|
172
|
0
|
|
|
0
|
|
0
|
| LPAREN expr RPAREN { YYY $_[2] } |
|
173
|
|
|
|
|
|
|
; |
|
174
|
|
|
|
|
|
|
|
|
175
|
|
|
|
|
|
|
|
|
176
|
|
|
|
|
|
|
element: TEST |
|
177
|
|
|
|
|
|
|
| ACTION |
|
178
|
|
|
|
|
|
|
{ |
|
179
|
6
|
50
|
|
6
|
|
185
|
$_[0]->YYData->{SAW_NON_PRUNE_ACTION} = true if $_[1] ne 'prune'; |
|
180
|
6
|
|
|
|
|
31
|
YYY $_[1]; |
|
181
|
|
|
|
|
|
|
} |
|
182
|
20
|
|
|
|
|
1225
|
; |
|
183
|
|
|
|
|
|
|
|
|
184
|
|
|
|
|
|
|
%% |
|
185
|
20
|
|
|
|
|
1105
|
|
|
186
|
|
|
|
|
|
|
############################################################################# |
|
187
|
|
|
|
|
|
|
# Footer |
|
188
|
|
|
|
|
|
|
|
|
189
|
|
|
|
|
|
|
# Helpers for the tokenizer {{{1 |
|
190
|
|
|
|
|
|
|
|
|
191
|
|
|
|
|
|
|
# Flag a ref as invalid without using regexes. |
|
192
|
|
|
|
|
|
|
# Implements https://git-scm.com/docs/git-check-ref-format as archived at |
|
193
|
|
|
|
|
|
|
# https://web.archive.org/web/20190725153529/https://git-scm.com/docs/git-check-ref-format |
|
194
|
|
|
|
|
|
|
|
|
195
|
|
|
|
|
|
|
sub _is_ref_ok { |
|
196
|
2676
|
50
|
|
2676
|
|
3610
|
my $arg = @_ ? $_[0] : $_; |
|
197
|
|
|
|
|
|
|
|
|
198
|
2676
|
50
|
33
|
|
|
5647
|
return false unless defined $arg and length($arg)>0; |
|
199
|
|
|
|
|
|
|
|
|
200
|
|
|
|
|
|
|
#1 - restrictions on slash-separated components |
|
201
|
2676
|
100
|
|
|
|
3526
|
if(index($arg, '/') != -1) { |
|
202
|
40
|
50
|
33
|
|
|
166
|
return false if index($arg, '/.') != -1 #internal components |
|
|
|
|
33
|
|
|
|
|
|
|
|
|
33
|
|
|
|
|
|
203
|
|
|
|
|
|
|
|| index($arg, '.lock/') != -1 |
|
204
|
|
|
|
|
|
|
|| substr($arg, 0, 1) eq '.' #components at start/end |
|
205
|
|
|
|
|
|
|
|| substr($arg, -5) eq '.lock'; |
|
206
|
|
|
|
|
|
|
} |
|
207
|
|
|
|
|
|
|
|
|
208
|
|
|
|
|
|
|
# Ignore #2 - assume --allow-onelevel |
|
209
|
|
|
|
|
|
|
|
|
210
|
|
|
|
|
|
|
#3 |
|
211
|
2676
|
50
|
|
|
|
3435
|
return false if index($arg, '..') != -1; |
|
212
|
|
|
|
|
|
|
|
|
213
|
|
|
|
|
|
|
#4 - require the caller to check that |
|
214
|
|
|
|
|
|
|
#5 - require the caller to check that - assume NOT --refspec-pattern |
|
215
|
|
|
|
|
|
|
|
|
216
|
|
|
|
|
|
|
#6 - assume NOT --normalize |
|
217
|
2676
|
100
|
66
|
|
|
8609
|
return false if substr($arg, 0, 1) eq '/' |
|
|
|
|
66
|
|
|
|
|
|
218
|
|
|
|
|
|
|
|| substr($arg, -1) eq '/' |
|
219
|
|
|
|
|
|
|
|| index($arg, '//') != -1; |
|
220
|
|
|
|
|
|
|
|
|
221
|
|
|
|
|
|
|
# #7. Also prohibits ".", which is OK for git-find since it is |
|
222
|
|
|
|
|
|
|
# fairly ambiguous between a ref/rev and a path. |
|
223
|
2670
|
100
|
|
|
|
3358
|
return false if substr($arg, -1) eq '.'; |
|
224
|
|
|
|
|
|
|
|
|
225
|
|
|
|
|
|
|
#8 |
|
226
|
2649
|
50
|
|
|
|
3077
|
return false if index($arg, '@{') != -1; |
|
227
|
|
|
|
|
|
|
|
|
228
|
|
|
|
|
|
|
#9 ('@') - ignore this one for simplicity in the rev test below. |
|
229
|
|
|
|
|
|
|
|
|
230
|
|
|
|
|
|
|
#10 - require the caller to check that |
|
231
|
|
|
|
|
|
|
|
|
232
|
|
|
|
|
|
|
# Extra: Prohibit refs that start with '--' since they are arguably |
|
233
|
|
|
|
|
|
|
# ambiguous with command-line options (and I can't make them work |
|
234
|
|
|
|
|
|
|
# with git anyway). |
|
235
|
2649
|
100
|
|
|
|
5683
|
return false if substr($arg, 0, 2) eq '--'; |
|
236
|
|
|
|
|
|
|
|
|
237
|
1194
|
|
|
|
|
3635
|
return true; # It's OK if we got here |
|
238
|
|
|
|
|
|
|
} #_is_ref_ok() |
|
239
|
|
|
|
|
|
|
|
|
240
|
|
|
|
|
|
|
#use re 'debug'; |
|
241
|
|
|
|
|
|
|
|
|
242
|
|
|
|
|
|
|
# Regex to match a rev or range of revs, i.e., something we should pass to git |
|
243
|
|
|
|
|
|
|
my $_rev_regex = |
|
244
|
|
|
|
|
|
|
qr`(?xi) # backtick delimiter because it doesn't occur in the regex text |
|
245
|
|
|
|
|
|
|
(?&RevRange) |
|
246
|
|
|
|
|
|
|
|
|
247
|
|
|
|
|
|
|
(?(DEFINE) |
|
248
|
|
|
|
|
|
|
|
|
249
|
|
|
|
|
|
|
(?<RevRange> ^(?: |
|
250
|
|
|
|
|
|
|
# :/text, :/!-text, :/!!text |
|
251
|
|
|
|
|
|
|
(?::/ #(?{ print "# saw colon slash\n"; }) |
|
252
|
|
|
|
|
|
|
(?: |
|
253
|
|
|
|
|
|
|
 #(?{print "# 4\n";}) |
|
254
|
|
|
|
|
|
|
| [^!].* #(?{print "# 5\u";}) |
|
255
|
|
|
|
|
|
|
) |
|
256
|
|
|
|
|
|
|
) |
|
257
|
|
|
|
|
|
|
|
|
258
|
|
|
|
|
|
|
# :[n:]path. NOTE: we prohibit starting the path with |
|
259
|
|
|
|
|
|
|
# / if there is no number, in order to disambiguate |
|
260
|
|
|
|
|
|
|
# the :/ text-search cases. |
|
261
|
|
|
|
|
|
|
| :\d+:(?:.+) #(?{print "# 2\n";}) |
|
262
|
|
|
|
|
|
|
| :[^/].* #(?{print "# 3\n";}) |
|
263
|
|
|
|
|
|
|
|
|
264
|
|
|
|
|
|
|
# ^<rev> |
|
265
|
|
|
|
|
|
|
| \^(?&Rev) #(?{print "# 6\n";}) |
|
266
|
|
|
|
|
|
|
|
|
267
|
|
|
|
|
|
|
# rev:path |
|
268
|
|
|
|
|
|
|
| (?&Rev):(?:.+) #(?{print "# 7\n";}) |
|
269
|
|
|
|
|
|
|
|
|
270
|
|
|
|
|
|
|
# .. and ... differences, including x.., x..., x..y, |
|
271
|
|
|
|
|
|
|
# and x...y. Also handles the fallthrough |
|
272
|
|
|
|
|
|
|
# of revrange->rev->ref. |
|
273
|
|
|
|
|
|
|
| (?&Rev)(?:\.{2,3}(?&Rev)?)? |
|
274
|
|
|
|
|
|
|
#(?{print "# 8\n";}) |
|
275
|
|
|
|
|
|
|
|
|
276
|
|
|
|
|
|
|
# ..rev and ...rev |
|
277
|
|
|
|
|
|
|
| \.{2,3}(?&Rev) |
|
278
|
|
|
|
|
|
|
|
|
279
|
|
|
|
|
|
|
# at sign followed by braced item, and possibly |
|
280
|
|
|
|
|
|
|
# preceded by a REF (not a rev). E.g., |
|
281
|
|
|
|
|
|
|
# HEAD@{1}@{1} doesn't work. |
|
282
|
|
|
|
|
|
|
# refname - at sign - braced item (date, #, branch, "push") |
|
283
|
|
|
|
|
|
|
| (?&Ref)?\@\{[^\}]+\} |
|
284
|
|
|
|
|
|
|
#(?{print "# 9\n";}) |
|
285
|
|
|
|
|
|
|
|
|
286
|
|
|
|
|
|
|
# git-rev-parse "Options for Objects" forms |
|
287
|
|
|
|
|
|
|
| --all |
|
288
|
|
|
|
|
|
|
| --(?:branches|tags|remotes)(?:=.+)? |
|
289
|
|
|
|
|
|
|
| --(?:glob|exclude)=.+ |
|
290
|
|
|
|
|
|
|
| --disambiguate=[0-9a-f]{4,40} |
|
291
|
|
|
|
|
|
|
|
|
292
|
|
|
|
|
|
|
# git-rev-parse "Other Options" forms |
|
293
|
|
|
|
|
|
|
| --(since|after|until|before)=.+ |
|
294
|
|
|
|
|
|
|
|
|
295
|
|
|
|
|
|
|
)$) # End of RevRange |
|
296
|
|
|
|
|
|
|
|
|
297
|
|
|
|
|
|
|
(?<Rev> (?&Ref)(?&RefTrailer)* ) |
|
298
|
|
|
|
|
|
|
# This handles most of the cases. |
|
299
|
|
|
|
|
|
|
# SHA1s, possibly abbreviated, are refs, |
|
300
|
|
|
|
|
|
|
# as are git-describe outputs, whence RefTrailer* |
|
301
|
|
|
|
|
|
|
# instead of RefTrailer+. |
|
302
|
|
|
|
|
|
|
|
|
303
|
|
|
|
|
|
|
(?<RefTrailer> |
|
304
|
|
|
|
|
|
|
# For rev^[#] and rev~[#] forms |
|
305
|
|
|
|
|
|
|
[~\^]\d* |
|
306
|
|
|
|
|
|
|
|
|
307
|
|
|
|
|
|
|
# For rev^{} forms (empty braces OK) |
|
308
|
|
|
|
|
|
|
| \^\{[^\}]*\} |
|
309
|
|
|
|
|
|
|
|
|
310
|
|
|
|
|
|
|
# For rev^[@!] and rev^-n |
|
311
|
|
|
|
|
|
|
| \^(?: \@ | ! | -\d* ) |
|
312
|
|
|
|
|
|
|
) # End of RefTrailer |
|
313
|
|
|
|
|
|
|
|
|
314
|
|
|
|
|
|
|
(?<Ref> |
|
315
|
|
|
|
|
|
|
( \@ # '@' from git-rev-parse |
|
316
|
|
|
|
|
|
|
| (?:[^\000-\037\177\ ~\^:\\?*\[.@/] |
|
317
|
|
|
|
|
|
|
# git-check-ref-format #4, #5. |
|
318
|
|
|
|
|
|
|
# [.@/] are handled below |
|
319
|
|
|
|
|
|
|
| \.(?!\.) # . ok, but .. prohibited |
|
320
|
|
|
|
|
|
|
| \@(?!\{) # @ ok, but @{ prohibited |
|
321
|
|
|
|
|
|
|
| /(?!/) # / ok, but // prohibited |
|
322
|
|
|
|
|
|
|
|
|
323
|
|
|
|
|
|
|
)+? |
|
324
|
|
|
|
|
|
|
) |
|
325
|
|
|
|
|
|
|
(?(?{ _is_ref_ok($+) })|(?!)) |
|
326
|
|
|
|
|
|
|
# NOTE: $+ used since I couldn't get named capture groups |
|
327
|
|
|
|
|
|
|
# with either %+ or %- to work |
|
328
|
|
|
|
|
|
|
) # End of <Ref> |
|
329
|
|
|
|
|
|
|
|
|
330
|
|
|
|
|
|
|
) #End of (DEFINE) |
|
331
|
|
|
|
|
|
|
|
|
332
|
|
|
|
|
|
|
`xi; # End of qr`...` and an extra backtick to unconfuse vim-eyapp: ` |
|
333
|
|
|
|
|
|
|
|
|
334
|
|
|
|
|
|
|
sub _is_valid_rev { |
|
335
|
149
|
50
|
|
149
|
|
88726
|
my $arg = @_ ? $_[0] : $_; |
|
336
|
|
|
|
|
|
|
|
|
337
|
149
|
100
|
100
|
|
|
855
|
return false unless defined $arg and length($arg)>0; |
|
338
|
147
|
|
|
|
|
3509
|
return scalar($arg =~ m{$_rev_regex}); |
|
339
|
|
|
|
|
|
|
} #_is_valid_rev() |
|
340
|
|
|
|
|
|
|
|
|
341
|
|
|
|
|
|
|
# Get an expression element from the array passed in $_[0]. |
|
342
|
|
|
|
|
|
|
my $ARGTEST_cached = App::GitFind::Actions::ARGTEST(); |
|
343
|
|
|
|
|
|
|
sub _consume_expression_element { |
|
344
|
2710
|
|
|
34
|
|
3271
|
my $lrArgv = shift; |
|
345
|
34
|
|
|
|
|
33
|
my @retval; |
|
346
|
|
|
|
|
|
|
|
|
347
|
|
|
|
|
|
|
#say STDERR "# Trying >>$lrArgv->[0]<<"; |
|
348
|
|
|
|
|
|
|
# TODO find(1) positional options, global options? |
|
349
|
|
|
|
|
|
|
|
|
350
|
|
|
|
|
|
|
# Regular options |
|
351
|
34
|
100
|
|
|
|
188
|
if($lrArgv->[0] =~ $ARGTEST_cached) { |
|
352
|
|
|
|
|
|
|
#say STDERR "# - matched"; |
|
353
|
20
|
|
|
|
|
44
|
my $arg = $1; |
|
354
|
20
|
|
|
|
|
15
|
my %opts = %{App::GitFind::Actions::argdetails($arg)}; |
|
|
20
|
|
|
|
|
35
|
|
|
355
|
|
|
|
|
|
|
|
|
356
|
|
|
|
|
|
|
# Save any non-parser information from the argdetails to be |
|
357
|
|
|
|
|
|
|
# returned as part of the semantic value. |
|
358
|
20
|
|
|
|
|
47
|
my %extras = %opts; |
|
359
|
20
|
|
|
|
|
319
|
delete @extras{qw(token nparam)}; |
|
360
|
|
|
|
|
|
|
|
|
361
|
|
|
|
|
|
|
# No-argument tests or actions |
|
362
|
20
|
50
|
|
|
|
49
|
unless($opts{nparam}>0) { |
|
363
|
|
|
|
|
|
|
#say STDERR "# - No parameters"; |
|
364
|
20
|
|
|
|
|
19
|
shift @$lrArgv; |
|
365
|
20
|
|
|
|
|
85
|
return ($opts{token} => { name => $arg, %extras }) |
|
366
|
|
|
|
|
|
|
} |
|
367
|
|
|
|
|
|
|
|
|
368
|
|
|
|
|
|
|
# Consume additional arguments up to a regexp |
|
369
|
0
|
0
|
|
|
|
0
|
if(ref $opts{nparam} eq 'Regexp') { |
|
370
|
|
|
|
|
|
|
#say STDERR "# - parameters until $opts{nparam}"; |
|
371
|
0
|
0
|
|
|
|
0
|
die "Need argument(s) for --$arg" if @$lrArgv == 1; |
|
372
|
0
|
|
|
|
|
0
|
my $lastarg; |
|
373
|
|
|
|
|
|
|
#say STDERR "Args: ", join ' : ', @$lrArgv; |
|
374
|
0
|
|
|
|
|
0
|
for(1..$#$lrArgv) { |
|
375
|
0
|
0
|
|
|
|
0
|
$lastarg=$_, last if $lrArgv->[$_] =~ $opts{nparam}; |
|
376
|
|
|
|
|
|
|
} |
|
377
|
0
|
0
|
|
|
|
0
|
die "--$arg needs an argument terminator matching $opts{nparam}" |
|
378
|
|
|
|
|
|
|
unless defined $lastarg; |
|
379
|
|
|
|
|
|
|
|
|
380
|
|
|
|
|
|
|
# Set up to fall through to the numeric-params case |
|
381
|
0
|
|
|
|
|
0
|
$opts{nparam} = $lastarg; |
|
382
|
|
|
|
|
|
|
} |
|
383
|
|
|
|
|
|
|
|
|
384
|
|
|
|
|
|
|
# Consume additional positional arguments |
|
385
|
|
|
|
|
|
|
#say STDERR "# - $opts{nparam} parameters"; |
|
386
|
|
|
|
|
|
|
die "Not enough parameters after --$arg (need $opts{nparam})" |
|
387
|
0
|
0
|
|
|
|
0
|
unless @$lrArgv >= ($opts{nparam}+1); # +1 for $arg itself |
|
388
|
|
|
|
|
|
|
|
|
389
|
|
|
|
|
|
|
# Custom argument validation |
|
390
|
0
|
0
|
|
|
|
0
|
if($opts{validator}) { |
|
391
|
0
|
|
|
|
|
0
|
my $errmsg = $opts{validator}->(@{$lrArgv}[0..$opts{nparam}]); |
|
|
0
|
|
|
|
|
0
|
|
|
392
|
0
|
0
|
|
|
|
0
|
die "--$arg argument error: $errmsg" if defined($errmsg); |
|
393
|
|
|
|
|
|
|
} |
|
394
|
|
|
|
|
|
|
|
|
395
|
|
|
|
|
|
|
@retval = ($opts{token} => { |
|
396
|
|
|
|
|
|
|
name => $arg, |
|
397
|
0
|
|
|
|
|
0
|
params => [ @{$lrArgv}[1..$opts{nparam}] ], |
|
|
0
|
|
|
|
|
0
|
|
|
398
|
|
|
|
|
|
|
%extras, |
|
399
|
|
|
|
|
|
|
}); |
|
400
|
0
|
|
|
|
|
0
|
splice @$lrArgv, 0, $opts{nparam}+1; |
|
401
|
0
|
|
|
|
|
0
|
return @retval; |
|
402
|
|
|
|
|
|
|
} |
|
403
|
|
|
|
|
|
|
|
|
404
|
|
|
|
|
|
|
# Operators |
|
405
|
14
|
|
|
|
|
19
|
my $arg = $lrArgv->[0]; |
|
406
|
|
|
|
|
|
|
|
|
407
|
14
|
100
|
|
|
|
31
|
@retval = (COMMA => ',') if $arg eq ','; |
|
408
|
14
|
100
|
|
|
|
45
|
@retval = (OR => '-o') if $arg =~ /^(?:-o|--o|-or|--or|\|\|)$/; |
|
409
|
14
|
50
|
|
|
|
25
|
@retval = (AND => '-a') if $arg =~ /^(?:-a|--a|-and|--and|&&)$/; |
|
410
|
14
|
50
|
|
|
|
32
|
@retval = (NOT => '!') if $arg =~ /^(?:-not|--not|!|\^)$/; |
|
411
|
14
|
50
|
|
|
|
23
|
@retval = (LPAREN => '(') if $arg =~ /^[([]$/; |
|
412
|
14
|
50
|
|
|
|
23
|
@retval = (RPAREN => ')') if $arg =~ /^[])]$/; |
|
413
|
|
|
|
|
|
|
|
|
414
|
14
|
100
|
|
|
|
24
|
if(@retval) { |
|
415
|
3
|
|
|
|
|
4
|
shift @$lrArgv; |
|
416
|
3
|
|
|
|
|
6
|
return @retval; |
|
417
|
|
|
|
|
|
|
} |
|
418
|
|
|
|
|
|
|
|
|
419
|
11
|
|
|
|
|
16
|
return (); # Not an expression element |
|
420
|
|
|
|
|
|
|
} #_consume_expression_element |
|
421
|
|
|
|
|
|
|
|
|
422
|
|
|
|
|
|
|
# Get a switch from the array passed in $_[0], if any. |
|
423
|
|
|
|
|
|
|
# Removes the switch from the array if successful. |
|
424
|
|
|
|
|
|
|
# Returns the token on success, and () on failure. |
|
425
|
|
|
|
|
|
|
# TODO un-bundle switches; handle switches with args. |
|
426
|
|
|
|
|
|
|
sub _consume_switch { |
|
427
|
21
|
|
|
21
|
|
20
|
my $lrArgv = shift; |
|
428
|
21
|
100
|
|
|
|
64
|
if($lrArgv->[0] =~ /^-([a-zA-z0-9\?])$/) { # non-bundled switch |
|
|
|
50
|
|
|
|
|
|
|
429
|
11
|
|
|
|
|
16
|
shift @$lrArgv; |
|
430
|
11
|
|
|
|
|
34
|
return (SWITCH => $1) |
|
431
|
|
|
|
|
|
|
} elsif($lrArgv->[0] =~ /^--?(help|man|usage|version)$/) { # long switch |
|
432
|
0
|
|
|
|
|
0
|
shift @$lrArgv; |
|
433
|
0
|
|
|
|
|
0
|
return (SWITCH => $1); |
|
434
|
|
|
|
|
|
|
} |
|
435
|
|
|
|
|
|
|
|
|
436
|
10
|
|
|
|
|
12
|
return (); |
|
437
|
|
|
|
|
|
|
} #_consume_switch() |
|
438
|
|
|
|
|
|
|
|
|
439
|
|
|
|
|
|
|
# Consume a rev from the array in $_[0] |
|
440
|
|
|
|
|
|
|
sub _consume_rev { |
|
441
|
10
|
|
|
10
|
|
9
|
my $lrArgv = shift; |
|
442
|
10
|
|
|
|
|
11
|
my $arg = $lrArgv->[0]; |
|
443
|
10
|
50
|
|
|
|
15
|
if(_is_valid_rev($arg)) { |
|
444
|
10
|
|
|
|
|
17
|
shift @$lrArgv; |
|
445
|
10
|
|
|
|
|
21
|
return (REV => $arg); |
|
446
|
|
|
|
|
|
|
} |
|
447
|
|
|
|
|
|
|
|
|
448
|
0
|
|
|
|
|
0
|
return (); |
|
449
|
|
|
|
|
|
|
} #_consume_rev() |
|
450
|
|
|
|
|
|
|
|
|
451
|
|
|
|
|
|
|
# }}}1 |
|
452
|
|
|
|
|
|
|
# Tokenizer and error-reporting routine for Parse::Yapp {{{1 |
|
453
|
|
|
|
|
|
|
|
|
454
|
|
|
|
|
|
|
# The lexer |
|
455
|
|
|
|
|
|
|
sub _next_token { |
|
456
|
64
|
|
|
64
|
|
2720
|
my $parser = shift; |
|
457
|
64
|
|
|
|
|
94
|
my $lrArgv = $parser->YYData->{ARGV}; |
|
458
|
64
|
100
|
|
|
|
492
|
return ('', undef) unless @$lrArgv; # EOF |
|
459
|
47
|
|
|
|
|
49
|
my @retval; # The eventual token we will return |
|
460
|
|
|
|
|
|
|
|
|
461
|
|
|
|
|
|
|
# TODO? in the expression, split trailing commas into their |
|
462
|
|
|
|
|
|
|
# own arguments |
|
463
|
|
|
|
|
|
|
|
|
464
|
|
|
|
|
|
|
# Check for '--' |
|
465
|
47
|
100
|
|
|
|
70
|
if($lrArgv->[0] eq '--') { |
|
466
|
8
|
|
|
|
|
14
|
$parser->YYData->{ONLY_EXPRESSIONS} = true; |
|
467
|
8
|
100
|
|
|
|
44
|
return ('', undef) unless @$lrArgv > 1; |
|
468
|
|
|
|
|
|
|
# We are about to shift, so return EOF if this was the last arg. |
|
469
|
5
|
|
|
|
|
7
|
shift(@$lrArgv); |
|
470
|
|
|
|
|
|
|
} |
|
471
|
|
|
|
|
|
|
|
|
472
|
44
|
100
|
|
|
|
58
|
if($parser->YYData->{HAS_DASH_DASH}) { |
|
473
|
|
|
|
|
|
|
# Split-arg mode: don't look for expressions before '--', or |
|
474
|
|
|
|
|
|
|
# for switches or refs after '--'. |
|
475
|
16
|
100
|
|
|
|
101
|
if(!$parser->YYData->{ONLY_EXPRESSIONS}) { # Look for switches/refs |
|
476
|
|
|
|
|
|
|
|
|
477
|
10
|
|
|
|
|
48
|
@retval = _consume_switch($lrArgv); |
|
478
|
10
|
100
|
|
|
|
29
|
return @retval if @retval; |
|
479
|
|
|
|
|
|
|
|
|
480
|
5
|
|
|
|
|
17
|
@retval = _consume_rev($lrArgv); |
|
481
|
5
|
50
|
|
|
|
8
|
if(@retval) { # _consume_rev always gives us two elements |
|
482
|
5
|
50
|
|
|
|
11
|
if($retval[1] eq ']]') { |
|
483
|
0
|
|
0
|
|
|
0
|
$parser->YYData->{SAW_RR} ||= true; |
|
484
|
|
|
|
|
|
|
} else { |
|
485
|
5
|
|
50
|
|
|
8
|
$parser->YYData->{SAW_NON_RR} ||= true; |
|
486
|
|
|
|
|
|
|
} |
|
487
|
5
|
|
|
|
|
42
|
return @retval; |
|
488
|
|
|
|
|
|
|
} |
|
489
|
|
|
|
|
|
|
|
|
490
|
0
|
|
|
|
|
0
|
die "I don't understand argument '$lrArgv->[0]' before --"; |
|
491
|
|
|
|
|
|
|
|
|
492
|
|
|
|
|
|
|
} else { # Look for expressions |
|
493
|
6
|
|
|
|
|
29
|
@retval = _consume_expression_element($lrArgv); |
|
494
|
6
|
50
|
|
|
|
21
|
return @retval if @retval; |
|
495
|
0
|
|
|
|
|
0
|
die "I don't understand argument '$lrArgv->[0]' after --"; |
|
496
|
|
|
|
|
|
|
} |
|
497
|
|
|
|
|
|
|
|
|
498
|
|
|
|
|
|
|
} else { |
|
499
|
|
|
|
|
|
|
# Merged-arg mode: any arg could be anything |
|
500
|
|
|
|
|
|
|
|
|
501
|
|
|
|
|
|
|
# Check for expressions. Look for these before checking for refs so |
|
502
|
|
|
|
|
|
|
# that an expression that happens to look like a ref will be considered |
|
503
|
|
|
|
|
|
|
# an expression instead of a ref. |
|
504
|
28
|
|
|
|
|
139
|
my @retval = _consume_expression_element($lrArgv); |
|
505
|
28
|
100
|
|
|
|
72
|
return @retval if @retval; |
|
506
|
|
|
|
|
|
|
|
|
507
|
|
|
|
|
|
|
# Next, look for switches. These are after expression elements |
|
508
|
|
|
|
|
|
|
# so that -a and -o will not be parsed as switches. |
|
509
|
11
|
|
|
|
|
21
|
@retval = _consume_switch($lrArgv); |
|
510
|
11
|
100
|
|
|
|
33
|
return @retval if @retval; |
|
511
|
|
|
|
|
|
|
|
|
512
|
|
|
|
|
|
|
# Last of all, revs. |
|
513
|
5
|
|
|
|
|
7
|
@retval = _consume_rev($lrArgv); |
|
514
|
5
|
50
|
|
|
|
11
|
if(@retval) { # _consume_rev always gives us two elements |
|
515
|
5
|
50
|
|
|
|
8
|
if($retval[1] eq ']]') { |
|
516
|
0
|
|
0
|
|
|
0
|
$parser->YYData->{SAW_RR} ||= true; |
|
517
|
|
|
|
|
|
|
} else { |
|
518
|
5
|
|
50
|
|
|
12
|
$parser->YYData->{SAW_NON_RR} ||= true; |
|
519
|
|
|
|
|
|
|
} |
|
520
|
5
|
|
|
|
|
41
|
return @retval; |
|
521
|
|
|
|
|
|
|
} |
|
522
|
|
|
|
|
|
|
|
|
523
|
0
|
|
|
|
|
0
|
die "I don't understand argument $lrArgv->[0]"; |
|
524
|
|
|
|
|
|
|
} |
|
525
|
|
|
|
|
|
|
|
|
526
|
0
|
|
|
|
|
0
|
die "Unexpected error while processing argument $lrArgv->[0]"; # Shouldn't happen |
|
527
|
|
|
|
|
|
|
} #_next_token() |
|
528
|
|
|
|
|
|
|
|
|
529
|
|
|
|
|
|
|
# Report an error |
|
530
|
|
|
|
|
|
|
sub _report_error { |
|
531
|
0
|
|
|
0
|
|
0
|
my $parser = shift; |
|
532
|
0
|
|
0
|
|
|
0
|
my $got = $parser->YYCurtok || '<end of input>'; |
|
533
|
0
|
|
|
|
|
0
|
my $val=''; |
|
534
|
0
|
0
|
|
|
|
0
|
$val = ' (' . $parser->YYCurval . ')' if $parser->YYCurval; |
|
535
|
0
|
|
|
|
|
0
|
die 'Syntax error: could not understand ', $got, $val, "\n"; |
|
536
|
0
|
0
|
|
|
|
0
|
if(ref($parser->YYExpect) eq 'ARRAY') { |
|
537
|
0
|
|
|
|
|
0
|
print 'Expected one of: ', join(',', @{$parser->YYExpect}), "\n"; |
|
|
0
|
|
|
|
|
0
|
|
|
538
|
|
|
|
|
|
|
} |
|
539
|
0
|
|
|
|
|
0
|
return; |
|
540
|
|
|
|
|
|
|
} #_report_error() |
|
541
|
|
|
|
|
|
|
|
|
542
|
|
|
|
|
|
|
# }}}1 |
|
543
|
|
|
|
|
|
|
# Top-level parse function {{{1 |
|
544
|
|
|
|
|
|
|
|
|
545
|
|
|
|
|
|
|
=head2 Parse |
|
546
|
|
|
|
|
|
|
|
|
547
|
|
|
|
|
|
|
Parse arguments. Usage: |
|
548
|
|
|
|
|
|
|
|
|
549
|
|
|
|
|
|
|
my $hrArgs = App::GitFind::cmdline::Parse(\@ARGV); |
|
550
|
|
|
|
|
|
|
|
|
551
|
|
|
|
|
|
|
Modifies the C<@ARGV> array. |
|
552
|
|
|
|
|
|
|
|
|
553
|
|
|
|
|
|
|
=cut |
|
554
|
|
|
|
|
|
|
|
|
555
|
|
|
|
|
|
|
sub Parse { |
|
556
|
20
|
50
|
|
20
|
1
|
94621
|
my $lrArgv = shift or |
|
557
|
|
|
|
|
|
|
(require Carp, Carp::croak 'Parse: Need an argument list'); |
|
558
|
|
|
|
|
|
|
|
|
559
|
20
|
|
|
|
|
58
|
my $parser = __PACKAGE__->new; |
|
560
|
20
|
|
|
|
|
43
|
my $hrData = $parser->YYData; |
|
561
|
|
|
|
|
|
|
|
|
562
|
|
|
|
|
|
|
# Data we use while parsing |
|
563
|
20
|
|
|
|
|
116
|
$hrData->{HAS_DASH_DASH} = !!(scalar grep { $_ eq '--' } @$lrArgv); |
|
|
52
|
|
|
|
|
107
|
|
|
564
|
20
|
|
|
|
|
30
|
$hrData->{ONLY_EXPRESSIONS} = false; # true once we hit '--' |
|
565
|
20
|
|
|
|
|
23
|
$hrData->{ARGV} = $lrArgv; |
|
566
|
|
|
|
|
|
|
|
|
567
|
|
|
|
|
|
|
# Data we determine while parsing and return to the caller |
|
568
|
|
|
|
|
|
|
|
|
569
|
|
|
|
|
|
|
# Keep track of whether an action other than -prune has been seen. |
|
570
|
|
|
|
|
|
|
# If not, -print is added automatically. |
|
571
|
20
|
|
|
|
|
28
|
$hrData->{SAW_NON_PRUNE_ACTION} = false; |
|
572
|
|
|
|
|
|
|
|
|
573
|
|
|
|
|
|
|
# Keep track of the types of rev we've seen (]] or non-]]) |
|
574
|
20
|
|
|
|
|
29
|
$hrData->{SAW_RR} = false; |
|
575
|
20
|
|
|
|
|
22
|
$hrData->{SAW_NON_RR} = false; |
|
576
|
|
|
|
|
|
|
|
|
577
|
20
|
50
|
|
|
|
73
|
my $hrRetval = $parser->YYParse(yylex => \&_next_token, |
|
578
|
|
|
|
|
|
|
yyerror => \&_report_error, |
|
579
|
|
|
|
|
|
|
(@_ ? (yydebug => $_[0]) : ()), |
|
580
|
|
|
|
|
|
|
); |
|
581
|
|
|
|
|
|
|
|
|
582
|
|
|
|
|
|
|
# Add non-AST data to the retval |
|
583
|
20
|
50
|
|
|
|
1313
|
$hrRetval->{saw_nonprune_action} = $hrData->{SAW_NON_PRUNE_ACTION} if $hrRetval; |
|
584
|
20
|
|
|
|
|
42
|
$hrRetval->{saw_rr} = $hrData->{SAW_RR}; |
|
585
|
20
|
|
|
|
|
35
|
$hrRetval->{saw_non_rr} = $hrData->{SAW_NON_RR}; |
|
586
|
20
|
|
|
|
|
849
|
return $hrRetval; |
|
587
|
|
|
|
|
|
|
} #Parse() |
|
588
|
|
|
|
|
|
|
|
|
589
|
|
|
|
|
|
|
# }}}1 |
|
590
|
|
|
|
|
|
|
# Rest of the docs {{{1 |
|
591
|
|
|
|
|
|
|
|
|
592
|
|
|
|
|
|
|
=head1 AUTHOR |
|
593
|
|
|
|
|
|
|
|
|
594
|
|
|
|
|
|
|
Christopher White C<< <cxw@cpan.org> >> |
|
595
|
|
|
|
|
|
|
|
|
596
|
|
|
|
|
|
|
=head1 COPYRIGHT |
|
597
|
|
|
|
|
|
|
|
|
598
|
|
|
|
|
|
|
MIT |
|
599
|
|
|
|
|
|
|
|
|
600
|
|
|
|
|
|
|
=cut |
|
601
|
|
|
|
|
|
|
|
|
602
|
|
|
|
|
|
|
# }}}1 |
|
603
|
|
|
|
|
|
|
|
|
604
|
|
|
|
|
|
|
# vi: set fdm=marker: # |