Skip to content

A robust internationalization solution for Rust featuring compile-time validation, ISO 639-1 compliance, and TOML-based translation management.

License

Apache-2.0, MIT licenses found

Licenses found

Apache-2.0
LICENSE-APACHE
MIT
LICENSE-MIT
Notifications You must be signed in to change notification settings

FlakySL/translatable.rs

Translatable 🌐🗣️💬🌍

Crates.io Docs.rs

A robust internationalization solution for Rust featuring compile-time validation, ISO 639-1 compliance, and TOML-based translation management.

This library prioritizes ergonomics over raw performance. Our goal is not to be blazingly fast but to provide the most user-friendly experience for implementing translations—whether you're a first-time user or an experienced developer. If you require maximum performance, consider alternative libraries, a custom implementation, or even hard-coded values on the stack.

Table of Contents 📖

Features 🚀

  • ISO 639-1 Standard: Full support for 180+ language codes/names
  • Compile-Time Safety: Macro-based translation validation
  • TOML Structure: Hierarchical translation files with nesting
  • Smart Error Messages: Context-aware suggestions
  • Template Validation: Balanced bracket checking
  • Flexible Loading: Configurable file processing order
  • Conflict Resolution: Choose between first or last match priority

Installation 📦

Run the following command in your project directory:

cargo add translatable

Usage 🛠️

Configuration

There are things you can configure on how translations are loaded from the folder, for this you should make a translatable.toml in the root of the project, and abide by the following configuration values.

Key Value type Description
path String Where the translation files will be stored, non translation files in that folder will cause errors.
seek_mode "alphabetical" | "unalphabetical" The found translations are ordered by file name, based on this field.
overlap "overwrite" | "ignore" Orderly if a translation is found "overwrite" will keep searching for translations and "ignore" will preserve the current one.

seek_mode and overlap only reverse the translations as convenient, this way the process doesn't get repeated every time a translation is loaded.

Translation file format

All the translation files are going to be loaded from the path specified in the configuration, all the files inside the path must be TOML files and sub folders, a walk_dir algorithm is used to load all the translations inside that folder.

The translation files have three rules

  • Objects (including top level) can only contain objects and strings
  • If an object contains another object, it can only contain other objects (known as nested object)
  • If an object contains a string, it can only contain other strings (known as translation object)

Loading translations

The load configuration such as seek_mode and overlap is not relevant here, as previously specified, these configuration values only get applied once by reversing the translations conveniently.

To load translations you make use of the translatable::translation macro, that macro requires at least two parameters to be passed.

The first parameter consists of the language which can be passed dynamically as a variable or an expression that resolves to an impl Into<String>, or statically as a &'static str literal. Not mattering the way it's passed, the translation must comply with the ISO 639-1 standard.

The second parameter consists of the path, which can be passed dynamically as a variable or an expression that resolves to an impl Into<String> with the format path.to.translation, or statically with the following syntax static path::to::translation.

The rest of parameters are meta-variable patterns also known as key = value parameters or key-value pairs, these are processed as replaces, or format if the call is all-static. When a template ({}) is found with the name of a key inside it gets replaced for whatever is the Display implementation of the value. This meaning that the value must always implement Display. Otherwise, if you want to have a {} inside your translation, you can escape it the same way format! does, by using {{}}. Just like object construction works in rust, if you have a parameter like x = x, you can shorten it to x.

Depending on whether the parameters are static or dynamic the macro will act different, differing whether the checks are compile-time or run-time, the following table is a macro behavior matrix.

Parameters Compile-Time checks Return type
static language + static path (most optimized) Path existence, Language validity &'static str (stack) if there are no templates or String (heap) if there are.
dynamic language + dynamic path None Result<String, TranslatableError> (heap)
static language + dynamic path Language validity Result<String, TranslatableError> (heap)
dynamic language + static path (commonly used) Path existence Result<String, TranslatableError> (heap)
  • For the error handling, if you want to integrate this with thiserror you can use a #[from] translatable::TranslationError, as a nested error, all the errors implement display, for optimization purposes there are not the same amount of errors with dynamic parameters than there are with static parameters.

  • The runtime errors implement a cause() method that returns a heap allocated String with the error reason, essentially the error display.

Example implementation 📂

The following examples are an example application structure for a possible real project.

Example application tree

project-root/
├── Cargo.toml
├── translatable.toml
├── translations/
│    └── app.toml
└── src/
     └── main.rs

Example translation file (translations/app.toml)

Notice how common.greeting has a template named name.

[welcome_message]
en = "Welcome to our app!"
es = "¡Bienvenido a nuestra aplicación!"

[common.greeting]
en = "Hello {name}!"
es = "¡Hola {name}!"

Example application usage

Notice how there is a template, this template is being replaced by the name = "john" key value pair passed as third parameter.

extern crate translatable;
use translatable::translation;

fn main() {
    let dynamic_lang = "es";
    let dynamic_path = "common.greeting"

    assert!(translation!("es", static common::greeting) == "¡Hola john!", name = "john");
    assert!(translation!("es", dynamic_path).unwrap() == "¡Hola john!".into(), name = "john");
    assert!(translation!(dynamic_lang, static common::greeting).unwrap() == "¡Hola john!".into(), name = "john");
    assert!(translation!(dynamic_lang, dynamic_path).unwrap() == "¡Hola john!".into(), name = "john");
}

License 📜

This repository is licensed under either of Apache License, Version 2.0 or MIT license at your option.
Unless you explicitly state any contribution intentionally submitted for inclusion in translatable by you, as defined in the Apache-2.0 license, shall be dual licensed as above, without any additional terms or conditions.

About

A robust internationalization solution for Rust featuring compile-time validation, ISO 639-1 compliance, and TOML-based translation management.

Topics

Resources

License

Apache-2.0, MIT licenses found

Licenses found

Apache-2.0
LICENSE-APACHE
MIT
LICENSE-MIT

Code of conduct

Security policy

Stars

Watchers

Forks

Releases

No releases published