idioma-cli
Version:
CLI for Idioma - Internationalization engine with smart defaults
97 lines (94 loc) • 11.7 kB
JavaScript
// 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": []
}