Simple and easy to use internationalizationlibrary inspired by Ruby i18n
Project description
PyI18n
PyI18n is GNU gettext free, simple and easy to use internationalization library for Python, inspired by Ruby i18n.
Documentation available at https://sectasy0.github.io/pyi18n.
Installation
You can install PyI18n via pip:
pip install pyi18n-v2
Getting Started
A few motivating and useful examples of how pyi18n can be used.
To use PyI18n in your application, you will first need to create a locales folder in the root directory of your project. Within this folder, you can create locale files in the format of your choice (e.g. YAML, JSON).
For example:
$ mkdir -p my_app/locales
$ touch my_app/locales/en.yml
$ touch my_app/locales/pl.yml
$ touch my_app/locales/de.yml
You can then create an instance of the PyI18n class, passing in the desired languages and, optionally, a custom locales directory.
from pyi18n import PyI18n
# default load_path is locales/
# you can change this path by specifying load_path parameter
i18n = PyI18n(("en", "pl", "de", "jp"), load_path="translations/")
_: callable = i18n.gettext
print(_("en", "hello.hello_user", user="John"))
#> Hello John!
print(_("pl", "hello.hello_user", user="John"))
#> Witaj John!
print(_("de", "hello.hello_user", user="John"))
#> Hallo John!
print(_("jp", "hello.hello_user", user="ジョンさん"))
#> こんにちは、ジョンさん!
Namespaces
PyI18n supports namespaces, which allows you to organize your translations into separate groups.
To use PyI18n with namespaces, you need to define the loader object yourself. Here's an example:
from pyi18n.loaders import PyI18nYamlLoader
from pyi18n import PyI18n
if __name__ == "__main__":
loader: PyI18nYamlLoader = PyI18nYamlLoader('locales/', namespaced=True)
pyi18n: PyI18n = PyI18n(('en_US', 'de_DE'), loader=loader)
In this example, we create an instance of the PyI18nYamlLoader class with the namespaced parameter set to True. This tells the loader to look for namespaced locales in separate folders instead of one single file for one locale.
Here's an example of the expected file structure for the locales:
locales
en_US
common.yml
analysis.yml
de_DE
common.yml
analysis.yml
To get a key that is located in the common namespace, you should use the dot notation in your translation call:
_(locale, 'common.greetings')
Integrate pyi18n with Django project
To integrate pyi18n into your Django project, you will need to first add a locale field to your user model class. This field will store the user's preferred language, which will be used to retrieve the appropriate translations from the locales directory.
Next, you will need to configure pyi18n in your settings.py file by creating an instance of the PyI18n class and specifying the available languages. You can also create a gettext function for ease of use.
In your views, you can then use the gettext function to retrieve translations based on the user's preferred language. To use translations in templates, you will need to create a custom template tag that utilizes the gettext function.
settings.py
from pyi18n import PyI18n
i18n: PyI18n = PyI18n(['pl', 'en'])
_: callable = i18n.gettext
views.py
from mysite.settings import _
def index(request):
translated: str = _(request.user.locale, 'hello', name="John")
return HttpResponse(f"This is an example view. {translated}")
register template tag
from django import template
from mysite.settings import _
register = template.Library()
@register.simple_tag
def translate(locale: str, path: str, **kwargs):
return _(locale, path, **kwargs)
usage in templates
NOTE: Wrap this tag inside jinja2 special characters
translate request.current_user.locale, "hello", name="John"
That's it, you have now successfully installed and configured PyI18n for your project. You can now use the provided gettext function to easily retrieve translations based on the user's preferred language. Additionally, you can use the provided template tag to easily retrieve translations in your templates. And if you need to use custom loaders you can use the PyI18nBaseLoader to create your own loaders.
Creating custom loader class
To create custom locale loader you have to create a class which will inherit from PyI18nBaseLoader and override load
method with all required parameters (see below). You can see an example of custom locale loader in examples/custom_xml_loader.py
.
from pyi18n.loaders import PyI18nBaseLoader
class MyCustomLoader(PyI18nBaseLoader):
def load(self, locales: tuple, load_path: str):
# load_path is the path where your loader will look for locales files
# locales is a tuple of locales which will be loaded
# return a dictionary with locale data
...your custom loader logic...
return {}
Then pass your custom loader to PyI18n class.
from pyi18n.loaders import PyI18nBaseLoader
class MyCustomLoader(PyI18nBaseLoader):
def load(self, locales: tuple, load_path: str):
# load_path is the path where your loader will look for locales files
# locales is a tuple of locales which will be loaded
...your custom loader logic...
# have to return a dictionary
return {}
# don't use load_path in `PyI18n` constructor, if not using default yaml loader
if __name__ == "__main__":
load_path: str = "locales/"
loader: PyI18nBaseLoader = MyCustomLoader(load_path=load_path)
i18n: PyI18n = PyI18n(("en",), loader=loader)
_: callable = i18n.gettext
print(_("en", "hello.hello_user", user="John"))
#> Hello John!
Tasks
Tasks usage
$ pyi18n-tasks
usage: pyi18n-tasks [-h] [-p PATH] normalize
pyi18n-tasks: error: the following arguments are required: normalize
Normalization
Normalization process will sort locales alphabetically. The default normalization path is locales/
, you can change it by passing -p
argument.
$ pyi18n-tasks normalize
$ pyi18n-tasks normalize -p my_app/locales/
Run tests
python3 tests/run_tests.py
For any questions and suggestions or bugs please create an issue.
Limitations
- Normalization task will not work for custom loader classes except xml, cause it's based on loader type field ( If you have an idea how to solve this differently please open the issue with a description ), if you need that use one of build in loaders or user XML loader from example.
Roadmap
See issues, If I have enough time and come up with a good idea on how this package can be improved, I'll post it there, along with tip.
Release History
Release History available at https://sectasy0.github.io/pyi18n/home/release-history/.
Contributing
- Fork it (https://github.com/sectasy0/pyi18n)
- Create your feature branch (
git checkout -b feature/fooBar
) - Commit your changes (
git commit -am 'feat: Add some fooBar'
) - Push to the branch (
git push origin feature/fooBar
) - Create a new Pull Request
Project details
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
File details
Details for the file pyi18n-v2-1.2.2.tar.gz
.
File metadata
- Download URL: pyi18n-v2-1.2.2.tar.gz
- Upload date:
- Size: 17.6 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/3.1.1 pkginfo/1.4.2 requests/2.22.0 setuptools/45.2.0 requests-toolbelt/0.8.0 tqdm/4.30.0 CPython/3.8.10
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | 96919717c933f6cfa4abfd496ce9f6d0ccad0ac74361475883b2e236fc5c14e7 |
|
MD5 | 4add4ebae31e6386bbe7d75f015e09f5 |
|
BLAKE2b-256 | 45c9b7da06f31595185aa01f1b2ad4d0bc187f499278cc98b5b29bc5d5f6ce61 |
File details
Details for the file pyi18n_v2-1.2.2-py3-none-any.whl
.
File metadata
- Download URL: pyi18n_v2-1.2.2-py3-none-any.whl
- Upload date:
- Size: 19.0 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/3.1.1 pkginfo/1.4.2 requests/2.22.0 setuptools/45.2.0 requests-toolbelt/0.8.0 tqdm/4.30.0 CPython/3.8.10
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | 1a5f64f4d1fde0a559876288849893384ac0ef25ba8df06ff7eac25ac368e556 |
|
MD5 | 01e482215664cadab3f73d32cc94f0cc |
|
BLAKE2b-256 | 706162315bfda77798c049e2b7d3ea9182b93597ef7804460659d84fd131f661 |