Skip to content

Internationalization

pidgezero_one edited this page Jun 25, 2024 · 3 revisions

i18n Contributor's Guide

I18n pages allow for the translation of content to various languages, enabling users to access localized versions of a webpage based on their locale preferences. For instance, when a user accesses https://openlibrary.org/subjects, they are redirected to https://openlibrary.org/subjects.en or https://openlibrary.org/subjects.es, depending on their selected language. Any text that is visible to the patron should be internationalized. The basics of web.py's templator I18N support is described here: http://webpy.org/cookbook/i18n_support_in_template_file

To get started:

  1. Please kindly reach out to us via the volunteer page https://openlibrary.org/volunteer#translator
  2. Watch this overview:

archive org_details_openlibrary-tour-2020_openlibrary-i18n mp4

Open Library i18n is handled via the python Babel library, GNU gettext, and the message lists located https://github.com/internetarchive/openlibrary/tree/master/openlibrary/i18n

The messages file format used by the gettext toolset is described here, and in the gettext manual.

In case you want to get started here are the following steps:

Contributing translations

Option 1. Locate the right target language within the project (e.g. es for Spanish) and then click on the po file (the raw file where translation strings are contributed), e.g. this one for Spanish. Click on the pencil (edit) option which will bring you to an editable page like this where you can add or edit translations. When you're satisfied with your translations, scroll down to the bottom of the page where it shows Commit Changes, leave a description of your changes and make sure to select the radio button of Create a new branch. You can call "translations-es" or dash whatever language you're working with. Then, click Propose Changes button and you're done! We can follow up if there are any validation issues which may need to be addressed.

Option 2. If you prefer working with git you may instead fork / clone the repository from Github. Install git and follow the instructions on our Git Cheat Sheet to get set up.

Adding a new language

If you are starting a new language translation, copy the template to the correct place in the directory hierarchy, add the plural forms info at the top and replace the English version of the msgstr text values with the translated versions for your language. A new directory containing a translation template file must be created for each new language. These can be automatically generated if your Docker environment is set up (see our Docker README), or created manually.

Before creating the new directory, you will need to know your language's two-letter ISO 639-1 code. Make a note of the code once you have found it here: https://en.wikipedia.org/wiki/List_of_ISO_639-1_codes

If you find something which can't be translated correctly (perhaps because the text is being concatenated in the code rather than in the message formatting), please create an issue describing the location and what the problem is.

Adding a language in a Docker environment

  1. Run docker compose up -d

  2. Run docker compose exec -uroot web ./scripts/i18n-messages add [CODE], replacing [CODE] with your two-letter ISO 639-1 code.

Adding a language manually

  1. Create a new folder for your language. Create a new folder in /openlibrary/i18n/, using your two-letter ISO 639-1 code as the folder's name.

  2. Make a copy of the latest messages to translate. The messages template file, /openlibrary/i18n/messages.pot should be copied as messages.po (note the difference in extension, the t for template is dropped for the copy) to your newly created folder.

Updating UI to include new language

In order for a new language option to be available in our language drop-down and footer, the language_list.html template must be updated to include the new language. An Open Library staff member can do this if you are unfamiliar with HTML.

Submitting translations

You can edit the message.po file using your favourite editor, or a .po specific tool such as poedit, and send in a Pull Request. Pull Request Guidelines can be found on our CONTRIBUTING guide and our Git Cheat Sheet.

Viewing and testing your changes

In order to open your language version of the website in the browser, you will need to setup your docker environment (see our Docker README). After having run docker compose up -d, run docker compose run --rm -uroot home make i18n to build the translation files; then e.g. http://localhost:8080/?lang=fr should work. To view production Open Library in a preferred language, you will need to adjust your browser language preferences. You can also use the lang= parameter on the URL with a two character language code, e.g. https://openlibrary.org/?lang=fr

Updating an existing language

