use std::env;
use std::fs::File;
use std::io::Write;
use std::path::Path;
pub fn main() -> Result<(), Box<dyn std::error::Error>> {
let out_dir = env::var("OUT_DIR")?;
let mut embedded_file = File::create(Path::new(&out_dir).join("embedded_locales.rs"))?;
writeln!(embedded_file, "// Generated at compile time - do not edit")?;
writeln!(
embedded_file,
"// This file contains embedded English locale files"
)?;
writeln!(embedded_file)?;
writeln!(embedded_file)?;
writeln!(
embedded_file,
"pub fn get_embedded_locale(key: &str) -> Option<&'static str> {{"
)?;
writeln!(embedded_file, " match key {{")?;
let target_utility = detect_target_utility();
match target_utility {
Some(util_name) => {
embed_single_utility_locale(&mut embedded_file, &project_root()?, &util_name)?;
}
None => {
embed_all_utility_locales(&mut embedded_file, &project_root()?)?;
}
}
writeln!(embedded_file, " _ => None,")?;
writeln!(embedded_file, " }}")?;
writeln!(embedded_file, "}}")?;
embedded_file.flush()?;
Ok(())
}
fn project_root() -> Result<std::path::PathBuf, Box<dyn std::error::Error>> {
let manifest_dir = env::var("CARGO_MANIFEST_DIR")?;
let uucore_path = std::path::Path::new(&manifest_dir);
let project_root = uucore_path
.parent() .and_then(|p| p.parent()) .ok_or("Could not determine project root")?;
Ok(project_root.to_path_buf())
}
fn detect_target_utility() -> Option<String> {
use std::fs;
println!("cargo:rerun-if-env-changed=UUCORE_TARGET_UTIL");
if let Ok(target_util) = env::var("UUCORE_TARGET_UTIL") {
if !target_util.is_empty() {
return Some(target_util);
}
}
if let Ok(pkg_name) = env::var("CARGO_PKG_NAME") {
if let Some(util_name) = pkg_name.strip_prefix("uu_") {
println!("cargo:warning=Auto-detected utility name: {util_name}");
return Some(util_name.to_string());
}
}
if let Ok(target_dir) = env::var("CARGO_TARGET_DIR") {
let config_path = std::path::Path::new(&target_dir).join("uucore_target_util.txt");
if let Ok(content) = fs::read_to_string(&config_path) {
let util_name = content.trim();
if !util_name.is_empty() && util_name != "multicall" {
return Some(util_name.to_string());
}
}
}
if let Ok(project_root) = project_root() {
let config_path = project_root.join("target/uucore_target_util.txt");
if let Ok(content) = fs::read_to_string(&config_path) {
let util_name = content.trim();
if !util_name.is_empty() && util_name != "multicall" {
return Some(util_name.to_string());
}
}
}
None
}
fn embed_single_utility_locale(
embedded_file: &mut std::fs::File,
project_root: &Path,
util_name: &str,
) -> Result<(), Box<dyn std::error::Error>> {
use std::fs;
let locale_path = project_root
.join("src/uu")
.join(util_name)
.join("locales/en-US.ftl");
if locale_path.exists() {
let content = fs::read_to_string(&locale_path)?;
writeln!(embedded_file, " // Locale for {util_name}")?;
writeln!(
embedded_file,
" \"{util_name}/en-US.ftl\" => Some(r###\"{content}\"###),"
)?;
println!("cargo:rerun-if-changed={}", locale_path.display());
}
let uucore_locale_path = project_root.join("src/uucore/locales/en-US.ftl");
if uucore_locale_path.exists() {
let content = fs::read_to_string(&uucore_locale_path)?;
writeln!(embedded_file, " // Common uucore locale")?;
writeln!(
embedded_file,
" \"uucore/en-US.ftl\" => Some(r###\"{content}\"###),"
)?;
println!("cargo:rerun-if-changed={}", uucore_locale_path.display());
}
Ok(())
}
fn embed_all_utility_locales(
embedded_file: &mut std::fs::File,
project_root: &Path,
) -> Result<(), Box<dyn std::error::Error>> {
use std::fs;
let src_uu_dir = project_root.join("src/uu");
if !src_uu_dir.exists() {
embed_static_utility_locales(embedded_file)?;
return Ok(());
}
let mut util_dirs = Vec::new();
for entry in fs::read_dir(&src_uu_dir)? {
let entry = entry?;
if entry.file_type()?.is_dir() {
if let Some(dir_name) = entry.file_name().to_str() {
util_dirs.push(dir_name.to_string());
}
}
}
util_dirs.sort();
for util_name in &util_dirs {
let locale_path = src_uu_dir.join(util_name).join("locales/en-US.ftl");
if locale_path.exists() {
let content = fs::read_to_string(&locale_path)?;
writeln!(embedded_file, " // Locale for {util_name}")?;
writeln!(
embedded_file,
" \"{util_name}/en-US.ftl\" => Some(r###\"{content}\"###),"
)?;
println!("cargo:rerun-if-changed={}", locale_path.display());
}
}
let uucore_locale_path = project_root.join("src/uucore/locales/en-US.ftl");
if uucore_locale_path.exists() {
let content = fs::read_to_string(&uucore_locale_path)?;
writeln!(embedded_file, " // Common uucore locale")?;
writeln!(
embedded_file,
" \"uucore/en-US.ftl\" => Some(r###\"{content}\"###),"
)?;
println!("cargo:rerun-if-changed={}", uucore_locale_path.display());
}
embedded_file.flush()?;
Ok(())
}
fn embed_static_utility_locales(
embedded_file: &mut std::fs::File,
) -> Result<(), Box<dyn std::error::Error>> {
use std::env;
writeln!(
embedded_file,
" // Static utility locales for crates.io builds"
)?;
let manifest_dir = env::var("CARGO_MANIFEST_DIR").unwrap_or_default();
let Some(registry_dir) = Path::new(&manifest_dir).parent() else {
return Ok(()); };
let uucore_locale_file = Path::new(&manifest_dir).join("locales/en-US.ftl");
if uucore_locale_file.is_file() {
let content = std::fs::read_to_string(&uucore_locale_file)?;
writeln!(embedded_file, " // Common uucore locale")?;
writeln!(
embedded_file,
" \"uucore/en-US.ftl\" => Some(r###\"{content}\"###),"
)?;
}
let mut entries: Vec<_> = std::fs::read_dir(registry_dir)?
.filter_map(Result::ok)
.collect();
entries.sort_by_key(|e| e.file_name());
for entry in entries {
let file_name = entry.file_name();
if let Some(dir_name) = file_name.to_str() {
if let Some((util_part, _)) = dir_name.split_once('-') {
if let Some(util_name) = util_part.strip_prefix("uu_") {
let locale_file = entry.path().join("locales/en-US.ftl");
if locale_file.is_file() {
let content = std::fs::read_to_string(&locale_file)?;
writeln!(embedded_file, " // Locale for {util_name}")?;
writeln!(
embedded_file,
" \"{util_name}/en-US.ftl\" => Some(r###\"{content}\"###),"
)?;
}
}
}
}
}
Ok(())
}