| line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
|
1
|
|
|
|
|
|
|
#! perl |
|
2
|
|
|
|
|
|
|
|
|
3
|
|
|
|
|
|
|
package ChordPro::Utils; |
|
4
|
|
|
|
|
|
|
|
|
5
|
79
|
|
|
79
|
|
981
|
use v5.26; |
|
|
79
|
|
|
|
|
342
|
|
|
6
|
79
|
|
|
79
|
|
1443
|
use utf8; |
|
|
79
|
|
|
|
|
280
|
|
|
|
79
|
|
|
|
|
454
|
|
|
7
|
79
|
|
|
79
|
|
1804
|
use Carp; |
|
|
79
|
|
|
|
|
188
|
|
|
|
79
|
|
|
|
|
4691
|
|
|
8
|
79
|
|
|
79
|
|
594
|
use feature qw( signatures ); |
|
|
79
|
|
|
|
|
264
|
|
|
|
79
|
|
|
|
|
7422
|
|
|
9
|
79
|
|
|
79
|
|
657
|
no warnings "experimental::signatures"; |
|
|
79
|
|
|
|
|
285
|
|
|
|
79
|
|
|
|
|
4096
|
|
|
10
|
79
|
|
|
79
|
|
574
|
use parent qw(Exporter); |
|
|
79
|
|
|
|
|
282
|
|
|
|
79
|
|
|
|
|
2236
|
|
|
11
|
|
|
|
|
|
|
|
|
12
|
|
|
|
|
|
|
our @EXPORT; |
|
13
|
|
|
|
|
|
|
|
|
14
|
|
|
|
|
|
|
################ Platforms ################ |
|
15
|
|
|
|
|
|
|
|
|
16
|
79
|
50
|
|
79
|
|
11935
|
use constant MSWIN => $^O =~ /MSWin|Windows_NT/i ? 1 : 0; |
|
|
79
|
|
|
|
|
226
|
|
|
|
79
|
|
|
|
|
19957
|
|
|
17
|
|
|
|
|
|
|
|
|
18
|
0
|
|
|
0
|
0
|
0
|
sub is_msw () { MSWIN } |
|
|
0
|
|
|
|
|
0
|
|
|
|
0
|
|
|
|
|
0
|
|
|
19
|
0
|
|
|
0
|
0
|
0
|
sub is_macos () { $^O =~ /darwin/ } |
|
|
0
|
|
|
|
|
0
|
|
|
|
0
|
|
|
|
|
0
|
|
|
20
|
|
|
|
|
|
|
|
|
21
|
|
|
|
|
|
|
push( @EXPORT, 'is_msw', 'is_macos' ); |
|
22
|
|
|
|
|
|
|
|
|
23
|
|
|
|
|
|
|
################ Filenames ################ |
|
24
|
|
|
|
|
|
|
|
|
25
|
79
|
|
|
79
|
|
681
|
use File::Glob ( ":bsd_glob" ); |
|
|
79
|
|
|
|
|
259
|
|
|
|
79
|
|
|
|
|
20406
|
|
|
26
|
79
|
|
|
79
|
|
663
|
use File::Spec; |
|
|
79
|
|
|
|
|
205
|
|
|
|
79
|
|
|
|
|
9287
|
|
|
27
|
|
|
|
|
|
|
|
|
28
|
|
|
|
|
|
|
# Derived from Path::ExpandTilde. |
|
29
|
|
|
|
|
|
|
|
|
30
|
79
|
50
|
|
|
|
7048
|
use constant BSD_GLOB_FLAGS => GLOB_NOCHECK | GLOB_QUOTE | GLOB_TILDE | GLOB_ERR |
|
31
|
|
|
|
|
|
|
# add GLOB_NOCASE as in File::Glob |
|
32
|
79
|
|
|
79
|
|
610
|
| ($^O =~ m/\A(?:MSWin32|VMS|os2|dos|riscos)\z/ ? GLOB_NOCASE : 0); |
|
|
79
|
|
|
|
|
221
|
|
|
33
|
|
|
|
|
|
|
|
|
34
|
|
|
|
|
|
|
# File::Glob did not try %USERPROFILE% (set in Windows NT derivatives) for ~ before 5.16 |
|
35
|
79
|
|
|
79
|
|
641
|
use constant WINDOWS_USERPROFILE => MSWIN && $] < 5.016; |
|
|
79
|
|
|
|
|
240
|
|
|
|
79
|
|
|
|
|
106795
|
|
|
36
|
|
|
|
|
|
|
|
|
37
|
368
|
|
|
368
|
0
|
739
|
sub expand_tilde ( $dir ) { |
|
|
368
|
|
|
|
|
854
|
|
|
|
368
|
|
|
|
|
659
|
|
|
38
|
|
|
|
|
|
|
|
|
39
|
368
|
50
|
|
|
|
1040
|
return undef unless defined $dir; |
|
40
|
368
|
50
|
|
|
|
3442
|
return File::Spec->canonpath($dir) unless $dir =~ m/^~/; |
|
41
|
|
|
|
|
|
|
|
|
42
|
|
|
|
|
|
|
# Parse path into segments. |
|
43
|
0
|
|
|
|
|
0
|
my ( $volume, $directories, $file ) = File::Spec->splitpath( $dir, 1 ); |
|
44
|
0
|
|
|
|
|
0
|
my @parts = File::Spec->splitdir($directories); |
|
45
|
0
|
|
|
|
|
0
|
my $first = shift( @parts ); |
|
46
|
0
|
0
|
|
|
|
0
|
return File::Spec->canonpath($dir) unless defined $first; |
|
47
|
|
|
|
|
|
|
|
|
48
|
|
|
|
|
|
|
# Expand first segment. |
|
49
|
0
|
|
|
|
|
0
|
my $expanded; |
|
50
|
0
|
|
|
|
|
0
|
if ( WINDOWS_USERPROFILE and $first eq '~' ) { |
|
51
|
|
|
|
|
|
|
$expanded = $ENV{HOME} || $ENV{USERPROFILE}; |
|
52
|
|
|
|
|
|
|
} |
|
53
|
|
|
|
|
|
|
else { |
|
54
|
0
|
|
|
|
|
0
|
( my $pattern = $first ) =~ s/([\\*?{[])/\\$1/g; |
|
55
|
0
|
|
|
|
|
0
|
($expanded) = bsd_glob( $pattern, BSD_GLOB_FLAGS ); |
|
56
|
0
|
0
|
|
|
|
0
|
croak( "Failed to expand $first: $!") if GLOB_ERROR; |
|
57
|
|
|
|
|
|
|
} |
|
58
|
0
|
0
|
0
|
|
|
0
|
return File::Spec->canonpath($dir) |
|
59
|
|
|
|
|
|
|
if !defined $expanded or $expanded eq $first; |
|
60
|
|
|
|
|
|
|
|
|
61
|
|
|
|
|
|
|
# Replace first segment with new path. |
|
62
|
0
|
|
|
|
|
0
|
( $volume, $directories ) = File::Spec->splitpath( $expanded, 1 ); |
|
63
|
0
|
|
|
|
|
0
|
$directories = File::Spec->catdir( $directories, @parts ); |
|
64
|
0
|
|
|
|
|
0
|
return File::Spec->catpath($volume, $directories, $file); |
|
65
|
|
|
|
|
|
|
} |
|
66
|
|
|
|
|
|
|
|
|
67
|
|
|
|
|
|
|
push( @EXPORT, 'expand_tilde' ); |
|
68
|
|
|
|
|
|
|
|
|
69
|
0
|
|
|
0
|
0
|
0
|
sub findexe ( $prog ) { |
|
|
0
|
|
|
|
|
0
|
|
|
|
0
|
|
|
|
|
0
|
|
|
70
|
0
|
|
|
|
|
0
|
my @path; |
|
71
|
0
|
|
|
|
|
0
|
if ( MSWIN ) { |
|
72
|
|
|
|
|
|
|
$prog .= ".exe" unless $prog =~ /\.\w+$/; |
|
73
|
|
|
|
|
|
|
@path = split( ';', $ENV{PATH} ); |
|
74
|
|
|
|
|
|
|
unshift( @path, '.' ); |
|
75
|
|
|
|
|
|
|
} |
|
76
|
|
|
|
|
|
|
else { |
|
77
|
0
|
|
|
|
|
0
|
@path = split( ':', $ENV{PATH} ); |
|
78
|
|
|
|
|
|
|
} |
|
79
|
0
|
|
|
|
|
0
|
foreach ( @path ) { |
|
80
|
0
|
|
|
|
|
0
|
my $try = "$_/$prog"; |
|
81
|
0
|
0
|
|
|
|
0
|
if ( -f -x $try ) { |
|
82
|
|
|
|
|
|
|
#warn("Found $prog in $_\n"); |
|
83
|
0
|
|
|
|
|
0
|
return $try; |
|
84
|
|
|
|
|
|
|
} |
|
85
|
|
|
|
|
|
|
} |
|
86
|
|
|
|
|
|
|
warn("Could not find $prog in ", |
|
87
|
0
|
|
|
|
|
0
|
join(" ", map { qq{"$_"} } @path), "\n"); |
|
|
0
|
|
|
|
|
0
|
|
|
88
|
0
|
|
|
|
|
0
|
return; |
|
89
|
|
|
|
|
|
|
} |
|
90
|
|
|
|
|
|
|
|
|
91
|
|
|
|
|
|
|
push( @EXPORT, 'findexe' ); |
|
92
|
|
|
|
|
|
|
|
|
93
|
0
|
|
|
0
|
0
|
0
|
sub sys ( @cmd ) { |
|
|
0
|
|
|
|
|
0
|
|
|
|
0
|
|
|
|
|
0
|
|
|
94
|
0
|
0
|
|
|
|
0
|
warn("+ @cmd\n") if $::options->{trace}; |
|
95
|
|
|
|
|
|
|
# Use outer defined subroutine, depends on Wx or not. |
|
96
|
0
|
|
|
|
|
0
|
my $res = ::sys(@cmd); |
|
97
|
0
|
0
|
|
|
|
0
|
warn( sprintf("=%02x=> @cmd", $res), "\n" ) if $res; |
|
98
|
0
|
|
|
|
|
0
|
return $res; |
|
99
|
|
|
|
|
|
|
} |
|
100
|
|
|
|
|
|
|
|
|
101
|
|
|
|
|
|
|
push( @EXPORT, 'sys' ); |
|
102
|
|
|
|
|
|
|
|
|
103
|
|
|
|
|
|
|
################ (Pre)Processing ################ |
|
104
|
|
|
|
|
|
|
|
|
105
|
169
|
|
|
169
|
0
|
347
|
sub make_preprocessor ( $prp ) { |
|
|
169
|
|
|
|
|
356
|
|
|
|
169
|
|
|
|
|
299
|
|
|
106
|
169
|
50
|
|
|
|
693
|
return unless $prp; |
|
107
|
|
|
|
|
|
|
|
|
108
|
169
|
|
|
|
|
341
|
my $prep; |
|
109
|
169
|
|
|
|
|
360
|
foreach my $linetype ( keys %{ $prp } ) { |
|
|
169
|
|
|
|
|
685
|
|
|
110
|
507
|
|
|
|
|
757
|
my @targets; |
|
111
|
507
|
|
|
|
|
817
|
my $code = ""; |
|
112
|
507
|
|
|
|
|
774
|
foreach ( @{ $prp->{$linetype} } ) { |
|
|
507
|
|
|
|
|
1113
|
|
|
113
|
0
|
|
0
|
|
|
0
|
my $flags = $_->{flags} // "g"; |
|
114
|
|
|
|
|
|
|
$code .= "m\0" . $_->{select} . "\0 && " |
|
115
|
0
|
0
|
|
|
|
0
|
if $_->{select}; |
|
116
|
0
|
0
|
|
|
|
0
|
if ( $_->{pattern} ) { |
|
117
|
|
|
|
|
|
|
$code .= "s\0" . $_->{pattern} . "\0" |
|
118
|
0
|
|
|
|
|
0
|
. $_->{replace} . "\0$flags;\n"; |
|
119
|
|
|
|
|
|
|
} |
|
120
|
|
|
|
|
|
|
else { |
|
121
|
|
|
|
|
|
|
$code .= "s\0" . quotemeta($_->{target}) . "\0" |
|
122
|
0
|
|
|
|
|
0
|
. quotemeta($_->{replace}) . "\0$flags;\n"; |
|
123
|
|
|
|
|
|
|
} |
|
124
|
|
|
|
|
|
|
} |
|
125
|
507
|
50
|
|
|
|
1307
|
if ( $code ) { |
|
126
|
0
|
|
|
|
|
0
|
my $t = "sub { for (\$_[0]) {\n" . $code . "}}"; |
|
127
|
0
|
|
|
|
|
0
|
$prep->{$linetype} = eval $t; |
|
128
|
0
|
0
|
|
|
|
0
|
die( "CODE : $t\n$@" ) if $@; |
|
129
|
|
|
|
|
|
|
} |
|
130
|
|
|
|
|
|
|
} |
|
131
|
169
|
|
|
|
|
628
|
$prep; |
|
132
|
|
|
|
|
|
|
} |
|
133
|
|
|
|
|
|
|
|
|
134
|
|
|
|
|
|
|
push( @EXPORT, 'make_preprocessor' ); |
|
135
|
|
|
|
|
|
|
|
|
136
|
|
|
|
|
|
|
################ Utilities ################ |
|
137
|
|
|
|
|
|
|
|
|
138
|
|
|
|
|
|
|
# Split (pseudo) command line into key/value pairs. |
|
139
|
|
|
|
|
|
|
|
|
140
|
6
|
|
|
6
|
0
|
6030
|
sub parse_kv ( @lines ) { |
|
|
6
|
|
|
|
|
17
|
|
|
|
6
|
|
|
|
|
11
|
|
|
141
|
|
|
|
|
|
|
|
|
142
|
79
|
|
|
79
|
|
42506
|
use Text::ParseWords qw(shellwords); |
|
|
79
|
|
|
|
|
110323
|
|
|
|
79
|
|
|
|
|
115175
|
|
|
143
|
6
|
|
|
|
|
23
|
my @words = shellwords(@lines); |
|
144
|
|
|
|
|
|
|
|
|
145
|
6
|
|
|
|
|
1071
|
my $res = {}; |
|
146
|
6
|
|
|
|
|
17
|
foreach ( @words ) { |
|
147
|
22
|
100
|
|
|
|
125
|
if ( /^(.*?)=(.+)/ ) { |
|
|
|
100
|
|
|
|
|
|
|
148
|
14
|
|
|
|
|
46
|
$res->{$1} = $2; |
|
149
|
|
|
|
|
|
|
} |
|
150
|
|
|
|
|
|
|
elsif ( /^no[-_]?(.+)/ ) { |
|
151
|
2
|
|
|
|
|
8
|
$res->{$1} = 0; |
|
152
|
|
|
|
|
|
|
} |
|
153
|
|
|
|
|
|
|
else { |
|
154
|
6
|
|
|
|
|
17
|
$res->{$_}++; |
|
155
|
|
|
|
|
|
|
} |
|
156
|
|
|
|
|
|
|
} |
|
157
|
|
|
|
|
|
|
|
|
158
|
6
|
|
|
|
|
31
|
return $res; |
|
159
|
|
|
|
|
|
|
} |
|
160
|
|
|
|
|
|
|
|
|
161
|
|
|
|
|
|
|
push( @EXPORT, 'parse_kv' ); |
|
162
|
|
|
|
|
|
|
|
|
163
|
|
|
|
|
|
|
# Map true/false etc to true / false. |
|
164
|
|
|
|
|
|
|
|
|
165
|
449
|
|
|
449
|
0
|
757
|
sub is_true ( $arg ) { |
|
|
449
|
|
|
|
|
694
|
|
|
|
449
|
|
|
|
|
627
|
|
|
166
|
449
|
50
|
33
|
|
|
1703
|
return if !defined($arg) || $arg eq ''; |
|
167
|
449
|
100
|
|
|
|
1885
|
return if $arg =~ /^(false|null|no|none|off|\s+|0)$/i; |
|
168
|
436
|
|
|
|
|
1610
|
return !!$arg; |
|
169
|
|
|
|
|
|
|
} |
|
170
|
|
|
|
|
|
|
|
|
171
|
|
|
|
|
|
|
push( @EXPORT, 'is_true' ); |
|
172
|
|
|
|
|
|
|
|
|
173
|
|
|
|
|
|
|
# Stricter form of true. |
|
174
|
9
|
|
|
9
|
0
|
21
|
sub is_ttrue ( $arg ) { |
|
|
9
|
|
|
|
|
19
|
|
|
|
9
|
|
|
|
|
15
|
|
|
175
|
9
|
50
|
|
|
|
31
|
return if !defined($arg); |
|
176
|
9
|
|
|
|
|
70
|
$arg =~ /^(on|true|1)$/i; |
|
177
|
|
|
|
|
|
|
} |
|
178
|
|
|
|
|
|
|
|
|
179
|
|
|
|
|
|
|
push( @EXPORT, 'is_ttrue' ); |
|
180
|
|
|
|
|
|
|
|
|
181
|
|
|
|
|
|
|
# Fix apos -> quote. |
|
182
|
|
|
|
|
|
|
|
|
183
|
1179
|
|
|
1179
|
0
|
1619
|
sub fq ( $arg ) { |
|
|
1179
|
|
|
|
|
1853
|
|
|
|
1179
|
|
|
|
|
1544
|
|
|
184
|
1179
|
|
|
|
|
2647
|
$arg =~ s/'/\x{2019}/g; |
|
185
|
1179
|
|
|
|
|
3776
|
$arg; |
|
186
|
|
|
|
|
|
|
} |
|
187
|
|
|
|
|
|
|
|
|
188
|
|
|
|
|
|
|
push( @EXPORT, 'fq' ); |
|
189
|
|
|
|
|
|
|
|
|
190
|
|
|
|
|
|
|
# Quote a string if needed unless forced. |
|
191
|
|
|
|
|
|
|
|
|
192
|
11
|
|
|
11
|
0
|
23
|
sub qquote ( $arg, $force = 0 ) { |
|
|
11
|
|
|
|
|
22
|
|
|
|
11
|
|
|
|
|
19
|
|
|
|
11
|
|
|
|
|
14
|
|
|
193
|
11
|
|
|
|
|
31
|
for ( $arg ) { |
|
194
|
11
|
|
|
|
|
32
|
s/([\\\"])/\\$1/g; |
|
195
|
11
|
|
|
|
|
36
|
s/([[:^print:]])/sprintf("\\u%04x", ord($1))/ge; |
|
|
0
|
|
|
|
|
0
|
|
|
196
|
11
|
100
|
100
|
|
|
65
|
return $_ unless /[\\\s]/ || $force; |
|
197
|
9
|
|
|
|
|
43
|
return qq("$_"); |
|
198
|
|
|
|
|
|
|
} |
|
199
|
|
|
|
|
|
|
} |
|
200
|
|
|
|
|
|
|
|
|
201
|
|
|
|
|
|
|
push( @EXPORT, 'qquote' ); |
|
202
|
|
|
|
|
|
|
|
|
203
|
|
|
|
|
|
|
# Turn foo.bar.blech=blah into { foo => { bar => { blech ==> "blah" } } }. |
|
204
|
|
|
|
|
|
|
|
|
205
|
119
|
|
|
119
|
0
|
307
|
sub prp2cfg ( $defs, $cfg ) { |
|
|
119
|
|
|
|
|
290
|
|
|
|
119
|
|
|
|
|
296
|
|
|
|
119
|
|
|
|
|
266
|
|
|
206
|
119
|
|
|
|
|
349
|
my $ccfg = {}; |
|
207
|
119
|
|
50
|
|
|
514
|
$cfg //= {}; |
|
208
|
119
|
|
|
|
|
827
|
while ( my ($k, $v) = each(%$defs) ) { |
|
209
|
10
|
|
|
|
|
85
|
my @k = split( /[:.]/, $k ); |
|
210
|
10
|
|
|
|
|
25
|
my $c = \$ccfg; # new |
|
211
|
10
|
|
|
|
|
25
|
my $o = $cfg; # current |
|
212
|
10
|
|
|
|
|
28
|
my $lk = pop(@k); # last key |
|
213
|
|
|
|
|
|
|
|
|
214
|
|
|
|
|
|
|
# Step through the keys. |
|
215
|
10
|
|
|
|
|
30
|
foreach ( @k ) { |
|
216
|
13
|
|
|
|
|
38
|
$c = \($$c->{$_}); |
|
217
|
13
|
|
|
|
|
34
|
$o = $o->{$_}; |
|
218
|
|
|
|
|
|
|
} |
|
219
|
|
|
|
|
|
|
|
|
220
|
|
|
|
|
|
|
# Final key. Merge array if so. |
|
221
|
10
|
50
|
33
|
|
|
63
|
if ( $lk =~ /^\d+$/ && ref($o) eq 'ARRAY' ) { |
|
222
|
0
|
0
|
|
|
|
0
|
unless ( ref($$c) eq 'ARRAY' ) { |
|
223
|
|
|
|
|
|
|
# Only copy orig values the first time. |
|
224
|
0
|
|
|
|
|
0
|
$$c->[$_] = $o->[$_] for 0..scalar(@{$o})-1; |
|
|
0
|
|
|
|
|
0
|
|
|
225
|
|
|
|
|
|
|
} |
|
226
|
0
|
|
|
|
|
0
|
$$c->[$lk] = $v; |
|
227
|
|
|
|
|
|
|
} |
|
228
|
|
|
|
|
|
|
else { |
|
229
|
10
|
|
|
|
|
80
|
$$c->{$lk} = $v; |
|
230
|
|
|
|
|
|
|
} |
|
231
|
|
|
|
|
|
|
} |
|
232
|
119
|
|
|
|
|
448
|
return $ccfg; |
|
233
|
|
|
|
|
|
|
} |
|
234
|
|
|
|
|
|
|
|
|
235
|
|
|
|
|
|
|
push( @EXPORT, 'prp2cfg' ); |
|
236
|
|
|
|
|
|
|
|
|
237
|
|
|
|
|
|
|
# Remove markup. |
|
238
|
2254
|
|
|
2254
|
0
|
18170
|
sub demarkup ( $t ) { |
|
|
2254
|
|
|
|
|
3474
|
|
|
|
2254
|
|
|
|
|
2907
|
|
|
239
|
2254
|
|
|
|
|
4225
|
return join( '', grep { ! /^\ } splitmarkup($t) ); |
|
|
2287
|
|
|
|
|
11396
|
|
|
240
|
|
|
|
|
|
|
} |
|
241
|
|
|
|
|
|
|
push( @EXPORT, 'demarkup' ); |
|
242
|
|
|
|
|
|
|
|
|
243
|
|
|
|
|
|
|
# Split into markup/nonmarkup segments. |
|
244
|
2276
|
|
|
2276
|
0
|
3067
|
sub splitmarkup ( $t ) { |
|
|
2276
|
|
|
|
|
3180
|
|
|
|
2276
|
|
|
|
|
3332
|
|
|
245
|
2276
|
|
|
|
|
11431
|
my @t = split( qr;(?(?:[-\w]+|span\s.*?)>);, $t ); |
|
246
|
2276
|
|
|
|
|
7056
|
return @t; |
|
247
|
|
|
|
|
|
|
} |
|
248
|
|
|
|
|
|
|
push( @EXPORT, 'splitmarkup' ); |
|
249
|
|
|
|
|
|
|
|
|
250
|
|
|
|
|
|
|
# For conditional filling of hashes. |
|
251
|
46
|
|
|
46
|
0
|
68
|
sub maybe ( $key, $value, @rest ) { |
|
|
46
|
|
|
|
|
81
|
|
|
|
46
|
|
|
|
|
59
|
|
|
|
46
|
|
|
|
|
78
|
|
|
|
46
|
|
|
|
|
81
|
|
|
252
|
46
|
50
|
33
|
|
|
189
|
if (defined $key and defined $value) { |
|
253
|
0
|
|
|
|
|
0
|
return ( $key, $value, @rest ); |
|
254
|
|
|
|
|
|
|
} |
|
255
|
|
|
|
|
|
|
else { |
|
256
|
46
|
50
|
33
|
|
|
344
|
( defined($key) || @rest ) ? @rest : (); |
|
257
|
|
|
|
|
|
|
} |
|
258
|
|
|
|
|
|
|
} |
|
259
|
|
|
|
|
|
|
push( @EXPORT, "maybe" ); |
|
260
|
|
|
|
|
|
|
|
|
261
|
|
|
|
|
|
|
1; |