UNPKG

idioma-cli

Version:

CLI for Idioma - Internationalization engine with smart defaults

97 lines (94 loc) 11.7 kB
// src/cli/background.ts import { spawn as nodeSpawn } from "node:child_process"; import fs from "node:fs/promises"; import os from "node:os"; import path from "node:path"; var __dirname = "/Users/ryanwaits/Code/projects/idioma/packages/cli/src/cli"; var STATUS_DIR = path.join(os.tmpdir(), "idioma-translations"); var STATUS_FILE = path.join(STATUS_DIR, "status.json"); var PID_FILE = path.join(STATUS_DIR, "process.pid"); async function startBackgroundTranslation(args) { await fs.mkdir(STATUS_DIR, { recursive: true }); const isRunning = await isTranslationRunning(); if (isRunning) { console.log("❌ A translation is already running in the background."); console.log('Run "idioma status" to check progress or "idioma stop" to cancel it.'); return; } const isBun = typeof Bun !== "undefined"; const runtime = isBun ? "bun" : "node"; const scriptPath = path.join(__dirname, "worker.js"); const proc = nodeSpawn(runtime, [scriptPath, ...args], { cwd: process.cwd(), env: process.env, detached: true, stdio: "ignore" }); proc.unref(); await fs.writeFile(PID_FILE, proc.pid.toString()); const initialStatus = { status: "running", startTime: new Date().toISOString(), totalFiles: 0, processedFiles: 0, errors: [], pid: proc.pid }; await fs.writeFile(STATUS_FILE, JSON.stringify(initialStatus, null, 2)); console.log(`✅ Translation started in background (PID: ${proc.pid})`); console.log('Run "idioma status" to check progress.'); } async function getTranslationStatus() { try { const statusData = await fs.readFile(STATUS_FILE, "utf-8"); return JSON.parse(statusData); } catch { return null; } } async function stopBackgroundTranslation() { try { const pidData = await fs.readFile(PID_FILE, "utf-8"); const pid = parseInt(pidData, 10); process.kill(pid, "SIGTERM"); await fs.unlink(PID_FILE).catch(() => {}); await fs.unlink(STATUS_FILE).catch(() => {}); console.log("✅ Background translation stopped."); return true; } catch (_error) { console.log("❌ No background translation running."); return false; } } async function isTranslationRunning() { try { const pidData = await fs.readFile(PID_FILE, "utf-8"); const pid = parseInt(pidData, 10); try { process.kill(pid, 0); return true; } catch { await fs.unlink(PID_FILE).catch(() => {}); return false; } } catch { return false; } } async function updateStatus(updates) { const current = await getTranslationStatus() || { status: "running", startTime: new Date().toISOString(), totalFiles: 0, processedFiles: 0, errors: [] }; const updated = { ...current, ...updates }; if (updates.errors && Array.isArray(updates.errors)) { updated.errors = [...current.errors || [], ...updates.errors]; } await fs.writeFile(STATUS_FILE, JSON.stringify(updated, null, 2)); } export { startBackgroundTranslation, getTranslationStatus, stopBackgroundTranslation, updateStatus }; //# debugId=E20AC452E3FF854464756E2164756E21 //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/cli/background.ts"],
  "sourcesContent": [
    "import { spawn as nodeSpawn } from 'node:child_process';\nimport fs from 'node:fs/promises';\nimport os from 'node:os';\nimport path from 'node:path';\n\nconst STATUS_DIR = path.join(os.tmpdir(), 'idioma-translations');\nconst STATUS_FILE = path.join(STATUS_DIR, 'status.json');\nconst PID_FILE = path.join(STATUS_DIR, 'process.pid');\n\nexport interface TranslationStatus {\n  status: 'running' | 'completed' | 'failed';\n  startTime: string;\n  endTime?: string;\n  totalFiles: number;\n  processedFiles: number;\n  currentFile?: string;\n  errors: string[];\n  pid?: number;\n}\n\nexport async function startBackgroundTranslation(args: string[]): Promise<void> {\n  // Ensure status directory exists\n  await fs.mkdir(STATUS_DIR, { recursive: true });\n\n  // Check if translation is already running\n  const isRunning = await isTranslationRunning();\n  if (isRunning) {\n    console.log('❌ A translation is already running in the background.');\n    console.log('Run \"idioma status\" to check progress or \"idioma stop\" to cancel it.');\n    return;\n  }\n\n  // Prepare the command - detect runtime\n  const isBun = typeof Bun !== 'undefined';\n  const runtime = isBun ? 'bun' : 'node';\n  const scriptPath = path.join(__dirname, 'worker.js'); // Use .js for compatibility\n\n  // Spawn the background process using Node's child_process\n  const proc = nodeSpawn(runtime, [scriptPath, ...args], {\n    cwd: process.cwd(),\n    env: process.env,\n    detached: true,\n    stdio: 'ignore',\n  });\n\n  // Detach the process so it runs independently\n  proc.unref();\n\n  // Save the PID\n  await fs.writeFile(PID_FILE, proc.pid.toString());\n\n  // Initialize status file\n  const initialStatus: TranslationStatus = {\n    status: 'running',\n    startTime: new Date().toISOString(),\n    totalFiles: 0,\n    processedFiles: 0,\n    errors: [],\n    pid: proc.pid,\n  };\n  await fs.writeFile(STATUS_FILE, JSON.stringify(initialStatus, null, 2));\n\n  console.log(`✅ Translation started in background (PID: ${proc.pid})`);\n  console.log('Run \"idioma status\" to check progress.');\n}\n\nexport async function getTranslationStatus(): Promise<TranslationStatus | null> {\n  try {\n    const statusData = await fs.readFile(STATUS_FILE, 'utf-8');\n    return JSON.parse(statusData);\n  } catch {\n    return null;\n  }\n}\n\nexport async function stopBackgroundTranslation(): Promise<boolean> {\n  try {\n    const pidData = await fs.readFile(PID_FILE, 'utf-8');\n    const pid = parseInt(pidData, 10);\n\n    // Kill the process\n    process.kill(pid, 'SIGTERM');\n\n    // Clean up files\n    await fs.unlink(PID_FILE).catch(() => {});\n    await fs.unlink(STATUS_FILE).catch(() => {});\n\n    console.log('✅ Background translation stopped.');\n    return true;\n  } catch (_error) {\n    console.log('❌ No background translation running.');\n    return false;\n  }\n}\n\nexport async function isTranslationRunning(): Promise<boolean> {\n  try {\n    const pidData = await fs.readFile(PID_FILE, 'utf-8');\n    const pid = parseInt(pidData, 10);\n\n    // Check if process is still running\n    try {\n      process.kill(pid, 0); // Signal 0 checks if process exists\n      return true;\n    } catch {\n      // Process doesn't exist, clean up stale files\n      await fs.unlink(PID_FILE).catch(() => {});\n      return false;\n    }\n  } catch {\n    return false;\n  }\n}\n\nexport async function updateStatus(updates: Partial<TranslationStatus>): Promise<void> {\n  const current = (await getTranslationStatus()) || {\n    status: 'running',\n    startTime: new Date().toISOString(),\n    totalFiles: 0,\n    processedFiles: 0,\n    errors: [],\n  };\n\n  const updated = { ...current, ...updates };\n\n  // Handle errors array specially - append instead of replace\n  if (updates.errors && Array.isArray(updates.errors)) {\n    updated.errors = [...(current.errors || []), ...updates.errors];\n  }\n\n  await fs.writeFile(STATUS_FILE, JSON.stringify(updated, null, 2));\n}\n"
  ],
  "mappings": "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",
  "debugId": "E20AC452E3FF854464756E2164756E21",
  "names": []
}