agent-insights 0.0.1 → 0.0.7

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.
package/dist/cli.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/cli.ts","../src/commands/cursor.ts","../src/config/config.ts","../src/config/paths.ts","../src/privacy/sanitize.ts","../src/util/identity.ts","../src/privacy/hash.ts","../src/util/log.ts","../src/commands/disable.ts","../src/adapters/claude-code.ts","../src/adapters/cursor.ts","../src/adapters/registry.ts","../src/commands/doctor.ts","../src/transcript/store.ts","../src/commands/hook.ts","../src/commands/init.ts","../src/commands/status.ts"],"sourcesContent":["import { Command } from \"commander\";\nimport { runCursorWrapper } from \"./commands/cursor.js\";\nimport { runDisable } from \"./commands/disable.js\";\nimport { runDoctor } from \"./commands/doctor.js\";\nimport { runHook } from \"./commands/hook.js\";\nimport { runInit } from \"./commands/init.js\";\nimport { runStatus } from \"./commands/status.js\";\nimport { log } from \"./util/log.js\";\n\nconst VERSION = \"0.0.1\";\n\nconst program = new Command();\n\nprogram\n .name(\"agent-insights\")\n .description(\n \"Internal CLI for AI coding agent observability (Claude Code, Cursor).\",\n )\n .version(VERSION);\n\nprogram\n .command(\"init\")\n .description(\"Set up agent-insights in this repo.\")\n .option(\n \"--agents <list>\",\n \"Comma-separated agent ids (claude-code,cursor)\",\n \"claude-code,cursor\",\n )\n .option(\"--transcripts\", \"Opt into transcript sync to Vercel Blob\")\n .option(\"--no-transcripts\", \"Skip transcript sync\")\n .option(\"--analysis\", \"Opt into SessionEnd analysis (Phase 2)\")\n .option(\"--telemetry\", \"Opt into OTLP event telemetry (Phase 3)\")\n .option(\"--no-hooks\", \"Skip installing per-agent hooks\")\n .option(\"--force\", \"Overwrite existing config\")\n .action(async (opts) => {\n await runInit(opts);\n });\n\nprogram\n .command(\"status\")\n .description(\"Show current config, consent, and adapter state.\")\n .option(\"--json\", \"Emit machine-readable JSON\")\n .action(async (opts) => {\n await runStatus(opts);\n });\n\nprogram\n .command(\"doctor\")\n .description(\"Verify config, Vercel link, Blob, and per-agent hook install.\")\n .option(\"--json\", \"Emit machine-readable JSON\")\n .action(async (opts) => {\n await runDoctor(opts);\n });\n\nprogram\n .command(\"disable\")\n .description(\"Disable agent-insights and/or remove installed hooks.\")\n .option(\"--agent <name>\", \"Disable a specific adapter (claude-code|cursor)\")\n .option(\"--purge\", \"Also delete the local config and caches\")\n .action(async (opts) => {\n await runDisable(opts);\n });\n\nprogram\n .command(\"hook <eventName>\")\n .description(\"Internal: invoked by an agent's hook configuration.\")\n .option(\"--agent <name>\", \"Override agent inference (claude-code|cursor)\")\n .action(async (eventName: string, opts: { agent?: string }) => {\n await runHook(eventName, opts);\n });\n\nprogram\n .command(\"cursor [path]\")\n .description(\"Launch Cursor with session start/end emitted (fallback).\")\n .action(async (path: string | undefined) => {\n await runCursorWrapper({ ...(path !== undefined ? { path } : {}) });\n });\n\nprogram.parseAsync(process.argv).catch((err: unknown) => {\n log.fail((err as Error).message ?? String(err));\n process.exit(1);\n});\n","import { spawn } from \"node:child_process\";\nimport { randomUUID } from \"node:crypto\";\nimport { loadConfig } from \"../config/config.js\";\nimport { sanitize } from \"../privacy/sanitize.js\";\nimport type { AgentEvent } from \"../types.js\";\nimport { repoHash, userHash, workspaceHash } from \"../util/identity.js\";\nimport { debug, log } from \"../util/log.js\";\n\nexport type CursorOptions = {\n path?: string;\n};\n\n/**\n * Wrapper for launching Cursor when native hooks aren't installed yet.\n * Emits session.start / session.end without prompts or tool inputs.\n */\nexport async function runCursorWrapper(opts: CursorOptions): Promise<void> {\n const cfg = await loadConfig();\n if (!cfg) {\n log.fail(\"agent-insights is not initialized — run `agent-insights init`\");\n process.exitCode = 1;\n return;\n }\n const path = opts.path ?? process.cwd();\n const sessionId = randomUUID();\n\n const baseEvent = {\n agent: \"cursor\" as const,\n sessionId,\n userHash: userHash(),\n repoHash: repoHash(),\n workspaceHash: workspaceHash(),\n };\n\n emit({\n ...baseEvent,\n type: \"session.start\",\n timestamp: new Date().toISOString(),\n });\n\n const code = await spawnCursor(path);\n\n emit({\n ...baseEvent,\n type: \"session.end\",\n timestamp: new Date().toISOString(),\n outcome: code === 0 ? \"success\" : \"error\",\n });\n\n process.exitCode = code;\n}\n\nfunction spawnCursor(path: string): Promise<number> {\n return new Promise((resolve) => {\n const child = spawn(\"cursor\", [path], {\n stdio: \"inherit\",\n env: process.env,\n });\n child.on(\"error\", (err) => {\n log.fail(`failed to launch cursor: ${err.message}`);\n resolve(127);\n });\n child.on(\"exit\", (code) => resolve(code ?? 0));\n });\n}\n\nfunction emit(event: AgentEvent): void {\n debug(\"cursor wrapper event\", {\n type: event.type,\n sessionId: event.sessionId,\n });\n // Phase 3 (OTLP) will export sanitized events here.\n void sanitize(event);\n}\n","import { mkdir, readFile, writeFile } from \"node:fs/promises\";\nimport { dirname } from \"node:path\";\nimport { configFile, logsDir, sessionCacheDir } from \"./paths.js\";\n\nexport type ExporterConfig = {\n type: \"otlp\";\n endpoint: string;\n headers: Record<string, string>;\n dataset: string;\n};\n\nexport type TranscriptStoreConfig =\n | {\n type: \"blob\";\n tokenEnv: string;\n prefix: string;\n }\n | {\n type: \"none\";\n };\n\nexport type SessionAnalysisConfig = {\n /** Server-side enablement; user opts in via `userConsent.sessionAnalysis`. */\n enabled: boolean;\n /** Analyzer endpoint base URL. CLI POSTs to `${analyzerUrl}/api/sessions`. */\n analyzerUrl: string;\n /** Env var name holding an optional shared secret. Sent as `x-agent-insights-secret`. */\n secretEnv: string;\n /** GitHub repo for pre-filled new-issue URLs (display only; constructed by analyzer). */\n githubIssueRepo: string;\n};\n\nexport type AgentConfig = {\n enabled: boolean;\n};\n\nexport type AgentInsightsConfig = {\n version: 1;\n enabled: boolean;\n vercel: {\n team: string;\n project: string;\n };\n userConsent: {\n eventTelemetry: boolean;\n transcriptSync: boolean;\n sessionAnalysis: boolean;\n };\n exporter: ExporterConfig;\n transcriptStore: TranscriptStoreConfig;\n sessionAnalysis: SessionAnalysisConfig;\n agents: Record<\"claudeCode\" | \"cursor\", AgentConfig>;\n privacy: {\n capturePrompts: false;\n captureMessages: false;\n captureToolInputs: false;\n captureFileContents: false;\n };\n};\n\nexport function defaultConfig(): AgentInsightsConfig {\n return {\n version: 1,\n enabled: true,\n vercel: {\n team: \"vercel-labs\",\n project: \"agent-insights\",\n },\n userConsent: {\n eventTelemetry: false,\n transcriptSync: false,\n sessionAnalysis: false,\n },\n exporter: {\n type: \"otlp\",\n endpoint: process.env.AGENT_INSIGHTS_OTEL_ENDPOINT ?? \"\",\n headers: {},\n dataset: \"agent-insights\",\n },\n transcriptStore: {\n type: \"blob\",\n tokenEnv: \"BLOB_READ_WRITE_TOKEN\",\n prefix: \"sessions/\",\n },\n sessionAnalysis: {\n enabled: false,\n analyzerUrl: process.env.AGENT_INSIGHTS_ANALYZER_URL ?? \"\",\n secretEnv: \"AGENT_INSIGHTS_INGEST_SECRET\",\n githubIssueRepo: \"vercel-labs/agent-insights\",\n },\n agents: {\n claudeCode: { enabled: false },\n cursor: { enabled: false },\n },\n privacy: {\n capturePrompts: false,\n captureMessages: false,\n captureToolInputs: false,\n captureFileContents: false,\n },\n };\n}\n\nexport async function loadConfig(): Promise<AgentInsightsConfig | undefined> {\n try {\n const raw = await readFile(configFile(), \"utf8\");\n const parsed = JSON.parse(raw) as Partial<AgentInsightsConfig>;\n return mergeConfig(defaultConfig(), parsed);\n } catch (err) {\n if ((err as NodeJS.ErrnoException).code === \"ENOENT\") return undefined;\n throw err;\n }\n}\n\nexport async function saveConfig(cfg: AgentInsightsConfig): Promise<void> {\n await ensureDirs();\n await writeFile(configFile(), `${JSON.stringify(cfg, null, 2)}\\n`, \"utf8\");\n}\n\nexport async function ensureDirs(): Promise<void> {\n await mkdir(dirname(configFile()), { recursive: true });\n await mkdir(sessionCacheDir(), { recursive: true });\n await mkdir(logsDir(), { recursive: true });\n}\n\nfunction mergeConfig(\n base: AgentInsightsConfig,\n patch: Partial<AgentInsightsConfig>,\n): AgentInsightsConfig {\n return {\n ...base,\n ...patch,\n vercel: { ...base.vercel, ...(patch.vercel ?? {}) },\n userConsent: { ...base.userConsent, ...(patch.userConsent ?? {}) },\n exporter: { ...base.exporter, ...(patch.exporter ?? {}) },\n transcriptStore: (patch.transcriptStore ??\n base.transcriptStore) as TranscriptStoreConfig,\n sessionAnalysis: {\n ...base.sessionAnalysis,\n ...(patch.sessionAnalysis ?? {}),\n },\n agents: {\n claudeCode: {\n ...base.agents.claudeCode,\n ...(patch.agents?.claudeCode ?? {}),\n },\n cursor: { ...base.agents.cursor, ...(patch.agents?.cursor ?? {}) },\n },\n privacy: base.privacy,\n };\n}\n","import { homedir } from \"node:os\";\nimport { join } from \"node:path\";\n\n/** Root directory for agent-insights state. Honors `AGENT_INSIGHTS_HOME`. */\nexport function configRoot(): string {\n return process.env.AGENT_INSIGHTS_HOME ?? join(homedir(), \".agent-insights\");\n}\n\nexport function configFile(): string {\n return join(configRoot(), \"config.json\");\n}\n\nexport function sessionCacheDir(): string {\n return join(configRoot(), \"session-cache\");\n}\n\nexport function logsDir(): string {\n return join(configRoot(), \"logs\");\n}\n\nexport const CLAUDE_SETTINGS_FILE = \".claude/settings.json\";\nexport const CURSOR_HOOKS_FILE = \".cursor/hooks.json\";\n","/**\n * Keys that must never appear in any payload that leaves the machine.\n *\n * Matching is case-insensitive and snake_case/camelCase agnostic — we strip a\n * key if its normalized form equals one of these values.\n */\nconst FORBIDDEN_KEYS: ReadonlySet<string> = new Set([\n \"prompt\",\n \"prompts\",\n \"message\",\n \"messages\",\n \"content\",\n \"text\",\n \"transcript\",\n \"toolinput\",\n \"input\",\n \"arguments\",\n \"args\",\n \"params\",\n \"file\",\n \"filecontent\",\n \"filecontents\",\n \"diff\",\n \"patch\",\n \"secret\",\n \"secrets\",\n \"token\",\n \"apikey\",\n \"authorization\",\n \"password\",\n \"credentials\",\n]);\n\nconst MAX_STRING_LENGTH = 256;\n\n/** Allowed primitive value types in sanitized metadata. */\nexport type SanitizedPrimitive = string | number | boolean | null;\n\nfunction normalizeKey(key: string): string {\n return key.toLowerCase().replace(/[_\\-\\s]/g, \"\");\n}\n\nfunction truncate(value: string): string {\n if (value.length <= MAX_STRING_LENGTH) return value;\n return `${value.slice(0, MAX_STRING_LENGTH)}…[truncated]`;\n}\n\nfunction sanitizePrimitive(value: unknown): SanitizedPrimitive | undefined {\n if (value === null) return null;\n switch (typeof value) {\n case \"string\":\n return truncate(value);\n case \"number\":\n return Number.isFinite(value) ? value : null;\n case \"boolean\":\n return value;\n default:\n return undefined;\n }\n}\n\n/**\n * Recursively remove forbidden keys from an arbitrary payload.\n *\n * Returned shape:\n * - Objects keep allowed primitive fields (truncated to {@link MAX_STRING_LENGTH}).\n * - Arrays keep allowed primitive elements; nested objects/arrays are recursed.\n * - Non-primitive, non-container values (functions, symbols, Buffers, …) are dropped.\n */\nexport function sanitize(input: unknown): unknown {\n return sanitizeInner(input);\n}\n\nfunction sanitizeInner(value: unknown): unknown {\n if (Array.isArray(value)) {\n const out: unknown[] = [];\n for (const item of value) {\n const sanitizedItem = sanitizeInner(item);\n if (sanitizedItem !== undefined) {\n out.push(sanitizedItem);\n }\n }\n return out;\n }\n\n if (value && typeof value === \"object\") {\n const out: Record<string, unknown> = {};\n for (const [rawKey, rawValue] of Object.entries(value)) {\n if (FORBIDDEN_KEYS.has(normalizeKey(rawKey))) {\n continue;\n }\n if (rawValue && typeof rawValue === \"object\") {\n out[rawKey] = sanitizeInner(rawValue);\n } else {\n const primitive = sanitizePrimitive(rawValue);\n if (primitive !== undefined) {\n out[rawKey] = primitive;\n }\n }\n }\n return out;\n }\n\n return sanitizePrimitive(value);\n}\n\nexport const __testing = { FORBIDDEN_KEYS, normalizeKey };\n","import { execSync } from \"node:child_process\";\nimport { hostname, userInfo } from \"node:os\";\nimport { sha256 } from \"../privacy/hash.js\";\n\n/** Stable hashed identity of the local user. */\nexport function userHash(): string {\n try {\n const user = userInfo().username;\n return sha256(`${user}@${hostname()}`);\n } catch {\n return sha256(\"unknown\");\n }\n}\n\n/**\n * Stable hashed identity of a repository, derived from `git remote get-url\n * origin` when available, else from `cwd`.\n */\nexport function repoHash(cwd: string = process.cwd()): string {\n const remote = tryGit([\"config\", \"--get\", \"remote.origin.url\"], cwd);\n return sha256(remote ?? cwd);\n}\n\n/** Stable hashed identity of the workspace path. */\nexport function workspaceHash(cwd: string = process.cwd()): string {\n return sha256(cwd);\n}\n\nfunction tryGit(args: string[], cwd: string): string | undefined {\n try {\n const out = execSync(`git ${args.join(\" \")}`, {\n cwd,\n stdio: [\"ignore\", \"pipe\", \"ignore\"],\n encoding: \"utf8\",\n timeout: 1500,\n });\n const trimmed = out.trim();\n return trimmed === \"\" ? undefined : trimmed;\n } catch {\n return undefined;\n }\n}\n","import { createHash } from \"node:crypto\";\n\n/**\n * Stable SHA-256 hash of a value, hex-encoded.\n *\n * Used to anonymize usernames, repo remote URLs, workspace paths, and\n * session identifiers before they leave the machine.\n */\nexport function sha256(value: string): string {\n return createHash(\"sha256\").update(value, \"utf8\").digest(\"hex\");\n}\n\n/** Short prefix of `sha256(value)` for human-readable dashboards. */\nexport function shortHash(value: string, length = 12): string {\n return sha256(value).slice(0, length);\n}\n","import kleur from \"kleur\";\n\nexport const log = {\n info: (msg: string): void => {\n process.stdout.write(`${msg}\\n`);\n },\n ok: (msg: string): void => {\n process.stdout.write(`${kleur.green(\"✓\")} ${msg}\\n`);\n },\n warn: (msg: string): void => {\n process.stdout.write(`${kleur.yellow(\"!\")} ${msg}\\n`);\n },\n fail: (msg: string): void => {\n process.stderr.write(`${kleur.red(\"✗\")} ${msg}\\n`);\n },\n hint: (msg: string): void => {\n process.stdout.write(`${kleur.dim(msg)}\\n`);\n },\n json: (obj: unknown): void => {\n process.stdout.write(`${JSON.stringify(obj, null, 2)}\\n`);\n },\n};\n\nexport function debug(msg: string, extra?: Record<string, unknown>): void {\n if (process.env.AGENT_INSIGHTS_DEBUG) {\n const payload = extra ? ` ${JSON.stringify(extra)}` : \"\";\n process.stderr.write(`[agent-insights] ${msg}${payload}\\n`);\n }\n}\n","import { rm } from \"node:fs/promises\";\nimport { adapterList } from \"../adapters/registry.js\";\nimport { loadConfig, saveConfig } from \"../config/config.js\";\nimport { configRoot } from \"../config/paths.js\";\nimport { log } from \"../util/log.js\";\nimport type { AgentId } from \"../adapters/types.js\";\n\nexport type DisableOptions = {\n agent?: string;\n purge?: boolean;\n};\n\nexport async function runDisable(opts: DisableOptions): Promise<void> {\n const repoRoot = process.cwd();\n const targets = opts.agent ? [opts.agent] : adapterList.map((a) => a.id);\n\n for (const id of targets) {\n if (id !== \"claude-code\" && id !== \"cursor\") {\n log.warn(`Unknown agent id \"${id}\"`);\n continue;\n }\n const adapter = adapterList.find((a) => a.id === (id as AgentId));\n if (!adapter) continue;\n const { removed, path } = await adapter.uninstall(repoRoot);\n if (removed) log.ok(`Removed ${adapter.label} hooks from ${path}`);\n else log.hint(`${adapter.label}: no hooks to remove (${path})`);\n }\n\n const cfg = await loadConfig();\n if (cfg) {\n if (!opts.agent || opts.agent === \"claude-code\") {\n cfg.agents.claudeCode.enabled = false;\n }\n if (!opts.agent || opts.agent === \"cursor\") {\n cfg.agents.cursor.enabled = false;\n }\n if (!opts.agent) cfg.enabled = false;\n await saveConfig(cfg);\n }\n\n if (opts.purge) {\n await rm(configRoot(), { recursive: true, force: true });\n log.ok(`Purged ${configRoot()}`);\n }\n}\n","import { mkdir, readFile, writeFile } from \"node:fs/promises\";\nimport { dirname, join } from \"node:path\";\nimport type { AgentEventType } from \"../types.js\";\nimport type { Adapter, HookPayload, MappedEvent } from \"./types.js\";\n\nconst HOOK_COMMAND_NAME = \"agent-insights\";\n\nconst HOOK_MAP: Record<string, AgentEventType> = {\n SessionStart: \"session.start\",\n SessionEnd: \"session.end\",\n UserPromptSubmit: \"user.prompt.submit\",\n PreToolUse: \"tool.start\",\n PostToolUse: \"tool.end\",\n PostToolUseFailure: \"tool.failure\",\n PermissionRequest: \"permission.request\",\n PermissionDenied: \"permission.denied\",\n SubagentStart: \"subagent.start\",\n SubagentStop: \"subagent.end\",\n Stop: \"agent.stop\",\n Notification: \"notification\",\n PreCompact: \"context.compact.start\",\n PostCompact: \"context.compact.end\",\n};\n\nexport const CLAUDE_HOOK_EVENTS: ReadonlyArray<string> = Object.keys(HOOK_MAP);\n\ntype ClaudeHookEntry = {\n type: \"command\";\n command: string;\n};\n\ntype ClaudeSettings = {\n hooks?: Record<string, ClaudeHookEntry[]>;\n [key: string]: unknown;\n};\n\nexport const claudeCodeAdapter: Adapter = {\n id: \"claude-code\",\n agent: \"claude-code\",\n label: \"Claude Code\",\n settingsFile: \".claude/settings.json\",\n\n map(payload: HookPayload): MappedEvent | undefined {\n const type = HOOK_MAP[payload.event];\n if (!type) return undefined;\n\n const data = payload.data ?? {};\n const sessionId =\n asString(data[\"session_id\"]) ?? asString(data[\"sessionId\"]);\n const toolName =\n asString(data[\"tool_name\"]) ??\n asString(data[\"toolName\"]) ??\n asString((data[\"tool\"] as Record<string, unknown> | undefined)?.[\"name\"]);\n const transcriptPath =\n asString(data[\"transcript_path\"]) ?? asString(data[\"transcriptPath\"]);\n\n return {\n type,\n ...(sessionId !== undefined ? { sessionId } : {}),\n ...(toolName !== undefined ? { toolName } : {}),\n ...(transcriptPath !== undefined ? { transcriptPath } : {}),\n };\n },\n\n async install(repoRoot: string) {\n const path = join(repoRoot, \".claude/settings.json\");\n const settings = await readJson<ClaudeSettings>(path);\n const next: ClaudeSettings = { ...(settings ?? {}) };\n const hooks: Record<string, ClaudeHookEntry[]> = { ...(next.hooks ?? {}) };\n\n for (const event of CLAUDE_HOOK_EVENTS) {\n const existing = (hooks[event] ?? []).filter(\n (h) => !isAgentInsightsCommand(h.command),\n );\n existing.push({\n type: \"command\",\n command: `${HOOK_COMMAND_NAME} hook ${event}`,\n });\n hooks[event] = existing;\n }\n\n next.hooks = hooks;\n await mkdir(dirname(path), { recursive: true });\n await writeFile(path, `${JSON.stringify(next, null, 2)}\\n`, \"utf8\");\n return { written: true, path };\n },\n\n async uninstall(repoRoot: string) {\n const path = join(repoRoot, \".claude/settings.json\");\n const settings = await readJson<ClaudeSettings>(path);\n if (!settings?.hooks) return { removed: false, path };\n const hooks = { ...settings.hooks };\n let touched = false;\n for (const [event, entries] of Object.entries(hooks)) {\n const filtered = entries.filter(\n (h) => !isAgentInsightsCommand(h.command),\n );\n if (filtered.length !== entries.length) touched = true;\n if (filtered.length === 0) {\n delete hooks[event];\n } else {\n hooks[event] = filtered;\n }\n }\n settings.hooks = hooks;\n await writeFile(path, `${JSON.stringify(settings, null, 2)}\\n`, \"utf8\");\n return { removed: touched, path };\n },\n\n async isInstalled(repoRoot: string) {\n const settings = await readJson<ClaudeSettings>(\n join(repoRoot, \".claude/settings.json\"),\n );\n if (!settings?.hooks) return false;\n return Object.values(settings.hooks).some((entries) =>\n entries.some((h) => isAgentInsightsCommand(h.command)),\n );\n },\n};\n\nfunction isAgentInsightsCommand(cmd: string | undefined): boolean {\n if (!cmd) return false;\n return cmd.startsWith(`${HOOK_COMMAND_NAME} hook`);\n}\n\nasync function readJson<T>(path: string): Promise<T | undefined> {\n try {\n const raw = await readFile(path, \"utf8\");\n return JSON.parse(raw) as T;\n } catch (err) {\n if ((err as NodeJS.ErrnoException).code === \"ENOENT\") return undefined;\n throw err;\n }\n}\n\nfunction asString(v: unknown): string | undefined {\n return typeof v === \"string\" && v.length > 0 ? v : undefined;\n}\n","import { mkdir, readFile, writeFile } from \"node:fs/promises\";\nimport { dirname, join } from \"node:path\";\nimport type { AgentEventType } from \"../types.js\";\nimport type { Adapter, HookPayload, MappedEvent } from \"./types.js\";\n\nconst HOOK_COMMAND_NAME = \"agent-insights\";\n\n/**\n * Cursor hook names → common schema. Confirm against current Cursor docs when\n * implementing; align with Claude where event names match.\n */\nconst HOOK_MAP: Record<string, AgentEventType> = {\n SessionStart: \"session.start\",\n SessionEnd: \"session.end\",\n UserPromptSubmit: \"user.prompt.submit\",\n PreToolUse: \"tool.start\",\n PostToolUse: \"tool.end\",\n Stop: \"agent.stop\",\n};\n\nexport const CURSOR_HOOK_EVENTS: ReadonlyArray<string> = Object.keys(HOOK_MAP);\n\ntype CursorHookEntry = {\n command: string;\n};\n\ntype CursorHooksConfig = {\n hooks?: Record<string, CursorHookEntry[]>;\n [key: string]: unknown;\n};\n\nexport const cursorAdapter: Adapter = {\n id: \"cursor\",\n agent: \"cursor\",\n label: \"Cursor\",\n settingsFile: \".cursor/hooks.json\",\n\n map(payload: HookPayload): MappedEvent | undefined {\n const type = HOOK_MAP[payload.event];\n if (!type) return undefined;\n\n const data = payload.data ?? {};\n const sessionId =\n asString(data[\"session_id\"]) ?? asString(data[\"sessionId\"]);\n const toolName = asString(data[\"tool_name\"]) ?? asString(data[\"toolName\"]);\n const transcriptPath =\n asString(data[\"transcript_path\"]) ?? asString(data[\"transcriptPath\"]);\n\n return {\n type,\n ...(sessionId !== undefined ? { sessionId } : {}),\n ...(toolName !== undefined ? { toolName } : {}),\n ...(transcriptPath !== undefined ? { transcriptPath } : {}),\n };\n },\n\n async install(repoRoot: string) {\n const path = join(repoRoot, \".cursor/hooks.json\");\n const settings = await readJson<CursorHooksConfig>(path);\n const next: CursorHooksConfig = { ...(settings ?? {}) };\n const hooks: Record<string, CursorHookEntry[]> = { ...(next.hooks ?? {}) };\n\n for (const event of CURSOR_HOOK_EVENTS) {\n const existing = (hooks[event] ?? []).filter(\n (h) => !isAgentInsightsCommand(h.command),\n );\n existing.push({ command: `${HOOK_COMMAND_NAME} hook ${event}` });\n hooks[event] = existing;\n }\n\n next.hooks = hooks;\n await mkdir(dirname(path), { recursive: true });\n await writeFile(path, `${JSON.stringify(next, null, 2)}\\n`, \"utf8\");\n return { written: true, path };\n },\n\n async uninstall(repoRoot: string) {\n const path = join(repoRoot, \".cursor/hooks.json\");\n const settings = await readJson<CursorHooksConfig>(path);\n if (!settings?.hooks) return { removed: false, path };\n const hooks = { ...settings.hooks };\n let touched = false;\n for (const [event, entries] of Object.entries(hooks)) {\n const filtered = entries.filter(\n (h) => !isAgentInsightsCommand(h.command),\n );\n if (filtered.length !== entries.length) touched = true;\n if (filtered.length === 0) {\n delete hooks[event];\n } else {\n hooks[event] = filtered;\n }\n }\n settings.hooks = hooks;\n await writeFile(path, `${JSON.stringify(settings, null, 2)}\\n`, \"utf8\");\n return { removed: touched, path };\n },\n\n async isInstalled(repoRoot: string) {\n const settings = await readJson<CursorHooksConfig>(\n join(repoRoot, \".cursor/hooks.json\"),\n );\n if (!settings?.hooks) return false;\n return Object.values(settings.hooks).some((entries) =>\n entries.some((h) => isAgentInsightsCommand(h.command)),\n );\n },\n};\n\nfunction isAgentInsightsCommand(cmd: string | undefined): boolean {\n if (!cmd) return false;\n return cmd.startsWith(`${HOOK_COMMAND_NAME} hook`);\n}\n\nasync function readJson<T>(path: string): Promise<T | undefined> {\n try {\n const raw = await readFile(path, \"utf8\");\n return JSON.parse(raw) as T;\n } catch (err) {\n if ((err as NodeJS.ErrnoException).code === \"ENOENT\") return undefined;\n throw err;\n }\n}\n\nfunction asString(v: unknown): string | undefined {\n return typeof v === \"string\" && v.length > 0 ? v : undefined;\n}\n","import { claudeCodeAdapter } from \"./claude-code.js\";\nimport { cursorAdapter } from \"./cursor.js\";\nimport type { Adapter, AgentId } from \"./types.js\";\n\nexport const adapters: Record<AgentId, Adapter> = {\n \"claude-code\": claudeCodeAdapter,\n cursor: cursorAdapter,\n};\n\nexport const adapterList: ReadonlyArray<Adapter> = Object.values(adapters);\n\nexport function getAdapter(id: AgentId): Adapter {\n return adapters[id];\n}\n","import { existsSync } from \"node:fs\";\nimport { join } from \"node:path\";\nimport kleur from \"kleur\";\nimport { adapterList } from \"../adapters/registry.js\";\nimport { loadConfig } from \"../config/config.js\";\nimport { configFile } from \"../config/paths.js\";\nimport { createTranscriptStore } from \"../transcript/store.js\";\nimport { log } from \"../util/log.js\";\n\nexport type DoctorOptions = {\n json?: boolean;\n};\n\ntype Check = {\n name: string;\n ok: boolean;\n detail?: string;\n};\n\nexport async function runDoctor(opts: DoctorOptions): Promise<void> {\n const repoRoot = process.cwd();\n const checks: Check[] = [];\n\n const cfg = await loadConfig();\n checks.push({\n name: \"config found\",\n ok: cfg != null,\n detail: cfg ? configFile() : `missing ${configFile()}`,\n });\n\n if (cfg) {\n checks.push({\n name: \"vercel project linked\",\n ok: existsSync(join(repoRoot, \".vercel/project.json\")),\n detail: \"run `vercel link --scope vercel-labs --project agent-insights`\",\n });\n\n if (cfg.userConsent.transcriptSync) {\n try {\n const store = createTranscriptStore(cfg.transcriptStore);\n const ping = await store.ping();\n if (ping.ok) {\n checks.push({ name: \"transcript store reachable\", ok: true });\n } else {\n checks.push({\n name: \"transcript store reachable\",\n ok: false,\n detail: ping.reason,\n });\n }\n } catch (err) {\n checks.push({\n name: \"transcript store reachable\",\n ok: false,\n detail: (err as Error).message,\n });\n }\n }\n\n for (const adapter of adapterList) {\n const enabled =\n adapter.id === \"claude-code\"\n ? cfg.agents.claudeCode.enabled\n : cfg.agents.cursor.enabled;\n if (!enabled) continue;\n const installed = await adapter.isInstalled(repoRoot);\n checks.push({\n name: `${adapter.label} hooks installed`,\n ok: installed,\n detail: installed\n ? join(repoRoot, adapter.settingsFile)\n : \"run `agent-insights init`\",\n });\n }\n }\n\n if (opts.json) {\n log.json({ ok: checks.every((c) => c.ok), checks });\n } else {\n for (const c of checks) {\n const tag = c.ok ? kleur.green(\"✓\") : kleur.red(\"✗\");\n const detail = c.detail ? kleur.dim(` (${c.detail})`) : \"\";\n log.info(`${tag} ${c.name}${detail}`);\n }\n }\n\n process.exitCode = checks.every((c) => c.ok) ? 0 : 1;\n}\n","import { readFile } from \"node:fs/promises\";\nimport { put } from \"@vercel/blob\";\nimport type { TranscriptStoreConfig } from \"../config/config.js\";\n\nexport type TranscriptUploadInput = {\n /** Absolute path on disk to the transcript file. */\n transcriptPath: string;\n /** SHA256-hashed user identifier. Used in the upload pathname. */\n userHash: string;\n /** Session identifier supplied by the agent. */\n sessionId: string;\n /** Agent name (e.g. \"claude-code\"). */\n agent: string;\n};\n\nexport type TranscriptUploadResult = {\n pathname: string;\n url: string;\n size: number;\n};\n\nexport interface TranscriptStore {\n upload(input: TranscriptUploadInput): Promise<TranscriptUploadResult>;\n /** Lightweight health check used by `doctor`. */\n ping(): Promise<{ ok: true } | { ok: false; reason: string }>;\n}\n\nclass NoopTranscriptStore implements TranscriptStore {\n async upload(): Promise<TranscriptUploadResult> {\n throw new Error(\"transcript store disabled (transcriptStore.type=none)\");\n }\n async ping(): Promise<{ ok: false; reason: string }> {\n return { ok: false, reason: \"transcript store disabled\" };\n }\n}\n\nclass BlobTranscriptStore implements TranscriptStore {\n constructor(\n private readonly token: string,\n private readonly prefix: string,\n ) {}\n\n async upload(input: TranscriptUploadInput): Promise<TranscriptUploadResult> {\n const body = await readFile(input.transcriptPath);\n const requestedPath = buildPathname(\n this.prefix,\n input.userHash,\n input.sessionId,\n );\n // addRandomSuffix avoids guessable URLs: transcripts contain prompts and\n // tool inputs, so URL knowledge is the only access control until we move\n // to private blobs + signed reads.\n const blob = await put(requestedPath, body, {\n access: \"public\",\n token: this.token,\n contentType: \"application/jsonl\",\n addRandomSuffix: true,\n });\n return { pathname: blob.pathname, url: blob.url, size: body.byteLength };\n }\n\n async ping(): Promise<{ ok: true } | { ok: false; reason: string }> {\n if (!this.token) return { ok: false, reason: \"missing token\" };\n return { ok: true };\n }\n}\n\nexport function createTranscriptStore(\n cfg: TranscriptStoreConfig,\n): TranscriptStore {\n if (cfg.type === \"none\") return new NoopTranscriptStore();\n const token = process.env[cfg.tokenEnv];\n if (!token) {\n throw new Error(\n `transcript store: env ${cfg.tokenEnv} is not set (run 'vercel env pull .env.local')`,\n );\n }\n return new BlobTranscriptStore(token, cfg.prefix);\n}\n\nfunction buildPathname(\n prefix: string,\n userHash: string,\n sessionId: string,\n): string {\n const safePrefix = prefix.endsWith(\"/\") ? prefix : `${prefix}/`;\n const safeSession = encodeURIComponent(sessionId);\n return `${safePrefix}${userHash}/${safeSession}/transcript.jsonl`;\n}\n","import { adapterList } from \"../adapters/registry.js\";\nimport { loadConfig } from \"../config/config.js\";\nimport { sanitize } from \"../privacy/sanitize.js\";\nimport { createTranscriptStore } from \"../transcript/store.js\";\nimport type { Adapter, AgentId, HookPayload } from \"../adapters/types.js\";\nimport type { AgentEvent } from \"../types.js\";\nimport { repoHash, userHash, workspaceHash } from \"../util/identity.js\";\nimport { debug, log } from \"../util/log.js\";\n\nexport type HookOptions = {\n agent?: string;\n};\n\n/**\n * Hook entrypoint. Reads JSON payload from stdin, resolves the adapter, maps\n * the native event into the common schema, and runs transcript upload on\n * session.end. Never fails the hook — errors are logged, exit code is 0.\n */\nexport async function runHook(\n eventName: string,\n opts: HookOptions,\n): Promise<void> {\n const cfg = await loadConfig();\n if (!cfg || !cfg.enabled) {\n debug(\"hook skipped — agent-insights not enabled\");\n return;\n }\n\n const stdin = await readStdinSafely();\n const data = parseJsonObject(stdin);\n const payload: HookPayload = { event: eventName, data };\n\n const adapter = pickAdapter(opts.agent, eventName);\n if (!adapter) {\n debug(`no adapter resolved for event ${eventName}`);\n return;\n }\n\n const mapped = adapter.map(payload);\n if (!mapped) {\n debug(`adapter ${adapter.id} did not map event ${eventName}`);\n return;\n }\n\n const event = buildEvent(adapter, mapped);\n debug(\"hook event\", {\n agent: event.agent,\n type: event.type,\n sessionId: event.sessionId,\n });\n\n if (\n mapped.type === \"session.end\" &&\n cfg.userConsent.transcriptSync &&\n mapped.transcriptPath\n ) {\n try {\n const store = createTranscriptStore(cfg.transcriptStore);\n const result = await store.upload({\n transcriptPath: mapped.transcriptPath,\n userHash: event.userHash ?? userHash(),\n sessionId: mapped.sessionId ?? \"unknown\",\n agent: adapter.id,\n });\n debug(\"transcript uploaded\", {\n size: result.size,\n pathname: result.pathname,\n });\n\n if (\n cfg.userConsent.sessionAnalysis &&\n cfg.sessionAnalysis.analyzerUrl\n ) {\n await notifyAnalyzer({\n analyzerUrl: cfg.sessionAnalysis.analyzerUrl,\n secretEnv: cfg.sessionAnalysis.secretEnv,\n payload: {\n sessionId: mapped.sessionId ?? \"unknown\",\n agent: adapter.id,\n userHash: event.userHash ?? userHash(),\n blobUrl: result.url,\n timestamp: event.timestamp,\n },\n });\n }\n } catch (err) {\n log.fail(`transcript upload failed: ${(err as Error).message}`);\n }\n }\n\n // Phase 3 (OTLP) will export `event` here.\n void sanitize(event);\n}\n\ntype AnalyzerPayload = {\n sessionId: string;\n agent: string;\n userHash: string;\n blobUrl: string;\n timestamp: string;\n};\n\nasync function notifyAnalyzer(opts: {\n analyzerUrl: string;\n secretEnv: string;\n payload: AnalyzerPayload;\n}): Promise<void> {\n try {\n const headers: Record<string, string> = {\n \"content-type\": \"application/json\",\n };\n const secret = process.env[opts.secretEnv];\n if (secret) headers[\"x-agent-insights-secret\"] = secret;\n\n const url = new URL(\"/api/sessions\", opts.analyzerUrl).toString();\n const res = await fetch(url, {\n method: \"POST\",\n headers,\n body: JSON.stringify(opts.payload),\n });\n if (!res.ok) {\n debug(`analyzer notify failed: ${res.status} ${res.statusText}`);\n return;\n }\n debug(\"analyzer notified\", { sessionId: opts.payload.sessionId });\n } catch (err) {\n // Never fail the hook because of analyzer issues.\n debug(`analyzer notify error: ${(err as Error).message}`);\n }\n}\n\nfunction pickAdapter(\n override: string | undefined,\n eventName: string,\n): Adapter | undefined {\n if (override) {\n return adapterList.find((a) => a.id === override);\n }\n for (const a of adapterList) {\n if (a.map({ event: eventName })) return a;\n }\n return undefined;\n}\n\nfunction buildEvent(\n adapter: { id: AgentId; agent: AgentEvent[\"agent\"] },\n mapped: NonNullable<ReturnType<Adapter[\"map\"]>>,\n): AgentEvent {\n return {\n type: mapped.type,\n agent: adapter.agent,\n timestamp: new Date().toISOString(),\n ...(mapped.sessionId !== undefined ? { sessionId: mapped.sessionId } : {}),\n ...(mapped.toolName !== undefined ? { toolName: mapped.toolName } : {}),\n ...(mapped.permissionType !== undefined\n ? { permissionType: mapped.permissionType }\n : {}),\n ...(mapped.outcome !== undefined ? { outcome: mapped.outcome } : {}),\n userHash: userHash(),\n repoHash: repoHash(),\n workspaceHash: workspaceHash(),\n };\n}\n\n/**\n * Read stdin if data is available. Returns an empty string when stdin is a\n * TTY or when nothing arrives within `STDIN_TIMEOUT_MS`. Hooks invoked\n * without piped JSON should never hang.\n */\nconst STDIN_TIMEOUT_MS = 200;\n\nasync function readStdinSafely(): Promise<string> {\n if (process.stdin.isTTY) return \"\";\n return new Promise<string>((resolve) => {\n let buf = \"\";\n let settled = false;\n const finish = (): void => {\n if (settled) return;\n settled = true;\n process.stdin.removeAllListeners(\"data\");\n process.stdin.removeAllListeners(\"end\");\n process.stdin.removeAllListeners(\"error\");\n // Release the event loop — `setEncoding` + 'data' listener cause Node\n // to keep reading stdin even after we've stopped listening.\n try {\n process.stdin.pause();\n process.stdin.unref();\n } catch {\n // ignore — stdin may already be destroyed\n }\n resolve(buf);\n };\n const idle = setTimeout(finish, STDIN_TIMEOUT_MS);\n idle.unref();\n process.stdin.setEncoding(\"utf8\");\n process.stdin.on(\"data\", (chunk) => {\n buf += chunk;\n idle.refresh();\n });\n process.stdin.on(\"end\", () => {\n clearTimeout(idle);\n finish();\n });\n process.stdin.on(\"error\", () => {\n clearTimeout(idle);\n finish();\n });\n });\n}\n\nfunction parseJsonObject(raw: string): Record<string, unknown> | undefined {\n if (!raw) return undefined;\n try {\n const parsed = JSON.parse(raw) as unknown;\n if (parsed && typeof parsed === \"object\" && !Array.isArray(parsed)) {\n return parsed as Record<string, unknown>;\n }\n return undefined;\n } catch {\n return undefined;\n }\n}\n","import { existsSync } from \"node:fs\";\nimport { join } from \"node:path\";\nimport kleur from \"kleur\";\nimport { adapterList } from \"../adapters/registry.js\";\nimport {\n defaultConfig,\n loadConfig,\n saveConfig,\n type AgentInsightsConfig,\n} from \"../config/config.js\";\nimport { configFile, configRoot } from \"../config/paths.js\";\nimport { log } from \"../util/log.js\";\n\nexport type InitOptions = {\n agents?: string;\n transcripts?: boolean;\n analysis?: boolean;\n telemetry?: boolean;\n force?: boolean;\n hooks?: boolean;\n};\n\nconst CONSENT_TEXT = `\\\nAgent Insights is opt-in (internal).\n\nOTLP / lifecycle metrics: event names, timestamps, tool names, outcomes,\nhashed user/repo/workspace IDs — never prompts, messages, or tool inputs.\n\nVercel Blob (optional): full session transcripts for internal analysis;\nusername stored as SHA256 only.\n\nSessionEnd analysis (optional): an internal LLM reviews transcript for\ntooling gaps; may post to Slack with a GitHub issue link.\\\n`;\n\nexport async function runInit(opts: InitOptions): Promise<void> {\n log.info(kleur.bold(\"Agent Insights setup\\n\"));\n log.hint(CONSENT_TEXT);\n log.info(\"\");\n\n const existing = await loadConfig();\n if (existing && !opts.force) {\n log.warn(\n `Config already exists at ${configFile()} — re-run with --force to overwrite.`,\n );\n }\n\n const requested = parseAgents(opts.agents);\n const cfg: AgentInsightsConfig = existing ?? defaultConfig();\n\n cfg.userConsent.transcriptSync = opts.transcripts ?? true;\n cfg.userConsent.sessionAnalysis = opts.analysis ?? false;\n cfg.userConsent.eventTelemetry = opts.telemetry ?? false;\n\n cfg.agents.claudeCode.enabled = requested.has(\"claude-code\");\n cfg.agents.cursor.enabled = requested.has(\"cursor\");\n\n await saveConfig(cfg);\n log.ok(`Wrote config to ${configFile()}`);\n\n const installHooks = opts.hooks ?? true;\n const repoRoot = process.cwd();\n if (installHooks) {\n for (const adapter of adapterList) {\n const enabled =\n (adapter.id === \"claude-code\" && cfg.agents.claudeCode.enabled) ||\n (adapter.id === \"cursor\" && cfg.agents.cursor.enabled);\n if (!enabled) continue;\n const { path } = await adapter.install(repoRoot);\n log.ok(`Installed ${adapter.label} hooks → ${rel(path)}`);\n }\n }\n\n printVercelHint(repoRoot);\n log.info(\"\");\n log.info(`Config root: ${configRoot()}`);\n log.hint(\"Run `agent-insights doctor` to verify the setup.\");\n}\n\nfunction parseAgents(value: string | undefined): Set<\"claude-code\" | \"cursor\"> {\n const out = new Set<\"claude-code\" | \"cursor\">();\n const list = (value ?? \"claude-code,cursor\")\n .split(\",\")\n .map((s) => s.trim())\n .filter(Boolean);\n for (const id of list) {\n if (id === \"claude-code\" || id === \"cursor\") out.add(id);\n else log.warn(`Unknown agent id \"${id}\" — skipping.`);\n }\n return out;\n}\n\nfunction printVercelHint(repoRoot: string): void {\n const linked = existsSync(join(repoRoot, \".vercel/project.json\"));\n const envFile = existsSync(join(repoRoot, \".env.local\"));\n\n if (!linked) {\n log.hint(\n \"Vercel project not linked. Run `vercel link --scope vercel-labs --project agent-insights`.\",\n );\n }\n if (!envFile) {\n log.hint(\n \"No .env.local found. Run `vercel env pull .env.local` to fetch BLOB_READ_WRITE_TOKEN.\",\n );\n }\n}\n\nfunction rel(p: string): string {\n const cwd = process.cwd();\n return p.startsWith(cwd) ? p.slice(cwd.length + 1) : p;\n}\n","import { existsSync } from \"node:fs\";\nimport { join } from \"node:path\";\nimport kleur from \"kleur\";\nimport { adapterList } from \"../adapters/registry.js\";\nimport { loadConfig } from \"../config/config.js\";\nimport { configFile } from \"../config/paths.js\";\nimport { log } from \"../util/log.js\";\n\nexport type StatusOptions = {\n json?: boolean;\n};\n\nexport async function runStatus(opts: StatusOptions): Promise<void> {\n const cfg = await loadConfig();\n const repoRoot = process.cwd();\n\n if (!cfg) {\n if (opts.json) log.json({ initialized: false });\n else log.warn(`No config found at ${configFile()} — run \\`agent-insights init\\`.`);\n return;\n }\n\n const adapterStatus = await Promise.all(\n adapterList.map(async (a) => ({\n id: a.id,\n label: a.label,\n enabled:\n a.id === \"claude-code\"\n ? cfg.agents.claudeCode.enabled\n : cfg.agents.cursor.enabled,\n installed: await a.isInstalled(repoRoot),\n })),\n );\n\n if (opts.json) {\n log.json({\n initialized: true,\n configPath: configFile(),\n enabled: cfg.enabled,\n consent: cfg.userConsent,\n vercel: {\n linked: existsSync(join(repoRoot, \".vercel/project.json\")),\n envFile: existsSync(join(repoRoot, \".env.local\")),\n },\n adapters: adapterStatus,\n });\n return;\n }\n\n log.info(kleur.bold(\"Agent Insights status\\n\"));\n log.info(`config ${configFile()}`);\n log.info(`enabled ${cfg.enabled ? \"yes\" : \"no\"}`);\n log.info(\"\");\n log.info(kleur.bold(\"Consent\"));\n log.info(` event telemetry ${yn(cfg.userConsent.eventTelemetry)}`);\n log.info(` transcript sync ${yn(cfg.userConsent.transcriptSync)}`);\n log.info(` session analysis ${yn(cfg.userConsent.sessionAnalysis)}`);\n log.info(\"\");\n log.info(kleur.bold(\"Adapters\"));\n for (const a of adapterStatus) {\n log.info(\n ` ${a.label.padEnd(12)} ${a.enabled ? \"enabled\" : \"disabled\"} hooks ${\n a.installed ? \"installed\" : \"not installed\"\n }`,\n );\n }\n log.info(\"\");\n log.info(kleur.bold(\"Vercel\"));\n log.info(\n ` linked ${yn(existsSync(join(repoRoot, \".vercel/project.json\")))}`,\n );\n log.info(` .env.local ${yn(existsSync(join(repoRoot, \".env.local\")))}`);\n}\n\nfunction yn(v: boolean): string {\n return v ? kleur.green(\"yes\") : kleur.dim(\"no\");\n}\n"],"mappings":"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","names":["mkdir","readFile","writeFile","dirname","join","mkdir","readFile","writeFile","dirname","join","HOOK_COMMAND_NAME","HOOK_MAP","asString","readJson","isAgentInsightsCommand","join","kleur","readFile","userHash","join","kleur","existsSync","join","kleur","kleur","existsSync","join","existsSync","join","kleur","existsSync","join","kleur"]}
1
+ {"version":3,"sources":["../src/config/paths.ts","../src/auth/oauth.ts","../src/auth/store.ts","../src/cli.ts","../src/commands/cursor.ts","../src/config/config.ts","../src/privacy/sanitize.ts","../src/util/identity.ts","../src/privacy/hash.ts","../src/util/log.ts","../src/commands/disable.ts","../src/adapters/claude-code.ts","../src/adapters/cursor.ts","../src/adapters/registry.ts","../src/commands/doctor.ts","../src/transcript/store.ts","../src/commands/hook.ts","../src/commands/init.ts","../src/commands/login.ts","../src/commands/status.ts"],"sourcesContent":["import { homedir } from \"node:os\";\nimport { join } from \"node:path\";\n\n/** Root directory for agent-insights state. Honors `AGENT_INSIGHTS_HOME`. */\nexport function configRoot(): string {\n return process.env.AGENT_INSIGHTS_HOME ?? join(homedir(), \".agent-insights\");\n}\n\nexport function configFile(): string {\n return join(configRoot(), \"config.json\");\n}\n\n/** API key storage (`chmod 600`). Not checked into git. */\nexport function credentialsFile(): string {\n return join(configRoot(), \"credentials.json\");\n}\n\n/** OAuth token storage (`chmod 600`). Not checked into git. */\nexport function authFile(): string {\n return join(configRoot(), \"auth.json\");\n}\n\nexport function sessionCacheDir(): string {\n return join(configRoot(), \"session-cache\");\n}\n\nexport function logsDir(): string {\n return join(configRoot(), \"logs\");\n}\n\nexport const CLAUDE_SETTINGS_FILE = \".claude/settings.json\";\nexport const CURSOR_HOOKS_FILE = \".cursor/hooks.json\";\n\n/** User-level Claude Code settings (hooks apply in every repo). */\nexport function claudeUserSettingsPath(): string {\n return join(homedir(), \".claude\", \"settings.json\");\n}\n\n/** User-level Cursor hooks (hooks apply in every project). */\nexport function cursorUserHooksPath(): string {\n return join(homedir(), \".cursor\", \"hooks.json\");\n}\n","import { createHash, randomBytes } from \"node:crypto\";\nimport { exec } from \"node:child_process\";\nimport { createServer } from \"node:http\";\nimport { platform } from \"node:os\";\n\n// ---------------------------------------------------------------------------\n// OAuth app config — Sign-in-with-Vercel client ID (public, not a secret).\n// Override at runtime via AGENT_INSIGHTS_CLIENT_ID env var.\n// ---------------------------------------------------------------------------\nexport const CLIENT_ID =\n process.env.AGENT_INSIGHTS_CLIENT_ID ?? \"cl_TxqqNFTHcF9kEtM48LBFsYfwjhJ9T8sz\";\n\nconst AUTH_ENDPOINT = \"https://vercel.com/oauth/authorize\";\nconst TOKEN_ENDPOINT = \"https://api.vercel.com/login/oauth/token\";\nconst USERINFO_ENDPOINT = \"https://api.vercel.com/login/oauth/userinfo\";\nconst REVOKE_ENDPOINT = \"https://api.vercel.com/login/oauth/token/revoke\";\n\n// Fixed port — must match the redirect URI registered in the OAuth app.\nconst CALLBACK_PORT = 9797;\nexport const REDIRECT_URI = `http://localhost:${CALLBACK_PORT}/callback`;\n\nconst SCOPES = \"openid email profile offline_access\";\n\nexport type TokenResponse = {\n access_token: string;\n refresh_token?: string;\n expires_in: number;\n token_type: string;\n id_token?: string;\n};\n\nexport type UserInfo = {\n sub: string;\n email: string;\n email_verified: boolean;\n name?: string;\n preferred_username?: string;\n picture?: string;\n};\n\n// ---------------------------------------------------------------------------\n// PKCE helpers\n// ---------------------------------------------------------------------------\n\nfunction generateCodeVerifier(): string {\n return randomBytes(48).toString(\"base64url\");\n}\n\nfunction generateCodeChallenge(verifier: string): string {\n return createHash(\"sha256\").update(verifier).digest(\"base64url\");\n}\n\nfunction generateState(): string {\n return randomBytes(16).toString(\"hex\");\n}\n\n// ---------------------------------------------------------------------------\n// Authorization Code + PKCE flow\n// ---------------------------------------------------------------------------\n\nexport type AuthCodeResult =\n | { status: \"success\"; code: string; state: string }\n | { status: \"error\"; message: string };\n\n/**\n * Starts a localhost callback server, opens the Vercel authorization URL in\n * the browser, and waits for the redirect with the auth code.\n * Resolves (or rejects after timeout) when Vercel redirects back.\n */\nexport async function startPkceFlow(timeoutMs = 120_000): Promise<{\n code: string;\n codeVerifier: string;\n}> {\n const codeVerifier = generateCodeVerifier();\n const codeChallenge = generateCodeChallenge(codeVerifier);\n const state = generateState();\n\n const authUrl = new URL(AUTH_ENDPOINT);\n authUrl.searchParams.set(\"client_id\", CLIENT_ID);\n authUrl.searchParams.set(\"response_type\", \"code\");\n authUrl.searchParams.set(\"redirect_uri\", REDIRECT_URI);\n authUrl.searchParams.set(\"scope\", SCOPES);\n authUrl.searchParams.set(\"code_challenge\", codeChallenge);\n authUrl.searchParams.set(\"code_challenge_method\", \"S256\");\n authUrl.searchParams.set(\"state\", state);\n\n const url = authUrl.toString();\n const codePromise = listenForCallback(state, timeoutMs);\n openBrowser(url);\n // Always print the URL — browser open may fail silently\n process.stderr.write(`\\nIf the browser did not open, visit:\\n ${url}\\n\\n`);\n\n const code = await codePromise;\n return { code, codeVerifier };\n}\n\nfunction listenForCallback(\n expectedState: string,\n timeoutMs: number,\n): Promise<string> {\n return new Promise((resolve, reject) => {\n const server = createServer((req, res) => {\n const url = new URL(req.url ?? \"/\", `http://localhost:${CALLBACK_PORT}`);\n if (url.pathname !== \"/callback\") {\n res.writeHead(404);\n res.end(\"Not found\");\n return;\n }\n\n const code = url.searchParams.get(\"code\");\n const returnedState = url.searchParams.get(\"state\");\n const error = url.searchParams.get(\"error\");\n\n const html = (msg: string) =>\n `<html><body style=\"font-family:sans-serif;text-align:center;padding:80px\">` +\n `<h2>${msg}</h2><p>You can close this tab.</p></body></html>`;\n\n if (error) {\n res.writeHead(400, { \"content-type\": \"text/html\" });\n res.end(html(`Login failed: ${error}`));\n server.close();\n reject(new Error(`OAuth error: ${error}`));\n return;\n }\n\n if (returnedState !== expectedState) {\n res.writeHead(400, { \"content-type\": \"text/html\" });\n res.end(html(\"Invalid state parameter — please try again.\"));\n server.close();\n reject(new Error(\"State mismatch — possible CSRF\"));\n return;\n }\n\n if (!code) {\n res.writeHead(400, { \"content-type\": \"text/html\" });\n res.end(html(\"No authorization code received.\"));\n server.close();\n reject(new Error(\"No code in callback\"));\n return;\n }\n\n res.writeHead(200, { \"content-type\": \"text/html\" });\n res.end(html(\"✓ Logged in! You can close this tab.\"));\n server.close();\n resolve(code);\n });\n\n const timer = setTimeout(() => {\n server.close();\n reject(new Error(\"Login timed out — no response within 2 minutes.\"));\n }, timeoutMs);\n timer.unref();\n\n server.listen(CALLBACK_PORT, \"127.0.0.1\", () => {\n // Server is ready; browser will be opened by caller\n });\n\n server.on(\"error\", (err: NodeJS.ErrnoException) => {\n clearTimeout(timer);\n if (err.code === \"EADDRINUSE\") {\n reject(\n new Error(\n `Port ${CALLBACK_PORT} is already in use. Close any other agent-insights login process and try again.`,\n ),\n );\n } else {\n reject(err);\n }\n });\n });\n}\n\n// ---------------------------------------------------------------------------\n// Token exchange\n// ---------------------------------------------------------------------------\n\nexport async function exchangeCodeForToken(\n code: string,\n codeVerifier: string,\n): Promise<TokenResponse> {\n const res = await fetch(TOKEN_ENDPOINT, {\n method: \"POST\",\n headers: { \"content-type\": \"application/x-www-form-urlencoded\" },\n body: new URLSearchParams({\n grant_type: \"authorization_code\",\n client_id: CLIENT_ID,\n code,\n code_verifier: codeVerifier,\n redirect_uri: REDIRECT_URI,\n }),\n });\n if (!res.ok) {\n const body = await res.text();\n throw new Error(`Token exchange failed (${res.status}): ${body}`);\n }\n return res.json() as Promise<TokenResponse>;\n}\n\nexport async function refreshAccessToken(\n refreshToken: string,\n): Promise<TokenResponse> {\n const res = await fetch(TOKEN_ENDPOINT, {\n method: \"POST\",\n headers: { \"content-type\": \"application/x-www-form-urlencoded\" },\n body: new URLSearchParams({\n grant_type: \"refresh_token\",\n client_id: CLIENT_ID,\n refresh_token: refreshToken,\n }),\n });\n if (!res.ok) {\n const body = await res.text();\n throw new Error(`Token refresh failed (${res.status}): ${body}`);\n }\n return res.json() as Promise<TokenResponse>;\n}\n\nexport async function revokeToken(token: string): Promise<void> {\n await fetch(REVOKE_ENDPOINT, {\n method: \"POST\",\n headers: { \"content-type\": \"application/x-www-form-urlencoded\" },\n body: new URLSearchParams({ token }),\n }).catch(() => {\n // best-effort — don't fail logout if revocation fails\n });\n}\n\nexport async function fetchUserInfo(accessToken: string): Promise<UserInfo> {\n const res = await fetch(USERINFO_ENDPOINT, {\n headers: { authorization: `Bearer ${accessToken}` },\n });\n if (!res.ok) {\n throw new Error(`Failed to fetch user info (${res.status})`);\n }\n return res.json() as Promise<UserInfo>;\n}\n\n/** Open a URL in the system browser. Fails silently — callers always print the URL too. */\nexport function openBrowser(url: string): void {\n const os = platform();\n let cmd: string;\n if (os === \"darwin\") cmd = `open \"${url}\"`;\n else if (os === \"win32\") cmd = `start \"\" \"${url}\"`;\n else cmd = `xdg-open \"${url}\"`;\n\n exec(cmd, () => {\n // Silently ignore errors — caller has already printed the URL\n });\n}\n","import { chmod, readFile, writeFile } from \"node:fs/promises\";\nimport { dirname } from \"node:path\";\nimport { mkdir } from \"node:fs/promises\";\nimport { authFile } from \"../config/paths.js\";\nimport { refreshAccessToken } from \"./oauth.js\";\n\nexport type AuthState = {\n accessToken: string;\n refreshToken?: string;\n /** Unix epoch milliseconds when the access token expires. */\n expiresAt: number;\n};\n\n/** How many ms before expiry we proactively refresh. */\nconst REFRESH_BUFFER_MS = 60_000;\n\nexport async function loadAuth(): Promise<AuthState | undefined> {\n try {\n const raw = await readFile(authFile(), \"utf8\");\n const parsed = JSON.parse(raw) as Partial<AuthState>;\n if (typeof parsed.accessToken === \"string\" && parsed.accessToken.length > 0) {\n return parsed as AuthState;\n }\n return undefined;\n } catch (err) {\n if ((err as NodeJS.ErrnoException).code === \"ENOENT\") return undefined;\n throw err;\n }\n}\n\nexport async function saveAuth(state: AuthState): Promise<void> {\n const path = authFile();\n await mkdir(dirname(path), { recursive: true });\n await writeFile(path, `${JSON.stringify(state, null, 2)}\\n`, {\n encoding: \"utf8\",\n mode: 0o600,\n });\n try {\n await chmod(path, 0o600);\n } catch {\n // Windows doesn't support chmod — ignore\n }\n}\n\nexport async function clearAuth(): Promise<void> {\n const { unlink } = await import(\"node:fs/promises\");\n try {\n await unlink(authFile());\n } catch (err) {\n if ((err as NodeJS.ErrnoException).code !== \"ENOENT\") throw err;\n }\n}\n\n/**\n * Returns a valid access token, refreshing silently if needed.\n * Returns `undefined` if not logged in or refresh fails (caller should prompt login).\n */\nexport async function getValidToken(): Promise<string | undefined> {\n const auth = await loadAuth();\n if (!auth) return undefined;\n\n const needsRefresh = Date.now() >= auth.expiresAt - REFRESH_BUFFER_MS;\n if (!needsRefresh) return auth.accessToken;\n\n if (!auth.refreshToken) {\n // Access token expired, no refresh token — clear and force re-login\n await clearAuth();\n return undefined;\n }\n\n try {\n const tokens = await refreshAccessToken(auth.refreshToken);\n const refreshed: AuthState = {\n accessToken: tokens.access_token,\n refreshToken: tokens.refresh_token ?? auth.refreshToken,\n expiresAt: Date.now() + tokens.expires_in * 1000,\n };\n await saveAuth(refreshed);\n return refreshed.accessToken;\n } catch {\n // Refresh failed (revoked, expired beyond 30 days) — clear and force re-login\n await clearAuth();\n return undefined;\n }\n}\n","import { Command } from \"commander\";\nimport { runCursorWrapper } from \"./commands/cursor.js\";\nimport { runDisable } from \"./commands/disable.js\";\nimport { runDoctor } from \"./commands/doctor.js\";\nimport { runHook } from \"./commands/hook.js\";\nimport { runInit } from \"./commands/init.js\";\nimport { runLogin, runLogout } from \"./commands/login.js\";\nimport { runStatus } from \"./commands/status.js\";\nimport { log } from \"./util/log.js\";\n\ndeclare const __VERSION__: string;\n\nconst program = new Command();\n\nprogram\n .name(\"agent-insights\")\n .description(\n \"Internal CLI for AI coding agent observability (Claude Code, Cursor).\",\n )\n .version(__VERSION__);\n\nprogram\n .command(\"login\")\n .description(\"Authenticate with Vercel (Sign in with Vercel).\")\n .action(async () => {\n await runLogin();\n });\n\nprogram\n .command(\"logout\")\n .description(\"Clear stored credentials.\")\n .action(async () => {\n await runLogout();\n });\n\nprogram\n .command(\"init\")\n .description(\"Install agent hooks globally (run once per machine).\")\n .option(\n \"--agents <list>\",\n \"Comma-separated agent ids (claude-code,cursor)\",\n \"claude-code,cursor\",\n )\n .option(\"--transcripts\", \"Opt into transcript sync to Vercel Blob\")\n .option(\"--no-transcripts\", \"Skip transcript sync\")\n .option(\"--analysis\", \"Opt into SessionEnd analysis (Phase 2)\")\n .option(\"--telemetry\", \"Opt into OTLP event telemetry (Phase 3)\")\n .option(\"--no-hooks\", \"Skip installing per-agent hooks\")\n .option(\"--force\", \"Overwrite existing config\")\n .action(async (opts) => {\n await runInit(opts);\n });\n\nprogram\n .command(\"status\")\n .description(\"Show current config, consent, and adapter state.\")\n .option(\"--json\", \"Emit machine-readable JSON\")\n .action(async (opts) => {\n await runStatus(opts);\n });\n\nprogram\n .command(\"doctor\")\n .description(\"Verify config, Vercel link, Blob, and per-agent hook install.\")\n .option(\"--json\", \"Emit machine-readable JSON\")\n .action(async (opts) => {\n await runDoctor(opts);\n });\n\nprogram\n .command(\"disable\")\n .description(\"Disable agent-insights and/or remove installed hooks.\")\n .option(\"--agent <name>\", \"Disable a specific adapter (claude-code|cursor)\")\n .option(\"--purge\", \"Also delete the local config and caches\")\n .action(async (opts) => {\n await runDisable(opts);\n });\n\nprogram\n .command(\"hook <eventName>\")\n .description(\"Internal: invoked by an agent's hook configuration.\")\n .option(\"--agent <name>\", \"Override agent inference (claude-code|cursor)\")\n .action(async (eventName: string, opts: { agent?: string }) => {\n await runHook(eventName, opts);\n });\n\nprogram\n .command(\"cursor [path]\")\n .description(\"Launch Cursor with session start/end emitted (fallback).\")\n .action(async (path: string | undefined) => {\n await runCursorWrapper({ ...(path !== undefined ? { path } : {}) });\n });\n\nprogram.parseAsync(process.argv).catch((err: unknown) => {\n log.fail((err as Error).message ?? String(err));\n process.exit(1);\n});\n","import { spawn } from \"node:child_process\";\nimport { randomUUID } from \"node:crypto\";\nimport { loadConfig } from \"../config/config.js\";\nimport { sanitize } from \"../privacy/sanitize.js\";\nimport type { AgentEvent } from \"../types.js\";\nimport { repoHash, userHash, workspaceHash } from \"../util/identity.js\";\nimport { debug, log } from \"../util/log.js\";\n\nexport type CursorOptions = {\n path?: string;\n};\n\n/**\n * Wrapper for launching Cursor when native hooks aren't installed yet.\n * Emits session.start / session.end without prompts or tool inputs.\n */\nexport async function runCursorWrapper(opts: CursorOptions): Promise<void> {\n const cfg = await loadConfig();\n if (!cfg) {\n log.fail(\"agent-insights is not initialized — run `agent-insights init`\");\n process.exitCode = 1;\n return;\n }\n const path = opts.path ?? process.cwd();\n const sessionId = randomUUID();\n\n const baseEvent = {\n agent: \"cursor\" as const,\n sessionId,\n userHash: userHash(),\n repoHash: repoHash(),\n workspaceHash: workspaceHash(),\n };\n\n emit({\n ...baseEvent,\n type: \"session.start\",\n timestamp: new Date().toISOString(),\n });\n\n const code = await spawnCursor(path);\n\n emit({\n ...baseEvent,\n type: \"session.end\",\n timestamp: new Date().toISOString(),\n outcome: code === 0 ? \"success\" : \"error\",\n });\n\n process.exitCode = code;\n}\n\nfunction spawnCursor(path: string): Promise<number> {\n return new Promise((resolve) => {\n const child = spawn(\"cursor\", [path], {\n stdio: \"inherit\",\n env: process.env,\n });\n child.on(\"error\", (err) => {\n log.fail(`failed to launch cursor: ${err.message}`);\n resolve(127);\n });\n child.on(\"exit\", (code) => resolve(code ?? 0));\n });\n}\n\nfunction emit(event: AgentEvent): void {\n debug(\"cursor wrapper event\", {\n type: event.type,\n sessionId: event.sessionId,\n });\n // Phase 3 (OTLP) will export sanitized events here.\n void sanitize(event);\n}\n","import { mkdir, readFile, writeFile } from \"node:fs/promises\";\nimport { dirname } from \"node:path\";\nimport { configFile, logsDir, sessionCacheDir } from \"./paths.js\";\n\nexport type ExporterConfig = {\n type: \"otlp\";\n endpoint: string;\n headers: Record<string, string>;\n dataset: string;\n};\n\nexport type HooksScope = \"global\" | \"local\";\n\nexport type TranscriptStoreConfig =\n | {\n /** Upload via analyzer API using AGENT_INSIGHTS_API_KEY (preferred). */\n type: \"analyzer\";\n }\n | {\n /** Direct upload with a Blob RW token (legacy / advanced). */\n type: \"blob\";\n tokenEnv: string;\n prefix: string;\n }\n | {\n type: \"none\";\n };\n\nexport type SessionAnalysisConfig = {\n /** Server-side enablement; user opts in via `userConsent.sessionAnalysis`. */\n enabled: boolean;\n /** Analyzer endpoint base URL. CLI POSTs to `${analyzerUrl}/api/sessions`. */\n analyzerUrl: string;\n /** GitHub repo for pre-filled new-issue URLs (display only; constructed by analyzer). */\n githubIssueRepo: string;\n};\n\nexport type AgentConfig = {\n enabled: boolean;\n};\n\nexport type AgentInsightsConfig = {\n version: 1;\n enabled: boolean;\n /** Where agent hook entries are written. Default `global` (user home). */\n hooksScope: HooksScope;\n vercel: {\n team: string;\n project: string;\n };\n userConsent: {\n eventTelemetry: boolean;\n transcriptSync: boolean;\n sessionAnalysis: boolean;\n };\n exporter: ExporterConfig;\n transcriptStore: TranscriptStoreConfig;\n sessionAnalysis: SessionAnalysisConfig;\n agents: Record<\"claudeCode\" | \"cursor\", AgentConfig>;\n privacy: {\n capturePrompts: false;\n captureMessages: false;\n captureToolInputs: false;\n captureFileContents: false;\n };\n};\n\nexport function defaultConfig(): AgentInsightsConfig {\n return {\n version: 1,\n enabled: true,\n hooksScope: \"global\",\n vercel: {\n team: \"vercel-labs\",\n project: \"agent-insights\",\n },\n userConsent: {\n eventTelemetry: false,\n transcriptSync: false,\n sessionAnalysis: false,\n },\n exporter: {\n type: \"otlp\",\n endpoint: process.env.AGENT_INSIGHTS_OTEL_ENDPOINT ?? \"\",\n headers: {},\n dataset: \"agent-insights\",\n },\n transcriptStore: {\n type: \"blob\",\n tokenEnv: \"BLOB_READ_WRITE_TOKEN\",\n prefix: \"sessions/\",\n },\n sessionAnalysis: {\n enabled: false,\n analyzerUrl: process.env.AGENT_INSIGHTS_ANALYZER_URL ?? \"\",\n githubIssueRepo: \"vercel-labs/agent-insights\",\n },\n agents: {\n claudeCode: { enabled: false },\n cursor: { enabled: false },\n },\n privacy: {\n capturePrompts: false,\n captureMessages: false,\n captureToolInputs: false,\n captureFileContents: false,\n },\n };\n}\n\nexport async function loadConfig(): Promise<AgentInsightsConfig | undefined> {\n try {\n const raw = await readFile(configFile(), \"utf8\");\n const parsed = JSON.parse(raw) as Partial<AgentInsightsConfig>;\n return mergeConfig(defaultConfig(), parsed);\n } catch (err) {\n if ((err as NodeJS.ErrnoException).code === \"ENOENT\") return undefined;\n throw err;\n }\n}\n\nexport async function saveConfig(cfg: AgentInsightsConfig): Promise<void> {\n await ensureDirs();\n await writeFile(configFile(), `${JSON.stringify(cfg, null, 2)}\\n`, \"utf8\");\n}\n\nexport async function ensureDirs(): Promise<void> {\n await mkdir(dirname(configFile()), { recursive: true });\n await mkdir(sessionCacheDir(), { recursive: true });\n await mkdir(logsDir(), { recursive: true });\n}\n\nfunction mergeConfig(\n base: AgentInsightsConfig,\n patch: Partial<AgentInsightsConfig>,\n): AgentInsightsConfig {\n return {\n ...base,\n ...patch,\n vercel: { ...base.vercel, ...(patch.vercel ?? {}) },\n userConsent: { ...base.userConsent, ...(patch.userConsent ?? {}) },\n exporter: { ...base.exporter, ...(patch.exporter ?? {}) },\n transcriptStore: (patch.transcriptStore ??\n base.transcriptStore) as TranscriptStoreConfig,\n sessionAnalysis: {\n ...base.sessionAnalysis,\n ...(patch.sessionAnalysis ?? {}),\n },\n agents: {\n claudeCode: {\n ...base.agents.claudeCode,\n ...(patch.agents?.claudeCode ?? {}),\n },\n cursor: { ...base.agents.cursor, ...(patch.agents?.cursor ?? {}) },\n },\n privacy: base.privacy,\n };\n}\n","/**\n * Keys that must never appear in any payload that leaves the machine.\n *\n * Matching is case-insensitive and snake_case/camelCase agnostic — we strip a\n * key if its normalized form equals one of these values.\n */\nconst FORBIDDEN_KEYS: ReadonlySet<string> = new Set([\n \"prompt\",\n \"prompts\",\n \"message\",\n \"messages\",\n \"content\",\n \"text\",\n \"transcript\",\n \"toolinput\",\n \"input\",\n \"arguments\",\n \"args\",\n \"params\",\n \"file\",\n \"filecontent\",\n \"filecontents\",\n \"diff\",\n \"patch\",\n \"secret\",\n \"secrets\",\n \"token\",\n \"apikey\",\n \"authorization\",\n \"password\",\n \"credentials\",\n]);\n\nconst MAX_STRING_LENGTH = 256;\n\n/** Allowed primitive value types in sanitized metadata. */\nexport type SanitizedPrimitive = string | number | boolean | null;\n\nfunction normalizeKey(key: string): string {\n return key.toLowerCase().replace(/[_\\-\\s]/g, \"\");\n}\n\nfunction truncate(value: string): string {\n if (value.length <= MAX_STRING_LENGTH) return value;\n return `${value.slice(0, MAX_STRING_LENGTH)}…[truncated]`;\n}\n\nfunction sanitizePrimitive(value: unknown): SanitizedPrimitive | undefined {\n if (value === null) return null;\n switch (typeof value) {\n case \"string\":\n return truncate(value);\n case \"number\":\n return Number.isFinite(value) ? value : null;\n case \"boolean\":\n return value;\n default:\n return undefined;\n }\n}\n\n/**\n * Recursively remove forbidden keys from an arbitrary payload.\n *\n * Returned shape:\n * - Objects keep allowed primitive fields (truncated to {@link MAX_STRING_LENGTH}).\n * - Arrays keep allowed primitive elements; nested objects/arrays are recursed.\n * - Non-primitive, non-container values (functions, symbols, Buffers, …) are dropped.\n */\nexport function sanitize(input: unknown): unknown {\n return sanitizeInner(input);\n}\n\nfunction sanitizeInner(value: unknown): unknown {\n if (Array.isArray(value)) {\n const out: unknown[] = [];\n for (const item of value) {\n const sanitizedItem = sanitizeInner(item);\n if (sanitizedItem !== undefined) {\n out.push(sanitizedItem);\n }\n }\n return out;\n }\n\n if (value && typeof value === \"object\") {\n const out: Record<string, unknown> = {};\n for (const [rawKey, rawValue] of Object.entries(value)) {\n if (FORBIDDEN_KEYS.has(normalizeKey(rawKey))) {\n continue;\n }\n if (rawValue && typeof rawValue === \"object\") {\n out[rawKey] = sanitizeInner(rawValue);\n } else {\n const primitive = sanitizePrimitive(rawValue);\n if (primitive !== undefined) {\n out[rawKey] = primitive;\n }\n }\n }\n return out;\n }\n\n return sanitizePrimitive(value);\n}\n\nexport const __testing = { FORBIDDEN_KEYS, normalizeKey };\n","import { execSync } from \"node:child_process\";\nimport { hostname, userInfo } from \"node:os\";\nimport { sha256 } from \"../privacy/hash.js\";\n\n/** Stable hashed identity of the local user. */\nexport function userHash(): string {\n try {\n const user = userInfo().username;\n return sha256(`${user}@${hostname()}`);\n } catch {\n return sha256(\"unknown\");\n }\n}\n\n/**\n * Stable hashed identity of a repository, derived from `git remote get-url\n * origin` when available, else from `cwd`.\n */\nexport function repoHash(cwd: string = process.cwd()): string {\n const remote = tryGit([\"config\", \"--get\", \"remote.origin.url\"], cwd);\n return sha256(remote ?? cwd);\n}\n\n/** Stable hashed identity of the workspace path. */\nexport function workspaceHash(cwd: string = process.cwd()): string {\n return sha256(cwd);\n}\n\nfunction tryGit(args: string[], cwd: string): string | undefined {\n try {\n const out = execSync(`git ${args.join(\" \")}`, {\n cwd,\n stdio: [\"ignore\", \"pipe\", \"ignore\"],\n encoding: \"utf8\",\n timeout: 1500,\n });\n const trimmed = out.trim();\n return trimmed === \"\" ? undefined : trimmed;\n } catch {\n return undefined;\n }\n}\n","import { createHash } from \"node:crypto\";\n\n/**\n * Stable SHA-256 hash of a value, hex-encoded.\n *\n * Used to anonymize usernames, repo remote URLs, workspace paths, and\n * session identifiers before they leave the machine.\n */\nexport function sha256(value: string): string {\n return createHash(\"sha256\").update(value, \"utf8\").digest(\"hex\");\n}\n\n/** Short prefix of `sha256(value)` for human-readable dashboards. */\nexport function shortHash(value: string, length = 12): string {\n return sha256(value).slice(0, length);\n}\n","import kleur from \"kleur\";\n\nexport const log = {\n info: (msg: string): void => {\n process.stdout.write(`${msg}\\n`);\n },\n ok: (msg: string): void => {\n process.stdout.write(`${kleur.green(\"✓\")} ${msg}\\n`);\n },\n warn: (msg: string): void => {\n process.stdout.write(`${kleur.yellow(\"!\")} ${msg}\\n`);\n },\n fail: (msg: string): void => {\n process.stderr.write(`${kleur.red(\"✗\")} ${msg}\\n`);\n },\n hint: (msg: string): void => {\n process.stdout.write(`${kleur.dim(msg)}\\n`);\n },\n json: (obj: unknown): void => {\n process.stdout.write(`${JSON.stringify(obj, null, 2)}\\n`);\n },\n};\n\nexport function debug(msg: string, extra?: Record<string, unknown>): void {\n if (process.env.AGENT_INSIGHTS_DEBUG) {\n const payload = extra ? ` ${JSON.stringify(extra)}` : \"\";\n process.stderr.write(`[agent-insights] ${msg}${payload}\\n`);\n }\n}\n","import { rm } from \"node:fs/promises\";\nimport { adapterList } from \"../adapters/registry.js\";\nimport { loadConfig, saveConfig } from \"../config/config.js\";\nimport { configRoot } from \"../config/paths.js\";\nimport { log } from \"../util/log.js\";\nimport type { AgentId } from \"../adapters/types.js\";\n\nexport type DisableOptions = {\n agent?: string;\n purge?: boolean;\n};\n\nexport async function runDisable(opts: DisableOptions): Promise<void> {\n const repoRoot = process.cwd();\n const targets = opts.agent ? [opts.agent] : adapterList.map((a) => a.id);\n\n for (const id of targets) {\n if (id !== \"claude-code\" && id !== \"cursor\") {\n log.warn(`Unknown agent id \"${id}\"`);\n continue;\n }\n const adapter = adapterList.find((a) => a.id === (id as AgentId));\n if (!adapter) continue;\n const { removed, path } = await adapter.uninstall(repoRoot);\n if (removed) log.ok(`Removed ${adapter.label} hooks from ${path}`);\n else log.hint(`${adapter.label}: no hooks to remove (${path})`);\n }\n\n const cfg = await loadConfig();\n if (cfg) {\n if (!opts.agent || opts.agent === \"claude-code\") {\n cfg.agents.claudeCode.enabled = false;\n }\n if (!opts.agent || opts.agent === \"cursor\") {\n cfg.agents.cursor.enabled = false;\n }\n if (!opts.agent) cfg.enabled = false;\n await saveConfig(cfg);\n }\n\n if (opts.purge) {\n await rm(configRoot(), { recursive: true, force: true });\n log.ok(`Purged ${configRoot()}`);\n }\n}\n","import { mkdir, readFile, writeFile } from \"node:fs/promises\";\nimport { homedir } from \"node:os\";\nimport { dirname, join } from \"node:path\";\nimport type { HooksScope } from \"../config/config.js\";\nimport type { AgentEventType } from \"../types.js\";\nimport type { Adapter, HookPayload, MappedEvent } from \"./types.js\";\n\nconst HOOK_COMMAND_NAME = \"agent-insights\";\n\nconst HOOK_MAP: Record<string, AgentEventType> = {\n SessionStart: \"session.start\",\n SessionEnd: \"session.end\",\n UserPromptSubmit: \"user.prompt.submit\",\n PreToolUse: \"tool.start\",\n PostToolUse: \"tool.end\",\n PostToolUseFailure: \"tool.failure\",\n PermissionRequest: \"permission.request\",\n PermissionDenied: \"permission.denied\",\n SubagentStart: \"subagent.start\",\n SubagentStop: \"subagent.end\",\n Stop: \"agent.stop\",\n Notification: \"notification\",\n PreCompact: \"context.compact.start\",\n PostCompact: \"context.compact.end\",\n};\n\nexport const CLAUDE_HOOK_EVENTS: ReadonlyArray<string> = Object.keys(HOOK_MAP);\n\ntype ClaudeHookEntry = {\n type: \"command\";\n command: string;\n};\n\ntype ClaudeSettings = {\n hooks?: Record<string, ClaudeHookEntry[]>;\n [key: string]: unknown;\n};\n\nexport const claudeCodeAdapter: Adapter = {\n id: \"claude-code\",\n agent: \"claude-code\",\n label: \"Claude Code\",\n settingsFile: \".claude/settings.json\",\n\n map(payload: HookPayload): MappedEvent | undefined {\n const type = HOOK_MAP[payload.event];\n if (!type) return undefined;\n\n const data = payload.data ?? {};\n const sessionId =\n asString(data[\"session_id\"]) ?? asString(data[\"sessionId\"]);\n const toolName =\n asString(data[\"tool_name\"]) ??\n asString(data[\"toolName\"]) ??\n asString((data[\"tool\"] as Record<string, unknown> | undefined)?.[\"name\"]);\n const transcriptPath =\n asString(data[\"transcript_path\"]) ?? asString(data[\"transcriptPath\"]);\n\n return {\n type,\n ...(sessionId !== undefined ? { sessionId } : {}),\n ...(toolName !== undefined ? { toolName } : {}),\n ...(transcriptPath !== undefined ? { transcriptPath } : {}),\n };\n },\n\n async install(repoRoot: string, scope: HooksScope = \"global\") {\n const path = resolveClaudePath(repoRoot, scope);\n const settings = await readJson<ClaudeSettings>(path);\n const next: ClaudeSettings = { ...(settings ?? {}) };\n const hooks: Record<string, ClaudeHookEntry[]> = { ...(next.hooks ?? {}) };\n\n for (const event of CLAUDE_HOOK_EVENTS) {\n const existing = (hooks[event] ?? []).filter(\n (h) => !isAgentInsightsCommand(h.command),\n );\n existing.push({\n type: \"command\",\n command: `${HOOK_COMMAND_NAME} hook ${event}`,\n });\n hooks[event] = existing;\n }\n\n next.hooks = hooks;\n await mkdir(dirname(path), { recursive: true });\n await writeFile(path, `${JSON.stringify(next, null, 2)}\\n`, \"utf8\");\n return { written: true, path };\n },\n\n async uninstall(repoRoot: string, scope: HooksScope = \"global\") {\n const path = resolveClaudePath(repoRoot, scope);\n const settings = await readJson<ClaudeSettings>(path);\n if (!settings?.hooks) return { removed: false, path };\n const hooks = { ...settings.hooks };\n let touched = false;\n for (const [event, entries] of Object.entries(hooks)) {\n const filtered = entries.filter(\n (h) => !isAgentInsightsCommand(h.command),\n );\n if (filtered.length !== entries.length) touched = true;\n if (filtered.length === 0) {\n delete hooks[event];\n } else {\n hooks[event] = filtered;\n }\n }\n settings.hooks = hooks;\n await writeFile(path, `${JSON.stringify(settings, null, 2)}\\n`, \"utf8\");\n return { removed: touched, path };\n },\n\n async isInstalled(repoRoot: string, scope: HooksScope = \"global\") {\n const settings = await readJson<ClaudeSettings>(\n resolveClaudePath(repoRoot, scope),\n );\n if (!settings?.hooks) return false;\n return Object.values(settings.hooks).some((entries) =>\n entries.some((h) => isAgentInsightsCommand(h.command)),\n );\n },\n};\n\nfunction resolveClaudePath(repoRoot: string, scope: HooksScope): string {\n return scope === \"global\"\n ? join(homedir(), \".claude\", \"settings.json\")\n : join(repoRoot, \".claude\", \"settings.json\");\n}\n\nfunction isAgentInsightsCommand(cmd: string | undefined): boolean {\n if (!cmd) return false;\n return cmd.startsWith(`${HOOK_COMMAND_NAME} hook`);\n}\n\nasync function readJson<T>(path: string): Promise<T | undefined> {\n try {\n const raw = await readFile(path, \"utf8\");\n return JSON.parse(raw) as T;\n } catch (err) {\n if ((err as NodeJS.ErrnoException).code === \"ENOENT\") return undefined;\n throw err;\n }\n}\n\nfunction asString(v: unknown): string | undefined {\n return typeof v === \"string\" && v.length > 0 ? v : undefined;\n}\n","import { mkdir, readFile, writeFile } from \"node:fs/promises\";\nimport { homedir } from \"node:os\";\nimport { dirname, join } from \"node:path\";\nimport type { HooksScope } from \"../config/config.js\";\nimport type { AgentEventType } from \"../types.js\";\nimport type { Adapter, HookPayload, MappedEvent } from \"./types.js\";\n\nconst HOOK_COMMAND_NAME = \"agent-insights\";\n\n/**\n * Cursor native hook names (camelCase) → common schema. Reference:\n * https://cursor.com/docs/hooks\n */\nconst HOOK_MAP: Record<string, AgentEventType> = {\n sessionStart: \"session.start\",\n sessionEnd: \"session.end\",\n beforeSubmitPrompt: \"user.prompt.submit\",\n preToolUse: \"tool.start\",\n postToolUse: \"tool.end\",\n postToolUseFailure: \"tool.failure\",\n subagentStart: \"subagent.start\",\n subagentStop: \"subagent.end\",\n stop: \"agent.stop\",\n preCompact: \"context.compact.start\",\n // Additional events — mapped to existing schema types\n beforeShellExecution: \"tool.start\",\n afterShellExecution: \"tool.end\",\n afterFileEdit: \"tool.end\",\n workspaceOpen: \"session.start\",\n};\n\nexport const CURSOR_HOOK_EVENTS: ReadonlyArray<string> = Object.keys(HOOK_MAP);\n\ntype CursorHookEntry = {\n command: string;\n};\n\ntype CursorHooksConfig = {\n /** Cursor expects schema version 1 at the top level. */\n version?: number;\n hooks?: Record<string, CursorHookEntry[]>;\n [key: string]: unknown;\n};\n\nexport const cursorAdapter: Adapter = {\n id: \"cursor\",\n agent: \"cursor\",\n label: \"Cursor\",\n settingsFile: \".cursor/hooks.json\",\n\n map(payload: HookPayload): MappedEvent | undefined {\n const type = HOOK_MAP[payload.event];\n if (!type) return undefined;\n\n const data = payload.data ?? {};\n // Cursor uses `session_id` in payloads and is identical to `conversation_id`.\n const sessionId =\n asString(data[\"session_id\"]) ??\n asString(data[\"sessionId\"]) ??\n asString(data[\"conversation_id\"]);\n const toolName = asString(data[\"tool_name\"]) ?? asString(data[\"toolName\"]);\n\n // transcript_path is confirmed in all Cursor hook payloads.\n const transcriptPath =\n asString(data[\"transcript_path\"]) ??\n asString(data[\"transcriptPath\"]) ??\n asString(process.env.CURSOR_TRANSCRIPT_PATH);\n\n return {\n type,\n ...(sessionId !== undefined ? { sessionId } : {}),\n ...(toolName !== undefined ? { toolName } : {}),\n ...(transcriptPath !== undefined ? { transcriptPath } : {}),\n };\n },\n\n async install(repoRoot: string, scope: HooksScope = \"global\") {\n const path = resolveCursorPath(repoRoot, scope);\n const settings = await readJson<CursorHooksConfig>(path);\n const next: CursorHooksConfig = {\n version: 1,\n ...(settings ?? {}),\n };\n const hooks: Record<string, CursorHookEntry[]> = { ...(next.hooks ?? {}) };\n\n for (const event of CURSOR_HOOK_EVENTS) {\n const existing = (hooks[event] ?? []).filter(\n (h) => !isAgentInsightsCommand(h.command),\n );\n existing.push({ command: `${HOOK_COMMAND_NAME} hook ${event}` });\n hooks[event] = existing;\n }\n\n next.hooks = hooks;\n await mkdir(dirname(path), { recursive: true });\n await writeFile(path, `${JSON.stringify(next, null, 2)}\\n`, \"utf8\");\n return { written: true, path };\n },\n\n async uninstall(repoRoot: string, scope: HooksScope = \"global\") {\n const path = resolveCursorPath(repoRoot, scope);\n const settings = await readJson<CursorHooksConfig>(path);\n if (!settings?.hooks) return { removed: false, path };\n const hooks = { ...settings.hooks };\n let touched = false;\n for (const [event, entries] of Object.entries(hooks)) {\n const filtered = entries.filter(\n (h) => !isAgentInsightsCommand(h.command),\n );\n if (filtered.length !== entries.length) touched = true;\n if (filtered.length === 0) {\n delete hooks[event];\n } else {\n hooks[event] = filtered;\n }\n }\n settings.hooks = hooks;\n await writeFile(path, `${JSON.stringify(settings, null, 2)}\\n`, \"utf8\");\n return { removed: touched, path };\n },\n\n async isInstalled(repoRoot: string, scope: HooksScope = \"global\") {\n const settings = await readJson<CursorHooksConfig>(\n resolveCursorPath(repoRoot, scope),\n );\n if (!settings?.hooks) return false;\n return Object.values(settings.hooks).some((entries) =>\n entries.some((h) => isAgentInsightsCommand(h.command)),\n );\n },\n};\n\nfunction resolveCursorPath(repoRoot: string, scope: HooksScope): string {\n return scope === \"global\"\n ? join(homedir(), \".cursor\", \"hooks.json\")\n : join(repoRoot, \".cursor\", \"hooks.json\");\n}\n\nfunction isAgentInsightsCommand(cmd: string | undefined): boolean {\n if (!cmd) return false;\n return cmd.startsWith(`${HOOK_COMMAND_NAME} hook`);\n}\n\nasync function readJson<T>(path: string): Promise<T | undefined> {\n try {\n const raw = await readFile(path, \"utf8\");\n return JSON.parse(raw) as T;\n } catch (err) {\n if ((err as NodeJS.ErrnoException).code === \"ENOENT\") return undefined;\n throw err;\n }\n}\n\nfunction asString(v: unknown): string | undefined {\n return typeof v === \"string\" && v.length > 0 ? v : undefined;\n}\n","import { claudeCodeAdapter } from \"./claude-code.js\";\nimport { cursorAdapter } from \"./cursor.js\";\nimport type { Adapter, AgentId } from \"./types.js\";\n\nexport const adapters: Record<AgentId, Adapter> = {\n \"claude-code\": claudeCodeAdapter,\n cursor: cursorAdapter,\n};\n\nexport const adapterList: ReadonlyArray<Adapter> = Object.values(adapters);\n\nexport function getAdapter(id: AgentId): Adapter {\n return adapters[id];\n}\n","import kleur from \"kleur\";\nimport { adapterList } from \"../adapters/registry.js\";\nimport { loadConfig } from \"../config/config.js\";\nimport { configFile } from \"../config/paths.js\";\nimport { getValidToken } from \"../auth/store.js\";\nimport { createTranscriptStore } from \"../transcript/store.js\";\nimport { log } from \"../util/log.js\";\n\nexport type DoctorOptions = {\n json?: boolean;\n};\n\ntype Check = {\n name: string;\n ok: boolean;\n detail?: string;\n};\n\nexport async function runDoctor(opts: DoctorOptions): Promise<void> {\n const repoRoot = process.cwd();\n const checks: Check[] = [];\n\n const token = await getValidToken();\n checks.push({\n name: \"logged in\",\n ok: token != null,\n ...(token ? {} : { detail: \"run `agent-insights login`\" }),\n });\n\n const cfg = await loadConfig();\n checks.push({\n name: \"config found\",\n ok: cfg != null,\n detail: cfg ? configFile() : `missing ${configFile()}`,\n });\n\n if (cfg) {\n if (cfg.userConsent.transcriptSync) {\n try {\n const store = createTranscriptStore(cfg.transcriptStore);\n const ping = await store.ping();\n if (ping.ok) {\n checks.push({ name: \"transcript store reachable\", ok: true });\n } else {\n checks.push({\n name: \"transcript store reachable\",\n ok: false,\n detail: ping.reason,\n });\n }\n } catch (err) {\n checks.push({\n name: \"transcript store reachable\",\n ok: false,\n detail: (err as Error).message,\n });\n }\n }\n\n for (const adapter of adapterList) {\n const enabled =\n adapter.id === \"claude-code\"\n ? cfg.agents.claudeCode.enabled\n : cfg.agents.cursor.enabled;\n if (!enabled) continue;\n const installed = await adapter.isInstalled(repoRoot, cfg.hooksScope);\n checks.push({\n name: `${adapter.label} hooks installed`,\n ok: installed,\n ...(installed ? {} : { detail: \"run `agent-insights init`\" }),\n });\n }\n }\n\n if (opts.json) {\n log.json({ ok: checks.every((c) => c.ok), checks });\n } else {\n for (const c of checks) {\n const tag = c.ok ? kleur.green(\"✓\") : kleur.red(\"✗\");\n const detail = c.detail ? kleur.dim(` (${c.detail})`) : \"\";\n log.info(`${tag} ${c.name}${detail}`);\n }\n }\n\n process.exitCode = checks.every((c) => c.ok) ? 0 : 1;\n}\n","import { readFile } from \"node:fs/promises\";\nimport { put } from \"@vercel/blob\";\nimport type { TranscriptStoreConfig } from \"../config/config.js\";\n\nexport type TranscriptUploadInput = {\n /** Absolute path on disk to the transcript file. */\n transcriptPath: string;\n /** SHA256-hashed user identifier. Used in the upload pathname. */\n userHash: string;\n /** Session identifier supplied by the agent. */\n sessionId: string;\n /** Agent name (e.g. \"claude-code\"). */\n agent: string;\n};\n\nexport type TranscriptUploadResult = {\n pathname: string;\n url: string;\n size: number;\n};\n\nexport interface TranscriptStore {\n upload(input: TranscriptUploadInput): Promise<TranscriptUploadResult>;\n /** Lightweight health check used by `doctor`. */\n ping(): Promise<{ ok: true } | { ok: false; reason: string }>;\n}\n\nclass NoopTranscriptStore implements TranscriptStore {\n async upload(): Promise<TranscriptUploadResult> {\n throw new Error(\"transcript store disabled (transcriptStore.type=none)\");\n }\n async ping(): Promise<{ ok: false; reason: string }> {\n return { ok: false, reason: \"transcript store disabled\" };\n }\n}\n\nclass BlobTranscriptStore implements TranscriptStore {\n constructor(\n private readonly token: string,\n private readonly prefix: string,\n ) {}\n\n async upload(input: TranscriptUploadInput): Promise<TranscriptUploadResult> {\n const body = await readFile(input.transcriptPath);\n const requestedPath = buildPathname(\n this.prefix,\n input.userHash,\n input.sessionId,\n );\n const blob = await put(requestedPath, body, {\n access: \"private\",\n token: this.token,\n contentType: \"application/jsonl\",\n addRandomSuffix: true,\n });\n return { pathname: blob.pathname, url: blob.url, size: body.byteLength };\n }\n\n async ping(): Promise<{ ok: true } | { ok: false; reason: string }> {\n if (!this.token) return { ok: false, reason: \"missing token\" };\n return { ok: true };\n }\n}\n\nexport function createTranscriptStore(\n cfg: TranscriptStoreConfig,\n): TranscriptStore {\n if (cfg.type === \"none\") return new NoopTranscriptStore();\n if (cfg.type === \"analyzer\") {\n // Analyzer-proxied uploads are handled directly in hook.ts via getValidToken().\n // createTranscriptStore is not used for this path.\n throw new Error(\n \"transcript store: type=analyzer uploads are handled by the hook command directly\",\n );\n }\n // type === \"blob\"\n const token = process.env[cfg.tokenEnv];\n if (!token) {\n throw new Error(\n `transcript store: env ${cfg.tokenEnv} is not set (run 'vercel env pull .env.local')`,\n );\n }\n return new BlobTranscriptStore(token, cfg.prefix);\n}\n\nfunction buildPathname(\n prefix: string,\n userHash: string,\n sessionId: string,\n): string {\n const safePrefix = prefix.endsWith(\"/\") ? prefix : `${prefix}/`;\n const safeSession = encodeURIComponent(sessionId);\n return `${safePrefix}${userHash}/${safeSession}/transcript.jsonl`;\n}\n","import { adapterList } from \"../adapters/registry.js\";\nimport { loadConfig } from \"../config/config.js\";\nimport { sanitize } from \"../privacy/sanitize.js\";\nimport { createTranscriptStore } from \"../transcript/store.js\";\nimport type { Adapter, AgentId, HookPayload } from \"../adapters/types.js\";\nimport type { AgentEvent } from \"../types.js\";\nimport { repoHash, userHash, workspaceHash } from \"../util/identity.js\";\nimport { debug, log } from \"../util/log.js\";\n\nexport type HookOptions = {\n agent?: string;\n};\n\n/**\n * Hook entrypoint. Reads JSON payload from stdin, resolves the adapter, maps\n * the native event into the common schema, and runs transcript upload on\n * session.end. Never fails the hook — errors are logged, exit code is 0.\n */\nexport async function runHook(\n eventName: string,\n opts: HookOptions,\n): Promise<void> {\n const cfg = await loadConfig();\n if (!cfg || !cfg.enabled) {\n debug(\"hook skipped — agent-insights not enabled\");\n return;\n }\n\n const stdin = await readStdinSafely();\n const data = parseJsonObject(stdin);\n const payload: HookPayload = { event: eventName, data };\n\n const adapter = pickAdapter(opts.agent, eventName);\n if (!adapter) {\n debug(`no adapter resolved for event ${eventName}`);\n return;\n }\n\n const mapped = adapter.map(payload);\n if (!mapped) {\n debug(`adapter ${adapter.id} did not map event ${eventName}`);\n return;\n }\n\n const event = buildEvent(adapter, mapped);\n debug(\"hook event\", {\n agent: event.agent,\n type: event.type,\n sessionId: event.sessionId,\n });\n\n if (\n mapped.type === \"session.end\" &&\n cfg.userConsent.transcriptSync &&\n mapped.transcriptPath\n ) {\n try {\n const store = createTranscriptStore(cfg.transcriptStore);\n const result = await store.upload({\n transcriptPath: mapped.transcriptPath,\n userHash: event.userHash ?? userHash(),\n sessionId: mapped.sessionId ?? \"unknown\",\n agent: adapter.id,\n });\n debug(\"transcript uploaded\", {\n size: result.size,\n pathname: result.pathname,\n });\n\n if (\n cfg.userConsent.sessionAnalysis &&\n cfg.sessionAnalysis.analyzerUrl\n ) {\n await notifyAnalyzer({\n analyzerUrl: cfg.sessionAnalysis.analyzerUrl,\n payload: {\n sessionId: mapped.sessionId ?? \"unknown\",\n agent: adapter.id,\n userHash: event.userHash ?? userHash(),\n blobUrl: result.url,\n timestamp: event.timestamp,\n },\n });\n }\n } catch (err) {\n log.fail(`transcript upload failed: ${(err as Error).message}`);\n }\n }\n\n // Phase 3 (OTLP) will export `event` here.\n void sanitize(event);\n}\n\ntype AnalyzerPayload = {\n sessionId: string;\n agent: string;\n userHash: string;\n blobUrl: string;\n timestamp: string;\n};\n\nasync function notifyAnalyzer(opts: {\n analyzerUrl: string;\n payload: AnalyzerPayload;\n}): Promise<void> {\n try {\n const { getValidToken } = await import(\"../auth/store.js\");\n const token = await getValidToken();\n const headers: Record<string, string> = {\n \"content-type\": \"application/json\",\n };\n if (token) headers[\"authorization\"] = `Bearer ${token}`;\n\n const url = new URL(\"/api/sessions\", opts.analyzerUrl).toString();\n const res = await fetch(url, {\n method: \"POST\",\n headers,\n body: JSON.stringify(opts.payload),\n });\n if (!res.ok) {\n debug(`analyzer notify failed: ${res.status} ${res.statusText}`);\n return;\n }\n debug(\"analyzer notified\", { sessionId: opts.payload.sessionId });\n } catch (err) {\n // Never fail the hook because of analyzer issues.\n debug(`analyzer notify error: ${(err as Error).message}`);\n }\n}\n\nfunction pickAdapter(\n override: string | undefined,\n eventName: string,\n): Adapter | undefined {\n if (override) {\n return adapterList.find((a) => a.id === override);\n }\n for (const a of adapterList) {\n if (a.map({ event: eventName })) return a;\n }\n return undefined;\n}\n\nfunction buildEvent(\n adapter: { id: AgentId; agent: AgentEvent[\"agent\"] },\n mapped: NonNullable<ReturnType<Adapter[\"map\"]>>,\n): AgentEvent {\n return {\n type: mapped.type,\n agent: adapter.agent,\n timestamp: new Date().toISOString(),\n ...(mapped.sessionId !== undefined ? { sessionId: mapped.sessionId } : {}),\n ...(mapped.toolName !== undefined ? { toolName: mapped.toolName } : {}),\n ...(mapped.permissionType !== undefined\n ? { permissionType: mapped.permissionType }\n : {}),\n ...(mapped.outcome !== undefined ? { outcome: mapped.outcome } : {}),\n userHash: userHash(),\n repoHash: repoHash(),\n workspaceHash: workspaceHash(),\n };\n}\n\n/**\n * Read stdin if data is available. Returns an empty string when stdin is a\n * TTY or when nothing arrives within `STDIN_TIMEOUT_MS`. Hooks invoked\n * without piped JSON should never hang.\n */\nconst STDIN_TIMEOUT_MS = 200;\n\nasync function readStdinSafely(): Promise<string> {\n if (process.stdin.isTTY) return \"\";\n return new Promise<string>((resolve) => {\n let buf = \"\";\n let settled = false;\n const finish = (): void => {\n if (settled) return;\n settled = true;\n process.stdin.removeAllListeners(\"data\");\n process.stdin.removeAllListeners(\"end\");\n process.stdin.removeAllListeners(\"error\");\n // Release the event loop — `setEncoding` + 'data' listener cause Node\n // to keep reading stdin even after we've stopped listening.\n try {\n process.stdin.pause();\n process.stdin.unref();\n } catch {\n // ignore — stdin may already be destroyed\n }\n resolve(buf);\n };\n const idle = setTimeout(finish, STDIN_TIMEOUT_MS);\n idle.unref();\n process.stdin.setEncoding(\"utf8\");\n process.stdin.on(\"data\", (chunk) => {\n buf += chunk;\n idle.refresh();\n });\n process.stdin.on(\"end\", () => {\n clearTimeout(idle);\n finish();\n });\n process.stdin.on(\"error\", () => {\n clearTimeout(idle);\n finish();\n });\n });\n}\n\nfunction parseJsonObject(raw: string): Record<string, unknown> | undefined {\n if (!raw) return undefined;\n try {\n const parsed = JSON.parse(raw) as unknown;\n if (parsed && typeof parsed === \"object\" && !Array.isArray(parsed)) {\n return parsed as Record<string, unknown>;\n }\n return undefined;\n } catch {\n return undefined;\n }\n}\n","import kleur from \"kleur\";\nimport { adapterList } from \"../adapters/registry.js\";\nimport {\n defaultConfig,\n loadConfig,\n saveConfig,\n type AgentInsightsConfig,\n} from \"../config/config.js\";\nimport { configFile, configRoot } from \"../config/paths.js\";\nimport { getValidToken } from \"../auth/store.js\";\nimport { log } from \"../util/log.js\";\n\nexport type InitOptions = {\n agents?: string;\n transcripts?: boolean;\n analysis?: boolean;\n telemetry?: boolean;\n force?: boolean;\n hooks?: boolean;\n};\n\nconst CONSENT_TEXT = `\\\nAgent Insights is opt-in (internal).\n\nOTLP / lifecycle metrics: event names, timestamps, tool names, outcomes,\nhashed user/repo/workspace IDs — never prompts, messages, or tool inputs.\n\nVercel Blob (optional): full session transcripts for internal analysis;\nusername stored as SHA256 only.\n\nSessionEnd analysis (optional): an internal LLM reviews transcript for\ntooling gaps; may post to Slack with a GitHub issue link.\\\n`;\n\nexport async function runInit(opts: InitOptions): Promise<void> {\n const token = await getValidToken();\n if (!token) {\n log.fail(\"Not logged in. Run `agent-insights login` first.\");\n process.exit(1);\n }\n\n log.info(kleur.bold(\"Agent Insights setup\\n\"));\n log.hint(CONSENT_TEXT);\n log.info(\"\");\n\n const existing = await loadConfig();\n if (existing && !opts.force) {\n log.warn(\n `Config already exists at ${configFile()} — re-run with --force to overwrite.`,\n );\n }\n\n const requested = parseAgents(opts.agents);\n const cfg: AgentInsightsConfig = existing ?? defaultConfig();\n\n cfg.userConsent.transcriptSync = opts.transcripts ?? true;\n cfg.userConsent.sessionAnalysis = opts.analysis ?? false;\n cfg.userConsent.eventTelemetry = opts.telemetry ?? false;\n\n cfg.agents.claudeCode.enabled = requested.has(\"claude-code\");\n cfg.agents.cursor.enabled = requested.has(\"cursor\");\n\n await saveConfig(cfg);\n log.ok(`Wrote config to ${configFile()}`);\n\n const installHooks = opts.hooks ?? true;\n const repoRoot = process.cwd();\n if (installHooks) {\n for (const adapter of adapterList) {\n const enabled =\n (adapter.id === \"claude-code\" && cfg.agents.claudeCode.enabled) ||\n (adapter.id === \"cursor\" && cfg.agents.cursor.enabled);\n if (!enabled) continue;\n const { path } = await adapter.install(repoRoot, cfg.hooksScope);\n log.ok(`Installed ${adapter.label} hooks → ${rel(path)}`);\n }\n }\n\n log.info(\"\");\n log.info(`Config root: ${configRoot()}`);\n log.hint(\"Run `agent-insights doctor` to verify the setup.\");\n}\n\nfunction parseAgents(value: string | undefined): Set<\"claude-code\" | \"cursor\"> {\n const out = new Set<\"claude-code\" | \"cursor\">();\n const list = (value ?? \"claude-code,cursor\")\n .split(\",\")\n .map((s) => s.trim())\n .filter(Boolean);\n for (const id of list) {\n if (id === \"claude-code\" || id === \"cursor\") out.add(id);\n else log.warn(`Unknown agent id \"${id}\" — skipping.`);\n }\n return out;\n}\n\nfunction rel(p: string): string {\n const cwd = process.cwd();\n return p.startsWith(cwd) ? p.slice(cwd.length + 1) : p;\n}\n","import kleur from \"kleur\";\nimport {\n exchangeCodeForToken,\n fetchUserInfo,\n openBrowser,\n REDIRECT_URI,\n startPkceFlow,\n} from \"../auth/oauth.js\";\nimport { clearAuth, saveAuth } from \"../auth/store.js\";\nimport { log } from \"../util/log.js\";\n\nconst ALLOWED_DOMAIN = \"vercel.com\";\n\nexport async function runLogin(): Promise<void> {\n log.info(kleur.bold(\"Signing in with Vercel...\\n\"));\n\n // 1. Build auth URL + start local callback server\n let pkceResult: { code: string; codeVerifier: string };\n try {\n log.hint(`Listening on ${REDIRECT_URI} — opening browser...`);\n pkceResult = await startPkceFlow();\n } catch (err) {\n log.fail(`Login failed: ${(err as Error).message}`);\n process.exit(1);\n }\n\n // 2. Exchange code for tokens\n let tokens;\n try {\n tokens = await exchangeCodeForToken(pkceResult.code, pkceResult.codeVerifier);\n } catch (err) {\n log.fail(`Token exchange failed: ${(err as Error).message}`);\n process.exit(1);\n }\n\n // 3. Verify @vercel.com email\n let userInfo;\n try {\n userInfo = await fetchUserInfo(tokens.access_token);\n } catch (err) {\n log.fail(`Could not fetch user info: ${(err as Error).message}`);\n process.exit(1);\n }\n\n if (\n !userInfo.email?.endsWith(`@${ALLOWED_DOMAIN}`) ||\n !userInfo.email_verified\n ) {\n log.fail(\n `Only @${ALLOWED_DOMAIN} accounts are allowed. Got: ${userInfo.email ?? \"(no email)\"}`,\n );\n process.exit(1);\n }\n\n // 4. Save tokens\n await saveAuth({\n accessToken: tokens.access_token,\n ...(tokens.refresh_token !== undefined\n ? { refreshToken: tokens.refresh_token }\n : {}),\n expiresAt: Date.now() + tokens.expires_in * 1000,\n });\n\n const name = userInfo.preferred_username ?? userInfo.email;\n log.ok(`Logged in as ${kleur.bold(name)} (${userInfo.email})`);\n log.hint(\"Run `agent-insights init` to install hooks globally.\");\n}\n\nexport async function runLogout(): Promise<void> {\n await clearAuth();\n log.ok(\"Logged out.\");\n}\n","import { existsSync } from \"node:fs\";\nimport { join } from \"node:path\";\nimport kleur from \"kleur\";\nimport { adapterList } from \"../adapters/registry.js\";\nimport { loadConfig } from \"../config/config.js\";\nimport { configFile } from \"../config/paths.js\";\nimport { log } from \"../util/log.js\";\n\nexport type StatusOptions = {\n json?: boolean;\n};\n\nexport async function runStatus(opts: StatusOptions): Promise<void> {\n const cfg = await loadConfig();\n const repoRoot = process.cwd();\n\n if (!cfg) {\n if (opts.json) log.json({ initialized: false });\n else log.warn(`No config found at ${configFile()} — run \\`agent-insights init\\`.`);\n return;\n }\n\n const adapterStatus = await Promise.all(\n adapterList.map(async (a) => ({\n id: a.id,\n label: a.label,\n enabled:\n a.id === \"claude-code\"\n ? cfg.agents.claudeCode.enabled\n : cfg.agents.cursor.enabled,\n installed: await a.isInstalled(repoRoot),\n })),\n );\n\n if (opts.json) {\n log.json({\n initialized: true,\n configPath: configFile(),\n enabled: cfg.enabled,\n consent: cfg.userConsent,\n vercel: {\n linked: existsSync(join(repoRoot, \".vercel/project.json\")),\n envFile: existsSync(join(repoRoot, \".env.local\")),\n },\n adapters: adapterStatus,\n });\n return;\n }\n\n log.info(kleur.bold(\"Agent Insights status\\n\"));\n log.info(`config ${configFile()}`);\n log.info(`enabled ${cfg.enabled ? \"yes\" : \"no\"}`);\n log.info(\"\");\n log.info(kleur.bold(\"Consent\"));\n log.info(` event telemetry ${yn(cfg.userConsent.eventTelemetry)}`);\n log.info(` transcript sync ${yn(cfg.userConsent.transcriptSync)}`);\n log.info(` session analysis ${yn(cfg.userConsent.sessionAnalysis)}`);\n log.info(\"\");\n log.info(kleur.bold(\"Adapters\"));\n for (const a of adapterStatus) {\n log.info(\n ` ${a.label.padEnd(12)} ${a.enabled ? \"enabled\" : \"disabled\"} hooks ${\n a.installed ? \"installed\" : \"not installed\"\n }`,\n );\n }\n log.info(\"\");\n log.info(kleur.bold(\"Vercel\"));\n log.info(\n ` linked ${yn(existsSync(join(repoRoot, \".vercel/project.json\")))}`,\n );\n log.info(` .env.local ${yn(existsSync(join(repoRoot, \".env.local\")))}`);\n}\n\nfunction yn(v: boolean): string {\n return v ? kleur.green(\"yes\") : kleur.dim(\"no\");\n}\n"],"mappings":"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","names":["createHash","readFile","writeFile","dirname","mkdir","mkdir","readFile","writeFile","homedir","dirname","join","mkdir","readFile","writeFile","homedir","dirname","join","HOOK_COMMAND_NAME","HOOK_MAP","asString","readJson","isAgentInsightsCommand","kleur","readFile","userHash","kleur","getValidToken","kleur","kleur","kleur","kleur","userInfo","join","kleur","join","kleur"]}
package/dist/index.d.ts CHANGED
@@ -36,50 +36,18 @@ declare function shortHash(value: string, length?: number): string;
36
36
  */
