@wrongstack/core 0.275.1 → 0.276.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (83) hide show
  1. package/dist/{agent-bridge-D9JkPvJ0.d.ts → agent-bridge-D7A-eu3C.d.ts} +1 -1
  2. package/dist/{agent-subagent-runner-CArSFKFl.d.ts → agent-subagent-runner-CEuw4ATz.d.ts} +16 -10
  3. package/dist/{brain-DCkB5_e7.d.ts → brain-BLOyN5ZP.d.ts} +127 -1
  4. package/dist/{compactor-CzSvxM1g.d.ts → compactor-DcBpaJsI.d.ts} +1 -1
  5. package/dist/{config-BzFRKkg7.d.ts → config-Bf5mj-ad.d.ts} +20 -2
  6. package/dist/{context-BrLe8pJy.d.ts → context-CLnUMW5g.d.ts} +40 -2
  7. package/dist/coordination/index.d.ts +43 -24
  8. package/dist/coordination/index.js +849 -648
  9. package/dist/coordination/index.js.map +1 -1
  10. package/dist/defaults/index.d.ts +28 -28
  11. package/dist/defaults/index.js +1636 -845
  12. package/dist/defaults/index.js.map +1 -1
  13. package/dist/execution/index.d.ts +16 -16
  14. package/dist/execution/index.js +218 -49
  15. package/dist/execution/index.js.map +1 -1
  16. package/dist/execution/prompt-enhancer.d.ts +1 -1
  17. package/dist/extension/index.d.ts +7 -7
  18. package/dist/extension/index.js.map +1 -1
  19. package/dist/{global-mailbox-CXkugtNQ.d.ts → global-mailbox-Iqfkgmwu.d.ts} +3 -3
  20. package/dist/{goal-store-DUwdbdoY.d.ts → goal-store-DGb6b5Ed.d.ts} +1 -1
  21. package/dist/hq/index.d.ts +6 -6
  22. package/dist/hq/index.js +178 -75
  23. package/dist/hq/index.js.map +1 -1
  24. package/dist/{index-CtlizLTK.d.ts → index-Cn0NOshr.d.ts} +10 -5
  25. package/dist/{index-neOCEy6q.d.ts → index-L4RZN9jJ.d.ts} +2 -2
  26. package/dist/index.d.ts +56 -48
  27. package/dist/index.js +2789 -1546
  28. package/dist/index.js.map +1 -1
  29. package/dist/infrastructure/index.d.ts +6 -6
  30. package/dist/infrastructure/index.js +26 -7
  31. package/dist/infrastructure/index.js.map +1 -1
  32. package/dist/kernel/index.d.ts +20 -12
  33. package/dist/kernel/index.js +55 -9
  34. package/dist/kernel/index.js.map +1 -1
  35. package/dist/{mailbox-types-_7gaY0Rl.d.ts → mailbox-types-DTl7bRH3.d.ts} +3 -1
  36. package/dist/{mcp-servers-MLL6bMlv.d.ts → mcp-servers-CuZGf9fI.d.ts} +4 -4
  37. package/dist/models/index.d.ts +5 -5
  38. package/dist/models/index.js +223 -139
  39. package/dist/models/index.js.map +1 -1
  40. package/dist/{models-registry-CrkcxQ-g.d.ts → models-registry-8XOdxWQu.d.ts} +16 -1
  41. package/dist/{multi-agent-coordinator-Dc_HuG9p.d.ts → multi-agent-coordinator-CiRtKVTk.d.ts} +8 -1
  42. package/dist/{null-fleet-bus-BMZwMin7.d.ts → null-fleet-bus-d9G-bVy9.d.ts} +26 -22
  43. package/dist/observability/index.d.ts +2 -2
  44. package/dist/{path-resolver-uVK4BatM.d.ts → path-resolver-BhIb6mtd.d.ts} +8 -3
  45. package/dist/{permission-CJR1qfOi.d.ts → permission-BCbQDR2s.d.ts} +1 -1
  46. package/dist/{permission-policy-DLVKKk4w.d.ts → permission-policy-C0ikndX_.d.ts} +2 -18
  47. package/dist/{pipeline-BYR-Vdau.d.ts → pipeline-Dl6XbfE7.d.ts} +10 -6
  48. package/dist/{provider-model-resolve-iREK_1lG.d.ts → provider-model-resolve-B70epO19.d.ts} +3 -3
  49. package/dist/{provider-runner-i7SQXZuC.d.ts → provider-runner-DZ808MSM.d.ts} +3 -3
  50. package/dist/{retry-policy-BmY5ooh3.d.ts → retry-policy-Dt3_z8Aj.d.ts} +1 -1
  51. package/dist/sdd/index.d.ts +19 -10
  52. package/dist/sdd/index.js +411 -240
  53. package/dist/sdd/index.js.map +1 -1
  54. package/dist/{secret-vault-C9leEMzr.d.ts → secret-vault-BUJ2d1gB.d.ts} +1 -1
  55. package/dist/security/index.d.ts +5 -5
  56. package/dist/security/index.js +30 -6
  57. package/dist/security/index.js.map +1 -1
  58. package/dist/{selector-qjpee9BF.d.ts → selector-BCkWgdwy.d.ts} +1 -1
  59. package/dist/{session-event-bridge-m7y--I-H.d.ts → session-event-bridge-CMvIO59_.d.ts} +1 -1
  60. package/dist/{session-reader-BjLH4V9n.d.ts → session-reader-C8aiChUu.d.ts} +1 -1
  61. package/dist/skills/index.js +1 -0
  62. package/dist/skills/index.js.map +1 -1
  63. package/dist/storage/index.d.ts +68 -30
  64. package/dist/storage/index.js +839 -528
  65. package/dist/storage/index.js.map +1 -1
  66. package/dist/{strategy-compactor-C2bmlWYg.d.ts → strategy-compactor-DI1OHVbB.d.ts} +10 -10
  67. package/dist/{todos-checkpoint-oDS9IBNS.d.ts → todos-checkpoint-Ddd2CGr0.d.ts} +56 -9
  68. package/dist/{tool-executor-D4YdaJ-M.d.ts → tool-executor-Bmd5Ygoo.d.ts} +45 -10
  69. package/dist/tools/index.d.ts +2 -2
  70. package/dist/tools/index.js.map +1 -1
  71. package/dist/types/index.d.ts +20 -20
  72. package/dist/types/index.js +331 -98
  73. package/dist/types/index.js.map +1 -1
  74. package/dist/utils/index.d.ts +16 -3
  75. package/dist/utils/index.js +159 -83
  76. package/dist/utils/index.js.map +1 -1
  77. package/dist/{worktree-manager-A1Efnvs0.d.ts → worktree-manager-DBdl_5rs.d.ts} +4 -1
  78. package/instructions/agents/shadow-agent.md +3 -3
  79. package/instructions/coordination/director-preamble.md +3 -3
  80. package/instructions/modes/research-web.md +4 -4
  81. package/package.json +1 -1
  82. package/skills/research-web/SKILL.md +26 -26
  83. package/skills/research-web/SKILL.save.md +1 -1
@@ -1,4 +1,4 @@
1
- import { M as Message } from './context-BrLe8pJy.js';
1
+ import { M as Message } from './context-CLnUMW5g.js';
2
2
 
