Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Support character sets. #50

Merged
merged 2 commits into from
Sep 6, 2021
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 3 additions & 0 deletions .meta.toml
Original file line number Diff line number Diff line change
Expand Up @@ -41,4 +41,7 @@ ignore-bad-ideas = [
"src/zope/i18n/tests/locale2/en/LC_MESSAGES/zope-i18n.mo",
"src/zope/i18n/tests/locale3/en/LC_MESSAGES/zope-i18n.mo",
"src/zope/i18n/tests/pl-default.mo",
"src/zope/i18n/tests/sr-default.mo",
"src/zope/i18n/tests/sr@Cyrl-default.mo",
"src/zope/i18n/tests/sr@Latn-default.mo",
]
8 changes: 7 additions & 1 deletion CHANGES.rst
Original file line number Diff line number Diff line change
Expand Up @@ -2,9 +2,15 @@
CHANGES
=========

4.7.1 (unreleased)
4.8.0 (unreleased)
==================

- Support character sets.
Example: ``sr@Latn`` and ``sr@Cyrl`` will be added to language ``sr`` (Serbian).
See https://github.com/collective/plone.app.locales/issues/326
You can choose which one to use by setting either ``sr@Latn`` or ``sr@Cyrl``
in environment variable ``zope_i18n_allowed_languages``.

- Support and test Python 3.8 and 3.9.
Full supported list is now: 2.7, 3.5, 3.6, 3.7, 3.8, 3.9, PyPy, PyPy3.

Expand Down
3 changes: 3 additions & 0 deletions setup.cfg
Original file line number Diff line number Diff line change
Expand Up @@ -21,3 +21,6 @@ ignore-bad-ideas =
src/zope/i18n/tests/locale2/en/LC_MESSAGES/zope-i18n.mo
src/zope/i18n/tests/locale3/en/LC_MESSAGES/zope-i18n.mo
src/zope/i18n/tests/pl-default.mo
src/zope/i18n/tests/sr-default.mo
src/zope/i18n/tests/sr@Cyrl-default.mo
src/zope/i18n/tests/sr@Latn-default.mo
2 changes: 1 addition & 1 deletion setup.py
Original file line number Diff line number Diff line change
Expand Up @@ -62,7 +62,7 @@ def alltests():

