forked from Stellarium/stellarium
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathrefactor.pl
executable file
·316 lines (252 loc) · 7.27 KB
/
refactor.pl
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
#!/usr/bin/perl
use warnings;
use strict;
use Getopt::Long;
use Term::ANSIColor;
my $textRed = color 'red';
my $textGreen = color 'green';
my $textBlue = color 'blue';
my $textYellow = color 'yellow';
my $textReset = color 'reset';
my $flg_verbose = 0;
my $flg_yes = 0;
my $flg_no = 0;
GetOptions(
"verbose" => \$flg_verbose,
"yes" => \$flg_yes,
"no" => \$flg_no,
"help" => sub { lusage(0); },
) || usage(1);
# we expect three or more parameters (search pattern, replacement, file(s))
if ($#ARGV<2) { usage(2);}
if ($ARGV[1] eq ".") {
# if the replacement is just ".", guess what the replacement is
# from the search pattern by transforming it to camelCase from
# lc_underscore_style
my $s = shift || usage(1);
my $r = autoProcess($s);
shift;
refactorString($s, $r, @ARGV);
}
else {
# or if the second argument is not ".", use it as the replacemnt
my $s = shift || usage(1);
my $r = shift || usage(1);
refactorString($s, $r, @ARGV);
}
exit(0);
# dump usage message and exit. message is maded from POD tags in this file
sub usage {
system "pod2usage < $0";
exit $_[0] || 0;
}
# longer usage message
sub lusage {
system "pod2usage -verbose 1 < $0";
exit $_[0] || 0;
}
# guess a replacement string from the search pattern
# e.g. variable_name -> variableName
sub autoProcess {
my $s = shift;
# be rid of special characters
$s =~ s/\\b//g;
$s =~ s/\\z//g;
my $w;
if ($s =~ /^[A-Z\d_]+$/) {
# LIKE_THIS -> LikeThis
my @a = split(/_+/, $s);
$w = "";
foreach (@a) {
$_ = lc($_);
substr($_,0,1) = uc(substr($_,0,1));
$w .= $_;
}
}
else {
$w = $s;
$w =~ s/_([a-z])/uc($1)/ge;
$w =~ s/_//g;
}
return $w;
}
# do all the work
# args: search-pattern; replacement-string; file; [file]; ...
sub refactorString {
# sanity checks on parameters
my $s = shift || die "empty search string not good\n";
my $r = shift || die "empty replacement string not good\n";
my @allFileList = @_;
if ($s eq "") { die "empty string not good\n"; }
if ($s eq $r) {
die "re-factored string is the same as the original $s\n";
}
if (!$flg_yes) {
print "${textBlue}Will replace${textGreen} $s ${textBlue}with${textRed} $r ${textBlue}:${textReset}\n";
}
my %patchFileList;
my $textRed = color 'red';
my $textGreen = color 'green';
my $textReset = color 'reset';
# read all the files and build a list of the ones which contain the
# search pattern. the list is the keys in the hash %patchFileList
# also print previews of the replacements, with the text to be
# replaced in red.
my $candidates = 0;
foreach my $f (@allFileList) {
if ($f =~ /~$/) { warn "SKIPPING backup file: $f"; next; }
if ( ! open(F, "<$f") ) {
warn "cannot open $f : SKIPPING\n";
}
else {
while(<F>) {
# First, if we already have the replacement string
# colour it red because it could be a problem
if (s/($r)/$textRed$1$textReset/g) {
if (!$flg_yes || $flg_verbose) {
print "!! $f\[$.\]: $_";
}
}
# Now colour all target strings and preview them
if (s/($s)/$textGreen$1$textReset/g) {
$patchFileList{$f} = 1;
if (!$flg_yes || $flg_verbose) {
s/(\")/$textRed$1$textReset/g;
print " $f\[$.\]: $_";
}
$candidates++;
}
}
close(F);
}
}
if ($candidates==0) {
if ($flg_verbose) {
print "There is nothing to replace\n";
}
exit(0);
}
if ($flg_no) { exit(0); }
# prompt user if they want to do the replacement
my $do_it = $flg_yes;
my $prompt_replace = 0;
if ($flg_yes) {
$do_it = 1;
}
else {
print "\nAccept $candidates changes (${textGreen}y${textReset}es/${textGreen}n${textReset}o/${textGreen}p${textReset}rompt)? >";
my $res = <STDIN>;
chomp $res;
if (lc($res) eq "y") {
$do_it = 1;
}
elsif (lc($res) eq "p") {
$do_it = 1;
$prompt_replace = 1;
}
}
if ($do_it)
{
# iterate over the files identified in the first scan, doing the
# replacement. The original files will be renamed with a ~ suffix
# and the original named file will be re-created with the changes.
my $updatedLines = 0;
my $updatedFiles = 0;
foreach my $f (keys %patchFileList) {
my $orig = "$f" . "~";
rename "$f", "$orig" || die "cannot rename file $f to $orig : $!\n";
open(ORIG, "<$orig") || die "cannot open $orig for reading : $!\n";
open(NEWF, ">$f") || die "cannot open $f for writing : $!\n";
while(<ORIG>)
{
if (/$s/) {
if (!$prompt_replace) {
s/$s/$r/g;
$updatedLines ++;
}
else {
my $prompt = $_;
$prompt =~ s/($s)/$textGreen$1$textReset/g;
$prompt =~ s/(\")/$textRed$1$textReset/g;
print "$prompt";
print "patch this line (y/n)? >";
my $res = <STDIN>;
chomp $res;
if (lc($res) ne "n") {
s/$s/$r/g;
$updatedLines ++;
}
}
}
print NEWF $_;
}
close(ORIG);
close(NEWF);
$updatedFiles++;
if ($flg_verbose) {
print "patched $f (backup in $orig)\n";
}
}
if ($flg_verbose) {
print "Done updated $updatedLines lines in $updatedFiles files\n";
}
}
elsif ($flg_verbose) {
print "not patching\n";
}
exit;
}
__END__
=head1 NAME
refactor - replace patterns in bunch of files with a nice-ish preview
=head1 SYNOPSIS
refactor [options] pattern {replacement|.} file [file] ...
=head1 DESCRIPTION
Replaces Perl-RE I<pattern> with I<replacement> in all specified files. First the
program prints a preview of what will be done.
If I<replacement> is a period ("."), the replacement will be guesses at according to
Stellarium's variable coding standards (e.g. camel_case => camelCase).
When the replacement string already exists in one of the target files, the offending
string will be coloured red, and the line printed out with the prefix !!.
Strings which match the search pattern will be coloured green, and those lines
which will be modified will be printed with no prefix.
Once the preview output is completed, the user will be prompted if they wish to
proceed with the modificaion of the files. If there are no modifiections to do
the program will terminate.
On confirming the intention to proceed, the original files will be re-named with a ~
suffix, and a modified copy of the file with the original name will be created with
the changes made.
=head1 OPTIONS
=over
=item B<--help>
Print the command line syntax an option details.
=item B<--no>
Don't do anything (useful for previewing).
=item B<--verbose>
Print extra information about what is going on.
=item B<--yes>
Don't prompt for confirmation - just do it. This will also suppress the
display of preview and conflict information, unless the --verbose flag is used.
=head1 ENVIRONMENT
N/A
=head1 FILES
When refactor is run, all modified files are backed up into a file with the same name
as the original with a ~ suffix. refactor will not opereate on files with this name.
=head1 LICENSE
refactor is released under the GNU GPL (version 2, June 1991). A
copy of the license should have been provided in the distribution of
the software in a file called "LICENSE". If you can't find this, then
try here: http://www.gnu.org/copyleft/gpl.html
=head1 AUTHOR
Matthew Gates <[email protected]>
http://porpoisehead.net/
=head1 CHANGELOG
=over
=item Date:2008-06-26 Created, Author MNG
Original version.
=back
=head1 BUGS
Please report bugs to the author.
=head1 SEE ALSO
sed, grep and friends
=cut