From 8280c538a2093b6bbf37577620ef80a38eacde1e Mon Sep 17 00:00:00 2001 From: Martin Pitt Date: Mon, 13 Jan 2025 11:41:59 +0100 Subject: [PATCH 1/2] base1: Add unit test for zh_CN (one plural form) ngettext This doesn't reproduce #21486, but is a good test to have anyway. --- pkg/base1/test-locale.js | 27 +++++++++++++++++++++------ 1 file changed, 21 insertions(+), 6 deletions(-) diff --git a/pkg/base1/test-locale.js b/pkg/base1/test-locale.js index 9ae6686f0b78..b1993df3721c 100644 --- a/pkg/base1/test-locale.js +++ b/pkg/base1/test-locale.js @@ -37,6 +37,15 @@ const ru = { "$0 bit": ["$0 bits", "$0 бит", "$0 бита", "$0 бит"] }; +const zh_CN = { + "": { + language: "zh_CN", + "language-direction": "ltr", + "plural-forms": (n) => 0, + }, + "$0 important hit": [null, "$0 次命中(含关键命中)"], +}; + QUnit.test("public api", function (assert) { assert.equal(typeof cockpit.locale, "function", "cockpit.locale is a function"); }); @@ -83,12 +92,18 @@ QUnit.test("ngettext simple", function (assert) { QUnit.test("ngettext complex", function (assert) { cockpit.locale(null); /* clear it */ cockpit.locale(ru); - assert.equal(cockpit.ngettext("$0 bit", "$0 bits", 0), "$0 бит", "zero things"); - assert.equal(cockpit.ngettext("$0 bit", "$0 bits", 1), "$0 бит", "one thing"); - assert.equal(cockpit.ngettext("$0 bit", "$0 bits", 5), "$0 бит", "multiple things"); - assert.equal(cockpit.ngettext("$0 bit", "$0 bits", 23), "$0 бита", "genitive singular"); - assert.equal(cockpit.ngettext("$0 byte", "$0 bytes", 1), "$0 byte", "default one"); - assert.equal(cockpit.ngettext("$0 byte", "$0 bytes", 2), "$0 bytes", "default multiple"); + assert.equal(cockpit.ngettext("$0 bit", "$0 bits", 0), "$0 бит", "ru, zero things"); + assert.equal(cockpit.ngettext("$0 bit", "$0 bits", 1), "$0 бит", "ru, one thing"); + assert.equal(cockpit.ngettext("$0 bit", "$0 bits", 5), "$0 бит", "ru, multiple things"); + assert.equal(cockpit.ngettext("$0 bit", "$0 bits", 23), "$0 бита", "ru, genitive singular"); + assert.equal(cockpit.ngettext("$0 byte", "$0 bytes", 1), "$0 byte", "ru, default one"); + assert.equal(cockpit.ngettext("$0 byte", "$0 bytes", 2), "$0 bytes", "ru, default multiple"); + + cockpit.locale(null); /* clear it */ + cockpit.locale(zh_CN); + [0, 1, 2].forEach(i => + assert.equal(cockpit.ngettext("$0 important hit", "$0 XXX", i), "$0 次命中(含关键命中)", `zh_CN, ${i} things`) + ); }); QUnit.test("translate document", function (assert) { From c4a5357651147ee650b1dbe78c5be8d8c7e40f29 Mon Sep 17 00:00:00 2001 From: Martin Pitt Date: Mon, 13 Jan 2025 15:58:03 +0100 Subject: [PATCH 2/2] bridge: Drop translations from manifests.js, introduce m-i18n.js Pages don't (shouldn't) care about manifest translations when including `manifests.js` -- they usually do that to query for a package's existence or some feature flags. Only the Shell cares about the translations. So drop translations from `manifests.js`, and support a new `manifests-i18n.js` which only the shell uses. This isolates translations from different pages from another. Concretely, subscription-manager-cockpit and our systemd page both translate "$0 important hit" differently, and even to different data types (in s-m it results in an array, in systemd in a normal string). This caused a page crash, and also wrong strings, as s-m's translations are really not expected to be used on the Overview page. This also speeds up the loading of frames, as they now don't have to go through umpteen `cockpit.locale()` calls any more. Note that this still leaves i18n conflicts in the shell itself, so this isn't a full fix, just an "80%" mitigation. Fixes #21486 --- pkg/shell/index.html | 2 +- src/cockpit/packages.py | 31 +++++++++++++++++-------------- test/pytest/test_packages.py | 2 +- 3 files changed, 19 insertions(+), 16 deletions(-) diff --git a/pkg/shell/index.html b/pkg/shell/index.html index 644a32a52723..895d7e5e0703 100644 --- a/pkg/shell/index.html +++ b/pkg/shell/index.html @@ -7,7 +7,7 @@ - + diff --git a/src/cockpit/packages.py b/src/cockpit/packages.py index 13fbe3e83967..3377ded69e0a 100644 --- a/src/cockpit/packages.py +++ b/src/cockpit/packages.py @@ -516,26 +516,27 @@ def reload_hint(self): self.reload() self.saw_first_reload_hint = True - def load_manifests_js(self, headers: JsonObject) -> Document: + def load_manifests_js(self, headers: JsonObject, *, i18n: bool) -> Document: logger.debug('Serving /manifests.js') chunks: List[bytes] = [] # Send the translations required for the manifest files, from each package - locales = parse_accept_language(get_str(headers, 'Accept-Language', '')) - for name, package in self.packages.items(): - if name in ['static', 'base1']: - continue + if i18n: + locales = parse_accept_language(get_str(headers, 'Accept-Language', '')) + for name, package in self.packages.items(): + if name in ['static', 'base1']: + continue - # find_translation will always find at least 'en' - translation = package.load_translation('po.manifest.js', locales) - with translation.data: - if translation.content_encoding == 'gzip': - data = gzip.decompress(translation.data.read()) - else: - data = translation.data.read() + # find_translation will always find at least 'en' + translation = package.load_translation('po.manifest.js', locales) + with translation.data: + if translation.content_encoding == 'gzip': + data = gzip.decompress(translation.data.read()) + else: + data = translation.data.read() - chunks.append(data) + chunks.append(data) chunks.append(b""" (function (root, data) { @@ -573,7 +574,9 @@ def load_path(self, path: str, headers: JsonObject) -> Document: if packagename is not None: return self.packages[packagename].load_path(filename, headers) elif filename == 'manifests.js': - return self.load_manifests_js(headers) + return self.load_manifests_js(headers, i18n=False) + elif filename == 'manifests-i18n.js': + return self.load_manifests_js(headers, i18n=True) elif filename == 'manifests.json': return self.load_manifests_json() else: diff --git a/test/pytest/test_packages.py b/test/pytest/test_packages.py index eaa8b6827dfb..e3aed57af9aa 100644 --- a/test/pytest/test_packages.py +++ b/test/pytest/test_packages.py @@ -233,7 +233,7 @@ def test_translation(pkgdir): assert document.data.read() == b'' # make sure the manifest translations get sent along with manifests.js - document = packages.load_path('/manifests.js', {'Accept-Language': 'de'}) + document = packages.load_path('/manifests-i18n.js', {'Accept-Language': 'de'}) contents = document.data.read() assert b'eins\n' in contents assert b'zwo\n' in contents