3
3
  /**
4
4
  * Result of LLM-driven message importance analysis.
@@ -1,4 +1,4 @@
1
- import { S as SessionEvent, b as SessionWriter } from './context-BrLe8pJy.js';
1
+ import { S as SessionEvent, b as SessionWriter } from './context-CLnUMW5g.js';
2
2
 
3
3
  type AuditLevel = 'minimal' | 'standard' | 'full';
4
4
  /**
@@ -1,4 +1,4 @@
1
- import { i as ContentBlock, S as SessionEvent, j as SessionMetadata, k as SessionStore } from './context-BrLe8pJy.js';
1
+ import { i as ContentBlock, S as SessionEvent, j as SessionMetadata, k as SessionStore } from './context-CLnUMW5g.js';
2
2
 
3
3
  type AttachmentKind = 'text' | 'image' | 'file';
4
4
  interface AttachmentMeta {
@@ -142,6 +142,7 @@ var ERROR_CODES = {
142
142
  SDD_NOT_READY: "SDD_NOT_READY",
143
143
  // General
144
144
  VALIDATION_ERROR: "VALIDATION_ERROR",
145
+ PARSE_FAILED: "PARSE_FAILED",
145
146
  UNKNOWN: "UNKNOWN"
146
147
  };
147
148
  var WrongStackError = class extends Error {
@@ -1 +1 @@
1
- {"version":3,"sources":["../../src/utils/expect-defined.ts","../../src/utils/error.ts","../../src/utils/atomic-write.ts","../../src/types/errors.ts","../../src/skills/github-fetcher.ts","../../src/skills/manifest-store.ts","../../src/skills/skill-installer.ts"],"names":["path","fs","stat","resolve","fs2","path2","fs3","path3"],"mappings":"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","file":"index.js","sourcesContent":["/** Assert a value is neither null nor undefined. Throws if it is.\n * Useful after optional chaining and indexed access when the\n * control flow guarantees the value exists but TypeScript can't\n * prove it (e.g. after a check on a related field). */\nexport function expectDefined<T>(value: T | null | undefined, label?: string): T {\n if (value === null || value === undefined) {\n const err = new Error(label ? `Expected ${label} to be defined` : 'Expected value to be defined');\n err.name = 'ExpectDefinedError';\n throw err;\n }\n return value;\n}\n","/**\n * Converts an unknown error value to a human-readable string.\n * Used in 40+ files across the codebase to normalize error messaging.\n */\nexport function toErrorMessage(err: unknown): string {\n return err instanceof Error ? err.message : String(err);\n}\n","import { randomBytes } from 'node:crypto';\nimport * as fs from 'node:fs/promises';\nimport * as path from 'node:path';\n\nexport interface AtomicWriteOptions {\n mode?: number | undefined;\n encoding?: BufferEncoding | undefined;\n}\n\nexport interface FileLockOptions {\n timeoutMs?: number | undefined;\n staleMs?: number | undefined;\n}\n\nexport async function atomicWrite(\n targetPath: string,\n content: string | Uint8Array,\n opts: AtomicWriteOptions = {},\n): Promise<void> {\n const dir = path.dirname(targetPath);\n await fs.mkdir(dir, { recursive: true });\n const tmp = path.join(dir, `.${path.basename(targetPath)}.${randomBytes(6).toString('hex')}.tmp`);\n\n // Write content to tmp first; 'wx' ensures exclusive creation (fails if\n // tmp already exists — extremely unlikely with 6-byte random suffix).\n try {\n if (typeof content === 'string') {\n await fs.writeFile(tmp, content, { flag: 'wx', encoding: opts.encoding ?? 'utf8' });\n } else {\n await fs.writeFile(tmp, content, { flag: 'wx' });\n }\n try {\n const fh = await fs.open(tmp, 'r+');\n try {\n await fh.sync();\n } finally {\n await fh.close();\n }\n } catch {\n // fsync best-effort\n }\n // Now safely read mode from target (if it exists) and apply to tmp before rename.\n // Prefer opts.mode for new files; for existing files preserve their mode.\n let mode: number | undefined;\n try {\n const stat = await fs.stat(targetPath);\n mode = stat.mode & 0o777;\n } catch {\n mode = opts.mode;\n }\n if (mode !== undefined) {\n await fs.chmod(tmp, mode);\n }\n await renameWithRetry(tmp, targetPath);\n // P3 #20 (before-release.md): on Windows, fs.rename (MoveFileExW) does\n // not preserve Unix permission bits — the chmod above applies to the tmp\n // file, but the rename may reset the destination's mode to the Windows\n // default. Re-apply the mode after rename on win32 so an edited file\n // keeps its executable bit (or any non-default permission). On POSIX,\n // rename preserves metadata so this is a no-op (chmod is idempotent and\n // cheap), but we gate it on win32 to avoid the extra stat+chmod on the\n // common path.\n if (mode !== undefined && process.platform === 'win32') {\n try {\n await fs.chmod(targetPath, mode);\n } catch {\n // Best-effort: a transient EPERM (antivirus lock) should not fail\n // the write — the content is already on disk.\n }\n }\n } catch (err) {\n try {\n await fs.unlink(tmp);\n } catch {\n // ignore cleanup error\n }\n throw err;\n }\n}\n\nexport async function ensureDir(dir: string): Promise<void> {\n await fs.mkdir(dir, { recursive: true });\n}\n\nexport async function withFileLock<T>(\n targetPath: string,\n fn: () => Promise<T>,\n opts: FileLockOptions = {},\n): Promise<T> {\n const dir = path.dirname(targetPath);\n await fs.mkdir(dir, { recursive: true });\n const lockPath = path.join(dir, `.${path.basename(targetPath)}.lock`);\n const timeoutMs = opts.timeoutMs ?? 5_000;\n const staleMs = opts.staleMs ?? 30_000;\n const started = Date.now();\n let handle: fs.FileHandle | undefined;\n\n for (;;) {\n try {\n handle = await fs.open(lockPath, 'wx');\n await handle.writeFile(`${process.pid}:${Date.now()}`);\n break;\n } catch (err) {\n const code = (err as NodeJS.ErrnoException).code;\n // ENOENT means the directory was deleted (e.g. by concurrent cleanup).\n // Recreate it and retry acquiring the lock.\n if (code === 'ENOENT') {\n await fs.mkdir(dir, { recursive: true });\n continue;\n }\n if (code !== 'EEXIST') throw err;\n try {\n const stat = await fs.stat(lockPath);\n if (Date.now() - stat.mtimeMs > staleMs) {\n await fs.unlink(lockPath);\n continue;\n }\n } catch {\n continue;\n }\n if (Date.now() - started >= timeoutMs) {\n throw new Error(`Timed out waiting for file lock: ${targetPath}`);\n }\n await new Promise((resolve) => setTimeout(resolve, 25));\n }\n }\n\n try {\n return await fn();\n } finally {\n try {\n await handle?.close();\n } catch {\n // ignore\n }\n try {\n await fs.unlink(lockPath);\n } catch {\n // ignore\n }\n }\n}\n\n// On Windows, fs.rename over an existing file can fail with EPERM/EBUSY/EACCES\n// when antivirus, file indexers, editor file watchers, or a concurrent writer\n// briefly hold a handle on the destination. These are transient — retry with a\n// short backoff before giving up. POSIX renames are atomic and won't hit this.\nconst TRANSIENT_RENAME_CODES = new Set(['EPERM', 'EBUSY', 'EACCES', 'ENOTEMPTY']);\n\nasync function renameWithRetry(from: string, to: string): Promise<void> {\n if (process.platform !== 'win32') {\n await fs.rename(from, to);\n return;\n }\n const delays = [10, 25, 60, 120, 250];\n let lastErr: unknown;\n for (let i = 0; i <= delays.length; i++) {\n try {\n await fs.rename(from, to);\n return;\n } catch (err) {\n lastErr = err;\n const code = (err as NodeJS.ErrnoException)?.code;\n if (!code || !TRANSIENT_RENAME_CODES.has(code) || i === delays.length) {\n throw err;\n }\n await new Promise((resolve) => setTimeout(resolve, delays[i]));\n }\n }\n throw lastErr;\n}\n","import { toErrorMessage } from '../utils/index.js';\n\n/**\n * WrongStack error hierarchy.\n *\n * Every error thrown by the framework is a `WrongStackError` with a\n * machine-readable `code`, a `subsystem` tag, and a `severity` level.\n * This lets consumers (CLI, TUI, plugins, tests) branch on structured\n * data instead of parsing error messages.\n */\n\n// ── Error codes ──────────────────────────────────────────────────────\n\n/**\n * Machine-readable error codes as frozen constants.\n *\n * Use `ERROR_CODES.X` instead of raw string literals for:\n * - IDE autocomplete and compile-time validation\n * - Safe refactoring (rename updates all usages)\n * - Plugin extensibility (extend the object to add custom codes)\n *\n * The `ErrorCode` type is derived from this object, so adding a new\n * code here automatically updates the type without extra changes.\n */\nexport const ERROR_CODES = {\n // Provider\n PROVIDER_RATE_LIMITED: 'PROVIDER_RATE_LIMITED',\n PROVIDER_AUTH_FAILED: 'PROVIDER_AUTH_FAILED',\n PROVIDER_OVERLOADED: 'PROVIDER_OVERLOADED',\n PROVIDER_INVALID_REQUEST: 'PROVIDER_INVALID_REQUEST',\n PROVIDER_SERVER_ERROR: 'PROVIDER_SERVER_ERROR',\n PROVIDER_NETWORK_ERROR: 'PROVIDER_NETWORK_ERROR',\n PROVIDER_CONTEXT_OVERFLOW: 'PROVIDER_CONTEXT_OVERFLOW',\n // Tool\n TOOL_NOT_FOUND: 'TOOL_NOT_FOUND',\n TOOL_PERMISSION_DENIED: 'TOOL_PERMISSION_DENIED',\n TOOL_EXECUTION_FAILED: 'TOOL_EXECUTION_FAILED',\n TOOL_TIMEOUT: 'TOOL_TIMEOUT',\n TOOL_INPUT_INVALID: 'TOOL_INPUT_INVALID',\n // Config\n CONFIG_INVALID: 'CONFIG_INVALID',\n CONFIG_NOT_FOUND: 'CONFIG_NOT_FOUND',\n CONFIG_PARSE_FAILED: 'CONFIG_PARSE_FAILED',\n CONFIG_MIGRATION_NEEDED: 'CONFIG_MIGRATION_NEEDED',\n // Plugin\n PLUGIN_LOAD_FAILED: 'PLUGIN_LOAD_FAILED',\n PLUGIN_API_MISMATCH: 'PLUGIN_API_MISMATCH',\n PLUGIN_MISSING_DEPENDENCY: 'PLUGIN_MISSING_DEPENDENCY',\n // Agent\n AGENT_ITERATION_LIMIT: 'AGENT_ITERATION_LIMIT',\n AGENT_CONTEXT_OVERFLOW: 'AGENT_CONTEXT_OVERFLOW',\n AGENT_ABORTED: 'AGENT_ABORTED',\n AGENT_RUN_FAILED: 'AGENT_RUN_FAILED',\n // Session\n SESSION_NOT_FOUND: 'SESSION_NOT_FOUND',\n SESSION_CORRUPTED: 'SESSION_CORRUPTED',\n SESSION_WRITE_FAILED: 'SESSION_WRITE_FAILED',\n // Container / Registry\n CONTAINER_TOKEN_ALREADY_BOUND: 'CONTAINER_TOKEN_ALREADY_BOUND',\n CONTAINER_TOKEN_NOT_BOUND: 'CONTAINER_TOKEN_NOT_BOUND',\n CONTAINER_CIRCULAR_DEPENDENCY: 'CONTAINER_CIRCULAR_DEPENDENCY',\n REGISTRY_DUPLICATE: 'REGISTRY_DUPLICATE',\n REGISTRY_NOT_FOUND: 'REGISTRY_NOT_FOUND',\n REGISTRY_INVALID: 'REGISTRY_INVALID',\n // File system\n FS_READ_FAILED: 'FS_READ_FAILED',\n FS_WRITE_FAILED: 'FS_WRITE_FAILED',\n FS_MKDIR_FAILED: 'FS_MKDIR_FAILED',\n FS_DELETE_FAILED: 'FS_DELETE_FAILED',\n FS_ATOMIC_WRITE_FAILED: 'FS_ATOMIC_WRITE_FAILED',\n // SDD (Spec-Driven Development)\n SDD_VALIDATION_FAILED: 'SDD_VALIDATION_FAILED',\n SDD_PARSE_FAILED: 'SDD_PARSE_FAILED',\n SDD_INVALID_STATE: 'SDD_INVALID_STATE',\n SDD_NOT_READY: 'SDD_NOT_READY',\n // General\n VALIDATION_ERROR: 'VALIDATION_ERROR',\n UNKNOWN: 'UNKNOWN',\n} as const;\n\n/**\n * Union type derived from `ERROR_CODES`. Using `typeof ERROR_CODES[keyof typeof ERROR_CODES]`\n * instead of a string literal union means TypeScript auto-updates the type whenever\n * a new code is added to `ERROR_CODES` — no need to keep two lists in sync.\n */\nexport type ErrorCode = (typeof ERROR_CODES)[keyof typeof ERROR_CODES];\n\nexport type ErrorSubsystem =\n | 'provider'\n | 'tool'\n | 'config'\n | 'plugin'\n | 'agent'\n | 'session'\n | 'sdd'\n | 'container'\n | 'fs'\n | 'general';\nexport type ErrorSeverity = 'fatal' | 'error' | 'warning';\n\n// ── Base error class ─────────────────────────────────────────────────\n\nexport class WrongStackError extends Error {\n readonly code: ErrorCode;\n readonly subsystem: ErrorSubsystem;\n readonly severity: ErrorSeverity;\n readonly recoverable: boolean;\n readonly context?: Record<string, unknown> | undefined;\n\n constructor(opts: {\n message: string;\n code: ErrorCode;\n subsystem: ErrorSubsystem;\n severity?: ErrorSeverity | undefined;\n recoverable?: boolean | undefined;\n context?: Record<string, unknown> | undefined;\n cause?: unknown | undefined;\n }) {\n super(opts.message, { cause: opts.cause });\n this.name = 'WrongStackError';\n this.code = opts.code;\n this.subsystem = opts.subsystem;\n this.severity = opts.severity ?? 'error';\n this.recoverable = opts.recoverable ?? false;\n this.context = opts.context;\n }\n\n /**\n * Render a one-line user-facing description.\n * Subclasses should override for domain-specific formatting.\n */\n describe(): string {\n const ctx = this.context ? ` ${formatContext(this.context)}` : '';\n return `${this.code}: ${this.message}${ctx}`;\n }\n}\n\nfunction formatContext(ctx: Record<string, unknown>): string {\n const parts = Object.entries(ctx)\n .filter(([, v]) => v !== undefined)\n .slice(0, 3)\n .map(([k, v]) => `${k}=${String(v)}`);\n return parts.length > 0 ? `[${parts.join(' ')}]` : '';\n}\n\n// ── Specific error classes ───────────────────────────────────────────\n\n/**\n * Tool execution errors — thrown by ToolExecutor and individual tools.\n */\nexport class ToolError extends WrongStackError {\n readonly toolName: string;\n\n constructor(opts: {\n message: string;\n code: Extract<\n ErrorCode,\n | 'TOOL_NOT_FOUND'\n | 'TOOL_PERMISSION_DENIED'\n | 'TOOL_EXECUTION_FAILED'\n | 'TOOL_TIMEOUT'\n | 'TOOL_INPUT_INVALID'\n >;\n toolName: string;\n recoverable?: boolean | undefined;\n context?: Record<string, unknown> | undefined;\n cause?: unknown | undefined;\n }) {\n super({\n message: opts.message,\n code: opts.code,\n subsystem: 'tool',\n recoverable: opts.recoverable,\n context: { tool: opts.toolName, ...opts.context },\n cause: opts.cause,\n });\n this.name = 'ToolError';\n this.toolName = opts.toolName;\n }\n}\n\n/**\n * Config loading / validation errors.\n */\nexport class ConfigError extends WrongStackError {\n constructor(opts: {\n message: string;\n code: Extract<\n ErrorCode,\n 'CONFIG_INVALID' | 'CONFIG_NOT_FOUND' | 'CONFIG_PARSE_FAILED' | 'CONFIG_MIGRATION_NEEDED'\n >;\n context?: Record<string, unknown> | undefined;\n cause?: unknown | undefined;\n }) {\n super({\n message: opts.message,\n code: opts.code,\n subsystem: 'config',\n severity: 'fatal',\n recoverable: false,\n context: opts.context,\n cause: opts.cause,\n });\n this.name = 'ConfigError';\n }\n}\n\n/**\n * Plugin loading / lifecycle errors.\n */\nexport class PluginError extends WrongStackError {\n readonly pluginName: string;\n\n constructor(opts: {\n message: string;\n code: Extract<\n ErrorCode,\n 'PLUGIN_LOAD_FAILED' | 'PLUGIN_API_MISMATCH' | 'PLUGIN_MISSING_DEPENDENCY'\n >;\n pluginName: string;\n context?: Record<string, unknown> | undefined;\n cause?: unknown | undefined;\n }) {\n super({\n message: opts.message,\n code: opts.code,\n subsystem: 'plugin',\n severity: 'error',\n recoverable: opts.code === ERROR_CODES.PLUGIN_MISSING_DEPENDENCY,\n context: { plugin: opts.pluginName, ...opts.context },\n cause: opts.cause,\n });\n this.name = 'PluginError';\n this.pluginName = opts.pluginName;\n }\n}\n\n/**\n * Agent runtime errors — thrown by Agent.run when a non-WrongStackError\n * escapes the inner loop, so callers always see a structured error.\n */\nexport class AgentError extends WrongStackError {\n constructor(opts: {\n message: string;\n code: Extract<\n ErrorCode,\n 'AGENT_ITERATION_LIMIT' | 'AGENT_CONTEXT_OVERFLOW' | 'AGENT_ABORTED' | 'AGENT_RUN_FAILED'\n >;\n recoverable?: boolean | undefined;\n context?: Record<string, unknown> | undefined;\n cause?: unknown | undefined;\n }) {\n super({\n message: opts.message,\n code: opts.code,\n subsystem: 'agent',\n severity: opts.code === ERROR_CODES.AGENT_ABORTED ? 'warning' : 'error',\n recoverable: opts.recoverable ?? opts.code === ERROR_CODES.AGENT_ITERATION_LIMIT,\n context: opts.context,\n cause: opts.cause,\n });\n this.name = 'AgentError';\n }\n}\n\n/**\n * Wrap an arbitrary thrown value into a `WrongStackError` so the caller\n * always gets a structured error. Pass-throughs WrongStackError instances\n * unchanged; raw `Error`s and primitives get an `AGENT_RUN_FAILED` wrapper\n * with the original preserved as `cause`.\n */\nexport function toWrongStackError(\n err: unknown,\n code: Extract<ErrorCode, 'AGENT_RUN_FAILED' | 'AGENT_ABORTED' | 'UNKNOWN'> = ERROR_CODES.AGENT_RUN_FAILED,\n): WrongStackError {\n if (err instanceof WrongStackError) return err;\n const message = toErrorMessage(err);\n return new AgentError({\n message,\n code: code === 'UNKNOWN' ? ERROR_CODES.AGENT_RUN_FAILED : code,\n cause: err,\n });\n}\n\n/**\n * Session storage errors.\n */\nexport class SessionError extends WrongStackError {\n readonly sessionId?: string | undefined;\n\n constructor(opts: {\n message: string;\n code: Extract<ErrorCode, 'SESSION_NOT_FOUND' | 'SESSION_CORRUPTED' | 'SESSION_WRITE_FAILED'>;\n sessionId?: string | undefined;\n context?: Record<string, unknown> | undefined;\n cause?: unknown | undefined;\n }) {\n super({\n message: opts.message,\n code: opts.code,\n subsystem: 'session',\n severity: opts.code === ERROR_CODES.SESSION_WRITE_FAILED ? 'error' : 'warning',\n recoverable: opts.code !== ERROR_CODES.SESSION_CORRUPTED,\n context: { sessionId: opts.sessionId, ...opts.context },\n cause: opts.cause,\n });\n this.name = 'SessionError';\n this.sessionId = opts.sessionId;\n }\n}\n\n/**\n * SDD (Spec-Driven Development) errors — spec validation, parsing, and\n * state machine violations in the AISpecBuilder, TaskFlow, and TaskTracker.\n */\nexport class SddError extends WrongStackError {\n constructor(opts: {\n message: string;\n code: Extract<\n ErrorCode,\n 'SDD_VALIDATION_FAILED' | 'SDD_PARSE_FAILED' | 'SDD_INVALID_STATE' | 'SDD_NOT_READY'\n >;\n context?: Record<string, unknown> | undefined;\n cause?: unknown | undefined;\n }) {\n super({\n message: opts.message,\n code: opts.code,\n subsystem: 'sdd',\n severity: opts.code === ERROR_CODES.SDD_PARSE_FAILED ? 'warning' : 'error',\n recoverable: opts.code === ERROR_CODES.SDD_NOT_READY,\n context: opts.context,\n cause: opts.cause,\n });\n this.name = 'SddError';\n }\n}\n\n/**\n * File system operation errors.\n */\nexport class FsError extends WrongStackError {\n readonly path?: string | undefined;\n\n constructor(opts: {\n message: string;\n code: Extract<\n ErrorCode,\n 'FS_READ_FAILED' | 'FS_WRITE_FAILED' | 'FS_MKDIR_FAILED' | 'FS_DELETE_FAILED' | 'FS_ATOMIC_WRITE_FAILED'\n >;\n path?: string | undefined;\n context?: Record<string, unknown> | undefined;\n cause?: unknown | undefined;\n }) {\n super({\n message: opts.message,\n code: opts.code,\n subsystem: 'fs',\n severity: 'error',\n recoverable: opts.code !== ERROR_CODES.FS_READ_FAILED,\n context: { path: opts.path, ...opts.context },\n cause: opts.cause,\n });\n this.name = 'FsError';\n this.path = opts.path;\n }\n}\n\n/**\n * HTTP fetch error — thrown when a network request returns a non-OK status.\n * Carries the response status so {@link classifyToolError} can branch on it\n * (429 → transient, 404 → not_found, 401 → permission) without duck-typing\n * the error via `'response' in err`.\n *\n * P3 #18 (before-release.md): the previous `'response' in err` check caught\n * any Error with a `response` property, including custom errors, proxy\n * objects, or mocked errors in tests. `instanceof FetchError` is reliable.\n *\n * Tools and providers that make HTTP requests and need the executor to\n * classify their failures should throw `new FetchError({ status, message })`\n * instead of a bare `Error` with an ad-hoc `response` field.\n */\nexport class FetchError extends WrongStackError {\n readonly status: number;\n\n constructor(opts: {\n message: string;\n status: number;\n context?: Record<string, unknown> | undefined;\n cause?: unknown | undefined;\n }) {\n super({\n message: opts.message,\n code: ERROR_CODES.VALIDATION_ERROR,\n subsystem: 'general',\n severity: 'error',\n recoverable: opts.status === 429 || opts.status >= 500,\n context: { status: opts.status, ...opts.context },\n cause: opts.cause,\n });\n this.name = 'FetchError';\n this.status = opts.status;\n }\n}\n\n/**\n * Tool input validation error — thrown when a tool's input fails a validation\n * check that the JSON Schema cannot express (e.g. `old_string === new_string`\n * in edit, or a cross-field invariant). Use this instead of a bare\n * `throw new Error('...validation...')` so {@link classifyToolError} can\n * match on `instanceof` rather than a locale-dependent message substring.\n *\n * P2 #6 (before-release.md): the previous `err.message.includes('validation')`\n * check misclassified any error whose message happened to contain \"validation\"\n * (e.g. a third-party \"input validation timeout\") as a VALIDATION error.\n *\n * Named `ToolValidationError` (not `ValidationError`) to avoid colliding with\n * the existing `ValidationError` interface exported by json-schema-validate.ts\n * (a validation-result shape, not an Error subclass).\n */\nexport class ToolValidationError extends WrongStackError {\n constructor(opts: {\n message: string;\n /** Field path or tool name that failed validation, for diagnostics. */\n field?: string | undefined;\n context?: Record<string, unknown> | undefined;\n cause?: unknown | undefined;\n }) {\n super({\n message: opts.message,\n code: ERROR_CODES.VALIDATION_ERROR,\n subsystem: 'general',\n severity: 'error',\n recoverable: false,\n context: { field: opts.field, ...opts.context },\n cause: opts.cause,\n });\n this.name = 'ToolValidationError';\n }\n}\n\n// ── Type guards ──────────────────────────────────────────────────────\n\nexport function isWrongStackError(err: unknown): err is WrongStackError {\n return err instanceof WrongStackError;\n}\n\nexport function isToolError(err: unknown): err is ToolError {\n return err instanceof ToolError;\n}\n\nexport function isConfigError(err: unknown): err is ConfigError {\n return err instanceof ConfigError;\n}\n\nexport function isPluginError(err: unknown): err is PluginError {\n return err instanceof PluginError;\n}\n\nexport function isSessionError(err: unknown): err is SessionError {\n return err instanceof SessionError;\n}\n\nexport function isAgentError(err: unknown): err is AgentError {\n return err instanceof AgentError;\n}\n\nexport function isFsError(err: unknown): err is FsError {\n return err instanceof FsError;\n}\n\nexport function isToolValidationError(err: unknown): err is ToolValidationError {\n return err instanceof ToolValidationError;\n}\n\nexport function isFetchError(err: unknown): err is FetchError {\n return err instanceof FetchError;\n}\n\nexport function isSddError(err: unknown): err is SddError {\n return err instanceof SddError;\n}\n","import { expectDefined } from '../utils/expect-defined.js';\nimport * as fs from 'node:fs/promises';\nimport * as os from 'node:os';\nimport * as path from 'node:path';\nimport { createGunzip } from 'node:zlib';\nimport { Readable } from 'node:stream';\nimport { pipeline } from 'node:stream/promises';\nimport { WrongStackError, ERROR_CODES } from '../types/errors.js';\nexport interface ParsedRef {\n owner: string;\n repo: string;\n ref: string;\n}\n\n/**\n * Parse a skill reference string.\n * Formats: `user/repo` (default ref: main), `user/repo@ref`\n */\nexport function parseSkillRef(input: string): ParsedRef {\n const trimmed = input.trim().replace(/^https?:\\/\\/github\\.com\\//, '').replace(/\\.git$/, '');\n const atIdx = trimmed.indexOf('@');\n let refPath: string;\n let ref: string;\n if (atIdx > 0) {\n refPath = trimmed.slice(0, atIdx);\n ref = trimmed.slice(atIdx + 1);\n } else {\n refPath = trimmed;\n ref = 'main';\n }\n const parts = refPath.split('/').filter(Boolean);\n if (parts.length < 2) {\n throw new WrongStackError({\n message: `Invalid skill reference \"${input}\". Expected format: user/repo or user/repo@ref`,\n code: ERROR_CODES.UNKNOWN,\n subsystem: 'general',\n context: { input },\n });\n }\n return { owner: expectDefined(parts[0]), repo: expectDefined(parts[1]), ref };\n}\n\nexport interface DownloadResult {\n /** Temp directory containing the extracted repo. Caller must clean up. */\n tempDir: string;\n}\n\nconst MAX_TARBALL_SIZE = 50 * 1024 * 1024; // 50MB\n\n/**\n * Download and extract a GitHub repository tarball.\n * Uses the public GitHub API — no auth token required for public repos.\n * Returns the path to a temp directory with the extracted contents.\n */\nexport async function downloadGitHubTarball(parsed: ParsedRef): Promise<DownloadResult> {\n const url = `https://api.github.com/repos/${parsed.owner}/${parsed.repo}/tarball/${parsed.ref}`;\n\n const response = await fetch(url, {\n signal: AbortSignal.timeout(30_000),\n headers: {\n Accept: 'application/vnd.github+json',\n 'User-Agent': 'wrongstack-skill-installer',\n },\n redirect: 'follow',\n });\n\n if (!response.ok) {\n if (response.status === 404) {\n throw new WrongStackError({\n message:\n `Repository not found: ${parsed.owner}/${parsed.repo}` +\n (parsed.ref !== 'main' ? ` (ref: ${parsed.ref})` : ''),\n code: ERROR_CODES.UNKNOWN,\n subsystem: 'general',\n context: { owner: parsed.owner, repo: parsed.repo, ref: parsed.ref, status: 404 },\n });\n }\n if (response.status === 403) {\n throw new WrongStackError({\n message: `Access denied: ${parsed.owner}/${parsed.repo}. The repository may be private or rate-limited.`,\n code: ERROR_CODES.UNKNOWN,\n subsystem: 'general',\n context: { owner: parsed.owner, repo: parsed.repo, status: 403 },\n });\n }\n throw new WrongStackError({\n message: `GitHub API error (${response.status}): ${response.statusText}`,\n code: ERROR_CODES.UNKNOWN,\n subsystem: 'general',\n context: { owner: parsed.owner, repo: parsed.repo, status: response.status },\n });\n }\n\n const contentLength = response.headers.get('content-length');\n if (contentLength && Number.parseInt(contentLength, 10) > MAX_TARBALL_SIZE) {\n throw new WrongStackError({\n message:\n `Tarball too large (${(Number.parseInt(contentLength, 10) / 1024 / 1024).toFixed(1)}MB). ` +\n `Max: ${MAX_TARBALL_SIZE / 1024 / 1024}MB`,\n code: ERROR_CODES.UNKNOWN,\n subsystem: 'general',\n context: {\n owner: parsed.owner,\n repo: parsed.repo,\n contentLengthBytes: Number.parseInt(contentLength, 10),\n maxBytes: MAX_TARBALL_SIZE,\n },\n });\n }\n\n const tempDir = await fs.mkdtemp(path.join(os.tmpdir(), 'wskill-'));\n\n try {\n if (!response.body) {\n throw new WrongStackError({\n message: 'Empty response body from GitHub API',\n code: ERROR_CODES.UNKNOWN,\n subsystem: 'general',\n context: { owner: parsed.owner, repo: parsed.repo },\n });\n }\n\n // Gunzip the response body, then extract the tar stream\n const nodeStream = Readable.fromWeb(response.body as import('node:stream/web').ReadableStream);\n const gunzip = createGunzip();\n\n // Collect the uncompressed tar data into a buffer, then extract\n const chunks: Buffer[] = [];\n await pipeline(nodeStream, gunzip, async (source) => {\n for await (const chunk of source) {\n chunks.push(Buffer.from(chunk));\n }\n });\n const tarBuf = Buffer.concat(chunks);\n\n // Extract tar archive (POSIX ustar format)\n await extractTar(tarBuf, tempDir);\n\n return { tempDir };\n } catch (err) {\n await fs.rm(tempDir, { recursive: true, force: true }).catch(() => {});\n throw err;\n }\n}\n\n/**\n * Minimal POSIX tar extractor. Handles the ustar format produced by GitHub.\n * Only extracts regular files and directories — symlinks and special entries\n * are skipped for security.\n */\nasync function extractTar(buf: Buffer, destDir: string): Promise<void> {\n let offset = 0;\n\n while (offset + 512 <= buf.length) {\n // Check for end-of-archive (two consecutive zero blocks)\n const header = buf.subarray(offset, offset + 512);\n if (header.every((b) => b === 0)) break;\n\n // Parse ustar header\n const name = readTarString(buf, offset, 100); // name field\n const prefix = readTarString(buf, offset + 345, 155); // ustar prefix\n const size = Number.parseInt(readTarString(buf, offset + 124, 12).trim(), 8) || 0;\n const typeflag = buf[offset + 156] ?? 0;\n\n // Full path: prefix/name (ustar) or just name\n const fullPath = prefix ? `${prefix}/${name}` : name;\n // Strip the top-level directory (GitHub tarballs have owner-repo-sha/)\n const relPath = stripTopDir(fullPath);\n\n if (relPath && relPath !== '.' && relPath !== '..') {\n const destPath = path.join(destDir, relPath);\n\n // Zip-slip guard: reject any entry whose resolved path escapes destDir\n // (e.g. a crafted entry name like `x/../../../etc/cron.d/evil`). GitHub\n // tarballs are built from git trees and so cannot carry `..` components,\n // but this extractor is generic — never trust archive entry names.\n const resolvedDest = path.resolve(destPath);\n const resolvedRoot = path.resolve(destDir);\n if (resolvedDest !== resolvedRoot && !resolvedDest.startsWith(resolvedRoot + path.sep)) {\n // Skip the entry entirely; advance past its data below.\n offset += 512 + Math.ceil(size / 512) * 512;\n continue;\n }\n\n // typeflag: '0' or '\\0' = regular file, '5' = directory\n if (typeflag === 0x35 || typeflag === 0) {\n // Directory\n if (relPath.endsWith('/') || typeflag === 0x35) {\n await fs.mkdir(destPath, { recursive: true });\n }\n }\n\n if ((typeflag === 0x30 || typeflag === 0 || typeflag === 0x00) && size > 0) {\n // Regular file\n const dir = path.dirname(destPath);\n await fs.mkdir(dir, { recursive: true });\n const dataStart = offset + 512;\n const dataEnd = dataStart + size;\n if (dataEnd > buf.length) break; // truncated archive\n await fs.writeFile(destPath, buf.subarray(dataStart, dataEnd));\n }\n }\n\n // Advance: 512-byte header + data padded to 512-byte boundary\n offset += 512 + Math.ceil(size / 512) * 512;\n }\n}\n\nfunction readTarString(buf: Buffer, start: number, maxLen: number): string {\n let end = start;\n while (end < start + maxLen && end < buf.length && buf[end] !== 0) {\n end++;\n }\n return buf.subarray(start, end).toString('utf8');\n}\n\n/**\n * Strip the top-level directory from a tar path.\n * GitHub tarballs have a single root dir like `owner-repo-sha/`.\n */\nfunction stripTopDir(p: string): string {\n const idx = p.indexOf('/');\n if (idx === -1) return ''; // top-level file, skip\n return p.slice(idx + 1);\n}\n","import * as fs from 'node:fs/promises';\r\nimport * as path from 'node:path';\r\nimport { atomicWrite } from '../utils/atomic-write.js';\r\n\r\nexport interface InstalledSkillEntry {\r\n name: string;\r\n /** Source identifier, e.g. \"github:user/repo\" */\r\n source: string;\r\n /** Git ref that was installed (branch, tag, commit) */\r\n ref: string;\r\n /** Installation scope */\r\n scope: 'project' | 'user';\r\n /** Project hash — only set when scope=project */\r\n projectHash?: string | undefined;\r\n /** ISO 8601 timestamp */\r\n installedAt: string;\r\n /** List of files that were installed (relative to skill dir) */\r\n files: string[];\r\n}\r\n\r\nexport interface ManifestData {\r\n skills: InstalledSkillEntry[];\r\n}\r\n\r\nexport class SkillManifestStore {\r\n private readonly manifestPath: string;\r\n private cache?: ManifestData | undefined;\r\n\r\n constructor(manifestPath: string) {\r\n this.manifestPath = manifestPath;\r\n }\r\n\r\n async read(): Promise<ManifestData> {\r\n if (this.cache) return this.cache;\r\n try {\r\n const raw = await fs.readFile(this.manifestPath, 'utf8');\r\n const data = JSON.parse(raw) as ManifestData;\r\n if (!Array.isArray(data.skills)) {\r\n this.cache = { skills: [] };\r\n } else {\r\n this.cache = data;\r\n }\r\n } catch {\r\n this.cache = { skills: [] };\r\n }\r\n return this.cache;\r\n }\r\n\r\n async write(data: ManifestData): Promise<void> {\r\n const dir = path.dirname(this.manifestPath);\r\n await fs.mkdir(dir, { recursive: true });\r\n await atomicWrite(this.manifestPath, JSON.stringify(data, null, 2) + '\\n');\r\n this.cache = data;\r\n }\r\n\r\n async addEntry(entry: InstalledSkillEntry): Promise<void> {\r\n const data = await this.read();\r\n // Remove existing entry with the same name + scope\r\n data.skills = data.skills.filter(\r\n (s) => !(s.name === entry.name && s.scope === entry.scope),\r\n );\r\n data.skills.push(entry);\r\n await this.write(data);\r\n }\r\n\r\n async removeEntry(name: string, scope: 'project' | 'user'): Promise<boolean> {\r\n const data = await this.read();\r\n const before = data.skills.length;\r\n data.skills = data.skills.filter(\r\n (s) => !(s.name === name && s.scope === scope),\r\n );\r\n if (data.skills.length === before) return false;\r\n await this.write(data);\r\n return true;\r\n }\r\n\r\n async findByName(name: string): Promise<InstalledSkillEntry[]> {\r\n const data = await this.read();\r\n return data.skills.filter((s) => s.name === name);\r\n }\r\n\r\n async findBySource(source: string): Promise<InstalledSkillEntry[]> {\r\n const data = await this.read();\r\n return data.skills.filter((s) => s.source === source);\r\n }\r\n\r\n async listAll(): Promise<InstalledSkillEntry[]> {\r\n const data = await this.read();\r\n return data.skills;\r\n }\r\n\r\n /** Invalidate the in-memory cache (e.g. after external file changes). */\r\n invalidateCache(): void {\r\n this.cache = undefined;\r\n }\r\n}\r\n","import { expectDefined } from '../utils/expect-defined.js';\nimport { toErrorMessage } from '../utils/error.js';\nimport * as fs from 'node:fs/promises';\nimport * as path from 'node:path';\nimport type { SkillLoader } from '../types/skill.js';\nimport { downloadGitHubTarball, parseSkillRef } from './github-fetcher.js';\nimport { type InstalledSkillEntry, SkillManifestStore } from './manifest-store.js';\nimport { FsError, WrongStackError, ERROR_CODES } from '../types/errors.js';\nexport interface SkillInstallerOptions {\n /** Path to the manifest file (~/.wrongstack/installed-skills.json) */\n manifestPath: string;\n /** Path to project-level skills dir (<project>/.wrongstack/skills/) */\n projectSkillsDir: string;\n /** Path to user-global skills dir (~/.wrongstack/skills/) */\n globalSkillsDir: string;\n /** Current project hash (for manifest tracking) */\n projectHash: string;\n /** Skill loader — cache will be invalidated after mutations */\n skillLoader?: SkillLoader | undefined;\n /** Logger for status messages */\n log?: (((msg: string) => void)) | undefined;\n}\n\nexport interface InstallResult {\n name: string;\n path: string;\n scope: 'project' | 'user';\n source: string;\n ref: string;\n skillCount: number;\n}\n\nexport interface UpdateResult {\n updated: Array<{ name: string; oldRef: string; newRef: string }>;\n unchanged: string[];\n errors: Array<{ name: string; error: string }>;\n}\n\nconst MAX_SKILL_FILE_SIZE = 100 * 1024; // 100KB\n\nexport class SkillInstaller {\n private readonly opts: SkillInstallerOptions;\n private readonly manifest: SkillManifestStore;\n\n constructor(opts: SkillInstallerOptions) {\n this.opts = opts;\n this.manifest = new SkillManifestStore(opts.manifestPath);\n }\n\n /**\n * Install skills from a GitHub repository.\n * Supports both single-skill repos (SKILL.md at root) and multi-skill repos (skills/ subdirectory).\n */\n async install(refInput: string, opts?: { global?: boolean | undefined }): Promise<InstallResult[]> {\n const parsed = parseSkillRef(refInput);\n const scope: 'project' | 'user' = opts?.global ? 'user' : 'project';\n const targetDir = scope === 'project' ? this.opts.projectSkillsDir : this.opts.globalSkillsDir;\n const source = `github:${parsed.owner}/${parsed.repo}`;\n\n this.opts.log?.(`Downloading ${parsed.owner}/${parsed.repo}@${parsed.ref}...`);\n\n const { tempDir } = await downloadGitHubTarball(parsed);\n\n try {\n // Detect skill structure\n const skills = await this.detectSkills(tempDir);\n\n if (skills.length === 0) {\n throw new WrongStackError({\n message: 'No skills found in repository. Expected SKILL.md at root or skills/ subdirectory.',\n code: ERROR_CODES.VALIDATION_ERROR,\n subsystem: 'general',\n context: { owner: parsed.owner, repo: parsed.repo, ref: parsed.ref },\n });\n }\n\n const results: InstallResult[] = [];\n\n for (const skill of skills) {\n // Check for overwrite\n const existing = await this.manifest.findByName(skill.name);\n const existingInScope = existing.find((e) => e.scope === scope);\n if (existingInScope) {\n this.opts.log?.(`Overwriting existing skill \"${skill.name}\" (${scope})...`);\n await this.removeSkillFiles(skill.name, scope);\n }\n\n // Copy skill files\n const destDir = path.join(targetDir, skill.name);\n await fs.mkdir(destDir, { recursive: true });\n const copiedFiles: string[] = [];\n\n for (const file of skill.files) {\n const srcPath = path.join(skill.baseDir, file);\n const destPath = path.join(destDir, file);\n\n // Path traversal check\n const resolved = path.resolve(destPath);\n if (!resolved.startsWith(path.resolve(destDir))) {\n throw new FsError({\n message: `Path traversal detected in skill file: ${file}`,\n code: ERROR_CODES.FS_DELETE_FAILED,\n path: destPath,\n context: { reason: 'path_traversal', skillName: skill.name },\n });\n }\n\n // Size check\n const stat = await fs.stat(srcPath);\n if (stat.size > MAX_SKILL_FILE_SIZE) {\n throw new FsError({\n message:\n `Skill file \"${file}\" is too large (${(stat.size / 1024).toFixed(1)}KB). ` +\n `Max: ${MAX_SKILL_FILE_SIZE / 1024}KB`,\n code: ERROR_CODES.FS_WRITE_FAILED,\n path: srcPath,\n context: { skillName: skill.name, fileSize: stat.size, maxSize: MAX_SKILL_FILE_SIZE },\n });\n }\n\n await fs.mkdir(path.dirname(destPath), { recursive: true });\n await fs.copyFile(srcPath, destPath);\n copiedFiles.push(file);\n }\n\n // Write manifest entry\n const entry: InstalledSkillEntry = {\n name: skill.name,\n source,\n ref: parsed.ref,\n scope,\n projectHash: scope === 'project' ? this.opts.projectHash : undefined,\n installedAt: new Date().toISOString(),\n files: copiedFiles,\n };\n await this.manifest.addEntry(entry);\n\n results.push({\n name: skill.name,\n path: destDir,\n scope,\n source,\n ref: parsed.ref,\n skillCount: 1,\n });\n }\n\n this.invalidateLoaderCache();\n return results;\n } finally {\n // Clean up temp directory\n await fs.rm(tempDir, { recursive: true, force: true }).catch(() => {});\n }\n }\n\n /**\n * Update installed skills.\n * - No args: update all\n * - Name: update that specific skill\n * - Name + newRef: update to a different ref\n */\n async update(\n nameOrRef?: string | undefined,\n _opts?: { global?: boolean | undefined } | undefined,\n ): Promise<UpdateResult> {\n const result: UpdateResult = { updated: [], unchanged: [], errors: [] };\n const allEntries = await this.manifest.listAll();\n\n let targets: InstalledSkillEntry[];\n if (nameOrRef) {\n // Check if it's a name or a ref (user/repo@ref)\n const byName = allEntries.filter((e) => e.name === nameOrRef);\n if (byName.length > 0) {\n targets = byName;\n } else {\n // Treat as a new ref — find matching source\n try {\n const parsed = parseSkillRef(nameOrRef);\n const source = `github:${parsed.owner}/${parsed.repo}`;\n targets = allEntries.filter((e) => e.source === source);\n if (targets.length === 0) {\n result.errors.push({\n name: nameOrRef,\n error: `No installed skills found matching \"${nameOrRef}\"`,\n });\n return result;\n }\n } catch {\n result.errors.push({\n name: nameOrRef,\n error: `Invalid reference: ${nameOrRef}`,\n });\n return result;\n }\n }\n } else {\n targets = allEntries;\n }\n\n // Group by source to avoid downloading the same repo multiple times\n const bySource = new Map<string, InstalledSkillEntry[]>();\n for (const entry of targets) {\n const key = `${entry.source}@${entry.ref}`;\n if (!bySource.has(key)) bySource.set(key, []);\n bySource.get(key)?.push(entry);\n }\n\n for (const [, entries] of bySource) {\n const first = expectDefined(entries[0]);\n const scope = first.scope;\n const isGlobal = scope === 'user';\n\n try {\n // Parse the original source to get the ref\n const sourceRepo = first.source.replace('github:', '');\n let refToInstall = first.ref;\n\n // If nameOrRef looks like a new ref, use it\n if (nameOrRef && !allEntries.find((e) => e.name === nameOrRef)) {\n try {\n const parsed = parseSkillRef(nameOrRef);\n refToInstall = parsed.ref;\n } catch {\n // keep original ref\n }\n }\n\n this.opts.log?.(`Updating ${first.source}@${refToInstall}...`);\n const results = await this.install(`${sourceRepo}@${refToInstall}`, { global: isGlobal });\n\n for (const r of results) {\n result.updated.push({\n name: r.name,\n oldRef: first.ref,\n newRef: refToInstall,\n });\n }\n } catch (err) {\n const msg = toErrorMessage(err);\n for (const entry of entries) {\n result.errors.push({ name: entry.name, error: msg });\n }\n }\n }\n\n return result;\n }\n\n /**\n * Uninstall a skill by name.\n */\n async uninstall(name: string, opts?: { global?: boolean | undefined }): Promise<void> {\n const scope: 'project' | 'user' = opts?.global ? 'user' : 'project';\n const entries = await this.manifest.findByName(name);\n const entry = entries.find((e) => e.scope === scope);\n\n if (!entry) {\n throw new WrongStackError({\n message: `Skill \"${name}\" is not installed${scope === 'user' ? ' (global)' : ''}.`,\n code: ERROR_CODES.VALIDATION_ERROR,\n subsystem: 'general',\n context: { skillName: name, scope },\n });\n }\n\n // Remove files\n await this.removeSkillFiles(name, scope);\n\n // Remove from manifest\n await this.manifest.removeEntry(name, scope);\n this.invalidateLoaderCache();\n }\n\n /**\n * List all installed skills from the manifest.\n */\n async listInstalled(): Promise<InstalledSkillEntry[]> {\n return this.manifest.listAll();\n }\n\n // ── Private helpers ──────────────────────────────────────────────\n\n /**\n * Detect skills in an extracted repository.\n * Returns an array of detected skills with their files.\n */\n private async detectSkills(\n baseDir: string,\n ): Promise<Array<{ name: string; baseDir: string; files: string[] }>> {\n const results: Array<{ name: string; baseDir: string; files: string[] }> = [];\n\n // Check for SKILL.md at root (single-skill repo)\n const rootSkillMd = path.join(baseDir, 'SKILL.md');\n try {\n await fs.access(rootSkillMd);\n const content = await fs.readFile(rootSkillMd, 'utf8');\n const meta = parseFrontmatter(content);\n if (meta.name && meta.description) {\n results.push({\n name: meta.name,\n baseDir,\n files: ['SKILL.md'],\n });\n return results; // Single-skill repo — don't look for skills/\n }\n } catch {\n // No root SKILL.md\n }\n\n // Check for skills/ subdirectory (multi-skill repo)\n const skillsDir = path.join(baseDir, 'skills');\n try {\n const entries = await fs.readdir(skillsDir, { withFileTypes: true });\n for (const entry of entries) {\n if (!entry.isDirectory()) continue;\n const skillFile = path.join(skillsDir, entry.name, 'SKILL.md');\n try {\n const content = await fs.readFile(skillFile, 'utf8');\n const meta = parseFrontmatter(content);\n if (meta.name && meta.description) {\n // Collect all files in the skill directory\n const skillDir = path.join(skillsDir, entry.name);\n const files = await collectFiles(skillDir, skillDir);\n results.push({\n name: meta.name,\n baseDir: skillDir,\n files,\n });\n }\n } catch {\n // Skip malformed skills\n }\n }\n } catch {\n // No skills/ directory\n }\n\n return results;\n }\n\n /**\n * Remove all files for an installed skill.\n */\n private async removeSkillFiles(name: string, scope: 'project' | 'user'): Promise<void> {\n const targetDir =\n scope === 'project' ? this.opts.projectSkillsDir : this.opts.globalSkillsDir;\n const skillDir = path.join(targetDir, name);\n await fs.rm(skillDir, { recursive: true, force: true });\n }\n\n /**\n * Invalidate the skill loader's cache so newly installed skills appear.\n */\n private invalidateLoaderCache(): void {\n // The SkillLoader interface has a cache internally.\n // We access it via the 'any' cast to call invalidateCache if available.\n const loader = this.opts.skillLoader as never as { invalidateCache?: () => void };\n if (loader && typeof loader.invalidateCache === 'function') {\n loader.invalidateCache();\n }\n }\n}\n\n// ── Utilities ──────────────────────────────────────────────────────\n\ninterface Frontmatter {\n name?: string | undefined;\n description?: string | undefined;\n}\n\nfunction parseFrontmatter(raw: string): Frontmatter {\n if (!raw.startsWith('---')) return {};\n const end = raw.indexOf('\\n---', 4);\n if (end === -1) return {};\n const block = raw.slice(4, end);\n const out: Frontmatter = {};\n let key: keyof Frontmatter | null = null;\n let value: string[] = [];\n const flush = () => {\n if (key) {\n out[key] = value.join('\\n').trim();\n }\n key = null;\n value = [];\n };\n for (const line of block.split('\\n')) {\n const m = /^([a-zA-Z_]+):\\s*(\\|?)\\s*(.*)$/.exec(line);\n if (m) {\n flush();\n key = (m[1] ?? '') as keyof Frontmatter;\n const pipe = m[2];\n const rest = m[3] ?? '';\n if (pipe === '|') {\n value = [];\n } else if (rest) {\n value = [rest];\n } else {\n value = [];\n }\n } else if (key) {\n value.push(line.replace(/^\\s+/, ''));\n }\n }\n flush();\n return out;\n}\n\n/**\n * Recursively collect all files in a directory (relative paths).\n */\nasync function collectFiles(dir: string, baseDir: string): Promise<string[]> {\n const results: string[] = [];\n const entries = await fs.readdir(dir, { withFileTypes: true });\n for (const entry of entries) {\n const fullPath = path.join(dir, entry.name);\n const relPath = path.relative(baseDir, fullPath);\n if (entry.isDirectory()) {\n // Skip hidden dirs and node_modules\n if (entry.name.startsWith('.') || entry.name === 'node_modules') continue;\n results.push(...(await collectFiles(fullPath, baseDir)));\n } else {\n results.push(relPath);\n }\n }\n return results;\n}\n"]}
1
+ {"version":3,"sources":["../../src/utils/expect-defined.ts","../../src/utils/error.ts","../../src/utils/atomic-write.ts","../../src/types/errors.ts","../../src/skills/github-fetcher.ts","../../src/skills/manifest-store.ts","../../src/skills/skill-installer.ts"],"names":["path","fs","stat","resolve","fs2","path2","fs3","path3"],"mappings":"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","file":"index.js","sourcesContent":["/** Assert a value is neither null nor undefined. Throws if it is.\n * Useful after optional chaining and indexed access when the\n * control flow guarantees the value exists but TypeScript can't\n * prove it (e.g. after a check on a related field). */\nexport function expectDefined<T>(value: T | null | undefined, label?: string): T {\n if (value === null || value === undefined) {\n const err = new Error(label ? `Expected ${label} to be defined` : 'Expected value to be defined');\n err.name = 'ExpectDefinedError';\n throw err;\n }\n return value;\n}\n","/**\n * Converts an unknown error value to a human-readable string.\n * Used in 40+ files across the codebase to normalize error messaging.\n */\nexport function toErrorMessage(err: unknown): string {\n return err instanceof Error ? err.message : String(err);\n}\n","import { randomBytes } from 'node:crypto';\nimport * as fs from 'node:fs/promises';\nimport * as path from 'node:path';\nimport { FsError } from '../types/errors.js';\n\nexport interface AtomicWriteOptions {\n mode?: number | undefined;\n encoding?: BufferEncoding | undefined;\n}\n\nexport interface FileLockOptions {\n timeoutMs?: number | undefined;\n staleMs?: number | undefined;\n}\n\nexport async function atomicWrite(\n targetPath: string,\n content: string | Uint8Array,\n opts: AtomicWriteOptions = {},\n): Promise<void> {\n const dir = path.dirname(targetPath);\n await fs.mkdir(dir, { recursive: true });\n const tmp = path.join(dir, `.${path.basename(targetPath)}.${randomBytes(6).toString('hex')}.tmp`);\n\n // Write content to tmp first; 'wx' ensures exclusive creation (fails if\n // tmp already exists — extremely unlikely with 6-byte random suffix).\n try {\n if (typeof content === 'string') {\n await fs.writeFile(tmp, content, { flag: 'wx', encoding: opts.encoding ?? 'utf8' });\n } else {\n await fs.writeFile(tmp, content, { flag: 'wx' });\n }\n try {\n const fh = await fs.open(tmp, 'r+');\n try {\n await fh.sync();\n } finally {\n await fh.close();\n }\n } catch {\n // fsync best-effort\n }\n // Now safely read mode from target (if it exists) and apply to tmp before rename.\n // Prefer opts.mode for new files; for existing files preserve their mode.\n let mode: number | undefined;\n try {\n const stat = await fs.stat(targetPath);\n mode = stat.mode & 0o777;\n } catch {\n mode = opts.mode;\n }\n if (mode !== undefined) {\n await fs.chmod(tmp, mode);\n }\n await renameWithRetry(tmp, targetPath);\n // P3 #20 (before-release.md): on Windows, fs.rename (MoveFileExW) does\n // not preserve Unix permission bits — the chmod above applies to the tmp\n // file, but the rename may reset the destination's mode to the Windows\n // default. Re-apply the mode after rename on win32 so an edited file\n // keeps its executable bit (or any non-default permission). On POSIX,\n // rename preserves metadata so this is a no-op (chmod is idempotent and\n // cheap), but we gate it on win32 to avoid the extra stat+chmod on the\n // common path.\n if (mode !== undefined && process.platform === 'win32') {\n try {\n await fs.chmod(targetPath, mode);\n } catch {\n // Best-effort: a transient EPERM (antivirus lock) should not fail\n // the write — the content is already on disk.\n }\n }\n } catch (err) {\n try {\n await fs.unlink(tmp);\n } catch {\n // ignore cleanup error\n }\n throw err;\n }\n}\n\nexport async function ensureDir(dir: string): Promise<void> {\n await fs.mkdir(dir, { recursive: true });\n}\n\nexport async function withFileLock<T>(\n targetPath: string,\n fn: () => Promise<T>,\n opts: FileLockOptions = {},\n): Promise<T> {\n const dir = path.dirname(targetPath);\n await fs.mkdir(dir, { recursive: true });\n const lockPath = path.join(dir, `.${path.basename(targetPath)}.lock`);\n const timeoutMs = opts.timeoutMs ?? 5_000;\n const staleMs = opts.staleMs ?? 30_000;\n const started = Date.now();\n let handle: fs.FileHandle | undefined;\n\n for (;;) {\n try {\n handle = await fs.open(lockPath, 'wx');\n await handle.writeFile(`${process.pid}:${Date.now()}`);\n break;\n } catch (err) {\n const code = (err as NodeJS.ErrnoException).code;\n // ENOENT means the directory was deleted (e.g. by concurrent cleanup).\n // Recreate it and retry acquiring the lock.\n if (code === 'ENOENT') {\n await fs.mkdir(dir, { recursive: true });\n continue;\n }\n if (code !== 'EEXIST') throw err;\n try {\n const stat = await fs.stat(lockPath);\n if (Date.now() - stat.mtimeMs > staleMs) {\n await fs.unlink(lockPath);\n continue;\n }\n } catch {\n continue;\n }\n if (Date.now() - started >= timeoutMs) {\n throw new FsError({\n message: `Timed out waiting for file lock: ${targetPath}`,\n code: 'FS_ATOMIC_WRITE_FAILED',\n path: targetPath,\n context: { timeoutMs },\n });\n }\n await new Promise((resolve) => setTimeout(resolve, 25));\n }\n }\n\n try {\n return await fn();\n } finally {\n try {\n await handle?.close();\n } catch {\n // ignore\n }\n try {\n await fs.unlink(lockPath);\n } catch {\n // ignore\n }\n }\n}\n\n// On Windows, fs.rename over an existing file can fail with EPERM/EBUSY/EACCES\n// when antivirus, file indexers, editor file watchers, or a concurrent writer\n// briefly hold a handle on the destination. These are transient — retry with a\n// short backoff before giving up. POSIX renames are atomic and won't hit this.\nconst TRANSIENT_RENAME_CODES = new Set(['EPERM', 'EBUSY', 'EACCES', 'ENOTEMPTY']);\n\nasync function renameWithRetry(from: string, to: string): Promise<void> {\n if (process.platform !== 'win32') {\n await fs.rename(from, to);\n return;\n }\n const delays = [10, 25, 60, 120, 250];\n let lastErr: unknown;\n for (let i = 0; i <= delays.length; i++) {\n try {\n await fs.rename(from, to);\n return;\n } catch (err) {\n lastErr = err;\n const code = (err as NodeJS.ErrnoException)?.code;\n if (!code || !TRANSIENT_RENAME_CODES.has(code) || i === delays.length) {\n throw err;\n }\n await new Promise((resolve) => setTimeout(resolve, delays[i]));\n }\n }\n throw lastErr;\n}\n","import { toErrorMessage } from '../utils/index.js';\n\n/**\n * WrongStack error hierarchy.\n *\n * Every error thrown by the framework is a `WrongStackError` with a\n * machine-readable `code`, a `subsystem` tag, and a `severity` level.\n * This lets consumers (CLI, TUI, plugins, tests) branch on structured\n * data instead of parsing error messages.\n */\n\n// ── Error codes ──────────────────────────────────────────────────────\n\n/**\n * Machine-readable error codes as frozen constants.\n *\n * Use `ERROR_CODES.X` instead of raw string literals for:\n * - IDE autocomplete and compile-time validation\n * - Safe refactoring (rename updates all usages)\n * - Plugin extensibility (extend the object to add custom codes)\n *\n * The `ErrorCode` type is derived from this object, so adding a new\n * code here automatically updates the type without extra changes.\n */\nexport const ERROR_CODES = {\n // Provider\n PROVIDER_RATE_LIMITED: 'PROVIDER_RATE_LIMITED',\n PROVIDER_AUTH_FAILED: 'PROVIDER_AUTH_FAILED',\n PROVIDER_OVERLOADED: 'PROVIDER_OVERLOADED',\n PROVIDER_INVALID_REQUEST: 'PROVIDER_INVALID_REQUEST',\n PROVIDER_SERVER_ERROR: 'PROVIDER_SERVER_ERROR',\n PROVIDER_NETWORK_ERROR: 'PROVIDER_NETWORK_ERROR',\n PROVIDER_CONTEXT_OVERFLOW: 'PROVIDER_CONTEXT_OVERFLOW',\n // Tool\n TOOL_NOT_FOUND: 'TOOL_NOT_FOUND',\n TOOL_PERMISSION_DENIED: 'TOOL_PERMISSION_DENIED',\n TOOL_EXECUTION_FAILED: 'TOOL_EXECUTION_FAILED',\n TOOL_TIMEOUT: 'TOOL_TIMEOUT',\n TOOL_INPUT_INVALID: 'TOOL_INPUT_INVALID',\n // Config\n CONFIG_INVALID: 'CONFIG_INVALID',\n CONFIG_NOT_FOUND: 'CONFIG_NOT_FOUND',\n CONFIG_PARSE_FAILED: 'CONFIG_PARSE_FAILED',\n CONFIG_MIGRATION_NEEDED: 'CONFIG_MIGRATION_NEEDED',\n // Plugin\n PLUGIN_LOAD_FAILED: 'PLUGIN_LOAD_FAILED',\n PLUGIN_API_MISMATCH: 'PLUGIN_API_MISMATCH',\n PLUGIN_MISSING_DEPENDENCY: 'PLUGIN_MISSING_DEPENDENCY',\n // Agent\n AGENT_ITERATION_LIMIT: 'AGENT_ITERATION_LIMIT',\n AGENT_CONTEXT_OVERFLOW: 'AGENT_CONTEXT_OVERFLOW',\n AGENT_ABORTED: 'AGENT_ABORTED',\n AGENT_RUN_FAILED: 'AGENT_RUN_FAILED',\n // Session\n SESSION_NOT_FOUND: 'SESSION_NOT_FOUND',\n SESSION_CORRUPTED: 'SESSION_CORRUPTED',\n SESSION_WRITE_FAILED: 'SESSION_WRITE_FAILED',\n // Container / Registry\n CONTAINER_TOKEN_ALREADY_BOUND: 'CONTAINER_TOKEN_ALREADY_BOUND',\n CONTAINER_TOKEN_NOT_BOUND: 'CONTAINER_TOKEN_NOT_BOUND',\n CONTAINER_CIRCULAR_DEPENDENCY: 'CONTAINER_CIRCULAR_DEPENDENCY',\n REGISTRY_DUPLICATE: 'REGISTRY_DUPLICATE',\n REGISTRY_NOT_FOUND: 'REGISTRY_NOT_FOUND',\n REGISTRY_INVALID: 'REGISTRY_INVALID',\n // File system\n FS_READ_FAILED: 'FS_READ_FAILED',\n FS_WRITE_FAILED: 'FS_WRITE_FAILED',\n FS_MKDIR_FAILED: 'FS_MKDIR_FAILED',\n FS_DELETE_FAILED: 'FS_DELETE_FAILED',\n FS_ATOMIC_WRITE_FAILED: 'FS_ATOMIC_WRITE_FAILED',\n // SDD (Spec-Driven Development)\n SDD_VALIDATION_FAILED: 'SDD_VALIDATION_FAILED',\n SDD_PARSE_FAILED: 'SDD_PARSE_FAILED',\n SDD_INVALID_STATE: 'SDD_INVALID_STATE',\n SDD_NOT_READY: 'SDD_NOT_READY',\n // General\n VALIDATION_ERROR: 'VALIDATION_ERROR',\n PARSE_FAILED: 'PARSE_FAILED',\n UNKNOWN: 'UNKNOWN',\n} as const;\n\n/**\n * Union type derived from `ERROR_CODES`. Using `typeof ERROR_CODES[keyof typeof ERROR_CODES]`\n * instead of a string literal union means TypeScript auto-updates the type whenever\n * a new code is added to `ERROR_CODES` — no need to keep two lists in sync.\n */\nexport type ErrorCode = (typeof ERROR_CODES)[keyof typeof ERROR_CODES];\n\nexport type ErrorSubsystem =\n | 'provider'\n | 'tool'\n | 'config'\n | 'plugin'\n | 'agent'\n | 'session'\n | 'sdd'\n | 'container'\n | 'fs'\n | 'general';\nexport type ErrorSeverity = 'fatal' | 'error' | 'warning';\n\n// ── Base error class ─────────────────────────────────────────────────\n\nexport class WrongStackError extends Error {\n readonly code: ErrorCode;\n readonly subsystem: ErrorSubsystem;\n readonly severity: ErrorSeverity;\n readonly recoverable: boolean;\n readonly context?: Record<string, unknown> | undefined;\n\n constructor(opts: {\n message: string;\n code: ErrorCode;\n subsystem: ErrorSubsystem;\n severity?: ErrorSeverity | undefined;\n recoverable?: boolean | undefined;\n context?: Record<string, unknown> | undefined;\n cause?: unknown | undefined;\n }) {\n super(opts.message, { cause: opts.cause });\n this.name = 'WrongStackError';\n this.code = opts.code;\n this.subsystem = opts.subsystem;\n this.severity = opts.severity ?? 'error';\n this.recoverable = opts.recoverable ?? false;\n this.context = opts.context;\n }\n\n /**\n * Render a one-line user-facing description.\n * Subclasses should override for domain-specific formatting.\n */\n describe(): string {\n const ctx = this.context ? ` ${formatContext(this.context)}` : '';\n return `${this.code}: ${this.message}${ctx}`;\n }\n}\n\nfunction formatContext(ctx: Record<string, unknown>): string {\n const parts = Object.entries(ctx)\n .filter(([, v]) => v !== undefined)\n .slice(0, 3)\n .map(([k, v]) => `${k}=${String(v)}`);\n return parts.length > 0 ? `[${parts.join(' ')}]` : '';\n}\n\n// ── Specific error classes ───────────────────────────────────────────\n\n/**\n * Tool execution errors — thrown by ToolExecutor and individual tools.\n */\nexport class ToolError extends WrongStackError {\n readonly toolName: string;\n\n constructor(opts: {\n message: string;\n code: Extract<\n ErrorCode,\n | 'TOOL_NOT_FOUND'\n | 'TOOL_PERMISSION_DENIED'\n | 'TOOL_EXECUTION_FAILED'\n | 'TOOL_TIMEOUT'\n | 'TOOL_INPUT_INVALID'\n >;\n toolName: string;\n recoverable?: boolean | undefined;\n context?: Record<string, unknown> | undefined;\n cause?: unknown | undefined;\n }) {\n super({\n message: opts.message,\n code: opts.code,\n subsystem: 'tool',\n recoverable: opts.recoverable,\n context: { tool: opts.toolName, ...opts.context },\n cause: opts.cause,\n });\n this.name = 'ToolError';\n this.toolName = opts.toolName;\n }\n}\n\n/**\n * Config loading / validation errors.\n */\nexport class ConfigError extends WrongStackError {\n constructor(opts: {\n message: string;\n code: Extract<\n ErrorCode,\n 'CONFIG_INVALID' | 'CONFIG_NOT_FOUND' | 'CONFIG_PARSE_FAILED' | 'CONFIG_MIGRATION_NEEDED'\n >;\n context?: Record<string, unknown> | undefined;\n cause?: unknown | undefined;\n }) {\n super({\n message: opts.message,\n code: opts.code,\n subsystem: 'config',\n severity: 'fatal',\n recoverable: false,\n context: opts.context,\n cause: opts.cause,\n });\n this.name = 'ConfigError';\n }\n}\n\n/**\n * Plugin loading / lifecycle errors.\n */\nexport class PluginError extends WrongStackError {\n readonly pluginName: string;\n\n constructor(opts: {\n message: string;\n code: Extract<\n ErrorCode,\n 'PLUGIN_LOAD_FAILED' | 'PLUGIN_API_MISMATCH' | 'PLUGIN_MISSING_DEPENDENCY'\n >;\n pluginName: string;\n context?: Record<string, unknown> | undefined;\n cause?: unknown | undefined;\n }) {\n super({\n message: opts.message,\n code: opts.code,\n subsystem: 'plugin',\n severity: 'error',\n recoverable: opts.code === ERROR_CODES.PLUGIN_MISSING_DEPENDENCY,\n context: { plugin: opts.pluginName, ...opts.context },\n cause: opts.cause,\n });\n this.name = 'PluginError';\n this.pluginName = opts.pluginName;\n }\n}\n\n/**\n * Agent runtime errors — thrown by Agent.run when a non-WrongStackError\n * escapes the inner loop, so callers always see a structured error.\n */\nexport class AgentError extends WrongStackError {\n constructor(opts: {\n message: string;\n code: Extract<\n ErrorCode,\n 'AGENT_ITERATION_LIMIT' | 'AGENT_CONTEXT_OVERFLOW' | 'AGENT_ABORTED' | 'AGENT_RUN_FAILED'\n >;\n recoverable?: boolean | undefined;\n context?: Record<string, unknown> | undefined;\n cause?: unknown | undefined;\n }) {\n super({\n message: opts.message,\n code: opts.code,\n subsystem: 'agent',\n severity: opts.code === ERROR_CODES.AGENT_ABORTED ? 'warning' : 'error',\n recoverable: opts.recoverable ?? opts.code === ERROR_CODES.AGENT_ITERATION_LIMIT,\n context: opts.context,\n cause: opts.cause,\n });\n this.name = 'AgentError';\n }\n}\n\n/**\n * Wrap an arbitrary thrown value into a `WrongStackError` so the caller\n * always gets a structured error. Pass-throughs WrongStackError instances\n * unchanged; raw `Error`s and primitives get an `AGENT_RUN_FAILED` wrapper\n * with the original preserved as `cause`.\n */\nexport function toWrongStackError(\n err: unknown,\n code: Extract<ErrorCode, 'AGENT_RUN_FAILED' | 'AGENT_ABORTED' | 'UNKNOWN'> = ERROR_CODES.AGENT_RUN_FAILED,\n): WrongStackError {\n if (err instanceof WrongStackError) return err;\n const message = toErrorMessage(err);\n return new AgentError({\n message,\n code: code === 'UNKNOWN' ? ERROR_CODES.AGENT_RUN_FAILED : code,\n cause: err,\n });\n}\n\n/**\n * Session storage errors.\n */\nexport class SessionError extends WrongStackError {\n readonly sessionId?: string | undefined;\n\n constructor(opts: {\n message: string;\n code: Extract<ErrorCode, 'SESSION_NOT_FOUND' | 'SESSION_CORRUPTED' | 'SESSION_WRITE_FAILED'>;\n sessionId?: string | undefined;\n context?: Record<string, unknown> | undefined;\n cause?: unknown | undefined;\n }) {\n super({\n message: opts.message,\n code: opts.code,\n subsystem: 'session',\n severity: opts.code === ERROR_CODES.SESSION_WRITE_FAILED ? 'error' : 'warning',\n recoverable: opts.code !== ERROR_CODES.SESSION_CORRUPTED,\n context: { sessionId: opts.sessionId, ...opts.context },\n cause: opts.cause,\n });\n this.name = 'SessionError';\n this.sessionId = opts.sessionId;\n }\n}\n\n/**\n * SDD (Spec-Driven Development) errors — spec validation, parsing, and\n * state machine violations in the AISpecBuilder, TaskFlow, and TaskTracker.\n */\nexport class SddError extends WrongStackError {\n constructor(opts: {\n message: string;\n code: Extract<\n ErrorCode,\n 'SDD_VALIDATION_FAILED' | 'SDD_PARSE_FAILED' | 'SDD_INVALID_STATE' | 'SDD_NOT_READY'\n >;\n context?: Record<string, unknown> | undefined;\n cause?: unknown | undefined;\n }) {\n super({\n message: opts.message,\n code: opts.code,\n subsystem: 'sdd',\n severity: opts.code === ERROR_CODES.SDD_PARSE_FAILED ? 'warning' : 'error',\n recoverable: opts.code === ERROR_CODES.SDD_NOT_READY,\n context: opts.context,\n cause: opts.cause,\n });\n this.name = 'SddError';\n }\n}\n\n/**\n * File system operation errors.\n */\nexport class FsError extends WrongStackError {\n readonly path?: string | undefined;\n\n constructor(opts: {\n message: string;\n code: Extract<\n ErrorCode,\n 'FS_READ_FAILED' | 'FS_WRITE_FAILED' | 'FS_MKDIR_FAILED' | 'FS_DELETE_FAILED' | 'FS_ATOMIC_WRITE_FAILED'\n >;\n path?: string | undefined;\n context?: Record<string, unknown> | undefined;\n cause?: unknown | undefined;\n }) {\n super({\n message: opts.message,\n code: opts.code,\n subsystem: 'fs',\n severity: 'error',\n recoverable: opts.code !== ERROR_CODES.FS_READ_FAILED,\n context: { path: opts.path, ...opts.context },\n cause: opts.cause,\n });\n this.name = 'FsError';\n this.path = opts.path;\n }\n}\n\n/**\n * HTTP fetch error — thrown when a network request returns a non-OK status.\n * Carries the response status so {@link classifyToolError} can branch on it\n * (429 → transient, 404 → not_found, 401 → permission) without duck-typing\n * the error via `'response' in err`.\n *\n * P3 #18 (before-release.md): the previous `'response' in err` check caught\n * any Error with a `response` property, including custom errors, proxy\n * objects, or mocked errors in tests. `instanceof FetchError` is reliable.\n *\n * Tools and providers that make HTTP requests and need the executor to\n * classify their failures should throw `new FetchError({ status, message })`\n * instead of a bare `Error` with an ad-hoc `response` field.\n */\nexport class FetchError extends WrongStackError {\n readonly status: number;\n\n constructor(opts: {\n message: string;\n status: number;\n context?: Record<string, unknown> | undefined;\n cause?: unknown | undefined;\n }) {\n super({\n message: opts.message,\n code: ERROR_CODES.VALIDATION_ERROR,\n subsystem: 'general',\n severity: 'error',\n recoverable: opts.status === 429 || opts.status >= 500,\n context: { status: opts.status, ...opts.context },\n cause: opts.cause,\n });\n this.name = 'FetchError';\n this.status = opts.status;\n }\n}\n\n/**\n * Tool input validation error — thrown when a tool's input fails a validation\n * check that the JSON Schema cannot express (e.g. `old_string === new_string`\n * in edit, or a cross-field invariant). Use this instead of a bare\n * `throw new Error('...validation...')` so {@link classifyToolError} can\n * match on `instanceof` rather than a locale-dependent message substring.\n *\n * P2 #6 (before-release.md): the previous `err.message.includes('validation')`\n * check misclassified any error whose message happened to contain \"validation\"\n * (e.g. a third-party \"input validation timeout\") as a VALIDATION error.\n *\n * Named `ToolValidationError` (not `ValidationError`) to avoid colliding with\n * the existing `ValidationError` interface exported by json-schema-validate.ts\n * (a validation-result shape, not an Error subclass).\n */\nexport class ToolValidationError extends WrongStackError {\n constructor(opts: {\n message: string;\n /** Field path or tool name that failed validation, for diagnostics. */\n field?: string | undefined;\n context?: Record<string, unknown> | undefined;\n cause?: unknown | undefined;\n }) {\n super({\n message: opts.message,\n code: ERROR_CODES.VALIDATION_ERROR,\n subsystem: 'general',\n severity: 'error',\n recoverable: false,\n context: { field: opts.field, ...opts.context },\n cause: opts.cause,\n });\n this.name = 'ToolValidationError';\n }\n}\n\n/**\n * Response / payload parse error — thrown when an upstream HTTP response,\n * file, or data structure is well-formed at the transport layer (HTTP 200,\n * valid JSON) but is missing required fields or has an unexpected shape.\n *\n * Distinct from `ConfigError(CONFIG_PARSE_FAILED)` (which is specifically\n * for config-file parsing) and `FetchError` (which covers HTTP non-OK\n * responses). `ParseError` fills the gap: the request succeeded but the\n * response body couldn't be interpreted.\n *\n * Common sites: OAuth token responses missing `access_token`, device-code\n * responses missing `device_code`, registry responses with unexpected\n * schemas.\n */\nexport class ParseError extends WrongStackError {\n readonly source?: string | undefined;\n\n constructor(opts: {\n message: string;\n /**\n * What was being parsed — e.g. `'oauth-token-response'`,\n * `'device-code-response'`. Lets consumers distinguish parse failures\n * from different upstream APIs without parsing the message.\n */\n source?: string | undefined;\n context?: Record<string, unknown> | undefined;\n cause?: unknown | undefined;\n }) {\n super({\n message: opts.message,\n code: ERROR_CODES.PARSE_FAILED,\n subsystem: 'general',\n severity: 'error',\n recoverable: false,\n context: { source: opts.source, ...opts.context },\n cause: opts.cause,\n });\n this.name = 'ParseError';\n this.source = opts.source;\n }\n}\n\n// ── Type guards ──────────────────────────────────────────────────────\n\nexport function isWrongStackError(err: unknown): err is WrongStackError {\n return err instanceof WrongStackError;\n}\n\nexport function isToolError(err: unknown): err is ToolError {\n return err instanceof ToolError;\n}\n\nexport function isConfigError(err: unknown): err is ConfigError {\n return err instanceof ConfigError;\n}\n\nexport function isPluginError(err: unknown): err is PluginError {\n return err instanceof PluginError;\n}\n\nexport function isSessionError(err: unknown): err is SessionError {\n return err instanceof SessionError;\n}\n\nexport function isAgentError(err: unknown): err is AgentError {\n return err instanceof AgentError;\n}\n\nexport function isFsError(err: unknown): err is FsError {\n return err instanceof FsError;\n}\n\nexport function isToolValidationError(err: unknown): err is ToolValidationError {\n return err instanceof ToolValidationError;\n}\n\nexport function isFetchError(err: unknown): err is FetchError {\n return err instanceof FetchError;\n}\n\nexport function isParseError(err: unknown): err is ParseError {\n return err instanceof ParseError;\n}\n\nexport function isSddError(err: unknown): err is SddError {\n return err instanceof SddError;\n}\n","import { expectDefined } from '../utils/expect-defined.js';\nimport * as fs from 'node:fs/promises';\nimport * as os from 'node:os';\nimport * as path from 'node:path';\nimport { createGunzip } from 'node:zlib';\nimport { Readable } from 'node:stream';\nimport { pipeline } from 'node:stream/promises';\nimport { WrongStackError, ERROR_CODES } from '../types/errors.js';\nexport interface ParsedRef {\n owner: string;\n repo: string;\n ref: string;\n}\n\n/**\n * Parse a skill reference string.\n * Formats: `user/repo` (default ref: main), `user/repo@ref`\n */\nexport function parseSkillRef(input: string): ParsedRef {\n const trimmed = input.trim().replace(/^https?:\\/\\/github\\.com\\//, '').replace(/\\.git$/, '');\n const atIdx = trimmed.indexOf('@');\n let refPath: string;\n let ref: string;\n if (atIdx > 0) {\n refPath = trimmed.slice(0, atIdx);\n ref = trimmed.slice(atIdx + 1);\n } else {\n refPath = trimmed;\n ref = 'main';\n }\n const parts = refPath.split('/').filter(Boolean);\n if (parts.length < 2) {\n throw new WrongStackError({\n message: `Invalid skill reference \"${input}\". Expected format: user/repo or user/repo@ref`,\n code: ERROR_CODES.UNKNOWN,\n subsystem: 'general',\n context: { input },\n });\n }\n return { owner: expectDefined(parts[0]), repo: expectDefined(parts[1]), ref };\n}\n\nexport interface DownloadResult {\n /** Temp directory containing the extracted repo. Caller must clean up. */\n tempDir: string;\n}\n\nconst MAX_TARBALL_SIZE = 50 * 1024 * 1024; // 50MB\n\n/**\n * Download and extract a GitHub repository tarball.\n * Uses the public GitHub API — no auth token required for public repos.\n * Returns the path to a temp directory with the extracted contents.\n */\nexport async function downloadGitHubTarball(parsed: ParsedRef): Promise<DownloadResult> {\n const url = `https://api.github.com/repos/${parsed.owner}/${parsed.repo}/tarball/${parsed.ref}`;\n\n const response = await fetch(url, {\n signal: AbortSignal.timeout(30_000),\n headers: {\n Accept: 'application/vnd.github+json',\n 'User-Agent': 'wrongstack-skill-installer',\n },\n redirect: 'follow',\n });\n\n if (!response.ok) {\n if (response.status === 404) {\n throw new WrongStackError({\n message:\n `Repository not found: ${parsed.owner}/${parsed.repo}` +\n (parsed.ref !== 'main' ? ` (ref: ${parsed.ref})` : ''),\n code: ERROR_CODES.UNKNOWN,\n subsystem: 'general',\n context: { owner: parsed.owner, repo: parsed.repo, ref: parsed.ref, status: 404 },\n });\n }\n if (response.status === 403) {\n throw new WrongStackError({\n message: `Access denied: ${parsed.owner}/${parsed.repo}. The repository may be private or rate-limited.`,\n code: ERROR_CODES.UNKNOWN,\n subsystem: 'general',\n context: { owner: parsed.owner, repo: parsed.repo, status: 403 },\n });\n }\n throw new WrongStackError({\n message: `GitHub API error (${response.status}): ${response.statusText}`,\n code: ERROR_CODES.UNKNOWN,\n subsystem: 'general',\n context: { owner: parsed.owner, repo: parsed.repo, status: response.status },\n });\n }\n\n const contentLength = response.headers.get('content-length');\n if (contentLength && Number.parseInt(contentLength, 10) > MAX_TARBALL_SIZE) {\n throw new WrongStackError({\n message:\n `Tarball too large (${(Number.parseInt(contentLength, 10) / 1024 / 1024).toFixed(1)}MB). ` +\n `Max: ${MAX_TARBALL_SIZE / 1024 / 1024}MB`,\n code: ERROR_CODES.UNKNOWN,\n subsystem: 'general',\n context: {\n owner: parsed.owner,\n repo: parsed.repo,\n contentLengthBytes: Number.parseInt(contentLength, 10),\n maxBytes: MAX_TARBALL_SIZE,\n },\n });\n }\n\n const tempDir = await fs.mkdtemp(path.join(os.tmpdir(), 'wskill-'));\n\n try {\n if (!response.body) {\n throw new WrongStackError({\n message: 'Empty response body from GitHub API',\n code: ERROR_CODES.UNKNOWN,\n subsystem: 'general',\n context: { owner: parsed.owner, repo: parsed.repo },\n });\n }\n\n // Gunzip the response body, then extract the tar stream\n const nodeStream = Readable.fromWeb(response.body as import('node:stream/web').ReadableStream);\n const gunzip = createGunzip();\n\n // Collect the uncompressed tar data into a buffer, then extract\n const chunks: Buffer[] = [];\n await pipeline(nodeStream, gunzip, async (source) => {\n for await (const chunk of source) {\n chunks.push(Buffer.from(chunk));\n }\n });\n const tarBuf = Buffer.concat(chunks);\n\n // Extract tar archive (POSIX ustar format)\n await extractTar(tarBuf, tempDir);\n\n return { tempDir };\n } catch (err) {\n await fs.rm(tempDir, { recursive: true, force: true }).catch(() => {});\n throw err;\n }\n}\n\n/**\n * Minimal POSIX tar extractor. Handles the ustar format produced by GitHub.\n * Only extracts regular files and directories — symlinks and special entries\n * are skipped for security.\n */\nasync function extractTar(buf: Buffer, destDir: string): Promise<void> {\n let offset = 0;\n\n while (offset + 512 <= buf.length) {\n // Check for end-of-archive (two consecutive zero blocks)\n const header = buf.subarray(offset, offset + 512);\n if (header.every((b) => b === 0)) break;\n\n // Parse ustar header\n const name = readTarString(buf, offset, 100); // name field\n const prefix = readTarString(buf, offset + 345, 155); // ustar prefix\n const size = Number.parseInt(readTarString(buf, offset + 124, 12).trim(), 8) || 0;\n const typeflag = buf[offset + 156] ?? 0;\n\n // Full path: prefix/name (ustar) or just name\n const fullPath = prefix ? `${prefix}/${name}` : name;\n // Strip the top-level directory (GitHub tarballs have owner-repo-sha/)\n const relPath = stripTopDir(fullPath);\n\n if (relPath && relPath !== '.' && relPath !== '..') {\n const destPath = path.join(destDir, relPath);\n\n // Zip-slip guard: reject any entry whose resolved path escapes destDir\n // (e.g. a crafted entry name like `x/../../../etc/cron.d/evil`). GitHub\n // tarballs are built from git trees and so cannot carry `..` components,\n // but this extractor is generic — never trust archive entry names.\n const resolvedDest = path.resolve(destPath);\n const resolvedRoot = path.resolve(destDir);\n if (resolvedDest !== resolvedRoot && !resolvedDest.startsWith(resolvedRoot + path.sep)) {\n // Skip the entry entirely; advance past its data below.\n offset += 512 + Math.ceil(size / 512) * 512;\n continue;\n }\n\n // typeflag: '0' or '\\0' = regular file, '5' = directory\n if (typeflag === 0x35 || typeflag === 0) {\n // Directory\n if (relPath.endsWith('/') || typeflag === 0x35) {\n await fs.mkdir(destPath, { recursive: true });\n }\n }\n\n if ((typeflag === 0x30 || typeflag === 0 || typeflag === 0x00) && size > 0) {\n // Regular file\n const dir = path.dirname(destPath);\n await fs.mkdir(dir, { recursive: true });\n const dataStart = offset + 512;\n const dataEnd = dataStart + size;\n if (dataEnd > buf.length) break; // truncated archive\n await fs.writeFile(destPath, buf.subarray(dataStart, dataEnd));\n }\n }\n\n // Advance: 512-byte header + data padded to 512-byte boundary\n offset += 512 + Math.ceil(size / 512) * 512;\n }\n}\n\nfunction readTarString(buf: Buffer, start: number, maxLen: number): string {\n let end = start;\n while (end < start + maxLen && end < buf.length && buf[end] !== 0) {\n end++;\n }\n return buf.subarray(start, end).toString('utf8');\n}\n\n/**\n * Strip the top-level directory from a tar path.\n * GitHub tarballs have a single root dir like `owner-repo-sha/`.\n */\nfunction stripTopDir(p: string): string {\n const idx = p.indexOf('/');\n if (idx === -1) return ''; // top-level file, skip\n return p.slice(idx + 1);\n}\n","import * as fs from 'node:fs/promises';\r\nimport * as path from 'node:path';\r\nimport { atomicWrite } from '../utils/atomic-write.js';\r\n\r\nexport interface InstalledSkillEntry {\r\n name: string;\r\n /** Source identifier, e.g. \"github:user/repo\" */\r\n source: string;\r\n /** Git ref that was installed (branch, tag, commit) */\r\n ref: string;\r\n /** Installation scope */\r\n scope: 'project' | 'user';\r\n /** Project hash — only set when scope=project */\r\n projectHash?: string | undefined;\r\n /** ISO 8601 timestamp */\r\n installedAt: string;\r\n /** List of files that were installed (relative to skill dir) */\r\n files: string[];\r\n}\r\n\r\nexport interface ManifestData {\r\n skills: InstalledSkillEntry[];\r\n}\r\n\r\nexport class SkillManifestStore {\r\n private readonly manifestPath: string;\r\n private cache?: ManifestData | undefined;\r\n\r\n constructor(manifestPath: string) {\r\n this.manifestPath = manifestPath;\r\n }\r\n\r\n async read(): Promise<ManifestData> {\r\n if (this.cache) return this.cache;\r\n try {\r\n const raw = await fs.readFile(this.manifestPath, 'utf8');\r\n const data = JSON.parse(raw) as ManifestData;\r\n if (!Array.isArray(data.skills)) {\r\n this.cache = { skills: [] };\r\n } else {\r\n this.cache = data;\r\n }\r\n } catch {\r\n this.cache = { skills: [] };\r\n }\r\n return this.cache;\r\n }\r\n\r\n async write(data: ManifestData): Promise<void> {\r\n const dir = path.dirname(this.manifestPath);\r\n await fs.mkdir(dir, { recursive: true });\r\n await atomicWrite(this.manifestPath, JSON.stringify(data, null, 2) + '\\n');\r\n this.cache = data;\r\n }\r\n\r\n async addEntry(entry: InstalledSkillEntry): Promise<void> {\r\n const data = await this.read();\r\n // Remove existing entry with the same name + scope\r\n data.skills = data.skills.filter(\r\n (s) => !(s.name === entry.name && s.scope === entry.scope),\r\n );\r\n data.skills.push(entry);\r\n await this.write(data);\r\n }\r\n\r\n async removeEntry(name: string, scope: 'project' | 'user'): Promise<boolean> {\r\n const data = await this.read();\r\n const before = data.skills.length;\r\n data.skills = data.skills.filter(\r\n (s) => !(s.name === name && s.scope === scope),\r\n );\r\n if (data.skills.length === before) return false;\r\n await this.write(data);\r\n return true;\r\n }\r\n\r\n async findByName(name: string): Promise<InstalledSkillEntry[]> {\r\n const data = await this.read();\r\n return data.skills.filter((s) => s.name === name);\r\n }\r\n\r\n async findBySource(source: string): Promise<InstalledSkillEntry[]> {\r\n const data = await this.read();\r\n return data.skills.filter((s) => s.source === source);\r\n }\r\n\r\n async listAll(): Promise<InstalledSkillEntry[]> {\r\n const data = await this.read();\r\n return data.skills;\r\n }\r\n\r\n /** Invalidate the in-memory cache (e.g. after external file changes). */\r\n invalidateCache(): void {\r\n this.cache = undefined;\r\n }\r\n}\r\n","import { expectDefined } from '../utils/expect-defined.js';\nimport { toErrorMessage } from '../utils/error.js';\nimport * as fs from 'node:fs/promises';\nimport * as path from 'node:path';\nimport type { SkillLoader } from '../types/skill.js';\nimport { downloadGitHubTarball, parseSkillRef } from './github-fetcher.js';\nimport { type InstalledSkillEntry, SkillManifestStore } from './manifest-store.js';\nimport { FsError, WrongStackError, ERROR_CODES } from '../types/errors.js';\nexport interface SkillInstallerOptions {\n /** Path to the manifest file (~/.wrongstack/installed-skills.json) */\n manifestPath: string;\n /** Path to project-level skills dir (<project>/.wrongstack/skills/) */\n projectSkillsDir: string;\n /** Path to user-global skills dir (~/.wrongstack/skills/) */\n globalSkillsDir: string;\n /** Current project hash (for manifest tracking) */\n projectHash: string;\n /** Skill loader — cache will be invalidated after mutations */\n skillLoader?: SkillLoader | undefined;\n /** Logger for status messages */\n log?: (((msg: string) => void)) | undefined;\n}\n\nexport interface InstallResult {\n name: string;\n path: string;\n scope: 'project' | 'user';\n source: string;\n ref: string;\n skillCount: number;\n}\n\nexport interface UpdateResult {\n updated: Array<{ name: string; oldRef: string; newRef: string }>;\n unchanged: string[];\n errors: Array<{ name: string; error: string }>;\n}\n\nconst MAX_SKILL_FILE_SIZE = 100 * 1024; // 100KB\n\nexport class SkillInstaller {\n private readonly opts: SkillInstallerOptions;\n private readonly manifest: SkillManifestStore;\n\n constructor(opts: SkillInstallerOptions) {\n this.opts = opts;\n this.manifest = new SkillManifestStore(opts.manifestPath);\n }\n\n /**\n * Install skills from a GitHub repository.\n * Supports both single-skill repos (SKILL.md at root) and multi-skill repos (skills/ subdirectory).\n */\n async install(refInput: string, opts?: { global?: boolean | undefined }): Promise<InstallResult[]> {\n const parsed = parseSkillRef(refInput);\n const scope: 'project' | 'user' = opts?.global ? 'user' : 'project';\n const targetDir = scope === 'project' ? this.opts.projectSkillsDir : this.opts.globalSkillsDir;\n const source = `github:${parsed.owner}/${parsed.repo}`;\n\n this.opts.log?.(`Downloading ${parsed.owner}/${parsed.repo}@${parsed.ref}...`);\n\n const { tempDir } = await downloadGitHubTarball(parsed);\n\n try {\n // Detect skill structure\n const skills = await this.detectSkills(tempDir);\n\n if (skills.length === 0) {\n throw new WrongStackError({\n message: 'No skills found in repository. Expected SKILL.md at root or skills/ subdirectory.',\n code: ERROR_CODES.VALIDATION_ERROR,\n subsystem: 'general',\n context: { owner: parsed.owner, repo: parsed.repo, ref: parsed.ref },\n });\n }\n\n const results: InstallResult[] = [];\n\n for (const skill of skills) {\n // Check for overwrite\n const existing = await this.manifest.findByName(skill.name);\n const existingInScope = existing.find((e) => e.scope === scope);\n if (existingInScope) {\n this.opts.log?.(`Overwriting existing skill \"${skill.name}\" (${scope})...`);\n await this.removeSkillFiles(skill.name, scope);\n }\n\n // Copy skill files\n const destDir = path.join(targetDir, skill.name);\n await fs.mkdir(destDir, { recursive: true });\n const copiedFiles: string[] = [];\n\n for (const file of skill.files) {\n const srcPath = path.join(skill.baseDir, file);\n const destPath = path.join(destDir, file);\n\n // Path traversal check\n const resolved = path.resolve(destPath);\n if (!resolved.startsWith(path.resolve(destDir))) {\n throw new FsError({\n message: `Path traversal detected in skill file: ${file}`,\n code: ERROR_CODES.FS_DELETE_FAILED,\n path: destPath,\n context: { reason: 'path_traversal', skillName: skill.name },\n });\n }\n\n // Size check\n const stat = await fs.stat(srcPath);\n if (stat.size > MAX_SKILL_FILE_SIZE) {\n throw new FsError({\n message:\n `Skill file \"${file}\" is too large (${(stat.size / 1024).toFixed(1)}KB). ` +\n `Max: ${MAX_SKILL_FILE_SIZE / 1024}KB`,\n code: ERROR_CODES.FS_WRITE_FAILED,\n path: srcPath,\n context: { skillName: skill.name, fileSize: stat.size, maxSize: MAX_SKILL_FILE_SIZE },\n });\n }\n\n await fs.mkdir(path.dirname(destPath), { recursive: true });\n await fs.copyFile(srcPath, destPath);\n copiedFiles.push(file);\n }\n\n // Write manifest entry\n const entry: InstalledSkillEntry = {\n name: skill.name,\n source,\n ref: parsed.ref,\n scope,\n projectHash: scope === 'project' ? this.opts.projectHash : undefined,\n installedAt: new Date().toISOString(),\n files: copiedFiles,\n };\n await this.manifest.addEntry(entry);\n\n results.push({\n name: skill.name,\n path: destDir,\n scope,\n source,\n ref: parsed.ref,\n skillCount: 1,\n });\n }\n\n this.invalidateLoaderCache();\n return results;\n } finally {\n // Clean up temp directory\n await fs.rm(tempDir, { recursive: true, force: true }).catch(() => {});\n }\n }\n\n /**\n * Update installed skills.\n * - No args: update all\n * - Name: update that specific skill\n * - Name + newRef: update to a different ref\n */\n async update(\n nameOrRef?: string | undefined,\n _opts?: { global?: boolean | undefined } | undefined,\n ): Promise<UpdateResult> {\n const result: UpdateResult = { updated: [], unchanged: [], errors: [] };\n const allEntries = await this.manifest.listAll();\n\n let targets: InstalledSkillEntry[];\n if (nameOrRef) {\n // Check if it's a name or a ref (user/repo@ref)\n const byName = allEntries.filter((e) => e.name === nameOrRef);\n if (byName.length > 0) {\n targets = byName;\n } else {\n // Treat as a new ref — find matching source\n try {\n const parsed = parseSkillRef(nameOrRef);\n const source = `github:${parsed.owner}/${parsed.repo}`;\n targets = allEntries.filter((e) => e.source === source);\n if (targets.length === 0) {\n result.errors.push({\n name: nameOrRef,\n error: `No installed skills found matching \"${nameOrRef}\"`,\n });\n return result;\n }\n } catch {\n result.errors.push({\n name: nameOrRef,\n error: `Invalid reference: ${nameOrRef}`,\n });\n return result;\n }\n }\n } else {\n targets = allEntries;\n }\n\n // Group by source to avoid downloading the same repo multiple times\n const bySource = new Map<string, InstalledSkillEntry[]>();\n for (const entry of targets) {\n const key = `${entry.source}@${entry.ref}`;\n if (!bySource.has(key)) bySource.set(key, []);\n bySource.get(key)?.push(entry);\n }\n\n for (const [, entries] of bySource) {\n const first = expectDefined(entries[0]);\n const scope = first.scope;\n const isGlobal = scope === 'user';\n\n try {\n // Parse the original source to get the ref\n const sourceRepo = first.source.replace('github:', '');\n let refToInstall = first.ref;\n\n // If nameOrRef looks like a new ref, use it\n if (nameOrRef && !allEntries.find((e) => e.name === nameOrRef)) {\n try {\n const parsed = parseSkillRef(nameOrRef);\n refToInstall = parsed.ref;\n } catch {\n // keep original ref\n }\n }\n\n this.opts.log?.(`Updating ${first.source}@${refToInstall}...`);\n const results = await this.install(`${sourceRepo}@${refToInstall}`, { global: isGlobal });\n\n for (const r of results) {\n result.updated.push({\n name: r.name,\n oldRef: first.ref,\n newRef: refToInstall,\n });\n }\n } catch (err) {\n const msg = toErrorMessage(err);\n for (const entry of entries) {\n result.errors.push({ name: entry.name, error: msg });\n }\n }\n }\n\n return result;\n }\n\n /**\n * Uninstall a skill by name.\n */\n async uninstall(name: string, opts?: { global?: boolean | undefined }): Promise<void> {\n const scope: 'project' | 'user' = opts?.global ? 'user' : 'project';\n const entries = await this.manifest.findByName(name);\n const entry = entries.find((e) => e.scope === scope);\n\n if (!entry) {\n throw new WrongStackError({\n message: `Skill \"${name}\" is not installed${scope === 'user' ? ' (global)' : ''}.`,\n code: ERROR_CODES.VALIDATION_ERROR,\n subsystem: 'general',\n context: { skillName: name, scope },\n });\n }\n\n // Remove files\n await this.removeSkillFiles(name, scope);\n\n // Remove from manifest\n await this.manifest.removeEntry(name, scope);\n this.invalidateLoaderCache();\n }\n\n /**\n * List all installed skills from the manifest.\n */\n async listInstalled(): Promise<InstalledSkillEntry[]> {\n return this.manifest.listAll();\n }\n\n // ── Private helpers ──────────────────────────────────────────────\n\n /**\n * Detect skills in an extracted repository.\n * Returns an array of detected skills with their files.\n */\n private async detectSkills(\n baseDir: string,\n ): Promise<Array<{ name: string; baseDir: string; files: string[] }>> {\n const results: Array<{ name: string; baseDir: string; files: string[] }> = [];\n\n // Check for SKILL.md at root (single-skill repo)\n const rootSkillMd = path.join(baseDir, 'SKILL.md');\n try {\n await fs.access(rootSkillMd);\n const content = await fs.readFile(rootSkillMd, 'utf8');\n const meta = parseFrontmatter(content);\n if (meta.name && meta.description) {\n results.push({\n name: meta.name,\n baseDir,\n files: ['SKILL.md'],\n });\n return results; // Single-skill repo — don't look for skills/\n }\n } catch {\n // No root SKILL.md\n }\n\n // Check for skills/ subdirectory (multi-skill repo)\n const skillsDir = path.join(baseDir, 'skills');\n try {\n const entries = await fs.readdir(skillsDir, { withFileTypes: true });\n for (const entry of entries) {\n if (!entry.isDirectory()) continue;\n const skillFile = path.join(skillsDir, entry.name, 'SKILL.md');\n try {\n const content = await fs.readFile(skillFile, 'utf8');\n const meta = parseFrontmatter(content);\n if (meta.name && meta.description) {\n // Collect all files in the skill directory\n const skillDir = path.join(skillsDir, entry.name);\n const files = await collectFiles(skillDir, skillDir);\n results.push({\n name: meta.name,\n baseDir: skillDir,\n files,\n });\n }\n } catch {\n // Skip malformed skills\n }\n }\n } catch {\n // No skills/ directory\n }\n\n return results;\n }\n\n /**\n * Remove all files for an installed skill.\n */\n private async removeSkillFiles(name: string, scope: 'project' | 'user'): Promise<void> {\n const targetDir =\n scope === 'project' ? this.opts.projectSkillsDir : this.opts.globalSkillsDir;\n const skillDir = path.join(targetDir, name);\n await fs.rm(skillDir, { recursive: true, force: true });\n }\n\n /**\n * Invalidate the skill loader's cache so newly installed skills appear.\n */\n private invalidateLoaderCache(): void {\n // The SkillLoader interface has a cache internally.\n // We access it via the 'any' cast to call invalidateCache if available.\n const loader = this.opts.skillLoader as never as { invalidateCache?: () => void };\n if (loader && typeof loader.invalidateCache === 'function') {\n loader.invalidateCache();\n }\n }\n}\n\n// ── Utilities ──────────────────────────────────────────────────────\n\ninterface Frontmatter {\n name?: string | undefined;\n description?: string | undefined;\n}\n\nfunction parseFrontmatter(raw: string): Frontmatter {\n if (!raw.startsWith('---')) return {};\n const end = raw.indexOf('\\n---', 4);\n if (end === -1) return {};\n const block = raw.slice(4, end);\n const out: Frontmatter = {};\n let key: keyof Frontmatter | null = null;\n let value: string[] = [];\n const flush = () => {\n if (key) {\n out[key] = value.join('\\n').trim();\n }\n key = null;\n value = [];\n };\n for (const line of block.split('\\n')) {\n const m = /^([a-zA-Z_]+):\\s*(\\|?)\\s*(.*)$/.exec(line);\n if (m) {\n flush();\n key = (m[1] ?? '') as keyof Frontmatter;\n const pipe = m[2];\n const rest = m[3] ?? '';\n if (pipe === '|') {\n value = [];\n } else if (rest) {\n value = [rest];\n } else {\n value = [];\n }\n } else if (key) {\n value.push(line.replace(/^\\s+/, ''));\n }\n }\n flush();\n return out;\n}\n\n/**\n * Recursively collect all files in a directory (relative paths).\n */\nasync function collectFiles(dir: string, baseDir: string): Promise<string[]> {\n const results: string[] = [];\n const entries = await fs.readdir(dir, { withFileTypes: true });\n for (const entry of entries) {\n const fullPath = path.join(dir, entry.name);\n const relPath = path.relative(baseDir, fullPath);\n if (entry.isDirectory()) {\n // Skip hidden dirs and node_modules\n if (entry.name.startsWith('.') || entry.name === 'node_modules') continue;\n results.push(...(await collectFiles(fullPath, baseDir)));\n } else {\n results.push(relPath);\n }\n }\n return results;\n}\n"]}
@@ -1,41 +1,42 @@
1
- import { M as MemoryBackend, j as FileMemoryBackendOptions } from '../todos-checkpoint-oDS9IBNS.js';
2
- export { A as AbandonedSession, a as AttachmentStoreOptions, C as ConfigLoaderOptions, b as ConfigMigration, c as ConfigMigrationError, d as ConfigSource, D as DEFAULT_CONFIG_MIGRATIONS, e as DefaultAttachmentStore, f as DefaultConfigLoader, g as DefaultConfigStore, h as DefaultMemoryStore, i as DefaultSessionStore, F as FileMemoryBackend, k as MemoryStoreOptions, l as MigrationContext, m as MigrationResult, P as PersistedQueueItem, n as PlanFile, o as PlanItem, p as PlanTemplate, Q as QueueStore, R as RecoveryLock, q as RecoveryLockOptions, S as SessionAnalyzer, r as SessionStoreOptions, T as TodosCheckpointFile, s as addPlanItem, t as attachPlanCheckpoint, u as attachTodosCheckpoint, v as clearPlan, w as deriveTodosFromPlanItem, x as emptyPlan, y as formatPlan, z as formatPlanTemplates, B as getPlanTemplate, E as listPlanTemplates, G as loadPlan, H as loadTodosCheckpoint, I as mutatePlan, J as parseEntries, K as removePlanItem, L as runConfigMigrations, N as savePlan, O as saveTodosCheckpoint, U as setPlanItemStatus } from '../todos-checkpoint-oDS9IBNS.js';
3
- import { M as MemoryScope, b as MemoryEntry, c as MemoryStore, E as EventBus } from '../brain-DCkB5_e7.js';
4
- import { i as AgentExtension, k as AfterRunHook } from '../index-CtlizLTK.js';
5
- import { P as Provider, c as Request, d as Response, S as SessionEvent } from '../context-BrLe8pJy.js';
6
- export { D as DefaultSessionReader, f as DefaultSessionReaderOptions, S as SessionReader } from '../session-reader-BjLH4V9n.js';
1
+ import { M as MemoryBackend, j as FileMemoryBackendOptions } from '../todos-checkpoint-Ddd2CGr0.js';
2
+ export { A as AbandonedSession, a as AttachmentStoreOptions, C as ConfigLoaderOptions, b as ConfigMigration, c as ConfigMigrationError, d as ConfigSource, D as DEFAULT_CONFIG_MIGRATIONS, e as DefaultAttachmentStore, f as DefaultConfigLoader, g as DefaultConfigStore, h as DefaultMemoryStore, i as DefaultSessionStore, F as FileMemoryBackend, k as MemoryStoreOptions, l as MigrationContext, m as MigrationResult, P as PersistedQueueItem, n as PlanFile, o as PlanItem, p as PlanTemplate, Q as QueueStore, R as RecoveryLock, q as RecoveryLockOptions, S as SessionAnalyzer, r as SessionStoreOptions, T as TodosCheckpointFile, s as addPlanItem, t as attachPlanCheckpoint, u as attachTodosCheckpoint, v as clearPlan, w as deriveTodosFromPlanItem, x as emptyPlan, y as formatPlan, z as formatPlanTemplates, B as getPlanTemplate, E as listPlanTemplates, G as loadPlan, H as loadTodosCheckpoint, I as mutatePlan, J as parseEntries, K as removePlanItem, L as runConfigMigrations, N as savePlan, O as saveTodosCheckpoint, U as setPlanItemStatus } from '../todos-checkpoint-Ddd2CGr0.js';
3
+ import { M as MemoryScope, b as MemoryEntry, c as MemoryStore, E as EventBus } from '../brain-BLOyN5ZP.js';
4
+ import { i as AgentExtension, k as AfterRunHook } from '../index-Cn0NOshr.js';
5
+ import { P as Provider, c as Request, d as Response, S as SessionEvent } from '../context-CLnUMW5g.js';
6
+ import { P as ProviderConfig, Z as SyncCategory, k as SyncConfig } from '../config-Bf5mj-ad.js';
7
+ import { S as SecretVault } from '../secret-vault-BAKpgFw_.js';
8
+ export { D as DefaultSessionReader, f as DefaultSessionReaderOptions, S as SessionReader } from '../session-reader-C8aiChUu.js';
7
9
  import { S as SessionRewinder, C as CheckpointInfo, a as RewindResultExtended } from '../session-rewinder-C9HnMkhP.js';
8
10
  import { T as TaskItem } from '../task-format-vGOIftmK.js';
9
11
  export { a as DirectorStateCheckpoint, D as DirectorStateSnapshot, b as DirectorSubagentState, c as DirectorTaskState, l as loadDirectorState } from '../director-state-BfeCUbmk.js';
10
- export { G as GoalFile, J as JournalEntry, M as MAX_JOURNAL_ENTRIES, a as MAX_PROGRESS_HISTORY, P as ProgressSnapshot, b as appendJournal, e as emptyGoal, f as formatGoal, g as goalFilePath, l as loadGoal, p as parseProgressFromText, r as recordProgress, s as saveGoal, c as setProgress, d as summarizeUsage } from '../goal-store-DUwdbdoY.js';
12
+ export { G as GoalFile, J as JournalEntry, M as MAX_JOURNAL_ENTRIES, a as MAX_PROGRESS_HISTORY, P as ProgressSnapshot, b as appendJournal, e as emptyGoal, f as formatGoal, g as goalFilePath, l as loadGoal, p as parseProgressFromText, r as recordProgress, s as saveGoal, c as setProgress, d as summarizeUsage } from '../goal-store-DGb6b5Ed.js';
11
13
  import { a as PromptEntry } from '../prompt-DLd35n4Q.js';
12
14
  import { W as WstackPaths } from '../wstack-paths-_NrRovdr.js';
13
- import { Z as SyncCategory, k as SyncConfig } from '../config-BzFRKkg7.js';
14
- export { A as AuditLevel, C as CORE_RECONSTRUCT_EVENTS, a as STANDARD_AUDIT_EVENTS, S as SessionEventBridge, b as SessionEventBridgeOptions, c as SessionSamplingOptions, T as ToolProgressSamplingOptions, d as createSessionEventBridge, r as resolveAuditLevel, e as resolveSessionLoggingConfig } from '../session-event-bridge-m7y--I-H.js';
15
- import '../secret-vault-BAKpgFw_.js';
16
- import '../permission-CJR1qfOi.js';
15
+ export { A as AuditLevel, C as CORE_RECONSTRUCT_EVENTS, a as STANDARD_AUDIT_EVENTS, S as SessionEventBridge, b as SessionEventBridgeOptions, c as SessionSamplingOptions, T as ToolProgressSamplingOptions, d as createSessionEventBridge, r as resolveAuditLevel, e as resolveSessionLoggingConfig } from '../session-event-bridge-CMvIO59_.js';
16
+ import '../permission-BCbQDR2s.js';
17
17
  import '../logger-B63L5bTg.js';
18
- import '../pipeline-BYR-Vdau.js';
19
- import '../mailbox-types-_7gaY0Rl.js';
18
+ import '../pipeline-Dl6XbfE7.js';
19
+ import '../mailbox-types-DTl7bRH3.js';
20
20
  import '../observability-D-HZN_mF.js';
21
21
  import '../task-graph-u1q9Jkyk.js';
22
22
 
23
- /** Sanitize a model name for use in filenames: alphanumeric + dash + underscore. */
23
+ /**
24
+ * @deprecated Legacy helper kept for callers that still need filename-safe
25
+ * labels. New session ids are opaque and do not include model/provider names.
26
+ */
24
27
  declare function sanitizeModel(model: string): string;
25
28
  /**
26
29
  * Generate a session ID in the format:
27
- * `YYYY-MM-DD/HH-MM-SSZ[_model]_xxxx.jsonl`
30
+ * `YYYY-MM-DD/sess_<ULID>`
28
31
  *
29
32
  * Examples:
30
- * `2026-06-06/12-30-45Z_claude-sonnet_a1b2.jsonl`
31
- * `2026-06-06/14-22-10Z_a1b2.jsonl` (no model)
33
+ * `2026-06-06/sess_01JX2S9V7T5M6N7P8Q9R0STXVW`
32
34
  *
33
35
  * The date prefix becomes a subdirectory so sessions group naturally by day.
34
- * The model name (when available) lets you see at a glance which provider was
35
- * used, without opening the file. The 4-byte random suffix prevents collisions
36
- * within the same second.
36
+ * The leaf is an opaque sortable id; provider/model names belong in metadata,
37
+ * not file paths. Older IDs that contain model/provider text remain readable.
37
38
  */
38
- declare function generateSessionId(startedAt: string, model?: string): string;
39
+ declare function generateSessionId(startedAt: string, _model?: string): string;
39
40
 
40
41
  interface GraphNode {
41
42
  id: string;
@@ -197,6 +198,35 @@ declare class SessionMemoryConsolidator implements AgentExtension {
197
198
  afterRun: AfterRunHook;
198
199
  }
199
200
 
201
+ /**
202
+ * A decrypted snapshot of the credential-bearing slice of `config.json`.
203
+ * `providers` is always present (empty object when the file has none);
204
+ * `apiKey`/`baseUrl` are the top-level fallbacks a provider inherits when its
205
+ * saved entry omits them.
206
+ */
207
+ interface ProviderConfigSnapshot {
208
+ providers: Record<string, ProviderConfig>;
209
+ apiKey?: string;
210
+ baseUrl?: string;
211
+ }
212
+ interface WatchProviderConfigOptions {
213
+ /** Surface non-fatal read/parse/decrypt issues. */
214
+ warn?: (msg: string) => void;
215
+ /** Coalesce bursts of fs events (default 200ms). */
216
+ debounceMs?: number;
217
+ }
218
+ /**
219
+ * Watch `configPath` for credential changes and invoke `onChange` with a
220
+ * freshly-decrypted snapshot whenever the `providers`/`apiKey`/`baseUrl` slice
221
+ * actually changes. Returns a `close()` that stops watching.
222
+ *
223
+ * The initial on-disk state is read once to seed the no-op guard but does NOT
224
+ * fire `onChange` — only subsequent changes do.
225
+ */
226
+ declare function watchProviderConfig(configPath: string, vault: SecretVault, onChange: (snapshot: ProviderConfigSnapshot) => void, opts?: WatchProviderConfigOptions): {
227
+ close: () => void;
228
+ };
229
+
200
230
  /**
201
231
  * L2-B: AnnotationsStore — sidecar storage for collaboration annotations
202
232
  * (Phase 2 of idea #13 from IDEAS.md).
@@ -212,8 +242,9 @@ declare class SessionMemoryConsolidator implements AgentExtension {
212
242
  * on the history.
213
243
  *
214
244
  * So we keep annotations in a sibling file: one JSON document per
215
- * session, at `<sessionDir>/<sessionId>.annotations.json`. The
216
- * shape is a simple versioned array, written atomically.
245
+ * session, resolved with `sessionScopedPath(sessionDir, sessionId,
246
+ * '.annotations.json')`. The shape is a simple versioned array,
247
+ * written atomically.
217
248
  *
218
249
  * Concurrency model:
219
250
  *
@@ -247,7 +278,7 @@ interface Annotation {
247
278
  resolvedBy?: string | undefined;
248
279
  }
249
280
  interface AnnotationsStoreOptions {
250
- /** Directory where `<sessionId>.annotations.json` files live. */
281
+ /** Root sessions directory used with `sessionScopedPath(..., '.annotations.json')`. */
251
282
  dir: string;
252
283
  events?: EventBus;
253
284
  traceId?: string;
@@ -323,7 +354,8 @@ declare class AnnotationsStore {
323
354
  * the event log would inflate every read for replay-irrelevant
324
355
  * paths.
325
356
  *
326
- * File layout: `<dir>/<sessionId>.replay.jsonl`, one entry per line.
357
+ * File layout: `sessionScopedPath(dir, sessionId, '.replay.jsonl')`,
358
+ * one entry per line.
327
359
  * Each entry: `{ hash, ts, request, response }`. The `hash` is
328
360
  * computed via `hashRequest` so lookups are O(1) by hash.
329
361
  *
@@ -338,7 +370,7 @@ interface ReplayEntry {
338
370
  response: Response;
339
371
  }
340
372
  interface ReplayLogStoreOptions {
341
- /** Directory where `<sessionId>.replay.jsonl` files live. */
373
+ /** Root sessions directory used with `sessionScopedPath(..., '.replay.jsonl')`. */
342
374
  dir: string;
343
375
  /**
344
376
  * Cap on the number of entries per session. When a `record` would
@@ -524,8 +556,8 @@ declare class SessionRecovery {
524
556
  * - The agent itself misbehaving; this is post-hoc audit, not
525
557
  * real-time enforcement. Use `PermissionPolicy` for that.
526
558
  *
527
- * File layout: `<dir>/<sessionId>.audit.jsonl`, one entry per
528
- * line. The chain starts with a `genesis` entry whose
559
+ * File layout: `sessionScopedPath(dir, sessionId, '.audit.jsonl')`,
560
+ * one entry per line. The chain starts with a `genesis` entry whose
529
561
  * `prevHash` is all zeros.
530
562
  */
531
563
  interface AuditEntry {
@@ -554,7 +586,7 @@ type VerifyResult = {
554
586
  reason: string;
555
587
  };
556
588
  interface ToolAuditLogOptions {
557
- /** Directory where `<sessionId>.audit.jsonl` files live. */
589
+ /** Root sessions directory used with `sessionScopedPath(..., '.audit.jsonl')`. */
558
590
  dir: string;
559
591
  /**
560
592
  * Flush the file system cache to disk every N writes per session.
@@ -775,6 +807,8 @@ declare function hasSessionRegistry(): boolean;
775
807
  interface AgentStatusTrackerOptions {
776
808
  events: EventBus;
777
809
  registry: SessionRegistry;
810
+ /** Session id whose live agent state is mirrored by this tracker. */
811
+ sessionId?: string | (() => string | undefined) | undefined;
778
812
  /** Leader agent name shown in the registry. Default: "leader". */
779
813
  leaderName?: string | undefined;
780
814
  /**
@@ -787,6 +821,7 @@ interface AgentStatusTrackerOptions {
787
821
  declare class AgentStatusTracker {
788
822
  private readonly events;
789
823
  private readonly registry;
824
+ private readonly sessionId;
790
825
  private readonly leaderName;
791
826
  private agents;
792
827
  private lastAgents;
@@ -823,6 +858,8 @@ declare class AgentStatusTracker {
823
858
  */
824
859
  private sweep;
825
860
  private flush;
861
+ private currentSessionId;
862
+ private acceptsSession;
826
863
  private markLeaderStarted;
827
864
  private captureLeaderContext;
828
865
  }
@@ -868,6 +905,7 @@ declare class DefaultSessionRewinder implements SessionRewinder {
868
905
  private readonly sessionsDir;
869
906
  private readonly projectRoot;
870
907
  constructor(sessionsDir: string, projectRoot: string);
908
+ private sessionFile;
871
909
  listCheckpoints(sessionId: string): Promise<CheckpointInfo[]>;
872
910
  rewindToCheckpoint(sessionId: string, checkpointIndex: number): Promise<RewindResultExtended>;
873
911
  rewindLastN(sessionId: string, n: number): Promise<RewindResultExtended>;
@@ -1010,4 +1048,4 @@ declare class CloudSync {
1010
1048
  private walkDir;
1011
1049
  }
1012
1050
 
1013
- export { ALL_SYNC_CATEGORIES, type AgentEntry, type AgentLiveStatus, AgentStatusTracker, type AgentStatusTrackerOptions, type Annotation, AnnotationsStore, type AnnotationsStoreOptions, type AuditEntry, CloudSync, type ConsolidationOp, DefaultPromptStore, DefaultSessionRewinder, FileMemoryBackendOptions, FleetNotifier, type FleetNotifierOptions, GraphMemoryBackend, type GraphMemoryBackendOptions, MemoryBackend, type MemoryConsolidatorOptions, PromptEntry, type PromptStore, type PromptUsage, PromptUsageStore, type RecoveryPlan, type ReplayEntry, ReplayLogStore, type ReplayLogStoreOptions, type SessionLiveStatus, SessionMemoryConsolidator, SessionRecovery, SessionRegistry, type SessionRegistryEntry, type SessionRewinderOptions, type StaleSession, SyncCategory, SyncConfig, type SyncResult, type TaskFile, ToolAuditLog, type ToolAuditLogOptions, type VerifyResult, emptyTaskFile, generateSessionId, getSessionRegistry, hasSessionRegistry, loadTasks, migratePromptEntry, mutateTasks, promptChecksum, sanitizeModel, saveTasks };
1051
+ export { ALL_SYNC_CATEGORIES, type AgentEntry, type AgentLiveStatus, AgentStatusTracker, type AgentStatusTrackerOptions, type Annotation, AnnotationsStore, type AnnotationsStoreOptions, type AuditEntry, CloudSync, type ConsolidationOp, DefaultPromptStore, DefaultSessionRewinder, FileMemoryBackendOptions, FleetNotifier, type FleetNotifierOptions, GraphMemoryBackend, type GraphMemoryBackendOptions, MemoryBackend, type MemoryConsolidatorOptions, PromptEntry, type PromptStore, type PromptUsage, PromptUsageStore, type ProviderConfigSnapshot, type RecoveryPlan, type ReplayEntry, ReplayLogStore, type ReplayLogStoreOptions, type SessionLiveStatus, SessionMemoryConsolidator, SessionRecovery, SessionRegistry, type SessionRegistryEntry, type SessionRewinderOptions, type StaleSession, SyncCategory, SyncConfig, type SyncResult, type TaskFile, ToolAuditLog, type ToolAuditLogOptions, type VerifyResult, type WatchProviderConfigOptions, emptyTaskFile, generateSessionId, getSessionRegistry, hasSessionRegistry, loadTasks, migratePromptEntry, mutateTasks, promptChecksum, sanitizeModel, saveTasks, watchProviderConfig };