If changes have been made to the .pot file, to reflect those changes to a given language you need to merge the two files. After setting up your docker environment (see our Docker README, run the following, replacing [CODE] with your two-letter language code:

docker compose run --rm -uroot home ./scripts/i18n-messages update [CODE]

If you don't have a Docker set-up, you can also use Poedit to merge PO and POT via 'Catalog -> Update from POT file' and use it to review and translate changed and missing strings.

See our i18n guideline in the wiki for important and useful tips.

If you are updating an existing translation, run scripts/i18n-messages update to merge the new msgids from the i18n/messages.pot message templates into your i18n/<lang>/messages.po message catalog. Review all fuzzy matches and either remove the fuzzy keyword, if a correct match, or update and remove the fuzzy keyword. Make sure it's an exact match before removing the fuzzy label. Sometimes there are minor, but important changes like datatype changes, e.g. %(count)s to %(count)d. Also review all entries with an empty msgstr and add correct translations for them.

If the text was revised and the update/matching algorithm didn't think it was a close enough match to even label a fuzzy match, you may find it at the bottom of the file in the section commented out with tildes (~). Any text in that section which is not useful to reuse or save, can be deleted.

Remember:

  • keep the substitution variable names and data types unchanged in your translated text (e.g. %(count)%s)
  • don't translate embedded HTML markup (e.g. <b>, <a href=>, etc)
  • do escape any embedded quotes (e.g. \")

Validating your translations

Before submitting a PR with your translations, we recommend correcting any validation errors identified by the following script (replace [CODE] with your language code):

docker compose exec -uroot web ./scripts/i18n-messages validate [CODE]

Extracting strings from HTML/python files (generating the .pot file)

To add i18n support to Open Library, templates and macros are modified to use gettext function calls. For brevity, the gettext function is abbreviated as:

<a href="..">$_("More search options")</a>

The messages in the the templates and macros are extracted and .pot file is created. After setting up your docker environment (see our Docker README, run:

docker compose run --rm -uroot home ./scripts/i18n-messages extract

The .pot file contains a msgid and a msgstr for each translation used. The msgstr field for each entry is filled with the translation of the required language and that file is placed at openlibrary/i18n/$locale/messages.po:

mkdir openlibrary/i18n/te
cp openlibrary/i18n/messages.pot openlibrary/i18n/te/messages.po
# edit openlibrary/i18n/te/messages.po and fill the translations

The .po files are compiled to .mo files to be able to use them by gettext system. This is done by make i18n automatically when the code is deployed, but needs to be done manually by a maintainer when deploying to dev.openlibrary.org .

Glossary

  • .po - Portable object file: This is the file where you will translators will add translations to.
  • .pot - Portable object template file: This is the file that lists all the strings in Open Library before translation.
  • .mo - Machine object file: This file is generated by make i18n, and is what is used by the actual site.

Deprecated messages that need updating to the new babel/gettext method:

The codebase has deprecated translations in the /openlibrary/i18n directory. In directories of older translations, the messages.po file will be replaced with a legacy-strings.{ISO 639-1 Code}.yml file.

Languages with deprecated translations:

  • hi Hindi
  • kn Kannada
  • mr Marathi
  • nl Dutch

Internationalizing Infogami Pages

@cdrini has a script to help automate the creation if i18n versions of openlibrary.org infogami pages (e.g. https://openlibrary.org/about v. https://openlibrary.org/about.es):

https://gist.github.com/cdrini/615d75653e1e47115930fa394e83ab17

Internationalization (i18n) Developer's Guide

Any text that will be visible to the user should be internationalized. The basics of web.py's templetor i18n support is described here.

The two primary i18n message functions are:

  • gettext() which is bound to '_' as a convenience since it's commonly used
  • ngettext() (or ungettext() as we've historically used) which currently needs to be spelled out, but is commonly bound to N_, so that's a convention we may adopt.

HTML Files

Text Type Example Text Example Syntax Notes
Plain inner HTML <p>About this book:</p> <p>$_("About this book:")</p> Wrap it in $_(" ")
title or alt attributes <button title="Submit" alt="Submit button"> <button title="$_('Submit')" alt="$_('Submit button')"> Wrap it in $_(' ') - note the use of single quotes
value attributes <button value="Submit" /> or <input value="Enter your name" <button value="$_('Submit')" /> & <input value="$_('Enter your name...')" Note: You only need to use the i18n syntax if the value is being used to set visible placeholder text -- values like this one can be left as is: <button value="add">$_("Add")</button>
Text that contains ' or " title="That's strange!" or <p>Click "Add"</p> title="$_('That\'s strange!')" or <p>$_("Click \"Add\"")</p> If there's a conflict between the outer and inner quotes, i.e. ' or "" inside "$_(' ')" or "" inside $_(" "), escape the inner quote/apostrophe with a \
Text that contains a variable title="Photo of $author_name" title="$_('Photo of %(author)s', author=author_name) Wrap a stand-in for the variable in %( )s if the variable is a string, or %( )d if the variable is a number, and then assign your stand-in to the initial variable; Note: It's important to use a meaningful stand-in name here, like "count" or "author" to give translators context
Text that contains nested HTML tags <p>Would you like to <strong>return</strong>?</p> <p>$:_('Would you like to <strong>return</strong>?')</p> Wrap each full sentence and punctuation in $:_(' ') -- note the : and single quotes*
Text that contains a plural ("There is one person waiting for this book.", "There are {wlsize} people waiting for this book.") $ungettext("There is %(count)d person waiting for this book.", "There are %(count)d people waiting for this book.", wlsize, count=wlsize) Write out ungettext instead of the traditional _ and sub in the variable (see variable instructions)**

Everything enclosed within any version of the $_(' ') syntax will be extracted into messages.pot file. Review the messages.pot file to get an idea of what the extracted messages look like.

When syntaxing messages that are split up by nested HTML tags, consider if the message can be translated in chunks, or if the meaning of the chunks depends on the sentence as a whole. If it can be broken into chunks, it will save the translators the effort (and potential risk) of copy & pasting HTML. However, translation accuracy is the priority.

Note:

  • There's no need to do nested i18n, i.e. the title in $:_('Click <a href="$link" title="External Link">Here</a>') would already be translated with the rest of the string.
  • Be sure to include punctuation marks inside parentheses, i.e. the . at the end of >$:_(' Please read our <a href="/https/github.com/help/faq">FAQ</a>.')
  • Exclude HTML syntax where possible to avoid unnecessary copy and pasting for translators, i.e. you can leave out the &nbsp; here like so: &nbsp;&nbsp;$_("Add some subjects")
  • But if you include an HTML closing tag, be sure to include the opening tag as well, i.e. $:_('<strong>Download the DAISY.zip</strong>.)
  • While working, avoid moving extra spaces to the beginning or end <a></a> tags, which will cause weird links that look like this
  • You may run across some untranslated HTML in the JavaScript test files; this is fine, and should be left as is

Enforcement

If your pull request includes new or changed HTML files that don't follow the above guidelines, these changes will be flagged by the detect_missing_i18n pre-commit script and will fail CI. The CI output may be truncated, so you are encouraged to install the pre-commit framework and run pre-commit run detect-missing-i18n to see which lines in your changes are missing i18n.

If the i18n error or warning is a false positive, you may append $# detect-missing-i18n-skip-line to the end of your line (example). You may also insert a line above the detected error that only contains $# detect-missing-i18n-skip-line (example).

Do not add your file to the EXCLUDE_LIST array within the detect_missing_i18n script. This list is intended only for files that had existing errors at the time the pre-commit check was created, and these errors will be addressed and taken out of the list over time.


* You should try to avoid this where possible because it requires the translator to copy the HTML exactly—but sometimes you can't avoid it. Note you should not split up the sentence; it might not make sense in other languages.

These sentences however can be represented without i18n-ing the HTML by using python template strings:

$def cc0_link():
  <a href="https://creativecommons.org/publicdomain/zero/1.0/" target="_blank" title="$_('This link to Creative Commons will open in a new window')">$_('CC0')</a>

$:(_('By saving a change to this wiki, you agree that your contribution is given freely to the world under %s. Yippee!') % str(cc0_link()).strip())

In this way, only the text is presented to the translators.

** In the translation file, this would look like:

#: borrow.html:114
#, python-format
msgid "There is %(count)d person waiting for this book."
msgid_plural "There are %(count)d people waiting for this book."
msgstr[0] "%(count)d personne attend ce livre."
msgstr[1] "%(count)d personnes attendent ce livre."

The top of the file declares the number of different plural forms for the language since this varies widely among languages. There is more information on plural forms support here.

Python Files

  • Follow the HTML instructions above, but omit the $. E.g.
from openlibrary.i18n import gettext as _, ungettext

_('My translated string!')

Note: When adding Python translations, you may encounter an AttributeError like 'ThreadedDict' object has no attribute 'lang'. This can happen if translated text is at the top of the file rather than nested in a function, as the gettext function _ requires the language context (web.ctx.lang) to be ready before it runs.

A simple workaround for this problem is nesting the desired text inside a function, either directly where it will be used or as a helper function near the top of the file.

For instance, a dict like this:

LOGIN_ERRORS = {
  "invalid_email": _("The email address you entered is invalid"),
  "account_blocked": _("This account has been blocked"),
  "account_locked": _("This account has been locked"),
  ...
}

can be made translation-safe if nested in a function like so:

def get_login_error(error_key):
    LOGIN_ERRORS = {
        "invalid_email": _("The email address you entered is invalid"),
        "account_blocked": _("This account has been blocked"),
        "account_locked": _("This account has been locked"),
        ...
    }

    return LOGIN_ERRORS[error_key]

JavaScript Files

Unlike our Python code and HTML templates, there is currently no way to extract strings from our Javascript code for localization. To bypass this limitation, we have been including localized strings in a data-i18n attribute.

As an example, the following code will display the localized Hello World message in the .greeting-example span when the greeting.html template is rendered on a page.

In the greeting.html template, we first create a dict that will contain all of the localized UI strings that will be used by the client-side code. In this case, we only have a single string. This dict is then added to the root element of the template in a data-i18n attribute. The json_encode call simply converts a Python dict to a JSON string.

/openlibrary/templates/greeting.html:

$def with ()

$ i18n_strings = {
$   'greeting': _('Hello World')
$ }

<span class="greeting-example" data-i18n="$json_encode(i18n_strings)"></span>

In the main index.js file, we add code that initializes the .greeting-example element if it exists on the page.

js/index.js:

const greetingElement = document.querySelector('.greeting-example')
if (greetingElement) {
    import('greeting')
        .then((module) => module.initGreeting(greetingElement))
}

In initGreeting, we parse the value of data-i18n and set the textContent of the greeting span to the localized "Hello World" string.

js/greeting.js:

/**
 * Sets text content of the given element to the localized greeting.
 *
 * @param {HTMLElement} greetingElement
 */
export function initGreeting(greetingElement) {
    const i18nStrings = JSON.parse(greetingElement.dataset.i18n)
    greetingElement.textContent = i18nStrings.greeting
}

Best Practices

Must DOs:

  • Internationalize all user visible strings, including HTML title and alt which are used
  • Use consistent terminology and phrasing throughout the UI to reduce the amount of text which needs to be translated
  • Use meaningful mnemonic parameter names to help the translators understand the context. e.g. "%(editioncount)d editions"
  • Double check that the format string types match the types of parameters being passed. Mismatches will cause errors at runtime.
  • Be sure to escape embedded quotes and apostrophes, if necessary, after wrapping strings in single/double quotes. e.g. "_$('Mustn't forget to escape')"
  • Be sure to remove extra dollar signs ($) when wrapping expressions with $_()
  • Generate a message catalog template (messages.pot) when user visible strings are added or change
  • Make changes incrementally in small batches. Multiple preprocessors (templator + babel) can make error messages obscure, so it's easier to debug if you know what changed.

DON'Ts:

  • Don't display internal status / keyword values from the code directly to the user. These can't be internationalized.
  • Don't do pluralization or string concatenation in code or templates. This mandates ordering in ways that can't be translated. Give the translators completed sentences or phrases, with embedded replacements, to work with so they can create natural translations.
  • Don't use inline styling or links in text, if at all possible. e.g. <em>, <a href=foo>
  • Don't update the translated message catalogs. Because the merging process is inexact, it's better for the translators to handle this so that they can validate the results. Do update the message templates though (ie messages.pot)
  • Don't hard code in 1 for singular nouns (e.g. 1 edition) because in some languages, 0 editions is singular and translated as 0 edition. Instead, substitute a variable, as you would with the plural. E.g. $ungettext("There is %(count)d person waiting for this book.", "There are %(count)d people waiting for this book.", wlsize, count=wlsize) and not $ungettext("There is 1 person waiting for this book.", "There are %(count)d people waiting for this book.", wlsize, count=wlsize)

Internationalization (i18n) Page Conversion Guide

Internationalization (i18n) pages are specialized pages within the Infogami platform that enable users to contribute translations. This guide outlines the process of converting a standard page into an i18n page, using the example of converting https://openlibrary.org/librarians to its English (en) i18n version.

Follow these steps to convert a standard page into an i18n page. In this example, we'll use https://openlibrary.org/librarians as the source page to be converted into its English (en) i18n version.

Prerequisites

  1. Administrator permissions on the Infogami platform.

Steps

  1. Create the English (en) Version of the Page
  2. Edit the Un-Suffixed Page
  3. Test

This concludes the process of converting a standard page into an i18n page, making it accessible in multiple languages. Ensure that you have made the necessary updates to the key and type fields as specified above.

Clone this wiki locally