@jefuriiij/synthra 0.1.20 → 0.1.21
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/CHANGELOG.md +23 -0
- package/dist/cli/index.js +56 -2
- package/dist/cli/index.js.map +1 -1
- package/dist/dashboard/index.js +1 -1
- package/dist/dashboard/index.js.map +1 -1
- package/dist/server/index.js +55 -1
- package/dist/server/index.js.map +1 -1
- package/package.json +1 -1
package/dist/server/index.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../../src/server/http.ts","../../src/activity/activity-log.ts","../../src/activity/file-watcher.ts","../../src/shared/logger.ts","../../src/activity/git-watcher.ts","../../src/cli/scan-command.ts","../../src/scanner/extract.ts","../../src/scanner/hash.ts","../../src/scanner/keywords.ts","../../src/scanner/parser.ts","../../src/scanner/parsers/_generic.ts","../../src/scanner/parsers/c.ts","../../src/scanner/parsers/cpp.ts","../../src/scanner/parsers/csharp.ts","../../src/scanner/parsers/dart.ts","../../src/scanner/parsers/go.ts","../../src/scanner/parsers/java.ts","../../src/scanner/parsers/kotlin.ts","../../src/scanner/parsers/php.ts","../../src/scanner/parsers/python.ts","../../src/scanner/parsers/ruby.ts","../../src/scanner/parsers/rust.ts","../../src/scanner/parsers/typescript.ts","../../src/scanner/parsers/svelte.ts","../../src/scanner/parsers/vue.ts","../../src/scanner/walker.ts","../../src/graph/store.ts","../../src/shared/paths.ts","../../src/cli/bootstrap.ts","../../src/hooks/claude-md.ts","../../src/graph/rank.ts","../../src/graph/retrieve.ts","../../src/memory/branches.ts","../../src/memory/context-md.ts","../../src/memory/context-store.ts","../../src/memory/index.ts","../../src/packer/format.ts","../../src/packer/inline.ts","../../src/packer/signatures.ts","../../src/packer/tests.ts","../../src/packer/index.ts","../../src/server/mcp.ts","../../src/server/port.ts","../../src/server/routes/activity.ts","../../src/server/routes/context-update.ts","../../src/server/routes/gate.ts","../../src/server/routes/log.ts","../../src/server/routes/pack.ts","../../src/server/routes/prime.ts"],"sourcesContent":["// HTTP server (Hono). Hosts the routes hooks need (/prime, /pack, /log,\n// /gate, /activity) and serves the loaded graph from memory. The MCP-protocol\n// envelope (/mcp endpoint, JSON-RPC) is wired in M3.\n\nimport { serve } from \"@hono/node-server\";\nimport { Hono } from \"hono\";\nimport { writeFile } from \"node:fs/promises\";\n\nimport { ActivityStore } from \"../activity/activity-log.js\";\nimport { createFileWatcher, type FileWatcher } from \"../activity/file-watcher.js\";\nimport { createGitWatcher, type GitWatcher } from \"../activity/git-watcher.js\";\nimport { scanProject } from \"../cli/scan-command.js\";\nimport { readGraph, readSymbolIndex } from \"../graph/store.js\";\nimport { log } from \"../shared/logger.js\";\nimport type { SynthraPaths } from \"../shared/paths.js\";\nimport type { ServerContext } from \"./context.js\";\nimport { handleMcpRequest } from \"./mcp.js\";\nimport { findFreePort } from \"./port.js\";\nimport { handleActivity } from \"./routes/activity.js\";\nimport { handleContextUpdate } from \"./routes/context-update.js\";\nimport { handleGate } from \"./routes/gate.js\";\nimport { handleLog } from \"./routes/log.js\";\nimport { handlePack } from \"./routes/pack.js\";\nimport { handlePrime } from \"./routes/prime.js\";\n\nexport interface ServerHandle {\n port: number;\n url: string;\n stop(): Promise<void>;\n}\n\nexport interface StartOptions {\n /** Override the port range search. */\n port?: number;\n}\n\nasync function loadContext(paths: SynthraPaths): Promise<ServerContext> {\n try {\n const [graph, symbolIndex] = await Promise.all([\n readGraph(paths.infoGraph),\n readSymbolIndex(paths.symbolIndex),\n ]);\n const activity = new ActivityStore(paths.activityLog);\n return { paths, graph, symbolIndex, activity };\n } catch (err) {\n throw new Error(\n `failed to load graph from ${paths.infoGraph}: ${(err as Error).message}. ` +\n `Run \\`syn scan\\` first.`,\n );\n }\n}\n\nfunction buildApp(ctx: ServerContext, port: number): Hono {\n const app = new Hono();\n\n app.get(\"/\", (c) =>\n c.json({\n service: \"synthra\",\n version: \"0.0.1\",\n port,\n file_count: ctx.graph.file_count,\n symbol_count: ctx.graph.symbol_count,\n generated_at: ctx.graph.generated_at,\n }),\n );\n\n app.get(\"/health\", (c) => c.json({ ok: true }));\n\n app.get(\"/prime\", async (c) => c.json(await handlePrime(ctx, port)));\n\n app.post(\"/pack\", async (c) => {\n const body = await c.req.json().catch(() => ({}));\n return c.json(await handlePack(body, ctx));\n });\n\n app.post(\"/log\", async (c) => {\n const body = await c.req.json().catch(() => ({}));\n return c.json(await handleLog(body, ctx));\n });\n\n app.post(\"/gate\", async (c) => {\n const body = await c.req.json().catch(() => ({}));\n return c.json(await handleGate(body, ctx));\n });\n\n app.get(\"/activity\", async (c) => {\n const sinceParam = c.req.query(\"since\");\n const sinceMs = sinceParam ? Number(sinceParam) : undefined;\n return c.json(\n await handleActivity(Number.isFinite(sinceMs) ? sinceMs : undefined, ctx),\n );\n });\n\n app.post(\"/context-update\", async (c) => {\n const body = await c.req.json().catch(() => ({}));\n return c.json(await handleContextUpdate(body, ctx));\n });\n\n app.post(\"/mcp\", async (c) => {\n const body = await c.req.json().catch(() => null);\n return c.json(await handleMcpRequest(body, ctx));\n });\n\n app.onError((err, c) => {\n log.error(\"route error:\", err.message);\n return c.json({ error: err.message }, 400);\n });\n\n return app;\n}\n\nexport async function startServer(\n paths: SynthraPaths,\n options: StartOptions = {},\n): Promise<ServerHandle> {\n const ctx = await loadContext(paths);\n const port = options.port ?? (await findFreePort());\n\n const app = buildApp(ctx, port);\n const nodeServer = serve({ fetch: app.fetch, port, hostname: \"127.0.0.1\" });\n\n await writeFile(paths.mcpPort, String(port), \"utf8\");\n\n // Spin up the human-activity watchers. Both are best-effort — if chokidar\n // can't watch (e.g. unsupported FS) or .git is missing, they no-op silently.\n const fileWatcher: FileWatcher = createFileWatcher(paths.projectRoot, (e) =>\n ctx.activity.add(e),\n );\n const gitWatcher: GitWatcher = createGitWatcher(paths.projectRoot, async (e) => {\n await ctx.activity.add(e);\n // Per-branch graph: rebuild on branch switch so the in-memory graph\n // matches whichever branch is currently checked out.\n if (e.kind === \"branch-switch\") {\n try {\n const to = (e.details as { to?: string } | undefined)?.to ?? \"unknown\";\n log.info(`branch switched to '${to}' — rebuilding graph…`);\n await scanProject(paths.projectRoot, { silent: true });\n const [g, idx] = await Promise.all([\n readGraph(paths.infoGraph),\n readSymbolIndex(paths.symbolIndex),\n ]);\n ctx.graph = g;\n ctx.symbolIndex = idx;\n log.info(`graph rebuilt for '${to}' (${g.symbol_count} symbols).`);\n } catch (err) {\n log.warn(`branch rescan failed: ${(err as Error).message}`);\n }\n }\n });\n try {\n await fileWatcher.start();\n } catch (err) {\n log.warn(`file watcher failed to start: ${(err as Error).message}`);\n }\n try {\n await gitWatcher.start();\n } catch (err) {\n log.warn(`git watcher failed to start: ${(err as Error).message}`);\n }\n\n const url = `http://127.0.0.1:${port}`;\n\n return {\n port,\n url,\n async stop() {\n await fileWatcher.stop().catch(() => undefined);\n await gitWatcher.stop().catch(() => undefined);\n await new Promise<void>((resolve, reject) => {\n nodeServer.close((err) => (err ? reject(err) : resolve()));\n });\n },\n };\n}\n","// Rolling JSONL log of human activity, written to .synthra-graph/activity.jsonl.\n// In-memory ring buffer for fast queries; disk append for durability.\n//\n// The buffer is bounded (defaults to 100 events) so we don't unbounded-grow\n// memory in long sessions. Disk gets every event so the dashboard / future\n// audit tooling can replay history.\n\nimport { appendFile, mkdir } from \"node:fs/promises\";\nimport { dirname } from \"node:path\";\n\nexport interface FileEvent {\n kind: \"save\" | \"create\" | \"delete\";\n path: string;\n ts: string;\n}\n\nexport interface GitEvent {\n kind: \"branch-switch\" | \"stage\" | \"unstage\" | \"diff-change\";\n details: Record<string, unknown>;\n ts: string;\n}\n\nexport type ActivityEvent = FileEvent | GitEvent;\n\nconst DEFAULT_RING_SIZE = 100;\n\nexport class ActivityStore {\n private ring: ActivityEvent[] = [];\n private readonly maxRingSize: number;\n private readonly persistPath: string;\n\n constructor(persistPath: string, maxRingSize = DEFAULT_RING_SIZE) {\n this.persistPath = persistPath;\n this.maxRingSize = maxRingSize;\n }\n\n async add(event: ActivityEvent): Promise<void> {\n this.ring.push(event);\n while (this.ring.length > this.maxRingSize) this.ring.shift();\n await this.persist(event);\n }\n\n /** Get events newer than `sinceMs` (epoch ms). If omitted, returns the full ring. */\n getEvents(sinceMs?: number): ActivityEvent[] {\n if (!sinceMs || !Number.isFinite(sinceMs)) return this.ring.slice();\n const cutoff = new Date(sinceMs).toISOString();\n return this.ring.filter((e) => e.ts >= cutoff);\n }\n\n /** Project-relative file paths that have a save/create event newer than `maxAgeMs` ms ago. */\n recentFilePaths(maxAgeMs: number): string[] {\n const cutoff = new Date(Date.now() - maxAgeMs).toISOString();\n const out = new Set<string>();\n for (const e of this.ring) {\n if (\"path\" in e && (e.kind === \"save\" || e.kind === \"create\") && e.ts >= cutoff) {\n out.add(e.path);\n }\n }\n return Array.from(out);\n }\n\n size(): number {\n return this.ring.length;\n }\n\n private async persist(event: ActivityEvent): Promise<void> {\n try {\n await mkdir(dirname(this.persistPath), { recursive: true });\n await appendFile(this.persistPath, JSON.stringify(event) + \"\\n\", \"utf8\");\n } catch {\n // Durability is best-effort; an unwritable disk shouldn't crash the server.\n }\n }\n}\n","// chokidar-based file watcher. Emits save/create/delete events for human\n// edits inside the project. Respects .gitignore + .synthraignore plus a\n// hard-coded list of always-ignored directories (.git, .synthra*, .claude,\n// node_modules, dist, build, coverage).\n\nimport chokidar, { type FSWatcher } from \"chokidar\";\nimport { readFile } from \"node:fs/promises\";\nimport { join, relative, sep } from \"node:path\";\nimport ignore, { type Ignore } from \"ignore\";\n\nimport { log } from \"../shared/logger.js\";\nimport type { FileEvent } from \"./activity-log.js\";\n\nconst ALWAYS_IGNORE = [\n \".git\",\n \".synthra\",\n \".synthra-graph\",\n \".claude\",\n \"node_modules\",\n \"dist\",\n \"build\",\n \"out\",\n \"coverage\",\n \".next\",\n \".nuxt\",\n \".svelte-kit\",\n \".turbo\",\n \".cache\",\n \".vscode\",\n \".idea\",\n];\n\nexport interface FileWatcher {\n start(): Promise<void>;\n stop(): Promise<void>;\n}\n\nexport type FileEventHandler = (e: FileEvent) => void | Promise<void>;\n\nasync function readIgnoreFile(path: string): Promise<string[]> {\n try {\n const text = await readFile(path, \"utf8\");\n return text\n .split(/\\r?\\n/)\n .map((l) => l.trim())\n .filter((l) => l.length > 0 && !l.startsWith(\"#\"));\n } catch {\n return [];\n }\n}\n\nasync function buildMatcher(root: string): Promise<Ignore> {\n const ig = ignore();\n ig.add(ALWAYS_IGNORE.map((d) => `${d}/`));\n ig.add(await readIgnoreFile(join(root, \".gitignore\")));\n ig.add(await readIgnoreFile(join(root, \".synthraignore\")));\n return ig;\n}\n\nfunction toPosixRel(root: string, abs: string): string {\n const rel = relative(root, abs);\n return sep === \"/\" ? rel : rel.split(sep).join(\"/\");\n}\n\nexport function createFileWatcher(root: string, onEvent: FileEventHandler): FileWatcher {\n let watcher: FSWatcher | null = null;\n let ig: Ignore | null = null;\n\n const emit = async (kind: FileEvent[\"kind\"], abs: string) => {\n if (!ig) return;\n const rel = toPosixRel(root, abs);\n if (!rel || rel.startsWith(\"..\")) return;\n if (ig.ignores(rel)) return;\n try {\n await onEvent({ kind, path: rel, ts: new Date().toISOString() });\n } catch {\n // swallow handler errors — watcher must keep going\n }\n };\n\n return {\n async start() {\n ig = await buildMatcher(root);\n watcher = chokidar.watch(root, {\n // Cross-platform glob ignore. We match both the directory itself and\n // anything inside it. picomatch (chokidar's matcher) normalizes path\n // separators so a single set of forward-slash globs handles\n // Windows + POSIX. Function-based ignore was unreliable on Windows\n // and let chokidar descend into .git/, which crashed on transient\n // index.lock files held exclusively by git.\n ignored: ALWAYS_IGNORE.flatMap((d) => [`**/${d}`, `**/${d}/**`]),\n ignoreInitial: true,\n persistent: true,\n awaitWriteFinish: { stabilityThreshold: 150, pollInterval: 50 },\n });\n\n // Chokidar emits \"error\" for transient OS-level issues — most commonly\n // EPERM/ENOENT on rapidly created+deleted files. We never want one of\n // these to crash the syn process. Log + swallow.\n watcher.on(\"error\", (err) => {\n const e = err as NodeJS.ErrnoException;\n log.debug(`file watcher error (swallowed): ${e?.code ?? \"\"} ${e?.message ?? String(err)}`);\n });\n\n watcher.on(\"add\", (path) => emit(\"create\", path));\n watcher.on(\"change\", (path) => emit(\"save\", path));\n watcher.on(\"unlink\", (path) => emit(\"delete\", path));\n },\n\n async stop() {\n if (watcher) {\n await watcher.close();\n watcher = null;\n }\n },\n };\n}\n","// Minimal logger. Prefixes Synthra output with [syn].\n\ntype Level = \"debug\" | \"info\" | \"warn\" | \"error\";\n\nconst LEVEL_PRIORITY: Record<Level, number> = {\n debug: 10,\n info: 20,\n warn: 30,\n error: 40,\n};\n\nlet activeLevel: Level = (process.env.SYN_LOG_LEVEL as Level) ?? \"info\";\n\nexport function setLevel(level: Level): void {\n activeLevel = level;\n}\n\nfunction shouldLog(level: Level): boolean {\n return LEVEL_PRIORITY[level] >= LEVEL_PRIORITY[activeLevel];\n}\n\nfunction emit(level: Level, msg: string, ...args: unknown[]): void {\n if (!shouldLog(level)) return;\n const stream = level === \"error\" || level === \"warn\" ? process.stderr : process.stdout;\n stream.write(`[syn] ${msg}${args.length ? \" \" + args.map(String).join(\" \") : \"\"}\\n`);\n}\n\nexport const log = {\n debug: (m: string, ...a: unknown[]) => emit(\"debug\", m, ...a),\n info: (m: string, ...a: unknown[]) => emit(\"info\", m, ...a),\n warn: (m: string, ...a: unknown[]) => emit(\"warn\", m, ...a),\n error: (m: string, ...a: unknown[]) => emit(\"error\", m, ...a),\n};\n","// Watches `.git/HEAD` for branch switches (via fs.watch) and polls\n// `git status --porcelain` every ~2s to surface uncommitted-diff changes.\n// Always best-effort: in a non-git directory or when git is missing, the\n// watcher simply emits nothing.\n\nimport { execFile } from \"node:child_process\";\nimport { watch, type FSWatcher } from \"node:fs\";\nimport { readFile } from \"node:fs/promises\";\nimport { join } from \"node:path\";\nimport { promisify } from \"node:util\";\n\nimport type { GitEvent } from \"./activity-log.js\";\n\nconst execFileAsync = promisify(execFile);\n\nconst POLL_MS = 2000;\n\nexport interface GitWatcher {\n start(): Promise<void>;\n stop(): Promise<void>;\n}\n\nexport type GitEventHandler = (e: GitEvent) => void | Promise<void>;\n\nasync function readHeadBranch(projectRoot: string): Promise<string | null> {\n try {\n const head = await readFile(join(projectRoot, \".git\", \"HEAD\"), \"utf8\");\n const m = head.trim().match(/^ref:\\s+refs\\/heads\\/(.+)$/);\n return m?.[1] ?? null;\n } catch {\n return null;\n }\n}\n\nasync function readStatusPorcelain(projectRoot: string): Promise<string | null> {\n try {\n const { stdout } = await execFileAsync(\"git\", [\"status\", \"--porcelain\"], {\n cwd: projectRoot,\n });\n return stdout;\n } catch {\n return null;\n }\n}\n\nexport function createGitWatcher(root: string, onEvent: GitEventHandler): GitWatcher {\n let headWatcher: FSWatcher | null = null;\n let pollTimer: NodeJS.Timeout | null = null;\n let lastBranch: string | null = null;\n let lastStatus: string | null = null;\n\n const emitSafe = async (event: GitEvent) => {\n try {\n await onEvent(event);\n } catch {\n // swallow\n }\n };\n\n const checkHead = async () => {\n const branch = await readHeadBranch(root);\n if (branch && branch !== lastBranch) {\n const prev = lastBranch;\n lastBranch = branch;\n if (prev !== null) {\n await emitSafe({\n kind: \"branch-switch\",\n details: { from: prev, to: branch },\n ts: new Date().toISOString(),\n });\n }\n }\n };\n\n const pollStatus = async () => {\n const status = await readStatusPorcelain(root);\n if (status === null) return;\n if (lastStatus !== null && status !== lastStatus) {\n const prevFiles = parseStatusFiles(lastStatus);\n const nowFiles = parseStatusFiles(status);\n const added = nowFiles.filter((f) => !prevFiles.includes(f));\n const removed = prevFiles.filter((f) => !nowFiles.includes(f));\n await emitSafe({\n kind: \"diff-change\",\n details: {\n changed_count: nowFiles.length,\n newly_dirty: added,\n newly_clean: removed,\n },\n ts: new Date().toISOString(),\n });\n }\n lastStatus = status;\n };\n\n return {\n async start() {\n // Seed initial branch + status so the first real change emits an event\n // rather than a stale \"from null\".\n lastBranch = await readHeadBranch(root);\n lastStatus = await readStatusPorcelain(root);\n\n try {\n headWatcher = watch(join(root, \".git\", \"HEAD\"), () => {\n void checkHead();\n });\n // fs.watch emits \"error\" for transient OS issues (EPERM on lock\n // files, ENOENT when refs get rewritten). Swallow them — we never\n // want a transient FS event to crash the syn process.\n headWatcher.on(\"error\", () => {\n // silent — branch-switch detection is best-effort\n });\n } catch {\n // .git/HEAD not present — silently no-op\n }\n\n pollTimer = setInterval(() => {\n void pollStatus();\n }, POLL_MS);\n pollTimer.unref?.();\n },\n\n async stop() {\n if (headWatcher) {\n headWatcher.close();\n headWatcher = null;\n }\n if (pollTimer) {\n clearInterval(pollTimer);\n pollTimer = null;\n }\n },\n };\n}\n\nfunction parseStatusFiles(porcelain: string): string[] {\n return porcelain\n .split(/\\r?\\n/)\n .map((l) => l.slice(3).trim())\n .filter((l) => l.length > 0);\n}\n","// `syn scan [path]` — bootstrap then walk + parse + write graph.\n// Also invoked by the default `syn .` flow (M3 will chain start-claude after).\n\nimport { resolve } from \"node:path\";\n\nimport { buildGraph, buildSymbolIndex } from \"../scanner/extract.js\";\nimport { parseFile, type ParsedFile } from \"../scanner/parser.js\";\nimport { walk, type WalkedFile } from \"../scanner/walker.js\";\nimport { writeGraph, writeSymbolIndex } from \"../graph/store.js\";\nimport { log } from \"../shared/logger.js\";\nimport { resolvePaths } from \"../shared/paths.js\";\nimport { bootstrap } from \"./bootstrap.js\";\n\nconst PARSABLE_EXTS = new Set([\n \".ts\", \".tsx\", \".cts\", \".mts\",\n \".js\", \".jsx\", \".cjs\", \".mjs\",\n \".py\", \".pyi\",\n \".svelte\",\n \".vue\",\n \".go\",\n \".rs\",\n \".java\",\n \".kt\", \".kts\",\n \".php\",\n \".rb\",\n \".c\", \".h\",\n \".cpp\", \".cc\", \".cxx\", \".hpp\", \".hh\", \".hxx\",\n \".dart\",\n \".cs\",\n]);\n\nexport interface ScanResult {\n walked: number;\n parsed: number;\n symbolCount: number;\n edgeCount: number;\n durationMs: number;\n}\n\nexport interface ScanOptions {\n /** Suppress per-step log output (used for branch-switch rescans). */\n silent?: boolean;\n}\n\n/**\n * Core scan pipeline — bootstrap + walk + parse + write graph. Importable\n * from anywhere (server, CLI, tests). `scanCommand` is just a logging wrapper\n * around this. Pass `silent: true` to skip the chatty progress output.\n */\nexport async function scanProject(projectRootRaw: string, opts: ScanOptions = {}): Promise<ScanResult> {\n const projectRoot = resolve(projectRootRaw);\n const paths = resolvePaths(projectRoot);\n const start = Date.now();\n const verbose = !opts.silent;\n\n if (verbose) log.info(`scanning ${projectRoot}`);\n\n const boot = await bootstrap(paths);\n if (verbose) {\n if (boot.graphCreated) log.info(\" created .synthra-graph/\");\n if (boot.contextCreated) log.info(\" created .synthra/\");\n if (boot.gitignoreUpdated) log.info(\" updated .gitignore\");\n if (boot.claudeMdCreated) {\n log.info(\" created CLAUDE.md — onboarding skeleton for the agent\");\n log.info(\" ↳ fill in Build / Conventions / Decisions (or run /init in Claude to auto-draft)\");\n } else if (boot.claudeMdUpdated) {\n log.info(\" updated CLAUDE.md\");\n }\n }\n\n const walked: WalkedFile[] = [];\n for await (const file of walk(projectRoot)) walked.push(file);\n if (verbose) log.info(` walked ${walked.length} files`);\n\n const parsable = walked.filter((f) => PARSABLE_EXTS.has(f.ext));\n const parsed: ParsedFile[] = [];\n let parseErrors = 0;\n for (const file of parsable) {\n try {\n parsed.push(await parseFile(file));\n } catch (err) {\n parseErrors += 1;\n if (verbose) log.debug(` parse failed: ${file.relPath} — ${(err as Error).message}`);\n }\n }\n if (verbose) {\n log.info(\n ` parsed ${parsed.length} files (${walked.length - parsable.length} skipped` +\n (parseErrors ? `, ${parseErrors} errored` : \"\") +\n \")\",\n );\n }\n\n const graph = await buildGraph(projectRoot, parsed);\n const symbolIndex = buildSymbolIndex(graph);\n\n await writeGraph(paths.infoGraph, graph);\n await writeSymbolIndex(paths.symbolIndex, symbolIndex);\n\n if (verbose) {\n log.info(\n ` wrote ${paths.infoGraph} — ${graph.symbol_count} symbols, ${graph.edge_count} edges`,\n );\n log.info(` wrote ${paths.symbolIndex} — ${Object.keys(symbolIndex).length} names`);\n }\n\n const durationMs = Date.now() - start;\n if (verbose) log.info(`done in ${(durationMs / 1000).toFixed(2)}s`);\n\n return {\n walked: walked.length,\n parsed: parsed.length,\n symbolCount: graph.symbol_count,\n edgeCount: graph.edge_count,\n durationMs,\n };\n}\n\n// Thin alias so the CLI command keeps its current name. Drop in v0.2 if we\n// settle on a single public function.\nexport async function scanCommand(rawPath: string): Promise<ScanResult> {\n return scanProject(rawPath);\n}\n","// Turns ParsedFile[] into a GraphSchema (nodes + edges).\n//\n// Edges produced in M1:\n// defines : file → symbol\n// imports : file → file (when the import target resolves inside the project)\n// tests : test-file → source-file (for foo.test.ts ↔ foo.ts)\n\nimport { dirname, join, posix } from \"node:path\";\n\nimport type { Edge, FileNode, GraphSchema, SymbolIndex, SymbolNode } from \"../graph/types.js\";\nimport { fileHash } from \"./hash.js\";\nimport { extractKeywords } from \"./keywords.js\";\nimport type { ParsedFile, ParsedSymbol } from \"./parser.js\";\n\nconst RESOLVE_EXTS = [\".ts\", \".tsx\", \".js\", \".jsx\", \".mjs\", \".cjs\", \".py\", \".svelte\", \".vue\", \".dart\"];\nconst INDEX_FILES = [\"index.ts\", \"index.tsx\", \"index.js\", \"index.jsx\", \"__init__.py\"];\n\nfunction fileId(relPath: string): string {\n return `file:${relPath}`;\n}\n\nfunction symbolId(relPath: string, sym: ParsedSymbol): string {\n return `symbol:${relPath}::${sym.name}:${sym.startLine}`;\n}\n\nfunction toFileNode(parsed: ParsedFile): FileNode {\n const content = parsed.source;\n return {\n id: fileId(parsed.file.relPath),\n kind: \"file\",\n path: parsed.file.relPath,\n ext: parsed.file.ext,\n size: parsed.file.size,\n keywords: extractKeywords(content, parsed.file.ext),\n content,\n summary: extractSummary(content),\n file_hash: fileHash(content),\n };\n}\n\nfunction extractSummary(content: string): string {\n // First leading comment block (// ... or /** ... */ or # ...), trimmed to ~200 chars.\n const trimmed = content.replace(/^\\s+/, \"\");\n const slashMatch = trimmed.match(/^\\/\\/\\s?(.*(?:\\r?\\n\\/\\/\\s?.*)*)/);\n if (slashMatch?.[1]) return slashMatch[1].split(/\\r?\\n/).join(\" \").trim().slice(0, 200);\n const blockMatch = trimmed.match(/^\\/\\*\\*?([\\s\\S]*?)\\*\\//);\n if (blockMatch?.[1]) {\n return blockMatch[1]\n .split(/\\r?\\n/)\n .map((l) => l.replace(/^\\s*\\*\\s?/, \"\"))\n .join(\" \")\n .trim()\n .slice(0, 200);\n }\n const hashMatch = trimmed.match(/^#\\s?(.*(?:\\r?\\n#\\s?.*)*)/);\n if (hashMatch?.[1]) return hashMatch[1].split(/\\r?\\n/).join(\" \").trim().slice(0, 200);\n return \"\";\n}\n\nfunction toSymbolNode(parsed: ParsedFile, sym: ParsedSymbol): SymbolNode {\n return {\n id: symbolId(parsed.file.relPath, sym),\n kind: \"symbol\",\n symbol_kind: sym.kind,\n name: sym.name,\n file: parsed.file.relPath,\n start_line: sym.startLine,\n end_line: sym.endLine,\n signature: sym.signature,\n };\n}\n\n/**\n * Resolve an import specifier to a project-relative path if it refers to a\n * file inside the project. Returns `null` for external packages (no leading\n * dot) or specifiers that don't match any known file.\n */\n// Strip a trailing JS-family extension so a spec like \"./crypto.js\" can\n// resolve to \"crypto.ts\". TypeScript-style `.js` imports are common.\nconst REWRITE_EXT_RE = /\\.(js|jsx|mjs|cjs)$/;\n\nfunction resolveImport(\n fromRelPath: string,\n spec: string,\n filesByPath: Map<string, true>,\n): string | null {\n if (!spec.startsWith(\".\")) return null;\n const fromDir = posix.dirname(toPosix(fromRelPath));\n const base = posix.normalize(posix.join(fromDir, toPosix(spec)));\n\n const candidates = [base];\n const rewritten = base.replace(REWRITE_EXT_RE, \"\");\n if (rewritten !== base) candidates.push(rewritten);\n\n for (const c of candidates) {\n if (filesByPath.has(c)) return c;\n for (const ext of RESOLVE_EXTS) {\n if (filesByPath.has(c + ext)) return c + ext;\n }\n for (const idx of INDEX_FILES) {\n const candidate = posix.join(c, idx);\n if (filesByPath.has(candidate)) return candidate;\n }\n }\n return null;\n}\n\nfunction toPosix(p: string): string {\n return p.split(/[\\\\/]/).join(\"/\");\n}\n\nconst TEST_RE = /^(?<base>.+?)\\.(test|spec)\\.(?<ext>[tj]sx?|py)$/;\n\nfunction testTarget(relPath: string, filesByPath: Map<string, true>): string | null {\n const fileName = relPath.split(\"/\").pop() ?? relPath;\n const match = TEST_RE.exec(fileName);\n if (!match) return null;\n const dir = relPath.includes(\"/\") ? relPath.slice(0, relPath.lastIndexOf(\"/\") + 1) : \"\";\n const base = match.groups?.base ?? \"\";\n const ext = match.groups?.ext ?? \"\";\n if (!base || !ext) return null;\n const candidate = `${dir}${base}.${ext}`;\n if (filesByPath.has(candidate)) return candidate;\n // Try sibling extensions (e.g. foo.test.ts → foo.tsx)\n for (const e of RESOLVE_EXTS) {\n const alt = `${dir}${base}${e}`;\n if (filesByPath.has(alt)) return alt;\n }\n return null;\n}\n\nexport async function buildGraph(root: string, parsed: ParsedFile[]): Promise<GraphSchema> {\n const filesByPath = new Map<string, true>();\n for (const p of parsed) filesByPath.set(p.file.relPath, true);\n\n const nodes: (FileNode | SymbolNode)[] = [];\n const edges: Edge[] = [];\n\n for (const p of parsed) {\n const fileNode = toFileNode(p);\n nodes.push(fileNode);\n\n for (const sym of p.symbols) {\n const symNode = toSymbolNode(p, sym);\n nodes.push(symNode);\n edges.push({ from: fileNode.id, to: symNode.id, kind: \"defines\" });\n }\n\n const importEdges = new Set<string>();\n for (const spec of p.imports) {\n const target = resolveImport(p.file.relPath, spec, filesByPath);\n if (!target) continue;\n const key = `${fileNode.id}->${fileId(target)}`;\n if (importEdges.has(key)) continue;\n importEdges.add(key);\n edges.push({ from: fileNode.id, to: fileId(target), kind: \"imports\" });\n }\n\n const testTargetPath = testTarget(p.file.relPath, filesByPath);\n if (testTargetPath && testTargetPath !== p.file.relPath) {\n edges.push({ from: fileNode.id, to: fileId(testTargetPath), kind: \"tests\" });\n }\n }\n\n const symbolCount = nodes.filter((n) => n.kind === \"symbol\").length;\n const fileCount = nodes.length - symbolCount;\n\n return {\n root,\n node_count: nodes.length,\n edge_count: edges.length,\n file_count: fileCount,\n symbol_count: symbolCount,\n nodes,\n edges,\n generated_at: new Date().toISOString(),\n schema_version: 1,\n };\n}\n\nexport function buildSymbolIndex(graph: GraphSchema): SymbolIndex {\n // Null-prototype map: symbol names like \"toString\" or \"constructor\" (common\n // in Dart, where every class overrides toString) would otherwise resolve to\n // an inherited Object.prototype member and crash on the .push below.\n const out: SymbolIndex = Object.create(null);\n for (const node of graph.nodes) {\n if (node.kind !== \"symbol\") continue;\n const list = out[node.name] ?? (out[node.name] = []);\n list.push({ file: node.file, line: node.start_line, kind: node.symbol_kind });\n }\n return out;\n}\n\n// Re-export node path helpers in case downstream wants the canonical id format\nexport { fileId, symbolId };\n// Suppress unused-import lint for dirname/join from node:path — kept reserved for incremental updates.\nvoid dirname;\nvoid join;\n","// Stable, short content hash for files. Used to detect changed files\n// during incremental rescans (post-v0.1; M1 does full re-parse).\n// TODO: M1 (minimal); post-v0.1 (incremental)\n\nimport { createHash } from \"node:crypto\";\n\nexport function fileHash(content: string): string {\n return createHash(\"sha1\").update(content).digest(\"hex\").slice(0, 8);\n}\n","// Per-file keyword extraction. Used for query-time relevance ranking.\n// Tokenizes identifiers + comment words, splits camelCase/snake_case, filters\n// stopwords, and returns the top-N rare tokens scored by inverse frequency\n// against a small built-in english/code corpus.\n\nconst STOPWORDS = new Set([\n \"a\", \"an\", \"and\", \"are\", \"as\", \"at\", \"be\", \"but\", \"by\", \"do\", \"for\", \"from\",\n \"has\", \"have\", \"he\", \"if\", \"in\", \"is\", \"it\", \"its\", \"not\", \"of\", \"on\", \"or\",\n \"she\", \"that\", \"the\", \"they\", \"this\", \"to\", \"was\", \"we\", \"were\", \"will\", \"with\",\n \"you\", \"your\", \"i\", \"me\", \"my\", \"our\", \"us\", \"their\", \"them\", \"his\", \"her\",\n // common code words that add no signal\n \"function\", \"const\", \"let\", \"var\", \"class\", \"interface\", \"type\", \"enum\",\n \"import\", \"export\", \"from\", \"default\", \"return\", \"if\", \"else\", \"for\", \"while\",\n \"do\", \"switch\", \"case\", \"break\", \"continue\", \"new\", \"this\", \"super\", \"throw\",\n \"try\", \"catch\", \"finally\", \"async\", \"await\", \"yield\", \"true\", \"false\", \"null\",\n \"undefined\", \"void\", \"any\", \"string\", \"number\", \"boolean\", \"object\", \"array\",\n \"self\", \"cls\", \"def\", \"lambda\", \"pass\", \"raise\", \"with\", \"as\", \"in\",\n \"todo\", \"fixme\", \"note\",\n]);\n\nconst COMMON_CODE = new Set([\n \"value\", \"data\", \"result\", \"args\", \"kwargs\", \"options\", \"config\", \"params\",\n \"name\", \"id\", \"key\", \"index\", \"item\", \"items\", \"list\", \"map\", \"set\", \"get\",\n \"set\", \"add\", \"remove\", \"delete\", \"create\", \"update\", \"find\", \"fetch\", \"load\",\n \"save\", \"init\", \"main\", \"run\", \"start\", \"stop\", \"test\", \"check\", \"validate\",\n \"error\", \"err\", \"warn\", \"info\", \"debug\", \"log\", \"trace\", \"msg\", \"message\",\n \"path\", \"file\", \"dir\", \"url\", \"host\", \"port\", \"size\", \"length\", \"count\",\n \"input\", \"output\", \"source\", \"target\", \"callback\", \"handler\", \"listener\",\n \"props\", \"state\", \"context\", \"render\", \"component\", \"node\", \"tree\", \"root\",\n]);\n\n// Frequency weight — common-code words count for less than rare identifiers\nfunction score(token: string): number {\n if (STOPWORDS.has(token)) return 0;\n if (COMMON_CODE.has(token)) return 0.2;\n if (token.length <= 2) return 0.1;\n return 1;\n}\n\nfunction splitIdentifier(id: string): string[] {\n // snake_case + kebab-case → words\n const partsRaw = id.split(/[_\\-./]+/).filter(Boolean);\n const out: string[] = [];\n for (const part of partsRaw) {\n // camelCase / PascalCase → words. Handles \"XMLHttp\" → [\"XML\", \"Http\"]\n const camelParts = part.match(/[A-Z]+(?=[A-Z][a-z])|[A-Z]?[a-z]+|[A-Z]+|[0-9]+/g);\n if (camelParts) out.push(...camelParts);\n else out.push(part);\n }\n return out.map((w) => w.toLowerCase()).filter((w) => /[a-z]/.test(w));\n}\n\nexport function extractKeywords(content: string, _ext: string): string[] {\n // Identifiers + alphanumeric words. Picks up both code and comment text.\n const tokens = content.match(/[A-Za-z_][A-Za-z0-9_]{1,40}/g) ?? [];\n const counts = new Map<string, number>();\n for (const tok of tokens) {\n for (const word of splitIdentifier(tok)) {\n const w = score(word);\n if (w === 0) continue;\n counts.set(word, (counts.get(word) ?? 0) + w);\n }\n }\n return Array.from(counts.entries())\n .sort((a, b) => b[1] - a[1])\n .slice(0, 32)\n .map(([w]) => w);\n}\n","// Dispatches a file to its language-specific parser based on extension.\n// Tree-sitter WASM grammars are loaded lazily via tree-sitter-wasms and\n// cached per language for the lifetime of the process.\n\nimport { readFile } from \"node:fs/promises\";\nimport { createRequire } from \"node:module\";\nimport { Language, Parser } from \"web-tree-sitter\";\n\nimport type { SymbolKind } from \"../graph/types.js\";\nimport { parseC } from \"./parsers/c.js\";\nimport { parseCpp } from \"./parsers/cpp.js\";\nimport { parseCSharp } from \"./parsers/csharp.js\";\nimport { parseDart } from \"./parsers/dart.js\";\nimport { parseGo } from \"./parsers/go.js\";\nimport { parseJava } from \"./parsers/java.js\";\nimport { parseKotlin } from \"./parsers/kotlin.js\";\nimport { parsePhp } from \"./parsers/php.js\";\nimport { parsePython } from \"./parsers/python.js\";\nimport { parseRuby } from \"./parsers/ruby.js\";\nimport { parseRust } from \"./parsers/rust.js\";\nimport { parseSvelte } from \"./parsers/svelte.js\";\nimport { parseTypeScript } from \"./parsers/typescript.js\";\nimport { parseVue } from \"./parsers/vue.js\";\nimport type { WalkedFile } from \"./walker.js\";\n\nexport interface ParsedSymbol {\n name: string;\n kind: SymbolKind;\n startLine: number;\n endLine: number;\n signature: string;\n}\n\nexport interface ParsedFile {\n file: WalkedFile;\n source: string;\n symbols: ParsedSymbol[];\n imports: string[];\n calls: Array<{ from: string; to: string }>;\n}\n\nconst require = createRequire(import.meta.url);\n\nexport type GrammarName =\n | \"typescript\"\n | \"tsx\"\n | \"javascript\"\n | \"python\"\n | \"go\"\n | \"rust\"\n | \"java\"\n | \"kotlin\"\n | \"php\"\n | \"ruby\"\n | \"c\"\n | \"cpp\"\n | \"dart\"\n | \"csharp\";\n\nconst GRAMMAR_FILES: Record<GrammarName, string> = {\n typescript: \"tree-sitter-wasms/out/tree-sitter-typescript.wasm\",\n tsx: \"tree-sitter-wasms/out/tree-sitter-tsx.wasm\",\n javascript: \"tree-sitter-wasms/out/tree-sitter-javascript.wasm\",\n python: \"tree-sitter-wasms/out/tree-sitter-python.wasm\",\n go: \"tree-sitter-wasms/out/tree-sitter-go.wasm\",\n rust: \"tree-sitter-wasms/out/tree-sitter-rust.wasm\",\n java: \"tree-sitter-wasms/out/tree-sitter-java.wasm\",\n kotlin: \"tree-sitter-wasms/out/tree-sitter-kotlin.wasm\",\n php: \"tree-sitter-wasms/out/tree-sitter-php.wasm\",\n ruby: \"tree-sitter-wasms/out/tree-sitter-ruby.wasm\",\n c: \"tree-sitter-wasms/out/tree-sitter-c.wasm\",\n cpp: \"tree-sitter-wasms/out/tree-sitter-cpp.wasm\",\n dart: \"tree-sitter-wasms/out/tree-sitter-dart.wasm\",\n csharp: \"tree-sitter-wasms/out/tree-sitter-c_sharp.wasm\",\n};\n\nlet parserInit: Promise<void> | null = null;\nconst languageCache = new Map<GrammarName, Language>();\n\nasync function ensureParserInit(): Promise<void> {\n if (!parserInit) {\n parserInit = Parser.init();\n }\n return parserInit;\n}\n\nexport async function loadGrammar(name: GrammarName): Promise<Language> {\n await ensureParserInit();\n const cached = languageCache.get(name);\n if (cached) return cached;\n const wasmPath = require.resolve(GRAMMAR_FILES[name]);\n const lang = await Language.load(wasmPath);\n languageCache.set(name, lang);\n return lang;\n}\n\nexport interface LoadedParser {\n parser: Parser;\n language: Language;\n}\n\nexport async function createParser(name: GrammarName): Promise<LoadedParser> {\n const language = await loadGrammar(name);\n const parser = new Parser();\n parser.setLanguage(language);\n return { parser, language };\n}\n\nfunction emptyParsed(file: WalkedFile, source: string): ParsedFile {\n return { file, source, symbols: [], imports: [], calls: [] };\n}\n\nexport async function parseFile(f: WalkedFile): Promise<ParsedFile> {\n let source: string;\n try {\n source = await readFile(f.absPath, \"utf8\");\n } catch {\n return emptyParsed(f, \"\");\n }\n\n switch (f.ext) {\n case \".ts\":\n case \".tsx\":\n case \".cts\":\n case \".mts\":\n case \".js\":\n case \".jsx\":\n case \".cjs\":\n case \".mjs\":\n return parseTypeScript(f, source);\n case \".py\":\n case \".pyi\":\n return parsePython(f, source);\n case \".svelte\":\n return parseSvelte(f, source);\n case \".vue\":\n return parseVue(f, source);\n case \".go\":\n return parseGo(f, source);\n case \".rs\":\n return parseRust(f, source);\n case \".java\":\n return parseJava(f, source);\n case \".kt\":\n case \".kts\":\n return parseKotlin(f, source);\n case \".php\":\n return parsePhp(f, source);\n case \".rb\":\n return parseRuby(f, source);\n case \".c\":\n case \".h\":\n return parseC(f, source);\n case \".cpp\":\n case \".cc\":\n case \".cxx\":\n case \".hpp\":\n case \".hh\":\n case \".hxx\":\n return parseCpp(f, source);\n case \".dart\":\n return parseDart(f, source);\n case \".cs\":\n return parseCSharp(f, source);\n default:\n return emptyParsed(f, source);\n }\n}\n","// Generic tree-sitter parser used by the simpler-grammar languages\n// (Go, Rust, Java, Kotlin, PHP, Ruby, C, C++, Dart, C#).\n//\n// Each language file defines:\n// - which tree-sitter grammar to load\n// - a query string with capture names like `@function`, `@function.name`\n// - a `decls` table mapping declaration-capture pairs to SymbolKind\n// - optional `importCapture` for collecting import edges\n// Everything else (parser init, error handling, dedupe) lives here.\n\nimport { Query, type Node } from \"web-tree-sitter\";\n\nimport type { SymbolKind } from \"../../graph/types.js\";\nimport { createParser, type GrammarName, type ParsedFile, type ParsedSymbol } from \"../parser.js\";\nimport type { WalkedFile } from \"../walker.js\";\n\nexport interface DeclCapture {\n /** Capture name for the declaration node, e.g. \"function\". */\n declCapture: string;\n /** Capture name for the symbol's name node, e.g. \"function.name\". */\n nameCapture: string;\n /** SymbolKind to assign. */\n kind: SymbolKind;\n}\n\nexport interface GenericParserConfig {\n grammar: GrammarName;\n query: string;\n decls: DeclCapture[];\n /** Capture name for import-source nodes. Skipped when omitted. */\n importCapture?: string;\n}\n\nexport function firstLine(text: string, max = 200): string {\n const line = text.split(/\\r?\\n/, 1)[0] ?? \"\";\n return line.length > max ? line.slice(0, max) + \"…\" : line;\n}\n\nfunction cleanImport(s: string): string {\n // Strip surrounding string-literal quotes (used by Go, Dart, C/C++).\n // Strip angle brackets used by C/C++ system includes.\n return s.replace(/^[\"'`<]+|[\"'`>]+$/g, \"\").trim();\n}\n\nexport async function runGenericParser(\n config: GenericParserConfig,\n f: WalkedFile,\n source: string,\n): Promise<ParsedFile> {\n let symbols: ParsedSymbol[] = [];\n let imports: string[] = [];\n\n try {\n const { parser, language } = await createParser(config.grammar);\n const tree = parser.parse(source);\n if (!tree) return { file: f, source, symbols, imports, calls: [] };\n\n const query = new Query(language, config.query);\n const matches = query.matches(tree.rootNode);\n\n for (const match of matches) {\n const byName = new Map<string, Node>();\n for (const cap of match.captures) byName.set(cap.name, cap.node);\n\n let matched: DeclCapture | null = null;\n for (const d of config.decls) {\n if (byName.has(d.declCapture) && byName.has(d.nameCapture)) {\n matched = d;\n break;\n }\n }\n\n if (matched) {\n const declNode = byName.get(matched.declCapture)!;\n const nameNode = byName.get(matched.nameCapture)!;\n symbols.push({\n name: nameNode.text,\n kind: matched.kind,\n startLine: declNode.startPosition.row + 1,\n endLine: declNode.endPosition.row + 1,\n signature: firstLine(declNode.text),\n });\n continue;\n }\n\n if (config.importCapture) {\n const imp = byName.get(config.importCapture);\n if (imp) imports.push(cleanImport(imp.text));\n }\n }\n\n const seen = new Set<string>();\n symbols = symbols.filter((s) => {\n const k = `${s.name}:${s.startLine}`;\n if (seen.has(k)) return false;\n seen.add(k);\n return true;\n });\n imports = Array.from(new Set(imports)).filter((s) => s.length > 0);\n } catch {\n // Query compile or parse failure — return what we have. Silent so a single\n // bad file doesn't abort the whole scan.\n }\n\n return { file: f, source, symbols, imports, calls: [] };\n}\n","// C parser. Function definitions, structs, enums, typedefs, #include directives.\n\nimport type { ParsedFile } from \"../parser.js\";\nimport type { WalkedFile } from \"../walker.js\";\nimport { runGenericParser } from \"./_generic.js\";\n\nconst QUERY = `\n(function_definition declarator: (function_declarator declarator: (identifier) @function.name)) @function\n(struct_specifier name: (type_identifier) @struct.name) @struct\n(enum_specifier name: (type_identifier) @enum.name) @enum\n(type_definition declarator: (type_identifier) @type.name) @type\n(preproc_include path: (string_literal) @import)\n(preproc_include path: (system_lib_string) @import)\n`;\n\nexport async function parseC(f: WalkedFile, source: string): Promise<ParsedFile> {\n return runGenericParser(\n {\n grammar: \"c\",\n query: QUERY,\n decls: [\n { declCapture: \"function\", nameCapture: \"function.name\", kind: \"function\" },\n { declCapture: \"struct\", nameCapture: \"struct.name\", kind: \"class\" },\n { declCapture: \"enum\", nameCapture: \"enum.name\", kind: \"enum\" },\n { declCapture: \"type\", nameCapture: \"type.name\", kind: \"type\" },\n ],\n importCapture: \"import\",\n },\n f,\n source,\n );\n}\n","// C++ parser. Functions, classes, structs, enums, namespaces, #includes.\n\nimport type { ParsedFile } from \"../parser.js\";\nimport type { WalkedFile } from \"../walker.js\";\nimport { runGenericParser } from \"./_generic.js\";\n\nconst QUERY = `\n(function_definition declarator: (function_declarator declarator: (identifier) @function.name)) @function\n(function_definition declarator: (function_declarator declarator: (qualified_identifier) @method.name)) @method\n(class_specifier name: (type_identifier) @class.name) @class\n(struct_specifier name: (type_identifier) @struct.name) @struct\n(enum_specifier name: (type_identifier) @enum.name) @enum\n(namespace_definition name: (namespace_identifier) @namespace.name) @namespace\n(preproc_include path: (string_literal) @import)\n(preproc_include path: (system_lib_string) @import)\n`;\n\nexport async function parseCpp(f: WalkedFile, source: string): Promise<ParsedFile> {\n return runGenericParser(\n {\n grammar: \"cpp\",\n query: QUERY,\n decls: [\n { declCapture: \"function\", nameCapture: \"function.name\", kind: \"function\" },\n { declCapture: \"method\", nameCapture: \"method.name\", kind: \"method\" },\n { declCapture: \"class\", nameCapture: \"class.name\", kind: \"class\" },\n { declCapture: \"struct\", nameCapture: \"struct.name\", kind: \"class\" },\n { declCapture: \"enum\", nameCapture: \"enum.name\", kind: \"enum\" },\n { declCapture: \"namespace\", nameCapture: \"namespace.name\", kind: \"class\" },\n ],\n importCapture: \"import\",\n },\n f,\n source,\n );\n}\n","// C# (.NET) parser. Classes, interfaces, structs, methods, namespaces.\n\nimport type { ParsedFile } from \"../parser.js\";\nimport type { WalkedFile } from \"../walker.js\";\nimport { runGenericParser } from \"./_generic.js\";\n\nconst QUERY = `\n(class_declaration name: (identifier) @class.name) @class\n(interface_declaration name: (identifier) @interface.name) @interface\n(struct_declaration name: (identifier) @struct.name) @struct\n(enum_declaration name: (identifier) @enum.name) @enum\n(method_declaration name: (identifier) @method.name) @method\n(namespace_declaration name: (_) @namespace.name) @namespace\n(using_directive (_) @import)\n`;\n\nexport async function parseCSharp(f: WalkedFile, source: string): Promise<ParsedFile> {\n return runGenericParser(\n {\n grammar: \"csharp\",\n query: QUERY,\n decls: [\n { declCapture: \"class\", nameCapture: \"class.name\", kind: \"class\" },\n { declCapture: \"interface\", nameCapture: \"interface.name\", kind: \"interface\" },\n { declCapture: \"struct\", nameCapture: \"struct.name\", kind: \"class\" },\n { declCapture: \"enum\", nameCapture: \"enum.name\", kind: \"enum\" },\n { declCapture: \"method\", nameCapture: \"method.name\", kind: \"method\" },\n { declCapture: \"namespace\", nameCapture: \"namespace.name\", kind: \"class\" },\n ],\n importCapture: \"import\",\n },\n f,\n source,\n );\n}\n","// Dart parser. v0.1.11 — real symbol extraction + import parsing on top of the\n// ABI-v15 grammar that ships in tree-sitter-wasms.\n//\n// Distinguishes top-level function_signature (kind: function) from\n// function_signature nested under method_signature (kind: method) by\n// anchoring the top-level pattern under `program`.\n//\n// Imports: `package:foo/bar.dart` and `dart:async` are stripped — they cross\n// the project boundary. Bare `foo.dart` is normalized to `./foo.dart` so the\n// shared resolveImport() (which requires a leading `.`) treats it as a\n// same-directory relative import.\n\nimport { Query, type Node } from \"web-tree-sitter\";\nimport type { SymbolKind } from \"../../graph/types.js\";\nimport { createParser, type ParsedFile, type ParsedSymbol } from \"../parser.js\";\nimport type { WalkedFile } from \"../walker.js\";\n\nconst QUERY = `\n(class_definition name: (identifier) @class.name) @class\n(mixin_declaration (identifier) @mixin.name) @mixin\n(extension_declaration name: (identifier) @ext.name) @ext\n(enum_declaration name: (identifier) @enum.name) @enum\n(type_alias (type_identifier) @typedef.name) @typedef\n\n(program (function_signature name: (identifier) @function.name) @function)\n\n(method_signature (function_signature name: (identifier) @method.name)) @method\n(method_signature (getter_signature name: (identifier) @getter.name)) @getter\n(method_signature (setter_signature name: (identifier) @setter.name)) @setter\n(constructor_signature name: (identifier) @ctor.name) @ctor\n\n(import_or_export (library_import (import_specification (configurable_uri (uri (string_literal) @import)))))\n`;\n\ninterface DeclShape {\n declCap: string;\n nameCap: string;\n kind: SymbolKind;\n}\n\nconst DECLS: DeclShape[] = [\n { declCap: \"class\", nameCap: \"class.name\", kind: \"class\" },\n { declCap: \"mixin\", nameCap: \"mixin.name\", kind: \"class\" },\n { declCap: \"ext\", nameCap: \"ext.name\", kind: \"class\" },\n { declCap: \"enum\", nameCap: \"enum.name\", kind: \"enum\" },\n { declCap: \"typedef\", nameCap: \"typedef.name\", kind: \"type\" },\n { declCap: \"function\", nameCap: \"function.name\", kind: \"function\" },\n { declCap: \"method\", nameCap: \"method.name\", kind: \"method\" },\n { declCap: \"getter\", nameCap: \"getter.name\", kind: \"method\" },\n { declCap: \"setter\", nameCap: \"setter.name\", kind: \"method\" },\n { declCap: \"ctor\", nameCap: \"ctor.name\", kind: \"method\" },\n];\n\nfunction firstLine(text: string, max = 200): string {\n const line = text.split(/\\r?\\n/, 1)[0] ?? \"\";\n return line.length > max ? line.slice(0, max) + \"…\" : line;\n}\n\n// Strip surrounding string-literal quotes and normalize bare same-directory\n// imports (Dart allows `import 'foo.dart';` without a leading `./`) so\n// resolveImport() — which keys off a leading dot — can match them.\nfunction normalizeDartImport(raw: string): string | null {\n const stripped = raw.replace(/^['\"]|['\"]$/g, \"\");\n if (!stripped) return null;\n if (stripped.startsWith(\"package:\")) return null;\n if (stripped.startsWith(\"dart:\")) return null;\n if (stripped.startsWith(\".\") || stripped.startsWith(\"/\")) return stripped;\n return `./${stripped}`;\n}\n\nexport async function parseDart(f: WalkedFile, source: string): Promise<ParsedFile> {\n let symbols: ParsedSymbol[] = [];\n let imports: string[] = [];\n\n try {\n const { parser, language } = await createParser(\"dart\");\n const tree = parser.parse(source);\n if (!tree) return { file: f, source, symbols, imports, calls: [] };\n\n const query = new Query(language, QUERY);\n const matches = query.matches(tree.rootNode);\n\n for (const match of matches) {\n const byName = new Map<string, Node>();\n for (const cap of match.captures) byName.set(cap.name, cap.node);\n\n let matched: DeclShape | null = null;\n for (const d of DECLS) {\n if (byName.has(d.declCap) && byName.has(d.nameCap)) {\n matched = d;\n break;\n }\n }\n\n if (matched) {\n const declNode = byName.get(matched.declCap)!;\n const nameNode = byName.get(matched.nameCap)!;\n symbols.push({\n name: nameNode.text,\n kind: matched.kind,\n startLine: declNode.startPosition.row + 1,\n endLine: declNode.endPosition.row + 1,\n signature: firstLine(declNode.text),\n });\n continue;\n }\n\n const importNode = byName.get(\"import\");\n if (importNode) {\n const norm = normalizeDartImport(importNode.text);\n if (norm) imports.push(norm);\n }\n }\n\n const seen = new Set<string>();\n symbols = symbols.filter((s) => {\n const k = `${s.name}:${s.startLine}`;\n if (seen.has(k)) return false;\n seen.add(k);\n return true;\n });\n imports = Array.from(new Set(imports));\n } catch {\n // swallow — see _generic.ts for the rationale (single bad file shouldn't\n // abort the whole scan).\n }\n\n return { file: f, source, symbols, imports, calls: [] };\n}\n","// Go parser. Functions, methods, type declarations, imports.\n\nimport type { ParsedFile } from \"../parser.js\";\nimport type { WalkedFile } from \"../walker.js\";\nimport { runGenericParser } from \"./_generic.js\";\n\nconst QUERY = `\n(function_declaration name: (identifier) @function.name) @function\n(method_declaration name: (field_identifier) @method.name) @method\n(type_spec name: (type_identifier) @type.name) @type\n(import_spec path: (interpreted_string_literal) @import)\n`;\n\nexport async function parseGo(f: WalkedFile, source: string): Promise<ParsedFile> {\n return runGenericParser(\n {\n grammar: \"go\",\n query: QUERY,\n decls: [\n { declCapture: \"function\", nameCapture: \"function.name\", kind: \"function\" },\n { declCapture: \"method\", nameCapture: \"method.name\", kind: \"method\" },\n { declCapture: \"type\", nameCapture: \"type.name\", kind: \"type\" },\n ],\n importCapture: \"import\",\n },\n f,\n source,\n );\n}\n","// Java parser. Classes, interfaces, methods, imports.\n\nimport type { ParsedFile } from \"../parser.js\";\nimport type { WalkedFile } from \"../walker.js\";\nimport { runGenericParser } from \"./_generic.js\";\n\nconst QUERY = `\n(class_declaration name: (identifier) @class.name) @class\n(interface_declaration name: (identifier) @interface.name) @interface\n(method_declaration name: (identifier) @method.name) @method\n(enum_declaration name: (identifier) @enum.name) @enum\n(import_declaration (scoped_identifier) @import)\n`;\n\nexport async function parseJava(f: WalkedFile, source: string): Promise<ParsedFile> {\n return runGenericParser(\n {\n grammar: \"java\",\n query: QUERY,\n decls: [\n { declCapture: \"class\", nameCapture: \"class.name\", kind: \"class\" },\n { declCapture: \"interface\", nameCapture: \"interface.name\", kind: \"interface\" },\n { declCapture: \"method\", nameCapture: \"method.name\", kind: \"method\" },\n { declCapture: \"enum\", nameCapture: \"enum.name\", kind: \"enum\" },\n ],\n importCapture: \"import\",\n },\n f,\n source,\n );\n}\n","// Kotlin parser. Functions, classes, objects, interfaces, imports.\n\nimport type { ParsedFile } from \"../parser.js\";\nimport type { WalkedFile } from \"../walker.js\";\nimport { runGenericParser } from \"./_generic.js\";\n\nconst QUERY = `\n(function_declaration (simple_identifier) @function.name) @function\n(class_declaration (type_identifier) @class.name) @class\n(object_declaration (type_identifier) @object.name) @object\n(import_header (identifier) @import)\n`;\n\nexport async function parseKotlin(f: WalkedFile, source: string): Promise<ParsedFile> {\n return runGenericParser(\n {\n grammar: \"kotlin\",\n query: QUERY,\n decls: [\n { declCapture: \"function\", nameCapture: \"function.name\", kind: \"function\" },\n { declCapture: \"class\", nameCapture: \"class.name\", kind: \"class\" },\n { declCapture: \"object\", nameCapture: \"object.name\", kind: \"class\" },\n ],\n importCapture: \"import\",\n },\n f,\n source,\n );\n}\n","// PHP parser. Functions, classes, interfaces, methods, traits.\n\nimport type { ParsedFile } from \"../parser.js\";\nimport type { WalkedFile } from \"../walker.js\";\nimport { runGenericParser } from \"./_generic.js\";\n\nconst QUERY = `\n(function_definition name: (name) @function.name) @function\n(class_declaration name: (name) @class.name) @class\n(interface_declaration name: (name) @interface.name) @interface\n(trait_declaration name: (name) @trait.name) @trait\n(method_declaration name: (name) @method.name) @method\n`;\n\nexport async function parsePhp(f: WalkedFile, source: string): Promise<ParsedFile> {\n return runGenericParser(\n {\n grammar: \"php\",\n query: QUERY,\n decls: [\n { declCapture: \"function\", nameCapture: \"function.name\", kind: \"function\" },\n { declCapture: \"class\", nameCapture: \"class.name\", kind: \"class\" },\n { declCapture: \"interface\", nameCapture: \"interface.name\", kind: \"interface\" },\n { declCapture: \"trait\", nameCapture: \"trait.name\", kind: \"class\" },\n { declCapture: \"method\", nameCapture: \"method.name\", kind: \"method\" },\n ],\n },\n f,\n source,\n );\n}\n","// Python parser using tree-sitter-python WASM.\n// Extracts: function/class definitions, methods, and import statements.\n\nimport { Query, type Node } from \"web-tree-sitter\";\nimport { createParser, type ParsedFile, type ParsedSymbol } from \"../parser.js\";\nimport type { WalkedFile } from \"../walker.js\";\n\nconst QUERY = `\n(function_definition name: (identifier) @function.name) @function\n(class_definition name: (identifier) @class.name) @class\n(import_statement name: (dotted_name) @import.module)\n(import_from_statement module_name: (dotted_name) @import.from)\n(import_from_statement module_name: (relative_import) @import.from)\n`;\n\nfunction firstLine(text: string, max = 200): string {\n const line = text.split(/\\r?\\n/, 1)[0] ?? \"\";\n return line.length > max ? line.slice(0, max) + \"…\" : line;\n}\n\nexport async function parsePython(f: WalkedFile, source: string): Promise<ParsedFile> {\n let symbols: ParsedSymbol[] = [];\n let imports: string[] = [];\n\n try {\n const { parser, language } = await createParser(\"python\");\n const tree = parser.parse(source);\n if (!tree) return { file: f, source, symbols, imports, calls: [] };\n\n const query = new Query(language, QUERY);\n const matches = query.matches(tree.rootNode);\n\n for (const match of matches) {\n const byName = new Map<string, Node>();\n for (const cap of match.captures) byName.set(cap.name, cap.node);\n\n const funcDecl = byName.get(\"function\");\n const funcName = byName.get(\"function.name\");\n if (funcDecl && funcName) {\n const parentType = funcDecl.parent?.parent?.type;\n const isMethod = parentType === \"class_definition\";\n symbols.push({\n name: funcName.text,\n kind: isMethod ? \"method\" : \"function\",\n startLine: funcDecl.startPosition.row + 1,\n endLine: funcDecl.endPosition.row + 1,\n signature: firstLine(funcDecl.text),\n });\n continue;\n }\n\n const classDecl = byName.get(\"class\");\n const className = byName.get(\"class.name\");\n if (classDecl && className) {\n symbols.push({\n name: className.text,\n kind: \"class\",\n startLine: classDecl.startPosition.row + 1,\n endLine: classDecl.endPosition.row + 1,\n signature: firstLine(classDecl.text),\n });\n continue;\n }\n\n const importNode = byName.get(\"import.module\") ?? byName.get(\"import.from\");\n if (importNode) imports.push(importNode.text);\n }\n\n const seen = new Set<string>();\n symbols = symbols.filter((s) => {\n const key = `${s.name}:${s.startLine}`;\n if (seen.has(key)) return false;\n seen.add(key);\n return true;\n });\n imports = Array.from(new Set(imports));\n } catch {\n // swallow parse errors\n }\n\n return { file: f, source, symbols, imports, calls: [] };\n}\n","// Ruby parser. Methods, classes, modules.\n// Imports omitted — Ruby's `require` is dynamic and hard to capture cleanly;\n// keyword indexing still surfaces dependencies.\n\nimport type { ParsedFile } from \"../parser.js\";\nimport type { WalkedFile } from \"../walker.js\";\nimport { runGenericParser } from \"./_generic.js\";\n\nconst QUERY = `\n(method name: (identifier) @function.name) @function\n(singleton_method name: (identifier) @method.name) @method\n(class name: (constant) @class.name) @class\n(module name: (constant) @module.name) @module\n`;\n\nexport async function parseRuby(f: WalkedFile, source: string): Promise<ParsedFile> {\n return runGenericParser(\n {\n grammar: \"ruby\",\n query: QUERY,\n decls: [\n { declCapture: \"function\", nameCapture: \"function.name\", kind: \"function\" },\n { declCapture: \"method\", nameCapture: \"method.name\", kind: \"method\" },\n { declCapture: \"class\", nameCapture: \"class.name\", kind: \"class\" },\n { declCapture: \"module\", nameCapture: \"module.name\", kind: \"class\" },\n ],\n },\n f,\n source,\n );\n}\n","// Rust parser. Functions, structs, enums, traits, impls.\n// Import capture is omitted — `use` paths are nested and complex; the file\n// will still be walked + keyword-indexed.\n\nimport type { ParsedFile } from \"../parser.js\";\nimport type { WalkedFile } from \"../walker.js\";\nimport { runGenericParser } from \"./_generic.js\";\n\nconst QUERY = `\n(function_item name: (identifier) @function.name) @function\n(struct_item name: (type_identifier) @struct.name) @struct\n(enum_item name: (type_identifier) @enum.name) @enum\n(trait_item name: (type_identifier) @trait.name) @trait\n(impl_item type: (type_identifier) @impl.name) @impl\n`;\n\nexport async function parseRust(f: WalkedFile, source: string): Promise<ParsedFile> {\n return runGenericParser(\n {\n grammar: \"rust\",\n query: QUERY,\n decls: [\n { declCapture: \"function\", nameCapture: \"function.name\", kind: \"function\" },\n { declCapture: \"struct\", nameCapture: \"struct.name\", kind: \"class\" },\n { declCapture: \"enum\", nameCapture: \"enum.name\", kind: \"enum\" },\n { declCapture: \"trait\", nameCapture: \"trait.name\", kind: \"interface\" },\n { declCapture: \"impl\", nameCapture: \"impl.name\", kind: \"class\" },\n ],\n },\n f,\n source,\n );\n}\n","// TS/JS parser using tree-sitter-typescript / -tsx WASM grammars.\r\n// Extracts: function/class/interface/type/enum declarations, exported consts,\r\n// arrow functions assigned to const, and import sources.\r\n\r\nimport { Query, type Node } from \"web-tree-sitter\";\r\nimport type { SymbolKind } from \"../../graph/types.js\";\r\nimport { createParser, type GrammarName, type ParsedFile, type ParsedSymbol } from \"../parser.js\";\r\nimport type { WalkedFile } from \"../walker.js\";\r\n\r\n// TS / TSX query — uses the type-identifier node type for class names, includes\r\n// interface / type-alias / enum declarations that don't exist in plain JS.\r\nconst TS_QUERY = `\r\n(function_declaration name: (identifier) @function.name) @function\r\n(class_declaration name: (type_identifier) @class.name) @class\r\n(interface_declaration name: (type_identifier) @interface.name) @interface\r\n(type_alias_declaration name: (type_identifier) @type.name) @type\r\n(enum_declaration name: (identifier) @enum.name) @enum\r\n(method_definition name: (property_identifier) @method.name) @method\r\n(lexical_declaration (variable_declarator name: (identifier) @const-fn.name value: [(arrow_function) (function_expression)])) @const-fn\r\n(import_statement source: (string) @import)\r\n`;\r\n\r\n// JS query — class names are plain identifiers (JS grammar has no\r\n// type_identifier node). No interface / type_alias / enum since JS lacks them.\r\n// Adds a call_expression capture for CommonJS require('x'); filtered in the\r\n// matching loop by checking the function identifier text equals \"require\".\r\nconst JS_QUERY = `\r\n(function_declaration name: (identifier) @function.name) @function\r\n(class_declaration name: (identifier) @class.name) @class\r\n(method_definition name: (property_identifier) @method.name) @method\r\n(lexical_declaration (variable_declarator name: (identifier) @const-fn.name value: [(arrow_function) (function_expression)])) @const-fn\r\n(import_statement source: (string) @import)\r\n(call_expression function: (identifier) @_require_fn arguments: (arguments . (string) @require_source))\r\n`;\r\n\r\nfunction grammarFor(ext: string): GrammarName {\r\n if (ext === \".tsx\" || ext === \".jsx\") return \"tsx\";\r\n if (ext === \".js\" || ext === \".cjs\" || ext === \".mjs\") return \"javascript\";\r\n return \"typescript\";\r\n}\r\n\r\nfunction queryFor(grammar: GrammarName): string {\r\n return grammar === \"javascript\" ? JS_QUERY : TS_QUERY;\r\n}\r\n\r\nfunction unquote(s: string): string {\r\n return s.replace(/^[\"'`]|[\"'`]$/g, \"\");\r\n}\r\n\r\nfunction firstLine(text: string, max = 200): string {\r\n const line = text.split(/\\r?\\n/, 1)[0] ?? \"\";\r\n return line.length > max ? line.slice(0, max) + \"…\" : line;\r\n}\r\n\r\ninterface DeclShape {\r\n decl: Node;\r\n name: Node;\r\n kind: SymbolKind;\r\n}\r\n\r\nfunction shapeFromCaptures(captures: Map<string, Node>): DeclShape | null {\r\n const findDecl = (k: string, sk: SymbolKind): DeclShape | null => {\r\n const decl = captures.get(k);\r\n const name = captures.get(`${k}.name`);\r\n return decl && name ? { decl, name, kind: sk } : null;\r\n };\r\n\r\n return (\r\n findDecl(\"function\", \"function\") ??\r\n findDecl(\"class\", \"class\") ??\r\n findDecl(\"interface\", \"interface\") ??\r\n findDecl(\"type\", \"type\") ??\r\n findDecl(\"enum\", \"enum\") ??\r\n findDecl(\"method\", \"method\") ??\r\n findDecl(\"const-fn\", \"function\")\r\n );\r\n}\r\n\r\nexport async function parseTypeScript(f: WalkedFile, source: string): Promise<ParsedFile> {\r\n const grammar = grammarFor(f.ext);\r\n let symbols: ParsedSymbol[] = [];\r\n let imports: string[] = [];\r\n\r\n try {\r\n const { parser, language } = await createParser(grammar);\r\n const tree = parser.parse(source);\r\n if (!tree) return { file: f, source, symbols, imports, calls: [] };\r\n\r\n const query = new Query(language, queryFor(grammar));\r\n const matches = query.matches(tree.rootNode);\r\n\r\n for (const match of matches) {\r\n const byName = new Map<string, Node>();\r\n for (const cap of match.captures) byName.set(cap.name, cap.node);\r\n\r\n const shape = shapeFromCaptures(byName);\r\n if (shape) {\r\n symbols.push({\r\n name: shape.name.text,\r\n kind: shape.kind,\r\n startLine: shape.decl.startPosition.row + 1,\r\n endLine: shape.decl.endPosition.row + 1,\r\n signature: firstLine(shape.decl.text),\r\n });\r\n continue;\r\n }\r\n const importNode = byName.get(\"import\");\r\n if (importNode) {\r\n imports.push(unquote(importNode.text));\r\n continue;\r\n }\r\n // CommonJS require('x') — only captured by JS_QUERY. The identifier\r\n // must literally be \"require\" (not setTimeout, console, etc).\r\n const requireFn = byName.get(\"_require_fn\");\r\n const requireSource = byName.get(\"require_source\");\r\n if (requireFn && requireSource && requireFn.text === \"require\") {\r\n imports.push(unquote(requireSource.text));\r\n }\r\n }\r\n\r\n const seen = new Set<string>();\r\n symbols = symbols.filter((s) => {\r\n const key = `${s.name}:${s.startLine}`;\r\n if (seen.has(key)) return false;\r\n seen.add(key);\r\n return true;\r\n });\r\n imports = Array.from(new Set(imports));\r\n } catch {\r\n // Parse failure shouldn't abort the whole scan — return what we have.\r\n }\r\n\r\n return { file: f, source, symbols, imports, calls: [] };\r\n}\r\n","// Svelte parser. Extracts <script> and <script lang=\"ts\"> blocks and parses\n// their contents with the TypeScript parser. Tracks the original line offset\n// so reported symbol positions match the .svelte source.\n\nimport { parseTypeScript } from \"./typescript.js\";\nimport type { ParsedFile } from \"../parser.js\";\nimport type { WalkedFile } from \"../walker.js\";\n\nconst SCRIPT_RE = /<script\\b[^>]*>([\\s\\S]*?)<\\/script>/gi;\n\ninterface ScriptBlock {\n source: string;\n startLine: number; // 1-based line number where the script content begins\n isTsx: boolean;\n}\n\nfunction extractScripts(source: string): ScriptBlock[] {\n const out: ScriptBlock[] = [];\n for (const match of source.matchAll(SCRIPT_RE)) {\n const full = match[0];\n const inner = match[1] ?? \"\";\n const openTag = full.slice(0, full.indexOf(\">\") + 1);\n const tagStart = match.index ?? 0;\n const contentStart = tagStart + openTag.length;\n const startLine = source.slice(0, contentStart).split(/\\r?\\n/).length;\n const isTsx = /\\blang\\s*=\\s*[\"']?(ts|tsx|typescript)[\"']?/i.test(openTag);\n out.push({ source: inner, startLine, isTsx });\n }\n return out;\n}\n\nexport async function parseSvelte(f: WalkedFile, source: string): Promise<ParsedFile> {\n const blocks = extractScripts(source);\n const out: ParsedFile = { file: f, source, symbols: [], imports: [], calls: [] };\n\n for (const block of blocks) {\n const virtual: WalkedFile = { ...f, ext: block.isTsx ? \".ts\" : \".js\" };\n const parsed = await parseTypeScript(virtual, block.source);\n const offset = block.startLine - 1;\n for (const sym of parsed.symbols) {\n out.symbols.push({\n ...sym,\n startLine: sym.startLine + offset,\n endLine: sym.endLine + offset,\n });\n }\n for (const imp of parsed.imports) out.imports.push(imp);\n }\n\n // The .svelte file itself is treated as a component.\n out.symbols.push({\n name: f.relPath.split(\"/\").pop()?.replace(/\\.svelte$/i, \"\") ?? f.relPath,\n kind: \"component\",\n startLine: 1,\n endLine: source.split(/\\r?\\n/).length,\n signature: f.relPath,\n });\n out.imports = Array.from(new Set(out.imports));\n return out;\n}\n","// Vue SFC parser. Extracts <script> / <script setup> / <script lang=\"ts\">\n// blocks and parses them with the TypeScript parser, preserving line offsets.\n\nimport { parseTypeScript } from \"./typescript.js\";\nimport type { ParsedFile } from \"../parser.js\";\nimport type { WalkedFile } from \"../walker.js\";\n\nconst SCRIPT_RE = /<script\\b[^>]*>([\\s\\S]*?)<\\/script>/gi;\n\ninterface ScriptBlock {\n source: string;\n startLine: number;\n isTs: boolean;\n}\n\nfunction extractScripts(source: string): ScriptBlock[] {\n const out: ScriptBlock[] = [];\n for (const match of source.matchAll(SCRIPT_RE)) {\n const full = match[0];\n const inner = match[1] ?? \"\";\n const openTag = full.slice(0, full.indexOf(\">\") + 1);\n const tagStart = match.index ?? 0;\n const contentStart = tagStart + openTag.length;\n const startLine = source.slice(0, contentStart).split(/\\r?\\n/).length;\n const isTs = /\\blang\\s*=\\s*[\"']?(ts|tsx|typescript)[\"']?/i.test(openTag);\n out.push({ source: inner, startLine, isTs });\n }\n return out;\n}\n\nexport async function parseVue(f: WalkedFile, source: string): Promise<ParsedFile> {\n const blocks = extractScripts(source);\n const out: ParsedFile = { file: f, source, symbols: [], imports: [], calls: [] };\n\n for (const block of blocks) {\n const virtual: WalkedFile = { ...f, ext: block.isTs ? \".ts\" : \".js\" };\n const parsed = await parseTypeScript(virtual, block.source);\n const offset = block.startLine - 1;\n for (const sym of parsed.symbols) {\n out.symbols.push({\n ...sym,\n startLine: sym.startLine + offset,\n endLine: sym.endLine + offset,\n });\n }\n for (const imp of parsed.imports) out.imports.push(imp);\n }\n\n out.symbols.push({\n name: f.relPath.split(\"/\").pop()?.replace(/\\.vue$/i, \"\") ?? f.relPath,\n kind: \"component\",\n startLine: 1,\n endLine: source.split(/\\r?\\n/).length,\n signature: f.relPath,\n });\n out.imports = Array.from(new Set(out.imports));\n return out;\n}\n","// Walks project root, yields files to parse.\n// Honors .gitignore + .synthraignore (additive — entries in either are ignored).\n// Defensive defaults skip VCS, build, and dependency directories even if absent\n// from .gitignore.\n\nimport { readFile, readdir, stat } from \"node:fs/promises\";\nimport { extname, join, relative, sep } from \"node:path\";\nimport ignore, { type Ignore } from \"ignore\";\n\nexport interface WalkedFile {\n absPath: string;\n relPath: string;\n ext: string;\n size: number;\n}\n\nexport interface WalkOptions {\n /** Maximum file size to yield (bytes). Defaults to 2 MB. */\n maxFileSize?: number;\n /** Additional ignore patterns layered on top of .gitignore + .synthraignore. */\n extraIgnore?: string[];\n}\n\nconst DEFAULT_IGNORE = [\n \".git/\",\n \".synthra/\",\n \".synthra-graph/\",\n \".claude/\",\n \"node_modules/\",\n \"dist/\",\n \"build/\",\n \"out/\",\n \"coverage/\",\n \".next/\",\n \".nuxt/\",\n \".svelte-kit/\",\n \".turbo/\",\n \".cache/\",\n \".vscode/\",\n \".idea/\",\n \".vs/\",\n // Flutter / Dart build caches — IDE-rehydrated, contain third-party\n // type stubs (typescript.d.ts, babylon.js etc.) that contaminate the graph.\n \".dart_tool/\",\n \".flutter-plugins\",\n \".flutter-plugins-dependencies\",\n // Android / Java / Kotlin / Rust\n \".gradle/\",\n \"target/\",\n // iOS / Xcode\n \"Pods/\",\n \"DerivedData/\",\n // Python\n \"__pycache__/\",\n \".venv/\",\n \"venv/\",\n \".tox/\",\n \".pytest_cache/\",\n \".mypy_cache/\",\n \".ruff_cache/\",\n // .NET\n \"obj/\",\n];\n\nconst BINARY_EXTS = new Set([\n \".png\", \".jpg\", \".jpeg\", \".gif\", \".webp\", \".svg\", \".ico\", \".bmp\",\n \".pdf\", \".zip\", \".tar\", \".gz\", \".7z\", \".rar\",\n \".mp3\", \".mp4\", \".mov\", \".avi\", \".webm\", \".wav\", \".ogg\",\n \".ttf\", \".otf\", \".woff\", \".woff2\", \".eot\",\n \".exe\", \".dll\", \".so\", \".dylib\", \".bin\", \".wasm\",\n \".lock\", \".lockb\",\n]);\n\nasync function readIgnoreFile(path: string): Promise<string[]> {\n try {\n const text = await readFile(path, \"utf8\");\n return text\n .split(/\\r?\\n/)\n .map((l) => l.trim())\n .filter((l) => l.length > 0 && !l.startsWith(\"#\"));\n } catch {\n return [];\n }\n}\n\nasync function buildMatcher(root: string, extra: string[]): Promise<Ignore> {\n const ig = ignore();\n ig.add(DEFAULT_IGNORE);\n ig.add(await readIgnoreFile(join(root, \".gitignore\")));\n ig.add(await readIgnoreFile(join(root, \".synthraignore\")));\n if (extra.length) ig.add(extra);\n return ig;\n}\n\nfunction toPosix(p: string): string {\n return sep === \"/\" ? p : p.split(sep).join(\"/\");\n}\n\nexport async function* walk(\n root: string,\n options: WalkOptions = {},\n): AsyncGenerator<WalkedFile> {\n const maxFileSize = options.maxFileSize ?? 2_000_000;\n const ig = await buildMatcher(root, options.extraIgnore ?? []);\n\n async function* recurse(dir: string): AsyncGenerator<WalkedFile> {\n let entries;\n try {\n entries = await readdir(dir, { withFileTypes: true });\n } catch {\n return;\n }\n for (const entry of entries) {\n const abs = join(dir, entry.name);\n const rel = relative(root, abs);\n if (!rel) continue;\n const relPosix = toPosix(rel);\n const matchPath = entry.isDirectory() ? `${relPosix}/` : relPosix;\n if (ig.ignores(matchPath)) continue;\n\n if (entry.isDirectory()) {\n yield* recurse(abs);\n } else if (entry.isFile()) {\n const ext = extname(entry.name).toLowerCase();\n if (BINARY_EXTS.has(ext)) continue;\n let size: number;\n try {\n const s = await stat(abs);\n size = s.size;\n } catch {\n continue;\n }\n if (size > maxFileSize) continue;\n yield { absPath: abs, relPath: relPosix, ext, size };\n }\n }\n }\n\n yield* recurse(root);\n}\n","// Reads/writes info_graph.json and symbol_index.json.\n\nimport { mkdir, readFile, writeFile } from \"node:fs/promises\";\nimport { dirname } from \"node:path\";\n\nimport type { GraphSchema, SymbolIndex } from \"./types.js\";\n\nasync function writeJson(path: string, data: unknown, pretty: boolean): Promise<void> {\n await mkdir(dirname(path), { recursive: true });\n const text = pretty ? JSON.stringify(data, null, 2) : JSON.stringify(data);\n await writeFile(path, text + \"\\n\", \"utf8\");\n}\n\nasync function readJson<T>(path: string): Promise<T> {\n const text = await readFile(path, \"utf8\");\n return JSON.parse(text) as T;\n}\n\nexport async function writeGraph(path: string, graph: GraphSchema): Promise<void> {\n // Pretty-printing a graph with full file contents balloons disk size and\n // the JSON is only ever read by machines; keep it compact.\n await writeJson(path, graph, false);\n}\n\nexport async function readGraph(path: string): Promise<GraphSchema> {\n return readJson<GraphSchema>(path);\n}\n\nexport async function writeSymbolIndex(path: string, index: SymbolIndex): Promise<void> {\n await writeJson(path, index, true);\n}\n\nexport async function readSymbolIndex(path: string): Promise<SymbolIndex> {\n // Re-home onto a null prototype so name lookups (e.g. index[\"toString\"])\n // never resolve to an inherited Object.prototype member. Mirrors\n // buildSymbolIndex, which builds the index the same way.\n const parsed = await readJson<SymbolIndex>(path);\n return Object.assign(Object.create(null), parsed);\n}\n","// Resolves Synthra's storage locations inside a project root.\n\nimport { join } from \"node:path\";\n\nexport interface SynthraPaths {\n projectRoot: string;\n graphDir: string;\n contextDir: string;\n infoGraph: string;\n symbolIndex: string;\n sessionState: string;\n activityLog: string;\n tokenLog: string;\n gateLog: string;\n mcpPort: string;\n mcpServerLog: string;\n mcpServerErrLog: string;\n contextStore: string;\n contextMd: string;\n branchesDir: string;\n claudeDir: string;\n claudeSettings: string;\n claudeHooksDir: string;\n claudeMd: string;\n gitignore: string;\n}\n\nexport function resolvePaths(projectRoot: string): SynthraPaths {\n const graphDir = join(projectRoot, \".synthra-graph\");\n const contextDir = join(projectRoot, \".synthra\");\n const claudeDir = join(projectRoot, \".claude\");\n\n return {\n projectRoot,\n graphDir,\n contextDir,\n infoGraph: join(graphDir, \"info_graph.json\"),\n symbolIndex: join(graphDir, \"symbol_index.json\"),\n sessionState: join(graphDir, \"session.json\"),\n activityLog: join(graphDir, \"activity.jsonl\"),\n tokenLog: join(graphDir, \"token_log.jsonl\"),\n gateLog: join(graphDir, \"gate_log.jsonl\"),\n mcpPort: join(graphDir, \"mcp_port\"),\n mcpServerLog: join(graphDir, \"mcp_server.log\"),\n mcpServerErrLog: join(graphDir, \"mcp_server.err.log\"),\n contextStore: join(contextDir, \"context-store.json\"),\n contextMd: join(contextDir, \"CONTEXT.md\"),\n branchesDir: join(contextDir, \"branches\"),\n claudeDir,\n claudeSettings: join(claudeDir, \"settings.local.json\"),\n claudeHooksDir: join(claudeDir, \"hooks\"),\n claudeMd: join(projectRoot, \"CLAUDE.md\"),\n gitignore: join(projectRoot, \".gitignore\"),\n };\n}\n","// Project bootstrap: creates .synthra-graph/, .synthra/, updates .gitignore,\r\n// patches CLAUDE.md with the versioned policy block.\r\n\r\nimport { mkdir, readFile, stat, writeFile } from \"node:fs/promises\";\r\nimport { basename } from \"node:path\";\r\n\r\nimport { patchClaudeMd } from \"../hooks/claude-md.js\";\r\nimport type { SynthraPaths } from \"../shared/paths.js\";\r\n\r\nexport interface BootstrapResult {\r\n graphCreated: boolean;\r\n contextCreated: boolean;\r\n gitignoreUpdated: boolean;\r\n claudeMdUpdated: boolean;\r\n claudeMdCreated: boolean;\r\n}\r\n\r\n// Entries Synthra appends to the project .gitignore on bootstrap.\r\n// Each is gated by a check: if the entry is already present (any\r\n// indentation, trimmed match), it's skipped. Comments are per-entry so\r\n// users understand why each line is there and can remove what they don't\r\n// want without breaking the rest.\r\nconst GITIGNORE_ENTRIES: { comment: string; entry: string }[] = [\r\n {\r\n comment: \"added by synthra (heavy generated state — gitignored by design)\",\r\n entry: \".synthra-graph/\",\r\n },\r\n {\r\n comment:\r\n \"added by synthra — MCP registration. Remove this line if you want \" +\r\n \"to share the synthra MCP entry with teammates via committed .mcp.json\",\r\n entry: \".mcp.json\",\r\n },\r\n];\r\n\r\nasync function exists(path: string): Promise<boolean> {\r\n try {\r\n await stat(path);\r\n return true;\r\n } catch {\r\n return false;\r\n }\r\n}\r\n\r\nasync function ensureDir(path: string): Promise<boolean> {\r\n const had = await exists(path);\r\n await mkdir(path, { recursive: true });\r\n return !had;\r\n}\r\n\r\nasync function patchGitignore(path: string): Promise<boolean> {\r\n let existing = \"\";\r\n try {\r\n existing = await readFile(path, \"utf8\");\r\n } catch {\r\n /* file may not exist */\r\n }\r\n const trimmed = new Set(existing.split(/\\r?\\n/).map((l) => l.trim()));\r\n const missing = GITIGNORE_ENTRIES.filter((e) => !trimmed.has(e.entry));\r\n if (missing.length === 0) return false;\r\n\r\n const block =\r\n missing.map((m) => `# ${m.comment}\\n${m.entry}`).join(\"\\n\") + \"\\n\";\r\n const appendix =\r\n (existing.length === 0 || existing.endsWith(\"\\n\") ? \"\" : \"\\n\") +\r\n (existing.length ? \"\\n\" : \"\") +\r\n block;\r\n await writeFile(path, existing + appendix, \"utf8\");\r\n return true;\r\n}\r\n\r\nexport async function bootstrap(paths: SynthraPaths): Promise<BootstrapResult> {\r\n const graphCreated = await ensureDir(paths.graphDir);\r\n const contextCreated = await ensureDir(paths.contextDir);\r\n const gitignoreUpdated = await patchGitignore(paths.gitignore);\r\n\r\n const claudeMdExistedBefore = await exists(paths.claudeMd);\r\n const patch = await patchClaudeMd(paths.claudeMd, basename(paths.projectRoot));\r\n\r\n return {\r\n graphCreated,\r\n contextCreated,\r\n gitignoreUpdated,\r\n claudeMdUpdated: patch.updated,\r\n claudeMdCreated: patch.created && !claudeMdExistedBefore,\r\n };\r\n}\r\n","// Idempotent patcher for the project's CLAUDE.md. Manages a single block\r\n// bounded by <!-- synthra-policy v<N> BEGIN --> ... <!-- synthra-policy v<N> END -->.\r\n// On each run, any prior synthra-policy block (any version) is removed and the\r\n// current-version block is appended at the end.\r\n\r\nimport { readFile, writeFile } from \"node:fs/promises\";\r\nimport { basename, dirname } from \"node:path\";\r\n\r\nexport const POLICY_VERSION = 4;\r\nexport const POLICY_BEGIN = `<!-- synthra-policy v${POLICY_VERSION} BEGIN -->`;\r\nexport const POLICY_END = `<!-- synthra-policy v${POLICY_VERSION} END -->`;\r\n\r\n// Matches a synthra-policy block of any version, e.g. v1, v2 …\r\nconst ANY_BLOCK_RE = /<!--\\s*synthra-policy\\s+v\\d+\\s+BEGIN\\s*-->[\\s\\S]*?<!--\\s*synthra-policy\\s+v\\d+\\s+END\\s*-->\\s*/g;\r\n\r\nexport interface PatchResult {\r\n created: boolean;\r\n updated: boolean;\r\n skipped: boolean;\r\n}\r\n\r\nexport function policyBlock(): string {\r\n return [\r\n POLICY_BEGIN,\r\n \"## Synthra context policy\",\r\n \"\",\r\n \"Synthra has pre-loaded structured context into this session and exposes\",\r\n \"the project's code graph through MCP tools. **Prefer these tools over\",\r\n \"Grep / Glob / Read** — they are faster, cheaper, and already filtered\",\r\n \"to relevant files.\",\r\n \"\",\r\n \"> **Tool namespace.** Synthra's MCP tools are exposed as\",\r\n \"> `mcp__synthra__graph_continue`, `mcp__synthra__graph_read`, and\",\r\n \"> `mcp__synthra__graph_register_edit`. Below they are referred to by\",\r\n \"> their short names (`graph_continue` etc.) for readability — use the\",\r\n \"> full namespaced form when actually invoking them.\",\r\n \"\",\r\n \"### Tools\",\r\n \"\",\r\n \"- **`graph_continue(query)`** — returns a `Confidence` label, the list\",\r\n \" of relevant `Files`, and signatures + top function bodies for those\",\r\n \" files. Your default first move when you need project context.\",\r\n \"- **`graph_read(target)`** — fetch source. Prefer the\",\r\n \" `\\\"file/path.ts::SymbolName\\\"` form over a bare file path — reading one\",\r\n \" symbol is ~50 tokens, reading a whole file is thousands.\",\r\n \"- **`graph_register_edit(files)`** — after you edit files, call this so\",\r\n \" subsequent turns weight your changes and avoid stale snapshots.\",\r\n \"\",\r\n \"### When to call `graph_continue` — and when to skip\",\r\n \"\",\r\n \"**Call `graph_continue` only when you do NOT already know the relevant\",\r\n \"files.**\",\r\n \"\",\r\n \"Call it when:\",\r\n \"- This is the first message of a new task or conversation\",\r\n \"- The task shifts to a different area of the codebase\",\r\n \"- You need files you haven't seen yet in this session\",\r\n \"\",\r\n \"**Skip `graph_continue` when:**\",\r\n \"- You already identified the relevant files earlier in this conversation\",\r\n \"- You are doing follow-up work on files already read (verify, refactor,\",\r\n \" test, docs, cleanup, commit)\",\r\n \"- The task is pure text (commit message, explanation, summary)\",\r\n \"\",\r\n \"If skipping, go directly to `graph_read(\\\"file.ts::symbol\\\")` on what\",\r\n \"you already know.\",\r\n \"\",\r\n \"### Confidence caps\",\r\n \"\",\r\n \"When `graph_continue` returns:\",\r\n \"\",\r\n \"- **`Confidence: high`** → Stop. Do NOT Grep, Glob, or further explore\",\r\n \" for this query. The graph already has it.\",\r\n \"- **`Confidence: medium`** → Read the listed `Files` directly via\",\r\n \" `graph_read(\\\"file::symbol\\\")` *before* trying Grep. The graph has\",\r\n \" narrowed the search space — use it, don't bypass it.\",\r\n \"- **`Confidence: low`** → You may use Grep / Glob, but the PreToolUse\",\r\n \" hook may still block redundant calls.\",\r\n \"\",\r\n \"### Reading code\",\r\n \"\",\r\n \"- **Always use `file::symbol` notation** with `graph_read`. Whole-file\",\r\n \" reads should be rare — only when you genuinely need the full file.\",\r\n \"- If `graph_continue`'s `Files` list contains a `::` entry, pass it\",\r\n \" verbatim to `graph_read`.\",\r\n \"\",\r\n \"### Editing a file\",\r\n \"\",\r\n \"Claude Code's `Edit` tool (and `Write` when overwriting) only accepts a\",\r\n \"file that was opened with the **`Read` tool** — a `graph_read` slice does\",\r\n \"not count, and editing such a file fails with *\\\"File has not been read\",\r\n \"yet.\\\"* So before editing a file you only know through `graph_read`: take\",\r\n \"the line range from its header (e.g. `…::handler (L120-168)`), `Read` that\",\r\n \"file with a matching `offset`/`limit`, then `Edit`. That satisfies the\",\r\n \"gate while keeping the read small — don't whole-file `Read` unless the\",\r\n \"edit spans most of the file.\",\r\n \"\",\r\n \"### Don'ts\",\r\n \"\",\r\n \"- Don't Grep / Glob before calling `graph_continue` when required — the\",\r\n \" PreToolUse hook may block it.\",\r\n \"- Don't call `graph_continue` more than once per turn.\",\r\n \"- Don't read whole files when a symbol-level read would suffice.\",\r\n \"\",\r\n \"### Session-end resume note\",\r\n \"\",\r\n \"When the user signals they're done (e.g. \\\"bye\\\", \\\"wrap up\\\", \\\"done\\\"),\",\r\n \"persist the resume state by calling `context_remember` once per bullet.\",\r\n \"Synthra re-renders `.synthra/CONTEXT.md` from those entries at session\",\r\n \"end — do **NOT** write to `CONTEXT.md` directly, it is a derived view\",\r\n \"and direct edits are overwritten by the Stop hook.\",\r\n \"\",\r\n \"Use these `kind` values:\",\r\n \"\",\r\n \"- **`kind: \\\"task\\\"`** — what is being worked on right now (1 entry)\",\r\n \"- **`kind: \\\"decision\\\"`** — non-obvious choices made this session (max 3)\",\r\n \"- **`kind: \\\"next\\\"`** — concrete next steps (max 3)\",\r\n \"\",\r\n \"Tag entries with the relevant area (`tags: [\\\"auth\\\"]`) and the files\",\r\n \"they touch (`files: [\\\"src/auth.ts\\\"]`) so later `context_recall` queries\",\r\n \"can filter. Keep each `text` to 1–2 sentences.\",\r\n \"\",\r\n \"_This block is managed by Synthra. Edits inside the BEGIN/END markers\",\r\n \"are overwritten on every `syn .` run._\",\r\n \"\",\r\n POLICY_END,\r\n ].join(\"\\n\");\r\n}\r\n\r\n// A lean, agent-facing onboarding skeleton written ONLY when a project has no\r\n// CLAUDE.md yet. It captures the durable \"why/how\" the graph can't infer\r\n// (build/test, conventions, decisions, gotchas). It lives OUTSIDE the\r\n// synthra-policy markers, so later `syn .` runs — which strip and re-add the\r\n// policy block — never touch what the user fills in here.\r\nexport function onboardingSkeleton(projectName: string): string {\r\n return [\r\n `# ${projectName}`,\r\n \"\",\r\n \"> Onboarding notes for AI coding agents. Synthra's graph already knows the\",\r\n \"> code's *structure* (files, symbols, imports) — this file is for what the\",\r\n \"> graph can't infer: how to run the project, its conventions, and the\",\r\n \"> decisions behind them. Keep it lean and current; delete prompts you don't need.\",\r\n \"\",\r\n \"## Build & test\",\r\n \"\",\r\n \"- TODO: install deps / build\",\r\n \"- TODO: run tests / lint / typecheck\",\r\n \"- TODO: run the app locally\",\r\n \"\",\r\n \"## Conventions\",\r\n \"\",\r\n \"- TODO: code style, naming, file layout the agent should follow\",\r\n \"\",\r\n \"## Key decisions\",\r\n \"\",\r\n \"- TODO: non-obvious choices and *why* (\\\"we use X not Y because …\\\")\",\r\n \"\",\r\n \"## Gotchas\",\r\n \"\",\r\n \"- TODO: traps, footguns, \\\"don't touch X without Y\\\"\",\r\n \"\",\r\n \"_Tip: run `/init` in Claude Code to auto-draft the sections above, then trim\",\r\n \"to the durable bits. Synthra manages its own block below — leave it._\",\r\n \"\",\r\n ].join(\"\\n\");\r\n}\r\n\r\nexport async function patchClaudeMd(path: string, projectName?: string): Promise<PatchResult> {\r\n let existing: string | null;\r\n try {\r\n existing = await readFile(path, \"utf8\");\r\n } catch {\r\n existing = null;\r\n }\r\n\r\n const block = policyBlock();\r\n\r\n if (existing === null) {\r\n // First creation: scaffold the onboarding skeleton (user-owned, written\r\n // once) followed by Synthra's managed policy block.\r\n const name = projectName || basename(dirname(path)) || \"this project\";\r\n await writeFile(path, onboardingSkeleton(name) + \"\\n\" + block + \"\\n\", \"utf8\");\r\n return { created: true, updated: false, skipped: false };\r\n }\r\n\r\n const stripped = existing.replace(ANY_BLOCK_RE, \"\");\r\n const hadBlock = stripped !== existing;\r\n\r\n const desired = stripped.endsWith(\"\\n\")\r\n ? stripped + \"\\n\" + block + \"\\n\"\r\n : (stripped.length ? stripped + \"\\n\\n\" : \"\") + block + \"\\n\";\r\n\r\n if (hadBlock && desired === existing) {\r\n return { created: false, updated: false, skipped: true };\r\n }\r\n\r\n await writeFile(path, desired, \"utf8\");\r\n return { created: false, updated: true, skipped: false };\r\n}\r\n","// Scoring for retrieved files. Combines:\n// - keyword overlap with the (already-tokenized) query\n// - symbol-name overlap (boosted: hits in a file's defined symbols matter more)\n// - import-graph proximity from session-known + recent paths (boost adjacent files)\n// - recency boost from activity log (placeholder — wired in M5)\n\nimport type { Edge, FileNode, GraphSchema, SymbolNode } from \"./types.js\";\n\nexport interface RankInputs {\n candidates: FileNode[];\n query: string;\n graph?: GraphSchema;\n recentlyEditedPaths?: string[];\n sessionKnownPaths?: string[];\n}\n\nexport interface ScoredFile {\n file: FileNode;\n score: number;\n reasons: string[];\n /** Total symbol-match weight (exact name = 3, partial substring = 1). */\n symHits: number;\n /** Count of symbols whose name a query token matched *exactly*. The gate\n * uses this — a partial substring match is too noisy (\"id\" ⊂ \"width\"). */\n exactSym: number;\n}\n\nconst STOPWORDS = new Set([\n \"a\", \"an\", \"and\", \"are\", \"as\", \"at\", \"be\", \"by\", \"for\", \"from\",\n \"has\", \"have\", \"in\", \"is\", \"it\", \"of\", \"on\", \"or\", \"that\", \"the\",\n \"this\", \"to\", \"was\", \"we\", \"with\", \"what\", \"where\", \"when\", \"why\",\n \"how\", \"do\", \"does\", \"i\", \"me\", \"my\", \"you\", \"your\", \"code\", \"file\",\n]);\n\nexport function tokenizeQuery(query: string): string[] {\n const tokens = query\n .toLowerCase()\n .split(/[^a-z0-9_]+/g)\n .filter((t) => t.length > 1 && !STOPWORDS.has(t));\n // Also split camelCase/snake_case for queries like \"AuthService\"\n const expanded = new Set<string>();\n for (const t of tokens) {\n expanded.add(t);\n const parts = t.match(/[a-z]+|[0-9]+/g) ?? [];\n for (const p of parts) if (p.length > 1) expanded.add(p);\n }\n return Array.from(expanded);\n}\n\nfunction indexSymbolsByFile(graph: GraphSchema | undefined): Map<string, SymbolNode[]> {\n const out = new Map<string, SymbolNode[]>();\n if (!graph) return out;\n for (const n of graph.nodes) {\n if (n.kind !== \"symbol\") continue;\n const list = out.get(n.file) ?? [];\n list.push(n);\n out.set(n.file, list);\n }\n return out;\n}\n\nfunction indexImportEdges(graph: GraphSchema | undefined): Map<string, Set<string>> {\n // file path → set of file paths it imports (1-hop)\n const out = new Map<string, Set<string>>();\n if (!graph) return out;\n const idToPath = new Map<string, string>();\n for (const n of graph.nodes) if (n.kind === \"file\") idToPath.set(n.id, n.path);\n for (const e of graph.edges as Edge[]) {\n if (e.kind !== \"imports\") continue;\n const from = idToPath.get(e.from);\n const to = idToPath.get(e.to);\n if (!from || !to) continue;\n const s = out.get(from) ?? new Set<string>();\n s.add(to);\n out.set(from, s);\n }\n return out;\n}\n\nexport function scoreFiles(inputs: RankInputs): ScoredFile[] {\n const qTokens = new Set(tokenizeQuery(inputs.query));\n const symbolsByFile = indexSymbolsByFile(inputs.graph);\n const importsFrom = indexImportEdges(inputs.graph);\n\n const seeds = new Set<string>(inputs.sessionKnownPaths ?? []);\n for (const p of inputs.recentlyEditedPaths ?? []) seeds.add(p);\n\n // First pass: keyword + symbol score\n const scored: ScoredFile[] = [];\n for (const file of inputs.candidates) {\n const reasons: string[] = [];\n let score = 0;\n\n // Keyword overlap\n let kwHits = 0;\n for (const kw of file.keywords) if (qTokens.has(kw)) kwHits += 1;\n if (kwHits) {\n score += kwHits * 2;\n reasons.push(`kw=${kwHits}`);\n }\n\n // Symbol-name overlap (higher signal than file-level keywords)\n const symbols = symbolsByFile.get(file.path) ?? [];\n let symHits = 0;\n let exactSym = 0;\n for (const sym of symbols) {\n const name = sym.name.toLowerCase();\n if (qTokens.has(name)) {\n symHits += 3;\n exactSym += 1;\n } else {\n // partial match: any query token is a substring of, or contained by, the symbol name\n for (const t of qTokens) {\n if (name.includes(t) || t.includes(name)) {\n symHits += 1;\n break;\n }\n }\n }\n }\n if (symHits) {\n score += symHits;\n reasons.push(`sym=${symHits}`);\n }\n\n // Path match: file path contains a query token\n const pathLower = file.path.toLowerCase();\n let pathHits = 0;\n for (const t of qTokens) if (pathLower.includes(t)) pathHits += 1;\n if (pathHits) {\n score += pathHits;\n reasons.push(`path=${pathHits}`);\n }\n\n if (seeds.has(file.path)) {\n score += 5;\n reasons.push(\"seed\");\n }\n\n scored.push({ file, score, reasons, symHits, exactSym });\n }\n\n // Second pass: 1-hop import-graph boost from any file already scored > 0\n const positivePaths = new Set(scored.filter((s) => s.score > 0).map((s) => s.file.path));\n if (positivePaths.size > 0) {\n for (const s of scored) {\n if (s.score > 0) continue;\n // Does any file that imports this one have a positive score?\n let importBoost = 0;\n for (const [from, tos] of importsFrom) {\n if (!positivePaths.has(from)) continue;\n if (tos.has(s.file.path)) {\n importBoost += 1;\n break;\n }\n }\n if (importBoost) {\n s.score += importBoost * 0.5;\n s.reasons.push(\"imp-adj\");\n }\n }\n }\n\n scored.sort((a, b) => b.score - a.score);\n return scored;\n}\n\nexport function rank(inputs: RankInputs): FileNode[] {\n return scoreFiles(inputs).map((s) => s.file);\n}\n","// Query-time retrieval. Given a natural-language query (or a symbol name),\n// returns a ranked list of FileNodes plus a confidence label.\n\nimport { scoreFiles, tokenizeQuery, type RankInputs } from \"./rank.js\";\nimport type { FileNode, GraphSchema } from \"./types.js\";\n\nexport interface RetrievalResult {\n files: FileNode[];\n confidence: \"high\" | \"medium\" | \"low\";\n reason: string;\n /** True if any returned file has a symbol whose name a query token matched\n * exactly — i.e. graph_read can serve a real `file::symbol` slice for this\n * query. When false, a Grep/Glob block would only force a fallback Read. */\n symbolMatched: boolean;\n}\n\nexport interface RetrieveOptions {\n topK?: number;\n recentlyEditedPaths?: string[];\n sessionKnownPaths?: string[];\n}\n\nexport async function retrieve(\n graph: GraphSchema,\n query: string,\n options: RetrieveOptions = {},\n): Promise<RetrievalResult> {\n const topK = options.topK ?? 12;\n const qTokens = tokenizeQuery(query);\n\n const allFiles: FileNode[] = graph.nodes.filter(\n (n): n is FileNode => n.kind === \"file\",\n );\n\n if (allFiles.length === 0 || qTokens.length === 0) {\n return {\n files: [],\n confidence: \"low\",\n reason: qTokens.length === 0 ? \"empty query\" : \"empty graph\",\n symbolMatched: false,\n };\n }\n\n const rankInputs: RankInputs = {\n candidates: allFiles,\n query,\n graph,\n recentlyEditedPaths: options.recentlyEditedPaths,\n sessionKnownPaths: options.sessionKnownPaths,\n };\n const scored = scoreFiles(rankInputs);\n const positive = scored.filter((s) => s.score > 0);\n\n if (positive.length === 0) {\n return {\n files: [],\n confidence: \"low\",\n reason: `no matches for ${JSON.stringify(qTokens)}`,\n symbolMatched: false,\n };\n }\n\n const topScored = positive.slice(0, topK);\n const top = topScored.map((s) => s.file);\n const symbolMatched = topScored.some((s) => s.exactSym > 0);\n const topScore = positive[0]?.score ?? 0;\n const secondScore = positive[1]?.score ?? 0;\n\n // confidence: high = clear top match (2x next or only one hit)\n // medium = several hits but no dominant one\n // low = a few weak hits\n let confidence: \"high\" | \"medium\" | \"low\";\n if (positive.length === 1) confidence = \"high\";\n else if (topScore >= 6 && topScore >= secondScore * 2) confidence = \"high\";\n else if (topScore >= 3) confidence = \"medium\";\n else confidence = \"low\";\n\n const reasons = positive\n .slice(0, Math.min(3, top.length))\n .map((s) => `${s.file.path} (${s.reasons.join(\",\")})`)\n .join(\"; \");\n\n return {\n files: top,\n confidence,\n reason: `top: ${reasons}`,\n symbolMatched,\n };\n}\n","// Branch-aware routing for the context store.\n// On the default branch, reads/writes go to .synthra/context-store.json.\n// On a feature branch, they go to .synthra/branches/<sanitized-branch>/context-store.json.\n\nimport { execFile } from \"node:child_process\";\nimport { readFile } from \"node:fs/promises\";\nimport { join } from \"node:path\";\nimport { promisify } from \"node:util\";\n\nconst execFileAsync = promisify(execFile);\n\nexport async function currentBranch(projectRoot: string): Promise<string> {\n // Try .git/HEAD first — avoids the subprocess cost and works for detached\n // worktrees too (.git is a file there).\n try {\n const headPath = join(projectRoot, \".git\", \"HEAD\");\n const head = await readFile(headPath, \"utf8\");\n const trimmed = head.trim();\n const match = trimmed.match(/^ref:\\s+refs\\/heads\\/(.+)$/);\n if (match?.[1]) return match[1];\n // Detached HEAD — fall through\n } catch {\n // .git/HEAD unreadable (worktree file, submodule, or not a git repo)\n }\n\n try {\n const { stdout } = await execFileAsync(\"git\", [\"branch\", \"--show-current\"], {\n cwd: projectRoot,\n });\n const name = stdout.trim();\n if (name) return name;\n } catch {\n // git not on PATH or not a repo\n }\n\n return \"main\";\n}\n\nexport async function defaultBranch(projectRoot: string): Promise<string> {\n try {\n const { stdout } = await execFileAsync(\n \"git\",\n [\"symbolic-ref\", \"refs/remotes/origin/HEAD\", \"--short\"],\n { cwd: projectRoot },\n );\n const trimmed = stdout.trim();\n const match = trimmed.match(/^origin\\/(.+)$/);\n if (match?.[1]) return match[1];\n } catch {\n // No origin/HEAD set — fall back to heuristic\n }\n return \"main\";\n}\n\nexport function sanitizeBranchName(name: string): string {\n return name.replaceAll(\"/\", \"-\").replaceAll(\"\\\\\", \"-\");\n}\n\nexport interface BranchScopedPaths {\n contextStore: string;\n contextMd: string;\n branchDir: string | null;\n}\n\nexport function resolveBranchPaths(\n contextDir: string,\n branch: string,\n isDefault: boolean,\n): BranchScopedPaths {\n if (isDefault) {\n return {\n contextStore: join(contextDir, \"context-store.json\"),\n contextMd: join(contextDir, \"CONTEXT.md\"),\n branchDir: null,\n };\n }\n const branchDir = join(contextDir, \"branches\", sanitizeBranchName(branch));\n return {\n contextStore: join(branchDir, \"context-store.json\"),\n contextMd: join(branchDir, \"CONTEXT.md\"),\n branchDir,\n };\n}\n","// Free-form CONTEXT.md narrative. Updated by Stop hook at session end with:\n// - Current Task (1 sentence)\n// - Key Decisions (max 3 bullets)\n// - Next Steps (max 3 bullets)\n// Capped at ~20 visible content lines.\n\nimport { mkdir, readFile, writeFile } from \"node:fs/promises\";\nimport { dirname } from \"node:path\";\n\nimport type { ContextEntry } from \"./context-store.js\";\n\nexport interface ContextMd {\n branch: string;\n currentTask: string;\n keyDecisions: string[];\n nextSteps: string[];\n date: string;\n}\n\nconst MAX_BULLETS = 3;\n\nexport function deriveContextMd(entries: ContextEntry[], branch: string): ContextMd {\n // Latest pending task drives \"current task\".\n const tasks = entries.filter((e) => e.type === \"task\").reverse();\n const currentTask = tasks[0]?.content ?? \"\";\n\n const keyDecisions = entries\n .filter((e) => e.type === \"decision\")\n .slice(-MAX_BULLETS)\n .map((e) => e.content);\n\n const nextSteps = entries\n .filter((e) => e.type === \"next\")\n .slice(-MAX_BULLETS)\n .map((e) => e.content);\n\n return {\n branch,\n currentTask,\n keyDecisions,\n nextSteps,\n date: new Date().toISOString(),\n };\n}\n\nexport function formatContextMd(ctx: ContextMd): string {\n const lines: string[] = [];\n lines.push(`# Context — ${ctx.branch}`);\n lines.push(\"\");\n lines.push(`_Updated: ${ctx.date}_`);\n lines.push(\"\");\n\n if (ctx.currentTask) {\n lines.push(`## Current task`);\n lines.push(ctx.currentTask);\n lines.push(\"\");\n }\n\n if (ctx.keyDecisions.length) {\n lines.push(`## Key decisions`);\n for (const d of ctx.keyDecisions) lines.push(`- ${d}`);\n lines.push(\"\");\n }\n\n if (ctx.nextSteps.length) {\n lines.push(`## Next steps`);\n for (const n of ctx.nextSteps) lines.push(`- ${n}`);\n lines.push(\"\");\n }\n\n if (!ctx.currentTask && !ctx.keyDecisions.length && !ctx.nextSteps.length) {\n lines.push(\"_(no context entries yet — use `context_remember` to add one)_\");\n lines.push(\"\");\n }\n\n return lines.join(\"\\n\");\n}\n\nexport async function writeContextMd(path: string, ctx: ContextMd): Promise<void> {\n await mkdir(dirname(path), { recursive: true });\n await writeFile(path, formatContextMd(ctx), \"utf8\");\n}\n\nexport async function readContextMd(path: string): Promise<string | null> {\n try {\n return await readFile(path, \"utf8\");\n } catch {\n return null;\n }\n}\n","// Structured decisions/tasks/facts that persist across sessions.\n// Stored in .synthra/ (GIT-TRACKED) so teammates inherit them.\n// Branch-partitioned via branches.ts.\n\nimport { mkdir, readFile, writeFile } from \"node:fs/promises\";\nimport { dirname } from \"node:path\";\n\nexport type EntryKind = \"decision\" | \"task\" | \"next\" | \"fact\" | \"blocker\";\n\nexport interface ContextEntry {\n type: EntryKind;\n content: string;\n tags: string[];\n files: string[];\n date: string;\n}\n\ninterface Store {\n schema_version: number;\n entries: ContextEntry[];\n}\n\nconst SCHEMA_VERSION = 1;\n\nexport async function readEntries(path: string): Promise<ContextEntry[]> {\n try {\n const raw = await readFile(path, \"utf8\");\n const parsed = JSON.parse(raw) as Partial<Store>;\n return Array.isArray(parsed.entries) ? parsed.entries : [];\n } catch {\n return [];\n }\n}\n\nexport async function writeEntries(path: string, entries: ContextEntry[]): Promise<void> {\n await mkdir(dirname(path), { recursive: true });\n const store: Store = { schema_version: SCHEMA_VERSION, entries };\n await writeFile(path, JSON.stringify(store, null, 2) + \"\\n\", \"utf8\");\n}\n\nexport async function appendEntry(path: string, entry: ContextEntry): Promise<void> {\n const entries = await readEntries(path);\n entries.push(entry);\n await writeEntries(path, entries);\n}\n","// High-level orchestration: branch detection + store routing + CONTEXT.md\n// refresh in one call. Used by the MCP tools and the /context-update route.\n\nimport type { SynthraPaths } from \"../shared/paths.js\";\nimport {\n currentBranch,\n defaultBranch,\n resolveBranchPaths,\n type BranchScopedPaths,\n} from \"./branches.js\";\nimport { deriveContextMd, writeContextMd } from \"./context-md.js\";\nimport { appendEntry, readEntries, type ContextEntry, type EntryKind } from \"./context-store.js\";\n\nexport interface ActiveBranch {\n branch: string;\n isDefault: boolean;\n paths: BranchScopedPaths;\n}\n\nexport async function resolveActiveBranch(\n paths: SynthraPaths,\n override?: string,\n): Promise<ActiveBranch> {\n const branch = override ?? (await currentBranch(paths.projectRoot));\n const def = await defaultBranch(paths.projectRoot);\n const isDefault = branch === def;\n return {\n branch,\n isDefault,\n paths: resolveBranchPaths(paths.contextDir, branch, isDefault),\n };\n}\n\nexport interface RememberInput {\n text: string;\n kind: EntryKind;\n tags?: string[];\n files?: string[];\n}\n\nexport interface RememberResult {\n entry: ContextEntry;\n branch: string;\n storePath: string;\n contextMdPath: string;\n}\n\nexport async function rememberEntry(\n paths: SynthraPaths,\n input: RememberInput,\n): Promise<RememberResult> {\n const active = await resolveActiveBranch(paths);\n const entry: ContextEntry = {\n type: input.kind,\n content: input.text,\n tags: input.tags ?? [],\n files: input.files ?? [],\n date: new Date().toISOString(),\n };\n await appendEntry(active.paths.contextStore, entry);\n\n // Refresh CONTEXT.md so the narrative stays in sync with the structured store.\n const entries = await readEntries(active.paths.contextStore);\n const md = deriveContextMd(entries, active.branch);\n await writeContextMd(active.paths.contextMd, md);\n\n return {\n entry,\n branch: active.branch,\n storePath: active.paths.contextStore,\n contextMdPath: active.paths.contextMd,\n };\n}\n\nexport interface RecallInput {\n kind?: EntryKind;\n branch?: string;\n limit?: number;\n}\n\nexport interface RecallResult {\n branch: string;\n entries: ContextEntry[];\n storePath: string;\n}\n\nexport async function recallEntries(\n paths: SynthraPaths,\n input: RecallInput = {},\n): Promise<RecallResult> {\n const active = await resolveActiveBranch(paths, input.branch);\n let entries = await readEntries(active.paths.contextStore);\n if (input.kind) entries = entries.filter((e) => e.type === input.kind);\n if (input.limit && input.limit > 0) entries = entries.slice(-input.limit);\n return {\n branch: active.branch,\n entries,\n storePath: active.paths.contextStore,\n };\n}\n\nexport async function refreshContextMd(paths: SynthraPaths, branchOverride?: string) {\n const active = await resolveActiveBranch(paths, branchOverride);\n const entries = await readEntries(active.paths.contextStore);\n const md = deriveContextMd(entries, active.branch);\n await writeContextMd(active.paths.contextMd, md);\n return {\n branch: active.branch,\n path: active.paths.contextMd,\n entriesSeen: entries.length,\n };\n}\n","// Renders the final context pack as a single Markdown blob for Claude.\n\nexport interface FormatFileSection {\n path: string;\n reason?: string;\n signatures: string[];\n inlineBodies: string;\n associatedTests?: string[];\n}\n\nexport interface FormatInputs {\n query: string;\n files: FormatFileSection[];\n recentActivity?: string;\n truncated?: boolean;\n}\n\nexport function formatPack(inputs: FormatInputs): string {\n const parts: string[] = [];\n parts.push(`# Synthra context — query: ${JSON.stringify(inputs.query)}\\n`);\n\n if (inputs.files.length === 0) {\n parts.push(\"> No matching files found in the graph.\\n\");\n }\n\n for (const f of inputs.files) {\n const heading = f.reason ? `## ${f.path} _(${f.reason})_` : `## ${f.path}`;\n parts.push(heading);\n\n if (f.signatures.length === 0) {\n parts.push(\"_(no symbols extracted)_\");\n } else {\n parts.push(\"**Signatures:**\");\n for (const s of f.signatures) parts.push(`- ${s}`);\n }\n\n if (f.inlineBodies.trim().length > 0) {\n parts.push(\"\");\n parts.push(\"**Bodies:**\");\n parts.push(\"```\");\n parts.push(f.inlineBodies.trimEnd());\n parts.push(\"```\");\n }\n\n if (f.associatedTests?.length) {\n parts.push(\"\");\n parts.push(`**Tests:** ${f.associatedTests.join(\", \")}`);\n }\n\n parts.push(\"\");\n }\n\n if (inputs.recentActivity?.trim()) {\n parts.push(\"---\");\n parts.push(\"## Recent human activity\");\n parts.push(inputs.recentActivity.trim());\n parts.push(\"\");\n }\n\n if (inputs.truncated) {\n parts.push(\"> _(pack truncated to fit budget)_\");\n }\n\n return parts.join(\"\\n\");\n}\n","// Picks the top function bodies from a file (by relevance to the query) and\n// inlines them, respecting a char-based budget. Truncates oversized bodies.\n\nimport { tokenizeQuery } from \"../graph/rank.js\";\nimport type { FileNode, SymbolKind, SymbolNode } from \"../graph/types.js\";\n\nexport interface InlineSelection {\n text: string;\n charsUsed: number;\n functionsInlined: string[];\n}\n\nconst INLINABLE_KINDS = new Set<SymbolKind>([\"function\", \"method\", \"class\"]);\nconst MAX_BODY_CHARS = 1500;\n\nfunction sliceLines(content: string, startLine: number, endLine: number): string {\n const lines = content.split(/\\r?\\n/);\n return lines.slice(Math.max(0, startLine - 1), endLine).join(\"\\n\");\n}\n\nfunction scoreSymbol(name: string, qTokens: Set<string>): number {\n const lower = name.toLowerCase();\n if (qTokens.has(lower)) return 3;\n for (const t of qTokens) {\n if (lower.includes(t) || t.includes(lower)) return 1;\n }\n return 0;\n}\n\nfunction truncate(body: string): string {\n if (body.length <= MAX_BODY_CHARS) return body;\n return body.slice(0, MAX_BODY_CHARS).trimEnd() + \"\\n // … truncated\";\n}\n\nexport function selectInlineBodies(\n file: FileNode,\n symbols: SymbolNode[],\n query: string,\n budgetChars: number,\n): InlineSelection {\n if (budgetChars <= 0) {\n return { text: \"\", charsUsed: 0, functionsInlined: [] };\n }\n\n const qTokens = new Set(tokenizeQuery(query));\n const mine = symbols.filter((s) => s.file === file.path && INLINABLE_KINDS.has(s.symbol_kind));\n\n const scored = mine\n .map((s) => ({ sym: s, score: scoreSymbol(s.name, qTokens) }))\n .sort((a, b) => {\n if (b.score !== a.score) return b.score - a.score;\n // Tie-break: smaller bodies first (so we fit more)\n const aSpan = (a.sym.end_line - a.sym.start_line) || 1;\n const bSpan = (b.sym.end_line - b.sym.start_line) || 1;\n return aSpan - bSpan;\n });\n\n const parts: string[] = [];\n const inlined: string[] = [];\n let used = 0;\n\n for (const { sym, score } of scored) {\n // Skip irrelevant symbols entirely when we have positive hits available;\n // fall back to top-by-size if no query match landed.\n if (score === 0 && inlined.length > 0) break;\n\n const body = truncate(sliceLines(file.content, sym.start_line, sym.end_line));\n const header = `${file.path}::${sym.name} (L${sym.start_line}-${sym.end_line})`;\n const block = `${header}\\n${body}\\n`;\n if (used + block.length > budgetChars) {\n if (inlined.length > 0) break;\n // No room and nothing yet — take a head-only slice.\n const remaining = Math.max(0, budgetChars - used - header.length - 16);\n if (remaining <= 0) break;\n const partial = body.slice(0, remaining).trimEnd() + \"\\n // … truncated\";\n const finalBlock = `${header}\\n${partial}\\n`;\n parts.push(finalBlock);\n inlined.push(sym.name);\n used += finalBlock.length;\n break;\n }\n parts.push(block);\n inlined.push(sym.name);\n used += block.length;\n }\n\n return { text: parts.join(\"\\n\"), charsUsed: used, functionsInlined: inlined };\n}\n","// Extracts function/class signatures from a FileNode (no bodies).\n// Signatures are the first line of each symbol in the file, sorted by line.\n\nimport type { FileNode, SymbolNode } from \"../graph/types.js\";\n\nexport function extractSignatures(file: FileNode, symbols: SymbolNode[]): string[] {\n const mine = symbols\n .filter((s) => s.file === file.path)\n .slice()\n .sort((a, b) => a.start_line - b.start_line);\n\n return mine.map((s) => `L${s.start_line}: ${s.signature.trim()}`);\n}\n","// Test ↔ source co-retrieval. Given a source file, returns the test files\n// linked to it via `tests` edges in the graph (foo.test.ts → foo.ts).\n\nimport type { FileNode, GraphSchema } from \"../graph/types.js\";\n\nexport function findTestsForFile(graph: GraphSchema, file: FileNode): FileNode[] {\n // tests edges run from test file → source file\n const fileNodesById = new Map<string, FileNode>();\n for (const n of graph.nodes) {\n if (n.kind === \"file\") fileNodesById.set(n.id, n);\n }\n\n const out: FileNode[] = [];\n for (const e of graph.edges) {\n if (e.kind !== \"tests\" || e.to !== file.id) continue;\n const testFile = fileNodesById.get(e.from);\n if (testFile && !out.includes(testFile)) out.push(testFile);\n }\n return out;\n}\n","// Compresses a list of retrieved files into a structured context pack:\n// signatures + top function bodies + tests co-retrieved + dependency edges.\n// Budget is enforced in characters (~ tokens × 4) — see SYN_HARD_MAX_READ_CHARS.\n\nimport type { FileNode, GraphSchema, SymbolNode } from \"../graph/types.js\";\nimport { formatPack, type FormatFileSection } from \"./format.js\";\nimport { selectInlineBodies } from \"./inline.js\";\nimport { extractSignatures } from \"./signatures.js\";\nimport { findTestsForFile } from \"./tests.js\";\n\nexport interface PackOptions {\n query: string;\n graph: GraphSchema;\n /** Soft target for total tokens (≈ chars/4). Default: 4000. */\n budgetTokens?: number;\n /** Fraction of remaining budget to spend on a single file's inline bodies. Default: 0.5. */\n inlineBodyRatio?: number;\n /** Co-retrieve linked test files when packing source files. Default: true. */\n includeTests?: boolean;\n /** Optional: file path → reason string from the ranker, surfaced in the pack heading. */\n reasons?: Map<string, string>;\n}\n\nexport interface ContextPack {\n text: string;\n tokenEstimate: number;\n filesUsed: string[];\n testsCoRetrieved: string[];\n truncated: boolean;\n}\n\nconst STATIC_OVERHEAD_PER_FILE = 200; // headers + bullet markdown + spacing\nconst MAX_INLINE_CHARS_PER_FILE = 2500;\n\nfunction indexSymbolsByFile(graph: GraphSchema): SymbolNode[] {\n return graph.nodes.filter((n): n is SymbolNode => n.kind === \"symbol\");\n}\n\nexport async function pack(files: FileNode[], opts: PackOptions): Promise<ContextPack> {\n const budgetTokens = opts.budgetTokens ?? 4000;\n const budgetChars = budgetTokens * 4;\n const inlineRatio = opts.inlineBodyRatio ?? 0.5;\n const includeTests = opts.includeTests ?? true;\n const reasons = opts.reasons ?? new Map<string, string>();\n\n const symbols = indexSymbolsByFile(opts.graph);\n\n const sections: FormatFileSection[] = [];\n const testsCoRetrieved: string[] = [];\n let used = 0;\n let truncated = false;\n\n for (const file of files) {\n const sig = extractSignatures(file, symbols);\n const testFiles = includeTests ? findTestsForFile(opts.graph, file) : [];\n const testPaths = testFiles.map((t) => t.path);\n\n const staticCost =\n file.path.length +\n sig.join(\"\\n\").length +\n testPaths.join(\",\").length +\n STATIC_OVERHEAD_PER_FILE;\n\n if (used + staticCost > budgetChars) {\n truncated = true;\n break;\n }\n\n const remaining = budgetChars - used - staticCost;\n const inlineBudget = Math.min(Math.floor(remaining * inlineRatio), MAX_INLINE_CHARS_PER_FILE);\n\n const inline = selectInlineBodies(file, symbols, opts.query, inlineBudget);\n\n sections.push({\n path: file.path,\n reason: reasons.get(file.path),\n signatures: sig,\n inlineBodies: inline.text,\n associatedTests: testPaths,\n });\n\n used += staticCost + inline.charsUsed;\n for (const t of testPaths) if (!testsCoRetrieved.includes(t)) testsCoRetrieved.push(t);\n\n if (used >= budgetChars) {\n truncated = true;\n break;\n }\n }\n\n if (sections.length < files.length) truncated = true;\n\n const text = formatPack({\n query: opts.query,\n files: sections,\n truncated,\n });\n const tokenEstimate = Math.ceil(text.length / 4);\n\n return {\n text,\n tokenEstimate,\n filesUsed: sections.map((s) => s.path),\n testsCoRetrieved,\n truncated,\n };\n}\n","// MCP-over-HTTP (streamable) protocol handler. Exposes Synthra's graph tools\n// to Claude Code via JSON-RPC 2.0 messages POSTed to /mcp.\n//\n// Tools:\n// graph_continue(query) — retrieve + pack a context bundle\n// graph_read(target) — return source for \"file\" or \"file::symbol\"\n// graph_register_edit(files) — Claude tells Synthra it edited files\n//\n// Spec: https://modelcontextprotocol.io/specification\n\nimport { retrieve } from \"../graph/retrieve.js\";\nimport type { FileNode, SymbolNode } from \"../graph/types.js\";\nimport { recallEntries, rememberEntry } from \"../memory/index.js\";\nimport type { EntryKind } from \"../memory/context-store.js\";\nimport { pack } from \"../packer/index.js\";\nimport type { ServerContext } from \"./context.js\";\n\nconst PROTOCOL_VERSION = \"2024-11-05\";\nconst SERVER_INFO = { name: \"synthra\", version: \"0.0.1\" } as const;\n\ntype JsonRpcId = string | number | null;\n\nexport interface JsonRpcRequest {\n jsonrpc: \"2.0\";\n id?: JsonRpcId;\n method: string;\n params?: Record<string, unknown>;\n}\n\nexport interface JsonRpcResponse {\n jsonrpc: \"2.0\";\n id: JsonRpcId;\n result?: unknown;\n error?: { code: number; message: string; data?: unknown };\n}\n\nconst ERR = {\n parse: -32700,\n invalidRequest: -32600,\n methodNotFound: -32601,\n invalidParams: -32602,\n internal: -32603,\n} as const;\n\nfunction ok(id: JsonRpcId, result: unknown): JsonRpcResponse {\n return { jsonrpc: \"2.0\", id, result };\n}\n\nfunction err(id: JsonRpcId, code: number, message: string, data?: unknown): JsonRpcResponse {\n return { jsonrpc: \"2.0\", id, error: { code, message, data } };\n}\n\nfunction textContent(text: string) {\n return { content: [{ type: \"text\", text }], isError: false };\n}\n\nfunction errorContent(message: string) {\n return { content: [{ type: \"text\", text: message }], isError: true };\n}\n\nconst TOOLS = [\n {\n name: \"graph_continue\",\n description:\n \"Returns the project context most relevant to a query — function signatures, top function bodies, and linked test files. Use this BEFORE Grep/Glob. If `confidence` is 'high', do not call Grep/Glob for the same query.\",\n inputSchema: {\n type: \"object\",\n properties: {\n query: { type: \"string\", description: \"Natural-language description of what you're looking for.\" },\n },\n required: [\"query\"],\n },\n },\n {\n name: \"graph_read\",\n description:\n \"Return the source code for a specific file or symbol. Target is either a project-relative file path (e.g. 'src/auth.ts') or 'file::symbol' (e.g. 'src/auth.ts::AuthService').\",\n inputSchema: {\n type: \"object\",\n properties: {\n target: { type: \"string\", description: \"File path or file::symbol notation.\" },\n },\n required: [\"target\"],\n },\n },\n {\n name: \"graph_register_edit\",\n description:\n \"Tell Synthra that you (the AI) have edited these files. Lets Synthra rank them higher in subsequent retrieval and avoid surfacing stale context.\",\n inputSchema: {\n type: \"object\",\n properties: {\n files: {\n type: \"array\",\n items: { type: \"string\" },\n description: \"Project-relative file paths that were edited.\",\n },\n },\n required: [\"files\"],\n },\n },\n {\n name: \"context_remember\",\n description:\n \"Persist a decision/task/next-step/fact/blocker into the project's branch-aware context store. Use when the user makes a decision worth keeping, identifies a TODO, or surfaces a key fact. Entries land in `.synthra/context-store.json` on the default branch, or `.synthra/branches/<sanitized>/context-store.json` on a feature branch — git-tracked, so teammates inherit them and they merge naturally.\",\n inputSchema: {\n type: \"object\",\n properties: {\n text: { type: \"string\", description: \"The thing to remember (1–3 sentences).\" },\n kind: {\n type: \"string\",\n enum: [\"decision\", \"task\", \"next\", \"fact\", \"blocker\"],\n description: \"What kind of entry. Required.\",\n },\n tags: {\n type: \"array\",\n items: { type: \"string\" },\n description: \"Optional tags for grouping (e.g. 'auth', 'perf').\",\n },\n files: {\n type: \"array\",\n items: { type: \"string\" },\n description: \"Optional project-relative file paths this entry relates to.\",\n },\n },\n required: [\"text\", \"kind\"],\n },\n },\n {\n name: \"context_recall\",\n description:\n \"Read previously-stored decisions/tasks/facts from the project's branch-aware context store. Defaults to the current branch.\",\n inputSchema: {\n type: \"object\",\n properties: {\n kind: {\n type: \"string\",\n enum: [\"decision\", \"task\", \"next\", \"fact\", \"blocker\"],\n description: \"Filter to a single kind.\",\n },\n branch: { type: \"string\", description: \"Override which branch to read from.\" },\n limit: { type: \"number\", description: \"Return only the most recent N entries.\" },\n },\n },\n },\n {\n name: \"recent_activity\",\n description:\n \"What has the human been doing in the editor recently — file saves, branch switches, and uncommitted-diff changes. Use this to check whether the static context pack may be stale (e.g. before answering a question about a file that was just edited).\",\n inputSchema: {\n type: \"object\",\n properties: {\n since_ms: {\n type: \"number\",\n description:\n \"Epoch milliseconds. Only return events newer than this. Defaults to the last 60 minutes.\",\n },\n limit: { type: \"number\", description: \"Cap on returned events.\" },\n },\n },\n },\n {\n name: \"count_tokens\",\n description:\n \"Estimate token count for a piece of text using a char/4 approximation. Accurate within ~10% for English + code. Useful for budgeting prompt content before sending.\",\n inputSchema: {\n type: \"object\",\n properties: {\n text: { type: \"string\", description: \"The text to estimate tokens for.\" },\n },\n required: [\"text\"],\n },\n },\n {\n name: \"blast_radius\",\n description:\n \"Given a file (or 'file::symbol' target), return all files that depend on it transitively via imports + tests edges. Use BEFORE editing a widely-used file to see what could break. Symbol-level granularity is approximated at the file level (we don't track call edges in v0.1).\",\n inputSchema: {\n type: \"object\",\n properties: {\n target: { type: \"string\", description: \"File path or 'file::symbol' notation.\" },\n depth: { type: \"number\", description: \"Max hops to traverse. Default 3.\" },\n },\n required: [\"target\"],\n },\n },\n {\n name: \"dead_code\",\n description:\n \"Return files in the project that no other file imports and no test file references — strong candidates for unused/orphaned code. File-level granularity (v0.1 limitation — symbol-level needs call-graph edges). Common entry-point patterns (main, index, app, CLI, bin/) are excluded heuristically.\",\n inputSchema: {\n type: \"object\",\n properties: {\n limit: { type: \"number\", description: \"Cap on returned files. Default 50.\" },\n },\n },\n },\n] as const;\n\nasync function callTool(\n name: string,\n args: Record<string, unknown> | undefined,\n ctx: ServerContext,\n) {\n switch (name) {\n case \"graph_continue\":\n return graphContinue(args, ctx);\n case \"graph_read\":\n return graphRead(args, ctx);\n case \"graph_register_edit\":\n return graphRegisterEdit(args, ctx);\n case \"context_remember\":\n return contextRemember(args, ctx);\n case \"context_recall\":\n return contextRecall(args, ctx);\n case \"recent_activity\":\n return recentActivity(args, ctx);\n case \"count_tokens\":\n return countTokens(args);\n case \"blast_radius\":\n return blastRadius(args, ctx);\n case \"dead_code\":\n return deadCode(args, ctx);\n default:\n return errorContent(`Unknown tool: ${name}`);\n }\n}\n\nfunction countTokens(args: Record<string, unknown> | undefined) {\n const text = typeof args?.text === \"string\" ? args.text : \"\";\n if (!text) return errorContent(\"count_tokens: 'text' (string) is required\");\n const tokens = Math.ceil(text.length / 4);\n return textContent(JSON.stringify({ tokens, method: \"chars/4 estimate\", chars: text.length }));\n}\n\nfunction blastRadius(args: Record<string, unknown> | undefined, ctx: ServerContext) {\n const targetRaw = typeof args?.target === \"string\" ? args.target.trim() : \"\";\n const maxDepth = typeof args?.depth === \"number\" && args.depth > 0 ? Math.floor(args.depth) : 3;\n if (!targetRaw) return errorContent(\"blast_radius: 'target' (string) is required\");\n\n const filePath = targetRaw.split(\"::\", 1)[0]?.trim() ?? targetRaw;\n const root = ctx.graph.nodes.find(\n (n): n is FileNode => n.kind === \"file\" && n.path === filePath,\n );\n if (!root) return errorContent(`blast_radius: file not in graph: ${filePath}`);\n\n // Index reverse edges (to → [{from, kind}]) once per call.\n const incoming = new Map<string, Array<{ from: string; kind: string }>>();\n for (const e of ctx.graph.edges) {\n if (e.kind !== \"imports\" && e.kind !== \"tests\") continue;\n const list = incoming.get(e.to) ?? [];\n list.push({ from: e.from, kind: e.kind });\n incoming.set(e.to, list);\n }\n\n interface Hit {\n path: string;\n depth: number;\n via: string;\n }\n\n const visited = new Set<string>([root.id]);\n const hits: Hit[] = [];\n const pathById = new Map<string, string>();\n for (const n of ctx.graph.nodes) if (n.kind === \"file\") pathById.set(n.id, n.path);\n\n let frontier = [root.id];\n for (let d = 1; d <= maxDepth; d++) {\n const next: string[] = [];\n for (const cur of frontier) {\n const callers = incoming.get(cur) ?? [];\n for (const c of callers) {\n if (visited.has(c.from)) continue;\n visited.add(c.from);\n next.push(c.from);\n const path = pathById.get(c.from) ?? c.from;\n hits.push({ path, depth: d, via: c.kind });\n }\n }\n frontier = next;\n if (next.length === 0) break;\n }\n\n if (hits.length === 0) {\n return textContent(`# Blast radius for ${filePath}\\n\\n_(no dependents — file is isolated)_`);\n }\n\n hits.sort((a, b) => a.depth - b.depth || a.path.localeCompare(b.path));\n const lines = [`# Blast radius for ${filePath} (depth ≤ ${maxDepth})`, \"\"];\n lines.push(`${hits.length} dependent file(s):`);\n for (const h of hits) {\n lines.push(`- **depth ${h.depth}** \\`${h.path}\\` _(via ${h.via})_`);\n }\n return textContent(lines.join(\"\\n\"));\n}\n\nconst LIKELY_ENTRY_PATTERNS = [\n /(?:^|\\/)main\\.[a-z0-9_]+$/i,\n /(?:^|\\/)index\\.[a-z0-9_]+$/i,\n /(?:^|\\/)app\\.[a-z0-9_]+$/i,\n /(?:^|\\/)entry\\.[a-z0-9_]+$/i,\n /(?:^|\\/)cli[\\/.]/i,\n /(?:^|\\/)bin[\\/.]/i,\n /(?:^|\\/)server\\.[a-z0-9_]+$/i,\n /\\.test\\.[a-z0-9_]+$/i,\n /\\.spec\\.[a-z0-9_]+$/i,\n /(?:^|\\/)tests?\\//i,\n /(?:^|\\/)__tests__\\//i,\n /(?:^|\\/)__init__\\.py$/i,\n];\n\nfunction isLikelyEntry(path: string): boolean {\n return LIKELY_ENTRY_PATTERNS.some((re) => re.test(path));\n}\n\nfunction deadCode(args: Record<string, unknown> | undefined, ctx: ServerContext) {\n const limit = typeof args?.limit === \"number\" && args.limit > 0 ? Math.floor(args.limit) : 50;\n\n const hasIncoming = new Set<string>();\n for (const e of ctx.graph.edges) {\n if (e.kind === \"imports\" || e.kind === \"tests\") hasIncoming.add(e.to);\n }\n\n const candidates = ctx.graph.nodes\n .filter((n): n is FileNode => n.kind === \"file\")\n .filter((f) => !hasIncoming.has(f.id))\n .filter((f) => !isLikelyEntry(f.path));\n\n if (candidates.length === 0) {\n return textContent(\n `# Dead code\\n\\n_(no file is unreferenced — every file is either imported by another, has a linked test, or matches an entry-point pattern)_`,\n );\n }\n\n candidates.sort((a, b) => a.path.localeCompare(b.path));\n const shown = candidates.slice(0, limit);\n const lines = [`# Dead code candidates (file-level, v0.1)`, \"\"];\n lines.push(\n `${shown.length} of ${candidates.length} unreferenced file(s) — no other file imports them and no test links them:`,\n );\n lines.push(\"\");\n for (const f of shown) {\n lines.push(`- \\`${f.path}\\``);\n }\n lines.push(\"\");\n lines.push(\n `_v0.1 caveat:_ this is file-level only. Symbol-level dead code (unused exports) needs call-graph edges, which land in v0.2.`,\n );\n return textContent(lines.join(\"\\n\"));\n}\n\nasync function graphContinue(args: Record<string, unknown> | undefined, ctx: ServerContext) {\n const query = typeof args?.query === \"string\" ? args.query : \"\";\n if (!query) return errorContent(\"graph_continue: 'query' (string) is required\");\n\n const retrieval = await retrieve(ctx.graph, query);\n const packed = await pack(retrieval.files, { query, graph: ctx.graph });\n\n const header =\n `Confidence: ${retrieval.confidence}\\n` +\n `Files: ${retrieval.files.map((f) => f.path).join(\", \") || \"(none)\"}\\n` +\n `Reason: ${retrieval.reason}\\n`;\n\n // The pack body already starts with a header — keep them concatenated.\n return textContent(`${header}\\n${packed.text}`);\n}\n\nfunction graphRead(args: Record<string, unknown> | undefined, ctx: ServerContext) {\n const target = typeof args?.target === \"string\" ? args.target : \"\";\n if (!target) return errorContent(\"graph_read: 'target' (string) is required\");\n\n const [rawFile, symbolName] = target.includes(\"::\") ? target.split(\"::\", 2) : [target, undefined];\n const filePath = (rawFile ?? \"\").trim();\n\n const fileNode = ctx.graph.nodes.find(\n (n): n is FileNode => n.kind === \"file\" && n.path === filePath,\n );\n if (!fileNode) return errorContent(`graph_read: file not found in graph: ${filePath}`);\n\n if (!symbolName) {\n return textContent(`# ${fileNode.path}\\n\\n${fileNode.content}`);\n }\n\n const cleanSym = symbolName.trim();\n const symbol = ctx.graph.nodes.find(\n (n): n is SymbolNode => n.kind === \"symbol\" && n.file === filePath && n.name === cleanSym,\n );\n if (!symbol) {\n return errorContent(`graph_read: symbol '${cleanSym}' not found in ${filePath}`);\n }\n\n const lines = fileNode.content.split(/\\r?\\n/);\n const body = lines.slice(symbol.start_line - 1, symbol.end_line).join(\"\\n\");\n return textContent(\n `# ${fileNode.path}::${symbol.name} (L${symbol.start_line}-${symbol.end_line})\\n\\n${body}`,\n );\n}\n\nconst editedFiles = new Set<string>();\n\nfunction graphRegisterEdit(args: Record<string, unknown> | undefined, _ctx: ServerContext) {\n const files = Array.isArray(args?.files) ? (args.files as unknown[]).filter((f) => typeof f === \"string\") : [];\n for (const f of files) editedFiles.add(f as string);\n return textContent(`Registered ${files.length} edited file(s). Total tracked this session: ${editedFiles.size}.`);\n}\n\nexport function getRegisteredEdits(): string[] {\n return Array.from(editedFiles);\n}\n\nconst VALID_KINDS = new Set<EntryKind>([\"decision\", \"task\", \"next\", \"fact\", \"blocker\"]);\n\nasync function contextRemember(args: Record<string, unknown> | undefined, ctx: ServerContext) {\n const text = typeof args?.text === \"string\" ? args.text.trim() : \"\";\n const kindRaw = typeof args?.kind === \"string\" ? args.kind : \"\";\n if (!text) return errorContent(\"context_remember: 'text' (string) is required\");\n if (!VALID_KINDS.has(kindRaw as EntryKind)) {\n return errorContent(\n `context_remember: 'kind' must be one of ${Array.from(VALID_KINDS).join(\", \")}`,\n );\n }\n const tags = Array.isArray(args?.tags)\n ? (args.tags as unknown[]).filter((t): t is string => typeof t === \"string\")\n : [];\n const files = Array.isArray(args?.files)\n ? (args.files as unknown[]).filter((f): f is string => typeof f === \"string\")\n : [];\n\n const result = await rememberEntry(ctx.paths, {\n text,\n kind: kindRaw as EntryKind,\n tags,\n files,\n });\n\n return textContent(\n `Remembered ${result.entry.type} on branch '${result.branch}'.\\n` +\n `Stored: ${result.storePath}\\n` +\n `CONTEXT.md refreshed: ${result.contextMdPath}`,\n );\n}\n\nconst DEFAULT_RECENT_WINDOW_MS = 60 * 60 * 1000;\n\nfunction recentActivity(args: Record<string, unknown> | undefined, ctx: ServerContext) {\n const sinceMs =\n typeof args?.since_ms === \"number\" && Number.isFinite(args.since_ms)\n ? args.since_ms\n : Date.now() - DEFAULT_RECENT_WINDOW_MS;\n const limit =\n typeof args?.limit === \"number\" && args.limit > 0 ? Math.floor(args.limit) : undefined;\n\n let events = ctx.activity.getEvents(sinceMs);\n if (limit) events = events.slice(-limit);\n\n if (events.length === 0) {\n return textContent(\n `No human-activity events since ${new Date(sinceMs).toISOString()}.`,\n );\n }\n\n const lines = [`# Recent human activity (${events.length} events)`, \"\"];\n for (const e of events) {\n if (\"path\" in e) {\n lines.push(`- **${e.kind}** ${e.path} _(${e.ts})_`);\n } else {\n const summary = JSON.stringify(e.details);\n lines.push(`- **${e.kind}** ${summary} _(${e.ts})_`);\n }\n }\n return textContent(lines.join(\"\\n\"));\n}\n\nasync function contextRecall(args: Record<string, unknown> | undefined, ctx: ServerContext) {\n const kind = typeof args?.kind === \"string\" && VALID_KINDS.has(args.kind as EntryKind)\n ? (args.kind as EntryKind)\n : undefined;\n const branch = typeof args?.branch === \"string\" ? args.branch : undefined;\n const limit = typeof args?.limit === \"number\" && args.limit > 0 ? Math.floor(args.limit) : undefined;\n\n const result = await recallEntries(ctx.paths, { kind, branch, limit });\n\n if (result.entries.length === 0) {\n const filter = kind ? ` of kind '${kind}'` : \"\";\n return textContent(`No context entries${filter} on branch '${result.branch}'.`);\n }\n\n const lines = [`# Context entries — branch: ${result.branch}`, \"\"];\n for (const e of result.entries) {\n const tags = e.tags.length ? ` [${e.tags.join(\", \")}]` : \"\";\n lines.push(`- **${e.type}**${tags} (${e.date}): ${e.content}`);\n if (e.files.length) lines.push(` files: ${e.files.join(\", \")}`);\n }\n return textContent(lines.join(\"\\n\"));\n}\n\nexport async function handleMcpRequest(\n body: unknown,\n ctx: ServerContext,\n): Promise<JsonRpcResponse> {\n if (!body || typeof body !== \"object\") {\n return err(null, ERR.invalidRequest, \"Request body must be a JSON-RPC 2.0 object.\");\n }\n\n const req = body as JsonRpcRequest;\n if (req.jsonrpc !== \"2.0\" || typeof req.method !== \"string\") {\n return err(req.id ?? null, ERR.invalidRequest, \"Invalid JSON-RPC envelope.\");\n }\n\n const id = req.id ?? null;\n\n try {\n switch (req.method) {\n case \"initialize\":\n return ok(id, {\n protocolVersion:\n typeof req.params?.protocolVersion === \"string\"\n ? req.params.protocolVersion\n : PROTOCOL_VERSION,\n capabilities: { tools: {} },\n serverInfo: SERVER_INFO,\n });\n\n case \"notifications/initialized\":\n // Client confirms initialization. No response required for notifications (id===undefined).\n return ok(id, {});\n\n case \"tools/list\":\n return ok(id, { tools: TOOLS });\n\n case \"tools/call\": {\n const params = req.params ?? {};\n const toolName = typeof params.name === \"string\" ? params.name : \"\";\n if (!toolName) return err(id, ERR.invalidParams, \"'name' is required for tools/call.\");\n const args = (params.arguments && typeof params.arguments === \"object\"\n ? (params.arguments as Record<string, unknown>)\n : {});\n const result = await callTool(toolName, args, ctx);\n return ok(id, result);\n }\n\n case \"ping\":\n return ok(id, {});\n\n default:\n return err(id, ERR.methodNotFound, `Method not found: ${req.method}`);\n }\n } catch (e) {\n return err(id, ERR.internal, (e as Error).message);\n }\n}\n\n// Exposed for code that wants to enumerate the tool catalogue without going\n// through JSON-RPC (e.g. CLI introspection in M3).\nexport function listTools(): Array<{ name: string; description: string; inputSchema: unknown }> {\n return TOOLS.map((t) => ({ name: t.name, description: t.description, inputSchema: t.inputSchema }));\n}\n","// Finds a free port in the 8080–8099 range. Writes the chosen port to\n// .synthra-graph/mcp_port so PowerShell/Bash hook scripts can read it.\n// TODO: M2\n\nimport { createServer } from \"node:net\";\n\nexport const PORT_RANGE_START = 8080;\nexport const PORT_RANGE_END = 8099;\n\nexport async function findFreePort(\n start = PORT_RANGE_START,\n end = PORT_RANGE_END,\n): Promise<number> {\n for (let port = start; port <= end; port++) {\n if (await isFree(port)) return port;\n }\n throw new Error(`Synthra: no free port in ${start}-${end}`);\n}\n\nfunction isFree(port: number): Promise<boolean> {\n return new Promise((resolve) => {\n const s = createServer();\n s.once(\"error\", () => resolve(false));\n s.once(\"listening\", () => s.close(() => resolve(true)));\n s.listen(port, \"127.0.0.1\");\n });\n}\n","// GET /activity?since=<ms> — returns recent human-activity events.\n// Backed by the in-memory ActivityStore (file-watcher + git-watcher feed it).\n// MCP tool `recent_activity` is a thin wrapper.\n\nimport type { ActivityEvent } from \"../../activity/activity-log.js\";\nimport type { ServerContext } from \"../context.js\";\n\nexport interface ActivityResponse {\n events: ActivityEvent[];\n since: string;\n ring_size: number;\n}\n\nexport async function handleActivity(\n sinceMs: number | undefined,\n ctx: ServerContext,\n): Promise<ActivityResponse> {\n const events = ctx.activity.getEvents(sinceMs);\n return {\n events,\n since: new Date(sinceMs ?? Date.now()).toISOString(),\n ring_size: ctx.activity.size(),\n };\n}\n","// POST /context-update — Stop hook calls this at session end.\n// For M4: re-renders CONTEXT.md from the branch-scoped store so the narrative\n// stays in sync with the structured entries that landed during the session.\n// Transcript-mining for new entries (auto \"we decided X\" → store) is v0.2.\n\nimport { refreshContextMd } from \"../../memory/index.js\";\nimport type { ServerContext } from \"../context.js\";\n\nexport interface ContextUpdateRequest {\n transcript_path?: string;\n branch?: string;\n}\n\nexport interface ContextUpdateResponse {\n updated: boolean;\n branch: string;\n path: string;\n entries: number;\n}\n\nexport async function handleContextUpdate(\n req: ContextUpdateRequest,\n ctx: ServerContext,\n): Promise<ContextUpdateResponse> {\n const r = await refreshContextMd(ctx.paths, req?.branch);\n return {\n updated: true,\n branch: r.branch,\n path: r.path,\n entries: r.entriesSeen,\n };\n}\n","// POST /gate — PreToolUse hook calls this with the tool name + arguments.\n// THE MOAT — improvement #1. Strategy:\n// - For Grep/Glob: extract the search pattern, run retrieve().\n// - If recent human activity touches a file matching the query → ALLOW\n// even at high confidence (the user's head is in that file; static\n// context may be stale).\n// - If confidence === \"high\" and no recent overlap → BLOCK with a reason\n// pointing at graph_continue / graph_read.\n// - Otherwise → ALLOW.\n\nimport { appendFile, mkdir } from \"node:fs/promises\";\nimport { dirname } from \"node:path\";\n\nimport { retrieve } from \"../../graph/retrieve.js\";\nimport { tokenizeQuery } from \"../../graph/rank.js\";\nimport type { ServerContext } from \"../context.js\";\n\nexport interface GateRequest {\n tool_name: string;\n tool_input: Record<string, unknown>;\n}\n\nexport interface GateResponse {\n decision: \"allow\" | \"block\";\n reason?: string;\n}\n\nconst BLOCKABLE_TOOLS = new Set([\"Grep\", \"Glob\"]);\nconst RECENT_ACTIVITY_WINDOW_MS = 5 * 60 * 1000;\n\nfunction extractQuery(toolName: string, input: Record<string, unknown>): string | null {\n if (toolName === \"Grep\") {\n const pattern = typeof input.pattern === \"string\" ? input.pattern : \"\";\n const query = typeof input.query === \"string\" ? input.query : \"\";\n return (pattern || query).trim() || null;\n }\n if (toolName === \"Glob\") {\n const pattern = typeof input.pattern === \"string\" ? input.pattern : \"\";\n return pattern.replace(/[*?/\\\\.]+/g, \" \").trim() || null;\n }\n return null;\n}\n\n// Heuristic: does this Grep pattern target markup / CSS / attributes / literals\n// rather than a code symbol? The graph only indexes symbols, so blocking these\n// and redirecting to graph_read just forces a fallback Read. Conservative — only\n// fires on syntax that never appears in a bare identifier search.\nfunction looksLikeNonSymbolQuery(pattern: string): boolean {\n // HTML / JSX tag: \"<div\", \"</\", \"<svg\"\n if (/<\\/?[a-zA-Z]/.test(pattern)) return true;\n // Hyphenated attribute assignment: \"data-tour=\", \"aria-label=\" ('-' is not a\n // valid identifier char, so this is markup, not a symbol).\n if (/[a-zA-Z][\\w-]*-[\\w-]*\\s*=/.test(pattern)) return true;\n // CSS rule / object brace: \".content{\", \"{ color\"\n if (/\\{/.test(pattern)) return true;\n // Escaped-dot class / member selector: \"\\.filter-bar\", \"\\.gs\"\n if (/\\\\\\.[a-zA-Z]/.test(pattern)) return true;\n // CSS property value or units: \": 100%\", \"12px\", \"1.5rem\", \"50%\"\n if (/:\\s*\\d/.test(pattern) || /\\d(?:px|rem|em|vh|vw)\\b/.test(pattern) || /\\d%/.test(pattern)) {\n return true;\n }\n return false;\n}\n\nfunction recentlyTouchedMatchesQuery(\n recentPaths: string[],\n queryTokens: Set<string>,\n): string[] {\n const matches: string[] = [];\n for (const path of recentPaths) {\n const lower = path.toLowerCase();\n for (const t of queryTokens) {\n if (lower.includes(t)) {\n matches.push(path);\n break;\n }\n }\n }\n return matches;\n}\n\nasync function logDecision(\n ctx: ServerContext,\n toolName: string,\n query: string | null,\n decision: \"allow\" | \"block\",\n reason: string | undefined,\n): Promise<void> {\n try {\n await mkdir(dirname(ctx.paths.gateLog), { recursive: true });\n const entry = {\n ts: new Date().toISOString(),\n tool: toolName,\n decision,\n query,\n reason,\n };\n await appendFile(ctx.paths.gateLog, JSON.stringify(entry) + \"\\n\", \"utf8\");\n } catch {\n // Durability is best-effort; an unwritable disk shouldn't fail the gate.\n }\n}\n\nexport async function handleGate(req: GateRequest, ctx: ServerContext): Promise<GateResponse> {\n if (!req?.tool_name || typeof req.tool_name !== \"string\") {\n return { decision: \"allow\", reason: \"no tool_name\" };\n }\n\n if (!BLOCKABLE_TOOLS.has(req.tool_name)) {\n return { decision: \"allow\" };\n }\n\n const input = (req.tool_input && typeof req.tool_input === \"object\" ? req.tool_input : {}) as Record<string, unknown>;\n const query = extractQuery(req.tool_name, input);\n if (!query) {\n const res: GateResponse = { decision: \"allow\", reason: \"no extractable query\" };\n await logDecision(ctx, req.tool_name, null, res.decision, res.reason);\n return res;\n }\n\n // Guard 1 — the query targets markup/CSS/attributes/literals, which the graph\n // does not index. Blocking would only force a fallback, so let Grep through.\n if (req.tool_name === \"Grep\" && looksLikeNonSymbolQuery(query)) {\n const res: GateResponse = {\n decision: \"allow\",\n reason: `\"${query}\" targets markup/CSS/attributes, not code symbols — letting Grep through (the graph indexes symbols).`,\n };\n await logDecision(ctx, req.tool_name, query, res.decision, res.reason);\n return res;\n }\n\n const retrieval = await retrieve(ctx.graph, query);\n // \"low\" = no real matches → let Grep through; Synthra has nothing useful.\n // \"medium\" + \"high\" = Synthra has structured context for this query →\n // bias toward blocking. The pitch (\"use graph_continue instead of Grep\")\n // holds at medium too — on real codebases of any size, \"high\" is rare\n // because almost every query matches multiple files.\n if (retrieval.confidence === \"low\") {\n const res: GateResponse = {\n decision: \"allow\",\n reason: `confidence=low — no graph context for \"${query}\", letting ${req.tool_name} through`,\n };\n await logDecision(ctx, req.tool_name, query, res.decision, res.reason);\n return res;\n }\n\n // Medium / high — but check if recent activity overlaps the query first.\n // If the user just touched a file matching the query, static context may\n // be stale and they probably want a fresh search.\n const qTokens = new Set(tokenizeQuery(query));\n const recentPaths = ctx.activity.recentFilePaths(RECENT_ACTIVITY_WINDOW_MS);\n const overlap = recentlyTouchedMatchesQuery(recentPaths, qTokens);\n\n if (overlap.length > 0) {\n const res: GateResponse = {\n decision: \"allow\",\n reason:\n `confidence=${retrieval.confidence} but human just touched ${overlap.slice(0, 3).join(\", \")} — ` +\n `static context may be stale, letting ${req.tool_name} through.`,\n };\n await logDecision(ctx, req.tool_name, query, res.decision, res.reason);\n return res;\n }\n\n // Guard 2 — the graph matched files only by keyword/path, not by a symbol the\n // query names, so graph_read can't return a real slice. A block would just\n // force a fallback Read; let the search through instead.\n if (!retrieval.symbolMatched) {\n const res: GateResponse = {\n decision: \"allow\",\n reason:\n `confidence=${retrieval.confidence} but only keyword/path matched (no symbol the query names) — ` +\n `graph_read can't slice it, letting ${req.tool_name} through.`,\n };\n await logDecision(ctx, req.tool_name, query, res.decision, res.reason);\n return res;\n }\n\n const top = retrieval.files.slice(0, 3).map((f) => f.path).join(\", \");\n const res: GateResponse = {\n decision: \"block\",\n reason:\n `Synthra has ${retrieval.confidence}-confidence context for \"${query}\" (top files: ${top}). ` +\n `Use the \\`graph_continue\\` MCP tool with this query instead of ${req.tool_name}, ` +\n `or read a specific file/symbol with \\`graph_read\\`.`,\n };\n await logDecision(ctx, req.tool_name, query, res.decision, res.reason);\n return res;\n}\n","// POST /log — Stop hook posts per-turn token usage parsed from Claude's\n// transcript JSONL. Synthra appends each entry as one line to token_log.jsonl.\n\nimport { appendFile, mkdir } from \"node:fs/promises\";\nimport { dirname } from \"node:path\";\n\nimport type { ServerContext } from \"../context.js\";\n\nexport interface LogEntry {\n input_tokens: number;\n output_tokens: number;\n cache_creation_input_tokens?: number;\n cache_read_input_tokens?: number;\n model: string;\n description?: string;\n project: string;\n}\n\nexport interface LogResponse {\n ok: true;\n written_at: string;\n}\n\nexport async function handleLog(entry: LogEntry, ctx: ServerContext): Promise<LogResponse> {\n if (!entry || typeof entry.input_tokens !== \"number\" || typeof entry.output_tokens !== \"number\") {\n throw new Error(\"log: input_tokens and output_tokens (number) are required\");\n }\n\n const written_at = new Date().toISOString();\n const record = { ...entry, written_at };\n await mkdir(dirname(ctx.paths.tokenLog), { recursive: true });\n await appendFile(ctx.paths.tokenLog, JSON.stringify(record) + \"\\n\", \"utf8\");\n\n return { ok: true, written_at };\n}\n","// POST /pack { query, maxTokens? } → ContextPack JSON.\n// Runs retrieve → pack against the in-memory graph.\n\nimport { retrieve } from \"../../graph/retrieve.js\";\nimport { scoreFiles } from \"../../graph/rank.js\";\nimport { pack, type ContextPack } from \"../../packer/index.js\";\nimport type { ServerContext } from \"../context.js\";\n\nexport interface PackRequest {\n query: string;\n maxTokens?: number;\n includeTests?: boolean;\n}\n\nexport interface PackResponse extends ContextPack {\n query: string;\n confidence: \"high\" | \"medium\" | \"low\";\n retrievalReason: string;\n}\n\nexport async function handlePack(req: PackRequest, ctx: ServerContext): Promise<PackResponse> {\n if (!req?.query || typeof req.query !== \"string\") {\n throw new Error(\"pack: 'query' (string) is required\");\n }\n\n const recentlyEditedPaths = ctx.activity.recentFilePaths(15 * 60 * 1000);\n const retrieval = await retrieve(ctx.graph, req.query, { recentlyEditedPaths });\n\n // Surface per-file scoring rationale in the rendered pack.\n const allFiles = ctx.graph.nodes.filter((n) => n.kind === \"file\");\n const scored = scoreFiles({\n candidates: allFiles as Parameters<typeof scoreFiles>[0][\"candidates\"],\n query: req.query,\n graph: ctx.graph,\n recentlyEditedPaths,\n });\n const reasons = new Map<string, string>();\n for (const s of scored) {\n if (s.reasons.length) reasons.set(s.file.path, s.reasons.join(\",\"));\n }\n\n const result = await pack(retrieval.files, {\n query: req.query,\n graph: ctx.graph,\n budgetTokens: req.maxTokens,\n includeTests: req.includeTests,\n reasons,\n });\n\n return {\n ...result,\n query: req.query,\n confidence: retrieval.confidence,\n retrievalReason: retrieval.reason,\n };\n}\n","// GET /prime — SessionStart and PreCompact hooks call this. Returns the\n// priming text Claude sees at session start.\n//\n// For M2 we return a minimal primer derived from the graph (file count + a\n// short top-level summary). M3 layers in CONTEXT.md narrative + recent\n// stored decisions.\n\nimport type { ServerContext } from \"../context.js\";\n\nexport interface PrimeResponse {\n primer: string;\n port: number;\n}\n\nexport async function handlePrime(ctx: ServerContext, port: number): Promise<PrimeResponse> {\n const g = ctx.graph;\n const fileCount = g.file_count;\n const symbolCount = g.symbol_count;\n\n const primer =\n `Synthra context loaded for ${g.root}.\\n` +\n `${fileCount} files indexed, ${symbolCount} symbols. ` +\n `Prefer the graph_* MCP tools over Grep/Glob for navigation.\\n` +\n `(Full primer wired in M3.)`;\n\n return { primer, port };\n}\n"],"mappings":"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","names":["writeFile","emit","err","readFile","join","dirname","join","readFile","QUERY","QUERY","Query","QUERY","firstLine","Query","QUERY","QUERY","QUERY","QUERY","Query","QUERY","firstLine","Query","QUERY","QUERY","Query","firstLine","Query","SCRIPT_RE","extractScripts","require","readFile","readFile","join","relative","sep","ignore","readIgnoreFile","buildMatcher","toPosix","mkdir","readFile","dirname","join","mkdir","readFile","stat","writeFile","basename","readFile","writeFile","dirname","stat","mkdir","readFile","writeFile","basename","err","STOPWORDS","score","execFile","readFile","join","promisify","execFileAsync","mkdir","readFile","writeFile","dirname","mkdir","readFile","writeFile","dirname","score","indexSymbolsByFile","resolve","appendFile","mkdir","dirname","mkdir","dirname","appendFile","res","appendFile","mkdir","dirname","err","writeFile","resolve"]}
|
|
1
|
+
{"version":3,"sources":["../../src/server/http.ts","../../src/activity/activity-log.ts","../../src/activity/file-watcher.ts","../../src/shared/logger.ts","../../src/activity/git-watcher.ts","../../src/cli/scan-command.ts","../../src/scanner/extract.ts","../../src/scanner/hash.ts","../../src/scanner/keywords.ts","../../src/scanner/parser.ts","../../src/scanner/parsers/_generic.ts","../../src/scanner/parsers/c.ts","../../src/scanner/parsers/cpp.ts","../../src/scanner/parsers/csharp.ts","../../src/scanner/parsers/dart.ts","../../src/scanner/parsers/go.ts","../../src/scanner/parsers/hubl.ts","../../src/scanner/parsers/java.ts","../../src/scanner/parsers/kotlin.ts","../../src/scanner/parsers/php.ts","../../src/scanner/parsers/python.ts","../../src/scanner/parsers/ruby.ts","../../src/scanner/parsers/rust.ts","../../src/scanner/parsers/typescript.ts","../../src/scanner/parsers/svelte.ts","../../src/scanner/parsers/vue.ts","../../src/scanner/walker.ts","../../src/graph/store.ts","../../src/shared/paths.ts","../../src/cli/bootstrap.ts","../../src/hooks/claude-md.ts","../../src/graph/rank.ts","../../src/graph/retrieve.ts","../../src/memory/branches.ts","../../src/memory/context-md.ts","../../src/memory/context-store.ts","../../src/memory/index.ts","../../src/packer/format.ts","../../src/packer/inline.ts","../../src/packer/signatures.ts","../../src/packer/tests.ts","../../src/packer/index.ts","../../src/server/mcp.ts","../../src/server/port.ts","../../src/server/routes/activity.ts","../../src/server/routes/context-update.ts","../../src/server/routes/gate.ts","../../src/server/routes/log.ts","../../src/server/routes/pack.ts","../../src/server/routes/prime.ts"],"sourcesContent":["// HTTP server (Hono). Hosts the routes hooks need (/prime, /pack, /log,\n// /gate, /activity) and serves the loaded graph from memory. The MCP-protocol\n// envelope (/mcp endpoint, JSON-RPC) is wired in M3.\n\nimport { serve } from \"@hono/node-server\";\nimport { Hono } from \"hono\";\nimport { writeFile } from \"node:fs/promises\";\n\nimport { ActivityStore } from \"../activity/activity-log.js\";\nimport { createFileWatcher, type FileWatcher } from \"../activity/file-watcher.js\";\nimport { createGitWatcher, type GitWatcher } from \"../activity/git-watcher.js\";\nimport { scanProject } from \"../cli/scan-command.js\";\nimport { readGraph, readSymbolIndex } from \"../graph/store.js\";\nimport { log } from \"../shared/logger.js\";\nimport type { SynthraPaths } from \"../shared/paths.js\";\nimport type { ServerContext } from \"./context.js\";\nimport { handleMcpRequest } from \"./mcp.js\";\nimport { findFreePort } from \"./port.js\";\nimport { handleActivity } from \"./routes/activity.js\";\nimport { handleContextUpdate } from \"./routes/context-update.js\";\nimport { handleGate } from \"./routes/gate.js\";\nimport { handleLog } from \"./routes/log.js\";\nimport { handlePack } from \"./routes/pack.js\";\nimport { handlePrime } from \"./routes/prime.js\";\n\nexport interface ServerHandle {\n port: number;\n url: string;\n stop(): Promise<void>;\n}\n\nexport interface StartOptions {\n /** Override the port range search. */\n port?: number;\n}\n\nasync function loadContext(paths: SynthraPaths): Promise<ServerContext> {\n try {\n const [graph, symbolIndex] = await Promise.all([\n readGraph(paths.infoGraph),\n readSymbolIndex(paths.symbolIndex),\n ]);\n const activity = new ActivityStore(paths.activityLog);\n return { paths, graph, symbolIndex, activity };\n } catch (err) {\n throw new Error(\n `failed to load graph from ${paths.infoGraph}: ${(err as Error).message}. ` +\n `Run \\`syn scan\\` first.`,\n );\n }\n}\n\nfunction buildApp(ctx: ServerContext, port: number): Hono {\n const app = new Hono();\n\n app.get(\"/\", (c) =>\n c.json({\n service: \"synthra\",\n version: \"0.0.1\",\n port,\n file_count: ctx.graph.file_count,\n symbol_count: ctx.graph.symbol_count,\n generated_at: ctx.graph.generated_at,\n }),\n );\n\n app.get(\"/health\", (c) => c.json({ ok: true }));\n\n app.get(\"/prime\", async (c) => c.json(await handlePrime(ctx, port)));\n\n app.post(\"/pack\", async (c) => {\n const body = await c.req.json().catch(() => ({}));\n return c.json(await handlePack(body, ctx));\n });\n\n app.post(\"/log\", async (c) => {\n const body = await c.req.json().catch(() => ({}));\n return c.json(await handleLog(body, ctx));\n });\n\n app.post(\"/gate\", async (c) => {\n const body = await c.req.json().catch(() => ({}));\n return c.json(await handleGate(body, ctx));\n });\n\n app.get(\"/activity\", async (c) => {\n const sinceParam = c.req.query(\"since\");\n const sinceMs = sinceParam ? Number(sinceParam) : undefined;\n return c.json(\n await handleActivity(Number.isFinite(sinceMs) ? sinceMs : undefined, ctx),\n );\n });\n\n app.post(\"/context-update\", async (c) => {\n const body = await c.req.json().catch(() => ({}));\n return c.json(await handleContextUpdate(body, ctx));\n });\n\n app.post(\"/mcp\", async (c) => {\n const body = await c.req.json().catch(() => null);\n return c.json(await handleMcpRequest(body, ctx));\n });\n\n app.onError((err, c) => {\n log.error(\"route error:\", err.message);\n return c.json({ error: err.message }, 400);\n });\n\n return app;\n}\n\nexport async function startServer(\n paths: SynthraPaths,\n options: StartOptions = {},\n): Promise<ServerHandle> {\n const ctx = await loadContext(paths);\n const port = options.port ?? (await findFreePort());\n\n const app = buildApp(ctx, port);\n const nodeServer = serve({ fetch: app.fetch, port, hostname: \"127.0.0.1\" });\n\n await writeFile(paths.mcpPort, String(port), \"utf8\");\n\n // Spin up the human-activity watchers. Both are best-effort — if chokidar\n // can't watch (e.g. unsupported FS) or .git is missing, they no-op silently.\n const fileWatcher: FileWatcher = createFileWatcher(paths.projectRoot, (e) =>\n ctx.activity.add(e),\n );\n const gitWatcher: GitWatcher = createGitWatcher(paths.projectRoot, async (e) => {\n await ctx.activity.add(e);\n // Per-branch graph: rebuild on branch switch so the in-memory graph\n // matches whichever branch is currently checked out.\n if (e.kind === \"branch-switch\") {\n try {\n const to = (e.details as { to?: string } | undefined)?.to ?? \"unknown\";\n log.info(`branch switched to '${to}' — rebuilding graph…`);\n await scanProject(paths.projectRoot, { silent: true });\n const [g, idx] = await Promise.all([\n readGraph(paths.infoGraph),\n readSymbolIndex(paths.symbolIndex),\n ]);\n ctx.graph = g;\n ctx.symbolIndex = idx;\n log.info(`graph rebuilt for '${to}' (${g.symbol_count} symbols).`);\n } catch (err) {\n log.warn(`branch rescan failed: ${(err as Error).message}`);\n }\n }\n });\n try {\n await fileWatcher.start();\n } catch (err) {\n log.warn(`file watcher failed to start: ${(err as Error).message}`);\n }\n try {\n await gitWatcher.start();\n } catch (err) {\n log.warn(`git watcher failed to start: ${(err as Error).message}`);\n }\n\n const url = `http://127.0.0.1:${port}`;\n\n return {\n port,\n url,\n async stop() {\n await fileWatcher.stop().catch(() => undefined);\n await gitWatcher.stop().catch(() => undefined);\n await new Promise<void>((resolve, reject) => {\n nodeServer.close((err) => (err ? reject(err) : resolve()));\n });\n },\n };\n}\n","// Rolling JSONL log of human activity, written to .synthra-graph/activity.jsonl.\n// In-memory ring buffer for fast queries; disk append for durability.\n//\n// The buffer is bounded (defaults to 100 events) so we don't unbounded-grow\n// memory in long sessions. Disk gets every event so the dashboard / future\n// audit tooling can replay history.\n\nimport { appendFile, mkdir } from \"node:fs/promises\";\nimport { dirname } from \"node:path\";\n\nexport interface FileEvent {\n kind: \"save\" | \"create\" | \"delete\";\n path: string;\n ts: string;\n}\n\nexport interface GitEvent {\n kind: \"branch-switch\" | \"stage\" | \"unstage\" | \"diff-change\";\n details: Record<string, unknown>;\n ts: string;\n}\n\nexport type ActivityEvent = FileEvent | GitEvent;\n\nconst DEFAULT_RING_SIZE = 100;\n\nexport class ActivityStore {\n private ring: ActivityEvent[] = [];\n private readonly maxRingSize: number;\n private readonly persistPath: string;\n\n constructor(persistPath: string, maxRingSize = DEFAULT_RING_SIZE) {\n this.persistPath = persistPath;\n this.maxRingSize = maxRingSize;\n }\n\n async add(event: ActivityEvent): Promise<void> {\n this.ring.push(event);\n while (this.ring.length > this.maxRingSize) this.ring.shift();\n await this.persist(event);\n }\n\n /** Get events newer than `sinceMs` (epoch ms). If omitted, returns the full ring. */\n getEvents(sinceMs?: number): ActivityEvent[] {\n if (!sinceMs || !Number.isFinite(sinceMs)) return this.ring.slice();\n const cutoff = new Date(sinceMs).toISOString();\n return this.ring.filter((e) => e.ts >= cutoff);\n }\n\n /** Project-relative file paths that have a save/create event newer than `maxAgeMs` ms ago. */\n recentFilePaths(maxAgeMs: number): string[] {\n const cutoff = new Date(Date.now() - maxAgeMs).toISOString();\n const out = new Set<string>();\n for (const e of this.ring) {\n if (\"path\" in e && (e.kind === \"save\" || e.kind === \"create\") && e.ts >= cutoff) {\n out.add(e.path);\n }\n }\n return Array.from(out);\n }\n\n size(): number {\n return this.ring.length;\n }\n\n private async persist(event: ActivityEvent): Promise<void> {\n try {\n await mkdir(dirname(this.persistPath), { recursive: true });\n await appendFile(this.persistPath, JSON.stringify(event) + \"\\n\", \"utf8\");\n } catch {\n // Durability is best-effort; an unwritable disk shouldn't crash the server.\n }\n }\n}\n","// chokidar-based file watcher. Emits save/create/delete events for human\n// edits inside the project. Respects .gitignore + .synthraignore plus a\n// hard-coded list of always-ignored directories (.git, .synthra*, .claude,\n// node_modules, dist, build, coverage).\n\nimport chokidar, { type FSWatcher } from \"chokidar\";\nimport { readFile } from \"node:fs/promises\";\nimport { join, relative, sep } from \"node:path\";\nimport ignore, { type Ignore } from \"ignore\";\n\nimport { log } from \"../shared/logger.js\";\nimport type { FileEvent } from \"./activity-log.js\";\n\nconst ALWAYS_IGNORE = [\n \".git\",\n \".synthra\",\n \".synthra-graph\",\n \".claude\",\n \"node_modules\",\n \"dist\",\n \"build\",\n \"out\",\n \"coverage\",\n \".next\",\n \".nuxt\",\n \".svelte-kit\",\n \".turbo\",\n \".cache\",\n \".vscode\",\n \".idea\",\n];\n\nexport interface FileWatcher {\n start(): Promise<void>;\n stop(): Promise<void>;\n}\n\nexport type FileEventHandler = (e: FileEvent) => void | Promise<void>;\n\nasync function readIgnoreFile(path: string): Promise<string[]> {\n try {\n const text = await readFile(path, \"utf8\");\n return text\n .split(/\\r?\\n/)\n .map((l) => l.trim())\n .filter((l) => l.length > 0 && !l.startsWith(\"#\"));\n } catch {\n return [];\n }\n}\n\nasync function buildMatcher(root: string): Promise<Ignore> {\n const ig = ignore();\n ig.add(ALWAYS_IGNORE.map((d) => `${d}/`));\n ig.add(await readIgnoreFile(join(root, \".gitignore\")));\n ig.add(await readIgnoreFile(join(root, \".synthraignore\")));\n return ig;\n}\n\nfunction toPosixRel(root: string, abs: string): string {\n const rel = relative(root, abs);\n return sep === \"/\" ? rel : rel.split(sep).join(\"/\");\n}\n\nexport function createFileWatcher(root: string, onEvent: FileEventHandler): FileWatcher {\n let watcher: FSWatcher | null = null;\n let ig: Ignore | null = null;\n\n const emit = async (kind: FileEvent[\"kind\"], abs: string) => {\n if (!ig) return;\n const rel = toPosixRel(root, abs);\n if (!rel || rel.startsWith(\"..\")) return;\n if (ig.ignores(rel)) return;\n try {\n await onEvent({ kind, path: rel, ts: new Date().toISOString() });\n } catch {\n // swallow handler errors — watcher must keep going\n }\n };\n\n return {\n async start() {\n ig = await buildMatcher(root);\n watcher = chokidar.watch(root, {\n // Cross-platform glob ignore. We match both the directory itself and\n // anything inside it. picomatch (chokidar's matcher) normalizes path\n // separators so a single set of forward-slash globs handles\n // Windows + POSIX. Function-based ignore was unreliable on Windows\n // and let chokidar descend into .git/, which crashed on transient\n // index.lock files held exclusively by git.\n ignored: ALWAYS_IGNORE.flatMap((d) => [`**/${d}`, `**/${d}/**`]),\n ignoreInitial: true,\n persistent: true,\n awaitWriteFinish: { stabilityThreshold: 150, pollInterval: 50 },\n });\n\n // Chokidar emits \"error\" for transient OS-level issues — most commonly\n // EPERM/ENOENT on rapidly created+deleted files. We never want one of\n // these to crash the syn process. Log + swallow.\n watcher.on(\"error\", (err) => {\n const e = err as NodeJS.ErrnoException;\n log.debug(`file watcher error (swallowed): ${e?.code ?? \"\"} ${e?.message ?? String(err)}`);\n });\n\n watcher.on(\"add\", (path) => emit(\"create\", path));\n watcher.on(\"change\", (path) => emit(\"save\", path));\n watcher.on(\"unlink\", (path) => emit(\"delete\", path));\n },\n\n async stop() {\n if (watcher) {\n await watcher.close();\n watcher = null;\n }\n },\n };\n}\n","// Minimal logger. Prefixes Synthra output with [syn].\n\ntype Level = \"debug\" | \"info\" | \"warn\" | \"error\";\n\nconst LEVEL_PRIORITY: Record<Level, number> = {\n debug: 10,\n info: 20,\n warn: 30,\n error: 40,\n};\n\nlet activeLevel: Level = (process.env.SYN_LOG_LEVEL as Level) ?? \"info\";\n\nexport function setLevel(level: Level): void {\n activeLevel = level;\n}\n\nfunction shouldLog(level: Level): boolean {\n return LEVEL_PRIORITY[level] >= LEVEL_PRIORITY[activeLevel];\n}\n\nfunction emit(level: Level, msg: string, ...args: unknown[]): void {\n if (!shouldLog(level)) return;\n const stream = level === \"error\" || level === \"warn\" ? process.stderr : process.stdout;\n stream.write(`[syn] ${msg}${args.length ? \" \" + args.map(String).join(\" \") : \"\"}\\n`);\n}\n\nexport const log = {\n debug: (m: string, ...a: unknown[]) => emit(\"debug\", m, ...a),\n info: (m: string, ...a: unknown[]) => emit(\"info\", m, ...a),\n warn: (m: string, ...a: unknown[]) => emit(\"warn\", m, ...a),\n error: (m: string, ...a: unknown[]) => emit(\"error\", m, ...a),\n};\n","// Watches `.git/HEAD` for branch switches (via fs.watch) and polls\n// `git status --porcelain` every ~2s to surface uncommitted-diff changes.\n// Always best-effort: in a non-git directory or when git is missing, the\n// watcher simply emits nothing.\n\nimport { execFile } from \"node:child_process\";\nimport { watch, type FSWatcher } from \"node:fs\";\nimport { readFile } from \"node:fs/promises\";\nimport { join } from \"node:path\";\nimport { promisify } from \"node:util\";\n\nimport type { GitEvent } from \"./activity-log.js\";\n\nconst execFileAsync = promisify(execFile);\n\nconst POLL_MS = 2000;\n\nexport interface GitWatcher {\n start(): Promise<void>;\n stop(): Promise<void>;\n}\n\nexport type GitEventHandler = (e: GitEvent) => void | Promise<void>;\n\nasync function readHeadBranch(projectRoot: string): Promise<string | null> {\n try {\n const head = await readFile(join(projectRoot, \".git\", \"HEAD\"), \"utf8\");\n const m = head.trim().match(/^ref:\\s+refs\\/heads\\/(.+)$/);\n return m?.[1] ?? null;\n } catch {\n return null;\n }\n}\n\nasync function readStatusPorcelain(projectRoot: string): Promise<string | null> {\n try {\n const { stdout } = await execFileAsync(\"git\", [\"status\", \"--porcelain\"], {\n cwd: projectRoot,\n });\n return stdout;\n } catch {\n return null;\n }\n}\n\nexport function createGitWatcher(root: string, onEvent: GitEventHandler): GitWatcher {\n let headWatcher: FSWatcher | null = null;\n let pollTimer: NodeJS.Timeout | null = null;\n let lastBranch: string | null = null;\n let lastStatus: string | null = null;\n\n const emitSafe = async (event: GitEvent) => {\n try {\n await onEvent(event);\n } catch {\n // swallow\n }\n };\n\n const checkHead = async () => {\n const branch = await readHeadBranch(root);\n if (branch && branch !== lastBranch) {\n const prev = lastBranch;\n lastBranch = branch;\n if (prev !== null) {\n await emitSafe({\n kind: \"branch-switch\",\n details: { from: prev, to: branch },\n ts: new Date().toISOString(),\n });\n }\n }\n };\n\n const pollStatus = async () => {\n const status = await readStatusPorcelain(root);\n if (status === null) return;\n if (lastStatus !== null && status !== lastStatus) {\n const prevFiles = parseStatusFiles(lastStatus);\n const nowFiles = parseStatusFiles(status);\n const added = nowFiles.filter((f) => !prevFiles.includes(f));\n const removed = prevFiles.filter((f) => !nowFiles.includes(f));\n await emitSafe({\n kind: \"diff-change\",\n details: {\n changed_count: nowFiles.length,\n newly_dirty: added,\n newly_clean: removed,\n },\n ts: new Date().toISOString(),\n });\n }\n lastStatus = status;\n };\n\n return {\n async start() {\n // Seed initial branch + status so the first real change emits an event\n // rather than a stale \"from null\".\n lastBranch = await readHeadBranch(root);\n lastStatus = await readStatusPorcelain(root);\n\n try {\n headWatcher = watch(join(root, \".git\", \"HEAD\"), () => {\n void checkHead();\n });\n // fs.watch emits \"error\" for transient OS issues (EPERM on lock\n // files, ENOENT when refs get rewritten). Swallow them — we never\n // want a transient FS event to crash the syn process.\n headWatcher.on(\"error\", () => {\n // silent — branch-switch detection is best-effort\n });\n } catch {\n // .git/HEAD not present — silently no-op\n }\n\n pollTimer = setInterval(() => {\n void pollStatus();\n }, POLL_MS);\n pollTimer.unref?.();\n },\n\n async stop() {\n if (headWatcher) {\n headWatcher.close();\n headWatcher = null;\n }\n if (pollTimer) {\n clearInterval(pollTimer);\n pollTimer = null;\n }\n },\n };\n}\n\nfunction parseStatusFiles(porcelain: string): string[] {\n return porcelain\n .split(/\\r?\\n/)\n .map((l) => l.slice(3).trim())\n .filter((l) => l.length > 0);\n}\n","// `syn scan [path]` — bootstrap then walk + parse + write graph.\n// Also invoked by the default `syn .` flow (M3 will chain start-claude after).\n\nimport { resolve } from \"node:path\";\n\nimport { buildGraph, buildSymbolIndex } from \"../scanner/extract.js\";\nimport { parseFile, type ParsedFile } from \"../scanner/parser.js\";\nimport { walk, type WalkedFile } from \"../scanner/walker.js\";\nimport { writeGraph, writeSymbolIndex } from \"../graph/store.js\";\nimport { log } from \"../shared/logger.js\";\nimport { resolvePaths } from \"../shared/paths.js\";\nimport { bootstrap } from \"./bootstrap.js\";\n\nconst PARSABLE_EXTS = new Set([\n \".ts\", \".tsx\", \".cts\", \".mts\",\n \".js\", \".jsx\", \".cjs\", \".mjs\",\n \".py\", \".pyi\",\n \".svelte\",\n \".vue\",\n \".go\",\n \".rs\",\n \".java\",\n \".kt\", \".kts\",\n \".php\",\n \".rb\",\n \".c\", \".h\",\n \".cpp\", \".cc\", \".cxx\", \".hpp\", \".hh\", \".hxx\",\n \".dart\",\n \".cs\",\n]);\n\nexport interface ScanResult {\n walked: number;\n parsed: number;\n symbolCount: number;\n edgeCount: number;\n durationMs: number;\n}\n\nexport interface ScanOptions {\n /** Suppress per-step log output (used for branch-switch rescans). */\n silent?: boolean;\n}\n\n/**\n * Core scan pipeline — bootstrap + walk + parse + write graph. Importable\n * from anywhere (server, CLI, tests). `scanCommand` is just a logging wrapper\n * around this. Pass `silent: true` to skip the chatty progress output.\n */\nexport async function scanProject(projectRootRaw: string, opts: ScanOptions = {}): Promise<ScanResult> {\n const projectRoot = resolve(projectRootRaw);\n const paths = resolvePaths(projectRoot);\n const start = Date.now();\n const verbose = !opts.silent;\n\n if (verbose) log.info(`scanning ${projectRoot}`);\n\n const boot = await bootstrap(paths);\n if (verbose) {\n if (boot.graphCreated) log.info(\" created .synthra-graph/\");\n if (boot.contextCreated) log.info(\" created .synthra/\");\n if (boot.gitignoreUpdated) log.info(\" updated .gitignore\");\n if (boot.claudeMdCreated) {\n log.info(\" created CLAUDE.md — onboarding skeleton for the agent\");\n log.info(\" ↳ fill in Build / Conventions / Decisions (or run /init in Claude to auto-draft)\");\n } else if (boot.claudeMdUpdated) {\n log.info(\" updated CLAUDE.md\");\n }\n }\n\n const walked: WalkedFile[] = [];\n for await (const file of walk(projectRoot)) walked.push(file);\n if (verbose) log.info(` walked ${walked.length} files`);\n\n const parsable = walked.filter((f) => PARSABLE_EXTS.has(f.ext));\n const parsed: ParsedFile[] = [];\n let parseErrors = 0;\n for (const file of parsable) {\n try {\n parsed.push(await parseFile(file));\n } catch (err) {\n parseErrors += 1;\n if (verbose) log.debug(` parse failed: ${file.relPath} — ${(err as Error).message}`);\n }\n }\n if (verbose) {\n log.info(\n ` parsed ${parsed.length} files (${walked.length - parsable.length} skipped` +\n (parseErrors ? `, ${parseErrors} errored` : \"\") +\n \")\",\n );\n }\n\n const graph = await buildGraph(projectRoot, parsed);\n const symbolIndex = buildSymbolIndex(graph);\n\n await writeGraph(paths.infoGraph, graph);\n await writeSymbolIndex(paths.symbolIndex, symbolIndex);\n\n if (verbose) {\n log.info(\n ` wrote ${paths.infoGraph} — ${graph.symbol_count} symbols, ${graph.edge_count} edges`,\n );\n log.info(` wrote ${paths.symbolIndex} — ${Object.keys(symbolIndex).length} names`);\n }\n\n const durationMs = Date.now() - start;\n if (verbose) log.info(`done in ${(durationMs / 1000).toFixed(2)}s`);\n\n return {\n walked: walked.length,\n parsed: parsed.length,\n symbolCount: graph.symbol_count,\n edgeCount: graph.edge_count,\n durationMs,\n };\n}\n\n// Thin alias so the CLI command keeps its current name. Drop in v0.2 if we\n// settle on a single public function.\nexport async function scanCommand(rawPath: string): Promise<ScanResult> {\n return scanProject(rawPath);\n}\n","// Turns ParsedFile[] into a GraphSchema (nodes + edges).\n//\n// Edges produced in M1:\n// defines : file → symbol\n// imports : file → file (when the import target resolves inside the project)\n// tests : test-file → source-file (for foo.test.ts ↔ foo.ts)\n\nimport { dirname, join, posix } from \"node:path\";\n\nimport type { Edge, FileNode, GraphSchema, SymbolIndex, SymbolNode } from \"../graph/types.js\";\nimport { fileHash } from \"./hash.js\";\nimport { extractKeywords } from \"./keywords.js\";\nimport type { ParsedFile, ParsedSymbol } from \"./parser.js\";\n\nconst RESOLVE_EXTS = [\".ts\", \".tsx\", \".js\", \".jsx\", \".mjs\", \".cjs\", \".py\", \".svelte\", \".vue\", \".dart\", \".html\", \".hubl\"];\nconst INDEX_FILES = [\"index.ts\", \"index.tsx\", \"index.js\", \"index.jsx\", \"__init__.py\"];\n\nfunction fileId(relPath: string): string {\n return `file:${relPath}`;\n}\n\nfunction symbolId(relPath: string, sym: ParsedSymbol): string {\n return `symbol:${relPath}::${sym.name}:${sym.startLine}`;\n}\n\nfunction toFileNode(parsed: ParsedFile): FileNode {\n const content = parsed.source;\n return {\n id: fileId(parsed.file.relPath),\n kind: \"file\",\n path: parsed.file.relPath,\n ext: parsed.file.ext,\n size: parsed.file.size,\n keywords: extractKeywords(content, parsed.file.ext),\n content,\n summary: extractSummary(content),\n file_hash: fileHash(content),\n };\n}\n\nfunction extractSummary(content: string): string {\n // First leading comment block (// ... or /** ... */ or # ...), trimmed to ~200 chars.\n const trimmed = content.replace(/^\\s+/, \"\");\n const slashMatch = trimmed.match(/^\\/\\/\\s?(.*(?:\\r?\\n\\/\\/\\s?.*)*)/);\n if (slashMatch?.[1]) return slashMatch[1].split(/\\r?\\n/).join(\" \").trim().slice(0, 200);\n const blockMatch = trimmed.match(/^\\/\\*\\*?([\\s\\S]*?)\\*\\//);\n if (blockMatch?.[1]) {\n return blockMatch[1]\n .split(/\\r?\\n/)\n .map((l) => l.replace(/^\\s*\\*\\s?/, \"\"))\n .join(\" \")\n .trim()\n .slice(0, 200);\n }\n const hashMatch = trimmed.match(/^#\\s?(.*(?:\\r?\\n#\\s?.*)*)/);\n if (hashMatch?.[1]) return hashMatch[1].split(/\\r?\\n/).join(\" \").trim().slice(0, 200);\n return \"\";\n}\n\nfunction toSymbolNode(parsed: ParsedFile, sym: ParsedSymbol): SymbolNode {\n return {\n id: symbolId(parsed.file.relPath, sym),\n kind: \"symbol\",\n symbol_kind: sym.kind,\n name: sym.name,\n file: parsed.file.relPath,\n start_line: sym.startLine,\n end_line: sym.endLine,\n signature: sym.signature,\n };\n}\n\n/**\n * Resolve an import specifier to a project-relative path if it refers to a\n * file inside the project. Returns `null` for external packages (no leading\n * dot) or specifiers that don't match any known file.\n */\n// Strip a trailing JS-family extension so a spec like \"./crypto.js\" can\n// resolve to \"crypto.ts\". TypeScript-style `.js` imports are common.\nconst REWRITE_EXT_RE = /\\.(js|jsx|mjs|cjs)$/;\n\nfunction resolveImport(\n fromRelPath: string,\n spec: string,\n filesByPath: Map<string, true>,\n): string | null {\n if (!spec.startsWith(\".\")) return null;\n const fromDir = posix.dirname(toPosix(fromRelPath));\n const base = posix.normalize(posix.join(fromDir, toPosix(spec)));\n\n const candidates = [base];\n const rewritten = base.replace(REWRITE_EXT_RE, \"\");\n if (rewritten !== base) candidates.push(rewritten);\n\n for (const c of candidates) {\n if (filesByPath.has(c)) return c;\n for (const ext of RESOLVE_EXTS) {\n if (filesByPath.has(c + ext)) return c + ext;\n }\n for (const idx of INDEX_FILES) {\n const candidate = posix.join(c, idx);\n if (filesByPath.has(candidate)) return candidate;\n }\n }\n return null;\n}\n\nfunction toPosix(p: string): string {\n return p.split(/[\\\\/]/).join(\"/\");\n}\n\nconst TEST_RE = /^(?<base>.+?)\\.(test|spec)\\.(?<ext>[tj]sx?|py)$/;\n\nfunction testTarget(relPath: string, filesByPath: Map<string, true>): string | null {\n const fileName = relPath.split(\"/\").pop() ?? relPath;\n const match = TEST_RE.exec(fileName);\n if (!match) return null;\n const dir = relPath.includes(\"/\") ? relPath.slice(0, relPath.lastIndexOf(\"/\") + 1) : \"\";\n const base = match.groups?.base ?? \"\";\n const ext = match.groups?.ext ?? \"\";\n if (!base || !ext) return null;\n const candidate = `${dir}${base}.${ext}`;\n if (filesByPath.has(candidate)) return candidate;\n // Try sibling extensions (e.g. foo.test.ts → foo.tsx)\n for (const e of RESOLVE_EXTS) {\n const alt = `${dir}${base}${e}`;\n if (filesByPath.has(alt)) return alt;\n }\n return null;\n}\n\nexport async function buildGraph(root: string, parsed: ParsedFile[]): Promise<GraphSchema> {\n const filesByPath = new Map<string, true>();\n for (const p of parsed) filesByPath.set(p.file.relPath, true);\n\n const nodes: (FileNode | SymbolNode)[] = [];\n const edges: Edge[] = [];\n\n for (const p of parsed) {\n const fileNode = toFileNode(p);\n nodes.push(fileNode);\n\n for (const sym of p.symbols) {\n const symNode = toSymbolNode(p, sym);\n nodes.push(symNode);\n edges.push({ from: fileNode.id, to: symNode.id, kind: \"defines\" });\n }\n\n const importEdges = new Set<string>();\n for (const spec of p.imports) {\n const target = resolveImport(p.file.relPath, spec, filesByPath);\n if (!target) continue;\n const key = `${fileNode.id}->${fileId(target)}`;\n if (importEdges.has(key)) continue;\n importEdges.add(key);\n edges.push({ from: fileNode.id, to: fileId(target), kind: \"imports\" });\n }\n\n const testTargetPath = testTarget(p.file.relPath, filesByPath);\n if (testTargetPath && testTargetPath !== p.file.relPath) {\n edges.push({ from: fileNode.id, to: fileId(testTargetPath), kind: \"tests\" });\n }\n }\n\n const symbolCount = nodes.filter((n) => n.kind === \"symbol\").length;\n const fileCount = nodes.length - symbolCount;\n\n return {\n root,\n node_count: nodes.length,\n edge_count: edges.length,\n file_count: fileCount,\n symbol_count: symbolCount,\n nodes,\n edges,\n generated_at: new Date().toISOString(),\n schema_version: 1,\n };\n}\n\nexport function buildSymbolIndex(graph: GraphSchema): SymbolIndex {\n // Null-prototype map: symbol names like \"toString\" or \"constructor\" (common\n // in Dart, where every class overrides toString) would otherwise resolve to\n // an inherited Object.prototype member and crash on the .push below.\n const out: SymbolIndex = Object.create(null);\n for (const node of graph.nodes) {\n if (node.kind !== \"symbol\") continue;\n const list = out[node.name] ?? (out[node.name] = []);\n list.push({ file: node.file, line: node.start_line, kind: node.symbol_kind });\n }\n return out;\n}\n\n// Re-export node path helpers in case downstream wants the canonical id format\nexport { fileId, symbolId };\n// Suppress unused-import lint for dirname/join from node:path — kept reserved for incremental updates.\nvoid dirname;\nvoid join;\n","// Stable, short content hash for files. Used to detect changed files\n// during incremental rescans (post-v0.1; M1 does full re-parse).\n// TODO: M1 (minimal); post-v0.1 (incremental)\n\nimport { createHash } from \"node:crypto\";\n\nexport function fileHash(content: string): string {\n return createHash(\"sha1\").update(content).digest(\"hex\").slice(0, 8);\n}\n","// Per-file keyword extraction. Used for query-time relevance ranking.\n// Tokenizes identifiers + comment words, splits camelCase/snake_case, filters\n// stopwords, and returns the top-N rare tokens scored by inverse frequency\n// against a small built-in english/code corpus.\n\nconst STOPWORDS = new Set([\n \"a\", \"an\", \"and\", \"are\", \"as\", \"at\", \"be\", \"but\", \"by\", \"do\", \"for\", \"from\",\n \"has\", \"have\", \"he\", \"if\", \"in\", \"is\", \"it\", \"its\", \"not\", \"of\", \"on\", \"or\",\n \"she\", \"that\", \"the\", \"they\", \"this\", \"to\", \"was\", \"we\", \"were\", \"will\", \"with\",\n \"you\", \"your\", \"i\", \"me\", \"my\", \"our\", \"us\", \"their\", \"them\", \"his\", \"her\",\n // common code words that add no signal\n \"function\", \"const\", \"let\", \"var\", \"class\", \"interface\", \"type\", \"enum\",\n \"import\", \"export\", \"from\", \"default\", \"return\", \"if\", \"else\", \"for\", \"while\",\n \"do\", \"switch\", \"case\", \"break\", \"continue\", \"new\", \"this\", \"super\", \"throw\",\n \"try\", \"catch\", \"finally\", \"async\", \"await\", \"yield\", \"true\", \"false\", \"null\",\n \"undefined\", \"void\", \"any\", \"string\", \"number\", \"boolean\", \"object\", \"array\",\n \"self\", \"cls\", \"def\", \"lambda\", \"pass\", \"raise\", \"with\", \"as\", \"in\",\n \"todo\", \"fixme\", \"note\",\n]);\n\nconst COMMON_CODE = new Set([\n \"value\", \"data\", \"result\", \"args\", \"kwargs\", \"options\", \"config\", \"params\",\n \"name\", \"id\", \"key\", \"index\", \"item\", \"items\", \"list\", \"map\", \"set\", \"get\",\n \"set\", \"add\", \"remove\", \"delete\", \"create\", \"update\", \"find\", \"fetch\", \"load\",\n \"save\", \"init\", \"main\", \"run\", \"start\", \"stop\", \"test\", \"check\", \"validate\",\n \"error\", \"err\", \"warn\", \"info\", \"debug\", \"log\", \"trace\", \"msg\", \"message\",\n \"path\", \"file\", \"dir\", \"url\", \"host\", \"port\", \"size\", \"length\", \"count\",\n \"input\", \"output\", \"source\", \"target\", \"callback\", \"handler\", \"listener\",\n \"props\", \"state\", \"context\", \"render\", \"component\", \"node\", \"tree\", \"root\",\n]);\n\n// Frequency weight — common-code words count for less than rare identifiers\nfunction score(token: string): number {\n if (STOPWORDS.has(token)) return 0;\n if (COMMON_CODE.has(token)) return 0.2;\n if (token.length <= 2) return 0.1;\n return 1;\n}\n\nfunction splitIdentifier(id: string): string[] {\n // snake_case + kebab-case → words\n const partsRaw = id.split(/[_\\-./]+/).filter(Boolean);\n const out: string[] = [];\n for (const part of partsRaw) {\n // camelCase / PascalCase → words. Handles \"XMLHttp\" → [\"XML\", \"Http\"]\n const camelParts = part.match(/[A-Z]+(?=[A-Z][a-z])|[A-Z]?[a-z]+|[A-Z]+|[0-9]+/g);\n if (camelParts) out.push(...camelParts);\n else out.push(part);\n }\n return out.map((w) => w.toLowerCase()).filter((w) => /[a-z]/.test(w));\n}\n\nexport function extractKeywords(content: string, _ext: string): string[] {\n // Identifiers + alphanumeric words. Picks up both code and comment text.\n const tokens = content.match(/[A-Za-z_][A-Za-z0-9_]{1,40}/g) ?? [];\n const counts = new Map<string, number>();\n for (const tok of tokens) {\n for (const word of splitIdentifier(tok)) {\n const w = score(word);\n if (w === 0) continue;\n counts.set(word, (counts.get(word) ?? 0) + w);\n }\n }\n return Array.from(counts.entries())\n .sort((a, b) => b[1] - a[1])\n .slice(0, 32)\n .map(([w]) => w);\n}\n","// Dispatches a file to its language-specific parser based on extension.\n// Tree-sitter WASM grammars are loaded lazily via tree-sitter-wasms and\n// cached per language for the lifetime of the process.\n\nimport { readFile } from \"node:fs/promises\";\nimport { createRequire } from \"node:module\";\nimport { Language, Parser } from \"web-tree-sitter\";\n\nimport type { SymbolKind } from \"../graph/types.js\";\nimport { parseC } from \"./parsers/c.js\";\nimport { parseCpp } from \"./parsers/cpp.js\";\nimport { parseCSharp } from \"./parsers/csharp.js\";\nimport { parseDart } from \"./parsers/dart.js\";\nimport { parseGo } from \"./parsers/go.js\";\nimport { parseHubL } from \"./parsers/hubl.js\";\nimport { parseJava } from \"./parsers/java.js\";\nimport { parseKotlin } from \"./parsers/kotlin.js\";\nimport { parsePhp } from \"./parsers/php.js\";\nimport { parsePython } from \"./parsers/python.js\";\nimport { parseRuby } from \"./parsers/ruby.js\";\nimport { parseRust } from \"./parsers/rust.js\";\nimport { parseSvelte } from \"./parsers/svelte.js\";\nimport { parseTypeScript } from \"./parsers/typescript.js\";\nimport { parseVue } from \"./parsers/vue.js\";\nimport type { WalkedFile } from \"./walker.js\";\n\nexport interface ParsedSymbol {\n name: string;\n kind: SymbolKind;\n startLine: number;\n endLine: number;\n signature: string;\n}\n\nexport interface ParsedFile {\n file: WalkedFile;\n source: string;\n symbols: ParsedSymbol[];\n imports: string[];\n calls: Array<{ from: string; to: string }>;\n}\n\nconst require = createRequire(import.meta.url);\n\nexport type GrammarName =\n | \"typescript\"\n | \"tsx\"\n | \"javascript\"\n | \"python\"\n | \"go\"\n | \"rust\"\n | \"java\"\n | \"kotlin\"\n | \"php\"\n | \"ruby\"\n | \"c\"\n | \"cpp\"\n | \"dart\"\n | \"csharp\";\n\nconst GRAMMAR_FILES: Record<GrammarName, string> = {\n typescript: \"tree-sitter-wasms/out/tree-sitter-typescript.wasm\",\n tsx: \"tree-sitter-wasms/out/tree-sitter-tsx.wasm\",\n javascript: \"tree-sitter-wasms/out/tree-sitter-javascript.wasm\",\n python: \"tree-sitter-wasms/out/tree-sitter-python.wasm\",\n go: \"tree-sitter-wasms/out/tree-sitter-go.wasm\",\n rust: \"tree-sitter-wasms/out/tree-sitter-rust.wasm\",\n java: \"tree-sitter-wasms/out/tree-sitter-java.wasm\",\n kotlin: \"tree-sitter-wasms/out/tree-sitter-kotlin.wasm\",\n php: \"tree-sitter-wasms/out/tree-sitter-php.wasm\",\n ruby: \"tree-sitter-wasms/out/tree-sitter-ruby.wasm\",\n c: \"tree-sitter-wasms/out/tree-sitter-c.wasm\",\n cpp: \"tree-sitter-wasms/out/tree-sitter-cpp.wasm\",\n dart: \"tree-sitter-wasms/out/tree-sitter-dart.wasm\",\n csharp: \"tree-sitter-wasms/out/tree-sitter-c_sharp.wasm\",\n};\n\nlet parserInit: Promise<void> | null = null;\nconst languageCache = new Map<GrammarName, Language>();\n\nasync function ensureParserInit(): Promise<void> {\n if (!parserInit) {\n parserInit = Parser.init();\n }\n return parserInit;\n}\n\nexport async function loadGrammar(name: GrammarName): Promise<Language> {\n await ensureParserInit();\n const cached = languageCache.get(name);\n if (cached) return cached;\n const wasmPath = require.resolve(GRAMMAR_FILES[name]);\n const lang = await Language.load(wasmPath);\n languageCache.set(name, lang);\n return lang;\n}\n\nexport interface LoadedParser {\n parser: Parser;\n language: Language;\n}\n\nexport async function createParser(name: GrammarName): Promise<LoadedParser> {\n const language = await loadGrammar(name);\n const parser = new Parser();\n parser.setLanguage(language);\n return { parser, language };\n}\n\nfunction emptyParsed(file: WalkedFile, source: string): ParsedFile {\n return { file, source, symbols: [], imports: [], calls: [] };\n}\n\nexport async function parseFile(f: WalkedFile): Promise<ParsedFile> {\n let source: string;\n try {\n source = await readFile(f.absPath, \"utf8\");\n } catch {\n return emptyParsed(f, \"\");\n }\n\n switch (f.ext) {\n case \".ts\":\n case \".tsx\":\n case \".cts\":\n case \".mts\":\n case \".js\":\n case \".jsx\":\n case \".cjs\":\n case \".mjs\":\n return parseTypeScript(f, source);\n case \".py\":\n case \".pyi\":\n return parsePython(f, source);\n case \".svelte\":\n return parseSvelte(f, source);\n case \".vue\":\n return parseVue(f, source);\n case \".html\":\n case \".hubl\":\n return parseHubL(f, source);\n case \".go\":\n return parseGo(f, source);\n case \".rs\":\n return parseRust(f, source);\n case \".java\":\n return parseJava(f, source);\n case \".kt\":\n case \".kts\":\n return parseKotlin(f, source);\n case \".php\":\n return parsePhp(f, source);\n case \".rb\":\n return parseRuby(f, source);\n case \".c\":\n case \".h\":\n return parseC(f, source);\n case \".cpp\":\n case \".cc\":\n case \".cxx\":\n case \".hpp\":\n case \".hh\":\n case \".hxx\":\n return parseCpp(f, source);\n case \".dart\":\n return parseDart(f, source);\n case \".cs\":\n return parseCSharp(f, source);\n default:\n return emptyParsed(f, source);\n }\n}\n","// Generic tree-sitter parser used by the simpler-grammar languages\n// (Go, Rust, Java, Kotlin, PHP, Ruby, C, C++, Dart, C#).\n//\n// Each language file defines:\n// - which tree-sitter grammar to load\n// - a query string with capture names like `@function`, `@function.name`\n// - a `decls` table mapping declaration-capture pairs to SymbolKind\n// - optional `importCapture` for collecting import edges\n// Everything else (parser init, error handling, dedupe) lives here.\n\nimport { Query, type Node } from \"web-tree-sitter\";\n\nimport type { SymbolKind } from \"../../graph/types.js\";\nimport { createParser, type GrammarName, type ParsedFile, type ParsedSymbol } from \"../parser.js\";\nimport type { WalkedFile } from \"../walker.js\";\n\nexport interface DeclCapture {\n /** Capture name for the declaration node, e.g. \"function\". */\n declCapture: string;\n /** Capture name for the symbol's name node, e.g. \"function.name\". */\n nameCapture: string;\n /** SymbolKind to assign. */\n kind: SymbolKind;\n}\n\nexport interface GenericParserConfig {\n grammar: GrammarName;\n query: string;\n decls: DeclCapture[];\n /** Capture name for import-source nodes. Skipped when omitted. */\n importCapture?: string;\n}\n\nexport function firstLine(text: string, max = 200): string {\n const line = text.split(/\\r?\\n/, 1)[0] ?? \"\";\n return line.length > max ? line.slice(0, max) + \"…\" : line;\n}\n\nfunction cleanImport(s: string): string {\n // Strip surrounding string-literal quotes (used by Go, Dart, C/C++).\n // Strip angle brackets used by C/C++ system includes.\n return s.replace(/^[\"'`<]+|[\"'`>]+$/g, \"\").trim();\n}\n\nexport async function runGenericParser(\n config: GenericParserConfig,\n f: WalkedFile,\n source: string,\n): Promise<ParsedFile> {\n let symbols: ParsedSymbol[] = [];\n let imports: string[] = [];\n\n try {\n const { parser, language } = await createParser(config.grammar);\n const tree = parser.parse(source);\n if (!tree) return { file: f, source, symbols, imports, calls: [] };\n\n const query = new Query(language, config.query);\n const matches = query.matches(tree.rootNode);\n\n for (const match of matches) {\n const byName = new Map<string, Node>();\n for (const cap of match.captures) byName.set(cap.name, cap.node);\n\n let matched: DeclCapture | null = null;\n for (const d of config.decls) {\n if (byName.has(d.declCapture) && byName.has(d.nameCapture)) {\n matched = d;\n break;\n }\n }\n\n if (matched) {\n const declNode = byName.get(matched.declCapture)!;\n const nameNode = byName.get(matched.nameCapture)!;\n symbols.push({\n name: nameNode.text,\n kind: matched.kind,\n startLine: declNode.startPosition.row + 1,\n endLine: declNode.endPosition.row + 1,\n signature: firstLine(declNode.text),\n });\n continue;\n }\n\n if (config.importCapture) {\n const imp = byName.get(config.importCapture);\n if (imp) imports.push(cleanImport(imp.text));\n }\n }\n\n const seen = new Set<string>();\n symbols = symbols.filter((s) => {\n const k = `${s.name}:${s.startLine}`;\n if (seen.has(k)) return false;\n seen.add(k);\n return true;\n });\n imports = Array.from(new Set(imports)).filter((s) => s.length > 0);\n } catch {\n // Query compile or parse failure — return what we have. Silent so a single\n // bad file doesn't abort the whole scan.\n }\n\n return { file: f, source, symbols, imports, calls: [] };\n}\n","// C parser. Function definitions, structs, enums, typedefs, #include directives.\n\nimport type { ParsedFile } from \"../parser.js\";\nimport type { WalkedFile } from \"../walker.js\";\nimport { runGenericParser } from \"./_generic.js\";\n\nconst QUERY = `\n(function_definition declarator: (function_declarator declarator: (identifier) @function.name)) @function\n(struct_specifier name: (type_identifier) @struct.name) @struct\n(enum_specifier name: (type_identifier) @enum.name) @enum\n(type_definition declarator: (type_identifier) @type.name) @type\n(preproc_include path: (string_literal) @import)\n(preproc_include path: (system_lib_string) @import)\n`;\n\nexport async function parseC(f: WalkedFile, source: string): Promise<ParsedFile> {\n return runGenericParser(\n {\n grammar: \"c\",\n query: QUERY,\n decls: [\n { declCapture: \"function\", nameCapture: \"function.name\", kind: \"function\" },\n { declCapture: \"struct\", nameCapture: \"struct.name\", kind: \"class\" },\n { declCapture: \"enum\", nameCapture: \"enum.name\", kind: \"enum\" },\n { declCapture: \"type\", nameCapture: \"type.name\", kind: \"type\" },\n ],\n importCapture: \"import\",\n },\n f,\n source,\n );\n}\n","// C++ parser. Functions, classes, structs, enums, namespaces, #includes.\n\nimport type { ParsedFile } from \"../parser.js\";\nimport type { WalkedFile } from \"../walker.js\";\nimport { runGenericParser } from \"./_generic.js\";\n\nconst QUERY = `\n(function_definition declarator: (function_declarator declarator: (identifier) @function.name)) @function\n(function_definition declarator: (function_declarator declarator: (qualified_identifier) @method.name)) @method\n(class_specifier name: (type_identifier) @class.name) @class\n(struct_specifier name: (type_identifier) @struct.name) @struct\n(enum_specifier name: (type_identifier) @enum.name) @enum\n(namespace_definition name: (namespace_identifier) @namespace.name) @namespace\n(preproc_include path: (string_literal) @import)\n(preproc_include path: (system_lib_string) @import)\n`;\n\nexport async function parseCpp(f: WalkedFile, source: string): Promise<ParsedFile> {\n return runGenericParser(\n {\n grammar: \"cpp\",\n query: QUERY,\n decls: [\n { declCapture: \"function\", nameCapture: \"function.name\", kind: \"function\" },\n { declCapture: \"method\", nameCapture: \"method.name\", kind: \"method\" },\n { declCapture: \"class\", nameCapture: \"class.name\", kind: \"class\" },\n { declCapture: \"struct\", nameCapture: \"struct.name\", kind: \"class\" },\n { declCapture: \"enum\", nameCapture: \"enum.name\", kind: \"enum\" },\n { declCapture: \"namespace\", nameCapture: \"namespace.name\", kind: \"class\" },\n ],\n importCapture: \"import\",\n },\n f,\n source,\n );\n}\n","// C# (.NET) parser. Classes, interfaces, structs, methods, namespaces.\n\nimport type { ParsedFile } from \"../parser.js\";\nimport type { WalkedFile } from \"../walker.js\";\nimport { runGenericParser } from \"./_generic.js\";\n\nconst QUERY = `\n(class_declaration name: (identifier) @class.name) @class\n(interface_declaration name: (identifier) @interface.name) @interface\n(struct_declaration name: (identifier) @struct.name) @struct\n(enum_declaration name: (identifier) @enum.name) @enum\n(method_declaration name: (identifier) @method.name) @method\n(namespace_declaration name: (_) @namespace.name) @namespace\n(using_directive (_) @import)\n`;\n\nexport async function parseCSharp(f: WalkedFile, source: string): Promise<ParsedFile> {\n return runGenericParser(\n {\n grammar: \"csharp\",\n query: QUERY,\n decls: [\n { declCapture: \"class\", nameCapture: \"class.name\", kind: \"class\" },\n { declCapture: \"interface\", nameCapture: \"interface.name\", kind: \"interface\" },\n { declCapture: \"struct\", nameCapture: \"struct.name\", kind: \"class\" },\n { declCapture: \"enum\", nameCapture: \"enum.name\", kind: \"enum\" },\n { declCapture: \"method\", nameCapture: \"method.name\", kind: \"method\" },\n { declCapture: \"namespace\", nameCapture: \"namespace.name\", kind: \"class\" },\n ],\n importCapture: \"import\",\n },\n f,\n source,\n );\n}\n","// Dart parser. v0.1.11 — real symbol extraction + import parsing on top of the\n// ABI-v15 grammar that ships in tree-sitter-wasms.\n//\n// Distinguishes top-level function_signature (kind: function) from\n// function_signature nested under method_signature (kind: method) by\n// anchoring the top-level pattern under `program`.\n//\n// Imports: `package:foo/bar.dart` and `dart:async` are stripped — they cross\n// the project boundary. Bare `foo.dart` is normalized to `./foo.dart` so the\n// shared resolveImport() (which requires a leading `.`) treats it as a\n// same-directory relative import.\n\nimport { Query, type Node } from \"web-tree-sitter\";\nimport type { SymbolKind } from \"../../graph/types.js\";\nimport { createParser, type ParsedFile, type ParsedSymbol } from \"../parser.js\";\nimport type { WalkedFile } from \"../walker.js\";\n\nconst QUERY = `\n(class_definition name: (identifier) @class.name) @class\n(mixin_declaration (identifier) @mixin.name) @mixin\n(extension_declaration name: (identifier) @ext.name) @ext\n(enum_declaration name: (identifier) @enum.name) @enum\n(type_alias (type_identifier) @typedef.name) @typedef\n\n(program (function_signature name: (identifier) @function.name) @function)\n\n(method_signature (function_signature name: (identifier) @method.name)) @method\n(method_signature (getter_signature name: (identifier) @getter.name)) @getter\n(method_signature (setter_signature name: (identifier) @setter.name)) @setter\n(constructor_signature name: (identifier) @ctor.name) @ctor\n\n(import_or_export (library_import (import_specification (configurable_uri (uri (string_literal) @import)))))\n`;\n\ninterface DeclShape {\n declCap: string;\n nameCap: string;\n kind: SymbolKind;\n}\n\nconst DECLS: DeclShape[] = [\n { declCap: \"class\", nameCap: \"class.name\", kind: \"class\" },\n { declCap: \"mixin\", nameCap: \"mixin.name\", kind: \"class\" },\n { declCap: \"ext\", nameCap: \"ext.name\", kind: \"class\" },\n { declCap: \"enum\", nameCap: \"enum.name\", kind: \"enum\" },\n { declCap: \"typedef\", nameCap: \"typedef.name\", kind: \"type\" },\n { declCap: \"function\", nameCap: \"function.name\", kind: \"function\" },\n { declCap: \"method\", nameCap: \"method.name\", kind: \"method\" },\n { declCap: \"getter\", nameCap: \"getter.name\", kind: \"method\" },\n { declCap: \"setter\", nameCap: \"setter.name\", kind: \"method\" },\n { declCap: \"ctor\", nameCap: \"ctor.name\", kind: \"method\" },\n];\n\nfunction firstLine(text: string, max = 200): string {\n const line = text.split(/\\r?\\n/, 1)[0] ?? \"\";\n return line.length > max ? line.slice(0, max) + \"…\" : line;\n}\n\n// Strip surrounding string-literal quotes and normalize bare same-directory\n// imports (Dart allows `import 'foo.dart';` without a leading `./`) so\n// resolveImport() — which keys off a leading dot — can match them.\nfunction normalizeDartImport(raw: string): string | null {\n const stripped = raw.replace(/^['\"]|['\"]$/g, \"\");\n if (!stripped) return null;\n if (stripped.startsWith(\"package:\")) return null;\n if (stripped.startsWith(\"dart:\")) return null;\n if (stripped.startsWith(\".\") || stripped.startsWith(\"/\")) return stripped;\n return `./${stripped}`;\n}\n\nexport async function parseDart(f: WalkedFile, source: string): Promise<ParsedFile> {\n let symbols: ParsedSymbol[] = [];\n let imports: string[] = [];\n\n try {\n const { parser, language } = await createParser(\"dart\");\n const tree = parser.parse(source);\n if (!tree) return { file: f, source, symbols, imports, calls: [] };\n\n const query = new Query(language, QUERY);\n const matches = query.matches(tree.rootNode);\n\n for (const match of matches) {\n const byName = new Map<string, Node>();\n for (const cap of match.captures) byName.set(cap.name, cap.node);\n\n let matched: DeclShape | null = null;\n for (const d of DECLS) {\n if (byName.has(d.declCap) && byName.has(d.nameCap)) {\n matched = d;\n break;\n }\n }\n\n if (matched) {\n const declNode = byName.get(matched.declCap)!;\n const nameNode = byName.get(matched.nameCap)!;\n symbols.push({\n name: nameNode.text,\n kind: matched.kind,\n startLine: declNode.startPosition.row + 1,\n endLine: declNode.endPosition.row + 1,\n signature: firstLine(declNode.text),\n });\n continue;\n }\n\n const importNode = byName.get(\"import\");\n if (importNode) {\n const norm = normalizeDartImport(importNode.text);\n if (norm) imports.push(norm);\n }\n }\n\n const seen = new Set<string>();\n symbols = symbols.filter((s) => {\n const k = `${s.name}:${s.startLine}`;\n if (seen.has(k)) return false;\n seen.add(k);\n return true;\n });\n imports = Array.from(new Set(imports));\n } catch {\n // swallow — see _generic.ts for the rationale (single bad file shouldn't\n // abort the whole scan).\n }\n\n return { file: f, source, symbols, imports, calls: [] };\n}\n","// Go parser. Functions, methods, type declarations, imports.\n\nimport type { ParsedFile } from \"../parser.js\";\nimport type { WalkedFile } from \"../walker.js\";\nimport { runGenericParser } from \"./_generic.js\";\n\nconst QUERY = `\n(function_declaration name: (identifier) @function.name) @function\n(method_declaration name: (field_identifier) @method.name) @method\n(type_spec name: (type_identifier) @type.name) @type\n(import_spec path: (interpreted_string_literal) @import)\n`;\n\nexport async function parseGo(f: WalkedFile, source: string): Promise<ParsedFile> {\n return runGenericParser(\n {\n grammar: \"go\",\n query: QUERY,\n decls: [\n { declCapture: \"function\", nameCapture: \"function.name\", kind: \"function\" },\n { declCapture: \"method\", nameCapture: \"method.name\", kind: \"method\" },\n { declCapture: \"type\", nameCapture: \"type.name\", kind: \"type\" },\n ],\n importCapture: \"import\",\n },\n f,\n source,\n );\n}\n","// HubL (HubSpot CMS templating) parser. HubL lives in .html / .hubl files and\n// has no tree-sitter grammar, so we extract its symbol-like constructs with\n// regex: `{% macro %}` and `{% block %}` become symbols, and\n// `{% include / extends / import / from \"path\" %}` become import edges.\n// Plain HTML with no HubL tags simply yields no symbols (same as before).\n\nimport type { ParsedFile, ParsedSymbol } from \"../parser.js\";\nimport type { WalkedFile } from \"../walker.js\";\n\n// `{%-` / `-%}` are HubL/Jinja whitespace-control variants — tolerate them.\nconst MACRO_RE = /\\{%-?\\s*macro\\s+([A-Za-z_]\\w*)\\s*\\(([^)]*)\\)/g;\nconst ENDMACRO_RE = /\\{%-?\\s*endmacro\\b/g;\nconst BLOCK_RE = /\\{%-?\\s*block\\s+([A-Za-z_]\\w*)/g;\nconst ENDBLOCK_RE = /\\{%-?\\s*endblock\\b/g;\n// include / extends / import / from — all take a quoted template path first.\nconst IMPORT_RE = /\\{%-?\\s*(?:include|extends|import|from)\\s+[\"']([^\"']+)[\"']/g;\n\nfunction lineAt(source: string, index: number): number {\n return source.slice(0, index).split(/\\r?\\n/).length;\n}\n\n// Line of the next matching close tag after `fromIndex`; falls back to the\n// start line if the template is unbalanced (no close found).\nfunction endLineAfter(source: string, fromIndex: number, endRe: RegExp, startLine: number): number {\n endRe.lastIndex = fromIndex;\n const m = endRe.exec(source);\n return m ? lineAt(source, m.index) : startLine;\n}\n\nexport function parseHubL(f: WalkedFile, source: string): ParsedFile {\n const symbols: ParsedSymbol[] = [];\n const imports: string[] = [];\n\n for (const m of source.matchAll(MACRO_RE)) {\n const name = m[1];\n if (!name) continue;\n const args = (m[2] ?? \"\").trim();\n const start = m.index ?? 0;\n const startLine = lineAt(source, start);\n symbols.push({\n name,\n kind: \"function\",\n startLine,\n endLine: endLineAfter(source, start + m[0].length, ENDMACRO_RE, startLine),\n signature: `macro ${name}(${args})`,\n });\n }\n\n for (const m of source.matchAll(BLOCK_RE)) {\n const name = m[1];\n if (!name) continue;\n const start = m.index ?? 0;\n const startLine = lineAt(source, start);\n symbols.push({\n name,\n kind: \"component\",\n startLine,\n endLine: endLineAfter(source, start + m[0].length, ENDBLOCK_RE, startLine),\n signature: `block ${name}`,\n });\n }\n\n for (const m of source.matchAll(IMPORT_RE)) {\n const spec = m[1];\n if (spec) imports.push(spec);\n }\n\n return { file: f, source, symbols, imports: Array.from(new Set(imports)), calls: [] };\n}\n","// Java parser. Classes, interfaces, methods, imports.\n\nimport type { ParsedFile } from \"../parser.js\";\nimport type { WalkedFile } from \"../walker.js\";\nimport { runGenericParser } from \"./_generic.js\";\n\nconst QUERY = `\n(class_declaration name: (identifier) @class.name) @class\n(interface_declaration name: (identifier) @interface.name) @interface\n(method_declaration name: (identifier) @method.name) @method\n(enum_declaration name: (identifier) @enum.name) @enum\n(import_declaration (scoped_identifier) @import)\n`;\n\nexport async function parseJava(f: WalkedFile, source: string): Promise<ParsedFile> {\n return runGenericParser(\n {\n grammar: \"java\",\n query: QUERY,\n decls: [\n { declCapture: \"class\", nameCapture: \"class.name\", kind: \"class\" },\n { declCapture: \"interface\", nameCapture: \"interface.name\", kind: \"interface\" },\n { declCapture: \"method\", nameCapture: \"method.name\", kind: \"method\" },\n { declCapture: \"enum\", nameCapture: \"enum.name\", kind: \"enum\" },\n ],\n importCapture: \"import\",\n },\n f,\n source,\n );\n}\n","// Kotlin parser. Functions, classes, objects, interfaces, imports.\n\nimport type { ParsedFile } from \"../parser.js\";\nimport type { WalkedFile } from \"../walker.js\";\nimport { runGenericParser } from \"./_generic.js\";\n\nconst QUERY = `\n(function_declaration (simple_identifier) @function.name) @function\n(class_declaration (type_identifier) @class.name) @class\n(object_declaration (type_identifier) @object.name) @object\n(import_header (identifier) @import)\n`;\n\nexport async function parseKotlin(f: WalkedFile, source: string): Promise<ParsedFile> {\n return runGenericParser(\n {\n grammar: \"kotlin\",\n query: QUERY,\n decls: [\n { declCapture: \"function\", nameCapture: \"function.name\", kind: \"function\" },\n { declCapture: \"class\", nameCapture: \"class.name\", kind: \"class\" },\n { declCapture: \"object\", nameCapture: \"object.name\", kind: \"class\" },\n ],\n importCapture: \"import\",\n },\n f,\n source,\n );\n}\n","// PHP parser. Functions, classes, interfaces, methods, traits.\n\nimport type { ParsedFile } from \"../parser.js\";\nimport type { WalkedFile } from \"../walker.js\";\nimport { runGenericParser } from \"./_generic.js\";\n\nconst QUERY = `\n(function_definition name: (name) @function.name) @function\n(class_declaration name: (name) @class.name) @class\n(interface_declaration name: (name) @interface.name) @interface\n(trait_declaration name: (name) @trait.name) @trait\n(method_declaration name: (name) @method.name) @method\n`;\n\nexport async function parsePhp(f: WalkedFile, source: string): Promise<ParsedFile> {\n return runGenericParser(\n {\n grammar: \"php\",\n query: QUERY,\n decls: [\n { declCapture: \"function\", nameCapture: \"function.name\", kind: \"function\" },\n { declCapture: \"class\", nameCapture: \"class.name\", kind: \"class\" },\n { declCapture: \"interface\", nameCapture: \"interface.name\", kind: \"interface\" },\n { declCapture: \"trait\", nameCapture: \"trait.name\", kind: \"class\" },\n { declCapture: \"method\", nameCapture: \"method.name\", kind: \"method\" },\n ],\n },\n f,\n source,\n );\n}\n","// Python parser using tree-sitter-python WASM.\n// Extracts: function/class definitions, methods, and import statements.\n\nimport { Query, type Node } from \"web-tree-sitter\";\nimport { createParser, type ParsedFile, type ParsedSymbol } from \"../parser.js\";\nimport type { WalkedFile } from \"../walker.js\";\n\nconst QUERY = `\n(function_definition name: (identifier) @function.name) @function\n(class_definition name: (identifier) @class.name) @class\n(import_statement name: (dotted_name) @import.module)\n(import_from_statement module_name: (dotted_name) @import.from)\n(import_from_statement module_name: (relative_import) @import.from)\n`;\n\nfunction firstLine(text: string, max = 200): string {\n const line = text.split(/\\r?\\n/, 1)[0] ?? \"\";\n return line.length > max ? line.slice(0, max) + \"…\" : line;\n}\n\nexport async function parsePython(f: WalkedFile, source: string): Promise<ParsedFile> {\n let symbols: ParsedSymbol[] = [];\n let imports: string[] = [];\n\n try {\n const { parser, language } = await createParser(\"python\");\n const tree = parser.parse(source);\n if (!tree) return { file: f, source, symbols, imports, calls: [] };\n\n const query = new Query(language, QUERY);\n const matches = query.matches(tree.rootNode);\n\n for (const match of matches) {\n const byName = new Map<string, Node>();\n for (const cap of match.captures) byName.set(cap.name, cap.node);\n\n const funcDecl = byName.get(\"function\");\n const funcName = byName.get(\"function.name\");\n if (funcDecl && funcName) {\n const parentType = funcDecl.parent?.parent?.type;\n const isMethod = parentType === \"class_definition\";\n symbols.push({\n name: funcName.text,\n kind: isMethod ? \"method\" : \"function\",\n startLine: funcDecl.startPosition.row + 1,\n endLine: funcDecl.endPosition.row + 1,\n signature: firstLine(funcDecl.text),\n });\n continue;\n }\n\n const classDecl = byName.get(\"class\");\n const className = byName.get(\"class.name\");\n if (classDecl && className) {\n symbols.push({\n name: className.text,\n kind: \"class\",\n startLine: classDecl.startPosition.row + 1,\n endLine: classDecl.endPosition.row + 1,\n signature: firstLine(classDecl.text),\n });\n continue;\n }\n\n const importNode = byName.get(\"import.module\") ?? byName.get(\"import.from\");\n if (importNode) imports.push(importNode.text);\n }\n\n const seen = new Set<string>();\n symbols = symbols.filter((s) => {\n const key = `${s.name}:${s.startLine}`;\n if (seen.has(key)) return false;\n seen.add(key);\n return true;\n });\n imports = Array.from(new Set(imports));\n } catch {\n // swallow parse errors\n }\n\n return { file: f, source, symbols, imports, calls: [] };\n}\n","// Ruby parser. Methods, classes, modules.\n// Imports omitted — Ruby's `require` is dynamic and hard to capture cleanly;\n// keyword indexing still surfaces dependencies.\n\nimport type { ParsedFile } from \"../parser.js\";\nimport type { WalkedFile } from \"../walker.js\";\nimport { runGenericParser } from \"./_generic.js\";\n\nconst QUERY = `\n(method name: (identifier) @function.name) @function\n(singleton_method name: (identifier) @method.name) @method\n(class name: (constant) @class.name) @class\n(module name: (constant) @module.name) @module\n`;\n\nexport async function parseRuby(f: WalkedFile, source: string): Promise<ParsedFile> {\n return runGenericParser(\n {\n grammar: \"ruby\",\n query: QUERY,\n decls: [\n { declCapture: \"function\", nameCapture: \"function.name\", kind: \"function\" },\n { declCapture: \"method\", nameCapture: \"method.name\", kind: \"method\" },\n { declCapture: \"class\", nameCapture: \"class.name\", kind: \"class\" },\n { declCapture: \"module\", nameCapture: \"module.name\", kind: \"class\" },\n ],\n },\n f,\n source,\n );\n}\n","// Rust parser. Functions, structs, enums, traits, impls.\n// Import capture is omitted — `use` paths are nested and complex; the file\n// will still be walked + keyword-indexed.\n\nimport type { ParsedFile } from \"../parser.js\";\nimport type { WalkedFile } from \"../walker.js\";\nimport { runGenericParser } from \"./_generic.js\";\n\nconst QUERY = `\n(function_item name: (identifier) @function.name) @function\n(struct_item name: (type_identifier) @struct.name) @struct\n(enum_item name: (type_identifier) @enum.name) @enum\n(trait_item name: (type_identifier) @trait.name) @trait\n(impl_item type: (type_identifier) @impl.name) @impl\n`;\n\nexport async function parseRust(f: WalkedFile, source: string): Promise<ParsedFile> {\n return runGenericParser(\n {\n grammar: \"rust\",\n query: QUERY,\n decls: [\n { declCapture: \"function\", nameCapture: \"function.name\", kind: \"function\" },\n { declCapture: \"struct\", nameCapture: \"struct.name\", kind: \"class\" },\n { declCapture: \"enum\", nameCapture: \"enum.name\", kind: \"enum\" },\n { declCapture: \"trait\", nameCapture: \"trait.name\", kind: \"interface\" },\n { declCapture: \"impl\", nameCapture: \"impl.name\", kind: \"class\" },\n ],\n },\n f,\n source,\n );\n}\n","// TS/JS parser using tree-sitter-typescript / -tsx WASM grammars.\r\n// Extracts: function/class/interface/type/enum declarations, exported consts,\r\n// arrow functions assigned to const, and import sources.\r\n\r\nimport { Query, type Node } from \"web-tree-sitter\";\r\nimport type { SymbolKind } from \"../../graph/types.js\";\r\nimport { createParser, type GrammarName, type ParsedFile, type ParsedSymbol } from \"../parser.js\";\r\nimport type { WalkedFile } from \"../walker.js\";\r\n\r\n// TS / TSX query — uses the type-identifier node type for class names, includes\r\n// interface / type-alias / enum declarations that don't exist in plain JS.\r\nconst TS_QUERY = `\r\n(function_declaration name: (identifier) @function.name) @function\r\n(class_declaration name: (type_identifier) @class.name) @class\r\n(interface_declaration name: (type_identifier) @interface.name) @interface\r\n(type_alias_declaration name: (type_identifier) @type.name) @type\r\n(enum_declaration name: (identifier) @enum.name) @enum\r\n(method_definition name: (property_identifier) @method.name) @method\r\n(lexical_declaration (variable_declarator name: (identifier) @const-fn.name value: [(arrow_function) (function_expression)])) @const-fn\r\n(import_statement source: (string) @import)\r\n`;\r\n\r\n// JS query — class names are plain identifiers (JS grammar has no\r\n// type_identifier node). No interface / type_alias / enum since JS lacks them.\r\n// Adds a call_expression capture for CommonJS require('x'); filtered in the\r\n// matching loop by checking the function identifier text equals \"require\".\r\nconst JS_QUERY = `\r\n(function_declaration name: (identifier) @function.name) @function\r\n(class_declaration name: (identifier) @class.name) @class\r\n(method_definition name: (property_identifier) @method.name) @method\r\n(lexical_declaration (variable_declarator name: (identifier) @const-fn.name value: [(arrow_function) (function_expression)])) @const-fn\r\n(import_statement source: (string) @import)\r\n(call_expression function: (identifier) @_require_fn arguments: (arguments . (string) @require_source))\r\n`;\r\n\r\nfunction grammarFor(ext: string): GrammarName {\r\n if (ext === \".tsx\" || ext === \".jsx\") return \"tsx\";\r\n if (ext === \".js\" || ext === \".cjs\" || ext === \".mjs\") return \"javascript\";\r\n return \"typescript\";\r\n}\r\n\r\nfunction queryFor(grammar: GrammarName): string {\r\n return grammar === \"javascript\" ? JS_QUERY : TS_QUERY;\r\n}\r\n\r\nfunction unquote(s: string): string {\r\n return s.replace(/^[\"'`]|[\"'`]$/g, \"\");\r\n}\r\n\r\nfunction firstLine(text: string, max = 200): string {\r\n const line = text.split(/\\r?\\n/, 1)[0] ?? \"\";\r\n return line.length > max ? line.slice(0, max) + \"…\" : line;\r\n}\r\n\r\ninterface DeclShape {\r\n decl: Node;\r\n name: Node;\r\n kind: SymbolKind;\r\n}\r\n\r\nfunction shapeFromCaptures(captures: Map<string, Node>): DeclShape | null {\r\n const findDecl = (k: string, sk: SymbolKind): DeclShape | null => {\r\n const decl = captures.get(k);\r\n const name = captures.get(`${k}.name`);\r\n return decl && name ? { decl, name, kind: sk } : null;\r\n };\r\n\r\n return (\r\n findDecl(\"function\", \"function\") ??\r\n findDecl(\"class\", \"class\") ??\r\n findDecl(\"interface\", \"interface\") ??\r\n findDecl(\"type\", \"type\") ??\r\n findDecl(\"enum\", \"enum\") ??\r\n findDecl(\"method\", \"method\") ??\r\n findDecl(\"const-fn\", \"function\")\r\n );\r\n}\r\n\r\nexport async function parseTypeScript(f: WalkedFile, source: string): Promise<ParsedFile> {\r\n const grammar = grammarFor(f.ext);\r\n let symbols: ParsedSymbol[] = [];\r\n let imports: string[] = [];\r\n\r\n try {\r\n const { parser, language } = await createParser(grammar);\r\n const tree = parser.parse(source);\r\n if (!tree) return { file: f, source, symbols, imports, calls: [] };\r\n\r\n const query = new Query(language, queryFor(grammar));\r\n const matches = query.matches(tree.rootNode);\r\n\r\n for (const match of matches) {\r\n const byName = new Map<string, Node>();\r\n for (const cap of match.captures) byName.set(cap.name, cap.node);\r\n\r\n const shape = shapeFromCaptures(byName);\r\n if (shape) {\r\n symbols.push({\r\n name: shape.name.text,\r\n kind: shape.kind,\r\n startLine: shape.decl.startPosition.row + 1,\r\n endLine: shape.decl.endPosition.row + 1,\r\n signature: firstLine(shape.decl.text),\r\n });\r\n continue;\r\n }\r\n const importNode = byName.get(\"import\");\r\n if (importNode) {\r\n imports.push(unquote(importNode.text));\r\n continue;\r\n }\r\n // CommonJS require('x') — only captured by JS_QUERY. The identifier\r\n // must literally be \"require\" (not setTimeout, console, etc).\r\n const requireFn = byName.get(\"_require_fn\");\r\n const requireSource = byName.get(\"require_source\");\r\n if (requireFn && requireSource && requireFn.text === \"require\") {\r\n imports.push(unquote(requireSource.text));\r\n }\r\n }\r\n\r\n const seen = new Set<string>();\r\n symbols = symbols.filter((s) => {\r\n const key = `${s.name}:${s.startLine}`;\r\n if (seen.has(key)) return false;\r\n seen.add(key);\r\n return true;\r\n });\r\n imports = Array.from(new Set(imports));\r\n } catch {\r\n // Parse failure shouldn't abort the whole scan — return what we have.\r\n }\r\n\r\n return { file: f, source, symbols, imports, calls: [] };\r\n}\r\n","// Svelte parser. Extracts <script> and <script lang=\"ts\"> blocks and parses\n// their contents with the TypeScript parser. Tracks the original line offset\n// so reported symbol positions match the .svelte source.\n\nimport { parseTypeScript } from \"./typescript.js\";\nimport type { ParsedFile } from \"../parser.js\";\nimport type { WalkedFile } from \"../walker.js\";\n\nconst SCRIPT_RE = /<script\\b[^>]*>([\\s\\S]*?)<\\/script>/gi;\n\ninterface ScriptBlock {\n source: string;\n startLine: number; // 1-based line number where the script content begins\n isTsx: boolean;\n}\n\nfunction extractScripts(source: string): ScriptBlock[] {\n const out: ScriptBlock[] = [];\n for (const match of source.matchAll(SCRIPT_RE)) {\n const full = match[0];\n const inner = match[1] ?? \"\";\n const openTag = full.slice(0, full.indexOf(\">\") + 1);\n const tagStart = match.index ?? 0;\n const contentStart = tagStart + openTag.length;\n const startLine = source.slice(0, contentStart).split(/\\r?\\n/).length;\n const isTsx = /\\blang\\s*=\\s*[\"']?(ts|tsx|typescript)[\"']?/i.test(openTag);\n out.push({ source: inner, startLine, isTsx });\n }\n return out;\n}\n\nexport async function parseSvelte(f: WalkedFile, source: string): Promise<ParsedFile> {\n const blocks = extractScripts(source);\n const out: ParsedFile = { file: f, source, symbols: [], imports: [], calls: [] };\n\n for (const block of blocks) {\n const virtual: WalkedFile = { ...f, ext: block.isTsx ? \".ts\" : \".js\" };\n const parsed = await parseTypeScript(virtual, block.source);\n const offset = block.startLine - 1;\n for (const sym of parsed.symbols) {\n out.symbols.push({\n ...sym,\n startLine: sym.startLine + offset,\n endLine: sym.endLine + offset,\n });\n }\n for (const imp of parsed.imports) out.imports.push(imp);\n }\n\n // The .svelte file itself is treated as a component.\n out.symbols.push({\n name: f.relPath.split(\"/\").pop()?.replace(/\\.svelte$/i, \"\") ?? f.relPath,\n kind: \"component\",\n startLine: 1,\n endLine: source.split(/\\r?\\n/).length,\n signature: f.relPath,\n });\n out.imports = Array.from(new Set(out.imports));\n return out;\n}\n","// Vue SFC parser. Extracts <script> / <script setup> / <script lang=\"ts\">\n// blocks and parses them with the TypeScript parser, preserving line offsets.\n\nimport { parseTypeScript } from \"./typescript.js\";\nimport type { ParsedFile } from \"../parser.js\";\nimport type { WalkedFile } from \"../walker.js\";\n\nconst SCRIPT_RE = /<script\\b[^>]*>([\\s\\S]*?)<\\/script>/gi;\n\ninterface ScriptBlock {\n source: string;\n startLine: number;\n isTs: boolean;\n}\n\nfunction extractScripts(source: string): ScriptBlock[] {\n const out: ScriptBlock[] = [];\n for (const match of source.matchAll(SCRIPT_RE)) {\n const full = match[0];\n const inner = match[1] ?? \"\";\n const openTag = full.slice(0, full.indexOf(\">\") + 1);\n const tagStart = match.index ?? 0;\n const contentStart = tagStart + openTag.length;\n const startLine = source.slice(0, contentStart).split(/\\r?\\n/).length;\n const isTs = /\\blang\\s*=\\s*[\"']?(ts|tsx|typescript)[\"']?/i.test(openTag);\n out.push({ source: inner, startLine, isTs });\n }\n return out;\n}\n\nexport async function parseVue(f: WalkedFile, source: string): Promise<ParsedFile> {\n const blocks = extractScripts(source);\n const out: ParsedFile = { file: f, source, symbols: [], imports: [], calls: [] };\n\n for (const block of blocks) {\n const virtual: WalkedFile = { ...f, ext: block.isTs ? \".ts\" : \".js\" };\n const parsed = await parseTypeScript(virtual, block.source);\n const offset = block.startLine - 1;\n for (const sym of parsed.symbols) {\n out.symbols.push({\n ...sym,\n startLine: sym.startLine + offset,\n endLine: sym.endLine + offset,\n });\n }\n for (const imp of parsed.imports) out.imports.push(imp);\n }\n\n out.symbols.push({\n name: f.relPath.split(\"/\").pop()?.replace(/\\.vue$/i, \"\") ?? f.relPath,\n kind: \"component\",\n startLine: 1,\n endLine: source.split(/\\r?\\n/).length,\n signature: f.relPath,\n });\n out.imports = Array.from(new Set(out.imports));\n return out;\n}\n","// Walks project root, yields files to parse.\n// Honors .gitignore + .synthraignore (additive — entries in either are ignored).\n// Defensive defaults skip VCS, build, and dependency directories even if absent\n// from .gitignore.\n\nimport { readFile, readdir, stat } from \"node:fs/promises\";\nimport { extname, join, relative, sep } from \"node:path\";\nimport ignore, { type Ignore } from \"ignore\";\n\nexport interface WalkedFile {\n absPath: string;\n relPath: string;\n ext: string;\n size: number;\n}\n\nexport interface WalkOptions {\n /** Maximum file size to yield (bytes). Defaults to 2 MB. */\n maxFileSize?: number;\n /** Additional ignore patterns layered on top of .gitignore + .synthraignore. */\n extraIgnore?: string[];\n}\n\nconst DEFAULT_IGNORE = [\n \".git/\",\n \".synthra/\",\n \".synthra-graph/\",\n \".claude/\",\n \"node_modules/\",\n \"dist/\",\n \"build/\",\n \"out/\",\n \"coverage/\",\n \".next/\",\n \".nuxt/\",\n \".svelte-kit/\",\n \".turbo/\",\n \".cache/\",\n \".vscode/\",\n \".idea/\",\n \".vs/\",\n // Flutter / Dart build caches — IDE-rehydrated, contain third-party\n // type stubs (typescript.d.ts, babylon.js etc.) that contaminate the graph.\n \".dart_tool/\",\n \".flutter-plugins\",\n \".flutter-plugins-dependencies\",\n // Android / Java / Kotlin / Rust\n \".gradle/\",\n \"target/\",\n // iOS / Xcode\n \"Pods/\",\n \"DerivedData/\",\n // Python\n \"__pycache__/\",\n \".venv/\",\n \"venv/\",\n \".tox/\",\n \".pytest_cache/\",\n \".mypy_cache/\",\n \".ruff_cache/\",\n // .NET\n \"obj/\",\n];\n\nconst BINARY_EXTS = new Set([\n \".png\", \".jpg\", \".jpeg\", \".gif\", \".webp\", \".svg\", \".ico\", \".bmp\",\n \".pdf\", \".zip\", \".tar\", \".gz\", \".7z\", \".rar\",\n \".mp3\", \".mp4\", \".mov\", \".avi\", \".webm\", \".wav\", \".ogg\",\n \".ttf\", \".otf\", \".woff\", \".woff2\", \".eot\",\n \".exe\", \".dll\", \".so\", \".dylib\", \".bin\", \".wasm\",\n \".lock\", \".lockb\",\n]);\n\nasync function readIgnoreFile(path: string): Promise<string[]> {\n try {\n const text = await readFile(path, \"utf8\");\n return text\n .split(/\\r?\\n/)\n .map((l) => l.trim())\n .filter((l) => l.length > 0 && !l.startsWith(\"#\"));\n } catch {\n return [];\n }\n}\n\nasync function buildMatcher(root: string, extra: string[]): Promise<Ignore> {\n const ig = ignore();\n ig.add(DEFAULT_IGNORE);\n ig.add(await readIgnoreFile(join(root, \".gitignore\")));\n ig.add(await readIgnoreFile(join(root, \".synthraignore\")));\n if (extra.length) ig.add(extra);\n return ig;\n}\n\nfunction toPosix(p: string): string {\n return sep === \"/\" ? p : p.split(sep).join(\"/\");\n}\n\nexport async function* walk(\n root: string,\n options: WalkOptions = {},\n): AsyncGenerator<WalkedFile> {\n const maxFileSize = options.maxFileSize ?? 2_000_000;\n const ig = await buildMatcher(root, options.extraIgnore ?? []);\n\n async function* recurse(dir: string): AsyncGenerator<WalkedFile> {\n let entries;\n try {\n entries = await readdir(dir, { withFileTypes: true });\n } catch {\n return;\n }\n for (const entry of entries) {\n const abs = join(dir, entry.name);\n const rel = relative(root, abs);\n if (!rel) continue;\n const relPosix = toPosix(rel);\n const matchPath = entry.isDirectory() ? `${relPosix}/` : relPosix;\n if (ig.ignores(matchPath)) continue;\n\n if (entry.isDirectory()) {\n yield* recurse(abs);\n } else if (entry.isFile()) {\n const ext = extname(entry.name).toLowerCase();\n if (BINARY_EXTS.has(ext)) continue;\n let size: number;\n try {\n const s = await stat(abs);\n size = s.size;\n } catch {\n continue;\n }\n if (size > maxFileSize) continue;\n yield { absPath: abs, relPath: relPosix, ext, size };\n }\n }\n }\n\n yield* recurse(root);\n}\n","// Reads/writes info_graph.json and symbol_index.json.\n\nimport { mkdir, readFile, writeFile } from \"node:fs/promises\";\nimport { dirname } from \"node:path\";\n\nimport type { GraphSchema, SymbolIndex } from \"./types.js\";\n\nasync function writeJson(path: string, data: unknown, pretty: boolean): Promise<void> {\n await mkdir(dirname(path), { recursive: true });\n const text = pretty ? JSON.stringify(data, null, 2) : JSON.stringify(data);\n await writeFile(path, text + \"\\n\", \"utf8\");\n}\n\nasync function readJson<T>(path: string): Promise<T> {\n const text = await readFile(path, \"utf8\");\n return JSON.parse(text) as T;\n}\n\nexport async function writeGraph(path: string, graph: GraphSchema): Promise<void> {\n // Pretty-printing a graph with full file contents balloons disk size and\n // the JSON is only ever read by machines; keep it compact.\n await writeJson(path, graph, false);\n}\n\nexport async function readGraph(path: string): Promise<GraphSchema> {\n return readJson<GraphSchema>(path);\n}\n\nexport async function writeSymbolIndex(path: string, index: SymbolIndex): Promise<void> {\n await writeJson(path, index, true);\n}\n\nexport async function readSymbolIndex(path: string): Promise<SymbolIndex> {\n // Re-home onto a null prototype so name lookups (e.g. index[\"toString\"])\n // never resolve to an inherited Object.prototype member. Mirrors\n // buildSymbolIndex, which builds the index the same way.\n const parsed = await readJson<SymbolIndex>(path);\n return Object.assign(Object.create(null), parsed);\n}\n","// Resolves Synthra's storage locations inside a project root.\n\nimport { join } from \"node:path\";\n\nexport interface SynthraPaths {\n projectRoot: string;\n graphDir: string;\n contextDir: string;\n infoGraph: string;\n symbolIndex: string;\n sessionState: string;\n activityLog: string;\n tokenLog: string;\n gateLog: string;\n mcpPort: string;\n mcpServerLog: string;\n mcpServerErrLog: string;\n contextStore: string;\n contextMd: string;\n branchesDir: string;\n claudeDir: string;\n claudeSettings: string;\n claudeHooksDir: string;\n claudeMd: string;\n gitignore: string;\n}\n\nexport function resolvePaths(projectRoot: string): SynthraPaths {\n const graphDir = join(projectRoot, \".synthra-graph\");\n const contextDir = join(projectRoot, \".synthra\");\n const claudeDir = join(projectRoot, \".claude\");\n\n return {\n projectRoot,\n graphDir,\n contextDir,\n infoGraph: join(graphDir, \"info_graph.json\"),\n symbolIndex: join(graphDir, \"symbol_index.json\"),\n sessionState: join(graphDir, \"session.json\"),\n activityLog: join(graphDir, \"activity.jsonl\"),\n tokenLog: join(graphDir, \"token_log.jsonl\"),\n gateLog: join(graphDir, \"gate_log.jsonl\"),\n mcpPort: join(graphDir, \"mcp_port\"),\n mcpServerLog: join(graphDir, \"mcp_server.log\"),\n mcpServerErrLog: join(graphDir, \"mcp_server.err.log\"),\n contextStore: join(contextDir, \"context-store.json\"),\n contextMd: join(contextDir, \"CONTEXT.md\"),\n branchesDir: join(contextDir, \"branches\"),\n claudeDir,\n claudeSettings: join(claudeDir, \"settings.local.json\"),\n claudeHooksDir: join(claudeDir, \"hooks\"),\n claudeMd: join(projectRoot, \"CLAUDE.md\"),\n gitignore: join(projectRoot, \".gitignore\"),\n };\n}\n","// Project bootstrap: creates .synthra-graph/, .synthra/, updates .gitignore,\r\n// patches CLAUDE.md with the versioned policy block.\r\n\r\nimport { mkdir, readFile, stat, writeFile } from \"node:fs/promises\";\r\nimport { basename } from \"node:path\";\r\n\r\nimport { patchClaudeMd } from \"../hooks/claude-md.js\";\r\nimport type { SynthraPaths } from \"../shared/paths.js\";\r\n\r\nexport interface BootstrapResult {\r\n graphCreated: boolean;\r\n contextCreated: boolean;\r\n gitignoreUpdated: boolean;\r\n claudeMdUpdated: boolean;\r\n claudeMdCreated: boolean;\r\n}\r\n\r\n// Entries Synthra appends to the project .gitignore on bootstrap.\r\n// Each is gated by a check: if the entry is already present (any\r\n// indentation, trimmed match), it's skipped. Comments are per-entry so\r\n// users understand why each line is there and can remove what they don't\r\n// want without breaking the rest.\r\nconst GITIGNORE_ENTRIES: { comment: string; entry: string }[] = [\r\n {\r\n comment: \"added by synthra (heavy generated state — gitignored by design)\",\r\n entry: \".synthra-graph/\",\r\n },\r\n {\r\n comment:\r\n \"added by synthra — MCP registration. Remove this line if you want \" +\r\n \"to share the synthra MCP entry with teammates via committed .mcp.json\",\r\n entry: \".mcp.json\",\r\n },\r\n];\r\n\r\nasync function exists(path: string): Promise<boolean> {\r\n try {\r\n await stat(path);\r\n return true;\r\n } catch {\r\n return false;\r\n }\r\n}\r\n\r\nasync function ensureDir(path: string): Promise<boolean> {\r\n const had = await exists(path);\r\n await mkdir(path, { recursive: true });\r\n return !had;\r\n}\r\n\r\nasync function patchGitignore(path: string): Promise<boolean> {\r\n let existing = \"\";\r\n try {\r\n existing = await readFile(path, \"utf8\");\r\n } catch {\r\n /* file may not exist */\r\n }\r\n const trimmed = new Set(existing.split(/\\r?\\n/).map((l) => l.trim()));\r\n const missing = GITIGNORE_ENTRIES.filter((e) => !trimmed.has(e.entry));\r\n if (missing.length === 0) return false;\r\n\r\n const block =\r\n missing.map((m) => `# ${m.comment}\\n${m.entry}`).join(\"\\n\") + \"\\n\";\r\n const appendix =\r\n (existing.length === 0 || existing.endsWith(\"\\n\") ? \"\" : \"\\n\") +\r\n (existing.length ? \"\\n\" : \"\") +\r\n block;\r\n await writeFile(path, existing + appendix, \"utf8\");\r\n return true;\r\n}\r\n\r\nexport async function bootstrap(paths: SynthraPaths): Promise<BootstrapResult> {\r\n const graphCreated = await ensureDir(paths.graphDir);\r\n const contextCreated = await ensureDir(paths.contextDir);\r\n const gitignoreUpdated = await patchGitignore(paths.gitignore);\r\n\r\n const claudeMdExistedBefore = await exists(paths.claudeMd);\r\n const patch = await patchClaudeMd(paths.claudeMd, basename(paths.projectRoot));\r\n\r\n return {\r\n graphCreated,\r\n contextCreated,\r\n gitignoreUpdated,\r\n claudeMdUpdated: patch.updated,\r\n claudeMdCreated: patch.created && !claudeMdExistedBefore,\r\n };\r\n}\r\n","// Idempotent patcher for the project's CLAUDE.md. Manages a single block\r\n// bounded by <!-- synthra-policy v<N> BEGIN --> ... <!-- synthra-policy v<N> END -->.\r\n// On each run, any prior synthra-policy block (any version) is removed and the\r\n// current-version block is appended at the end.\r\n\r\nimport { readFile, writeFile } from \"node:fs/promises\";\r\nimport { basename, dirname } from \"node:path\";\r\n\r\nexport const POLICY_VERSION = 4;\r\nexport const POLICY_BEGIN = `<!-- synthra-policy v${POLICY_VERSION} BEGIN -->`;\r\nexport const POLICY_END = `<!-- synthra-policy v${POLICY_VERSION} END -->`;\r\n\r\n// Matches a synthra-policy block of any version, e.g. v1, v2 …\r\nconst ANY_BLOCK_RE = /<!--\\s*synthra-policy\\s+v\\d+\\s+BEGIN\\s*-->[\\s\\S]*?<!--\\s*synthra-policy\\s+v\\d+\\s+END\\s*-->\\s*/g;\r\n\r\nexport interface PatchResult {\r\n created: boolean;\r\n updated: boolean;\r\n skipped: boolean;\r\n}\r\n\r\nexport function policyBlock(): string {\r\n return [\r\n POLICY_BEGIN,\r\n \"## Synthra context policy\",\r\n \"\",\r\n \"Synthra has pre-loaded structured context into this session and exposes\",\r\n \"the project's code graph through MCP tools. **Prefer these tools over\",\r\n \"Grep / Glob / Read** — they are faster, cheaper, and already filtered\",\r\n \"to relevant files.\",\r\n \"\",\r\n \"> **Tool namespace.** Synthra's MCP tools are exposed as\",\r\n \"> `mcp__synthra__graph_continue`, `mcp__synthra__graph_read`, and\",\r\n \"> `mcp__synthra__graph_register_edit`. Below they are referred to by\",\r\n \"> their short names (`graph_continue` etc.) for readability — use the\",\r\n \"> full namespaced form when actually invoking them.\",\r\n \"\",\r\n \"### Tools\",\r\n \"\",\r\n \"- **`graph_continue(query)`** — returns a `Confidence` label, the list\",\r\n \" of relevant `Files`, and signatures + top function bodies for those\",\r\n \" files. Your default first move when you need project context.\",\r\n \"- **`graph_read(target)`** — fetch source. Prefer the\",\r\n \" `\\\"file/path.ts::SymbolName\\\"` form over a bare file path — reading one\",\r\n \" symbol is ~50 tokens, reading a whole file is thousands.\",\r\n \"- **`graph_register_edit(files)`** — after you edit files, call this so\",\r\n \" subsequent turns weight your changes and avoid stale snapshots.\",\r\n \"\",\r\n \"### When to call `graph_continue` — and when to skip\",\r\n \"\",\r\n \"**Call `graph_continue` only when you do NOT already know the relevant\",\r\n \"files.**\",\r\n \"\",\r\n \"Call it when:\",\r\n \"- This is the first message of a new task or conversation\",\r\n \"- The task shifts to a different area of the codebase\",\r\n \"- You need files you haven't seen yet in this session\",\r\n \"\",\r\n \"**Skip `graph_continue` when:**\",\r\n \"- You already identified the relevant files earlier in this conversation\",\r\n \"- You are doing follow-up work on files already read (verify, refactor,\",\r\n \" test, docs, cleanup, commit)\",\r\n \"- The task is pure text (commit message, explanation, summary)\",\r\n \"\",\r\n \"If skipping, go directly to `graph_read(\\\"file.ts::symbol\\\")` on what\",\r\n \"you already know.\",\r\n \"\",\r\n \"### Confidence caps\",\r\n \"\",\r\n \"When `graph_continue` returns:\",\r\n \"\",\r\n \"- **`Confidence: high`** → Stop. Do NOT Grep, Glob, or further explore\",\r\n \" for this query. The graph already has it.\",\r\n \"- **`Confidence: medium`** → Read the listed `Files` directly via\",\r\n \" `graph_read(\\\"file::symbol\\\")` *before* trying Grep. The graph has\",\r\n \" narrowed the search space — use it, don't bypass it.\",\r\n \"- **`Confidence: low`** → You may use Grep / Glob, but the PreToolUse\",\r\n \" hook may still block redundant calls.\",\r\n \"\",\r\n \"### Reading code\",\r\n \"\",\r\n \"- **Always use `file::symbol` notation** with `graph_read`. Whole-file\",\r\n \" reads should be rare — only when you genuinely need the full file.\",\r\n \"- If `graph_continue`'s `Files` list contains a `::` entry, pass it\",\r\n \" verbatim to `graph_read`.\",\r\n \"\",\r\n \"### Editing a file\",\r\n \"\",\r\n \"Claude Code's `Edit` tool (and `Write` when overwriting) only accepts a\",\r\n \"file that was opened with the **`Read` tool** — a `graph_read` slice does\",\r\n \"not count, and editing such a file fails with *\\\"File has not been read\",\r\n \"yet.\\\"* So before editing a file you only know through `graph_read`: take\",\r\n \"the line range from its header (e.g. `…::handler (L120-168)`), `Read` that\",\r\n \"file with a matching `offset`/`limit`, then `Edit`. That satisfies the\",\r\n \"gate while keeping the read small — don't whole-file `Read` unless the\",\r\n \"edit spans most of the file.\",\r\n \"\",\r\n \"### Don'ts\",\r\n \"\",\r\n \"- Don't Grep / Glob before calling `graph_continue` when required — the\",\r\n \" PreToolUse hook may block it.\",\r\n \"- Don't call `graph_continue` more than once per turn.\",\r\n \"- Don't read whole files when a symbol-level read would suffice.\",\r\n \"\",\r\n \"### Session-end resume note\",\r\n \"\",\r\n \"When the user signals they're done (e.g. \\\"bye\\\", \\\"wrap up\\\", \\\"done\\\"),\",\r\n \"persist the resume state by calling `context_remember` once per bullet.\",\r\n \"Synthra re-renders `.synthra/CONTEXT.md` from those entries at session\",\r\n \"end — do **NOT** write to `CONTEXT.md` directly, it is a derived view\",\r\n \"and direct edits are overwritten by the Stop hook.\",\r\n \"\",\r\n \"Use these `kind` values:\",\r\n \"\",\r\n \"- **`kind: \\\"task\\\"`** — what is being worked on right now (1 entry)\",\r\n \"- **`kind: \\\"decision\\\"`** — non-obvious choices made this session (max 3)\",\r\n \"- **`kind: \\\"next\\\"`** — concrete next steps (max 3)\",\r\n \"\",\r\n \"Tag entries with the relevant area (`tags: [\\\"auth\\\"]`) and the files\",\r\n \"they touch (`files: [\\\"src/auth.ts\\\"]`) so later `context_recall` queries\",\r\n \"can filter. Keep each `text` to 1–2 sentences.\",\r\n \"\",\r\n \"_This block is managed by Synthra. Edits inside the BEGIN/END markers\",\r\n \"are overwritten on every `syn .` run._\",\r\n \"\",\r\n POLICY_END,\r\n ].join(\"\\n\");\r\n}\r\n\r\n// A lean, agent-facing onboarding skeleton written ONLY when a project has no\r\n// CLAUDE.md yet. It captures the durable \"why/how\" the graph can't infer\r\n// (build/test, conventions, decisions, gotchas). It lives OUTSIDE the\r\n// synthra-policy markers, so later `syn .` runs — which strip and re-add the\r\n// policy block — never touch what the user fills in here.\r\nexport function onboardingSkeleton(projectName: string): string {\r\n return [\r\n `# ${projectName}`,\r\n \"\",\r\n \"> Onboarding notes for AI coding agents. Synthra's graph already knows the\",\r\n \"> code's *structure* (files, symbols, imports) — this file is for what the\",\r\n \"> graph can't infer: how to run the project, its conventions, and the\",\r\n \"> decisions behind them. Keep it lean and current; delete prompts you don't need.\",\r\n \"\",\r\n \"## Build & test\",\r\n \"\",\r\n \"- TODO: install deps / build\",\r\n \"- TODO: run tests / lint / typecheck\",\r\n \"- TODO: run the app locally\",\r\n \"\",\r\n \"## Conventions\",\r\n \"\",\r\n \"- TODO: code style, naming, file layout the agent should follow\",\r\n \"\",\r\n \"## Key decisions\",\r\n \"\",\r\n \"- TODO: non-obvious choices and *why* (\\\"we use X not Y because …\\\")\",\r\n \"\",\r\n \"## Gotchas\",\r\n \"\",\r\n \"- TODO: traps, footguns, \\\"don't touch X without Y\\\"\",\r\n \"\",\r\n \"_Tip: run `/init` in Claude Code to auto-draft the sections above, then trim\",\r\n \"to the durable bits. Synthra manages its own block below — leave it._\",\r\n \"\",\r\n ].join(\"\\n\");\r\n}\r\n\r\nexport async function patchClaudeMd(path: string, projectName?: string): Promise<PatchResult> {\r\n let existing: string | null;\r\n try {\r\n existing = await readFile(path, \"utf8\");\r\n } catch {\r\n existing = null;\r\n }\r\n\r\n const block = policyBlock();\r\n\r\n if (existing === null) {\r\n // First creation: scaffold the onboarding skeleton (user-owned, written\r\n // once) followed by Synthra's managed policy block.\r\n const name = projectName || basename(dirname(path)) || \"this project\";\r\n await writeFile(path, onboardingSkeleton(name) + \"\\n\" + block + \"\\n\", \"utf8\");\r\n return { created: true, updated: false, skipped: false };\r\n }\r\n\r\n const stripped = existing.replace(ANY_BLOCK_RE, \"\");\r\n const hadBlock = stripped !== existing;\r\n\r\n const desired = stripped.endsWith(\"\\n\")\r\n ? stripped + \"\\n\" + block + \"\\n\"\r\n : (stripped.length ? stripped + \"\\n\\n\" : \"\") + block + \"\\n\";\r\n\r\n if (hadBlock && desired === existing) {\r\n return { created: false, updated: false, skipped: true };\r\n }\r\n\r\n await writeFile(path, desired, \"utf8\");\r\n return { created: false, updated: true, skipped: false };\r\n}\r\n","// Scoring for retrieved files. Combines:\n// - keyword overlap with the (already-tokenized) query\n// - symbol-name overlap (boosted: hits in a file's defined symbols matter more)\n// - import-graph proximity from session-known + recent paths (boost adjacent files)\n// - recency boost from activity log (placeholder — wired in M5)\n\nimport type { Edge, FileNode, GraphSchema, SymbolNode } from \"./types.js\";\n\nexport interface RankInputs {\n candidates: FileNode[];\n query: string;\n graph?: GraphSchema;\n recentlyEditedPaths?: string[];\n sessionKnownPaths?: string[];\n}\n\nexport interface ScoredFile {\n file: FileNode;\n score: number;\n reasons: string[];\n /** Total symbol-match weight (exact name = 3, partial substring = 1). */\n symHits: number;\n /** Count of symbols whose name a query token matched *exactly*. The gate\n * uses this — a partial substring match is too noisy (\"id\" ⊂ \"width\"). */\n exactSym: number;\n}\n\nconst STOPWORDS = new Set([\n \"a\", \"an\", \"and\", \"are\", \"as\", \"at\", \"be\", \"by\", \"for\", \"from\",\n \"has\", \"have\", \"in\", \"is\", \"it\", \"of\", \"on\", \"or\", \"that\", \"the\",\n \"this\", \"to\", \"was\", \"we\", \"with\", \"what\", \"where\", \"when\", \"why\",\n \"how\", \"do\", \"does\", \"i\", \"me\", \"my\", \"you\", \"your\", \"code\", \"file\",\n]);\n\nexport function tokenizeQuery(query: string): string[] {\n const tokens = query\n .toLowerCase()\n .split(/[^a-z0-9_]+/g)\n .filter((t) => t.length > 1 && !STOPWORDS.has(t));\n // Also split camelCase/snake_case for queries like \"AuthService\"\n const expanded = new Set<string>();\n for (const t of tokens) {\n expanded.add(t);\n const parts = t.match(/[a-z]+|[0-9]+/g) ?? [];\n for (const p of parts) if (p.length > 1) expanded.add(p);\n }\n return Array.from(expanded);\n}\n\nfunction indexSymbolsByFile(graph: GraphSchema | undefined): Map<string, SymbolNode[]> {\n const out = new Map<string, SymbolNode[]>();\n if (!graph) return out;\n for (const n of graph.nodes) {\n if (n.kind !== \"symbol\") continue;\n const list = out.get(n.file) ?? [];\n list.push(n);\n out.set(n.file, list);\n }\n return out;\n}\n\nfunction indexImportEdges(graph: GraphSchema | undefined): Map<string, Set<string>> {\n // file path → set of file paths it imports (1-hop)\n const out = new Map<string, Set<string>>();\n if (!graph) return out;\n const idToPath = new Map<string, string>();\n for (const n of graph.nodes) if (n.kind === \"file\") idToPath.set(n.id, n.path);\n for (const e of graph.edges as Edge[]) {\n if (e.kind !== \"imports\") continue;\n const from = idToPath.get(e.from);\n const to = idToPath.get(e.to);\n if (!from || !to) continue;\n const s = out.get(from) ?? new Set<string>();\n s.add(to);\n out.set(from, s);\n }\n return out;\n}\n\nexport function scoreFiles(inputs: RankInputs): ScoredFile[] {\n const qTokens = new Set(tokenizeQuery(inputs.query));\n const symbolsByFile = indexSymbolsByFile(inputs.graph);\n const importsFrom = indexImportEdges(inputs.graph);\n\n const seeds = new Set<string>(inputs.sessionKnownPaths ?? []);\n for (const p of inputs.recentlyEditedPaths ?? []) seeds.add(p);\n\n // First pass: keyword + symbol score\n const scored: ScoredFile[] = [];\n for (const file of inputs.candidates) {\n const reasons: string[] = [];\n let score = 0;\n\n // Keyword overlap\n let kwHits = 0;\n for (const kw of file.keywords) if (qTokens.has(kw)) kwHits += 1;\n if (kwHits) {\n score += kwHits * 2;\n reasons.push(`kw=${kwHits}`);\n }\n\n // Symbol-name overlap (higher signal than file-level keywords)\n const symbols = symbolsByFile.get(file.path) ?? [];\n let symHits = 0;\n let exactSym = 0;\n for (const sym of symbols) {\n const name = sym.name.toLowerCase();\n if (qTokens.has(name)) {\n symHits += 3;\n exactSym += 1;\n } else {\n // partial match: any query token is a substring of, or contained by, the symbol name\n for (const t of qTokens) {\n if (name.includes(t) || t.includes(name)) {\n symHits += 1;\n break;\n }\n }\n }\n }\n if (symHits) {\n score += symHits;\n reasons.push(`sym=${symHits}`);\n }\n\n // Path match: file path contains a query token\n const pathLower = file.path.toLowerCase();\n let pathHits = 0;\n for (const t of qTokens) if (pathLower.includes(t)) pathHits += 1;\n if (pathHits) {\n score += pathHits;\n reasons.push(`path=${pathHits}`);\n }\n\n if (seeds.has(file.path)) {\n score += 5;\n reasons.push(\"seed\");\n }\n\n scored.push({ file, score, reasons, symHits, exactSym });\n }\n\n // Second pass: 1-hop import-graph boost from any file already scored > 0\n const positivePaths = new Set(scored.filter((s) => s.score > 0).map((s) => s.file.path));\n if (positivePaths.size > 0) {\n for (const s of scored) {\n if (s.score > 0) continue;\n // Does any file that imports this one have a positive score?\n let importBoost = 0;\n for (const [from, tos] of importsFrom) {\n if (!positivePaths.has(from)) continue;\n if (tos.has(s.file.path)) {\n importBoost += 1;\n break;\n }\n }\n if (importBoost) {\n s.score += importBoost * 0.5;\n s.reasons.push(\"imp-adj\");\n }\n }\n }\n\n scored.sort((a, b) => b.score - a.score);\n return scored;\n}\n\nexport function rank(inputs: RankInputs): FileNode[] {\n return scoreFiles(inputs).map((s) => s.file);\n}\n","// Query-time retrieval. Given a natural-language query (or a symbol name),\n// returns a ranked list of FileNodes plus a confidence label.\n\nimport { scoreFiles, tokenizeQuery, type RankInputs } from \"./rank.js\";\nimport type { FileNode, GraphSchema } from \"./types.js\";\n\nexport interface RetrievalResult {\n files: FileNode[];\n confidence: \"high\" | \"medium\" | \"low\";\n reason: string;\n /** True if any returned file has a symbol whose name a query token matched\n * exactly — i.e. graph_read can serve a real `file::symbol` slice for this\n * query. When false, a Grep/Glob block would only force a fallback Read. */\n symbolMatched: boolean;\n}\n\nexport interface RetrieveOptions {\n topK?: number;\n recentlyEditedPaths?: string[];\n sessionKnownPaths?: string[];\n}\n\nexport async function retrieve(\n graph: GraphSchema,\n query: string,\n options: RetrieveOptions = {},\n): Promise<RetrievalResult> {\n const topK = options.topK ?? 12;\n const qTokens = tokenizeQuery(query);\n\n const allFiles: FileNode[] = graph.nodes.filter(\n (n): n is FileNode => n.kind === \"file\",\n );\n\n if (allFiles.length === 0 || qTokens.length === 0) {\n return {\n files: [],\n confidence: \"low\",\n reason: qTokens.length === 0 ? \"empty query\" : \"empty graph\",\n symbolMatched: false,\n };\n }\n\n const rankInputs: RankInputs = {\n candidates: allFiles,\n query,\n graph,\n recentlyEditedPaths: options.recentlyEditedPaths,\n sessionKnownPaths: options.sessionKnownPaths,\n };\n const scored = scoreFiles(rankInputs);\n const positive = scored.filter((s) => s.score > 0);\n\n if (positive.length === 0) {\n return {\n files: [],\n confidence: \"low\",\n reason: `no matches for ${JSON.stringify(qTokens)}`,\n symbolMatched: false,\n };\n }\n\n const topScored = positive.slice(0, topK);\n const top = topScored.map((s) => s.file);\n const symbolMatched = topScored.some((s) => s.exactSym > 0);\n const topScore = positive[0]?.score ?? 0;\n const secondScore = positive[1]?.score ?? 0;\n\n // confidence: high = clear top match (2x next or only one hit)\n // medium = several hits but no dominant one\n // low = a few weak hits\n let confidence: \"high\" | \"medium\" | \"low\";\n if (positive.length === 1) confidence = \"high\";\n else if (topScore >= 6 && topScore >= secondScore * 2) confidence = \"high\";\n else if (topScore >= 3) confidence = \"medium\";\n else confidence = \"low\";\n\n const reasons = positive\n .slice(0, Math.min(3, top.length))\n .map((s) => `${s.file.path} (${s.reasons.join(\",\")})`)\n .join(\"; \");\n\n return {\n files: top,\n confidence,\n reason: `top: ${reasons}`,\n symbolMatched,\n };\n}\n","// Branch-aware routing for the context store.\n// On the default branch, reads/writes go to .synthra/context-store.json.\n// On a feature branch, they go to .synthra/branches/<sanitized-branch>/context-store.json.\n\nimport { execFile } from \"node:child_process\";\nimport { readFile } from \"node:fs/promises\";\nimport { join } from \"node:path\";\nimport { promisify } from \"node:util\";\n\nconst execFileAsync = promisify(execFile);\n\nexport async function currentBranch(projectRoot: string): Promise<string> {\n // Try .git/HEAD first — avoids the subprocess cost and works for detached\n // worktrees too (.git is a file there).\n try {\n const headPath = join(projectRoot, \".git\", \"HEAD\");\n const head = await readFile(headPath, \"utf8\");\n const trimmed = head.trim();\n const match = trimmed.match(/^ref:\\s+refs\\/heads\\/(.+)$/);\n if (match?.[1]) return match[1];\n // Detached HEAD — fall through\n } catch {\n // .git/HEAD unreadable (worktree file, submodule, or not a git repo)\n }\n\n try {\n const { stdout } = await execFileAsync(\"git\", [\"branch\", \"--show-current\"], {\n cwd: projectRoot,\n });\n const name = stdout.trim();\n if (name) return name;\n } catch {\n // git not on PATH or not a repo\n }\n\n return \"main\";\n}\n\nexport async function defaultBranch(projectRoot: string): Promise<string> {\n try {\n const { stdout } = await execFileAsync(\n \"git\",\n [\"symbolic-ref\", \"refs/remotes/origin/HEAD\", \"--short\"],\n { cwd: projectRoot },\n );\n const trimmed = stdout.trim();\n const match = trimmed.match(/^origin\\/(.+)$/);\n if (match?.[1]) return match[1];\n } catch {\n // No origin/HEAD set — fall back to heuristic\n }\n return \"main\";\n}\n\nexport function sanitizeBranchName(name: string): string {\n return name.replaceAll(\"/\", \"-\").replaceAll(\"\\\\\", \"-\");\n}\n\nexport interface BranchScopedPaths {\n contextStore: string;\n contextMd: string;\n branchDir: string | null;\n}\n\nexport function resolveBranchPaths(\n contextDir: string,\n branch: string,\n isDefault: boolean,\n): BranchScopedPaths {\n if (isDefault) {\n return {\n contextStore: join(contextDir, \"context-store.json\"),\n contextMd: join(contextDir, \"CONTEXT.md\"),\n branchDir: null,\n };\n }\n const branchDir = join(contextDir, \"branches\", sanitizeBranchName(branch));\n return {\n contextStore: join(branchDir, \"context-store.json\"),\n contextMd: join(branchDir, \"CONTEXT.md\"),\n branchDir,\n };\n}\n","// Free-form CONTEXT.md narrative. Updated by Stop hook at session end with:\n// - Current Task (1 sentence)\n// - Key Decisions (max 3 bullets)\n// - Next Steps (max 3 bullets)\n// Capped at ~20 visible content lines.\n\nimport { mkdir, readFile, writeFile } from \"node:fs/promises\";\nimport { dirname } from \"node:path\";\n\nimport type { ContextEntry } from \"./context-store.js\";\n\nexport interface ContextMd {\n branch: string;\n currentTask: string;\n keyDecisions: string[];\n nextSteps: string[];\n date: string;\n}\n\nconst MAX_BULLETS = 3;\n\nexport function deriveContextMd(entries: ContextEntry[], branch: string): ContextMd {\n // Latest pending task drives \"current task\".\n const tasks = entries.filter((e) => e.type === \"task\").reverse();\n const currentTask = tasks[0]?.content ?? \"\";\n\n const keyDecisions = entries\n .filter((e) => e.type === \"decision\")\n .slice(-MAX_BULLETS)\n .map((e) => e.content);\n\n const nextSteps = entries\n .filter((e) => e.type === \"next\")\n .slice(-MAX_BULLETS)\n .map((e) => e.content);\n\n return {\n branch,\n currentTask,\n keyDecisions,\n nextSteps,\n date: new Date().toISOString(),\n };\n}\n\nexport function formatContextMd(ctx: ContextMd): string {\n const lines: string[] = [];\n lines.push(`# Context — ${ctx.branch}`);\n lines.push(\"\");\n lines.push(`_Updated: ${ctx.date}_`);\n lines.push(\"\");\n\n if (ctx.currentTask) {\n lines.push(`## Current task`);\n lines.push(ctx.currentTask);\n lines.push(\"\");\n }\n\n if (ctx.keyDecisions.length) {\n lines.push(`## Key decisions`);\n for (const d of ctx.keyDecisions) lines.push(`- ${d}`);\n lines.push(\"\");\n }\n\n if (ctx.nextSteps.length) {\n lines.push(`## Next steps`);\n for (const n of ctx.nextSteps) lines.push(`- ${n}`);\n lines.push(\"\");\n }\n\n if (!ctx.currentTask && !ctx.keyDecisions.length && !ctx.nextSteps.length) {\n lines.push(\"_(no context entries yet — use `context_remember` to add one)_\");\n lines.push(\"\");\n }\n\n return lines.join(\"\\n\");\n}\n\nexport async function writeContextMd(path: string, ctx: ContextMd): Promise<void> {\n await mkdir(dirname(path), { recursive: true });\n await writeFile(path, formatContextMd(ctx), \"utf8\");\n}\n\nexport async function readContextMd(path: string): Promise<string | null> {\n try {\n return await readFile(path, \"utf8\");\n } catch {\n return null;\n }\n}\n","// Structured decisions/tasks/facts that persist across sessions.\n// Stored in .synthra/ (GIT-TRACKED) so teammates inherit them.\n// Branch-partitioned via branches.ts.\n\nimport { mkdir, readFile, writeFile } from \"node:fs/promises\";\nimport { dirname } from \"node:path\";\n\nexport type EntryKind = \"decision\" | \"task\" | \"next\" | \"fact\" | \"blocker\";\n\nexport interface ContextEntry {\n type: EntryKind;\n content: string;\n tags: string[];\n files: string[];\n date: string;\n}\n\ninterface Store {\n schema_version: number;\n entries: ContextEntry[];\n}\n\nconst SCHEMA_VERSION = 1;\n\nexport async function readEntries(path: string): Promise<ContextEntry[]> {\n try {\n const raw = await readFile(path, \"utf8\");\n const parsed = JSON.parse(raw) as Partial<Store>;\n return Array.isArray(parsed.entries) ? parsed.entries : [];\n } catch {\n return [];\n }\n}\n\nexport async function writeEntries(path: string, entries: ContextEntry[]): Promise<void> {\n await mkdir(dirname(path), { recursive: true });\n const store: Store = { schema_version: SCHEMA_VERSION, entries };\n await writeFile(path, JSON.stringify(store, null, 2) + \"\\n\", \"utf8\");\n}\n\nexport async function appendEntry(path: string, entry: ContextEntry): Promise<void> {\n const entries = await readEntries(path);\n entries.push(entry);\n await writeEntries(path, entries);\n}\n","// High-level orchestration: branch detection + store routing + CONTEXT.md\n// refresh in one call. Used by the MCP tools and the /context-update route.\n\nimport type { SynthraPaths } from \"../shared/paths.js\";\nimport {\n currentBranch,\n defaultBranch,\n resolveBranchPaths,\n type BranchScopedPaths,\n} from \"./branches.js\";\nimport { deriveContextMd, writeContextMd } from \"./context-md.js\";\nimport { appendEntry, readEntries, type ContextEntry, type EntryKind } from \"./context-store.js\";\n\nexport interface ActiveBranch {\n branch: string;\n isDefault: boolean;\n paths: BranchScopedPaths;\n}\n\nexport async function resolveActiveBranch(\n paths: SynthraPaths,\n override?: string,\n): Promise<ActiveBranch> {\n const branch = override ?? (await currentBranch(paths.projectRoot));\n const def = await defaultBranch(paths.projectRoot);\n const isDefault = branch === def;\n return {\n branch,\n isDefault,\n paths: resolveBranchPaths(paths.contextDir, branch, isDefault),\n };\n}\n\nexport interface RememberInput {\n text: string;\n kind: EntryKind;\n tags?: string[];\n files?: string[];\n}\n\nexport interface RememberResult {\n entry: ContextEntry;\n branch: string;\n storePath: string;\n contextMdPath: string;\n}\n\nexport async function rememberEntry(\n paths: SynthraPaths,\n input: RememberInput,\n): Promise<RememberResult> {\n const active = await resolveActiveBranch(paths);\n const entry: ContextEntry = {\n type: input.kind,\n content: input.text,\n tags: input.tags ?? [],\n files: input.files ?? [],\n date: new Date().toISOString(),\n };\n await appendEntry(active.paths.contextStore, entry);\n\n // Refresh CONTEXT.md so the narrative stays in sync with the structured store.\n const entries = await readEntries(active.paths.contextStore);\n const md = deriveContextMd(entries, active.branch);\n await writeContextMd(active.paths.contextMd, md);\n\n return {\n entry,\n branch: active.branch,\n storePath: active.paths.contextStore,\n contextMdPath: active.paths.contextMd,\n };\n}\n\nexport interface RecallInput {\n kind?: EntryKind;\n branch?: string;\n limit?: number;\n}\n\nexport interface RecallResult {\n branch: string;\n entries: ContextEntry[];\n storePath: string;\n}\n\nexport async function recallEntries(\n paths: SynthraPaths,\n input: RecallInput = {},\n): Promise<RecallResult> {\n const active = await resolveActiveBranch(paths, input.branch);\n let entries = await readEntries(active.paths.contextStore);\n if (input.kind) entries = entries.filter((e) => e.type === input.kind);\n if (input.limit && input.limit > 0) entries = entries.slice(-input.limit);\n return {\n branch: active.branch,\n entries,\n storePath: active.paths.contextStore,\n };\n}\n\nexport async function refreshContextMd(paths: SynthraPaths, branchOverride?: string) {\n const active = await resolveActiveBranch(paths, branchOverride);\n const entries = await readEntries(active.paths.contextStore);\n const md = deriveContextMd(entries, active.branch);\n await writeContextMd(active.paths.contextMd, md);\n return {\n branch: active.branch,\n path: active.paths.contextMd,\n entriesSeen: entries.length,\n };\n}\n","// Renders the final context pack as a single Markdown blob for Claude.\n\nexport interface FormatFileSection {\n path: string;\n reason?: string;\n signatures: string[];\n inlineBodies: string;\n associatedTests?: string[];\n}\n\nexport interface FormatInputs {\n query: string;\n files: FormatFileSection[];\n recentActivity?: string;\n truncated?: boolean;\n}\n\nexport function formatPack(inputs: FormatInputs): string {\n const parts: string[] = [];\n parts.push(`# Synthra context — query: ${JSON.stringify(inputs.query)}\\n`);\n\n if (inputs.files.length === 0) {\n parts.push(\"> No matching files found in the graph.\\n\");\n }\n\n for (const f of inputs.files) {\n const heading = f.reason ? `## ${f.path} _(${f.reason})_` : `## ${f.path}`;\n parts.push(heading);\n\n if (f.signatures.length === 0) {\n parts.push(\"_(no symbols extracted)_\");\n } else {\n parts.push(\"**Signatures:**\");\n for (const s of f.signatures) parts.push(`- ${s}`);\n }\n\n if (f.inlineBodies.trim().length > 0) {\n parts.push(\"\");\n parts.push(\"**Bodies:**\");\n parts.push(\"```\");\n parts.push(f.inlineBodies.trimEnd());\n parts.push(\"```\");\n }\n\n if (f.associatedTests?.length) {\n parts.push(\"\");\n parts.push(`**Tests:** ${f.associatedTests.join(\", \")}`);\n }\n\n parts.push(\"\");\n }\n\n if (inputs.recentActivity?.trim()) {\n parts.push(\"---\");\n parts.push(\"## Recent human activity\");\n parts.push(inputs.recentActivity.trim());\n parts.push(\"\");\n }\n\n if (inputs.truncated) {\n parts.push(\"> _(pack truncated to fit budget)_\");\n }\n\n return parts.join(\"\\n\");\n}\n","// Picks the top function bodies from a file (by relevance to the query) and\n// inlines them, respecting a char-based budget. Truncates oversized bodies.\n\nimport { tokenizeQuery } from \"../graph/rank.js\";\nimport type { FileNode, SymbolKind, SymbolNode } from \"../graph/types.js\";\n\nexport interface InlineSelection {\n text: string;\n charsUsed: number;\n functionsInlined: string[];\n}\n\nconst INLINABLE_KINDS = new Set<SymbolKind>([\"function\", \"method\", \"class\"]);\nconst MAX_BODY_CHARS = 1500;\n\nfunction sliceLines(content: string, startLine: number, endLine: number): string {\n const lines = content.split(/\\r?\\n/);\n return lines.slice(Math.max(0, startLine - 1), endLine).join(\"\\n\");\n}\n\nfunction scoreSymbol(name: string, qTokens: Set<string>): number {\n const lower = name.toLowerCase();\n if (qTokens.has(lower)) return 3;\n for (const t of qTokens) {\n if (lower.includes(t) || t.includes(lower)) return 1;\n }\n return 0;\n}\n\nfunction truncate(body: string): string {\n if (body.length <= MAX_BODY_CHARS) return body;\n return body.slice(0, MAX_BODY_CHARS).trimEnd() + \"\\n // … truncated\";\n}\n\nexport function selectInlineBodies(\n file: FileNode,\n symbols: SymbolNode[],\n query: string,\n budgetChars: number,\n): InlineSelection {\n if (budgetChars <= 0) {\n return { text: \"\", charsUsed: 0, functionsInlined: [] };\n }\n\n const qTokens = new Set(tokenizeQuery(query));\n const mine = symbols.filter((s) => s.file === file.path && INLINABLE_KINDS.has(s.symbol_kind));\n\n const scored = mine\n .map((s) => ({ sym: s, score: scoreSymbol(s.name, qTokens) }))\n .sort((a, b) => {\n if (b.score !== a.score) return b.score - a.score;\n // Tie-break: smaller bodies first (so we fit more)\n const aSpan = (a.sym.end_line - a.sym.start_line) || 1;\n const bSpan = (b.sym.end_line - b.sym.start_line) || 1;\n return aSpan - bSpan;\n });\n\n const parts: string[] = [];\n const inlined: string[] = [];\n let used = 0;\n\n for (const { sym, score } of scored) {\n // Skip irrelevant symbols entirely when we have positive hits available;\n // fall back to top-by-size if no query match landed.\n if (score === 0 && inlined.length > 0) break;\n\n const body = truncate(sliceLines(file.content, sym.start_line, sym.end_line));\n const header = `${file.path}::${sym.name} (L${sym.start_line}-${sym.end_line})`;\n const block = `${header}\\n${body}\\n`;\n if (used + block.length > budgetChars) {\n if (inlined.length > 0) break;\n // No room and nothing yet — take a head-only slice.\n const remaining = Math.max(0, budgetChars - used - header.length - 16);\n if (remaining <= 0) break;\n const partial = body.slice(0, remaining).trimEnd() + \"\\n // … truncated\";\n const finalBlock = `${header}\\n${partial}\\n`;\n parts.push(finalBlock);\n inlined.push(sym.name);\n used += finalBlock.length;\n break;\n }\n parts.push(block);\n inlined.push(sym.name);\n used += block.length;\n }\n\n return { text: parts.join(\"\\n\"), charsUsed: used, functionsInlined: inlined };\n}\n","// Extracts function/class signatures from a FileNode (no bodies).\n// Signatures are the first line of each symbol in the file, sorted by line.\n\nimport type { FileNode, SymbolNode } from \"../graph/types.js\";\n\nexport function extractSignatures(file: FileNode, symbols: SymbolNode[]): string[] {\n const mine = symbols\n .filter((s) => s.file === file.path)\n .slice()\n .sort((a, b) => a.start_line - b.start_line);\n\n return mine.map((s) => `L${s.start_line}: ${s.signature.trim()}`);\n}\n","// Test ↔ source co-retrieval. Given a source file, returns the test files\n// linked to it via `tests` edges in the graph (foo.test.ts → foo.ts).\n\nimport type { FileNode, GraphSchema } from \"../graph/types.js\";\n\nexport function findTestsForFile(graph: GraphSchema, file: FileNode): FileNode[] {\n // tests edges run from test file → source file\n const fileNodesById = new Map<string, FileNode>();\n for (const n of graph.nodes) {\n if (n.kind === \"file\") fileNodesById.set(n.id, n);\n }\n\n const out: FileNode[] = [];\n for (const e of graph.edges) {\n if (e.kind !== \"tests\" || e.to !== file.id) continue;\n const testFile = fileNodesById.get(e.from);\n if (testFile && !out.includes(testFile)) out.push(testFile);\n }\n return out;\n}\n","// Compresses a list of retrieved files into a structured context pack:\n// signatures + top function bodies + tests co-retrieved + dependency edges.\n// Budget is enforced in characters (~ tokens × 4) — see SYN_HARD_MAX_READ_CHARS.\n\nimport type { FileNode, GraphSchema, SymbolNode } from \"../graph/types.js\";\nimport { formatPack, type FormatFileSection } from \"./format.js\";\nimport { selectInlineBodies } from \"./inline.js\";\nimport { extractSignatures } from \"./signatures.js\";\nimport { findTestsForFile } from \"./tests.js\";\n\nexport interface PackOptions {\n query: string;\n graph: GraphSchema;\n /** Soft target for total tokens (≈ chars/4). Default: 4000. */\n budgetTokens?: number;\n /** Fraction of remaining budget to spend on a single file's inline bodies. Default: 0.5. */\n inlineBodyRatio?: number;\n /** Co-retrieve linked test files when packing source files. Default: true. */\n includeTests?: boolean;\n /** Optional: file path → reason string from the ranker, surfaced in the pack heading. */\n reasons?: Map<string, string>;\n}\n\nexport interface ContextPack {\n text: string;\n tokenEstimate: number;\n filesUsed: string[];\n testsCoRetrieved: string[];\n truncated: boolean;\n}\n\nconst STATIC_OVERHEAD_PER_FILE = 200; // headers + bullet markdown + spacing\nconst MAX_INLINE_CHARS_PER_FILE = 2500;\n\nfunction indexSymbolsByFile(graph: GraphSchema): SymbolNode[] {\n return graph.nodes.filter((n): n is SymbolNode => n.kind === \"symbol\");\n}\n\nexport async function pack(files: FileNode[], opts: PackOptions): Promise<ContextPack> {\n const budgetTokens = opts.budgetTokens ?? 4000;\n const budgetChars = budgetTokens * 4;\n const inlineRatio = opts.inlineBodyRatio ?? 0.5;\n const includeTests = opts.includeTests ?? true;\n const reasons = opts.reasons ?? new Map<string, string>();\n\n const symbols = indexSymbolsByFile(opts.graph);\n\n const sections: FormatFileSection[] = [];\n const testsCoRetrieved: string[] = [];\n let used = 0;\n let truncated = false;\n\n for (const file of files) {\n const sig = extractSignatures(file, symbols);\n const testFiles = includeTests ? findTestsForFile(opts.graph, file) : [];\n const testPaths = testFiles.map((t) => t.path);\n\n const staticCost =\n file.path.length +\n sig.join(\"\\n\").length +\n testPaths.join(\",\").length +\n STATIC_OVERHEAD_PER_FILE;\n\n if (used + staticCost > budgetChars) {\n truncated = true;\n break;\n }\n\n const remaining = budgetChars - used - staticCost;\n const inlineBudget = Math.min(Math.floor(remaining * inlineRatio), MAX_INLINE_CHARS_PER_FILE);\n\n const inline = selectInlineBodies(file, symbols, opts.query, inlineBudget);\n\n sections.push({\n path: file.path,\n reason: reasons.get(file.path),\n signatures: sig,\n inlineBodies: inline.text,\n associatedTests: testPaths,\n });\n\n used += staticCost + inline.charsUsed;\n for (const t of testPaths) if (!testsCoRetrieved.includes(t)) testsCoRetrieved.push(t);\n\n if (used >= budgetChars) {\n truncated = true;\n break;\n }\n }\n\n if (sections.length < files.length) truncated = true;\n\n const text = formatPack({\n query: opts.query,\n files: sections,\n truncated,\n });\n const tokenEstimate = Math.ceil(text.length / 4);\n\n return {\n text,\n tokenEstimate,\n filesUsed: sections.map((s) => s.path),\n testsCoRetrieved,\n truncated,\n };\n}\n","// MCP-over-HTTP (streamable) protocol handler. Exposes Synthra's graph tools\n// to Claude Code via JSON-RPC 2.0 messages POSTed to /mcp.\n//\n// Tools:\n// graph_continue(query) — retrieve + pack a context bundle\n// graph_read(target) — return source for \"file\" or \"file::symbol\"\n// graph_register_edit(files) — Claude tells Synthra it edited files\n//\n// Spec: https://modelcontextprotocol.io/specification\n\nimport { retrieve } from \"../graph/retrieve.js\";\nimport type { FileNode, SymbolNode } from \"../graph/types.js\";\nimport { recallEntries, rememberEntry } from \"../memory/index.js\";\nimport type { EntryKind } from \"../memory/context-store.js\";\nimport { pack } from \"../packer/index.js\";\nimport type { ServerContext } from \"./context.js\";\n\nconst PROTOCOL_VERSION = \"2024-11-05\";\nconst SERVER_INFO = { name: \"synthra\", version: \"0.0.1\" } as const;\n\ntype JsonRpcId = string | number | null;\n\nexport interface JsonRpcRequest {\n jsonrpc: \"2.0\";\n id?: JsonRpcId;\n method: string;\n params?: Record<string, unknown>;\n}\n\nexport interface JsonRpcResponse {\n jsonrpc: \"2.0\";\n id: JsonRpcId;\n result?: unknown;\n error?: { code: number; message: string; data?: unknown };\n}\n\nconst ERR = {\n parse: -32700,\n invalidRequest: -32600,\n methodNotFound: -32601,\n invalidParams: -32602,\n internal: -32603,\n} as const;\n\nfunction ok(id: JsonRpcId, result: unknown): JsonRpcResponse {\n return { jsonrpc: \"2.0\", id, result };\n}\n\nfunction err(id: JsonRpcId, code: number, message: string, data?: unknown): JsonRpcResponse {\n return { jsonrpc: \"2.0\", id, error: { code, message, data } };\n}\n\nfunction textContent(text: string) {\n return { content: [{ type: \"text\", text }], isError: false };\n}\n\nfunction errorContent(message: string) {\n return { content: [{ type: \"text\", text: message }], isError: true };\n}\n\nconst TOOLS = [\n {\n name: \"graph_continue\",\n description:\n \"Returns the project context most relevant to a query — function signatures, top function bodies, and linked test files. Use this BEFORE Grep/Glob. If `confidence` is 'high', do not call Grep/Glob for the same query.\",\n inputSchema: {\n type: \"object\",\n properties: {\n query: { type: \"string\", description: \"Natural-language description of what you're looking for.\" },\n },\n required: [\"query\"],\n },\n },\n {\n name: \"graph_read\",\n description:\n \"Return the source code for a specific file or symbol. Target is either a project-relative file path (e.g. 'src/auth.ts') or 'file::symbol' (e.g. 'src/auth.ts::AuthService').\",\n inputSchema: {\n type: \"object\",\n properties: {\n target: { type: \"string\", description: \"File path or file::symbol notation.\" },\n },\n required: [\"target\"],\n },\n },\n {\n name: \"graph_register_edit\",\n description:\n \"Tell Synthra that you (the AI) have edited these files. Lets Synthra rank them higher in subsequent retrieval and avoid surfacing stale context.\",\n inputSchema: {\n type: \"object\",\n properties: {\n files: {\n type: \"array\",\n items: { type: \"string\" },\n description: \"Project-relative file paths that were edited.\",\n },\n },\n required: [\"files\"],\n },\n },\n {\n name: \"context_remember\",\n description:\n \"Persist a decision/task/next-step/fact/blocker into the project's branch-aware context store. Use when the user makes a decision worth keeping, identifies a TODO, or surfaces a key fact. Entries land in `.synthra/context-store.json` on the default branch, or `.synthra/branches/<sanitized>/context-store.json` on a feature branch — git-tracked, so teammates inherit them and they merge naturally.\",\n inputSchema: {\n type: \"object\",\n properties: {\n text: { type: \"string\", description: \"The thing to remember (1–3 sentences).\" },\n kind: {\n type: \"string\",\n enum: [\"decision\", \"task\", \"next\", \"fact\", \"blocker\"],\n description: \"What kind of entry. Required.\",\n },\n tags: {\n type: \"array\",\n items: { type: \"string\" },\n description: \"Optional tags for grouping (e.g. 'auth', 'perf').\",\n },\n files: {\n type: \"array\",\n items: { type: \"string\" },\n description: \"Optional project-relative file paths this entry relates to.\",\n },\n },\n required: [\"text\", \"kind\"],\n },\n },\n {\n name: \"context_recall\",\n description:\n \"Read previously-stored decisions/tasks/facts from the project's branch-aware context store. Defaults to the current branch.\",\n inputSchema: {\n type: \"object\",\n properties: {\n kind: {\n type: \"string\",\n enum: [\"decision\", \"task\", \"next\", \"fact\", \"blocker\"],\n description: \"Filter to a single kind.\",\n },\n branch: { type: \"string\", description: \"Override which branch to read from.\" },\n limit: { type: \"number\", description: \"Return only the most recent N entries.\" },\n },\n },\n },\n {\n name: \"recent_activity\",\n description:\n \"What has the human been doing in the editor recently — file saves, branch switches, and uncommitted-diff changes. Use this to check whether the static context pack may be stale (e.g. before answering a question about a file that was just edited).\",\n inputSchema: {\n type: \"object\",\n properties: {\n since_ms: {\n type: \"number\",\n description:\n \"Epoch milliseconds. Only return events newer than this. Defaults to the last 60 minutes.\",\n },\n limit: { type: \"number\", description: \"Cap on returned events.\" },\n },\n },\n },\n {\n name: \"count_tokens\",\n description:\n \"Estimate token count for a piece of text using a char/4 approximation. Accurate within ~10% for English + code. Useful for budgeting prompt content before sending.\",\n inputSchema: {\n type: \"object\",\n properties: {\n text: { type: \"string\", description: \"The text to estimate tokens for.\" },\n },\n required: [\"text\"],\n },\n },\n {\n name: \"blast_radius\",\n description:\n \"Given a file (or 'file::symbol' target), return all files that depend on it transitively via imports + tests edges. Use BEFORE editing a widely-used file to see what could break. Symbol-level granularity is approximated at the file level (we don't track call edges in v0.1).\",\n inputSchema: {\n type: \"object\",\n properties: {\n target: { type: \"string\", description: \"File path or 'file::symbol' notation.\" },\n depth: { type: \"number\", description: \"Max hops to traverse. Default 3.\" },\n },\n required: [\"target\"],\n },\n },\n {\n name: \"dead_code\",\n description:\n \"Return files in the project that no other file imports and no test file references — strong candidates for unused/orphaned code. File-level granularity (v0.1 limitation — symbol-level needs call-graph edges). Common entry-point patterns (main, index, app, CLI, bin/) are excluded heuristically.\",\n inputSchema: {\n type: \"object\",\n properties: {\n limit: { type: \"number\", description: \"Cap on returned files. Default 50.\" },\n },\n },\n },\n] as const;\n\nasync function callTool(\n name: string,\n args: Record<string, unknown> | undefined,\n ctx: ServerContext,\n) {\n switch (name) {\n case \"graph_continue\":\n return graphContinue(args, ctx);\n case \"graph_read\":\n return graphRead(args, ctx);\n case \"graph_register_edit\":\n return graphRegisterEdit(args, ctx);\n case \"context_remember\":\n return contextRemember(args, ctx);\n case \"context_recall\":\n return contextRecall(args, ctx);\n case \"recent_activity\":\n return recentActivity(args, ctx);\n case \"count_tokens\":\n return countTokens(args);\n case \"blast_radius\":\n return blastRadius(args, ctx);\n case \"dead_code\":\n return deadCode(args, ctx);\n default:\n return errorContent(`Unknown tool: ${name}`);\n }\n}\n\nfunction countTokens(args: Record<string, unknown> | undefined) {\n const text = typeof args?.text === \"string\" ? args.text : \"\";\n if (!text) return errorContent(\"count_tokens: 'text' (string) is required\");\n const tokens = Math.ceil(text.length / 4);\n return textContent(JSON.stringify({ tokens, method: \"chars/4 estimate\", chars: text.length }));\n}\n\nfunction blastRadius(args: Record<string, unknown> | undefined, ctx: ServerContext) {\n const targetRaw = typeof args?.target === \"string\" ? args.target.trim() : \"\";\n const maxDepth = typeof args?.depth === \"number\" && args.depth > 0 ? Math.floor(args.depth) : 3;\n if (!targetRaw) return errorContent(\"blast_radius: 'target' (string) is required\");\n\n const filePath = targetRaw.split(\"::\", 1)[0]?.trim() ?? targetRaw;\n const root = ctx.graph.nodes.find(\n (n): n is FileNode => n.kind === \"file\" && n.path === filePath,\n );\n if (!root) return errorContent(`blast_radius: file not in graph: ${filePath}`);\n\n // Index reverse edges (to → [{from, kind}]) once per call.\n const incoming = new Map<string, Array<{ from: string; kind: string }>>();\n for (const e of ctx.graph.edges) {\n if (e.kind !== \"imports\" && e.kind !== \"tests\") continue;\n const list = incoming.get(e.to) ?? [];\n list.push({ from: e.from, kind: e.kind });\n incoming.set(e.to, list);\n }\n\n interface Hit {\n path: string;\n depth: number;\n via: string;\n }\n\n const visited = new Set<string>([root.id]);\n const hits: Hit[] = [];\n const pathById = new Map<string, string>();\n for (const n of ctx.graph.nodes) if (n.kind === \"file\") pathById.set(n.id, n.path);\n\n let frontier = [root.id];\n for (let d = 1; d <= maxDepth; d++) {\n const next: string[] = [];\n for (const cur of frontier) {\n const callers = incoming.get(cur) ?? [];\n for (const c of callers) {\n if (visited.has(c.from)) continue;\n visited.add(c.from);\n next.push(c.from);\n const path = pathById.get(c.from) ?? c.from;\n hits.push({ path, depth: d, via: c.kind });\n }\n }\n frontier = next;\n if (next.length === 0) break;\n }\n\n if (hits.length === 0) {\n return textContent(`# Blast radius for ${filePath}\\n\\n_(no dependents — file is isolated)_`);\n }\n\n hits.sort((a, b) => a.depth - b.depth || a.path.localeCompare(b.path));\n const lines = [`# Blast radius for ${filePath} (depth ≤ ${maxDepth})`, \"\"];\n lines.push(`${hits.length} dependent file(s):`);\n for (const h of hits) {\n lines.push(`- **depth ${h.depth}** \\`${h.path}\\` _(via ${h.via})_`);\n }\n return textContent(lines.join(\"\\n\"));\n}\n\nconst LIKELY_ENTRY_PATTERNS = [\n /(?:^|\\/)main\\.[a-z0-9_]+$/i,\n /(?:^|\\/)index\\.[a-z0-9_]+$/i,\n /(?:^|\\/)app\\.[a-z0-9_]+$/i,\n /(?:^|\\/)entry\\.[a-z0-9_]+$/i,\n /(?:^|\\/)cli[\\/.]/i,\n /(?:^|\\/)bin[\\/.]/i,\n /(?:^|\\/)server\\.[a-z0-9_]+$/i,\n /\\.test\\.[a-z0-9_]+$/i,\n /\\.spec\\.[a-z0-9_]+$/i,\n /(?:^|\\/)tests?\\//i,\n /(?:^|\\/)__tests__\\//i,\n /(?:^|\\/)__init__\\.py$/i,\n];\n\nfunction isLikelyEntry(path: string): boolean {\n return LIKELY_ENTRY_PATTERNS.some((re) => re.test(path));\n}\n\nfunction deadCode(args: Record<string, unknown> | undefined, ctx: ServerContext) {\n const limit = typeof args?.limit === \"number\" && args.limit > 0 ? Math.floor(args.limit) : 50;\n\n const hasIncoming = new Set<string>();\n for (const e of ctx.graph.edges) {\n if (e.kind === \"imports\" || e.kind === \"tests\") hasIncoming.add(e.to);\n }\n\n const candidates = ctx.graph.nodes\n .filter((n): n is FileNode => n.kind === \"file\")\n .filter((f) => !hasIncoming.has(f.id))\n .filter((f) => !isLikelyEntry(f.path));\n\n if (candidates.length === 0) {\n return textContent(\n `# Dead code\\n\\n_(no file is unreferenced — every file is either imported by another, has a linked test, or matches an entry-point pattern)_`,\n );\n }\n\n candidates.sort((a, b) => a.path.localeCompare(b.path));\n const shown = candidates.slice(0, limit);\n const lines = [`# Dead code candidates (file-level, v0.1)`, \"\"];\n lines.push(\n `${shown.length} of ${candidates.length} unreferenced file(s) — no other file imports them and no test links them:`,\n );\n lines.push(\"\");\n for (const f of shown) {\n lines.push(`- \\`${f.path}\\``);\n }\n lines.push(\"\");\n lines.push(\n `_v0.1 caveat:_ this is file-level only. Symbol-level dead code (unused exports) needs call-graph edges, which land in v0.2.`,\n );\n return textContent(lines.join(\"\\n\"));\n}\n\nasync function graphContinue(args: Record<string, unknown> | undefined, ctx: ServerContext) {\n const query = typeof args?.query === \"string\" ? args.query : \"\";\n if (!query) return errorContent(\"graph_continue: 'query' (string) is required\");\n\n const retrieval = await retrieve(ctx.graph, query);\n const packed = await pack(retrieval.files, { query, graph: ctx.graph });\n\n const header =\n `Confidence: ${retrieval.confidence}\\n` +\n `Files: ${retrieval.files.map((f) => f.path).join(\", \") || \"(none)\"}\\n` +\n `Reason: ${retrieval.reason}\\n`;\n\n // The pack body already starts with a header — keep them concatenated.\n return textContent(`${header}\\n${packed.text}`);\n}\n\nfunction graphRead(args: Record<string, unknown> | undefined, ctx: ServerContext) {\n const target = typeof args?.target === \"string\" ? args.target : \"\";\n if (!target) return errorContent(\"graph_read: 'target' (string) is required\");\n\n const [rawFile, symbolName] = target.includes(\"::\") ? target.split(\"::\", 2) : [target, undefined];\n const filePath = (rawFile ?? \"\").trim();\n\n const fileNode = ctx.graph.nodes.find(\n (n): n is FileNode => n.kind === \"file\" && n.path === filePath,\n );\n if (!fileNode) return errorContent(`graph_read: file not found in graph: ${filePath}`);\n\n if (!symbolName) {\n return textContent(`# ${fileNode.path}\\n\\n${fileNode.content}`);\n }\n\n const cleanSym = symbolName.trim();\n const symbol = ctx.graph.nodes.find(\n (n): n is SymbolNode => n.kind === \"symbol\" && n.file === filePath && n.name === cleanSym,\n );\n if (!symbol) {\n return errorContent(`graph_read: symbol '${cleanSym}' not found in ${filePath}`);\n }\n\n const lines = fileNode.content.split(/\\r?\\n/);\n const body = lines.slice(symbol.start_line - 1, symbol.end_line).join(\"\\n\");\n return textContent(\n `# ${fileNode.path}::${symbol.name} (L${symbol.start_line}-${symbol.end_line})\\n\\n${body}`,\n );\n}\n\nconst editedFiles = new Set<string>();\n\nfunction graphRegisterEdit(args: Record<string, unknown> | undefined, _ctx: ServerContext) {\n const files = Array.isArray(args?.files) ? (args.files as unknown[]).filter((f) => typeof f === \"string\") : [];\n for (const f of files) editedFiles.add(f as string);\n return textContent(`Registered ${files.length} edited file(s). Total tracked this session: ${editedFiles.size}.`);\n}\n\nexport function getRegisteredEdits(): string[] {\n return Array.from(editedFiles);\n}\n\nconst VALID_KINDS = new Set<EntryKind>([\"decision\", \"task\", \"next\", \"fact\", \"blocker\"]);\n\nasync function contextRemember(args: Record<string, unknown> | undefined, ctx: ServerContext) {\n const text = typeof args?.text === \"string\" ? args.text.trim() : \"\";\n const kindRaw = typeof args?.kind === \"string\" ? args.kind : \"\";\n if (!text) return errorContent(\"context_remember: 'text' (string) is required\");\n if (!VALID_KINDS.has(kindRaw as EntryKind)) {\n return errorContent(\n `context_remember: 'kind' must be one of ${Array.from(VALID_KINDS).join(\", \")}`,\n );\n }\n const tags = Array.isArray(args?.tags)\n ? (args.tags as unknown[]).filter((t): t is string => typeof t === \"string\")\n : [];\n const files = Array.isArray(args?.files)\n ? (args.files as unknown[]).filter((f): f is string => typeof f === \"string\")\n : [];\n\n const result = await rememberEntry(ctx.paths, {\n text,\n kind: kindRaw as EntryKind,\n tags,\n files,\n });\n\n return textContent(\n `Remembered ${result.entry.type} on branch '${result.branch}'.\\n` +\n `Stored: ${result.storePath}\\n` +\n `CONTEXT.md refreshed: ${result.contextMdPath}`,\n );\n}\n\nconst DEFAULT_RECENT_WINDOW_MS = 60 * 60 * 1000;\n\nfunction recentActivity(args: Record<string, unknown> | undefined, ctx: ServerContext) {\n const sinceMs =\n typeof args?.since_ms === \"number\" && Number.isFinite(args.since_ms)\n ? args.since_ms\n : Date.now() - DEFAULT_RECENT_WINDOW_MS;\n const limit =\n typeof args?.limit === \"number\" && args.limit > 0 ? Math.floor(args.limit) : undefined;\n\n let events = ctx.activity.getEvents(sinceMs);\n if (limit) events = events.slice(-limit);\n\n if (events.length === 0) {\n return textContent(\n `No human-activity events since ${new Date(sinceMs).toISOString()}.`,\n );\n }\n\n const lines = [`# Recent human activity (${events.length} events)`, \"\"];\n for (const e of events) {\n if (\"path\" in e) {\n lines.push(`- **${e.kind}** ${e.path} _(${e.ts})_`);\n } else {\n const summary = JSON.stringify(e.details);\n lines.push(`- **${e.kind}** ${summary} _(${e.ts})_`);\n }\n }\n return textContent(lines.join(\"\\n\"));\n}\n\nasync function contextRecall(args: Record<string, unknown> | undefined, ctx: ServerContext) {\n const kind = typeof args?.kind === \"string\" && VALID_KINDS.has(args.kind as EntryKind)\n ? (args.kind as EntryKind)\n : undefined;\n const branch = typeof args?.branch === \"string\" ? args.branch : undefined;\n const limit = typeof args?.limit === \"number\" && args.limit > 0 ? Math.floor(args.limit) : undefined;\n\n const result = await recallEntries(ctx.paths, { kind, branch, limit });\n\n if (result.entries.length === 0) {\n const filter = kind ? ` of kind '${kind}'` : \"\";\n return textContent(`No context entries${filter} on branch '${result.branch}'.`);\n }\n\n const lines = [`# Context entries — branch: ${result.branch}`, \"\"];\n for (const e of result.entries) {\n const tags = e.tags.length ? ` [${e.tags.join(\", \")}]` : \"\";\n lines.push(`- **${e.type}**${tags} (${e.date}): ${e.content}`);\n if (e.files.length) lines.push(` files: ${e.files.join(\", \")}`);\n }\n return textContent(lines.join(\"\\n\"));\n}\n\nexport async function handleMcpRequest(\n body: unknown,\n ctx: ServerContext,\n): Promise<JsonRpcResponse> {\n if (!body || typeof body !== \"object\") {\n return err(null, ERR.invalidRequest, \"Request body must be a JSON-RPC 2.0 object.\");\n }\n\n const req = body as JsonRpcRequest;\n if (req.jsonrpc !== \"2.0\" || typeof req.method !== \"string\") {\n return err(req.id ?? null, ERR.invalidRequest, \"Invalid JSON-RPC envelope.\");\n }\n\n const id = req.id ?? null;\n\n try {\n switch (req.method) {\n case \"initialize\":\n return ok(id, {\n protocolVersion:\n typeof req.params?.protocolVersion === \"string\"\n ? req.params.protocolVersion\n : PROTOCOL_VERSION,\n capabilities: { tools: {} },\n serverInfo: SERVER_INFO,\n });\n\n case \"notifications/initialized\":\n // Client confirms initialization. No response required for notifications (id===undefined).\n return ok(id, {});\n\n case \"tools/list\":\n return ok(id, { tools: TOOLS });\n\n case \"tools/call\": {\n const params = req.params ?? {};\n const toolName = typeof params.name === \"string\" ? params.name : \"\";\n if (!toolName) return err(id, ERR.invalidParams, \"'name' is required for tools/call.\");\n const args = (params.arguments && typeof params.arguments === \"object\"\n ? (params.arguments as Record<string, unknown>)\n : {});\n const result = await callTool(toolName, args, ctx);\n return ok(id, result);\n }\n\n case \"ping\":\n return ok(id, {});\n\n default:\n return err(id, ERR.methodNotFound, `Method not found: ${req.method}`);\n }\n } catch (e) {\n return err(id, ERR.internal, (e as Error).message);\n }\n}\n\n// Exposed for code that wants to enumerate the tool catalogue without going\n// through JSON-RPC (e.g. CLI introspection in M3).\nexport function listTools(): Array<{ name: string; description: string; inputSchema: unknown }> {\n return TOOLS.map((t) => ({ name: t.name, description: t.description, inputSchema: t.inputSchema }));\n}\n","// Finds a free port in the 8080–8099 range. Writes the chosen port to\n// .synthra-graph/mcp_port so PowerShell/Bash hook scripts can read it.\n// TODO: M2\n\nimport { createServer } from \"node:net\";\n\nexport const PORT_RANGE_START = 8080;\nexport const PORT_RANGE_END = 8099;\n\nexport async function findFreePort(\n start = PORT_RANGE_START,\n end = PORT_RANGE_END,\n): Promise<number> {\n for (let port = start; port <= end; port++) {\n if (await isFree(port)) return port;\n }\n throw new Error(`Synthra: no free port in ${start}-${end}`);\n}\n\nfunction isFree(port: number): Promise<boolean> {\n return new Promise((resolve) => {\n const s = createServer();\n s.once(\"error\", () => resolve(false));\n s.once(\"listening\", () => s.close(() => resolve(true)));\n s.listen(port, \"127.0.0.1\");\n });\n}\n","// GET /activity?since=<ms> — returns recent human-activity events.\n// Backed by the in-memory ActivityStore (file-watcher + git-watcher feed it).\n// MCP tool `recent_activity` is a thin wrapper.\n\nimport type { ActivityEvent } from \"../../activity/activity-log.js\";\nimport type { ServerContext } from \"../context.js\";\n\nexport interface ActivityResponse {\n events: ActivityEvent[];\n since: string;\n ring_size: number;\n}\n\nexport async function handleActivity(\n sinceMs: number | undefined,\n ctx: ServerContext,\n): Promise<ActivityResponse> {\n const events = ctx.activity.getEvents(sinceMs);\n return {\n events,\n since: new Date(sinceMs ?? Date.now()).toISOString(),\n ring_size: ctx.activity.size(),\n };\n}\n","// POST /context-update — Stop hook calls this at session end.\n// For M4: re-renders CONTEXT.md from the branch-scoped store so the narrative\n// stays in sync with the structured entries that landed during the session.\n// Transcript-mining for new entries (auto \"we decided X\" → store) is v0.2.\n\nimport { refreshContextMd } from \"../../memory/index.js\";\nimport type { ServerContext } from \"../context.js\";\n\nexport interface ContextUpdateRequest {\n transcript_path?: string;\n branch?: string;\n}\n\nexport interface ContextUpdateResponse {\n updated: boolean;\n branch: string;\n path: string;\n entries: number;\n}\n\nexport async function handleContextUpdate(\n req: ContextUpdateRequest,\n ctx: ServerContext,\n): Promise<ContextUpdateResponse> {\n const r = await refreshContextMd(ctx.paths, req?.branch);\n return {\n updated: true,\n branch: r.branch,\n path: r.path,\n entries: r.entriesSeen,\n };\n}\n","// POST /gate — PreToolUse hook calls this with the tool name + arguments.\n// THE MOAT — improvement #1. Strategy:\n// - For Grep/Glob: extract the search pattern, run retrieve().\n// - If recent human activity touches a file matching the query → ALLOW\n// even at high confidence (the user's head is in that file; static\n// context may be stale).\n// - If confidence === \"high\" and no recent overlap → BLOCK with a reason\n// pointing at graph_continue / graph_read.\n// - Otherwise → ALLOW.\n\nimport { appendFile, mkdir } from \"node:fs/promises\";\nimport { dirname } from \"node:path\";\n\nimport { retrieve } from \"../../graph/retrieve.js\";\nimport { tokenizeQuery } from \"../../graph/rank.js\";\nimport type { ServerContext } from \"../context.js\";\n\nexport interface GateRequest {\n tool_name: string;\n tool_input: Record<string, unknown>;\n}\n\nexport interface GateResponse {\n decision: \"allow\" | \"block\";\n reason?: string;\n}\n\nconst BLOCKABLE_TOOLS = new Set([\"Grep\", \"Glob\"]);\nconst RECENT_ACTIVITY_WINDOW_MS = 5 * 60 * 1000;\n\nfunction extractQuery(toolName: string, input: Record<string, unknown>): string | null {\n if (toolName === \"Grep\") {\n const pattern = typeof input.pattern === \"string\" ? input.pattern : \"\";\n const query = typeof input.query === \"string\" ? input.query : \"\";\n return (pattern || query).trim() || null;\n }\n if (toolName === \"Glob\") {\n const pattern = typeof input.pattern === \"string\" ? input.pattern : \"\";\n return pattern.replace(/[*?/\\\\.]+/g, \" \").trim() || null;\n }\n return null;\n}\n\n// Heuristic: does this Grep pattern target markup / CSS / attributes / literals\n// rather than a code symbol? The graph only indexes symbols, so blocking these\n// and redirecting to graph_read just forces a fallback Read. Conservative — only\n// fires on syntax that never appears in a bare identifier search.\nfunction looksLikeNonSymbolQuery(pattern: string): boolean {\n // HTML / JSX tag: \"<div\", \"</\", \"<svg\"\n if (/<\\/?[a-zA-Z]/.test(pattern)) return true;\n // Hyphenated attribute assignment: \"data-tour=\", \"aria-label=\" ('-' is not a\n // valid identifier char, so this is markup, not a symbol).\n if (/[a-zA-Z][\\w-]*-[\\w-]*\\s*=/.test(pattern)) return true;\n // CSS rule / object brace: \".content{\", \"{ color\"\n if (/\\{/.test(pattern)) return true;\n // Escaped-dot class / member selector: \"\\.filter-bar\", \"\\.gs\"\n if (/\\\\\\.[a-zA-Z]/.test(pattern)) return true;\n // CSS property value or units: \": 100%\", \"12px\", \"1.5rem\", \"50%\"\n if (/:\\s*\\d/.test(pattern) || /\\d(?:px|rem|em|vh|vw)\\b/.test(pattern) || /\\d%/.test(pattern)) {\n return true;\n }\n return false;\n}\n\nfunction recentlyTouchedMatchesQuery(\n recentPaths: string[],\n queryTokens: Set<string>,\n): string[] {\n const matches: string[] = [];\n for (const path of recentPaths) {\n const lower = path.toLowerCase();\n for (const t of queryTokens) {\n if (lower.includes(t)) {\n matches.push(path);\n break;\n }\n }\n }\n return matches;\n}\n\nasync function logDecision(\n ctx: ServerContext,\n toolName: string,\n query: string | null,\n decision: \"allow\" | \"block\",\n reason: string | undefined,\n): Promise<void> {\n try {\n await mkdir(dirname(ctx.paths.gateLog), { recursive: true });\n const entry = {\n ts: new Date().toISOString(),\n tool: toolName,\n decision,\n query,\n reason,\n };\n await appendFile(ctx.paths.gateLog, JSON.stringify(entry) + \"\\n\", \"utf8\");\n } catch {\n // Durability is best-effort; an unwritable disk shouldn't fail the gate.\n }\n}\n\nexport async function handleGate(req: GateRequest, ctx: ServerContext): Promise<GateResponse> {\n if (!req?.tool_name || typeof req.tool_name !== \"string\") {\n return { decision: \"allow\", reason: \"no tool_name\" };\n }\n\n if (!BLOCKABLE_TOOLS.has(req.tool_name)) {\n return { decision: \"allow\" };\n }\n\n const input = (req.tool_input && typeof req.tool_input === \"object\" ? req.tool_input : {}) as Record<string, unknown>;\n const query = extractQuery(req.tool_name, input);\n if (!query) {\n const res: GateResponse = { decision: \"allow\", reason: \"no extractable query\" };\n await logDecision(ctx, req.tool_name, null, res.decision, res.reason);\n return res;\n }\n\n // Guard 1 — the query targets markup/CSS/attributes/literals, which the graph\n // does not index. Blocking would only force a fallback, so let Grep through.\n if (req.tool_name === \"Grep\" && looksLikeNonSymbolQuery(query)) {\n const res: GateResponse = {\n decision: \"allow\",\n reason: `\"${query}\" targets markup/CSS/attributes, not code symbols — letting Grep through (the graph indexes symbols).`,\n };\n await logDecision(ctx, req.tool_name, query, res.decision, res.reason);\n return res;\n }\n\n const retrieval = await retrieve(ctx.graph, query);\n // \"low\" = no real matches → let Grep through; Synthra has nothing useful.\n // \"medium\" + \"high\" = Synthra has structured context for this query →\n // bias toward blocking. The pitch (\"use graph_continue instead of Grep\")\n // holds at medium too — on real codebases of any size, \"high\" is rare\n // because almost every query matches multiple files.\n if (retrieval.confidence === \"low\") {\n const res: GateResponse = {\n decision: \"allow\",\n reason: `confidence=low — no graph context for \"${query}\", letting ${req.tool_name} through`,\n };\n await logDecision(ctx, req.tool_name, query, res.decision, res.reason);\n return res;\n }\n\n // Medium / high — but check if recent activity overlaps the query first.\n // If the user just touched a file matching the query, static context may\n // be stale and they probably want a fresh search.\n const qTokens = new Set(tokenizeQuery(query));\n const recentPaths = ctx.activity.recentFilePaths(RECENT_ACTIVITY_WINDOW_MS);\n const overlap = recentlyTouchedMatchesQuery(recentPaths, qTokens);\n\n if (overlap.length > 0) {\n const res: GateResponse = {\n decision: \"allow\",\n reason:\n `confidence=${retrieval.confidence} but human just touched ${overlap.slice(0, 3).join(\", \")} — ` +\n `static context may be stale, letting ${req.tool_name} through.`,\n };\n await logDecision(ctx, req.tool_name, query, res.decision, res.reason);\n return res;\n }\n\n // Guard 2 — the graph matched files only by keyword/path, not by a symbol the\n // query names, so graph_read can't return a real slice. A block would just\n // force a fallback Read; let the search through instead.\n if (!retrieval.symbolMatched) {\n const res: GateResponse = {\n decision: \"allow\",\n reason:\n `confidence=${retrieval.confidence} but only keyword/path matched (no symbol the query names) — ` +\n `graph_read can't slice it, letting ${req.tool_name} through.`,\n };\n await logDecision(ctx, req.tool_name, query, res.decision, res.reason);\n return res;\n }\n\n const top = retrieval.files.slice(0, 3).map((f) => f.path).join(\", \");\n const res: GateResponse = {\n decision: \"block\",\n reason:\n `Synthra has ${retrieval.confidence}-confidence context for \"${query}\" (top files: ${top}). ` +\n `Use the \\`graph_continue\\` MCP tool with this query instead of ${req.tool_name}, ` +\n `or read a specific file/symbol with \\`graph_read\\`.`,\n };\n await logDecision(ctx, req.tool_name, query, res.decision, res.reason);\n return res;\n}\n","// POST /log — Stop hook posts per-turn token usage parsed from Claude's\n// transcript JSONL. Synthra appends each entry as one line to token_log.jsonl.\n\nimport { appendFile, mkdir } from \"node:fs/promises\";\nimport { dirname } from \"node:path\";\n\nimport type { ServerContext } from \"../context.js\";\n\nexport interface LogEntry {\n input_tokens: number;\n output_tokens: number;\n cache_creation_input_tokens?: number;\n cache_read_input_tokens?: number;\n model: string;\n description?: string;\n project: string;\n}\n\nexport interface LogResponse {\n ok: true;\n written_at: string;\n}\n\nexport async function handleLog(entry: LogEntry, ctx: ServerContext): Promise<LogResponse> {\n if (!entry || typeof entry.input_tokens !== \"number\" || typeof entry.output_tokens !== \"number\") {\n throw new Error(\"log: input_tokens and output_tokens (number) are required\");\n }\n\n const written_at = new Date().toISOString();\n const record = { ...entry, written_at };\n await mkdir(dirname(ctx.paths.tokenLog), { recursive: true });\n await appendFile(ctx.paths.tokenLog, JSON.stringify(record) + \"\\n\", \"utf8\");\n\n return { ok: true, written_at };\n}\n","// POST /pack { query, maxTokens? } → ContextPack JSON.\n// Runs retrieve → pack against the in-memory graph.\n\nimport { retrieve } from \"../../graph/retrieve.js\";\nimport { scoreFiles } from \"../../graph/rank.js\";\nimport { pack, type ContextPack } from \"../../packer/index.js\";\nimport type { ServerContext } from \"../context.js\";\n\nexport interface PackRequest {\n query: string;\n maxTokens?: number;\n includeTests?: boolean;\n}\n\nexport interface PackResponse extends ContextPack {\n query: string;\n confidence: \"high\" | \"medium\" | \"low\";\n retrievalReason: string;\n}\n\nexport async function handlePack(req: PackRequest, ctx: ServerContext): Promise<PackResponse> {\n if (!req?.query || typeof req.query !== \"string\") {\n throw new Error(\"pack: 'query' (string) is required\");\n }\n\n const recentlyEditedPaths = ctx.activity.recentFilePaths(15 * 60 * 1000);\n const retrieval = await retrieve(ctx.graph, req.query, { recentlyEditedPaths });\n\n // Surface per-file scoring rationale in the rendered pack.\n const allFiles = ctx.graph.nodes.filter((n) => n.kind === \"file\");\n const scored = scoreFiles({\n candidates: allFiles as Parameters<typeof scoreFiles>[0][\"candidates\"],\n query: req.query,\n graph: ctx.graph,\n recentlyEditedPaths,\n });\n const reasons = new Map<string, string>();\n for (const s of scored) {\n if (s.reasons.length) reasons.set(s.file.path, s.reasons.join(\",\"));\n }\n\n const result = await pack(retrieval.files, {\n query: req.query,\n graph: ctx.graph,\n budgetTokens: req.maxTokens,\n includeTests: req.includeTests,\n reasons,\n });\n\n return {\n ...result,\n query: req.query,\n confidence: retrieval.confidence,\n retrievalReason: retrieval.reason,\n };\n}\n","// GET /prime — SessionStart and PreCompact hooks call this. Returns the\n// priming text Claude sees at session start.\n//\n// For M2 we return a minimal primer derived from the graph (file count + a\n// short top-level summary). M3 layers in CONTEXT.md narrative + recent\n// stored decisions.\n\nimport type { ServerContext } from \"../context.js\";\n\nexport interface PrimeResponse {\n primer: string;\n port: number;\n}\n\nexport async function handlePrime(ctx: ServerContext, port: number): Promise<PrimeResponse> {\n const g = ctx.graph;\n const fileCount = g.file_count;\n const symbolCount = g.symbol_count;\n\n const primer =\n `Synthra context loaded for ${g.root}.\\n` +\n `${fileCount} files indexed, ${symbolCount} symbols. ` +\n `Prefer the graph_* MCP tools over Grep/Glob for navigation.\\n` +\n `(Full primer wired in M3.)`;\n\n return { primer, port };\n}\n"],"mappings":"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","names":["writeFile","emit","err","readFile","join","dirname","join","readFile","QUERY","QUERY","Query","QUERY","firstLine","Query","QUERY","QUERY","QUERY","QUERY","Query","QUERY","firstLine","Query","QUERY","QUERY","Query","firstLine","Query","SCRIPT_RE","extractScripts","require","readFile","readFile","join","relative","sep","ignore","readIgnoreFile","buildMatcher","toPosix","mkdir","readFile","dirname","join","mkdir","readFile","stat","writeFile","basename","readFile","writeFile","dirname","stat","mkdir","readFile","writeFile","basename","err","STOPWORDS","score","execFile","readFile","join","promisify","execFileAsync","mkdir","readFile","writeFile","dirname","mkdir","readFile","writeFile","dirname","score","indexSymbolsByFile","resolve","appendFile","mkdir","dirname","mkdir","dirname","appendFile","res","appendFile","mkdir","dirname","err","writeFile","resolve"]}
|