37
37
  declare function sanitize(input: unknown): unknown;
38
38
 
39
- type AgentId = "claude-code" | "cursor";
40
- type HookPayload = {
41
- /** Native event name as emitted by the agent (e.g. `SessionStart`). */
42
- event: string;
43
- /** Parsed JSON body of the hook stdin (may be undefined for empty stdin). */
44
- data?: Record<string, unknown> | undefined;
45
- };
46
- type MappedEvent = {
47
- type: AgentEventType;
48
- sessionId?: string;
49
- toolName?: string;
50
- permissionType?: string;
51
- outcome?: AgentEvent["outcome"];
52
- /** Absolute path to a session transcript file, if the hook exposed one. */
53
- transcriptPath?: string;
54
- };
55
- type Adapter = {
56
- id: AgentId;
57
- agent: AgentName;
58
- label: string;
59
- settingsFile: string;
60
- map(payload: HookPayload): MappedEvent | undefined;
61
- install(repoRoot: string): Promise<{
62
- written: boolean;
63
- path: string;
64
- }>;
65
- uninstall(repoRoot: string): Promise<{
66
- removed: boolean;
67
- path: string;
68
- }>;
69
- isInstalled(repoRoot: string): Promise<boolean>;
70
- };
71
-
72
- declare const adapters: Record<AgentId, Adapter>;
73
- declare const adapterList: ReadonlyArray<Adapter>;
74
- declare function getAdapter(id: AgentId): Adapter;
75
-
76
39
  type ExporterConfig = {
77
40
  type: "otlp";
78
41
  endpoint: string;
79
42
  headers: Record<string, string>;
80
43
  dataset: string;
81
44
  };
