UNPKG

idioma-cli

Version:

CLI for Idioma - Internationalization engine with smart defaults

115 lines (112 loc) 13.6 kB
#!/usr/bin/env node import { updateStatus } from "../shared/chunk-3acbb6gb.js"; // src/cli/worker.ts import { glob } from "glob"; import { loadConfig, loadLock, saveLock, translateFile } from "idioma-sdk"; async function runTranslation() { try { await updateStatus({ status: "running", startTime: new Date().toISOString() }); const args = process.argv.slice(2); const showCosts = args.includes("--costs"); const providerIndex = args.indexOf("--provider"); const modelIndex = args.indexOf("--model"); const providerOverride = providerIndex !== -1 ? args[providerIndex + 1] : undefined; const modelOverride = modelIndex !== -1 ? args[modelIndex + 1] : undefined; const config = await loadConfig(); if (providerOverride || modelOverride) { config.translation = { ...config.translation || {}, ...providerOverride ? { provider: providerOverride } : {}, ...modelOverride ? { model: modelOverride } : {} }; if (providerOverride) { config.provider = providerOverride; } if (modelOverride) { config.model = modelOverride; } } const lock = await loadLock(); const sourceLocale = config.locale.source; const targetLocales = config.locale.targets; const includePatterns = Array.isArray(config.files) ? config.files : config.files?.include || []; let allSourceFiles = []; for (const pattern of includePatterns) { const sourcePattern = pattern.replace(/\[locale\]/g, sourceLocale); const files = await glob(sourcePattern); allSourceFiles = [...allSourceFiles, ...files]; } const totalFiles = allSourceFiles.length * targetLocales.length; let processedFiles = 0; await updateStatus({ totalFiles, processedFiles: 0 }); for (const sourceFile of allSourceFiles) { for (const targetLocale of targetLocales) { try { const fileName = sourceFile.replace(/.*\//, ""); await updateStatus({ currentFile: `${fileName} -> ${targetLocale}`, processedFiles }); if (processedFiles > 0) { await new Promise((resolve) => setTimeout(resolve, 250)); } const _result = await translateFile(sourceFile, sourceLocale, targetLocale, lock, config, { showCosts }); processedFiles++; await updateStatus({ processedFiles }); if (processedFiles % 1 === 0) { await saveLock(lock); } } catch (error) { console.error(`Error translating ${sourceFile} to ${targetLocale}:`, error); processedFiles++; await updateStatus({ processedFiles, errors: [`${sourceFile} -> ${targetLocale}: ${error}`] }); } } } await saveLock(lock); await updateStatus({ status: "completed", endTime: new Date().toISOString(), currentFile: undefined, processedFiles: totalFiles }); console.log("✅ Translation completed successfully!"); process.exit(0); } catch (error) { console.error("Translation failed:", error); await updateStatus({ status: "failed", endTime: new Date().toISOString(), currentFile: undefined, errors: [`Fatal error: ${error}`] }); process.exit(1); } } process.on("SIGTERM", async () => { console.log("Received SIGTERM, shutting down gracefully..."); await updateStatus({ status: "failed", endTime: new Date().toISOString(), errors: ["Process terminated by user"] }); process.exit(0); }); runTranslation(); //# debugId=1D309EA2A857961564756E2164756E21 //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/cli/worker.ts"],
  "sourcesContent": [
    "#!/usr/bin/env node\nimport { glob } from 'glob';\nimport { loadConfig, loadLock, saveLock, translateFile } from 'idioma-sdk';\nimport { updateStatus } from './background';\n\n// Worker process that runs the actual translation\nasync function runTranslation() {\n  try {\n    // Update status to running\n    await updateStatus({\n      status: 'running',\n      startTime: new Date().toISOString(),\n    });\n\n    // Parse command line arguments\n    const args = process.argv.slice(2);\n    const showCosts = args.includes('--costs');\n    const providerIndex = args.indexOf('--provider');\n    const modelIndex = args.indexOf('--model');\n    const providerOverride = providerIndex !== -1 ? args[providerIndex + 1] : undefined;\n    const modelOverride = modelIndex !== -1 ? args[modelIndex + 1] : undefined;\n\n    // Load config and lock\n    const config = await loadConfig();\n    if (providerOverride || modelOverride) {\n      config.translation = {\n        ...(config.translation || {}),\n        ...(providerOverride ? { provider: providerOverride } : {}),\n        ...(modelOverride ? { model: modelOverride } : {}),\n      };\n      if (providerOverride) {\n        config.provider = providerOverride;\n      }\n      if (modelOverride) {\n        config.model = modelOverride;\n      }\n    }\n    const lock = await loadLock();\n\n    // Calculate actual files to be processed\n    const sourceLocale = config.locale.source;\n    const targetLocales = config.locale.targets;\n    const includePatterns = Array.isArray(config.files)\n      ? config.files\n      : config.files?.include || [];\n\n    // Get all source files\n    let allSourceFiles: string[] = [];\n    for (const pattern of includePatterns) {\n      const sourcePattern = pattern.replace(/\\[locale\\]/g, sourceLocale);\n      const files = await glob(sourcePattern);\n      allSourceFiles = [...allSourceFiles, ...files];\n    }\n\n    // Calculate total translation tasks (files × target locales)\n    const totalFiles = allSourceFiles.length * targetLocales.length;\n    let processedFiles = 0;\n\n    await updateStatus({\n      totalFiles,\n      processedFiles: 0,\n    });\n\n    // Process files with progress tracking\n    for (const sourceFile of allSourceFiles) {\n      for (const targetLocale of targetLocales) {\n        try {\n          // Update current file being processed\n          const fileName = sourceFile.replace(/.*\\//, ''); // Get just filename\n          await updateStatus({\n            currentFile: `${fileName} -> ${targetLocale}`,\n            processedFiles,\n          });\n\n          // Add delay to avoid rate limits\n          if (processedFiles > 0) {\n            await new Promise((resolve) => setTimeout(resolve, 250));\n          }\n\n          // Translate the file\n          const _result = await translateFile(\n            sourceFile,\n            sourceLocale,\n            targetLocale,\n            lock,\n            config,\n            {\n              showCosts,\n            }\n          );\n\n          processedFiles++;\n\n          // Update progress\n          await updateStatus({\n            processedFiles,\n          });\n\n          // Save lock file periodically\n          if (processedFiles % 1 === 0) {\n            // Save after each file for safety\n            await saveLock(lock);\n          }\n        } catch (error) {\n          console.error(`Error translating ${sourceFile} to ${targetLocale}:`, error);\n          processedFiles++;\n\n          // Update progress even on error\n          await updateStatus({\n            processedFiles,\n            errors: [`${sourceFile} -> ${targetLocale}: ${error}`],\n          });\n        }\n      }\n    }\n\n    // Save final lock file\n    await saveLock(lock);\n\n    // Update status to completed\n    await updateStatus({\n      status: 'completed',\n      endTime: new Date().toISOString(),\n      currentFile: undefined,\n      processedFiles: totalFiles,\n    });\n\n    console.log('✅ Translation completed successfully!');\n    process.exit(0);\n  } catch (error) {\n    console.error('Translation failed:', error);\n\n    await updateStatus({\n      status: 'failed',\n      endTime: new Date().toISOString(),\n      currentFile: undefined,\n      errors: [`Fatal error: ${error}`],\n    });\n\n    process.exit(1);\n  }\n}\n\n// Handle graceful shutdown\nprocess.on('SIGTERM', async () => {\n  console.log('Received SIGTERM, shutting down gracefully...');\n\n  await updateStatus({\n    status: 'failed',\n    endTime: new Date().toISOString(),\n    errors: ['Process terminated by user'],\n  });\n\n  process.exit(0);\n});\n\n// Run the translation\nrunTranslation();\n"
  ],
  "mappings": "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",
  "debugId": "1D309EA2A857961564756E2164756E21",
  "names": []
}