Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
grasscutters
GitHub Repository: grasscutters/grasscutter
Path: blob/development/scripts/manage_languages.py
3153 views
1
# Written for Python 3.6+
2
# Older versions don't retain insertion order of regular dicts
3
import argparse
4
import cmd
5
import json
6
import os
7
import re
8
from pprint import pprint
9
10
INDENT = 2
11
PRIMARY_LANGUAGE = 'en-US.json'
12
PRIMARY_FALLBACK_PREFIX = '🇺🇸' # This is invisible in-game, terminal emulators might render it
13
LANGUAGE_FOLDER = 'src/main/resources/languages/'
14
LANGUAGE_FILENAMES = sorted(os.listdir(LANGUAGE_FOLDER), key=lambda x: 'AAA' if x == PRIMARY_LANGUAGE else x)
15
SOURCE_FOLDER = 'src/'
16
SOURCE_EXTENSIONS = ('java')
17
18
19
def ppprint(data):
20
pprint(data, width=130, sort_dicts=False, compact=True)
21
22
23
class JsonHelpers:
24
@staticmethod
25
def load(filename: str) -> dict:
26
with open(filename, 'r', encoding='utf-8') as file:
27
return json.load(file)
28
29
@staticmethod
30
def save(filename: str, data: dict) -> None:
31
with open(filename, 'w', encoding='utf-8', newline='\n') as file:
32
json.dump(data, file, ensure_ascii=False, indent=INDENT)
33
file.write('\n') # json.dump doesn't terminate last line
34
35
@staticmethod
36
def flatten(data: dict, prefix='') -> dict:
37
output = {}
38
for key, value in data.items():
39
if isinstance(value, dict):
40
for k,v in JsonHelpers.flatten(value, f'{prefix}{key}.').items():
41
output[k] = v
42
else:
43
output[f'{prefix}{key}'] = value
44
return output
45
46
@staticmethod
47
def unflatten(data: dict) -> dict:
48
output = {}
49
def add_key(k: list, value, d: dict):
50
if len(k) == 1:
51
d[k[0]] = value
52
else:
53
d[k[0]] = d.get(k[0], {})
54
add_key(k[1:], value, d[k[0]])
55
for key, value in data.items():
56
add_key(key.split('.'), value, output)
57
return output
58
59
@staticmethod
60
def pprint_keys(keys, indent=4) -> str:
61
# Only strip down to one level
62
padding = ' ' * indent
63
roots = {}
64
for key in keys:
65
root, _, k = key.rpartition('.')
66
roots[root] = roots.get(root, [])
67
roots[root].append(k)
68
lines = []
69
for root, ks in roots.items():
70
if len(ks) > 1:
71
lines.append(f'{padding}{root}.[{", ".join(ks)}]')
72
else:
73
lines.append(f'{padding}{root}.{ks[0]}')
74
return ',\n'.join(lines)
75
76
@staticmethod
77
def deep_clone_and_fill(d1: dict, d2: dict, fallback_prefix=PRIMARY_FALLBACK_PREFIX) -> dict:
78
out = {}
79
for key, value in d1.items():
80
if isinstance(value, dict):
81
out[key] = JsonHelpers.deep_clone_and_fill(value, d2.get(key, {}), fallback_prefix)
82
else:
83
v2 = d2.get(key, value)
84
if type(value) == str and v2 == value:
85
out[key] = fallback_prefix + value
86
else:
87
out[key] = v2
88
return out
89
90
91
class LanguageManager:
92
TRANSLATION_KEY = re.compile(r'[Tt]ranslate.*"(\w+\.[\w\.]+)"')
93
POTENTIAL_KEY = re.compile(r'"(\w+\.[\w\.]+)"')
94
COMMAND_LABEL = re.compile(r'@Command\s*\([\W\w]*?label\s*=\s*"(\w+)"', re.MULTILINE) # [\W\w] is a cheeky way to match everything including \n
95
96
def __init__(self):
97
self.load_jsons()
98
99
def load_jsons(self):
100
self.language_jsons = [JsonHelpers.load(LANGUAGE_FOLDER + filename) for filename in LANGUAGE_FILENAMES]
101
self.flattened_jsons = [JsonHelpers.flatten(j) for j in self.language_jsons]
102
self.update_keys()
103
104
def update_keys(self):
105
self.key_sets = [set(j.keys()) for j in self.flattened_jsons]
106
self.common_keys = set.intersection(*self.key_sets)
107
self.all_keys = set.union(*self.key_sets)
108
self.used_keys = self.find_all_used_keys(self.all_keys)
109
self.missing_keys = self.used_keys - self.common_keys
110
self.unused_keys = self.all_keys - self.used_keys
111
112
def find_all_used_keys(self, expected_keys=[]) -> set:
113
# Note that this will only find string literals passed to the translate() or sendTranslatedMessage() methods!
114
# String variables passed to them can be checked against expected_keys
115
used = set()
116
potential = set()
117
for root, dirs, files in os.walk(SOURCE_FOLDER):
118
for file in files:
119
if file.rpartition('.')[-1] in SOURCE_EXTENSIONS:
120
filename = os.path.join(root, file)
121
with open(filename, 'r', encoding='utf-8') as f:
122
data = f.read() # Loads in entire file at once
123
for k in self.TRANSLATION_KEY.findall(data):
124
used.add(k)
125
for k in self.POTENTIAL_KEY.findall(data):
126
potential.add(k)
127
for label in self.COMMAND_LABEL.findall(data):
128
used.add(f'commands.{label}.description')
129
return used | (potential & expected_keys)
130
131
def _lint_report_language(self, lang: str, keys: set, flattened: dict, primary_language_flattened: dict) -> None:
132
missing = self.used_keys - keys
133
unused = keys - self.used_keys
134
identical_keys = set() if (lang == PRIMARY_LANGUAGE) else {key for key in keys if primary_language_flattened.get(key, None) == flattened.get(key)}
135
placeholder_keys = {key for key in keys if flattened.get(key).startswith(PRIMARY_FALLBACK_PREFIX)}
136
p1 = f'Language {lang} has {len(missing)} missing keys and {len(unused)} unused keys.'
137
p2 = 'This is the primary language.' if (lang == PRIMARY_LANGUAGE) else f'{len(identical_keys)} match {PRIMARY_LANGUAGE}, {len(placeholder_keys)} have the placeholder mark.'
138
print(f'{p1} {p2}')
139
140
lint_categories = {
141
'Missing': missing,
142
'Unused': unused,
143
f'Matches {PRIMARY_LANGUAGE}': identical_keys,
144
'Placeholder': placeholder_keys,
145
}
146
for name, category in lint_categories.items():
147
if len(category) > 0:
148
print(name + ':')
149
print(JsonHelpers.pprint_keys(sorted(category)))
150
151
def lint_report(self) -> None:
152
print(f'There are {len(self.missing_keys)} translation keys in use that are missing from one or more language files.')
153
print(f'There are {len(self.unused_keys)} translation keys in language files that are not used.')
154
primary_language_flattened = self.flattened_jsons[LANGUAGE_FILENAMES.index(PRIMARY_LANGUAGE)]
155
for lang, keys, flattened in zip(LANGUAGE_FILENAMES, self.key_sets, self.flattened_jsons):
156
print('')
157
self._lint_report_language(lang, keys, flattened, primary_language_flattened)
158
159
def rename_keys(self, key_remappings: dict) -> None:
160
# Unfortunately we can't rename keys in-place preserving insertion order, so we have to make new dicts
161
for i in range(len(self.flattened_jsons)):
162
self.flattened_jsons[i] = {key_remappings.get(k,k):v for k,v in self.flattened_jsons[i].items()}
163
164
def update_secondary_languages(self):
165
# Push en_US fallback
166
primary_language_json = self.language_jsons[LANGUAGE_FILENAMES.index(PRIMARY_LANGUAGE)]
167
for filename, lang in zip(LANGUAGE_FILENAMES, self.language_jsons):
168
if filename != PRIMARY_LANGUAGE:
169
js = JsonHelpers.deep_clone_and_fill(primary_language_json, lang)
170
JsonHelpers.save(LANGUAGE_FOLDER + filename, js)
171
172
def update_all_languages_from_flattened(self):
173
for filename, flat in zip(LANGUAGE_FILENAMES, self.flattened_jsons):
174
JsonHelpers.save(LANGUAGE_FOLDER + filename, JsonHelpers.unflatten(flat))
175
176
def save_flattened_languages(self, prefix='flat_'):
177
for filename, flat in zip(LANGUAGE_FILENAMES, self.flattened_jsons):
178
JsonHelpers.save(prefix + filename, flat)
179
180
181
class InteractiveRename(cmd.Cmd):
182
intro = 'Welcome to the interactive rename shell. Type help or ? to list commands.\n'
183
prompt = '(rename) '
184
file = None
185
186
def __init__(self, language_manager: LanguageManager) -> None:
187
super().__init__()
188
self.language_manager = language_manager
189
self.flat_keys = [key for key in language_manager.flattened_jsons[LANGUAGE_FILENAMES.index(PRIMARY_LANGUAGE)].keys()]
190
self.mappings = {}
191
192
def do_add(self, arg):
193
'''
194
Prepare to rename an existing translation key. Will not actually rename anything until you confirm all your pending changes with 'rename'.
195
e.g. a single string: add commands.execution.argument_error commands.generic.invalid.argument
196
e.g. a group: add commands.enter_dungeon commands.new_enter_dungeon
197
'''
198
args = arg.split()
199
if len(args) < 2:
200
self.do_help('add')
201
return
202
old, new = args[:2]
203
if old in self.flat_keys:
204
self.mappings[old] = new
205
else:
206
# Check if we are renaming a higher level
207
if not old.endswith('.'):
208
old = old + '.'
209
results = [key for key in self.flat_keys if key.startswith(old)]
210
if len(results) > 0:
211
if not new.endswith('.'):
212
new = new + '.'
213
new_mappings = {key: key.replace(old, new) for key in results}
214
# Ask for confirmation
215
print('Will add the following mappings:')
216
ppprint(new_mappings)
217
print('Add these mappings? [y/N]')
218
if self.prompt_yn():
219
for k,v in new_mappings.items():
220
self.mappings[k] = v
221
else:
222
print('No translation keys matched!')
223
224
def complete_add(self, text: str, line: str, begidx: int, endidx: int) -> list:
225
if text == '':
226
return [k for k in {key.partition('.')[0] for key in self.flat_keys}]
227
results = [key for key in self.flat_keys if key.startswith(text)]
228
if len(results) > 40:
229
# Collapse categories
230
if text[-1] != '.':
231
text = text + '.'
232
level = text.count('.') + 1
233
new_results = {'.'.join(key.split('.')[:level]) for key in results}
234
return list(new_results)
235
return results
236
237
def do_remove(self, arg):
238
'''
239
Remove a pending rename mapping. Takes the old name of the key, not the new one.
240
e.g. a single key: remove commands.execution.argument_error
241
e.g. a group: remove commands.enter_dungeon
242
'''
243
old = arg.split()[0]
244
if old in self.mappings:
245
self.mappings.pop(old)
246
else:
247
# Check if we are renaming a higher level
248
if not old.endswith('.'):
249
old = old + '.'
250
results = [key for key in self.mappings if key.startswith(old)]
251
if len(results) > 0:
252
# Ask for confirmation
253
print('Will remove the following pending mappings:')
254
print(JsonHelpers.pprint_keys(results))
255
print('Delete these mappings? [y/N]')
256
if self.prompt_yn():
257
for key in results:
258
self.mappings.pop(key)
259
else:
260
print('No pending rename mappings matched!')
261
262
def complete_remove(self, text: str, line: str, begidx: int, endidx: int) -> list:
263
return [key for key in self.mappings if key.startswith(text)]
264
265
def do_rename(self, _arg):
266
'Applies pending renames and overwrites language jsons.'
267
# Ask for confirmation
268
print('Will perform the following mappings:')
269
ppprint(self.mappings)
270
print('Perform and save these rename mappings? [y/N]')
271
if self.prompt_yn():
272
self.language_manager.rename_keys(self.mappings)
273
self.language_manager.update_all_languages_from_flattened()
274
print('Renamed keys, closing')
275
return True
276
else:
277
print('Do you instead wish to quit without saving? [yes/N]')
278
if self.prompt_yn(True):
279
print('Left rename shell without renaming')
280
return True
281
282
def prompt_yn(self, strict_yes=False):
283
if strict_yes:
284
return input('(yes/N) ').lower() == 'yes'
285
return input('(y/N) ').lower()[0] == 'y'
286
287
288
def main(args: argparse.Namespace):
289
# print(args)
290
language_manager = LanguageManager()
291
errors = None
292
if args.lint_report:
293
language_manager.lint_report()
294
missing = language_manager.used_keys - language_manager.key_sets[LANGUAGE_FILENAMES.index(PRIMARY_LANGUAGE)]
295
if len(missing) > 0:
296
errors = f'[ERROR] {len(missing)} keys missing from primary language json!\n{JsonHelpers.pprint_keys(missing)}'
297
if prefix := args.save_flattened:
298
language_manager.save_flattened_languages(prefix)
299
if args.update:
300
print('Updating secondary languages')
301
language_manager.update_secondary_languages()
302
if args.interactive_rename:
303
language_manager.load_jsons() # Previous actions may have changed them on-disk
304
try:
305
InteractiveRename(language_manager).cmdloop()
306
except KeyboardInterrupt:
307
print('Left rename shell without renaming')
308
if errors:
309
print(errors)
310
exit(1)
311
312
313
314
if __name__ == "__main__":
315
parser = argparse.ArgumentParser(description="Manage Grasscutter's language json files.")
316
parser.add_argument('-u', '--update', action='store_true',
317
help=f'Update secondary language files to conform to the layout of the primary language file ({PRIMARY_LANGUAGE}) and contain any new keys from it.')
318
parser.add_argument('-l', '--lint-report', action='store_true',
319
help='Prints a lint report, listing unused, missing, and untranslated keys among all language jsons.')
320
parser.add_argument('-f', '--save-flattened', const='./flat_', metavar='prefix', nargs='?',
321
help='Save copies of all the language jsons in a flattened key form.')
322
parser.add_argument('-i', '--interactive-rename', action='store_true',
323
help='Enter interactive rename mode, in which you can specify keys in flattened form to be renamed.')
324
args = parser.parse_args()
325
main(args)
326