Skip to content

Locale containers

Motivation

As your product expands and starts to be used worldwide, it becomes essential to think about your audience and how to wide it. Localization can greatly improve the reach and impact of your application, making it more accessible for people in different countries.

So you decided to integrate it into your application.

You ended up creating a lang folder containing all translations...

lang/en.json
{
  "my_key_1": "My text",
  "my_key_2": "Wrong text"
}

...and a parser that reads and stores these files in a mapping.

1
2
3
4
5
6
import parser  # your handwritten parser

locales = parser.parse('/lang')
locale = locales.get('en')

print(locale.get('my_key_1'))  # My text

You can take any string you want by its key.

You may think "Sounds pretty simple"... and you'd be right.

This approach is pretty common and used in many applications and games (even Minecraft uses it).

But it's really error-prone. You can easily make a typo or refer to a different key with a similar name.

print(locale.get('my_key_I'))  # my_key_I
print(locale.get('my_key_2'))  # Wrong text

This becomes a real trouble once you change the schema of your translation files...

lang/en.json
{
  "my_text": "My text",
  "wrong_text": "Wrong text"
}
print(locale.get('my_key_1'))  # my_key_1

...or add new keys and forget about updating all the files.

lang/en.json
{
  "my_text": "My text", 
  "correct_text": "Correct text",
  "wrong_text": "Wrong text"
}
lang/de.json
{
  "my_text": "Mein Text",
  "wrong_text": "Falscher Text"
}
locale = locales.get('de')

print(locale.get('correct_text'))  # correct_text

What we just described here is extremely similar to the difference between statically typed and dynamically typed programming languages.

using System;

namespace Zero
{
    class Program
    {
        string myString = "0";
        static void Main(string[] args)
        {
            Console.WriteLine(myString);

            if (myString >= 0) {  // raises an error at compile time
                Console.WriteLine("What's kind of black magic is this?");
            }
        }
    }
}
1
2
3
4
5
my_string = '0'
print(my_string.lower())

if my_string >= 0:  # raises an error at runtime only when got to this statement (!!!)
    print("What's kind of black magic is this?")
1
2
3
4
5
6
let myString = "0";
console.log(myString.toLowerCase())

if (myString >= 0) {  // ...doesn't raise an error?!
    console.log("What's kind of black magic is this?")  // and in fact, this line also gets executed...
}
Fun fact about JavaScript

This is a valid JavaScript.

console.log("0" == -null)  // true
I don't even know anymore...

The thing is nowadays we have smart IDEs and code editors that can check for those errors and highlight them even for dynamically typed languages and even before we launched our program. That's why prototyping algorithms in languages like Python is much easier and faster now.

Sadly, we can't rely on IDE when it comes to checking what translation keys we are accessing as it would be really challenging to add this kind of functionality.

This is where the locale containers start to play.

The concept of locale container

my 300 billion dollar application localization spec:
- my_key_1
- my_key_2

The idea is to define a spec which determines what keys we can use in our application. This also creates a layer between translation files and our program.

That way we can clearly define what localization keys we have and what we haven't, which makes debugging it much simpler.

As an example, the spec allows us to check for undefined, unfilled or odd localization keys in the files.

It's really that simple.

Defining a locale container

In sl10n, locale containers are defined by subclassing SLocale:

1
2
3
4
5
6
7
8
from sl10n import SLocale


class MyLocale(SLocale):
    greetings_text: str
    main_menu_title: str
    success_text: str
    error_text: str

Note

All locale container strings, even multiline ones, have str type.

Then you need to reference it when defining an SL10n object:

1
2
3
4
5
6
7
8
9
from sl10n import SL10n, SLocale

class MyLocale(SLocale):
    greetings_text: str
    main_menu_title: str
    success_text: str
    error_text: str

sl10n = SL10n(MyLocale)

Warning

You can't use SLocale directly in SL10n. Treat it as an abstract dataclass.

That's it.