i18n + pluralization library with multi-plural form support and thread safe for web apps
Project description
Pluralize
Pluralize is a Python library for Internationalization (i18n) and Pluralization.
The library assumes a folder (for exaple "translations") that contains files like:
it.json
it-IT.json
fr.json
fr-FR.json
(etc)
Each file has the following structure, for example for Italian (it.json):
{"dog": {"0": "no cane", "1": "un cane", "2": "{n} cani", "10": "tantissimi cani"}}
The top level keys are the expressions to be translated and the associated value/dictionary maps a number to a translation. Different translations correspond to different plural forms of the expression,
Here is another example for the word "bed" in Czech
{"bed": {"0": "no postel", "1": "postel", "2": "postele", "5": "postelí"}}
A translation value may also be a plain string when no pluralization is needed:
{"hello": "ciao"}
When loaded, plain-string values are normalized in memory to {"0": "..."} and a warning is logged so the file can be cleaned up. The dict form is preferred, but the string form is accepted so that translation files written by other i18n tools work without modification.
To translate and pluralize a string "dog" one simply wraps the string in the T operator as follows:
>>> from pluralize import Translator
>>> T = Translator('translations')
>>> dog = T("dog")
>>> print(dog)
dog
>>> T.select('it')
>>> print(dog)
un cane
>>> print(dog.format(n=0))
no cane
>>> print(dog.format(n=1))
un cane
>>> print(dog.format(n=5))
5 cani
>>> print(dog.format(n=20))
tantissimi cani
The string can contain multiple placeholders but the {n} placeholder is special because the variable called n is used to determine the pluralization by best match (max dict key <= n).
T(...) returns a lazyT object: the actual translation lookup is deferred until the value is rendered to a string. This means a lazyT can be created at import time and resolved later, after T.select(...) has chosen a language.
lazyT objects support:
- Concatenation with each other and with regular strings:
T("hello") + " " + T("world"). .format(**kwargs)to bind placeholder values, including the specialnfor pluralization:T("dog").format(n=5).- The
%operator with a dict, equivalent to.format(**d):T("route {num}") % {"num": 66}. With a non-dict argument,%falls back to standard string%formatting on the translated text (for backward compatibility). .xml(), which returns the translated string. It is provided for interoperability withyatlHTML helpers, which callxml()on embedded values.
T.select(s) can parse a string s following the HTTP Accept-Language header format (e.g. "fr-CH, fr;q=0.9, en;q=0.8, *;q=0.5") and picks the best available match from the loaded languages. Sub-tags are tried as fallbacks (e.g. fr-CH falls back to fr).
Constructor options
Translator(folder=None, encoding="utf-8", comment_marker=None)
folder: directory ofxx.json/xx-YY.jsonfiles to load. If omitted, no files are loaded andT.languagesstarts empty.encoding: text encoding used to read and write translation files. Defaults toutf-8.comment_marker: when set (e.g."##"), any text after this marker is stripped from the original (untranslated) string before it is returned. This lets you disambiguate identical source strings that need different translations, e.g.T("Open ##verb")andT("Open ##adjective")— when no translation is selected, the user sees"Open".
Missing translations
Every source string that is looked up but has no entry in the currently selected language is added to T.missing (a set). This is useful for finding gaps after running your app against a real workload.
Update the translation files
Find all strings wrapped in T(...) in .py, .html, and .js files:
matches = T.find_matches('path/to/app/folder')
Add newly discovered entries in all supported languages
T.update_languages(matches)
Add a new supported language (for example german, "de")
T.languages['de'] = {}
Make sure all languages contain the same origin expressions
known_expressions = set()
for language in T.languages.values():
for expression in language:
known_expressions.add(expression)
T.update_languages(known_expressions))
Finally save the changes:
T.save('translations')
save() writes one JSON file per loaded language, sorted by key and indented. Pass ensure_ascii=False to keep non-ASCII characters as-is in the output.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file pluralize-20260519.2.tar.gz.
File metadata
- Download URL: pluralize-20260519.2.tar.gz
- Upload date:
- Size: 6.7 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
4dc83fa6a8b3f2e279499ba2aace43c98231e828d38f4a03fb221349e0708182
|
|
| MD5 |
8d8a0a704d4fd593f135c74718b613a3
|
|
| BLAKE2b-256 |
671a00444cc68f6ed512f16ca51841d80c3c8b7fcc19b1ac3f43363eaf981c46
|
File details
Details for the file pluralize-20260519.2-py3-none-any.whl.
File metadata
- Download URL: pluralize-20260519.2-py3-none-any.whl
- Upload date:
- Size: 6.0 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
03b3c80946286554a78103f77e13e05eb027b8e24666d1147adca46c1646852a
|
|
| MD5 |
d440ba306b1ec7d6e3f02d426709ceb2
|
|
| BLAKE2b-256 |
7e4179d2f79a719dd4a94fe36d536801c195bca662015750f142ef0340fda2c2
|