From f5139df9346fa484abe4691dce0b5cf37a8b8c1d Mon Sep 17 00:00:00 2001 From: Norbert Csaba Herczeg Date: Wed, 20 Nov 2024 11:31:35 +0100 Subject: [PATCH] JNG-6016 translation sync script --- .../src/main/resources/actor/README.md.hbs | 11 +++- .../resources/actor/sync-translations.py.hbs | 51 +++++++++++++++---- 2 files changed, 50 insertions(+), 12 deletions(-) diff --git a/judo-ui-react/src/main/resources/actor/README.md.hbs b/judo-ui-react/src/main/resources/actor/README.md.hbs index 41ee0e9c..ea38a564 100644 --- a/judo-ui-react/src/main/resources/actor/README.md.hbs +++ b/judo-ui-react/src/main/resources/actor/README.md.hbs @@ -88,8 +88,7 @@ If we modify the model in a way where we add new elements or remove previous one our translations by hand. To counter this we provide a sync script `sync-translations.py`, which can update our tracked translations according -to model changes. Don't worry, translations related to existing elements won't be modified, only new ones will be added -and removed ones will be cleaned up. +to model changes. > The script will run for both application and system translations as well @@ -97,6 +96,14 @@ and removed ones will be cleaned up. python3 sync-translations.py ``` +### Removing obsolete entries + +We can provide a `-d` flag to remove entries from the target files which are no longer present in the defaults: + +```bash +python3 sync-translations.py -d +``` + ## Other ### About pnpm-lock.yaml diff --git a/judo-ui-react/src/main/resources/actor/sync-translations.py.hbs b/judo-ui-react/src/main/resources/actor/sync-translations.py.hbs index 63b1eec6..d5f6ad42 100755 --- a/judo-ui-react/src/main/resources/actor/sync-translations.py.hbs +++ b/judo-ui-react/src/main/resources/actor/sync-translations.py.hbs @@ -1,7 +1,9 @@ import json import os +import argparse +from collections import OrderedDict -def sync_translation_files(source_path, target_path): +def sync_translation_files(source_path, target_path, delete_obsolete=False): # Prepend the paths with "public/i18n/" source_path = os.path.join("public", "i18n", source_path) target_path = os.path.join("public", "i18n", target_path) @@ -17,12 +19,15 @@ def sync_translation_files(source_path, target_path): source_translations = source_data.get("translation", {}) target_translations = target_data.get("translation", {}) - # Update target_translations based on source_translations - # Keep entries from the target if they exist in both; add new ones from the source - updated_translations = { - key: target_translations[key] if key in target_translations else source_translations[key] - for key in source_translations - } + if not delete_obsolete: + updated_translations = merge_json_by_index(source_translations, target_translations) + else: + # Update target_translations based on source_translations + # Keep entries from the target if they exist in both; add new ones from the source + updated_translations = { + key: target_translations[key] if key in target_translations else source_translations[key] + for key in source_translations + } # Create the updated target data structure with updated translations updated_target_data = {"translation": updated_translations} @@ -32,10 +37,36 @@ def sync_translation_files(source_path, target_path): json.dump(updated_target_data, target_file, indent=4, ensure_ascii=False) target_file.write('\n') # Ensure a single empty line at the end of the file + +def merge_json_by_index(source, target): + result = OrderedDict() + source_keys = list(source.keys()) + target_keys = list(target.keys()) + max_len = max(len(source_keys), len(target_keys)) + + for i in range(max_len): + source_key = source_keys[i] if i < len(source_keys) else None + target_key = target_keys[i] if i < len(target_keys) else None + + if source_key == target_key: + result[source_key] = target[target_key] + + if source_key is not None and source_key not in target: + result[source_key] = source[source_key] + + if target_key is not None and target_key not in source: + result[target_key] = target[target_key] + + return result + def main(): - sync_translation_files('system_default.json', 'system_hu-HU.json') - sync_translation_files('system_default.json', 'system_en-US.json') - sync_translation_files('application_default.json', 'application_{{ getDefaultLanguage application }}.json') + parser = argparse.ArgumentParser(description="Sync translation JSON files.") + parser.add_argument("-d", "--delete-obsolete", action="store_true", help="Delete entries from the target that are no longer in the source") + args = parser.parse_args() + + sync_translation_files('system_default.json', 'system_hu-HU.json', args.delete_obsolete) + sync_translation_files('system_default.json', 'system_en-US.json', args.delete_obsolete) + sync_translation_files('application_default.json', 'application_{{ getDefaultLanguage application }}.json', args.delete_obsolete) if __name__ == "__main__": main()