{"version":3,"file":"dev-logs.mjs","names":[],"sources":["../src/dev-logs.ts"],"sourcesContent":["import { existsSync } from \"node:fs\";\nimport { appendFile, mkdir, readFile, writeFile } from \"node:fs/promises\";\nimport { join } from \"node:path\";\n\nconst ESC = \"\\x1b\";\nconst BEL = \"\\x07\";\nconst ANSI_RE = new RegExp(`${ESC}\\\\[[0-?]*[ -/]*[@-~]|${ESC}\\\\][^${BEL}]*${BEL}`, \"g\");\n\nconst stripAnsi = (input: string): string => input.replace(ANSI_RE, \"\");\n\nexport interface LogEntry {\n  timestamp: number;\n  source: string;\n  line: string;\n  isError?: boolean;\n}\n\nexport interface DevLogger {\n  logFile: string;\n  latestFile: string;\n  write: (entry: LogEntry) => Promise<void>;\n  readLatest: (opts?: { tail?: number }) => Promise<string>;\n}\n\nexport function getBosDir(configDir: string): string {\n  return join(configDir, \".bos\");\n}\n\nexport function getLogsDir(configDir: string): string {\n  return join(getBosDir(configDir), \"logs\");\n}\n\nfunction formatLogLine(entry: LogEntry): string {\n  const ts = new Date(entry.timestamp).toISOString();\n  const prefix = entry.isError ? \"ERR\" : \"OUT\";\n  const clean = stripAnsi(entry.line);\n  return `[${ts}] [${entry.source}] [${prefix}] ${clean}`;\n}\n\nexport async function createDevLogger(configDir: string, description: string): Promise<DevLogger> {\n  const dir = getLogsDir(configDir);\n  if (!existsSync(dir)) {\n    await mkdir(dir, { recursive: true });\n  }\n\n  const now = new Date();\n  const ts = now.toISOString().replace(/[:.]/g, \"-\").slice(0, 19);\n  const logFile = join(dir, `dev-${ts}.log`);\n  const latestFile = join(dir, \"dev-latest.log\");\n\n  const header =\n    `# everything-dev dev session: ${description}\\n` + `# Started: ${now.toISOString()}\\n\\n`;\n  // Overwrite each run so dev-latest.log is always actionable.\n  await writeFile(logFile, header, \"utf8\");\n  await writeFile(latestFile, header, \"utf8\");\n\n  let chain = Promise.resolve();\n  const enqueue = (fn: () => Promise<void>) => {\n    chain = chain.then(fn, fn);\n    return chain;\n  };\n\n  return {\n    logFile,\n    latestFile,\n    write: (entry) =>\n      enqueue(async () => {\n        const line = `${formatLogLine(entry)}\\n`;\n        await appendFile(logFile, line);\n        await appendFile(latestFile, line);\n      }),\n    readLatest: async (opts) => {\n      const text = await readFile(latestFile, \"utf8\").catch(() => \"\");\n      const tail = opts?.tail;\n      if (!tail || tail <= 0) return text;\n      const lines = text.split(\"\\n\");\n      return lines.slice(Math.max(0, lines.length - tail)).join(\"\\n\");\n    },\n  };\n}\n\nexport async function readDevLatestLog(\n  configDir: string,\n  opts?: { tail?: number },\n): Promise<string> {\n  const latestFile = join(getLogsDir(configDir), \"dev-latest.log\");\n  const text = await readFile(latestFile, \"utf8\").catch(() => \"\");\n  const tail = opts?.tail;\n  if (!tail || tail <= 0) return text;\n  const lines = text.split(\"\\n\");\n  return lines.slice(Math.max(0, lines.length - tail)).join(\"\\n\");\n}\n"],"mappings":";;;;;AAIA,MAAM,MAAM;AACZ,MAAM,MAAM;AACZ,MAAM,UAAU,IAAI,OAAO,GAAG,IAAI,uBAAuB,IAAI,OAAO,IAAI,IAAI,OAAO,IAAI;AAEvF,MAAM,aAAa,UAA0B,MAAM,QAAQ,SAAS,GAAG;AAgBvE,SAAgB,UAAU,WAA2B;AACnD,QAAO,KAAK,WAAW,OAAO;;AAGhC,SAAgB,WAAW,WAA2B;AACpD,QAAO,KAAK,UAAU,UAAU,EAAE,OAAO;;AAG3C,SAAS,cAAc,OAAyB;CAC9C,MAAM,KAAK,IAAI,KAAK,MAAM,UAAU,CAAC,aAAa;CAClD,MAAM,SAAS,MAAM,UAAU,QAAQ;CACvC,MAAM,QAAQ,UAAU,MAAM,KAAK;AACnC,QAAO,IAAI,GAAG,KAAK,MAAM,OAAO,KAAK,OAAO,IAAI;;AAGlD,eAAsB,gBAAgB,WAAmB,aAAyC;CAChG,MAAM,MAAM,WAAW,UAAU;AACjC,KAAI,CAAC,WAAW,IAAI,CAClB,OAAM,MAAM,KAAK,EAAE,WAAW,MAAM,CAAC;CAGvC,MAAM,sBAAM,IAAI,MAAM;CAEtB,MAAM,UAAU,KAAK,KAAK,OADf,IAAI,aAAa,CAAC,QAAQ,SAAS,IAAI,CAAC,MAAM,GAAG,GACzB,CAAC,MAAM;CAC1C,MAAM,aAAa,KAAK,KAAK,iBAAiB;CAE9C,MAAM,SACJ,iCAAiC,YAAY,eAAoB,IAAI,aAAa,CAAC;AAErF,OAAM,UAAU,SAAS,QAAQ,OAAO;AACxC,OAAM,UAAU,YAAY,QAAQ,OAAO;CAE3C,IAAI,QAAQ,QAAQ,SAAS;CAC7B,MAAM,WAAW,OAA4B;AAC3C,UAAQ,MAAM,KAAK,IAAI,GAAG;AAC1B,SAAO;;AAGT,QAAO;EACL;EACA;EACA,QAAQ,UACN,QAAQ,YAAY;GAClB,MAAM,OAAO,GAAG,cAAc,MAAM,CAAC;AACrC,SAAM,WAAW,SAAS,KAAK;AAC/B,SAAM,WAAW,YAAY,KAAK;IAClC;EACJ,YAAY,OAAO,SAAS;GAC1B,MAAM,OAAO,MAAM,SAAS,YAAY,OAAO,CAAC,YAAY,GAAG;GAC/D,MAAM,OAAO,MAAM;AACnB,OAAI,CAAC,QAAQ,QAAQ,EAAG,QAAO;GAC/B,MAAM,QAAQ,KAAK,MAAM,KAAK;AAC9B,UAAO,MAAM,MAAM,KAAK,IAAI,GAAG,MAAM,SAAS,KAAK,CAAC,CAAC,KAAK,KAAK;;EAElE"}