idioma-cli
Version:
CLI for Idioma - Internationalization engine with smart defaults
319 lines (313 loc) • 40.5 kB
JavaScript
import {
getTranslationStatus,
startBackgroundTranslation,
stopBackgroundTranslation
} from "../shared/chunk-3acbb6gb.js";
// src/cli/index.ts
import { Command } from "commander";
import { readFileSync } from "fs";
import { dirname, join } from "path";
import { fileURLToPath } from "url";
// src/cli/commands.ts
import fs from "node:fs/promises";
import path from "node:path";
import { glob } from "glob";
import {
loadConfig,
loadLock,
processFiles,
replaceLocaleInPattern,
saveConfig,
saveLock
} from "idioma-sdk";
async function initCommand() {
const configPath = path.resolve("idioma.json");
try {
await fs.access(configPath);
console.log("Configuration file already exists.");
} catch {
const defaultConfig = {
provider: "anthropic",
locale: {
source: "en",
targets: []
},
files: {
include: ["**/*.mdx"]
}
};
await saveConfig(defaultConfig);
console.log("✓ Created idioma.json");
console.log(`
Next steps:`);
console.log("1. Add target locales: idioma add <locale>");
console.log("2. Configure your file patterns in idioma.json");
console.log("3. Run translation: idioma translate");
}
}
async function translateCommand(options) {
try {
const config = await loadConfig();
if (options.provider || options.model) {
config.translation = {
...config.translation || {},
...options.provider ? { provider: options.provider } : {},
...options.model ? { model: options.model } : {}
};
if (options.provider) {
config.provider = options.provider;
}
if (options.model) {
config.model = options.model;
}
}
const effectiveProvider = config.translation?.provider || config.provider || "anthropic";
const effectiveModel = config.translation?.model || config.model;
console.log(`[idioma] Using provider: ${effectiveProvider}${effectiveModel ? ` (model: ${effectiveModel})` : ""}`);
const lock = await loadLock();
const result = await processFiles(config, lock, { showCosts: options.costs });
await saveLock(result.lock);
console.log("Translation complete. Lockfile updated.");
} catch (error) {
if (error instanceof Error) {
console.error("Error:", error.message);
} else {
console.error("An unknown error occurred");
}
process.exit(1);
}
}
async function localeAddCommand(locales) {
try {
const config = await loadConfig();
const localeList = locales.split(",").map((l) => l.trim());
const added = [];
for (const locale of localeList) {
if (!config.locale.targets.includes(locale)) {
config.locale.targets.push(locale);
added.push(locale);
}
}
if (added.length > 0) {
await saveConfig(config);
console.log(`✓ Added locales: ${added.join(", ")}`);
} else {
console.log("All specified locales already exist.");
}
} catch (error) {
if (error instanceof Error) {
console.error("Error:", error.message);
} else {
console.error("An unknown error occurred");
}
process.exit(1);
}
}
async function localeRemoveCommand(locales) {
try {
const config = await loadConfig();
const localeList = locales.split(",").map((l) => l.trim());
const removed = [];
for (const locale of localeList) {
const index = config.locale.targets.indexOf(locale);
if (index !== -1) {
config.locale.targets.splice(index, 1);
removed.push(locale);
}
}
if (removed.length > 0) {
await saveConfig(config);
console.log(`✓ Removed locales: ${removed.join(", ")}`);
} else {
console.log("None of the specified locales were found.");
}
} catch (error) {
if (error instanceof Error) {
console.error("Error:", error.message);
} else {
console.error("An unknown error occurred");
}
process.exit(1);
}
}
async function localeListCommand() {
try {
const config = await loadConfig();
console.log("Source locale:", config.locale.source);
console.log("Target locales:", config.locale.targets.length ? config.locale.targets.join(", ") : "None");
} catch (error) {
if (error instanceof Error) {
console.error("Error:", error.message);
} else {
console.error("An unknown error occurred");
}
process.exit(1);
}
}
async function resetCommand() {
try {
const config = await loadConfig();
const lock = await loadLock();
if (config.locale.targets.length === 0) {
console.log("No target locales configured.");
return;
}
const deletedFiles = [];
for (const pattern of config.files) {
for (const targetLocale of config.locale.targets) {
const targetPattern = replaceLocaleInPattern(pattern, config.locale.source, targetLocale);
const files = await glob(targetPattern);
for (const file of files) {
try {
await fs.unlink(file);
deletedFiles.push(file);
} catch (_error) {}
}
if (files.length > 0) {
try {
const dirsToCheck = new Set;
for (const file of files) {
let dir = path.dirname(file);
while (dir.includes(`/${targetLocale}/`) || dir.endsWith(`/${targetLocale}`)) {
dirsToCheck.add(dir);
dir = path.dirname(dir);
}
}
const sortedDirs = Array.from(dirsToCheck).sort((a, b) => b.split("/").length - a.split("/").length);
for (const dir of sortedDirs) {
try {
await fs.rmdir(dir);
} catch {}
}
} catch {}
}
}
}
lock.files = {};
await saveLock(lock);
if (deletedFiles.length > 0) {
console.log(`✓ Reset complete. Removed ${deletedFiles.length} generated translation files:`);
deletedFiles.forEach((file) => console.log(` - ${file}`));
console.log(`
Translation status cleared. Run "idioma translate" to regenerate translations.`);
} else {
console.log("No translation files found. Lock file has been reset.");
}
} catch (error) {
if (error instanceof Error) {
console.error("Error:", error.message);
} else {
console.error("An unknown error occurred");
}
process.exit(1);
}
}
function displayStatus(status) {
const percentage = status.totalFiles > 0 ? Math.round(status.processedFiles / status.totalFiles * 100) : 0;
console.log(`
\uD83D\uDCCA Translation Status`);
console.log("─".repeat(40));
console.log(`Status: ${status.status === "running" ? "\uD83D\uDD04" : status.status === "completed" ? "✅" : "❌"} ${status.status}`);
console.log(`Progress: ${status.processedFiles}/${status.totalFiles} files (${percentage}%)`);
if (status.currentFile) {
console.log(`Current file: ${status.currentFile}`);
}
console.log(`Started: ${new Date(status.startTime).toLocaleString()}`);
if (status.endTime) {
console.log(`Ended: ${new Date(status.endTime).toLocaleString()}`);
const duration = new Date(status.endTime).getTime() - new Date(status.startTime).getTime();
const minutes = Math.floor(duration / 60000);
const seconds = Math.floor(duration % 60000 / 1000);
console.log(`Duration: ${minutes}m ${seconds}s`);
}
if (status.errors.length > 0) {
console.log(`
⚠️ Errors (${status.errors.length}):`);
status.errors.slice(-5).forEach((error) => {
console.log(` - ${error}`);
});
}
if (status.pid && status.status === "running") {
console.log(`
Process ID: ${status.pid}`);
console.log("To stop: idioma stop");
}
}
async function statusCommand(options = {}) {
if (options.tail) {
console.log("\uD83D\uDCE1 Real-time translation status (Press Ctrl+C to exit)");
console.log("═".repeat(50));
let lastStatus = null;
const updateDisplay = async () => {
const status = await getTranslationStatus();
if (!status) {
console.log(`
❌ No background translation is currently running.`);
process.exit(0);
}
const statusStr = JSON.stringify(status);
if (statusStr !== lastStatus) {
process.stdout.write("\x1B[2J\x1B[0f");
console.log("\uD83D\uDCE1 Real-time translation status (Press Ctrl+C to exit)");
console.log("═".repeat(50));
displayStatus(status);
if (status.status !== "running") {
console.log(`
\uD83C\uDFAF Translation finished!`);
process.exit(0);
}
lastStatus = statusStr;
}
};
await updateDisplay();
const interval = setInterval(updateDisplay, 2000);
process.on("SIGINT", () => {
clearInterval(interval);
console.log(`
\uD83D\uDC4B Exiting real-time status...`);
process.exit(0);
});
} else {
const status = await getTranslationStatus();
if (!status) {
console.log("No background translation is currently running.");
return;
}
displayStatus(status);
}
}
async function stopCommand() {
await stopBackgroundTranslation();
}
// src/cli/index.ts
var __filename2 = fileURLToPath(import.meta.url);
var __dirname2 = dirname(__filename2);
var packageJsonPath = join(__dirname2, "../../package.json");
var packageJson = JSON.parse(readFileSync(packageJsonPath, "utf8"));
var program = new Command;
program.name("idioma").description("Internationalization engine").version(packageJson.version);
program.command("init").description("Initialize Idioma configuration").action(initCommand);
program.command("translate").description("Translate files based on configuration").option("--costs", "Show translation costs based on token usage").option("--provider <provider>", "Override translation provider for this run").option("--model <model>", "Override translation model for this run").option("--background", "Run translation in the background").action(async (options) => {
if (options.background) {
const args = [];
if (options.costs)
args.push("--costs");
if (options.provider)
args.push("--provider", options.provider);
if (options.model)
args.push("--model", options.model);
await startBackgroundTranslation(args);
} else {
await translateCommand(options);
}
});
program.command("add <locales>").description("Add target locale(s) - supports comma-separated values (e.g., pt,fr)").action(localeAddCommand);
program.command("remove <locales>").description("Remove target locale(s) - supports comma-separated values (e.g., pt,fr)").action(localeRemoveCommand);
program.command("list").description("List all configured locales").action(localeListCommand);
program.command("reset").description("Reset translation status and remove generated translation files").action(resetCommand);
program.command("status").description("Check the status of a background translation").option("--tail", "Show real-time status updates").action(statusCommand);
program.command("stop").description("Stop a running background translation").action(stopCommand);
program.parse();
//# debugId=E649EAFE1D55390564756E2164756E21
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/cli/index.ts", "../src/cli/commands.ts"],
  "sourcesContent": [
    "#!/usr/bin/env node\nimport { Command } from 'commander';\nimport { readFileSync } from 'fs';\nimport { dirname, join } from 'path';\nimport { fileURLToPath } from 'url';\nimport { startBackgroundTranslation } from './background';\nimport {\n  initCommand,\n  localeAddCommand,\n  localeListCommand,\n  localeRemoveCommand,\n  resetCommand,\n  statusCommand,\n  stopCommand,\n  translateCommand,\n} from './commands';\n\nconst __filename = fileURLToPath(import.meta.url);\nconst __dirname = dirname(__filename);\nconst packageJsonPath = join(__dirname, '../../package.json');\nconst packageJson = JSON.parse(readFileSync(packageJsonPath, 'utf8'));\n\nconst program = new Command();\n\nprogram.name('idioma').description('Internationalization engine').version(packageJson.version);\n\n// Init command\nprogram.command('init').description('Initialize Idioma configuration').action(initCommand);\n\n// Translate command\nprogram\n  .command('translate')\n  .description('Translate files based on configuration')\n  .option('--costs', 'Show translation costs based on token usage')\n  .option('--provider <provider>', 'Override translation provider for this run')\n  .option('--model <model>', 'Override translation model for this run')\n  .option('--background', 'Run translation in the background')\n  .action(async (options) => {\n    if (options.background) {\n      const args = [];\n      if (options.costs) args.push('--costs');\n      if (options.provider) args.push('--provider', options.provider);\n      if (options.model) args.push('--model', options.model);\n      await startBackgroundTranslation(args);\n    } else {\n      await translateCommand(options);\n    }\n  });\n\n// Direct locale commands\nprogram\n  .command('add <locales>')\n  .description('Add target locale(s) - supports comma-separated values (e.g., pt,fr)')\n  .action(localeAddCommand);\n\nprogram\n  .command('remove <locales>')\n  .description('Remove target locale(s) - supports comma-separated values (e.g., pt,fr)')\n  .action(localeRemoveCommand);\n\nprogram.command('list').description('List all configured locales').action(localeListCommand);\n\n// Reset command\nprogram\n  .command('reset')\n  .description('Reset translation status and remove generated translation files')\n  .action(resetCommand);\n\n// Status command - check background translation status\nprogram\n  .command('status')\n  .description('Check the status of a background translation')\n  .option('--tail', 'Show real-time status updates')\n  .action(statusCommand);\n\n// Stop command - stop background translation\nprogram.command('stop').description('Stop a running background translation').action(stopCommand);\n\n// Parse command line arguments\nprogram.parse();\n",
    "import fs from 'node:fs/promises';\nimport path from 'node:path';\nimport { glob } from 'glob';\nimport {\n  type Config,\n  loadConfig,\n  loadLock,\n  processFiles,\n  replaceLocaleInPattern,\n  saveConfig,\n  saveLock,\n} from 'idioma-sdk';\nimport { getTranslationStatus, stopBackgroundTranslation } from './background';\n\n// Init command - create config file\nexport async function initCommand(): Promise<void> {\n  const configPath = path.resolve('idioma.json');\n\n  try {\n    await fs.access(configPath);\n    console.log('Configuration file already exists.');\n  } catch {\n    const defaultConfig: Config = {\n      provider: 'anthropic',\n      locale: {\n        source: 'en',\n        targets: [],\n      },\n      files: {\n        include: ['**/*.mdx'],\n      },\n    };\n\n    await saveConfig(defaultConfig);\n    console.log('✓ Created idioma.json');\n    console.log('\\nNext steps:');\n    console.log('1. Add target locales: idioma add <locale>');\n    console.log('2. Configure your file patterns in idioma.json');\n    console.log('3. Run translation: idioma translate');\n  }\n}\n\n// Translate command - process all files\nexport async function translateCommand(options: {\n  costs?: boolean;\n  provider?: string;\n  model?: string;\n}): Promise<void> {\n  try {\n    // Load configuration\n    const config = await loadConfig();\n    if (options.provider || options.model) {\n      config.translation = {\n        ...(config.translation || {}),\n        ...(options.provider ? { provider: options.provider } : {}),\n        ...(options.model ? { model: options.model } : {}),\n      };\n      if (options.provider) {\n        config.provider = options.provider;\n      }\n      if (options.model) {\n        config.model = options.model;\n      }\n    }\n    const effectiveProvider = config.translation?.provider || config.provider || 'anthropic';\n    const effectiveModel = config.translation?.model || config.model;\n    console.log(\n      `[idioma] Using provider: ${effectiveProvider}${\n        effectiveModel ? ` (model: ${effectiveModel})` : ''\n      }`\n    );\n    const lock = await loadLock();\n\n    // Process all files\n    const result = await processFiles(config, lock, { showCosts: options.costs });\n\n    // Save updated lock file\n    await saveLock(result.lock);\n    console.log('Translation complete. Lockfile updated.');\n  } catch (error) {\n    if (error instanceof Error) {\n      console.error('Error:', error.message);\n    } else {\n      console.error('An unknown error occurred');\n    }\n    process.exit(1);\n  }\n}\n\n// Locale add command - supports comma-separated locales\nexport async function localeAddCommand(locales: string): Promise<void> {\n  try {\n    const config = await loadConfig();\n    const localeList = locales.split(',').map((l) => l.trim());\n    const added: string[] = [];\n\n    for (const locale of localeList) {\n      if (!config.locale.targets.includes(locale)) {\n        config.locale.targets.push(locale);\n        added.push(locale);\n      }\n    }\n\n    if (added.length > 0) {\n      await saveConfig(config);\n      console.log(`✓ Added locales: ${added.join(', ')}`);\n    } else {\n      console.log('All specified locales already exist.');\n    }\n  } catch (error) {\n    if (error instanceof Error) {\n      console.error('Error:', error.message);\n    } else {\n      console.error('An unknown error occurred');\n    }\n    process.exit(1);\n  }\n}\n\n// Locale remove command - supports comma-separated locales\nexport async function localeRemoveCommand(locales: string): Promise<void> {\n  try {\n    const config = await loadConfig();\n    const localeList = locales.split(',').map((l) => l.trim());\n    const removed: string[] = [];\n\n    for (const locale of localeList) {\n      const index = config.locale.targets.indexOf(locale);\n      if (index !== -1) {\n        config.locale.targets.splice(index, 1);\n        removed.push(locale);\n      }\n    }\n\n    if (removed.length > 0) {\n      await saveConfig(config);\n      console.log(`✓ Removed locales: ${removed.join(', ')}`);\n    } else {\n      console.log('None of the specified locales were found.');\n    }\n  } catch (error) {\n    if (error instanceof Error) {\n      console.error('Error:', error.message);\n    } else {\n      console.error('An unknown error occurred');\n    }\n    process.exit(1);\n  }\n}\n\n// Locale list command\nexport async function localeListCommand(): Promise<void> {\n  try {\n    const config = await loadConfig();\n\n    console.log('Source locale:', config.locale.source);\n    console.log(\n      'Target locales:',\n      config.locale.targets.length ? config.locale.targets.join(', ') : 'None'\n    );\n  } catch (error) {\n    if (error instanceof Error) {\n      console.error('Error:', error.message);\n    } else {\n      console.error('An unknown error occurred');\n    }\n    process.exit(1);\n  }\n}\n\n// Reset command - reset translation status and remove generated files\nexport async function resetCommand(): Promise<void> {\n  try {\n    const config = await loadConfig();\n    const lock = await loadLock();\n\n    if (config.locale.targets.length === 0) {\n      console.log('No target locales configured.');\n      return;\n    }\n\n    const deletedFiles: string[] = [];\n\n    // Process each file pattern\n    for (const pattern of config.files) {\n      for (const targetLocale of config.locale.targets) {\n        // Replace [locale] placeholder with target locale\n        const targetPattern = replaceLocaleInPattern(pattern, config.locale.source, targetLocale);\n\n        // Find all files matching the target pattern\n        const files = await glob(targetPattern);\n\n        for (const file of files) {\n          try {\n            await fs.unlink(file);\n            deletedFiles.push(file);\n          } catch (_error) {\n            // File might not exist, continue\n          }\n        }\n\n        // Try to remove empty directories after deleting files\n        if (files.length > 0) {\n          try {\n            // Get unique directories from deleted files\n            const dirsToCheck = new Set<string>();\n            for (const file of files) {\n              let dir = path.dirname(file);\n              // Add all parent directories up to the locale directory\n              while (dir.includes(`/${targetLocale}/`) || dir.endsWith(`/${targetLocale}`)) {\n                dirsToCheck.add(dir);\n                dir = path.dirname(dir);\n              }\n            }\n\n            // Sort directories by depth (deepest first) to remove from bottom up\n            const sortedDirs = Array.from(dirsToCheck).sort(\n              (a, b) => b.split('/').length - a.split('/').length\n            );\n\n            for (const dir of sortedDirs) {\n              try {\n                await fs.rmdir(dir);\n              } catch {\n                // Directory not empty or doesn't exist, continue\n              }\n            }\n          } catch {\n            // Error removing directories, continue\n          }\n        }\n      }\n    }\n\n    // Reset lock file to initial state\n    lock.files = {};\n\n    await saveLock(lock);\n\n    if (deletedFiles.length > 0) {\n      console.log(`✓ Reset complete. Removed ${deletedFiles.length} generated translation files:`);\n      deletedFiles.forEach((file) => console.log(`  - ${file}`));\n      console.log(\n        '\\nTranslation status cleared. Run \"idioma translate\" to regenerate translations.'\n      );\n    } else {\n      console.log('No translation files found. Lock file has been reset.');\n    }\n  } catch (error) {\n    if (error instanceof Error) {\n      console.error('Error:', error.message);\n    } else {\n      console.error('An unknown error occurred');\n    }\n    process.exit(1);\n  }\n}\n\n// Display translation status (shared helper)\nfunction displayStatus(status: any): void {\n  const percentage =\n    status.totalFiles > 0 ? Math.round((status.processedFiles / status.totalFiles) * 100) : 0;\n\n  console.log('\\n📊 Translation Status');\n  console.log('─'.repeat(40));\n  console.log(\n    `Status: ${status.status === 'running' ? '🔄' : status.status === 'completed' ? '✅' : '❌'} ${status.status}`\n  );\n  console.log(`Progress: ${status.processedFiles}/${status.totalFiles} files (${percentage}%)`);\n\n  if (status.currentFile) {\n    console.log(`Current file: ${status.currentFile}`);\n  }\n\n  console.log(`Started: ${new Date(status.startTime).toLocaleString()}`);\n\n  if (status.endTime) {\n    console.log(`Ended: ${new Date(status.endTime).toLocaleString()}`);\n    const duration = new Date(status.endTime).getTime() - new Date(status.startTime).getTime();\n    const minutes = Math.floor(duration / 60000);\n    const seconds = Math.floor((duration % 60000) / 1000);\n    console.log(`Duration: ${minutes}m ${seconds}s`);\n  }\n\n  if (status.errors.length > 0) {\n    console.log(`\\n⚠️  Errors (${status.errors.length}):`);\n    status.errors.slice(-5).forEach((error) => {\n      console.log(`  - ${error}`);\n    });\n  }\n\n  if (status.pid && status.status === 'running') {\n    console.log(`\\nProcess ID: ${status.pid}`);\n    console.log('To stop: idioma stop');\n  }\n}\n\n// Status command - check background translation status\nexport async function statusCommand(options: { tail?: boolean } = {}): Promise<void> {\n  if (options.tail) {\n    // Real-time status updates\n    console.log('📡 Real-time translation status (Press Ctrl+C to exit)');\n    console.log('═'.repeat(50));\n\n    let lastStatus: any = null;\n\n    const updateDisplay = async () => {\n      const status = await getTranslationStatus();\n\n      if (!status) {\n        console.log('\\n❌ No background translation is currently running.');\n        process.exit(0);\n      }\n\n      // Only update display if status changed\n      const statusStr = JSON.stringify(status);\n      if (statusStr !== lastStatus) {\n        // Clear screen and show updated status\n        process.stdout.write('\\x1B[2J\\x1B[0f');\n        console.log('📡 Real-time translation status (Press Ctrl+C to exit)');\n        console.log('═'.repeat(50));\n        displayStatus(status);\n\n        // Exit if translation completed or failed\n        if (status.status !== 'running') {\n          console.log('\\n🎯 Translation finished!');\n          process.exit(0);\n        }\n\n        lastStatus = statusStr;\n      }\n    };\n\n    // Initial display\n    await updateDisplay();\n\n    // Update every 2 seconds\n    const interval = setInterval(updateDisplay, 2000);\n\n    // Handle Ctrl+C gracefully\n    process.on('SIGINT', () => {\n      clearInterval(interval);\n      console.log('\\n\\n👋 Exiting real-time status...');\n      process.exit(0);\n    });\n  } else {\n    // Single status check\n    const status = await getTranslationStatus();\n\n    if (!status) {\n      console.log('No background translation is currently running.');\n      return;\n    }\n\n    displayStatus(status);\n  }\n}\n\n// Stop command - stop background translation\nexport async function stopCommand(): Promise<void> {\n  await stopBackgroundTranslation();\n}\n"
  ],
  "mappings": "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",
  "debugId": "E649EAFE1D55390564756E2164756E21",
  "names": []
}