use std::collections::HashMap;
use std::env;
use std::sync::{LazyLock, Mutex};
use log::{debug, warn};
use time_tz::{timezones, Tz};
const ENV_VAR: &str = "IBAPI_TIMEZONE_ALIASES";
const TIMEZONE_ALIASES: &[(&str, &str)] = &[
("中国标准时间", "Asia/Shanghai"),
("北京时间", "Asia/Shanghai"),
("China Standard Time", "Asia/Shanghai"),
("Greenwich Mean Time", "Europe/London"),
("GMT Standard Time", "Europe/London"),
("British Summer Time", "Europe/London"),
("SGT", "Asia/Singapore"),
("E. Europe Standard Time", "Europe/Bucharest"),
("Eastern European Standard Time", "Europe/Athens"),
("Eastern European Summer Time", "Europe/Athens"),
("FLE Standard Time", "Europe/Helsinki"),
("GTB Standard Time", "Europe/Athens"),
("Central European Standard Time", "Europe/Warsaw"),
("Central European Summer Time", "Europe/Warsaw"),
("W. Europe Standard Time", "Europe/Berlin"),
("Romance Standard Time", "Europe/Paris"),
];
static TIMEZONE_REGISTRY: LazyLock<Mutex<HashMap<String, String>>> = LazyLock::new(|| Mutex::new(seed_from_env()));
pub fn register_timezone_alias(name: impl Into<String>, iana: impl Into<String>) {
let name = name.into().trim().to_string();
let iana = iana.into().trim().to_string();
if name.is_empty() || iana.is_empty() {
warn!("register_timezone_alias: ignoring empty name or iana value");
return;
}
let mut registry = TIMEZONE_REGISTRY.lock().unwrap_or_else(|e| e.into_inner());
registry.insert(name, iana);
}
pub fn find_timezone(name: &str) -> Vec<&'static Tz> {
let mapped = map_timezone_name(name);
timezones::find_by_name(&mapped)
}
fn map_timezone_name(name: &str) -> String {
let registry = TIMEZONE_REGISTRY.lock().unwrap_or_else(|e| e.into_inner());
map_timezone_name_with(®istry, name)
}
fn map_timezone_name_with(registry: &HashMap<String, String>, name: &str) -> String {
if let Some(iana) = registry.get(name) {
debug!("timezone alias matched (registry): {name:?} -> {iana:?}");
return iana.clone();
}
for &(alias, iana) in TIMEZONE_ALIASES {
if name == alias {
return iana.to_string();
}
}
if name.contains('\u{FFFD}') {
return "Asia/Shanghai".to_string();
}
name.to_string()
}
fn seed_from_env() -> HashMap<String, String> {
match env::var(ENV_VAR) {
Ok(raw) => parse_env_aliases(&raw).into_iter().collect(),
Err(_) => HashMap::new(),
}
}
fn parse_env_aliases(raw: &str) -> Vec<(String, String)> {
let mut out = Vec::new();
for entry in raw.split(';') {
let entry = entry.trim();
if entry.is_empty() {
continue;
}
match entry.split_once('=') {
Some((name, iana)) => {
let name = name.trim();
let iana = iana.trim();
if name.is_empty() || iana.is_empty() {
warn!("ignoring malformed {ENV_VAR} entry: {entry:?}");
continue;
}
out.push((name.to_string(), iana.to_string()));
}
None => {
warn!("ignoring malformed {ENV_VAR} entry (missing '='): {entry:?}");
}
}
}
out
}
#[cfg(test)]
mod tests;