setup(
name='zope.i18n',
version='4.7.1.dev0',
version='4.8.0.dev0',
author='Zope Foundation and Contributors',
author_email='zope-dev@zope.org',
description='Zope Internationalization Support',
Expand Down
Binary file added src/zope/i18n/tests/sr-default.mo
Binary file not shown.
12 changes: 12 additions & 0 deletions src/zope/i18n/tests/sr-default.po
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
msgid ""
msgstr ""
"Project-Id-Version: Zope 5\n"
"PO-Revision-Date: 2021/09/02\n"
"Last-Translator: Zope 3 contributors\n"
"MIME-Version: 1.0\n"
"Content-Type: text/plain; charset=UTF-8\n"
"Content-Transfer-Encoding: 8bit\n"
"Plural-Forms: nplurals=2; plural=n != 1;\n"

msgid "short_greeting"
msgstr "Hello in Serbian Standard!"
Binary file added src/zope/i18n/tests/sr@Cyrl-default.mo
Binary file not shown.
12 changes: 12 additions & 0 deletions src/zope/i18n/tests/sr@Cyrl-default.po
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
msgid ""
msgstr ""
"Project-Id-Version: Zope 5\n"
"PO-Revision-Date: 2021/09/02\n"
"Last-Translator: Zope 3 contributors\n"
"MIME-Version: 1.0\n"
"Content-Type: text/plain; charset=UTF-8\n"
"Content-Transfer-Encoding: 8bit\n"
"Plural-Forms: nplurals=2; plural=n != 1;\n"

msgid "short_greeting"
msgstr "Hello in српски!"
Binary file added src/zope/i18n/tests/sr@Latn-default.mo
Binary file not shown.
12 changes: 12 additions & 0 deletions src/zope/i18n/tests/sr@Latn-default.po
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
msgid ""
msgstr ""
"Project-Id-Version: Zope 5\n"
"PO-Revision-Date: 2021/09/02\n"
"Last-Translator: Zope 3 contributors\n"
"MIME-Version: 1.0\n"
"Content-Type: text/plain; charset=UTF-8\n"
"Content-Transfer-Encoding: 8bit\n"
"Plural-Forms: nplurals=2; plural=n != 1;\n"

msgid "short_greeting"
msgstr "Hello in Serbian Latin!"
71 changes: 71 additions & 0 deletions src/zope/i18n/tests/test_translationdomain.py
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
# -*- coding: utf-8 -*-
##############################################################################
#
# Copyright (c) 2001-2008 Zope Foundation and Contributors.
Expand Down Expand Up @@ -197,3 +198,73 @@ def test_releoadCatalogs(self):

with self.assertRaises(KeyError):
self._domain.reloadCatalogs(('dne',))

def test_character_sets(self):
"""Test two character sets for the same language.

Serbian can be written in Latin or Cyrillic,
where Latin is currently most used.
Interestingly, every Latin character can actually be mapped to
one Cyrillic character, and the other way around,
so you could write a script to turn one po-file into the other.

But most practical is to have two locales with names
sr@Latn and sr@Cyrl. These are then two character sets
for the same language. So they should end up together
under the same language in a translation domain.

The best way for an integrator to choose which one to use,
is to add an environment variable 'zope_i18n_allowed_languages'
and let this contain either sr@Latn or sr@Cyrl.

See https://github.com/collective/plone.app.locales/issues/326
"""
standard_file = os.path.join(testdir, 'sr-default.mo')
latin_file = os.path.join(testdir, 'sr@Latn-default.mo')
cyrillic_file = os.path.join(testdir, 'sr@Cyrl-default.mo')
standard_catalog = GettextMessageCatalog('sr', 'char', standard_file)
latin_catalog = GettextMessageCatalog('sr@Latn', 'char', latin_file)
cyrillic_catalog = GettextMessageCatalog(
'sr@Cyrl', 'char', cyrillic_file
)

# Test the standard file.
domain = TranslationDomain('char')
domain.addCatalog(standard_catalog)
self.assertEqual(
domain.translate('short_greeting', target_language='sr'),
u"Hello in Serbian Standard!",
)

# Test the Latin file.
domain = TranslationDomain('char')
domain.addCatalog(latin_catalog)
self.assertEqual(
domain.translate('short_greeting', target_language='sr'),
u"Hello in Serbian Latin!",
)
# Note that sr@Latn is not recognizes as language id.
self.assertEqual(
domain.translate('short_greeting', target_language='sr@Latn'),
u"short_greeting",
)

# Test the Cyrillic file.
domain = TranslationDomain('char')
domain.addCatalog(cyrillic_catalog)
self.assertEqual(
domain.translate('short_greeting', target_language='sr'),
u"Hello in српски!",
)

# When I have all three locales, this is the order that
# os.listdir gives them in:
domain = TranslationDomain('char')
domain.addCatalog(latin_catalog)
domain.addCatalog(cyrillic_catalog)
domain.addCatalog(standard_catalog)
# The Latin one is first, so it wins.
self.assertEqual(
domain.translate('short_greeting', target_language='sr'),
u"Hello in Serbian Latin!",
)
5 changes: 5 additions & 0 deletions src/zope/i18n/translationdomain.py
Original file line number Diff line number Diff line change
Expand Up @@ -50,6 +50,11 @@ def __init__(self, domain, fallbacks=None):

def _registerMessageCatalog(self, language, catalog_name):
key = language
if "@" in key:
# sr@Latn and sr@Cyrl are two character set variants of
# the same Serbian language.
# See https://github.com/collective/plone.app.locales/issues/326
key = key.split("@")[0]
mc = self._catalogs.setdefault(key, [])
mc.append(catalog_name)

Expand Down