45
+ type HooksScope = "global" | "local";
82
46
  type TranscriptStoreConfig = {
47
+ /** Upload via analyzer API using AGENT_INSIGHTS_API_KEY (preferred). */
48
+ type: "analyzer";
49
+ } | {
50
+ /** Direct upload with a Blob RW token (legacy / advanced). */
83
51
  type: "blob";
84
52
  tokenEnv: string;
85
53
  prefix: string;
@@ -91,8 +59,6 @@ type SessionAnalysisConfig = {
91
59
  enabled: boolean;
92
60
  /** Analyzer endpoint base URL. CLI POSTs to `${analyzerUrl}/api/sessions`. */
93
61
  analyzerUrl: string;
94
- /** Env var name holding an optional shared secret. Sent as `x-agent-insights-secret`. */
95
- secretEnv: string;
96
62
  /** GitHub repo for pre-filled new-issue URLs (display only; constructed by analyzer). */
97
63
  githubIssueRepo: string;
98
64
  };
@@ -102,6 +68,8 @@ type AgentConfig = {
102
68
  type AgentInsightsConfig = {
103
69
  version: 1;
104
70
  enabled: boolean;
71
+ /** Where agent hook entries are written. Default `global` (user home). */
72
+ hooksScope: HooksScope;
105
73
  vercel: {
106
74
  team: string;
107
75
  project: string;
@@ -126,6 +94,43 @@ declare function defaultConfig(): AgentInsightsConfig;
126
94
  declare function loadConfig(): Promise<AgentInsightsConfig | undefined>;
127
95
  declare function saveConfig(cfg: AgentInsightsConfig): Promise<void>;
128
96
 
97
+ type AgentId = "claude-code" | "cursor";
98
+ type HookPayload = {
99
+ /** Native event name as emitted by the agent (e.g. `SessionStart`). */
100
+ event: string;
101
+ /** Parsed JSON body of the hook stdin (may be undefined for empty stdin). */
102
+ data?: Record<string, unknown> | undefined;
103
+ };
104
+ type MappedEvent = {
105
+ type: AgentEventType;
106
+ sessionId?: string;
107
+ toolName?: string;
108
+ permissionType?: string;
109
+ outcome?: AgentEvent["outcome"];
110
+ /** Absolute path to a session transcript file, if the hook exposed one. */
111
+ transcriptPath?: string;
112
+ };
113
+ type Adapter = {
114
+ id: AgentId;
115
+ agent: AgentName;
116
+ label: string;
117
+ settingsFile: string;
118
+ map(payload: HookPayload): MappedEvent | undefined;
119
+ install(repoRoot: string, scope?: HooksScope): Promise<{
120
+ written: boolean;
121
+ path: string;
122
+ }>;
123
+ uninstall(repoRoot: string, scope?: HooksScope): Promise<{
124
+ removed: boolean;
125
+ path: string;
126
+ }>;
127
+ isInstalled(repoRoot: string, scope?: HooksScope): Promise<boolean>;
128
+ };
129
+
130
+ declare const adapters: Record<AgentId, Adapter>;
131
+ declare const adapterList: ReadonlyArray<Adapter>;
132
+ declare function getAdapter(id: AgentId): Adapter;
133
+
129
134
  type TranscriptUploadInput = {
130
135
  /** Absolute path on disk to the transcript file. */
131
136
  transcriptPath: string;