-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathreplace.py
executable file
·106 lines (81 loc) · 2.78 KB
/
replace.py
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
#!/usr/bin/env python3
#
# The script can be used for more complex replacements in the source code.
#
import argparse
import concurrent.futures
import os
import sys
EXTENSIONS = ('.h', '.c', '.cc', '.C', '.rst', '.f90', '.f', '.f95', '.inl', '.py', '.rs')
def handle_file_p(filename):
if 'testsuite' in filename:
return False
if all(not filename.endswith(ext) for ext in EXTENSIONS):
return False
return True
"""
Example replacement:
LOADED_FILES = [re.compile(fr'\b{re.escape(x)}\b') for x in open('/tmp/files.txt').read().splitlines()]
m = re.match(r'.*time_function\\(&([^,]*),', line)
if m:
e = m.end(1)
name = m.group(1)
line = line[:e + 1] + f' "{name}",' + line[e + 1:]
for x in LOADED_FILES:
for match in reversed(list(re.finditer(x, line))):
start = match.start()
end = match.end()
text = match.group()
line = line[:start] + text + 'c' + line[end:]
return line
"""
def modify_line(line, index, lines, filename):
return line
parser = argparse.ArgumentParser(description='Make a custom replacements '
'for source files')
parser.add_argument('directory', help='Root directory')
parser.add_argument('-v', action='store_true',
help='Verbose output')
parser.add_argument('-vv', action='store_true',
help='More verbose output')
args = parser.parse_args()
modified_files = 0
files_worklist = []
for root, _, files in os.walk(sys.argv[1]):
for file in files:
full = os.path.join(root, file)
if handle_file_p(full):
files_worklist.append(full)
def replace_file(full, i, n):
if args.vv:
print(f'.. {i + 1}/{len(files_worklist)}: {full}')
modified = False
try:
modified_lines = []
with open(full) as f:
lines = f.readlines()
for index, line in enumerate(lines):
modified_line = modify_line(line, index, lines, full)
if line != modified_line:
modified = True
modified_lines.append(modified_line)
if modified:
with open(full, 'w') as w:
w.write(''.join(modified_lines))
if args.v:
print(f'File modified: {full}')
return True
except UnicodeDecodeError as e:
print(f'Skipping file: {full} ({e})')
with concurrent.futures.ProcessPoolExecutor() as executor:
futures = []
for i, full in enumerate(files_worklist):
futures.append(executor.submit(replace_file, full, i, len(files_worklist)))
for future in futures:
r = future.result()
if r:
modified_files += 1
if args.vv:
print()
print(f'Visited files: {len(files_worklist)}')
print(f'Modified files: {modified_files}')