mirror of
https://github.com/zaaarf/fluent-fluently.git
synced 2024-11-22 03:14:48 +01:00
feat: initial, basic implementation
This commit is contained in:
commit
0a759cfc29
6 changed files with 133 additions and 0 deletions
6
.editorconfig
Normal file
6
.editorconfig
Normal file
|
@ -0,0 +1,6 @@
|
||||||
|
[*]
|
||||||
|
end_of_line = lf
|
||||||
|
insert_final_newline = true
|
||||||
|
charset = utf-8
|
||||||
|
indent_style = tab
|
||||||
|
indent_size = 2
|
2
.gitignore
vendored
Normal file
2
.gitignore
vendored
Normal file
|
@ -0,0 +1,2 @@
|
||||||
|
/target
|
||||||
|
Cargo.lock
|
11
Cargo.toml
Normal file
11
Cargo.toml
Normal file
|
@ -0,0 +1,11 @@
|
||||||
|
[package]
|
||||||
|
name = "fluent-fluently"
|
||||||
|
version = "0.1.0"
|
||||||
|
edition = "2021"
|
||||||
|
|
||||||
|
[dependencies]
|
||||||
|
fluent = "0.16.0"
|
||||||
|
fluent-syntax = "0.11.0"
|
||||||
|
intl-memoizer = "0.5.1"
|
||||||
|
unic-langid = "0.9.4"
|
||||||
|
walkdir = "2.4.0"
|
2
README.md
Normal file
2
README.md
Normal file
|
@ -0,0 +1,2 @@
|
||||||
|
# Fluent, fluently
|
||||||
|
I found myself needing to do localisation, one time. I decided I'd want it to happen at runtime, so that any spelling fixes may be applied without needing to recompile the whole program. [Fluent](https://github.com/projectfluent/fluent-rs) provides an API, but no real way to use it. I couldn't find any existing implementation fitting my use case, so I wrote this.
|
33
src/error.rs
Normal file
33
src/error.rs
Normal file
|
@ -0,0 +1,33 @@
|
||||||
|
use std::result::Result as StdResult;
|
||||||
|
|
||||||
|
use fluent::FluentResource;
|
||||||
|
|
||||||
|
pub type Result<T> = StdResult<T, Error>;
|
||||||
|
|
||||||
|
pub enum Error {
|
||||||
|
IoError(String)
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<std::io::Error> for Error {
|
||||||
|
fn from(value: std::io::Error) -> Self {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<(FluentResource, Vec<fluent_syntax::parser::ParserError>)> for Error {
|
||||||
|
fn from(value: (FluentResource, Vec<fluent_syntax::parser::ParserError>)) -> Self {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<Vec<fluent::FluentError>> for Error {
|
||||||
|
fn from(value: Vec<fluent::FluentError>) -> Self {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<unic_langid::LanguageIdentifierError> for Error {
|
||||||
|
fn from(value: unic_langid::LanguageIdentifierError) -> Self {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
}
|
79
src/lib.rs
Normal file
79
src/lib.rs
Normal file
|
@ -0,0 +1,79 @@
|
||||||
|
use std::{collections::HashMap, sync::Arc};
|
||||||
|
use fluent::{bundle::FluentBundle, FluentResource};
|
||||||
|
use intl_memoizer::concurrent::IntlLangMemoizer;
|
||||||
|
use unic_langid::LanguageIdentifier;
|
||||||
|
use crate::error::Result;
|
||||||
|
|
||||||
|
pub mod error;
|
||||||
|
|
||||||
|
type TypedFluentBundle = FluentBundle<Arc<FluentResource>, IntlLangMemoizer>;
|
||||||
|
pub struct Localiser {
|
||||||
|
bundles: HashMap<LanguageIdentifier, TypedFluentBundle>,
|
||||||
|
default_language: LanguageIdentifier
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Localiser {
|
||||||
|
pub fn try_load(path: String, default_language: String) -> Result<Self> {
|
||||||
|
let mut bundles = HashMap::new();
|
||||||
|
let paths = std::fs::read_dir(path)?
|
||||||
|
.filter_map(|res| res.ok())
|
||||||
|
.map(|dir_entry| dir_entry.path())
|
||||||
|
.filter_map(|path| {
|
||||||
|
if path.extension().map_or(false, |ext| ext == "ftl") || path.is_dir() {
|
||||||
|
Some(path)
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
}).collect::<Vec<_>>();
|
||||||
|
|
||||||
|
//TODO load default first and call bundle.add_resource_overriding(default_bundle) on others
|
||||||
|
let default_language = default_language.parse::<LanguageIdentifier>()?;
|
||||||
|
|
||||||
|
for path in paths {
|
||||||
|
// validate filename as language code
|
||||||
|
let language_code = path.file_stem()
|
||||||
|
.and_then(|f| f.to_str())
|
||||||
|
.map(|f| f.parse::<LanguageIdentifier>())
|
||||||
|
.and_then(|id| match id {
|
||||||
|
Ok(id) => Some(id),
|
||||||
|
Err(_) => None
|
||||||
|
});
|
||||||
|
|
||||||
|
if language_code.is_none() {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
let language_code = language_code.unwrap();
|
||||||
|
|
||||||
|
let mut bundle: TypedFluentBundle = fluent::bundle::FluentBundle::new_concurrent(vec![language_code.clone()]);
|
||||||
|
if path.is_dir() { //is a directory
|
||||||
|
for res in Self::path_to_resources(&path)? {
|
||||||
|
bundle.add_resource(res)?;
|
||||||
|
}
|
||||||
|
} else { //is a single file
|
||||||
|
bundle.add_resource(Self::file_to_resource(&path)?)?;
|
||||||
|
}
|
||||||
|
|
||||||
|
bundles.insert(language_code, bundle);
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(Self {
|
||||||
|
bundles,
|
||||||
|
default_language
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
fn path_to_resources(path: &std::path::PathBuf) -> Result<Vec<Arc<FluentResource>>> {
|
||||||
|
let mut res = Vec::new();
|
||||||
|
for entry in walkdir::WalkDir::new(path).follow_links(true).into_iter().filter_map(|e| e.ok()) {
|
||||||
|
let entry_path = entry.path().to_path_buf();
|
||||||
|
res.push(Self::file_to_resource(&entry_path)?);
|
||||||
|
}
|
||||||
|
Ok(res)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn file_to_resource(path: &std::path::PathBuf) -> Result<Arc<FluentResource>> {
|
||||||
|
let content = std::fs::read_to_string(path)?;
|
||||||
|
Ok(Arc::new(FluentResource::try_new(content)?))
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in a new issue