mirror of
https://github.com/shuaiplus/nodewarden.git
synced 2026-06-20 13:00:39 +00:00
7c58282e42
- Updated AuthViews component to conditionally show invite code field based on registrationInviteRequired prop. - Enhanced error handling in auth API functions to use translateServerError for better user feedback. - Added new translations for various server error messages in English, Spanish, Russian, Chinese (Simplified and Traditional). - Modified demo initial bootstrap state to include registrationInviteRequired flag. - Updated types to include registrationInviteRequired in WebBootstrapResponse.
152 lines
5.6 KiB
TypeScript
152 lines
5.6 KiB
TypeScript
// CONTRACT:
|
|
// Locale bundles are standalone and loaded on demand. Adding a locale requires
|
|
// updating Locale, AVAILABLE_LOCALES, browser-language detection, localeLoaders,
|
|
// scripts/i18n-utils.cjs, and the locale file itself.
|
|
//
|
|
// Do not call t() at module scope for exported arrays/constants; async init can
|
|
// otherwise leave raw txt_* keys in the rendered UI.
|
|
export type Locale =
|
|
| 'en'
|
|
| 'zh-CN'
|
|
| 'zh-TW'
|
|
| 'ru'
|
|
| 'es';
|
|
|
|
import enMessages from './i18n/locales/en';
|
|
|
|
const LOCALE_STORAGE_KEY = 'nodewarden.locale';
|
|
|
|
type MessageTable = Record<string, string>;
|
|
|
|
export const AVAILABLE_LOCALES: readonly { value: Locale; label: string }[] = [
|
|
{ value: 'en', label: 'English' },
|
|
{ value: 'zh-CN', label: '简体中文' },
|
|
{ value: 'zh-TW', label: '繁體中文' },
|
|
{ value: 'ru', label: 'Русский' },
|
|
{ value: 'es', label: 'Español' },
|
|
];
|
|
|
|
let locale: Locale = resolveInitialLocale();
|
|
let activeMessages: MessageTable = enMessages;
|
|
const loadedMessages = new Map<Locale, MessageTable>([['en', enMessages]]);
|
|
|
|
function isLocale(value: unknown): value is Locale {
|
|
return AVAILABLE_LOCALES.some((item) => item.value === value);
|
|
}
|
|
|
|
function resolveInitialLocale(): Locale {
|
|
try {
|
|
const saved = localStorage.getItem(LOCALE_STORAGE_KEY);
|
|
if (isLocale(saved)) return saved;
|
|
} catch {
|
|
// ignore storage errors
|
|
}
|
|
if (typeof navigator !== 'undefined') {
|
|
const langs = Array.isArray(navigator.languages) ? navigator.languages : [navigator.language];
|
|
for (const lang of langs) {
|
|
const normalized = String(lang || '').toLowerCase();
|
|
if (normalized === 'zh-tw' || normalized === 'zh-hk' || normalized === 'zh-mo' || normalized.includes('hant')) return 'zh-TW';
|
|
if (normalized.startsWith('zh')) return 'zh-CN';
|
|
if (normalized.startsWith('ru')) return 'ru';
|
|
if (normalized.startsWith('es')) return 'es';
|
|
}
|
|
}
|
|
return 'en';
|
|
}
|
|
|
|
const localeLoaders: Record<Locale, () => Promise<{ default: MessageTable }>> = {
|
|
en: () => Promise.resolve({ default: enMessages }),
|
|
'zh-CN': () => import('./i18n/locales/zh-CN'),
|
|
'zh-TW': () => import('./i18n/locales/zh-TW'),
|
|
ru: () => import('./i18n/locales/ru'),
|
|
es: () => import('./i18n/locales/es'),
|
|
};
|
|
|
|
async function loadLocaleMessages(next: Locale): Promise<MessageTable> {
|
|
const cached = loadedMessages.get(next);
|
|
if (cached) return cached;
|
|
|
|
const mod = await localeLoaders[next]();
|
|
loadedMessages.set(next, mod.default);
|
|
return mod.default;
|
|
}
|
|
|
|
async function loadFallbackMessages(): Promise<MessageTable> {
|
|
return enMessages;
|
|
}
|
|
|
|
export type I18nParams = Record<string, string | number | null | undefined>;
|
|
|
|
export async function initI18n(): Promise<void> {
|
|
try {
|
|
activeMessages = await loadLocaleMessages(locale);
|
|
} catch (error) {
|
|
console.error('Failed to load locale, falling back to English:', error);
|
|
locale = 'en';
|
|
activeMessages = await loadFallbackMessages();
|
|
}
|
|
}
|
|
|
|
export function t(key: string, params?: I18nParams): string {
|
|
const template = activeMessages[key] ?? key;
|
|
if (!params) return template;
|
|
return template.replace(/\{(\w+)\}/g, (_, name: string) => String(params[name] ?? ''));
|
|
}
|
|
|
|
export function translateServerError(message: string | null | undefined, fallback: string): string {
|
|
const normalized = String(message || '').trim();
|
|
if (!normalized) return fallback;
|
|
|
|
const rateLimitMatch = normalized.match(/^Rate limit exceeded\. Try again in (\d+) seconds\.$/i);
|
|
if (rateLimitMatch) {
|
|
return t('txt_rate_limit_try_again_seconds', { seconds: rateLimitMatch[1] });
|
|
}
|
|
|
|
const key = {
|
|
'Account is disabled': 'txt_server_error_account_disabled',
|
|
'Client IP is required': 'txt_server_error_client_ip_required',
|
|
'ClientId or clientSecret is incorrect. Try again': 'txt_server_error_client_credentials_incorrect',
|
|
'Email already registered': 'txt_server_error_email_already_registered',
|
|
'Email and password are required': 'txt_server_error_email_password_required',
|
|
'Email is required': 'txt_server_error_email_required',
|
|
'Invite code is invalid or expired': 'txt_server_error_invite_invalid_or_expired',
|
|
'Invite code is required': 'txt_server_error_invite_required',
|
|
'Invalid refresh token': 'txt_server_error_invalid_refresh_token',
|
|
'Invalid request payload': 'txt_server_error_invalid_request_payload',
|
|
'JWT_SECRET is not set': 'txt_server_error_jwt_secret_missing',
|
|
'JWT_SECRET is using the default/sample value. Please change it.': 'txt_server_error_jwt_secret_default',
|
|
'JWT_SECRET must be at least 32 characters': 'txt_server_error_jwt_secret_too_short',
|
|
'Parameter error': 'txt_server_error_parameter_error',
|
|
'Refresh token is required': 'txt_server_error_refresh_token_required',
|
|
'Registration is temporarily unavailable, retry once': 'txt_server_error_registration_retry',
|
|
'TOTP token is required': 'txt_server_error_totp_token_required',
|
|
'Two factor required.': 'txt_server_error_two_factor_required',
|
|
'Two-step token is invalid. Try again.': 'txt_server_error_two_factor_invalid',
|
|
'Username or password is incorrect. Try again': 'txt_server_error_username_password_incorrect',
|
|
}[normalized];
|
|
|
|
return key ? t(key) : normalized;
|
|
}
|
|
|
|
export function getLocale(): Locale {
|
|
return locale;
|
|
}
|
|
|
|
export async function setLocale(next: Locale): Promise<void> {
|
|
let nextMessages: MessageTable;
|
|
try {
|
|
nextMessages = await loadLocaleMessages(next);
|
|
} catch (error) {
|
|
console.error('Failed to load selected locale, falling back to English:', error);
|
|
next = 'en';
|
|
nextMessages = await loadFallbackMessages();
|
|
}
|
|
locale = next;
|
|
activeMessages = nextMessages;
|
|
try {
|
|
localStorage.setItem(LOCALE_STORAGE_KEY, next);
|
|
} catch {
|
|
// ignore storage errors
|
|
}
|
|
}
|