zam-core 0.3.7 → 0.3.11
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/.agent/skills/zam/SKILL.md +170 -7
- package/.agents/skills/zam/SKILL.md +170 -7
- package/.claude/skills/zam/SKILL.md +170 -7
- package/README.de.md +3 -3
- package/README.md +3 -3
- package/dist/cli/index.js +3145 -1427
- package/dist/cli/index.js.map +1 -1
- package/dist/index.d.ts +499 -151
- package/dist/index.js +1885 -864
- package/dist/index.js.map +1 -1
- package/package.json +1 -1
package/dist/index.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/kernel/analytics/stats.ts","../src/kernel/credentials.ts","../src/kernel/connectors/azure-devops.ts","../src/kernel/db/connection.ts","../src/kernel/db/schema.ts","../src/kernel/goals/engine.ts","../src/kernel/goals/parser.ts","../src/kernel/models/agent-skill.ts","../src/kernel/models/card.ts","../src/kernel/models/prerequisite.ts","../src/kernel/models/review.ts","../src/kernel/models/session.ts","../src/kernel/models/settings.ts","../src/kernel/models/token.ts","../src/kernel/observation/analyzer.ts","../src/kernel/observation/monitor-io.ts","../src/kernel/observation/shell-hooks.ts","../src/kernel/observation/skill-discovery.ts","../src/kernel/scheduler/blocker.ts","../src/kernel/recall/evaluator.ts","../src/kernel/scheduler/fsrs.ts","../src/kernel/recall/actions.ts","../src/kernel/recall/prompter.ts","../src/kernel/recall/reference-resolver.ts","../src/kernel/scheduler/interleaver.ts","../src/kernel/scheduler/queue.ts","../src/kernel/system/hooks.ts","../src/kernel/system/i18n.ts","../src/kernel/system/installer.ts","../src/kernel/system/locale.ts","../src/kernel/system/profiler.ts","../src/kernel/system/repos.ts"],"sourcesContent":["/**\n * Learning Analytics\n *\n * Progress statistics, competence tracking, and session summaries.\n * Ported from PoC's `stats` command with additions for FSRS and symbiosis modes.\n */\n\nimport type { Database } from \"../db/types.js\";\n\nexport interface UserStats {\n userId: string;\n totalTokens: number;\n cardsInDeck: number;\n dueToday: number;\n blocked: number;\n mature: number;\n avgStability: number | null;\n totalSessions: number;\n lastSession: string | null;\n}\n\nexport interface DomainCompetence {\n domain: string;\n totalCards: number;\n matureCards: number;\n avgStability: number;\n retentionRate: number;\n suggestedMode: \"shadowing\" | \"copilot\" | \"autonomy\";\n}\n\nfunction q(db: Database, sql: string, ...params: unknown[]) {\n return db.prepare(sql).get(...params) as Record<string, unknown>;\n}\n\n/**\n * Get overall learning stats for a user (ported from PoC's `stats` command).\n */\nexport function getUserStats(db: Database, userId: string): UserStats {\n return {\n userId,\n totalTokens: (q(db, \"SELECT COUNT(*) as n FROM tokens\") as { n: number }).n,\n cardsInDeck: (\n q(db, \"SELECT COUNT(*) as n FROM cards WHERE user_id = ?\", userId) as {\n n: number;\n }\n ).n,\n dueToday: (\n q(\n db,\n \"SELECT COUNT(*) as n FROM cards WHERE user_id = ? AND blocked = 0 AND due_at <= datetime('now')\",\n userId,\n ) as { n: number }\n ).n,\n blocked: (\n q(\n db,\n \"SELECT COUNT(*) as n FROM cards WHERE user_id = ? AND blocked = 1\",\n userId,\n ) as { n: number }\n ).n,\n mature: (\n q(\n db,\n \"SELECT COUNT(*) as n FROM cards WHERE user_id = ? AND reps >= 3 AND stability >= 21\",\n userId,\n ) as { n: number }\n ).n,\n avgStability: (() => {\n const v = q(\n db,\n \"SELECT AVG(stability) as v FROM cards WHERE user_id = ? AND reps > 0\",\n userId,\n ) as { v: number | null };\n return v.v ? Math.round(v.v * 100) / 100 : null;\n })(),\n totalSessions: (\n q(db, \"SELECT COUNT(*) as n FROM sessions WHERE user_id = ?\", userId) as {\n n: number;\n }\n ).n,\n lastSession: (() => {\n const r = db\n .prepare(\n \"SELECT started_at FROM sessions WHERE user_id = ? ORDER BY started_at DESC LIMIT 1\",\n )\n .get(userId) as { started_at: string } | undefined;\n return r?.started_at ?? null;\n })(),\n };\n}\n\n/**\n * Get competence per domain for a user.\n * Used to suggest symbiosis mode transitions.\n */\nexport function getDomainCompetence(\n db: Database,\n userId: string,\n): DomainCompetence[] {\n const domains = db\n .prepare(\n `SELECT DISTINCT t.domain FROM cards c\n JOIN tokens t ON t.id = c.token_id\n WHERE c.user_id = ? AND t.domain != ''`,\n )\n .all(userId) as { domain: string }[];\n\n return domains.map((d) => {\n const total = (\n q(\n db,\n `SELECT COUNT(*) as n FROM cards c\n JOIN tokens t ON t.id = c.token_id\n WHERE c.user_id = ? AND t.domain = ?`,\n userId,\n d.domain,\n ) as { n: number }\n ).n;\n\n const mature = (\n q(\n db,\n `SELECT COUNT(*) as n FROM cards c\n JOIN tokens t ON t.id = c.token_id\n WHERE c.user_id = ? AND t.domain = ? AND c.reps >= 3 AND c.stability >= 21`,\n userId,\n d.domain,\n ) as { n: number }\n ).n;\n\n const avgStab =\n (\n q(\n db,\n `SELECT AVG(c.stability) as v FROM cards c\n JOIN tokens t ON t.id = c.token_id\n WHERE c.user_id = ? AND t.domain = ? AND c.reps > 0`,\n userId,\n d.domain,\n ) as { v: number | null }\n ).v ?? 0;\n\n // Estimate retention from review history\n const reviews = q(\n db,\n `SELECT COUNT(*) as total,\n SUM(CASE WHEN rating >= 2 THEN 1 ELSE 0 END) as passed\n FROM review_logs\n WHERE user_id = ? AND token_id IN (SELECT id FROM tokens WHERE domain = ?)`,\n userId,\n d.domain,\n ) as { total: number; passed: number };\n\n const retentionRate =\n reviews.total > 0 ? reviews.passed / reviews.total : 0;\n\n let suggestedMode: DomainCompetence[\"suggestedMode\"];\n if (retentionRate > 0.9 && avgStab > 30) {\n suggestedMode = \"autonomy\";\n } else if (retentionRate > 0.7 && avgStab > 7) {\n suggestedMode = \"copilot\";\n } else {\n suggestedMode = \"shadowing\";\n }\n\n return {\n domain: d.domain,\n totalCards: total,\n matureCards: mature,\n avgStability: Math.round(avgStab * 100) / 100,\n retentionRate: Math.round(retentionRate * 1000) / 1000,\n suggestedMode,\n };\n });\n}\n","/**\n * Credential store — reads/writes ~/.zam/credentials.json\n *\n * Connector secrets (Turso URL/token, ADO PAT, etc.) live here instead of\n * inside the SQLite database. This ensures credentials survive db deletion,\n * which is required when migrating from plain SQLite to a libsql embedded\n * replica (Turso cloud sync).\n */\n\nimport { existsSync, mkdirSync, readFileSync, writeFileSync } from \"node:fs\";\nimport { homedir } from \"node:os\";\nimport { dirname, join } from \"node:path\";\n\nconst DEFAULT_CREDENTIALS_PATH = join(homedir(), \".zam\", \"credentials.json\");\n\nexport interface TursoCredentials {\n url: string;\n token: string;\n}\n\nexport interface ADOCredentials {\n org_url: string;\n project: string;\n pat: string;\n}\n\nexport interface Credentials {\n turso?: Partial<TursoCredentials>;\n ado?: Partial<ADOCredentials>;\n}\n\n/** Load credentials from ~/.zam/credentials.json. Returns empty object if missing. */\nexport function loadCredentials(path?: string): Credentials {\n const p = path ?? DEFAULT_CREDENTIALS_PATH;\n if (!existsSync(p)) return {};\n try {\n return JSON.parse(readFileSync(p, \"utf-8\")) as Credentials;\n } catch {\n return {};\n }\n}\n\n/** Save credentials to ~/.zam/credentials.json. */\nexport function saveCredentials(creds: Credentials, path?: string): void {\n const p = path ?? DEFAULT_CREDENTIALS_PATH;\n const dir = dirname(p);\n if (!existsSync(dir)) {\n mkdirSync(dir, { recursive: true });\n }\n writeFileSync(p, `${JSON.stringify(creds, null, 2)}\\n`, \"utf-8\");\n}\n\n/** Get complete Turso credentials, or null if incomplete. */\nexport function getTursoCredentials(path?: string): TursoCredentials | null {\n const creds = loadCredentials(path);\n if (creds.turso?.url && creds.turso?.token) {\n return { url: creds.turso.url, token: creds.turso.token };\n }\n return null;\n}\n\n/** Set Turso credentials. */\nexport function setTursoCredentials(\n url: string,\n token: string,\n path?: string,\n): void {\n const creds = loadCredentials(path);\n creds.turso = { url, token };\n saveCredentials(creds, path);\n}\n\n/** Clear Turso credentials. */\nexport function clearTursoCredentials(path?: string): void {\n const creds = loadCredentials(path);\n delete creds.turso;\n saveCredentials(creds, path);\n}\n\n/** Get complete ADO credentials, or null if incomplete. */\nexport function getADOCredentials(path?: string): ADOCredentials | null {\n const creds = loadCredentials(path);\n if (creds.ado?.org_url && creds.ado?.project && creds.ado?.pat) {\n return {\n org_url: creds.ado.org_url,\n project: creds.ado.project,\n pat: creds.ado.pat,\n };\n }\n return null;\n}\n\n/** Set ADO credentials. */\nexport function setADOCredentials(\n orgUrl: string,\n project: string,\n pat: string,\n path?: string,\n): void {\n const creds = loadCredentials(path);\n creds.ado = { org_url: orgUrl, project, pat };\n saveCredentials(creds, path);\n}\n\n/** Clear ADO credentials. */\nexport function clearADOCredentials(path?: string): void {\n const creds = loadCredentials(path);\n delete creds.ado;\n saveCredentials(creds, path);\n}\n","/**\n * Azure DevOps connector — fetches work items from ADO boards.\n */\n\nimport { getADOCredentials } from \"../credentials.js\";\n\nexport interface ADOConfig {\n orgUrl: string;\n project: string;\n pat: string;\n}\n\nexport interface WorkItem {\n id: number;\n title: string;\n state: string;\n type: string;\n assignedTo: string;\n}\n\n/** Load ADO config from credentials file. Returns null if not configured. */\nexport function loadADOConfig(): ADOConfig | null {\n const creds = getADOCredentials();\n if (!creds) return null;\n return {\n orgUrl: creds.org_url.replace(/\\/+$/, \"\"),\n project: creds.project,\n pat: creds.pat,\n };\n}\n\nfunction authHeader(pat: string): string {\n return `Basic ${Buffer.from(`:${pat}`).toString(\"base64\")}`;\n}\n\n/**\n * Fetch active work items assigned to the current user.\n * Uses WIQL to query, then batch-fetches work item details.\n */\nexport async function fetchActiveWorkItems(\n config: ADOConfig,\n): Promise<WorkItem[]> {\n const { orgUrl, project, pat } = config;\n\n // Step 1: WIQL query for work item IDs\n const wiqlUrl = `${orgUrl}/${project}/_apis/wit/wiql?api-version=7.1`;\n const wiqlBody = {\n query: `SELECT [System.Id] FROM WorkItems WHERE [System.AssignedTo] = @me AND [System.State] NOT IN ('Closed', 'Completed', 'Done', 'Removed') ORDER BY [Microsoft.VSTS.Common.Priority] ASC, [System.ChangedDate] DESC`,\n };\n\n const wiqlRes = await fetch(wiqlUrl, {\n method: \"POST\",\n headers: {\n Authorization: authHeader(pat),\n \"Content-Type\": \"application/json\",\n },\n body: JSON.stringify(wiqlBody),\n });\n\n if (!wiqlRes.ok) {\n const text = await wiqlRes.text();\n throw new Error(`ADO WIQL query failed (${wiqlRes.status}): ${text}`);\n }\n\n const wiqlData = (await wiqlRes.json()) as { workItems: { id: number }[] };\n const ids = wiqlData.workItems.map((wi) => wi.id);\n\n if (ids.length === 0) return [];\n\n // Step 2: Batch fetch work item details (max 200 per request)\n const batchIds = ids.slice(0, 200);\n const fields =\n \"System.Id,System.Title,System.State,System.WorkItemType,System.AssignedTo\";\n const detailUrl = `${orgUrl}/${project}/_apis/wit/workitems?ids=${batchIds.join(\",\")}&fields=${fields}&api-version=7.1`;\n\n const detailRes = await fetch(detailUrl, {\n headers: { Authorization: authHeader(pat) },\n });\n\n if (!detailRes.ok) {\n const text = await detailRes.text();\n throw new Error(\n `ADO work items fetch failed (${detailRes.status}): ${text}`,\n );\n }\n\n const detailData = (await detailRes.json()) as {\n value: Array<{\n id: number;\n fields: {\n \"System.Title\": string;\n \"System.State\": string;\n \"System.WorkItemType\": string;\n \"System.AssignedTo\"?: { displayName: string };\n };\n }>;\n };\n\n return detailData.value.map((wi) => ({\n id: wi.id,\n title: wi.fields[\"System.Title\"],\n state: wi.fields[\"System.State\"],\n type: wi.fields[\"System.WorkItemType\"],\n assignedTo: wi.fields[\"System.AssignedTo\"]?.displayName ?? \"\",\n }));\n}\n","import { existsSync, mkdirSync, readFileSync, rmSync } from \"node:fs\";\nimport { createRequire } from \"node:module\";\nimport { homedir } from \"node:os\";\nimport { dirname, join } from \"node:path\";\nimport BetterSqlite3 from \"better-sqlite3\";\nimport { getTursoCredentials } from \"../credentials.js\";\nimport { SCHEMA } from \"./schema.js\";\nimport type { Database as DatabaseType } from \"./types.js\";\n\nconst DEFAULT_DB_DIR = join(homedir(), \".zam\");\nconst DEFAULT_DB_PATH = join(DEFAULT_DB_DIR, \"zam.db\");\nconst require = createRequire(import.meta.url);\n\ntype LibsqlConstructor = new (\n path: string,\n options?: Record<string, unknown>,\n) => DatabaseType;\n\nexport interface ConnectionOptions {\n /** Path to the SQLite database file. Defaults to ~/.zam/zam.db */\n dbPath?: string;\n /** If true, create the directory and run schema migrations on open */\n initialize?: boolean;\n /** Turso sync URL for embedded replica mode (e.g. libsql://db-name.turso.io) */\n syncUrl?: string;\n /** Turso auth token for direct remote or embedded replica access */\n authToken?: string;\n /** If false, ignore ~/.zam/credentials.json and force the local/default database. */\n useConfiguredCloud?: boolean;\n}\n\nfunction isRemoteDatabasePath(dbPath: string): boolean {\n return /^(libsql|https?):\\/\\//i.test(dbPath);\n}\n\nfunction openLocalSqlite(dbPath: string): DatabaseType {\n return new BetterSqlite3(dbPath) as unknown as DatabaseType;\n}\n\nfunction loadLibsql(): LibsqlConstructor {\n try {\n const module = require(\"libsql\") as\n | LibsqlConstructor\n | { default: LibsqlConstructor };\n return \"default\" in module ? module.default : module;\n } catch (err) {\n const detail = err instanceof Error ? ` ${err.message}` : \"\";\n throw new Error(\n \"Turso sync requires the optional native libsql backend, which is not \" +\n `available for ${process.platform}/${process.arch}.${detail}`,\n );\n }\n}\n\n/**\n * Open (or create) the ZAM database.\n * Uses configured Turso credentials for the default database when present.\n * Falls back to local SQLite and WAL mode when no cloud credentials exist.\n * When syncUrl is provided explicitly, enables embedded replica sync with Turso.\n */\nexport function openDatabase(options: ConnectionOptions = {}): DatabaseType {\n const configuredCloud =\n options.useConfiguredCloud !== false && !options.dbPath && !options.syncUrl\n ? getTursoCredentials()\n : null;\n\n let requiresTurso = false;\n try {\n const configPath = join(process.cwd(), \".zam\", \"config.yaml\");\n if (existsSync(configPath)) {\n const configText = readFileSync(configPath, \"utf-8\");\n if (/[\\s\\S]*turso:[\\s\\S]*url:/m.test(configText)) {\n requiresTurso = true;\n }\n }\n } catch (_e) {}\n\n if (\n requiresTurso &&\n !configuredCloud &&\n options.useConfiguredCloud !== false &&\n !options.dbPath &&\n !options.syncUrl\n ) {\n throw new Error(\n \"Turso cloud database is configured in .zam/config.yaml but missing local credentials. Run: zam connector setup turso\",\n );\n }\n const dbPath = configuredCloud?.url ?? options.dbPath ?? DEFAULT_DB_PATH;\n const isRemote = isRemoteDatabasePath(dbPath);\n const isEmbeddedReplica = Boolean(options.syncUrl);\n\n if (options.initialize && !isRemote) {\n const dir = dirname(dbPath);\n if (!existsSync(dir)) {\n mkdirSync(dir, { recursive: true });\n }\n }\n\n // Build constructor options for the optional libsql cloud/sync backend.\n const dbOpts: Record<string, unknown> = {};\n if (options.syncUrl) {\n dbOpts.syncUrl = options.syncUrl;\n\n // When syncUrl is provided, the db must be a libsql embedded replica (not\n // plain SQLite). The presence of a companion .meta (or -info) file proves\n // it was created by libsql.\n //\n // If the db exists WITHOUT metadata, it was created before Turso was\n // configured — delete it so libsql can sync fresh from cloud.\n //\n // If metadata exists WITHOUT the db, libsql throws InvalidLocalState —\n // delete the metadata so it can start fresh.\n const metaPath = `${dbPath}.meta`;\n const infoPath = `${dbPath}-info`;\n\n if (existsSync(dbPath) && !existsSync(metaPath) && !existsSync(infoPath)) {\n for (const suffix of [\"\", \"-wal\", \"-shm\"]) {\n const f = `${dbPath}${suffix}`;\n if (existsSync(f)) rmSync(f, { force: true });\n }\n } else if (\n !existsSync(dbPath) &&\n (existsSync(metaPath) || existsSync(infoPath))\n ) {\n if (existsSync(metaPath)) rmSync(metaPath);\n if (existsSync(infoPath)) rmSync(infoPath);\n }\n }\n const authToken = configuredCloud?.token ?? options.authToken;\n if (authToken) {\n dbOpts.authToken = authToken;\n }\n\n let db: DatabaseType;\n if (isRemote || isEmbeddedReplica) {\n const LibsqlDatabase = loadLibsql();\n try {\n db = new LibsqlDatabase(dbPath, dbOpts);\n } catch (err) {\n const msg = (err as Error).message;\n if (msg.includes(\"InvalidLocalState\") && options.syncUrl) {\n // Last-ditch recovery: metadata is corrupt or mismatched\n const metaPath = `${dbPath}.meta`;\n const infoPath = `${dbPath}-info`;\n if (existsSync(metaPath)) rmSync(metaPath);\n if (existsSync(infoPath)) rmSync(infoPath);\n db = new LibsqlDatabase(dbPath, dbOpts);\n } else {\n throw err;\n }\n }\n } else {\n db = openLocalSqlite(dbPath);\n }\n\n // Enable WAL mode and foreign keys for local SQLite.\n // Remote Turso databases and embedded replicas manage their own journaling.\n if (!isRemote && !isEmbeddedReplica) {\n db.pragma(\"journal_mode = WAL\");\n }\n db.pragma(\"foreign_keys = ON\");\n if (!isRemote) {\n db.pragma(\"busy_timeout = 5000\");\n }\n\n // For embedded replicas: sync from cloud FIRST so the local file has the\n // primary's schema before we try to run migrations or create tables.\n if (isEmbeddedReplica) {\n db.sync?.();\n }\n\n if (options.initialize) {\n db.exec(SCHEMA);\n }\n\n runMigrations(db);\n\n return db;\n}\n\n/**\n * Open the database with Turso cloud credentials auto-detected.\n * Credentials live in ~/.zam/credentials.json (NOT in the db), so a fresh\n * machine only has to collect missing secrets instead of bootstrapping local\n * state first.\n */\nexport function openDatabaseWithSync(\n options: Omit<ConnectionOptions, \"syncUrl\" | \"authToken\"> = {},\n): DatabaseType {\n return openDatabase(options);\n}\n\n/** Get the default database path */\nexport function getDefaultDbPath(): string {\n return DEFAULT_DB_PATH;\n}\n\n/**\n * Run incremental schema migrations on every open.\n * Each migration is idempotent — safe to run repeatedly.\n */\nfunction runMigrations(db: DatabaseType): void {\n // M001: add execution_context to sessions\n const sessionCols = db.pragma(\"table_info(sessions)\") as Array<{\n name: string;\n }>;\n if (\n sessionCols.length > 0 &&\n !sessionCols.some((c) => c.name === \"execution_context\")\n ) {\n db.exec(\n `ALTER TABLE sessions ADD COLUMN execution_context TEXT NOT NULL DEFAULT 'shell'`,\n );\n }\n\n // M002: add deprecated_at to tokens\n const tokenCols = db.pragma(\"table_info(tokens)\") as Array<{ name: string }>;\n if (\n tokenCols.length > 0 &&\n !tokenCols.some((c) => c.name === \"deprecated_at\")\n ) {\n db.exec(`ALTER TABLE tokens ADD COLUMN deprecated_at TEXT`);\n }\n\n // M004: add source_link to tokens\n if (\n tokenCols.length > 0 &&\n !tokenCols.some((c) => c.name === \"source_link\")\n ) {\n db.exec(`ALTER TABLE tokens ADD COLUMN source_link TEXT`);\n }\n\n // M005: add question to tokens\n if (tokenCols.length > 0 && !tokenCols.some((c) => c.name === \"question\")) {\n db.exec(`ALTER TABLE tokens ADD COLUMN question TEXT`);\n }\n\n // M003: create agent_skills table (idempotent via IF NOT EXISTS in SCHEMA,\n // but also needed for databases that skipped the init path)\n db.exec(`\n CREATE TABLE IF NOT EXISTS agent_skills (\n id TEXT PRIMARY KEY,\n slug TEXT NOT NULL UNIQUE,\n description TEXT NOT NULL,\n steps TEXT NOT NULL DEFAULT '[]',\n token_slugs TEXT NOT NULL DEFAULT '[]',\n source TEXT NOT NULL DEFAULT 'learned',\n created_at TEXT NOT NULL DEFAULT (datetime('now')),\n updated_at TEXT NOT NULL DEFAULT (datetime('now'))\n )\n `);\n}\n","/**\n * ZAM Learning Kernel — SQLite Schema\n *\n * Evolves the PoC's schema with:\n * - FSRS scheduling fields (replaces SM-2's ef/interval_days)\n * - Bloom taxonomy levels on tokens\n * - Symbiosis modes (shadowing/copilot/autonomy)\n * - ULID-based IDs\n * - Immutable review log\n */\n\nexport const SCHEMA = `\n-- PRAGMAs (WAL, foreign_keys) are set programmatically in connection.ts,\n-- not here, because libsql embedded replicas manage their own WAL.\n\n-- Knowledge tokens: atomic concepts/facts with Bloom levels\nCREATE TABLE IF NOT EXISTS tokens (\n id TEXT PRIMARY KEY,\n slug TEXT UNIQUE NOT NULL,\n concept TEXT NOT NULL,\n domain TEXT NOT NULL DEFAULT '',\n bloom_level INTEGER NOT NULL DEFAULT 1 CHECK (bloom_level BETWEEN 1 AND 5),\n context TEXT NOT NULL DEFAULT '',\n symbiosis_mode TEXT CHECK (symbiosis_mode IN ('shadowing', 'copilot', 'autonomy')),\n source_link TEXT,\n created_at TEXT NOT NULL DEFAULT (datetime('now')),\n updated_at TEXT NOT NULL DEFAULT (datetime('now')),\n deprecated_at TEXT,\n question TEXT\n);\n\n-- Prerequisite dependency graph: \"to learn A, first know B\"\nCREATE TABLE IF NOT EXISTS prerequisites (\n token_id TEXT NOT NULL REFERENCES tokens(id) ON DELETE CASCADE,\n requires_id TEXT NOT NULL REFERENCES tokens(id) ON DELETE CASCADE,\n PRIMARY KEY (token_id, requires_id)\n);\n\n-- Per-user scheduling state for each token (FSRS fields)\nCREATE TABLE IF NOT EXISTS cards (\n id TEXT PRIMARY KEY,\n token_id TEXT NOT NULL REFERENCES tokens(id) ON DELETE CASCADE,\n user_id TEXT NOT NULL,\n stability REAL NOT NULL DEFAULT 0.0,\n difficulty REAL NOT NULL DEFAULT 0.5,\n elapsed_days REAL NOT NULL DEFAULT 0.0,\n scheduled_days REAL NOT NULL DEFAULT 0.0,\n reps INTEGER NOT NULL DEFAULT 0,\n lapses INTEGER NOT NULL DEFAULT 0,\n state TEXT NOT NULL DEFAULT 'new' CHECK (state IN ('new', 'learning', 'review', 'relearning')),\n due_at TEXT NOT NULL DEFAULT (datetime('now')),\n last_review_at TEXT,\n blocked INTEGER NOT NULL DEFAULT 0,\n UNIQUE(token_id, user_id)\n);\n\n-- Immutable review log: every rating event\nCREATE TABLE IF NOT EXISTS review_logs (\n id TEXT PRIMARY KEY,\n card_id TEXT NOT NULL REFERENCES cards(id) ON DELETE CASCADE,\n token_id TEXT NOT NULL REFERENCES tokens(id) ON DELETE CASCADE,\n user_id TEXT NOT NULL,\n rating INTEGER NOT NULL CHECK (rating BETWEEN 1 AND 4),\n response_time_ms INTEGER,\n reviewed_at TEXT NOT NULL DEFAULT (datetime('now')),\n scheduled_at TEXT NOT NULL,\n session_id TEXT REFERENCES sessions(id)\n);\n\n-- Work+learning sessions\nCREATE TABLE IF NOT EXISTS sessions (\n id TEXT PRIMARY KEY,\n user_id TEXT NOT NULL,\n task TEXT NOT NULL,\n started_at TEXT NOT NULL DEFAULT (datetime('now')),\n completed_at TEXT\n);\n\n-- Steps within a session: who did what\nCREATE TABLE IF NOT EXISTS session_steps (\n id TEXT PRIMARY KEY,\n session_id TEXT NOT NULL REFERENCES sessions(id) ON DELETE CASCADE,\n token_id TEXT NOT NULL REFERENCES tokens(id) ON DELETE CASCADE,\n done_by TEXT NOT NULL CHECK (done_by IN ('user', 'agent')),\n rating INTEGER CHECK (rating BETWEEN 1 AND 4),\n notes TEXT,\n created_at TEXT NOT NULL DEFAULT (datetime('now'))\n);\n\n-- User configuration\nCREATE TABLE IF NOT EXISTS user_config (\n key TEXT PRIMARY KEY,\n value TEXT NOT NULL,\n updated_at TEXT NOT NULL DEFAULT (datetime('now'))\n);\n\n-- Agent skills: task recipes the agent learns from user guidance\nCREATE TABLE IF NOT EXISTS agent_skills (\n id TEXT PRIMARY KEY,\n slug TEXT NOT NULL UNIQUE,\n description TEXT NOT NULL,\n steps TEXT NOT NULL DEFAULT '[]', -- JSON array of step strings\n token_slugs TEXT NOT NULL DEFAULT '[]', -- JSON array of related token slugs\n source TEXT NOT NULL DEFAULT 'learned'\n CHECK(source IN ('learned', 'builtin')),\n created_at TEXT NOT NULL DEFAULT (datetime('now')),\n updated_at TEXT NOT NULL DEFAULT (datetime('now'))\n);\n\n-- Performance indexes\nCREATE INDEX IF NOT EXISTS idx_tokens_domain ON tokens(domain);\nCREATE INDEX IF NOT EXISTS idx_tokens_slug ON tokens(slug);\nCREATE INDEX IF NOT EXISTS idx_prereqs_token ON prerequisites(token_id);\nCREATE INDEX IF NOT EXISTS idx_prereqs_requires ON prerequisites(requires_id);\nCREATE INDEX IF NOT EXISTS idx_cards_user_due ON cards(user_id, blocked, due_at);\nCREATE INDEX IF NOT EXISTS idx_cards_token_user ON cards(token_id, user_id);\nCREATE INDEX IF NOT EXISTS idx_review_logs_card ON review_logs(card_id);\nCREATE INDEX IF NOT EXISTS idx_review_logs_user ON review_logs(user_id, reviewed_at);\nCREATE INDEX IF NOT EXISTS idx_session_steps_session ON session_steps(session_id);\n`;\n","/**\n * Goal Engine — manages goal lifecycle via markdown files.\n *\n * Goals live as markdown files in a directory (typically the personal repo's\n * goals/ folder). The engine reads, creates, and updates these files.\n * It does not depend on the database — goals are git-tracked, not DB-tracked.\n */\n\nimport { existsSync, readdirSync, readFileSync, writeFileSync } from \"node:fs\";\nimport { basename, join } from \"node:path\";\nimport type { Goal, GoalStatus } from \"./parser.js\";\nimport {\n extractTasks,\n extractTokenRefs,\n parseGoalFile,\n serializeGoal,\n} from \"./parser.js\";\n\n// ── Types ────────────────────────────────────────────────────────────────────\n\nexport interface GoalSummary {\n slug: string;\n title: string;\n status: GoalStatus;\n parent: string | null;\n taskCount: number;\n tasksDone: number;\n tokenCount: number;\n}\n\nexport interface CreateGoalInput {\n slug: string;\n title: string;\n status?: GoalStatus;\n parent?: string;\n description?: string;\n}\n\n// ── Functions ────────────────────────────────────────────────────────────────\n\n/**\n * List all goals in the goals directory.\n * Returns summaries sorted by status (active first) then title.\n */\nexport function listGoals(goalsDir: string): GoalSummary[] {\n if (!existsSync(goalsDir)) return [];\n\n const files = readdirSync(goalsDir).filter(\n (f) => f.endsWith(\".md\") && f !== \"README.md\",\n );\n\n const summaries: GoalSummary[] = [];\n\n for (const file of files) {\n const filePath = join(goalsDir, file);\n const content = readFileSync(filePath, \"utf-8\");\n const slug = basename(file, \".md\");\n const goal = parseGoalFile(content, slug, filePath);\n const tasks = extractTasks(goal.body);\n const tokens = extractTokenRefs(goal.body);\n\n summaries.push({\n slug: goal.slug,\n title: goal.title,\n status: goal.status,\n parent: goal.parent,\n taskCount: tasks.length,\n tasksDone: tasks.filter((t) => t.done).length,\n tokenCount: tokens.length,\n });\n }\n\n const statusOrder: Record<GoalStatus, number> = {\n active: 0,\n paused: 1,\n completed: 2,\n abandoned: 3,\n };\n\n summaries.sort((a, b) => {\n const statusDiff = statusOrder[a.status] - statusOrder[b.status];\n if (statusDiff !== 0) return statusDiff;\n return a.title.localeCompare(b.title);\n });\n\n return summaries;\n}\n\n/**\n * Get a single goal by slug (filename without .md).\n * Returns undefined if the file doesn't exist.\n */\nexport function getGoal(goalsDir: string, slug: string): Goal | undefined {\n const filePath = join(goalsDir, `${slug}.md`);\n if (!existsSync(filePath)) return undefined;\n\n const content = readFileSync(filePath, \"utf-8\");\n return parseGoalFile(content, slug, filePath);\n}\n\n/**\n * Create a new goal file. Throws if a goal with this slug already exists.\n */\nexport function createGoal(goalsDir: string, input: CreateGoalInput): Goal {\n const filePath = join(goalsDir, `${input.slug}.md`);\n\n if (existsSync(filePath)) {\n throw new Error(`Goal already exists: ${input.slug}`);\n }\n\n const now = new Date().toISOString().slice(0, 10);\n\n const goal: Goal = {\n slug: input.slug,\n title: input.title,\n status: input.status ?? \"active\",\n parent: input.parent ?? null,\n created: now,\n updated: now,\n body: input.description\n ? `## Description\\n${input.description}\\n\\n## Tasks\\n\\n## Tokens`\n : \"## Description\\n\\n## Tasks\\n\\n## Tokens\",\n filePath,\n };\n\n writeFileSync(filePath, serializeGoal(goal), \"utf-8\");\n return goal;\n}\n\n/**\n * Update a goal's status. Writes the updated file back to disk.\n */\nexport function updateGoalStatus(\n goalsDir: string,\n slug: string,\n status: GoalStatus,\n): Goal {\n const goal = getGoal(goalsDir, slug);\n if (!goal) throw new Error(`Goal not found: ${slug}`);\n\n goal.status = status;\n goal.updated = new Date().toISOString().slice(0, 10);\n\n writeFileSync(goal.filePath, serializeGoal(goal), \"utf-8\");\n return goal;\n}\n\n/**\n * Get the goal tree — goals organized by parent relationships.\n * Returns root goals (no parent) with nested children.\n */\nexport function getGoalTree(\n goalsDir: string,\n): Array<GoalSummary & { children: GoalSummary[] }> {\n const all = listGoals(goalsDir);\n const bySlug = new Map(all.map((g) => [g.slug, g]));\n\n const roots: Array<GoalSummary & { children: GoalSummary[] }> = [];\n const children = new Map<string, GoalSummary[]>();\n\n for (const g of all) {\n if (g.parent && bySlug.has(g.parent)) {\n const list = children.get(g.parent) ?? [];\n list.push(g);\n children.set(g.parent, list);\n }\n }\n\n for (const g of all) {\n if (!g.parent || !bySlug.has(g.parent)) {\n roots.push({ ...g, children: children.get(g.slug) ?? [] });\n }\n }\n\n return roots;\n}\n","/**\n * Goal file parser — reads markdown files with YAML-style frontmatter.\n *\n * Goals are persisted as markdown files in the personal repo.\n * Each file has simple key: value frontmatter (no nested structures)\n * and a markdown body with description, tasks, and token references.\n */\n\n// ── Types ────────────────────────────────────────────────────────────────────\n\nexport type GoalStatus = \"active\" | \"completed\" | \"paused\" | \"abandoned\";\n\nexport interface Goal {\n slug: string; // derived from filename (e.g., \"learn-rust\" from \"learn-rust.md\")\n title: string;\n status: GoalStatus;\n parent: string | null; // slug of parent goal\n created: string; // ISO date\n updated: string; // ISO date\n body: string; // markdown body after frontmatter\n filePath: string; // absolute path to the file\n}\n\nexport interface GoalFrontmatter {\n title?: string;\n status?: string;\n parent?: string;\n created?: string;\n updated?: string;\n}\n\n// ── Parser ───────────────────────────────────────────────────────────────────\n\n/**\n * Parse a goal markdown file into a Goal object.\n *\n * Expected format:\n * ```\n * ---\n * title: Learn Rust fundamentals\n * status: active\n * parent: become-systems-programmer\n * created: 2026-03-28\n * updated: 2026-03-28\n * ---\n *\n * ## Description\n * ...\n * ```\n *\n * @param content - Raw file content\n * @param slug - Goal slug (derived from filename by caller)\n * @param filePath - Absolute path to the file\n */\nexport function parseGoalFile(\n content: string,\n slug: string,\n filePath: string,\n): Goal {\n const { frontmatter, body } = splitFrontmatter(content);\n\n const validStatuses: GoalStatus[] = [\n \"active\",\n \"completed\",\n \"paused\",\n \"abandoned\",\n ];\n const status = validStatuses.includes(frontmatter.status as GoalStatus)\n ? (frontmatter.status as GoalStatus)\n : \"active\";\n\n const now = new Date().toISOString().slice(0, 10);\n\n return {\n slug,\n title: frontmatter.title || slug,\n status,\n parent: frontmatter.parent || null,\n created: frontmatter.created || now,\n updated: frontmatter.updated || now,\n body,\n filePath,\n };\n}\n\n/**\n * Serialize a Goal back to markdown with frontmatter.\n */\nexport function serializeGoal(goal: Goal): string {\n const lines = [\"---\", `title: ${goal.title}`, `status: ${goal.status}`];\n\n if (goal.parent) {\n lines.push(`parent: ${goal.parent}`);\n }\n\n lines.push(`created: ${goal.created}`);\n lines.push(`updated: ${goal.updated}`);\n lines.push(\"---\");\n lines.push(\"\");\n\n if (goal.body.trim()) {\n lines.push(goal.body.trim());\n lines.push(\"\");\n }\n\n return lines.join(\"\\n\");\n}\n\n/**\n * Extract tasks (checklist items) from goal body.\n * Returns items like { text: \"Complete Rustlings\", done: false }.\n */\nexport function extractTasks(\n body: string,\n): Array<{ text: string; done: boolean }> {\n const tasks: Array<{ text: string; done: boolean }> = [];\n const taskRegex = /^[-*]\\s+\\[([ xX])\\]\\s+(.+)$/gm;\n let match: RegExpExecArray | null = taskRegex.exec(body);\n\n while (match !== null) {\n tasks.push({\n done: match[1] !== \" \",\n text: match[2].trim(),\n });\n match = taskRegex.exec(body);\n }\n\n return tasks;\n}\n\n/**\n * Extract token references from goal body.\n * Looks for lines like `- token/slug` under a \"## Tokens\" section.\n */\nexport function extractTokenRefs(body: string): string[] {\n const tokensSection = body.match(/## Tokens\\n([\\s\\S]*?)(?=\\n## |\\n*$)/);\n if (!tokensSection) return [];\n\n const refs: string[] = [];\n const lines = tokensSection[1].split(\"\\n\");\n\n for (const line of lines) {\n const match = line.match(/^[-*]\\s+(\\S+)/);\n if (match) {\n refs.push(match[1]);\n }\n }\n\n return refs;\n}\n\n// ── Internal helpers ─────────────────────────────────────────────────────────\n\nfunction splitFrontmatter(content: string): {\n frontmatter: GoalFrontmatter;\n body: string;\n} {\n const trimmed = content.trim();\n\n if (!trimmed.startsWith(\"---\")) {\n return { frontmatter: {}, body: trimmed };\n }\n\n const endIndex = trimmed.indexOf(\"---\", 3);\n if (endIndex === -1) {\n return { frontmatter: {}, body: trimmed };\n }\n\n const fmBlock = trimmed.slice(3, endIndex).trim();\n const body = trimmed.slice(endIndex + 3).trim();\n\n const frontmatter: GoalFrontmatter = {};\n for (const line of fmBlock.split(\"\\n\")) {\n const colonIndex = line.indexOf(\":\");\n if (colonIndex === -1) continue;\n\n const key = line.slice(0, colonIndex).trim();\n const value = line.slice(colonIndex + 1).trim();\n\n if (key && value) {\n (frontmatter as Record<string, string>)[key] = value;\n }\n }\n\n return { frontmatter, body };\n}\n","/**\n * Agent skills: task recipes the agent learns from user guidance.\n *\n * When the agent cannot execute a step, it admits it, asks for guidance,\n * and saves the successful approach here. Skills are linked to tokens so\n * FSRS decay naturally resurfaces them for review — automation ≠ retention.\n */\n\nimport { ulid } from \"ulid\";\nimport type { Database } from \"../db/types.js\";\n\n// ── Types ────────────────────────────────────────────────────────────────────\n\nexport type SkillSource = \"learned\" | \"builtin\";\n\nexport interface AgentSkill {\n id: string;\n slug: string;\n description: string;\n steps: string[]; // parsed from JSON\n token_slugs: string[]; // parsed from JSON\n source: SkillSource;\n created_at: string;\n updated_at: string;\n}\n\n/** Raw DB row — steps and token_slugs are stored as JSON strings */\ninterface AgentSkillRow {\n id: string;\n slug: string;\n description: string;\n steps: string;\n token_slugs: string;\n source: SkillSource;\n created_at: string;\n updated_at: string;\n}\n\nexport interface CreateAgentSkillInput {\n slug: string;\n description: string;\n steps: string[];\n token_slugs?: string[];\n source?: SkillSource;\n}\n\n// ── Helpers ──────────────────────────────────────────────────────────────────\n\nfunction parseRow(row: AgentSkillRow): AgentSkill {\n return {\n ...row,\n steps: JSON.parse(row.steps) as string[],\n token_slugs: JSON.parse(row.token_slugs) as string[],\n };\n}\n\n// ── Functions ────────────────────────────────────────────────────────────────\n\nexport function createAgentSkill(\n db: Database,\n input: CreateAgentSkillInput,\n): AgentSkill {\n const existing = db\n .prepare(\"SELECT * FROM agent_skills WHERE slug = ?\")\n .get(input.slug) as AgentSkillRow | undefined;\n\n if (existing) {\n throw new Error(`Agent skill already exists: ${input.slug}`);\n }\n\n const id = ulid();\n const now = new Date().toISOString();\n\n db.prepare(\n `INSERT INTO agent_skills (id, slug, description, steps, token_slugs, source, created_at, updated_at)\n VALUES (?, ?, ?, ?, ?, ?, ?, ?)`,\n ).run(\n id,\n input.slug,\n input.description,\n JSON.stringify(input.steps),\n JSON.stringify(input.token_slugs ?? []),\n input.source ?? \"learned\",\n now,\n now,\n );\n\n return parseRow(\n db\n .prepare(\"SELECT * FROM agent_skills WHERE id = ?\")\n .get(id) as AgentSkillRow,\n );\n}\n\nexport function getAgentSkill(\n db: Database,\n slug: string,\n): AgentSkill | undefined {\n const row = db\n .prepare(\"SELECT * FROM agent_skills WHERE slug = ?\")\n .get(slug) as AgentSkillRow | undefined;\n\n return row ? parseRow(row) : undefined;\n}\n\nexport function listAgentSkills(db: Database): AgentSkill[] {\n const rows = db\n .prepare(\"SELECT * FROM agent_skills ORDER BY created_at ASC\")\n .all() as AgentSkillRow[];\n\n return rows.map(parseRow);\n}\n","/**\n * Card repository — typed wrappers around the cards table.\n *\n * Each card tracks one user's scheduling state for one token,\n * using FSRS fields (stability, difficulty, elapsed_days, etc.).\n */\n\nimport { ulid } from \"ulid\";\nimport type { Database } from \"../db/types.js\";\n\n// ── Types ────────────────────────────────────────────────────────────────────\n\nexport type CardState = \"new\" | \"learning\" | \"review\" | \"relearning\";\n\nexport interface Card {\n id: string;\n token_id: string;\n user_id: string;\n stability: number;\n difficulty: number;\n elapsed_days: number;\n scheduled_days: number;\n reps: number;\n lapses: number;\n state: CardState;\n due_at: string;\n last_review_at: string | null;\n blocked: number; // 0 or 1\n}\n\nexport interface UpdateCardInput {\n stability?: number;\n difficulty?: number;\n elapsed_days?: number;\n scheduled_days?: number;\n reps?: number;\n lapses?: number;\n state?: CardState;\n due_at?: string;\n last_review_at?: string | null;\n blocked?: number;\n}\n\nexport interface CardDeletionImpact {\n review_logs: number;\n}\n\nexport interface DeleteCardResult {\n card: Card;\n impact: CardDeletionImpact;\n}\n\n/** A due card joined with its token details. */\nexport interface DueCard extends Card {\n slug: string;\n concept: string;\n domain: string;\n bloom_level: number;\n}\n\n/** A blocked card joined with its token details. */\nexport interface BlockedCard extends Card {\n slug: string;\n concept: string;\n domain: string;\n bloom_level: number;\n}\n\n// ── Functions ────────────────────────────────────────────────────────────────\n\n/**\n * Ensure a card exists for the given token+user pair.\n *\n * If one already exists, return it. Otherwise create a new card with\n * default FSRS values (due immediately) and return it.\n *\n * Ported from the PoC's ensureCard helper.\n */\nexport function ensureCard(\n db: Database,\n tokenId: string,\n userId: string,\n): Card {\n const existing = db\n .prepare(\"SELECT * FROM cards WHERE token_id = ? AND user_id = ?\")\n .get(tokenId, userId) as Card | undefined;\n\n if (existing) return existing;\n\n const id = ulid();\n const now = new Date().toISOString();\n\n db.prepare(\n `INSERT INTO cards (id, token_id, user_id, due_at)\n VALUES (?, ?, ?, ?)`,\n ).run(id, tokenId, userId, now);\n\n return db.prepare(\"SELECT * FROM cards WHERE id = ?\").get(id) as Card;\n}\n\n/**\n * Get a card by token+user. Returns undefined if no card exists.\n */\nexport function getCard(\n db: Database,\n tokenId: string,\n userId: string,\n): Card | undefined {\n return db\n .prepare(\"SELECT * FROM cards WHERE token_id = ? AND user_id = ?\")\n .get(tokenId, userId) as Card | undefined;\n}\n\n/**\n * Get a card by its ULID.\n */\nexport function getCardById(db: Database, cardId: string): Card | undefined {\n return db.prepare(\"SELECT * FROM cards WHERE id = ?\").get(cardId) as\n | Card\n | undefined;\n}\n\n/**\n * Update a card's scheduling fields.\n *\n * Only the fields present in `updates` are changed. Throws if the card\n * does not exist.\n */\nexport function updateCard(\n db: Database,\n cardId: string,\n updates: UpdateCardInput,\n): Card {\n const fields: string[] = [];\n const values: unknown[] = [];\n\n if (updates.stability !== undefined) {\n fields.push(\"stability = ?\");\n values.push(updates.stability);\n }\n if (updates.difficulty !== undefined) {\n fields.push(\"difficulty = ?\");\n values.push(updates.difficulty);\n }\n if (updates.elapsed_days !== undefined) {\n fields.push(\"elapsed_days = ?\");\n values.push(updates.elapsed_days);\n }\n if (updates.scheduled_days !== undefined) {\n fields.push(\"scheduled_days = ?\");\n values.push(updates.scheduled_days);\n }\n if (updates.reps !== undefined) {\n fields.push(\"reps = ?\");\n values.push(updates.reps);\n }\n if (updates.lapses !== undefined) {\n fields.push(\"lapses = ?\");\n values.push(updates.lapses);\n }\n if (updates.state !== undefined) {\n fields.push(\"state = ?\");\n values.push(updates.state);\n }\n if (updates.due_at !== undefined) {\n fields.push(\"due_at = ?\");\n values.push(updates.due_at);\n }\n if (updates.last_review_at !== undefined) {\n fields.push(\"last_review_at = ?\");\n values.push(updates.last_review_at);\n }\n if (updates.blocked !== undefined) {\n fields.push(\"blocked = ?\");\n values.push(updates.blocked);\n }\n\n if (fields.length === 0) {\n throw new Error(\"updateCard called with no fields to update\");\n }\n\n values.push(cardId);\n\n const result = db\n .prepare(`UPDATE cards SET ${fields.join(\", \")} WHERE id = ?`)\n .run(...values);\n\n if (result.changes === 0) {\n throw new Error(`Card not found: ${cardId}`);\n }\n\n return db.prepare(\"SELECT * FROM cards WHERE id = ?\").get(cardId) as Card;\n}\n\n/**\n * Preview the review-log rows that will be removed when deleting a user's card.\n */\nexport function getCardDeletionImpact(\n db: Database,\n tokenId: string,\n userId: string,\n): CardDeletionImpact {\n const card = getCard(db, tokenId, userId);\n if (!card) {\n throw new Error(`Card not found for token ${tokenId} and user ${userId}`);\n }\n\n const reviewLogs = db\n .prepare(\"SELECT COUNT(*) AS n FROM review_logs WHERE card_id = ?\")\n .get(card.id) as { n: number };\n\n return { review_logs: reviewLogs.n };\n}\n\n/**\n * Delete one user's card for a token. Review logs cascade via FK.\n */\nexport function deleteCardForUser(\n db: Database,\n tokenId: string,\n userId: string,\n): DeleteCardResult {\n const card = getCard(db, tokenId, userId);\n if (!card) {\n throw new Error(`Card not found for token ${tokenId} and user ${userId}`);\n }\n\n const impact = getCardDeletionImpact(db, tokenId, userId);\n db.prepare(\"DELETE FROM cards WHERE id = ?\").run(card.id);\n\n return { card, impact };\n}\n\n/**\n * Get all cards that are due for review.\n *\n * A card is due when it is not blocked and due_at <= now.\n * Results are ordered by bloom_level ascending (fundamentals first),\n * then by due_at ascending (oldest first).\n *\n * Ported from the PoC's due-tokens command.\n */\nexport function getDueCards(\n db: Database,\n userId: string,\n now?: string,\n): DueCard[] {\n const cutoff = now ?? new Date().toISOString();\n\n return db\n .prepare(\n `SELECT c.*, t.slug, t.concept, t.domain, t.bloom_level\n FROM cards c\n JOIN tokens t ON t.id = c.token_id\n WHERE c.user_id = ? AND c.blocked = 0 AND c.due_at <= ?\n ORDER BY t.bloom_level ASC, c.due_at ASC`,\n )\n .all(userId, cutoff) as DueCard[];\n}\n\n/**\n * Get all blocked cards for a user.\n *\n * Returns cards joined with their token details so the caller can\n * see what is waiting and why.\n */\nexport function getBlockedCards(db: Database, userId: string): BlockedCard[] {\n return db\n .prepare(\n `SELECT c.*, t.slug, t.concept, t.domain, t.bloom_level\n FROM cards c\n JOIN tokens t ON t.id = c.token_id\n WHERE c.user_id = ? AND c.blocked = 1\n ORDER BY t.bloom_level ASC, t.slug ASC`,\n )\n .all(userId) as BlockedCard[];\n}\n","/**\n * Prerequisite repository — typed wrappers around the prerequisites table.\n *\n * Models the dependency graph: \"to learn token A, first know token B.\"\n * The graph must remain acyclic — cycles are rejected at insert time.\n */\n\nimport type { Database } from \"../db/types.js\";\n\n// ── Types ────────────────────────────────────────────────────────────────────\n\nexport interface Prerequisite {\n token_id: string;\n requires_id: string;\n}\n\n/** A prerequisite row joined with the token it points to. */\nexport interface PrerequisiteWithToken extends Prerequisite {\n slug: string;\n concept: string;\n domain: string;\n bloom_level: number;\n}\n\n// ── Helpers ──────────────────────────────────────────────────────────────────\n\n/**\n * Collect all prerequisite edges as an adjacency map: child → parent set.\n * Only used for cycle detection; the full graph is loaded once per\n * addPrerequisite call (small N in practice).\n */\nfunction buildAncestorMap(db: Database): Map<string, Set<string>> {\n const rows = db\n .prepare(\"SELECT token_id, requires_id FROM prerequisites\")\n .all() as Array<{ token_id: string; requires_id: string }>;\n const map = new Map<string, Set<string>>();\n for (const row of rows) {\n let ancestors = map.get(row.token_id);\n if (!ancestors) {\n ancestors = new Set();\n map.set(row.token_id, ancestors);\n }\n ancestors.add(row.requires_id);\n }\n return map;\n}\n\n/**\n * Returns true if adding edge (tokenId → requiresId) would create a cycle.\n * Uses BFS from requiresId: if tokenId is reachable, adding the edge closes\n * a loop.\n */\nexport function wouldCreateCycle(\n db: Database,\n tokenId: string,\n requiresId: string,\n): boolean {\n if (tokenId === requiresId) return true;\n\n const ancestors = buildAncestorMap(db);\n const visited = new Set<string>();\n const queue = [requiresId];\n\n while (queue.length > 0) {\n const current = queue.shift()!;\n if (current === tokenId) return true;\n if (visited.has(current)) continue;\n visited.add(current);\n\n const parents = ancestors.get(current);\n if (parents) {\n for (const parent of parents) {\n if (!visited.has(parent)) queue.push(parent);\n }\n }\n }\n return false;\n}\n\n// ── Functions ────────────────────────────────────────────────────────────────\n\n/**\n * Add a prerequisite edge: tokenId requires requiresId.\n *\n * Idempotent — silently ignores duplicate edges.\n * Throws if either token ID does not exist (FK constraint).\n * Throws if a token is declared as its own prerequisite.\n * Throws if the edge would create a cycle in the prerequisite graph.\n */\nexport function addPrerequisite(\n db: Database,\n tokenId: string,\n requiresId: string,\n): void {\n if (tokenId === requiresId) {\n throw new Error(\"A token cannot be a prerequisite of itself\");\n }\n\n if (wouldCreateCycle(db, tokenId, requiresId)) {\n throw new Error(\n `Cannot add prerequisite: would create a cycle. ` +\n `${requiresId} already depends on ${tokenId} (directly or transitively).`,\n );\n }\n\n db.prepare(\n \"INSERT OR IGNORE INTO prerequisites (token_id, requires_id) VALUES (?, ?)\",\n ).run(tokenId, requiresId);\n}\n\n/**\n * Get the direct prerequisites of a token — \"what does token X require?\"\n *\n * Returns prerequisite rows joined with the required token's details.\n */\nexport function getPrerequisites(\n db: Database,\n tokenId: string,\n): PrerequisiteWithToken[] {\n return db\n .prepare(\n `SELECT p.token_id, p.requires_id, t.slug, t.concept, t.domain, t.bloom_level\n FROM prerequisites p\n JOIN tokens t ON t.id = p.requires_id\n WHERE p.token_id = ?`,\n )\n .all(tokenId) as PrerequisiteWithToken[];\n}\n\n/**\n * Get the direct dependents of a token — \"what depends on token X?\"\n *\n * Returns prerequisite rows joined with the dependent token's details.\n */\nexport function getDependents(\n db: Database,\n tokenId: string,\n): PrerequisiteWithToken[] {\n return db\n .prepare(\n `SELECT p.token_id, p.requires_id, t.slug, t.concept, t.domain, t.bloom_level\n FROM prerequisites p\n JOIN tokens t ON t.id = p.token_id\n WHERE p.requires_id = ?`,\n )\n .all(tokenId) as PrerequisiteWithToken[];\n}\n","/**\n * Review log repository — typed wrappers around the review_logs table.\n *\n * The review log is immutable: every rating event is appended, never\n * updated or deleted. This provides a complete audit trail of a user's\n * learning history.\n */\n\nimport { ulid } from \"ulid\";\nimport type { Database } from \"../db/types.js\";\n\n// ── Types ────────────────────────────────────────────────────────────────────\n\nexport interface ReviewLog {\n id: string;\n card_id: string;\n token_id: string;\n user_id: string;\n rating: number; // 1-4\n response_time_ms: number | null;\n reviewed_at: string;\n scheduled_at: string;\n session_id: string | null;\n}\n\nexport interface CreateReviewInput {\n card_id: string;\n token_id: string;\n user_id: string;\n rating: number; // 1-4\n scheduled_at: string;\n response_time_ms?: number | null;\n session_id?: string | null;\n}\n\nexport interface ListReviewsOptions {\n /** Maximum number of reviews to return. */\n limit?: number;\n /** Return reviews after this ISO timestamp. */\n after?: string;\n /** Return reviews before this ISO timestamp. */\n before?: string;\n}\n\n// ── Functions ────────────────────────────────────────────────────────────────\n\n/**\n * Log an immutable review event.\n *\n * Validates that the rating is between 1 and 4 (matching the schema CHECK).\n * Returns the created review log entry.\n */\nexport function logReview(db: Database, input: CreateReviewInput): ReviewLog {\n if (input.rating < 1 || input.rating > 4) {\n throw new Error(`Rating must be between 1 and 4, got ${input.rating}`);\n }\n\n const id = ulid();\n const now = new Date().toISOString();\n\n db.prepare(\n `INSERT INTO review_logs (id, card_id, token_id, user_id, rating, response_time_ms, reviewed_at, scheduled_at, session_id)\n VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?)`,\n ).run(\n id,\n input.card_id,\n input.token_id,\n input.user_id,\n input.rating,\n input.response_time_ms ?? null,\n now,\n input.scheduled_at,\n input.session_id ?? null,\n );\n\n return db\n .prepare(\"SELECT * FROM review_logs WHERE id = ?\")\n .get(id) as ReviewLog;\n}\n\n/**\n * Get all reviews for a specific card, ordered by reviewed_at ascending.\n */\nexport function getReviewsForCard(db: Database, cardId: string): ReviewLog[] {\n return db\n .prepare(\n \"SELECT * FROM review_logs WHERE card_id = ? ORDER BY reviewed_at ASC\",\n )\n .all(cardId) as ReviewLog[];\n}\n\n/**\n * Get reviews for a user, with optional filtering.\n *\n * Results are ordered by reviewed_at descending (most recent first).\n */\nexport function getReviewsForUser(\n db: Database,\n userId: string,\n options?: ListReviewsOptions,\n): ReviewLog[] {\n const conditions = [\"user_id = ?\"];\n const params: unknown[] = [userId];\n\n if (options?.after) {\n conditions.push(\"reviewed_at > ?\");\n params.push(options.after);\n }\n if (options?.before) {\n conditions.push(\"reviewed_at < ?\");\n params.push(options.before);\n }\n\n let sql = `SELECT * FROM review_logs WHERE ${conditions.join(\" AND \")} ORDER BY reviewed_at DESC`;\n\n if (options?.limit) {\n sql += \" LIMIT ?\";\n params.push(options.limit);\n }\n\n return db.prepare(sql).all(...params) as ReviewLog[];\n}\n","/**\n * Session repository — typed wrappers around sessions and session_steps.\n *\n * A session represents a work+learning episode. Steps within a session\n * record which tokens were touched and by whom (user or agent).\n */\n\nimport { ulid } from \"ulid\";\nimport type { Database } from \"../db/types.js\";\n\n// ── Types ────────────────────────────────────────────────────────────────────\n\nexport type ExecutionContext = \"shell\" | \"ui\" | \"reallife\";\n\nexport interface Session {\n id: string;\n user_id: string;\n task: string;\n execution_context: ExecutionContext;\n started_at: string;\n completed_at: string | null;\n}\n\nexport interface SessionStep {\n id: string;\n session_id: string;\n token_id: string;\n done_by: \"user\" | \"agent\";\n rating: number | null; // 1-4 or null\n notes: string | null;\n created_at: string;\n}\n\nexport interface CreateSessionInput {\n user_id: string;\n task: string;\n execution_context?: ExecutionContext;\n}\n\nexport interface LogStepInput {\n session_id: string;\n token_id: string;\n done_by: \"user\" | \"agent\";\n rating?: number | null;\n notes?: string | null;\n}\n\n/** A step joined with its token details, returned by getSessionSummary. */\nexport interface StepWithToken extends SessionStep {\n slug: string;\n concept: string;\n domain: string;\n bloom_level: number;\n}\n\nexport interface SessionSummary {\n session: Session;\n steps: StepWithToken[];\n}\n\n// ── Functions ────────────────────────────────────────────────────────────────\n\n/**\n * Start a new session. Returns the created session.\n *\n * Ported from the PoC's start-session command.\n */\nexport function startSession(db: Database, input: CreateSessionInput): Session {\n const id = ulid();\n const now = new Date().toISOString();\n const ctx = input.execution_context ?? \"shell\";\n\n db.prepare(\n `INSERT INTO sessions (id, user_id, task, execution_context, started_at)\n VALUES (?, ?, ?, ?, ?)`,\n ).run(id, input.user_id, input.task, ctx, now);\n\n return db.prepare(\"SELECT * FROM sessions WHERE id = ?\").get(id) as Session;\n}\n\n/**\n * End a session by setting its completed_at timestamp.\n *\n * Throws if the session does not exist or is already completed.\n *\n * Ported from the PoC's end-session command.\n */\nexport function endSession(db: Database, sessionId: string): Session {\n const session = db\n .prepare(\"SELECT * FROM sessions WHERE id = ?\")\n .get(sessionId) as Session | undefined;\n\n if (!session) {\n throw new Error(`Session not found: ${sessionId}`);\n }\n if (session.completed_at) {\n throw new Error(`Session already completed: ${sessionId}`);\n }\n\n const now = new Date().toISOString();\n db.prepare(\"UPDATE sessions SET completed_at = ? WHERE id = ?\").run(\n now,\n sessionId,\n );\n\n return db\n .prepare(\"SELECT * FROM sessions WHERE id = ?\")\n .get(sessionId) as Session;\n}\n\n/**\n * Log a step within a session.\n *\n * Validates that done_by is 'user' or 'agent' and that the rating\n * (if provided) is between 1 and 4.\n *\n * Ported from the PoC's log-step command.\n */\nexport function logStep(db: Database, input: LogStepInput): SessionStep {\n if (input.done_by !== \"user\" && input.done_by !== \"agent\") {\n throw new Error(\n `done_by must be 'user' or 'agent', got '${input.done_by}'`,\n );\n }\n if (input.rating != null && (input.rating < 1 || input.rating > 4)) {\n throw new Error(`Rating must be between 1 and 4, got ${input.rating}`);\n }\n\n // Verify the session exists\n const session = db\n .prepare(\"SELECT id FROM sessions WHERE id = ?\")\n .get(input.session_id);\n if (!session) {\n throw new Error(`Session not found: ${input.session_id}`);\n }\n\n const id = ulid();\n const now = new Date().toISOString();\n\n db.prepare(\n `INSERT INTO session_steps (id, session_id, token_id, done_by, rating, notes, created_at)\n VALUES (?, ?, ?, ?, ?, ?, ?)`,\n ).run(\n id,\n input.session_id,\n input.token_id,\n input.done_by,\n input.rating ?? null,\n input.notes ?? null,\n now,\n );\n\n return db\n .prepare(\"SELECT * FROM session_steps WHERE id = ?\")\n .get(id) as SessionStep;\n}\n\n/**\n * Get a full session summary: the session record plus all steps\n * joined with their token details.\n *\n * Ported from the PoC's session-summary command.\n * Throws if the session does not exist.\n */\nexport function getSessionSummary(\n db: Database,\n sessionId: string,\n): SessionSummary {\n const session = db\n .prepare(\"SELECT * FROM sessions WHERE id = ?\")\n .get(sessionId) as Session | undefined;\n\n if (!session) {\n throw new Error(`Session not found: ${sessionId}`);\n }\n\n const steps = db\n .prepare(\n `SELECT ss.*, t.slug, t.concept, t.domain, t.bloom_level\n FROM session_steps ss\n JOIN tokens t ON t.id = ss.token_id\n WHERE ss.session_id = ?\n ORDER BY ss.created_at ASC`,\n )\n .all(sessionId) as StepWithToken[];\n\n return { session, steps };\n}\n","/**\n * User settings — key/value store backed by the user_config table.\n */\n\nimport type { Database } from \"../db/types.js\";\n\nexport interface UserSetting {\n key: string;\n value: string;\n updated_at: string;\n}\n\n/** Get a single setting by key. Returns undefined if not set. */\nexport function getSetting(db: Database, key: string): string | undefined {\n const row = db\n .prepare(\"SELECT value FROM user_config WHERE key = ?\")\n .get(key) as { value: string } | undefined;\n return row?.value;\n}\n\n/** Get all settings as a key-value map. */\nexport function getAllSettings(db: Database): Record<string, string> {\n const rows = db\n .prepare(\"SELECT key, value FROM user_config ORDER BY key\")\n .all() as { key: string; value: string }[];\n const map: Record<string, string> = {};\n for (const row of rows) {\n map[row.key] = row.value;\n }\n return map;\n}\n\n/** Get all settings with metadata. */\nexport function getAllSettingsDetailed(db: Database): UserSetting[] {\n return db\n .prepare(\"SELECT key, value, updated_at FROM user_config ORDER BY key\")\n .all() as UserSetting[];\n}\n\n/** Set a setting (insert or update). */\nexport function setSetting(db: Database, key: string, value: string): void {\n db.prepare(\n `INSERT INTO user_config (key, value, updated_at)\n VALUES (?, ?, datetime('now'))\n ON CONFLICT(key) DO UPDATE SET value = excluded.value, updated_at = excluded.updated_at`,\n ).run(key, value);\n}\n\n/** Delete a setting. Returns true if it existed. */\nexport function deleteSetting(db: Database, key: string): boolean {\n const result = db.prepare(\"DELETE FROM user_config WHERE key = ?\").run(key);\n return result.changes > 0;\n}\n","/**\n * Token repository — typed wrappers around the tokens table.\n *\n * Tokens are atomic knowledge concepts with Bloom taxonomy levels\n * and optional symbiosis modes (shadowing / copilot / autonomy).\n */\n\nimport { ulid } from \"ulid\";\nimport type { Database } from \"../db/types.js\";\n\n// ── Types ────────────────────────────────────────────────────────────────────\n\nexport type BloomLevel = 1 | 2 | 3 | 4 | 5;\n\nexport type SymbiosisMode = \"shadowing\" | \"copilot\" | \"autonomy\";\n\nexport interface Token {\n id: string;\n slug: string;\n concept: string;\n domain: string;\n bloom_level: BloomLevel;\n context: string;\n symbiosis_mode: SymbiosisMode | null;\n source_link: string | null;\n question: string | null;\n created_at: string;\n updated_at: string;\n deprecated_at: string | null;\n}\n\nexport interface CreateTokenInput {\n slug: string;\n concept: string;\n domain?: string;\n bloom_level?: BloomLevel;\n context?: string;\n symbiosis_mode?: SymbiosisMode | null;\n source_link?: string | null;\n question?: string | null;\n}\n\nexport interface UpdateTokenInput {\n concept?: string;\n domain?: string;\n bloom_level?: BloomLevel;\n context?: string;\n symbiosis_mode?: SymbiosisMode | null;\n source_link?: string | null;\n question?: string | null;\n}\n\nexport interface ListTokensOptions {\n domain?: string;\n}\n\nexport interface TokenDeleteImpact {\n cards: number;\n review_logs: number;\n prerequisite_edges_from_token: number;\n prerequisite_edges_to_token: number;\n session_steps: number;\n sessions_touched: number;\n agent_skills: number;\n}\n\nexport interface DeleteTokenResult {\n token: Token;\n impact: TokenDeleteImpact;\n}\n\n// ── Scored result from fuzzy search ──────────────────────────────────────────\n\nexport interface ScoredToken extends Token {\n score: number;\n}\n\n// ── Functions ────────────────────────────────────────────────────────────────\n\n/**\n * Create a new knowledge token.\n * Throws if a token with the same slug already exists.\n */\nexport function createToken(db: Database, input: CreateTokenInput): Token {\n const id = ulid();\n const now = new Date().toISOString();\n\n const bloom = input.bloom_level ?? 1;\n if (bloom < 1 || bloom > 5) {\n throw new Error(`bloom_level must be between 1 and 5, got ${bloom}`);\n }\n\n db.prepare(`\n INSERT INTO tokens (id, slug, concept, domain, bloom_level, context, symbiosis_mode, source_link, question, created_at, updated_at)\n VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)\n `).run(\n id,\n input.slug,\n input.concept,\n input.domain ?? \"\",\n bloom,\n input.context ?? \"\",\n input.symbiosis_mode ?? null,\n input.source_link ?? null,\n input.question ?? null,\n now,\n now,\n );\n\n return getTokenById(db, id)!;\n}\n\n/**\n * Look up a token by its unique slug.\n * Returns undefined if not found.\n */\nexport function getTokenBySlug(db: Database, slug: string): Token | undefined {\n return db.prepare(\"SELECT * FROM tokens WHERE slug = ?\").get(slug) as\n | Token\n | undefined;\n}\n\n/**\n * Look up a token by its ULID.\n * Returns undefined if not found.\n */\nexport function getTokenById(db: Database, id: string): Token | undefined {\n return db.prepare(\"SELECT * FROM tokens WHERE id = ?\").get(id) as\n | Token\n | undefined;\n}\n\n/**\n * Update mutable fields on a token.\n *\n * Slug is intentionally immutable in v1 because it is referenced by other\n * parts of the system (for example agent skill metadata).\n */\nexport function updateToken(\n db: Database,\n slug: string,\n updates: UpdateTokenInput,\n): Token {\n const token = getTokenBySlug(db, slug);\n if (!token) {\n throw new Error(`Token not found: ${slug}`);\n }\n\n const fields: string[] = [];\n const values: unknown[] = [];\n\n if (updates.concept !== undefined) {\n fields.push(\"concept = ?\");\n values.push(updates.concept);\n }\n if (updates.domain !== undefined) {\n fields.push(\"domain = ?\");\n values.push(updates.domain);\n }\n if (updates.bloom_level !== undefined) {\n if (updates.bloom_level < 1 || updates.bloom_level > 5) {\n throw new Error(\n `bloom_level must be between 1 and 5, got ${updates.bloom_level}`,\n );\n }\n fields.push(\"bloom_level = ?\");\n values.push(updates.bloom_level);\n }\n if (updates.context !== undefined) {\n fields.push(\"context = ?\");\n values.push(updates.context);\n }\n if (updates.symbiosis_mode !== undefined) {\n const validModes = [\"shadowing\", \"copilot\", \"autonomy\"];\n if (\n updates.symbiosis_mode !== null &&\n !validModes.includes(updates.symbiosis_mode)\n ) {\n throw new Error(`Invalid symbiosis_mode: ${updates.symbiosis_mode}`);\n }\n fields.push(\"symbiosis_mode = ?\");\n values.push(updates.symbiosis_mode);\n }\n if (updates.source_link !== undefined) {\n fields.push(\"source_link = ?\");\n values.push(updates.source_link);\n }\n if (updates.question !== undefined) {\n fields.push(\"question = ?\");\n values.push(updates.question);\n }\n\n if (fields.length === 0) {\n throw new Error(\"updateToken called with no fields to update\");\n }\n\n fields.push(\"updated_at = ?\");\n values.push(new Date().toISOString());\n values.push(slug);\n\n db.prepare(`UPDATE tokens SET ${fields.join(\", \")} WHERE slug = ?`).run(\n ...values,\n );\n return getTokenBySlug(db, slug)!;\n}\n\n/**\n * Mark a token as deprecated. Deprecated tokens are excluded from review queues\n * and search results but are not deleted — they can still be consulted.\n *\n * Throws if the token does not exist or is already deprecated.\n */\nexport function deprecateToken(db: Database, slug: string): Token {\n const token = getTokenBySlug(db, slug);\n if (!token) {\n throw new Error(`Token not found: ${slug}`);\n }\n if (token.deprecated_at) {\n throw new Error(`Token already deprecated: ${slug}`);\n }\n\n const now = new Date().toISOString();\n db.prepare(\n \"UPDATE tokens SET deprecated_at = ?, updated_at = ? WHERE slug = ?\",\n ).run(now, now, slug);\n\n return getTokenBySlug(db, slug)!;\n}\n\n/**\n * Preview the rows that will be removed or updated when deleting a token.\n */\nexport function getTokenDeleteImpact(\n db: Database,\n slug: string,\n): TokenDeleteImpact {\n const token = getTokenBySlug(db, slug);\n if (!token) {\n throw new Error(`Token not found: ${slug}`);\n }\n\n const cards = db\n .prepare(\"SELECT COUNT(*) AS n FROM cards WHERE token_id = ?\")\n .get(token.id) as { n: number };\n const reviewLogs = db\n .prepare(\"SELECT COUNT(*) AS n FROM review_logs WHERE token_id = ?\")\n .get(token.id) as { n: number };\n const prereqsFrom = db\n .prepare(\"SELECT COUNT(*) AS n FROM prerequisites WHERE token_id = ?\")\n .get(token.id) as { n: number };\n const prereqsTo = db\n .prepare(\"SELECT COUNT(*) AS n FROM prerequisites WHERE requires_id = ?\")\n .get(token.id) as { n: number };\n const sessionSteps = db\n .prepare(\"SELECT COUNT(*) AS n FROM session_steps WHERE token_id = ?\")\n .get(token.id) as { n: number };\n const sessionsTouched = db\n .prepare(\n \"SELECT COUNT(DISTINCT session_id) AS n FROM session_steps WHERE token_id = ?\",\n )\n .get(token.id) as { n: number };\n\n const skillRows = db\n .prepare(\"SELECT token_slugs FROM agent_skills\")\n .all() as Array<{ token_slugs: string }>;\n const agentSkills = skillRows.filter((row) => {\n const tokenSlugs = JSON.parse(row.token_slugs) as string[];\n return tokenSlugs.includes(slug);\n }).length;\n\n return {\n cards: cards.n,\n review_logs: reviewLogs.n,\n prerequisite_edges_from_token: prereqsFrom.n,\n prerequisite_edges_to_token: prereqsTo.n,\n session_steps: sessionSteps.n,\n sessions_touched: sessionsTouched.n,\n agent_skills: agentSkills,\n };\n}\n\n/**\n * Hard-delete a token and clean up non-FK references that point at its slug.\n */\nexport function deleteToken(db: Database, slug: string): DeleteTokenResult {\n const token = getTokenBySlug(db, slug);\n if (!token) {\n throw new Error(`Token not found: ${slug}`);\n }\n\n const impact = getTokenDeleteImpact(db, slug);\n\n db.exec(\"BEGIN\");\n try {\n const now = new Date().toISOString();\n const skillRows = db\n .prepare(\"SELECT id, token_slugs FROM agent_skills\")\n .all() as Array<{ id: string; token_slugs: string }>;\n\n for (const row of skillRows) {\n const tokenSlugs = JSON.parse(row.token_slugs) as string[];\n const filtered = tokenSlugs.filter((tokenSlug) => tokenSlug !== slug);\n if (filtered.length !== tokenSlugs.length) {\n db.prepare(\n \"UPDATE agent_skills SET token_slugs = ?, updated_at = ? WHERE id = ?\",\n ).run(JSON.stringify(filtered), now, row.id);\n }\n }\n\n db.prepare(\"DELETE FROM tokens WHERE id = ?\").run(token.id);\n db.exec(\"COMMIT\");\n } catch (err) {\n db.exec(\"ROLLBACK\");\n throw err;\n }\n\n return { token, impact };\n}\n\n/**\n * Fuzzy search for tokens by keyword query.\n *\n * Uses SQLite LIKE queries on slug, concept, and domain to avoid loading\n * every non-deprecated token into memory. Each search term runs its own\n * LIKE query; results are aggregated in JS with a word-overlap score plus\n * a substring bonus on the concept field. Results are returned sorted by\n * relevance score descending.\n *\n * For very small search terms (< 3 chars) a light in-memory fallback is\n * used to avoid matching every token.\n */\nexport function findTokens(db: Database, query: string): ScoredToken[] {\n const normalised = query.toLowerCase();\n const searchTokens = normalised\n .split(/[\\s,.\\-_/\\\\:;!?()[\\]{}]+/)\n .filter((t) => t.length > 0);\n\n if (searchTokens.length === 0) return [];\n\n // Short terms: fall back to in-memory scan (cheap — few tokens match anyway)\n const shortTerms = searchTokens.filter((t) => t.length <= 2);\n const longTerms = searchTokens.filter((t) => t.length > 2);\n\n const scoreMap = new Map<string, { token: Token; score: number }>();\n\n // Per-term SQL LIKE queries for each substantive search token.\n const likeSQL =\n `SELECT * FROM tokens WHERE deprecated_at IS NULL AND ` +\n `(lower(slug) LIKE ? OR lower(concept) LIKE ? OR lower(domain) LIKE ?)`;\n\n for (const term of longTerms) {\n const pattern = `%${term}%`;\n const rows = db.prepare(likeSQL).all(pattern, pattern, pattern) as Token[];\n for (const row of rows) {\n const entry = scoreMap.get(row.id);\n if (entry) {\n entry.score++;\n } else {\n scoreMap.set(row.id, { token: row, score: 1 });\n }\n }\n }\n\n // If there were short terms, or all terms were short, scan in-memory.\n if (shortTerms.length > 0 || longTerms.length === 0) {\n const allTokens = db\n .prepare(\"SELECT * FROM tokens WHERE deprecated_at IS NULL\")\n .all() as Token[];\n\n for (const token of allTokens) {\n const words = `${token.slug} ${token.concept} ${token.domain}`\n .toLowerCase()\n .split(/[\\s,.\\-_/\\\\:;!?()[\\]{}]+/)\n .filter(Boolean);\n\n let matchCount = 0;\n for (const term of shortTerms.length > 0 ? shortTerms : searchTokens) {\n for (const w of words) {\n if (w === term) matchCount++;\n }\n }\n\n if (matchCount > 0) {\n const entry = scoreMap.get(token.id);\n if (entry) {\n entry.score += matchCount;\n } else {\n scoreMap.set(token.id, { token, score: matchCount });\n }\n }\n }\n }\n\n // Apply substring bonus and build result.\n const scored: ScoredToken[] = [];\n for (const { token, score } of scoreMap.values()) {\n let finalScore = score;\n if (token.concept.toLowerCase().includes(normalised.slice(0, 25))) {\n finalScore += 3;\n }\n scored.push({ score: finalScore, ...token });\n }\n\n scored.sort((a, b) => b.score - a.score);\n return scored;\n}\n\n/**\n * List all tokens, optionally filtered by domain.\n * Results are ordered by bloom_level then slug.\n */\nexport function listTokens(db: Database, options?: ListTokensOptions): Token[] {\n if (options?.domain) {\n return db\n .prepare(\n \"SELECT * FROM tokens WHERE domain = ? AND deprecated_at IS NULL ORDER BY bloom_level, slug\",\n )\n .all(options.domain) as Token[];\n }\n return db\n .prepare(\n \"SELECT * FROM tokens WHERE deprecated_at IS NULL ORDER BY bloom_level, domain, slug\",\n )\n .all() as Token[];\n}\n","/**\n * Monitor log analyzer — maps observed shell commands to token ratings.\n *\n * Pure functions, no DB or filesystem access. Takes parsed command records\n * and a token-to-pattern mapping, returns ratings with evidence.\n */\n\n// ── Types ────────────────────────────────────────────────────────────────────\n\nexport interface MonitorEvent {\n type: \"command_start\" | \"command_end\" | \"monitor_meta\";\n ts: string;\n seq?: number;\n pid?: number;\n command?: string;\n cwd?: string;\n exit_code?: number;\n event?: \"start\" | \"stop\";\n session_id?: string;\n shell?: string;\n}\n\nexport interface CommandRecord {\n seq: number;\n pid: number;\n command: string;\n cwd: string;\n startedAt: string;\n endedAt: string | null;\n durationMs: number | null;\n exitCode: number | null;\n}\n\nexport interface TokenPattern {\n slug: string;\n patterns: string[]; // command prefixes or regex strings\n}\n\nexport interface ObservationRating {\n tokenSlug: string;\n rating: 1 | 2 | 3 | 4 | null;\n confidence: \"high\" | \"medium\" | \"low\";\n evidence: {\n matchedCommands: number;\n helpSeeking: boolean;\n errorCount: number;\n selfCorrections: number;\n medianGapMs: number | null;\n thinkingGapMs: number | null;\n };\n matchedCommandTexts: string[];\n}\n\nexport interface AnalysisResult {\n ratings: ObservationRating[];\n unmatchedCommands: string[];\n timeSpan: { start: string; end: string; durationMs: number } | null;\n}\n\n// ── Parsing ──────────────────────────────────────────────────────────────────\n\n/**\n * Parse a JSONL string into MonitorEvent objects.\n * Skips malformed lines silently.\n */\nexport function parseMonitorLog(jsonl: string): MonitorEvent[] {\n const events: MonitorEvent[] = [];\n for (const line of jsonl.split(\"\\n\")) {\n const trimmed = line.trim();\n if (!trimmed) continue;\n try {\n events.push(JSON.parse(trimmed) as MonitorEvent);\n } catch {\n // skip malformed lines\n }\n }\n return events;\n}\n\n/**\n * Pair command_start and command_end events by (pid, seq) into CommandRecords.\n */\nexport function pairCommands(events: MonitorEvent[]): CommandRecord[] {\n const starts = new Map<string, MonitorEvent>();\n const records: CommandRecord[] = [];\n\n for (const e of events) {\n if (e.type === \"command_start\" && e.seq != null) {\n const key = `${e.pid ?? 0}:${e.seq}`;\n starts.set(key, e);\n } else if (e.type === \"command_end\" && e.seq != null) {\n const key = `${e.pid ?? 0}:${e.seq}`;\n const start = starts.get(key);\n if (start) {\n const startMs = new Date(start.ts).getTime();\n const endMs = new Date(e.ts).getTime();\n records.push({\n seq: e.seq,\n pid: e.pid ?? 0,\n command: start.command ?? \"\",\n cwd: start.cwd ?? \"\",\n startedAt: start.ts,\n endedAt: e.ts,\n durationMs: endMs - startMs,\n exitCode: e.exit_code ?? null,\n });\n starts.delete(key);\n }\n }\n }\n\n // Remaining unpaired starts (monitoring stopped mid-command)\n for (const [, start] of starts) {\n records.push({\n seq: start.seq ?? 0,\n pid: start.pid ?? 0,\n command: start.command ?? \"\",\n cwd: start.cwd ?? \"\",\n startedAt: start.ts,\n endedAt: null,\n durationMs: null,\n exitCode: null,\n });\n }\n\n records.sort(\n (a, b) => new Date(a.startedAt).getTime() - new Date(b.startedAt).getTime(),\n );\n return records;\n}\n\n// ── Analysis ─────────────────────────────────────────────────────────────────\n\nconst HELP_PATTERNS = [\"--help\", \"man \", \"tldr \", \"help \"];\nconst HELP_WINDOW_MS = 60_000;\nconst _RETRY_WINDOW_MS = 30_000;\n\nfunction matchesToken(command: string, patterns: string[]): boolean {\n const lower = command.toLowerCase();\n return patterns.some((p) => lower.includes(p.toLowerCase()));\n}\n\nfunction isHelpCommand(command: string): boolean {\n const lower = command.toLowerCase();\n return HELP_PATTERNS.some((p) => lower.includes(p));\n}\n\nfunction commandPrefix(command: string): string {\n return command.split(/\\s+/).slice(0, 2).join(\" \").toLowerCase();\n}\n\nfunction computeMedian(values: number[]): number | null {\n if (values.length === 0) return null;\n const sorted = [...values].sort((a, b) => a - b);\n const mid = Math.floor(sorted.length / 2);\n return sorted.length % 2 !== 0\n ? sorted[mid]\n : (sorted[mid - 1] + sorted[mid]) / 2;\n}\n\n/**\n * Analyze observed commands against token patterns and produce ratings.\n */\nexport function analyzeObservation(\n commands: CommandRecord[],\n tokenPatterns: TokenPattern[],\n): AnalysisResult {\n const matchedSet = new Set<number>(); // indices of commands matched to any token\n const ratings: ObservationRating[] = [];\n\n for (const tp of tokenPatterns) {\n const matchIndices: number[] = [];\n const matchedTexts: string[] = [];\n\n for (let i = 0; i < commands.length; i++) {\n if (matchesToken(commands[i].command, tp.patterns)) {\n matchIndices.push(i);\n matchedTexts.push(commands[i].command);\n matchedSet.add(i);\n }\n }\n\n if (matchIndices.length === 0) {\n ratings.push({\n tokenSlug: tp.slug,\n rating: null,\n confidence: \"low\",\n evidence: {\n matchedCommands: 0,\n helpSeeking: false,\n errorCount: 0,\n selfCorrections: 0,\n medianGapMs: null,\n thinkingGapMs: null,\n },\n matchedCommandTexts: [],\n });\n continue;\n }\n\n // Help-seeking: any help command within HELP_WINDOW_MS before a matched command\n let helpSeeking = false;\n for (const mi of matchIndices) {\n const matchTime = new Date(commands[mi].startedAt).getTime();\n for (let j = 0; j < commands.length; j++) {\n if (j === mi) continue;\n const cmdTime = new Date(commands[j].startedAt).getTime();\n if (cmdTime >= matchTime - HELP_WINDOW_MS && cmdTime < matchTime) {\n if (isHelpCommand(commands[j].command)) {\n helpSeeking = true;\n break;\n }\n }\n }\n if (helpSeeking) break;\n }\n\n // Error count: matched commands with non-zero exit code\n let errorCount = 0;\n for (const mi of matchIndices) {\n if (commands[mi].exitCode != null && commands[mi].exitCode !== 0) {\n errorCount++;\n }\n }\n\n // Self-corrections: same command prefix run multiple times with different args\n let selfCorrections = 0;\n const prefixGroups = new Map<string, number>();\n for (const mi of matchIndices) {\n const prefix = commandPrefix(commands[mi].command);\n prefixGroups.set(prefix, (prefixGroups.get(prefix) ?? 0) + 1);\n }\n for (const count of prefixGroups.values()) {\n if (count > 1) selfCorrections += count - 1;\n }\n\n // Speed: inter-command gaps between matched commands\n const gaps: number[] = [];\n for (let k = 1; k < matchIndices.length; k++) {\n const prev = commands[matchIndices[k - 1]];\n const curr = commands[matchIndices[k]];\n if (prev.endedAt) {\n const gap =\n new Date(curr.startedAt).getTime() - new Date(prev.endedAt).getTime();\n if (gap >= 0) gaps.push(gap);\n }\n }\n\n // Thinking gap: time before first matched command from previous command's end\n let thinkingGapMs: number | null = null;\n const firstMatchIdx = matchIndices[0];\n if (firstMatchIdx > 0) {\n const prev = commands[firstMatchIdx - 1];\n if (prev.endedAt) {\n thinkingGapMs =\n new Date(commands[firstMatchIdx].startedAt).getTime() -\n new Date(prev.endedAt).getTime();\n }\n }\n\n const medianGapMs = computeMedian(gaps);\n\n // Determine rating\n const rating = inferRating({\n helpSeeking,\n errorCount,\n selfCorrections,\n medianGapMs,\n thinkingGapMs,\n matchedCommands: matchIndices.length,\n });\n\n // Confidence: more matched commands = higher confidence\n const confidence =\n matchIndices.length >= 3\n ? \"high\"\n : matchIndices.length >= 2\n ? \"medium\"\n : \"low\";\n\n ratings.push({\n tokenSlug: tp.slug,\n rating,\n confidence,\n evidence: {\n matchedCommands: matchIndices.length,\n helpSeeking,\n errorCount,\n selfCorrections,\n medianGapMs,\n thinkingGapMs,\n },\n matchedCommandTexts: matchedTexts,\n });\n }\n\n // Unmatched commands\n const unmatchedCommands: string[] = [];\n for (let i = 0; i < commands.length; i++) {\n if (!matchedSet.has(i) && !isHelpCommand(commands[i].command)) {\n unmatchedCommands.push(commands[i].command);\n }\n }\n\n // Time span\n let timeSpan: AnalysisResult[\"timeSpan\"] = null;\n if (commands.length > 0) {\n const first = commands[0];\n const last = commands[commands.length - 1];\n const endTs = last.endedAt ?? last.startedAt;\n timeSpan = {\n start: first.startedAt,\n end: endTs,\n durationMs:\n new Date(endTs).getTime() - new Date(first.startedAt).getTime(),\n };\n }\n\n return { ratings, unmatchedCommands, timeSpan };\n}\n\n// ── Rating Inference ─────────────────────────────────────────────────────────\n\ninterface RatingSignals {\n helpSeeking: boolean;\n errorCount: number;\n selfCorrections: number;\n medianGapMs: number | null;\n thinkingGapMs: number | null;\n matchedCommands: number;\n}\n\nfunction inferRating(signals: RatingSignals): 1 | 2 | 3 | 4 {\n const {\n helpSeeking,\n errorCount,\n selfCorrections,\n medianGapMs,\n thinkingGapMs,\n } = signals;\n\n // Count negative signals\n let negatives = 0;\n if (helpSeeking) negatives += 2;\n if (errorCount >= 3) negatives += 3;\n else if (errorCount >= 1) negatives += 1;\n if (selfCorrections >= 2) negatives += 2;\n else if (selfCorrections >= 1) negatives += 1;\n if (medianGapMs != null && medianGapMs > 30_000) negatives += 2;\n else if (medianGapMs != null && medianGapMs > 10_000) negatives += 1;\n if (thinkingGapMs != null && thinkingGapMs > 30_000) negatives += 1;\n\n if (negatives >= 5) return 1;\n if (negatives >= 3) return 2;\n if (negatives >= 1) return 3;\n return 4;\n}\n","/**\n * Monitor I/O — read/write JSONL files for shell observation.\n *\n * Monitor logs live at ~/.zam/monitor/<session-id>.jsonl.\n * Separated from analyzer.ts so the analyzer remains pure-function testable.\n */\n\nimport {\n appendFileSync,\n existsSync,\n mkdirSync,\n readFileSync,\n statSync,\n} from \"node:fs\";\nimport { homedir } from \"node:os\";\nimport { join } from \"node:path\";\nimport type { MonitorEvent } from \"./analyzer.js\";\nimport { parseMonitorLog } from \"./analyzer.js\";\n\nconst MONITOR_DIR = join(homedir(), \".zam\", \"monitor\");\n\n/** Get the monitor directory path. */\nexport function getMonitorDir(): string {\n return MONITOR_DIR;\n}\n\n/** Get the JSONL file path for a session. */\nexport function getMonitorPath(sessionId: string): string {\n return join(MONITOR_DIR, `${sessionId}.jsonl`);\n}\n\n/** Ensure the monitor directory exists (mode 0700 for privacy). */\nexport function ensureMonitorDir(): void {\n if (!existsSync(MONITOR_DIR)) {\n mkdirSync(MONITOR_DIR, { recursive: true, mode: 0o700 });\n }\n}\n\n/** Append a single event to the session's JSONL file. */\nexport function writeMonitorEvent(\n sessionId: string,\n event: MonitorEvent,\n): void {\n ensureMonitorDir();\n const path = getMonitorPath(sessionId);\n appendFileSync(path, `${JSON.stringify(event)}\\n`);\n}\n\n/** Read and parse all events from a session's monitor log. */\nexport function readMonitorLog(sessionId: string): MonitorEvent[] {\n const path = getMonitorPath(sessionId);\n if (!existsSync(path)) {\n return [];\n }\n const content = readFileSync(path, \"utf-8\");\n return parseMonitorLog(content);\n}\n\n/** Check if a monitor log exists for a session. */\nexport function monitorLogExists(sessionId: string): boolean {\n return existsSync(getMonitorPath(sessionId));\n}\n\n/** Get basic stats about a monitor log without full parsing. */\nexport function getMonitorLogStats(sessionId: string): {\n exists: boolean;\n sizeBytes: number;\n lineCount: number;\n} {\n const path = getMonitorPath(sessionId);\n if (!existsSync(path)) {\n return { exists: false, sizeBytes: 0, lineCount: 0 };\n }\n const stat = statSync(path);\n const content = readFileSync(path, \"utf-8\");\n const lineCount = content.split(\"\\n\").filter((l) => l.trim()).length;\n return { exists: true, sizeBytes: stat.size, lineCount };\n}\n","/**\n * Shell hook code generation for zsh, bash, and PowerShell.\n *\n * Pure functions that return shell code strings. The CLI command\n * `zam monitor start/stop` calls these and prints to stdout.\n */\n\nfunction psSingleQuoted(value: string): string {\n return `'${value.replace(/'/g, \"''\")}'`;\n}\n\n/**\n * Generate zsh hooks that capture commands to a JSONL file.\n * Uses $EPOCHREALTIME for sub-second timestamp precision.\n */\nexport function generateZshHooks(\n monitorFile: string,\n sessionId: string,\n): string {\n return `\n# ZAM monitor hooks for session ${sessionId}\nexport __ZAM_MONITOR_FILE=\"${monitorFile}\"\nexport __ZAM_MONITOR_SEQ=0\nexport __ZAM_MONITOR_SESSION=\"${sessionId}\"\n\n__zam_ts() {\n if [[ -n \"\\${EPOCHREALTIME:-}\" ]]; then\n local sec=\"\\${EPOCHREALTIME%%.*}\"\n local frac=\"\\${EPOCHREALTIME##*.}\"\n frac=\"\\${frac:0:3}\"\n printf '%s.%sZ' \"$(date -u -r \"$sec\" '+%Y-%m-%dT%H:%M:%S' 2>/dev/null || date -u '+%Y-%m-%dT%H:%M:%S')\" \"$frac\"\n else\n date -u '+%Y-%m-%dT%H:%M:%SZ'\n fi\n}\n\n__zam_preexec() {\n (( __ZAM_MONITOR_SEQ++ ))\n local cmd=\"\\${1//\\\\\"/\\\\\\\\\\\\\"}\"\n local cwd=\"\\${PWD//\\\\\"/\\\\\\\\\\\\\"}\"\n local ts=\"$(__zam_ts)\"\n printf '{\"type\":\"command_start\",\"ts\":\"%s\",\"command\":\"%s\",\"cwd\":\"%s\",\"seq\":%d,\"pid\":%d}\\\\n' \\\\\n \"$ts\" \"$cmd\" \"$cwd\" \"$__ZAM_MONITOR_SEQ\" \"$$\" \\\\\n >> \"$__ZAM_MONITOR_FILE\"\n}\n\n__zam_precmd() {\n local exit_code=$?\n [[ $__ZAM_MONITOR_SEQ -eq 0 ]] && return\n local ts=\"$(__zam_ts)\"\n printf '{\"type\":\"command_end\",\"ts\":\"%s\",\"exit_code\":%d,\"seq\":%d,\"pid\":%d}\\\\n' \\\\\n \"$ts\" \"$exit_code\" \"$__ZAM_MONITOR_SEQ\" \"$$\" \\\\\n >> \"$__ZAM_MONITOR_FILE\"\n}\n\nautoload -Uz add-zsh-hook\nadd-zsh-hook preexec __zam_preexec\nadd-zsh-hook precmd __zam_precmd\n\necho \"ZAM monitor active for session $__ZAM_MONITOR_SESSION\"\n`.trim();\n}\n\n/**\n * Generate bash hooks that capture commands to a JSONL file.\n * Uses DEBUG trap for preexec, PROMPT_COMMAND for precmd.\n */\nexport function generateBashHooks(\n monitorFile: string,\n sessionId: string,\n): string {\n return `\n# ZAM monitor hooks for session ${sessionId}\nexport __ZAM_MONITOR_FILE=\"${monitorFile}\"\nexport __ZAM_MONITOR_SEQ=0\nexport __ZAM_MONITOR_SESSION=\"${sessionId}\"\nexport __ZAM_MONITOR_CMD_ACTIVE=0\n\n__zam_ts() {\n date -u '+%Y-%m-%dT%H:%M:%SZ'\n}\n\n__zam_debug_trap() {\n [[ \"$__ZAM_MONITOR_CMD_ACTIVE\" -eq 1 ]] && return\n __ZAM_MONITOR_CMD_ACTIVE=1\n (( __ZAM_MONITOR_SEQ++ ))\n local cmd=\"\\${BASH_COMMAND//\\\\\"/\\\\\\\\\\\\\"}\"\n local cwd=\"\\${PWD//\\\\\"/\\\\\\\\\\\\\"}\"\n local ts=\"$(__zam_ts)\"\n printf '{\"type\":\"command_start\",\"ts\":\"%s\",\"command\":\"%s\",\"cwd\":\"%s\",\"seq\":%d,\"pid\":%d}\\\\n' \\\\\n \"$ts\" \"$cmd\" \"$cwd\" \"$__ZAM_MONITOR_SEQ\" \"$$\" \\\\\n >> \"$__ZAM_MONITOR_FILE\"\n}\n\n__zam_prompt_cmd() {\n local exit_code=$?\n if [[ \"$__ZAM_MONITOR_CMD_ACTIVE\" -eq 1 ]]; then\n __ZAM_MONITOR_CMD_ACTIVE=0\n local ts=\"$(__zam_ts)\"\n printf '{\"type\":\"command_end\",\"ts\":\"%s\",\"exit_code\":%d,\"seq\":%d,\"pid\":%d}\\\\n' \\\\\n \"$ts\" \"$exit_code\" \"$__ZAM_MONITOR_SEQ\" \"$$\" \\\\\n >> \"$__ZAM_MONITOR_FILE\"\n fi\n}\n\ntrap '__zam_debug_trap' DEBUG\nPROMPT_COMMAND=\"__zam_prompt_cmd;\\${PROMPT_COMMAND:-}\"\n\necho \"ZAM monitor active for session $__ZAM_MONITOR_SESSION\"\n`.trim();\n}\n\n/**\n * Generate PowerShell hooks that capture completed commands to a JSONL file.\n * PowerShell has no zsh-style preexec hook, so this records the most recent\n * history item from the prompt function after each command completes.\n */\nexport function generatePowerShellHooks(\n monitorFile: string,\n sessionId: string,\n): string {\n return `\n# ZAM monitor hooks for session ${sessionId}\n$global:__ZAM_MONITOR_FILE = ${psSingleQuoted(monitorFile)}\n$global:__ZAM_MONITOR_SEQ = 0\n$global:__ZAM_MONITOR_SESSION = ${psSingleQuoted(sessionId)}\n$global:__ZAM_MONITOR_SKIP_NEXT_PROMPT = $true\n\nfunction global:__zam_write_monitor_event {\n param([hashtable]$Event)\n $json = $Event | ConvertTo-Json -Compress -Depth 4\n $utf8NoBom = New-Object System.Text.UTF8Encoding $false\n [System.IO.File]::AppendAllText($global:__ZAM_MONITOR_FILE, $json + [Environment]::NewLine, $utf8NoBom)\n}\n\nfunction global:__zam_iso_utc {\n param([datetime]$Date)\n if ($Date -eq [datetime]::MinValue) {\n return (Get-Date).ToUniversalTime().ToString(\"o\")\n }\n return $Date.ToUniversalTime().ToString(\"o\")\n}\n\nfunction global:__zam_update_last_history_id {\n $history = Get-History -Count 1\n if ($null -ne $history) {\n $global:__ZAM_MONITOR_LAST_HISTORY_ID = $history.Id\n } elseif ($null -eq $global:__ZAM_MONITOR_LAST_HISTORY_ID) {\n $global:__ZAM_MONITOR_LAST_HISTORY_ID = 0\n }\n}\n\nfunction global:__zam_record_last_history {\n param(\n [bool]$Success,\n [object]$NativeExitCode\n )\n\n $history = Get-History -Count 1\n if ($null -eq $history) { return }\n if ($history.Id -le $global:__ZAM_MONITOR_LAST_HISTORY_ID) { return }\n\n $global:__ZAM_MONITOR_LAST_HISTORY_ID = $history.Id\n $global:__ZAM_MONITOR_SEQ += 1\n\n $exitCode = 0\n if (-not $Success) {\n if ($NativeExitCode -is [int] -and $NativeExitCode -ne 0) {\n $exitCode = $NativeExitCode\n } else {\n $exitCode = 1\n }\n }\n\n $cwd = (Get-Location).Path\n __zam_write_monitor_event @{\n type = \"command_start\"\n ts = (__zam_iso_utc $history.StartExecutionTime)\n command = $history.CommandLine\n cwd = $cwd\n seq = $global:__ZAM_MONITOR_SEQ\n pid = $PID\n }\n __zam_write_monitor_event @{\n type = \"command_end\"\n ts = (__zam_iso_utc $history.EndExecutionTime)\n exit_code = $exitCode\n seq = $global:__ZAM_MONITOR_SEQ\n pid = $PID\n }\n}\n\nif (-not (Test-Path function:\\\\__zam_previous_prompt) -and (Test-Path function:\\\\prompt)) {\n Set-Item -Path function:\\\\__zam_previous_prompt -Value (Get-Item function:\\\\prompt).ScriptBlock\n}\n__zam_update_last_history_id\n\nfunction global:prompt {\n $zamSuccess = $?\n $zamNativeExitCode = $global:LASTEXITCODE\n\n if ($global:__ZAM_MONITOR_SKIP_NEXT_PROMPT) {\n __zam_update_last_history_id\n $global:__ZAM_MONITOR_SKIP_NEXT_PROMPT = $false\n } else {\n __zam_record_last_history -Success $zamSuccess -NativeExitCode $zamNativeExitCode\n }\n\n if (Test-Path function:\\\\__zam_previous_prompt) {\n & (Get-Item function:\\\\__zam_previous_prompt).ScriptBlock\n } else {\n \"PS $($executionContext.SessionState.Path.CurrentLocation)$('>' * ($nestedPromptLevel + 1)) \"\n }\n}\n\nWrite-Host \"ZAM monitor active for session $global:__ZAM_MONITOR_SESSION\"\n`.trim();\n}\n\n/** Generate zsh code to remove monitor hooks. */\nexport function generateZshUnhooks(): string {\n return `\n# Remove ZAM monitor hooks\nadd-zsh-hook -d preexec __zam_preexec 2>/dev/null\nadd-zsh-hook -d precmd __zam_precmd 2>/dev/null\nunset -f __zam_preexec __zam_precmd __zam_ts 2>/dev/null\nunset __ZAM_MONITOR_FILE __ZAM_MONITOR_SEQ __ZAM_MONITOR_SESSION 2>/dev/null\necho \"ZAM monitor stopped.\"\n`.trim();\n}\n\n/** Generate bash code to remove monitor hooks. */\nexport function generateBashUnhooks(): string {\n return `\n# Remove ZAM monitor hooks\ntrap - DEBUG\nPROMPT_COMMAND=\"\\${PROMPT_COMMAND/__zam_prompt_cmd;/}\"\nunset -f __zam_debug_trap __zam_prompt_cmd __zam_ts 2>/dev/null\nunset __ZAM_MONITOR_FILE __ZAM_MONITOR_SEQ __ZAM_MONITOR_SESSION __ZAM_MONITOR_CMD_ACTIVE 2>/dev/null\necho \"ZAM monitor stopped.\"\n`.trim();\n}\n\n/** Generate PowerShell code to remove monitor hooks. */\nexport function generatePowerShellUnhooks(): string {\n return `\n# Remove ZAM monitor hooks\nif (Test-Path function:\\\\__zam_previous_prompt) {\n Set-Item -Path function:\\\\prompt -Value (Get-Item function:\\\\__zam_previous_prompt).ScriptBlock\n Remove-Item function:\\\\__zam_previous_prompt -Force -ErrorAction SilentlyContinue\n}\nRemove-Item function:\\\\__zam_write_monitor_event,function:\\\\__zam_iso_utc,function:\\\\__zam_update_last_history_id,function:\\\\__zam_record_last_history -ErrorAction SilentlyContinue\nRemove-Variable -Name __ZAM_MONITOR_FILE,__ZAM_MONITOR_SEQ,__ZAM_MONITOR_SESSION,__ZAM_MONITOR_LAST_HISTORY_ID,__ZAM_MONITOR_SKIP_NEXT_PROMPT -Scope Global -ErrorAction SilentlyContinue\nWrite-Host \"ZAM monitor stopped.\"\n`.trim();\n}\n","/**\n * Skill Discovery — identifies recurring non-standard command patterns\n * across multiple sessions and proposes them as minimal reusable skills.\n *\n * The key insight from Increment 2: \"The human's demonstrated competence\n * is the gate for automation — not the other way around.\" A pattern must\n * appear consistently across sessions before being proposed as a skill.\n *\n * Pure functions — no DB access. Callers provide command records and\n * existing skills; this module returns proposed skills.\n */\n\nimport type { CommandRecord } from \"./analyzer.js\";\n\n// ── Types ────────────────────────────────────────────────────────────────────\n\nexport interface CommandSequence {\n /** The ordered command prefixes forming the pattern (e.g., [\"git checkout\", \"npm install\", \"npm run build\"]) */\n steps: string[];\n /** How many sessions contained this sequence */\n sessionCount: number;\n /** Total occurrences across all sessions */\n totalOccurrences: number;\n /** Example full commands from the most recent occurrence */\n examples: string[];\n}\n\nexport interface SkillProposal {\n /** Suggested slug for the skill */\n slug: string;\n /** Human-readable description of what the pattern does */\n description: string;\n /** The command steps forming the skill */\n steps: string[];\n /** How many sessions demonstrated this pattern */\n sessionCount: number;\n /** Confidence that this is a real, repeatable skill */\n confidence: \"high\" | \"medium\" | \"low\";\n /** Example commands from actual usage */\n examples: string[];\n}\n\nexport interface DiscoveryOptions {\n /** Minimum number of sessions a pattern must appear in (default: 2) */\n minSessions?: number;\n /** Minimum sequence length to consider (default: 2) */\n minSequenceLength?: number;\n /** Maximum sequence length to consider (default: 5) */\n maxSequenceLength?: number;\n /** Existing skill slugs to exclude from proposals */\n existingSkillSlugs?: string[];\n}\n\n// ── Discovery ────────────────────────────────────────────────────────────────\n\n/**\n * Discover recurring command patterns across multiple sessions.\n *\n * Takes a map of session ID → command records, finds command sequences\n * that appear in multiple sessions, and proposes them as skills.\n *\n * @param sessionCommands - Map of session ID to that session's commands\n * @param options - Discovery configuration\n * @returns Array of skill proposals, sorted by confidence then session count\n */\nexport function discoverSkills(\n sessionCommands: Map<string, CommandRecord[]>,\n options: DiscoveryOptions = {},\n): SkillProposal[] {\n const minSessions = options.minSessions ?? 2;\n const minLen = options.minSequenceLength ?? 2;\n const maxLen = options.maxSequenceLength ?? 5;\n const existing = new Set(options.existingSkillSlugs ?? []);\n\n // Step 1: Extract normalized command sequences from each session\n const sessionSequences = new Map<string, string[][]>();\n for (const [sessionId, commands] of sessionCommands) {\n const sequences = extractSequences(commands, minLen, maxLen);\n if (sequences.length > 0) {\n sessionSequences.set(sessionId, sequences);\n }\n }\n\n // Step 2: Count how many sessions contain each unique sequence\n const sequenceIndex = new Map<string, CommandSequence>();\n\n for (const [, sequences] of sessionSequences) {\n // Deduplicate within a session — count each sequence once per session\n const seen = new Set<string>();\n for (const seq of sequences) {\n const key = seq.join(\" → \");\n if (seen.has(key)) continue;\n seen.add(key);\n\n const entry = sequenceIndex.get(key);\n if (entry) {\n entry.sessionCount++;\n entry.totalOccurrences++;\n } else {\n sequenceIndex.set(key, {\n steps: seq,\n sessionCount: 1,\n totalOccurrences: 1,\n examples: [],\n });\n }\n }\n }\n\n // Step 3: Collect examples from the most recent session for qualifying sequences\n const lastSessionId = [...sessionCommands.keys()].pop();\n if (lastSessionId) {\n const lastCommands = sessionCommands.get(lastSessionId)!;\n for (const [_key, entry] of sequenceIndex) {\n if (entry.sessionCount >= minSessions) {\n entry.examples = findExamplesForSequence(lastCommands, entry.steps);\n }\n }\n }\n\n // Step 4: Filter to patterns that appear in enough sessions\n const candidates = [...sequenceIndex.values()].filter(\n (s) => s.sessionCount >= minSessions,\n );\n\n // Step 5: Remove subsequences of longer patterns (prefer maximal patterns)\n const pruned = removeSubsequences(candidates);\n\n // Step 6: Convert to skill proposals\n const proposals: SkillProposal[] = [];\n for (const seq of pruned) {\n const slug = generateSlug(seq.steps);\n\n // Skip if this skill already exists\n if (existing.has(slug)) continue;\n\n proposals.push({\n slug,\n description: describeSequence(seq.steps),\n steps: seq.steps,\n sessionCount: seq.sessionCount,\n confidence:\n seq.sessionCount >= 4\n ? \"high\"\n : seq.sessionCount >= 3\n ? \"medium\"\n : \"low\",\n examples: seq.examples,\n });\n }\n\n // Sort: high confidence first, then by session count\n const confidenceOrder = { high: 0, medium: 1, low: 2 };\n proposals.sort((a, b) => {\n const confDiff =\n confidenceOrder[a.confidence] - confidenceOrder[b.confidence];\n if (confDiff !== 0) return confDiff;\n return b.sessionCount - a.sessionCount;\n });\n\n return proposals;\n}\n\n// ── Helpers ──────────────────────────────────────────────────────────────────\n\n/**\n * Normalize a command to its tool + subcommand prefix.\n * \"git checkout -b feat/foo\" → \"git checkout\"\n * \"npm run build\" → \"npm run build\"\n * \"docker compose up -d\" → \"docker compose up\"\n */\nfunction normalizeCommand(command: string): string {\n const parts = command.trim().split(/\\s+/);\n\n // Known multi-word tools\n const multiWord = [\"docker compose\", \"npm run\", \"npx\", \"git\"];\n const lower = command.toLowerCase();\n\n for (const mw of multiWord) {\n if (lower.startsWith(mw) && parts.length >= mw.split(\" \").length + 1) {\n return parts\n .slice(0, mw.split(\" \").length + 1)\n .join(\" \")\n .toLowerCase();\n }\n }\n\n // Default: first two words\n return parts.slice(0, Math.min(2, parts.length)).join(\" \").toLowerCase();\n}\n\n/**\n * Extract all command subsequences of length minLen..maxLen from a session.\n * Uses normalized command prefixes for comparison.\n */\nfunction extractSequences(\n commands: CommandRecord[],\n minLen: number,\n maxLen: number,\n): string[][] {\n // Filter out trivial commands\n const filtered = commands.filter((c) => {\n const lower = c.command.toLowerCase().trim();\n return (\n lower.length > 0 &&\n !lower.startsWith(\"cd \") &&\n lower !== \"cd\" &&\n lower !== \"ls\" &&\n lower !== \"pwd\" &&\n lower !== \"clear\" &&\n lower !== \"exit\" &&\n !lower.startsWith(\"echo \")\n );\n });\n\n const normalized = filtered.map((c) => normalizeCommand(c.command));\n const sequences: string[][] = [];\n\n for (let len = minLen; len <= maxLen; len++) {\n for (let i = 0; i <= normalized.length - len; i++) {\n const seq = normalized.slice(i, i + len);\n // Only include if at least 2 distinct commands (not just \"git commit\" repeated)\n if (new Set(seq).size >= 2) {\n sequences.push(seq);\n }\n }\n }\n\n return sequences;\n}\n\n/**\n * Find example full commands for a given normalized sequence in a command list.\n */\nfunction findExamplesForSequence(\n commands: CommandRecord[],\n steps: string[],\n): string[] {\n const normalized = commands.map((c) => ({\n norm: normalizeCommand(c.command),\n full: c.command,\n }));\n\n for (let i = 0; i <= normalized.length - steps.length; i++) {\n let match = true;\n for (let j = 0; j < steps.length; j++) {\n if (normalized[i + j].norm !== steps[j]) {\n match = false;\n break;\n }\n }\n if (match) {\n return normalized.slice(i, i + steps.length).map((n) => n.full);\n }\n }\n\n return [];\n}\n\n/**\n * Remove sequences that are strict subsequences of longer qualifying sequences.\n */\nfunction removeSubsequences(candidates: CommandSequence[]): CommandSequence[] {\n // Sort by length descending so we check long sequences first\n const sorted = [...candidates].sort(\n (a, b) => b.steps.length - a.steps.length,\n );\n const result: CommandSequence[] = [];\n\n for (const candidate of sorted) {\n const key = candidate.steps.join(\" → \");\n const isSubsequence = result.some((longer) => {\n const longerKey = longer.steps.join(\" → \");\n return longerKey.includes(key) && longerKey !== key;\n });\n\n if (!isSubsequence) {\n result.push(candidate);\n }\n }\n\n return result;\n}\n\n/**\n * Generate a slug from command steps.\n * [\"git checkout\", \"npm install\", \"npm run build\"] → \"checkout-install-build\"\n */\nfunction generateSlug(steps: string[]): string {\n return steps\n .map((s) => {\n const parts = s.split(/\\s+/);\n return parts[parts.length - 1]; // last word of each step\n })\n .join(\"-\");\n}\n\n/**\n * Generate a human-readable description of what a command sequence does.\n */\nfunction describeSequence(steps: string[]): string {\n return `Recurring pattern: ${steps.join(\" → \")}`;\n}\n","/**\n * Cascade Block & Unblock — prerequisite-aware blocking logic.\n *\n * Ported from the PoC's cascade-block and unblock-ready commands.\n *\n * When a user rates a token as \"forgot\" (rating 1) and that token has\n * prerequisites, we block the token and surface its prerequisites into\n * the active deck. When all prerequisites are met, we unblock.\n */\n\nimport type { Database } from \"../db/types.js\";\nimport { ensureCard } from \"../models/card.js\";\nimport { getPrerequisites } from \"../models/prerequisite.js\";\nimport { getTokenBySlug } from \"../models/token.js\";\n\n// ── Types ────────────────────────────────────────────────────────────────────\n\nexport interface CascadeBlockResult {\n blockedSlug: string;\n prerequisites: Array<{ slug: string; concept: string; bloomLevel: number }>;\n}\n\nexport interface UnblockResult {\n unblocked: Array<{ slug: string; concept: string }>;\n}\n\n// ── Functions ────────────────────────────────────────────────────────────────\n\n/**\n * Block a token and surface its prerequisites.\n *\n * Called when a user rates a token as \"forgot\" (rating 1). The token is\n * marked as blocked so it won't appear in review queues. All direct\n * prerequisites are ensured to have cards (unblocked, due now) so they\n * appear in the user's next review session.\n *\n * @param db - Database connection\n * @param userId - The user whose card to block\n * @param tokenSlug - Slug of the token the user forgot\n * @returns Info about what was blocked and which prerequisites were surfaced\n */\nexport function cascadeBlock(\n db: Database,\n userId: string,\n tokenSlug: string,\n): CascadeBlockResult {\n const token = getTokenBySlug(db, tokenSlug);\n if (!token) {\n throw new Error(`Unknown token slug: ${tokenSlug}`);\n }\n\n // Ensure a card exists, then block it\n ensureCard(db, token.id, userId);\n db.prepare(\n \"UPDATE cards SET blocked = 1 WHERE token_id = ? AND user_id = ?\",\n ).run(token.id, userId);\n\n // Surface all direct prerequisites — ensure cards exist (unblocked, due now)\n const prereqs = getPrerequisites(db, token.id);\n const surfaced: Array<{ slug: string; concept: string; bloomLevel: number }> =\n [];\n\n for (const prereq of prereqs) {\n // ensureCard creates a new card if missing (defaults: blocked=0, due_at=now)\n const card = ensureCard(db, prereq.requires_id, userId);\n\n // If the prerequisite card was somehow blocked with no prereqs of its own,\n // make sure it's unblocked and due now so it surfaces\n if (card.blocked === 1) {\n const prereqOfPrereq = db\n .prepare(\"SELECT COUNT(*) as n FROM prerequisites WHERE token_id = ?\")\n .get(prereq.requires_id) as { n: number };\n\n // Only force-unblock if it has no prerequisites of its own\n if (prereqOfPrereq.n === 0) {\n const now = new Date().toISOString();\n db.prepare(\n \"UPDATE cards SET blocked = 0, due_at = ? WHERE token_id = ? AND user_id = ?\",\n ).run(now, prereq.requires_id, userId);\n }\n }\n\n surfaced.push({\n slug: prereq.slug,\n concept: prereq.concept,\n bloomLevel: prereq.bloom_level,\n });\n }\n\n return {\n blockedSlug: tokenSlug,\n prerequisites: surfaced,\n };\n}\n\n/**\n * Scan all blocked cards for a user and unblock any whose prerequisites are met.\n *\n * A blocked card is ready to unblock when ALL of its direct prerequisites have:\n * - reps >= 1 (the user has successfully recalled it at least once)\n * - blocked = 0 (the prerequisite itself is not blocked)\n *\n * If a blocked card has no prerequisites at all, it is unblocked immediately\n * (it was likely blocked in error or its prerequisites were removed).\n *\n * @param db - Database connection\n * @param userId - The user whose blocked cards to check\n * @returns List of cards that were unblocked\n */\nexport function unblockReady(db: Database, userId: string): UnblockResult {\n const blockedCards = db\n .prepare(\n `SELECT c.token_id, t.slug, t.concept\n FROM cards c\n JOIN tokens t ON t.id = c.token_id\n WHERE c.user_id = ? AND c.blocked = 1`,\n )\n .all(userId) as Array<{ token_id: string; slug: string; concept: string }>;\n\n const unblocked: Array<{ slug: string; concept: string }> = [];\n\n for (const card of blockedCards) {\n const totalPrereqs = db\n .prepare(\"SELECT COUNT(*) as n FROM prerequisites WHERE token_id = ?\")\n .get(card.token_id) as { n: number };\n\n const metPrereqs = db\n .prepare(\n `SELECT COUNT(*) as n FROM cards c\n JOIN prerequisites p ON p.requires_id = c.token_id\n WHERE p.token_id = ? AND c.user_id = ? AND c.reps >= 1 AND c.blocked = 0`,\n )\n .get(card.token_id, userId) as { n: number };\n\n if (totalPrereqs.n === 0 || metPrereqs.n === totalPrereqs.n) {\n const now = new Date().toISOString();\n db.prepare(\n \"UPDATE cards SET blocked = 0, due_at = ? WHERE token_id = ? AND user_id = ?\",\n ).run(now, card.token_id, userId);\n\n unblocked.push({ slug: card.slug, concept: card.concept });\n }\n }\n\n return { unblocked };\n}\n","/**\n * Rating Evaluator\n *\n * Processes a user's self-assessment rating after a recall attempt.\n * Coordinates between FSRS scheduling, review logging, and blocking.\n */\n\nimport { ulid } from \"ulid\";\nimport type { Database } from \"../db/types.js\";\nimport { updateCard } from \"../models/card.js\";\nimport type { Rating, SchedulingCard } from \"../scheduler/fsrs.js\";\nimport { createFSRS } from \"../scheduler/fsrs.js\";\n\nexport interface EvaluateInput {\n cardId: string;\n tokenId: string;\n userId: string;\n rating: Rating;\n sessionId?: string;\n responseTimeMs?: number;\n}\n\nexport interface EvaluateResult {\n nextDueAt: string;\n stability: number;\n difficulty: number;\n state: string;\n scheduledDays: number;\n reps: number;\n lapses: number;\n}\n\n/**\n * Process a rating: update the card via FSRS, log the review.\n * Returns the updated scheduling state.\n *\n * Note: blocking logic (cascade-block) is handled separately by the caller\n * when rating === 1 and the token has prerequisites.\n */\nexport function evaluateRating(\n db: Database,\n input: EvaluateInput,\n): EvaluateResult {\n // Get current card state\n const card = db\n .prepare(\"SELECT * FROM cards WHERE id = ?\")\n .get(input.cardId) as\n | {\n stability: number;\n difficulty: number;\n elapsed_days: number;\n scheduled_days: number;\n reps: number;\n lapses: number;\n state: string;\n due_at: string;\n last_review_at: string | null;\n }\n | undefined;\n\n if (!card) {\n throw new Error(`Card not found: ${input.cardId}`);\n }\n\n const now = new Date();\n const fsrs = createFSRS();\n\n // Build scheduling card from DB state\n const schedulingCard: SchedulingCard = {\n stability: card.stability,\n difficulty: card.difficulty,\n elapsedDays: card.elapsed_days,\n scheduledDays: card.scheduled_days,\n reps: card.reps,\n lapses: card.lapses,\n state: card.state as SchedulingCard[\"state\"],\n dueAt: new Date(card.due_at),\n lastReviewAt: card.last_review_at ? new Date(card.last_review_at) : null,\n };\n\n // Run FSRS\n const updated = fsrs.schedule(schedulingCard, input.rating, now);\n\n // Update the card in the DB\n updateCard(db, input.cardId, {\n stability: updated.stability,\n difficulty: updated.difficulty,\n elapsed_days: updated.elapsedDays,\n scheduled_days: updated.scheduledDays,\n reps: updated.reps,\n lapses: updated.lapses,\n state: updated.state,\n due_at: updated.dueAt.toISOString(),\n last_review_at: now.toISOString(),\n });\n\n // Log the review (immutable)\n db.prepare(\n `INSERT INTO review_logs (id, card_id, token_id, user_id, rating, response_time_ms, reviewed_at, scheduled_at, session_id)\n VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?)`,\n ).run(\n ulid(),\n input.cardId,\n input.tokenId,\n input.userId,\n input.rating,\n input.responseTimeMs ?? null,\n now.toISOString(),\n card.due_at,\n input.sessionId ?? null,\n );\n\n return {\n nextDueAt: updated.dueAt.toISOString(),\n stability: updated.stability,\n difficulty: updated.difficulty,\n state: updated.state,\n scheduledDays: updated.scheduledDays,\n reps: updated.reps,\n lapses: updated.lapses,\n };\n}\n","/**\n * FSRS-5 — Free Spaced Repetition Scheduler (v5)\n *\n * Pure-function implementation of the FSRS algorithm that replaces\n * the PoC's SM-2 scheduler. This is the mathematical heart of ZAM's\n * spaced-repetition engine.\n *\n * Reference: https://github.com/open-spaced-repetition/fsrs4anki/wiki/The-Algorithm\n */\n\n// ── Rating scale ────────────────────────────────────────────────────────────\n\n/** 1 = Again (forgot), 2 = Hard, 3 = Good, 4 = Easy */\nexport type Rating = 1 | 2 | 3 | 4;\n\n// ── Card states ─────────────────────────────────────────────────────────────\n\nexport type CardState = \"new\" | \"learning\" | \"review\" | \"relearning\";\n\n// ── Scheduling card ─────────────────────────────────────────────────────────\n\nexport interface SchedulingCard {\n /** Memory stability in days — expected half-life of recall probability. */\n stability: number;\n /** Intrinsic difficulty on a 1–10 scale. */\n difficulty: number;\n /** Days elapsed since the last review. */\n elapsedDays: number;\n /** Currently scheduled interval in days. */\n scheduledDays: number;\n /** Count of successful consecutive reviews. */\n reps: number;\n /** Times the card was forgotten (rated Again). */\n lapses: number;\n /** Current learning state. */\n state: CardState;\n /** When the card is next due. */\n dueAt: Date;\n /** When the card was last reviewed (null for new cards). */\n lastReviewAt: Date | null;\n}\n\n// ── Parameters ──────────────────────────────────────────────────────────────\n\nexport interface FSRSParameters {\n /** 19 optimised weight parameters (w0 – w18). */\n w: number[];\n /** Target retention rate, e.g. 0.9 means we aim for 90% recall. */\n requestRetention: number;\n}\n\n// ── FSRS-5 default weights ──────────────────────────────────────────────────\n\nconst DEFAULT_W: number[] = [\n 0.4072,\n 1.1829,\n 3.1262,\n 15.4722, // w0–w3: initial stability per rating\n 7.2102,\n 0.5316,\n 1.0651, // w4–w6: difficulty\n 0.0092,\n 1.5988,\n 0.1176,\n 1.0014, // w7–w10: stability after forgetting\n 2.0032,\n 0.0266,\n 0.3077,\n 0.15, // w11–w14: stability increase\n 0.0,\n 2.7849,\n 0.3477,\n 0.6831, // w15–w18: additional parameters\n];\n\nconst DEFAULT_REQUEST_RETENTION = 0.9;\n\n// ── FSRS object returned by the factory ─────────────────────────────────────\n\nexport interface FSRS {\n /** Return a fully updated card after applying a rating. Pure function. */\n schedule(card: SchedulingCard, rating: Rating, now?: Date): SchedulingCard;\n /** The parameters baked into this instance. */\n readonly params: Readonly<FSRSParameters>;\n}\n\n// ── Helpers ─────────────────────────────────────────────────────────────────\n\n/** Clamp a number to [lo, hi]. */\nfunction clamp(value: number, lo: number, hi: number): number {\n return Math.min(hi, Math.max(lo, value));\n}\n\n/** Days between two Date objects (may be fractional). */\nfunction daysBetween(a: Date, b: Date): number {\n return (b.getTime() - a.getTime()) / (1000 * 60 * 60 * 24);\n}\n\n// ── Core FSRS-5 formulas ────────────────────────────────────────────────────\n\n/**\n * Initial stability for a brand-new card.\n * S₀ = w[rating - 1]\n */\nfunction initialStability(w: number[], rating: Rating): number {\n return w[rating - 1];\n}\n\n/**\n * Initial difficulty for a brand-new card.\n * D₀(rating) = w4 - exp(w5 * (rating - 1)) + 1\n * Clamped to [1, 10].\n */\nfunction initialDifficulty(w: number[], rating: Rating): number {\n return clamp(w[4] - Math.exp(w[5] * (rating - 1)) + 1, 1, 10);\n}\n\n/**\n * Updated difficulty after a review.\n * D' = w7 * D₀(3) + (1 - w7) * (D - w6 * (rating - 3))\n * Clamped to [1, 10].\n */\nfunction nextDifficulty(w: number[], d: number, rating: Rating): number {\n const d0ForGood = initialDifficulty(w, 3);\n const updated = w[7] * d0ForGood + (1 - w[7]) * (d - w[6] * (rating - 3));\n return clamp(updated, 1, 10);\n}\n\n/**\n * Retrievability — probability of recall.\n * R = (1 + elapsed / (9 * S))^(-1)\n */\nfunction retrievability(elapsed: number, stability: number): number {\n if (stability <= 0) return 0;\n return (1 + elapsed / (9 * stability)) ** -1;\n}\n\n/**\n * Stability after a **successful** recall (rating >= 2).\n * S' = S * (exp(w8) * (11 - D) * S^(-w9) * (exp(w10 * (1 - R)) - 1) * hard_penalty * easy_bonus + 1)\n */\nfunction stabilityAfterSuccess(\n w: number[],\n s: number,\n d: number,\n r: number,\n rating: Rating,\n): number {\n const hardPenalty = rating === 2 ? w[15] : 1;\n const easyBonus = rating === 4 ? w[16] : 1;\n\n const inner =\n Math.exp(w[8]) *\n (11 - d) *\n s ** -w[9] *\n (Math.exp(w[10] * (1 - r)) - 1) *\n hardPenalty *\n easyBonus;\n\n return s * (inner + 1);\n}\n\n/**\n * Stability after **forgetting** (rating = 1).\n * S' = w11 * D^(-w12) * ((S+1)^w13 - 1) * exp(w14 * (1 - R))\n */\nfunction stabilityAfterForgetting(\n w: number[],\n s: number,\n d: number,\n r: number,\n): number {\n return (\n w[11] * d ** -w[12] * ((s + 1) ** w[13] - 1) * Math.exp(w[14] * (1 - r))\n );\n}\n\n/**\n * Optimal interval given new stability and target retention.\n * I = 9 * S' * (1/requestRetention - 1)\n * Minimum 1 day.\n */\nfunction nextInterval(stability: number, requestRetention: number): number {\n const interval = 9 * stability * (1 / requestRetention - 1);\n return Math.max(1, Math.round(interval));\n}\n\n// ── Factory ─────────────────────────────────────────────────────────────────\n\n/**\n * Create a new card with default initial values.\n */\nexport function createEmptyCard(now?: Date): SchedulingCard {\n const ts = now ?? new Date();\n return {\n stability: 0,\n difficulty: 0,\n elapsedDays: 0,\n scheduledDays: 0,\n reps: 0,\n lapses: 0,\n state: \"new\",\n dueAt: ts,\n lastReviewAt: null,\n };\n}\n\n/**\n * Create an FSRS scheduler instance.\n *\n * All scheduling is done through pure functions — no side effects,\n * no database access, no mutation of the input card.\n */\nexport function createFSRS(params?: Partial<FSRSParameters>): FSRS {\n const resolvedParams: FSRSParameters = {\n w: params?.w ?? [...DEFAULT_W],\n requestRetention: params?.requestRetention ?? DEFAULT_REQUEST_RETENTION,\n };\n\n function schedule(\n card: SchedulingCard,\n rating: Rating,\n now?: Date,\n ): SchedulingCard {\n const reviewTime = now ?? new Date();\n const w = resolvedParams.w;\n\n // Compute elapsed days since last review (or 0 for new cards).\n const elapsed =\n card.lastReviewAt !== null\n ? Math.max(0, daysBetween(card.lastReviewAt, reviewTime))\n : 0;\n\n // ── New card ──────────────────────────────────────────────────────\n if (card.state === \"new\") {\n const s = initialStability(w, rating);\n const d = initialDifficulty(w, rating);\n const interval = nextInterval(s, resolvedParams.requestRetention);\n\n const dueAt = new Date(reviewTime);\n dueAt.setDate(dueAt.getDate() + interval);\n\n // New cards always move to \"learning\" after first rating.\n\n return {\n stability: s,\n difficulty: d,\n elapsedDays: 0,\n scheduledDays: interval,\n reps: rating >= 2 ? 1 : 0,\n lapses: rating === 1 ? 1 : 0,\n state: \"learning\",\n dueAt,\n lastReviewAt: reviewTime,\n };\n }\n\n // ── Existing card (learning / review / relearning) ───────────────\n\n const r = retrievability(elapsed, card.stability);\n\n let newStability: number;\n let newDifficulty: number;\n let newReps: number;\n let newLapses: number;\n let newState: CardState;\n\n if (rating === 1) {\n // Forgot — apply forgetting formula\n newStability = stabilityAfterForgetting(\n w,\n card.stability,\n card.difficulty,\n r,\n );\n newDifficulty = nextDifficulty(w, card.difficulty, rating);\n newReps = 0;\n newLapses = card.lapses + 1;\n newState = \"relearning\";\n } else {\n // Recalled — apply success formula\n newStability = stabilityAfterSuccess(\n w,\n card.stability,\n card.difficulty,\n r,\n rating,\n );\n newDifficulty = nextDifficulty(w, card.difficulty, rating);\n newReps = card.reps + 1;\n newLapses = card.lapses;\n // Successful recall transitions any state to review.\n newState = \"review\";\n }\n\n const interval = nextInterval(\n newStability,\n resolvedParams.requestRetention,\n );\n\n const dueAt = new Date(reviewTime);\n dueAt.setDate(dueAt.getDate() + interval);\n\n return {\n stability: newStability,\n difficulty: newDifficulty,\n elapsedDays: elapsed,\n scheduledDays: interval,\n reps: newReps,\n lapses: newLapses,\n state: newState,\n dueAt,\n lastReviewAt: reviewTime,\n };\n }\n\n return {\n schedule,\n params: Object.freeze(resolvedParams),\n };\n}\n","import type { Database } from \"../db/types.js\";\nimport type { DeleteCardResult } from \"../models/card.js\";\nimport { deleteCardForUser, getCardById } from \"../models/card.js\";\nimport { getPrerequisites } from \"../models/prerequisite.js\";\nimport type {\n DeleteTokenResult,\n Token,\n UpdateTokenInput,\n} from \"../models/token.js\";\nimport {\n deleteToken,\n deprecateToken,\n getTokenById,\n updateToken,\n} from \"../models/token.js\";\nimport type { CascadeBlockResult } from \"../scheduler/blocker.js\";\nimport { cascadeBlock } from \"../scheduler/blocker.js\";\nimport type { Rating } from \"../scheduler/fsrs.js\";\nimport type { EvaluateResult } from \"./evaluator.js\";\nimport { evaluateRating } from \"./evaluator.js\";\n\nexport type ReviewActionType =\n | \"rate\"\n | \"skip\"\n | \"edit-token\"\n | \"deprecate-token\"\n | \"delete-token\"\n | \"delete-card\"\n | \"stop\";\n\nexport interface ExecuteReviewActionInput {\n cardId: string;\n userId: string;\n action: ReviewActionType;\n rating?: Rating;\n tokenUpdates?: UpdateTokenInput;\n}\n\nexport interface ReviewActionResult {\n action: ReviewActionType;\n token: Token;\n evaluation?: EvaluateResult;\n blocked?: CascadeBlockResult;\n updatedToken?: Token;\n deletedToken?: DeleteTokenResult;\n deletedCard?: DeleteCardResult;\n skipped?: boolean;\n stopped?: boolean;\n}\n\nfunction getReviewTarget(\n db: Database,\n cardId: string,\n userId: string,\n): { cardId: string; token: Token } {\n const card = getCardById(db, cardId);\n if (!card) {\n throw new Error(`Card not found: ${cardId}`);\n }\n if (card.user_id !== userId) {\n throw new Error(`Card ${cardId} does not belong to user ${userId}`);\n }\n\n const token = getTokenById(db, card.token_id);\n if (!token) {\n throw new Error(`Token not found for card ${cardId}`);\n }\n\n return { cardId: card.id, token };\n}\n\nexport function executeReviewAction(\n db: Database,\n input: ExecuteReviewActionInput,\n): ReviewActionResult {\n const target = getReviewTarget(db, input.cardId, input.userId);\n\n switch (input.action) {\n case \"rate\": {\n if (input.rating == null) {\n throw new Error(\"rating is required for action=rate\");\n }\n\n const evaluation = evaluateRating(db, {\n cardId: target.cardId,\n tokenId: target.token.id,\n userId: input.userId,\n rating: input.rating,\n });\n\n let blocked: CascadeBlockResult | undefined;\n if (input.rating === 1) {\n const prereqs = getPrerequisites(db, target.token.id);\n if (prereqs.length > 0) {\n blocked = cascadeBlock(db, input.userId, target.token.slug);\n }\n }\n\n return {\n action: input.action,\n token: target.token,\n evaluation,\n blocked,\n };\n }\n\n case \"skip\":\n return { action: input.action, token: target.token, skipped: true };\n\n case \"stop\":\n return { action: input.action, token: target.token, stopped: true };\n\n case \"edit-token\": {\n const updatedToken = updateToken(\n db,\n target.token.slug,\n input.tokenUpdates ?? {},\n );\n return {\n action: input.action,\n token: target.token,\n updatedToken,\n };\n }\n\n case \"deprecate-token\": {\n const updatedToken = deprecateToken(db, target.token.slug);\n return {\n action: input.action,\n token: target.token,\n updatedToken,\n };\n }\n\n case \"delete-token\": {\n const deletedToken = deleteToken(db, target.token.slug);\n return {\n action: input.action,\n token: target.token,\n deletedToken,\n };\n }\n\n case \"delete-card\": {\n const deletedCard = deleteCardForUser(db, target.token.id, input.userId);\n return {\n action: input.action,\n token: target.token,\n deletedCard,\n };\n }\n\n default: {\n const exhaustive: never = input.action;\n throw new Error(`Unsupported review action: ${exhaustive}`);\n }\n }\n}\n","/**\n * Active Recall Prompt Generation\n *\n * Generates review prompts from tokens, adapting the question style\n * to the token's Bloom taxonomy level. This is NOT an LLM call —\n * it's template-based prompt assembly for the CLI and bridge.\n */\n\nexport type BloomLevel = 1 | 2 | 3 | 4 | 5;\n\nexport interface RecallPrompt {\n cardId: string;\n tokenId: string;\n slug: string;\n question: string;\n concept: string;\n domain: string;\n bloomLevel: BloomLevel;\n bloomVerb: string;\n hints: string[];\n sourceLink?: string | null;\n}\n\nconst BLOOM_VERBS: Record<BloomLevel, string> = {\n 1: \"Remember\",\n 2: \"Understand\",\n 3: \"Apply\",\n 4: \"Analyze\",\n 5: \"Synthesize\",\n};\n\nfunction formatSlugForCue(slug: string): string {\n return slug.replace(/[-_]/g, \" \");\n}\n\nconst BLOOM_CUES: Record<BloomLevel, (slug: string) => string> = {\n 1: (slug) =>\n `Recall the definition and core concept of: ${formatSlugForCue(slug)}`,\n 2: (slug) => `Explain the concept and how ${formatSlugForCue(slug)} works.`,\n 3: (slug) =>\n `Describe how or where you would apply the concept of ${formatSlugForCue(slug)}.`,\n 4: (slug) =>\n `Analyze the trade-offs, advantages, or alternatives of ${formatSlugForCue(slug)}.`,\n 5: (slug) =>\n `How would you design a solution using the concept of ${formatSlugForCue(slug)}?`,\n};\n\nexport interface PromptInput {\n cardId: string;\n tokenId: string;\n slug: string;\n concept: string;\n domain: string;\n bloomLevel: BloomLevel;\n sourceLink?: string | null;\n question?: string | null;\n}\n\n/**\n * Generate a template-based concept-free recall cue using the slug and domain.\n */\nexport function generateConceptFreeCue(\n bloomLevel: BloomLevel,\n slug: string,\n _domain: string,\n): string {\n const bloom = (\n bloomLevel >= 1 && bloomLevel <= 5 ? bloomLevel : 1\n ) as BloomLevel;\n return BLOOM_CUES[bloom](slug);\n}\n\n/**\n * Generate a recall prompt for a token at its Bloom level.\n * When called from the CLI, the prompt is rendered in the terminal.\n * When called from the AI bridge, the JSON is returned for the AI to present conversationally.\n */\nexport function generatePrompt(input: PromptInput): RecallPrompt {\n const bloom = (\n input.bloomLevel >= 1 && input.bloomLevel <= 5 ? input.bloomLevel : 1\n ) as BloomLevel;\n\n const question = input.question?.trim()\n ? input.question.trim()\n : generateConceptFreeCue(bloom, input.slug, input.domain);\n\n return {\n cardId: input.cardId,\n tokenId: input.tokenId,\n slug: input.slug,\n question,\n concept: input.concept,\n domain: input.domain,\n bloomLevel: bloom,\n bloomVerb: BLOOM_VERBS[bloom],\n hints: [],\n sourceLink: input.sourceLink ?? null,\n };\n}\n","import { existsSync, readFileSync } from \"node:fs\";\nimport { dirname, join, resolve } from \"node:path\";\n\nexport interface ResolvedReference {\n sourceType: \"local\" | \"remote_web\" | \"dynamic_search\";\n content: string;\n filePath?: string;\n url?: string;\n}\n\n/**\n * A source reference resolved and bounded for inclusion in a review payload.\n * Same shape as ResolvedReference plus the originating link and a truncation flag.\n */\nexport interface ReviewContext {\n sourceLink: string;\n sourceType: ResolvedReference[\"sourceType\"];\n content: string;\n filePath?: string;\n url?: string;\n truncated: boolean;\n}\n\n/** Default cap on resolved content length, so bridge JSON / terminal output stays bounded. */\nexport const DEFAULT_REVIEW_CONTEXT_MAX_CHARS = 6000;\n\n/**\n * Strips HTML tags and attempts to convert basic structure to readable text/markdown.\n */\nfunction htmlToText(html: string): string {\n // Extract body if present\n let content = html;\n const bodyMatch = /<body[^>]*>([\\s\\S]*?)<\\/body>/i.exec(html);\n if (bodyMatch) {\n content = bodyMatch[1];\n }\n\n // Strip script, style, and head tags completely\n content = content.replace(/<(script|style|head)[^>]*>([\\s\\S]*?)<\\/\\1>/gi, \"\");\n // Replace headings\n content = content.replace(\n /<h[1-6][^>]*>([\\s\\S]*?)<\\/h[1-6]>/gi,\n \"\\n\\n# $1\\n\",\n );\n // Replace paragraph/div/li tags with line breaks\n content = content.replace(/<(p|div|li)[^>]*>/gi, \"\\n\");\n content = content.replace(/<\\/(p|div|li)>/gi, \"\\n\");\n content = content.replace(/<br\\s*\\/?>/gi, \"\\n\");\n // Strip all other HTML tags\n content = content.replace(/<[^>]+>/g, \"\");\n // Decode basic HTML entities\n content = content\n .replace(/ /g, \" \")\n .replace(/</g, \"<\")\n .replace(/>/g, \">\")\n .replace(/&/g, \"&\")\n .replace(/"/g, '\"')\n .replace(/'/g, \"'\");\n\n // Collapse consecutive newlines\n content = content.replace(/\\n{3,}/g, \"\\n\\n\").trim();\n return content;\n}\n\n/**\n * Parse line anchors like #L10-L25 or #L10 and extract the lines from file content.\n */\nfunction extractLines(content: string, anchor: string): string {\n const lines = content.split(/\\r?\\n/);\n const match = /#L(\\d+)(?:-L(\\d+))?$/i.exec(anchor);\n if (!match) return content;\n\n const start = Number.parseInt(match[1], 10) - 1; // 0-indexed\n const end = match[2] ? Number.parseInt(match[2], 10) - 1 : start;\n\n if (start < 0 || start >= lines.length) return content;\n\n const slice = lines.slice(start, Math.min(end + 1, lines.length));\n return slice.join(\"\\n\");\n}\n\n/**\n * Resolves a given token's source_link into readable textual content.\n */\nexport async function resolveReference(\n sourceLink: string,\n): Promise<ResolvedReference> {\n const cleaned = sourceLink.trim();\n\n // 1. Dynamic Web Search\n if (cleaned.startsWith(\"search://\")) {\n try {\n const url = new URL(cleaned);\n const query = url.searchParams.get(\"q\") || \"\";\n return {\n sourceType: \"dynamic_search\",\n content: `QUERY_DIRECTIVE: Run web search for \"${query}\"`,\n url: cleaned,\n };\n } catch {\n // Fallback if URL parsing fails\n const query = cleaned.replace(/^search:\\/\\/(\\??q=)?/, \"\");\n return {\n sourceType: \"dynamic_search\",\n content: `QUERY_DIRECTIVE: Run web search for \"${decodeURIComponent(query)}\"`,\n url: cleaned,\n };\n }\n }\n\n // 2. HTTP/HTTPS URLs\n if (cleaned.startsWith(\"http://\") || cleaned.startsWith(\"https://\")) {\n // 2.a GitHub URIs\n const gitHubMatch =\n /^https?:\\/\\/(?:www\\.)?github\\.com\\/([^/]+)\\/([^/]+)\\/blob\\/([^/]+)\\/(.+)$/i.exec(\n cleaned,\n );\n if (gitHubMatch) {\n const [_, owner, repo, branch, fullPathWithAnchor] = gitHubMatch;\n const anchorIndex = fullPathWithAnchor.indexOf(\"#\");\n const filePath =\n anchorIndex !== -1\n ? fullPathWithAnchor.slice(0, anchorIndex)\n : fullPathWithAnchor;\n const anchor =\n anchorIndex !== -1 ? fullPathWithAnchor.slice(anchorIndex) : \"\";\n\n // Try local resolution: check if repo folder exists in sibling directories\n const parentDir = dirname(process.cwd());\n const localRepoPath = join(parentDir, repo);\n const localFilePath = join(localRepoPath, filePath);\n\n if (existsSync(localFilePath)) {\n try {\n let fileContent = readFileSync(localFilePath, \"utf-8\");\n if (anchor) {\n fileContent = extractLines(fileContent, anchor);\n }\n return {\n sourceType: \"local\",\n content: fileContent,\n filePath: localFilePath,\n };\n } catch (_e) {\n // Fallback to fetch if file read fails\n }\n }\n\n // Remote fallback: fetch raw content from githubusercontent\n const rawUrl = `https://raw.githubusercontent.com/${owner}/${repo}/${branch}/${filePath}`;\n try {\n const response = await fetch(rawUrl);\n if (response.ok) {\n let rawText = await response.text();\n if (anchor) {\n rawText = extractLines(rawText, anchor);\n }\n return {\n sourceType: \"remote_web\",\n content: rawText,\n url: cleaned,\n };\n }\n } catch (_e) {\n // Fallback to generic URL loading\n }\n }\n\n // 2.b Generic HTTPS/HTTP URLs\n try {\n const response = await fetch(cleaned);\n if (response.ok) {\n const text = await response.text();\n const cleanText = htmlToText(text);\n return {\n sourceType: \"remote_web\",\n content: cleanText,\n url: cleaned,\n };\n }\n throw new Error(`HTTP error ${response.status}: ${response.statusText}`);\n } catch (err) {\n return {\n sourceType: \"remote_web\",\n content: `Error fetching URL reference: ${(err as Error).message}\\nLink: ${cleaned}`,\n url: cleaned,\n };\n }\n }\n\n // 3. Local Workspace Path (relative to process.cwd)\n const anchorIndex = cleaned.indexOf(\"#\");\n const relativePath =\n anchorIndex !== -1 ? cleaned.slice(0, anchorIndex) : cleaned;\n const anchor = anchorIndex !== -1 ? cleaned.slice(anchorIndex) : \"\";\n const absolutePath = resolve(process.cwd(), relativePath);\n\n if (existsSync(absolutePath)) {\n try {\n let fileContent = readFileSync(absolutePath, \"utf-8\");\n if (anchor) {\n fileContent = extractLines(fileContent, anchor);\n }\n return {\n sourceType: \"local\",\n content: fileContent,\n filePath: absolutePath,\n };\n } catch (_e) {\n // Fallback\n }\n }\n\n // Final fallback: return the path/link description as string\n return {\n sourceType: \"local\",\n content: `Local reference file not found or unreadable.\\nReference: ${cleaned}`,\n filePath: absolutePath,\n };\n}\n\n/**\n * Resolve a token's source_link into bounded, review-ready context.\n *\n * Wraps {@link resolveReference} for the review/bridge flow: returns `null`\n * for empty links and caps content length so the surrounding payload (bridge\n * JSON or terminal output) stays manageable, flagging when truncation occurred.\n */\nexport async function resolveReviewContext(\n sourceLink: string | null | undefined,\n opts: { maxChars?: number } = {},\n): Promise<ReviewContext | null> {\n const cleaned = sourceLink?.trim();\n if (!cleaned) return null;\n\n const maxChars = opts.maxChars ?? DEFAULT_REVIEW_CONTEXT_MAX_CHARS;\n const resolved = await resolveReference(cleaned);\n\n let content = resolved.content;\n let truncated = false;\n if (content.length > maxChars) {\n content = content.slice(0, maxChars);\n truncated = true;\n }\n\n return {\n sourceLink: cleaned,\n sourceType: resolved.sourceType,\n content,\n filePath: resolved.filePath,\n url: resolved.url,\n truncated,\n };\n}\n\n/**\n * Normalizes a path, stripping anchors and converting separators.\n */\nexport function normalizePath(p: string): string {\n const base = p.split(\"#\")[0].trim();\n return base.replace(/\\\\/g, \"/\").toLowerCase();\n}\n\n/**\n * Checks if a token's source_link references a changed file.\n */\nexport function matchesFilePath(\n sourceLink: string | null,\n changedFile: string,\n): boolean {\n if (!sourceLink) return false;\n\n const normSource = normalizePath(sourceLink);\n const normChanged = normalizePath(changedFile);\n\n if (!normSource || !normChanged) return false;\n\n // 1. GitHub URI matching\n const gitHubMatch =\n /^https?:\\/\\/(?:www\\.)?github\\.com\\/([^/]+)\\/([^/]+)\\/blob\\/([^/]+)\\/(.+)$/i.exec(\n normSource,\n );\n if (gitHubMatch) {\n const filePath = gitHubMatch[4];\n return filePath === normChanged;\n }\n\n // Generic URL check (don't match web references against local paths)\n if (normSource.startsWith(\"http://\") || normSource.startsWith(\"https://\")) {\n return false;\n }\n\n // 2. Relative/Absolute path matching\n return normSource.endsWith(normChanged) || normChanged.endsWith(normSource);\n}\n","/**\n * Cross-domain interleaving — prevents same-domain streaks in review queues.\n *\n * Given an array of items with a `domain` field, reorders them using a\n * round-robin strategy so that consecutive items come from different domains.\n * This leverages the interleaving effect: mixing topics during practice\n * strengthens discrimination and long-term retention.\n */\n\n// ── Types ────────────────────────────────────────────────────────────────────\n\nexport interface QueueItem {\n cardId: string;\n tokenId: string;\n domain: string;\n}\n\n// ── Functions ────────────────────────────────────────────────────────────────\n\n/**\n * Reorder items so no domain appears more than `maxConsecutive` times in a row.\n *\n * Algorithm: group items by domain, then round-robin across domain groups.\n * Each round picks one item from each non-exhausted domain. Within a domain,\n * the original order is preserved (so urgency sorting survives).\n *\n * If a domain has more items than others, its extras will appear after all\n * other domains are exhausted — but the `maxConsecutive` cap is still\n * respected by inserting items from the largest remaining domains first.\n *\n * @param items - Array of items to interleave. Not mutated.\n * @param maxConsecutive - Max consecutive items from the same domain. Defaults to 2.\n * @returns A new array with the same items in interleaved order.\n */\nexport function interleave<T extends { domain: string }>(\n items: T[],\n maxConsecutive: number = 2,\n): T[] {\n if (items.length <= 1) return [...items];\n\n // Group items by domain, preserving original order within each group\n const byDomain = new Map<string, T[]>();\n for (const item of items) {\n const group = byDomain.get(item.domain);\n if (group) {\n group.push(item);\n } else {\n byDomain.set(item.domain, [item]);\n }\n }\n\n // If there's only one domain, no interleaving possible\n if (byDomain.size === 1) return [...items];\n\n const result: T[] = [];\n let consecutiveCount = 0;\n let lastDomain: string | null = null;\n\n // Track how many items we've consumed from each domain\n const cursors = new Map<string, number>();\n for (const domain of byDomain.keys()) {\n cursors.set(domain, 0);\n }\n\n // Round-robin: sort domains by remaining count (largest first) each round\n while (result.length < items.length) {\n // Get domains that still have items, sorted by remaining count descending\n const activeDomains = [...byDomain.entries()]\n .filter(\n ([domain]) =>\n (cursors.get(domain) ?? 0) < (byDomain.get(domain)?.length ?? 0),\n )\n .sort((a, b) => {\n const remainA = a[1].length - (cursors.get(a[0]) ?? 0);\n const remainB = b[1].length - (cursors.get(b[0]) ?? 0);\n return remainB - remainA;\n });\n\n if (activeDomains.length === 0) break;\n\n let pickedThisRound = false;\n\n for (const [domain, group] of activeDomains) {\n const cursor = cursors.get(domain) ?? 0;\n if (cursor >= group.length) continue;\n\n // Check if adding from this domain would exceed maxConsecutive\n if (domain === lastDomain && consecutiveCount >= maxConsecutive) {\n // Try to find another domain first\n continue;\n }\n\n // Pick one item from this domain\n result.push(group[cursor]);\n cursors.set(domain, cursor + 1);\n pickedThisRound = true;\n\n if (domain === lastDomain) {\n consecutiveCount++;\n } else {\n lastDomain = domain;\n consecutiveCount = 1;\n }\n\n break;\n }\n\n // If we couldn't pick without exceeding maxConsecutive, we must accept\n // a streak from whatever domain has items left\n if (!pickedThisRound) {\n for (const [domain, group] of activeDomains) {\n const cursor = cursors.get(domain) ?? 0;\n if (cursor >= group.length) continue;\n\n result.push(group[cursor]);\n cursors.set(domain, cursor + 1);\n\n if (domain === lastDomain) {\n consecutiveCount++;\n } else {\n lastDomain = domain;\n consecutiveCount = 1;\n }\n\n break;\n }\n }\n }\n\n return result;\n}\n","/**\n * Review Queue Builder — assembles a session's review queue.\n *\n * Combines due-card fetching, new-card selection, urgency sorting,\n * and cross-domain interleaving into a single ready-to-review queue.\n */\n\nimport type { Database } from \"../db/types.js\";\nimport { interleave } from \"./interleaver.js\";\n\n// ── Types ────────────────────────────────────────────────────────────────────\n\nexport interface ReviewQueueOptions {\n userId: string;\n maxNew?: number; // default 10\n maxReviews?: number; // default 50\n now?: Date;\n}\n\nexport interface ReviewQueueItem {\n cardId: string;\n tokenId: string;\n slug: string;\n concept: string;\n domain: string;\n bloomLevel: number;\n state: string; // 'new' | 'learning' | 'review' | 'relearning'\n dueAt: string;\n sourceLink: string | null;\n question: string | null;\n}\n\nexport interface ReviewQueue {\n items: ReviewQueueItem[];\n newCount: number;\n reviewCount: number;\n relearnCount: number;\n totalDomains: string[];\n}\n\n// ── Internal row type from SQL queries ───────────────────────────────────────\n\ninterface CardRow {\n card_id: string;\n token_id: string;\n slug: string;\n concept: string;\n domain: string;\n bloom_level: number;\n state: string;\n due_at: string;\n source_link: string | null;\n question: string | null;\n}\n\n// ── Functions ────────────────────────────────────────────────────────────────\n\n/**\n * Build a review queue for a user's study session.\n *\n * The queue is assembled in stages:\n * 1. Fetch all due cards (not blocked, due_at <= now, state in review/relearning/learning)\n * 2. Fetch new cards (state = 'new', not blocked) up to maxNew\n * 3. Sort overdue cards by urgency — most overdue first\n * 4. Apply cross-domain interleaving to prevent same-domain streaks\n * 5. Intersperse new cards at regular intervals (every 5th position)\n * 6. Cap total at maxReviews\n *\n * @param db - Database connection\n * @param options - Queue building options\n * @returns The assembled review queue with metadata\n */\nexport function buildReviewQueue(\n db: Database,\n options: ReviewQueueOptions,\n): ReviewQueue {\n const maxNew = options.maxNew ?? 10;\n const maxReviews = options.maxReviews ?? 50;\n const now = options.now ?? new Date();\n const nowISO = now.toISOString();\n\n // ── Step 1: Fetch due cards (review, relearning, learning — not new) ───\n const dueRows = db\n .prepare(\n `SELECT\n c.id AS card_id,\n c.token_id AS token_id,\n t.slug AS slug,\n t.concept AS concept,\n t.domain AS domain,\n t.bloom_level AS bloom_level,\n c.state AS state,\n c.due_at AS due_at,\n t.source_link AS source_link,\n t.question AS question\n FROM cards c\n JOIN tokens t ON t.id = c.token_id\n WHERE c.user_id = ?\n AND c.blocked = 0\n AND c.due_at <= ?\n AND c.state IN ('review', 'relearning', 'learning')\n AND t.deprecated_at IS NULL\n ORDER BY c.due_at ASC`,\n )\n .all(options.userId, nowISO) as CardRow[];\n\n // ── Step 2: Fetch new cards ────────────────────────────────────────────\n const newRows = db\n .prepare(\n `SELECT\n c.id AS card_id,\n c.token_id AS token_id,\n t.slug AS slug,\n t.concept AS concept,\n t.domain AS domain,\n t.bloom_level AS bloom_level,\n c.state AS state,\n c.due_at AS due_at,\n t.source_link AS source_link,\n t.question AS question\n FROM cards c\n JOIN tokens t ON t.id = c.token_id\n WHERE c.user_id = ?\n AND c.blocked = 0\n AND c.state = 'new'\n AND t.deprecated_at IS NULL\n ORDER BY t.bloom_level ASC, t.slug ASC\n LIMIT ?`,\n )\n .all(options.userId, maxNew) as CardRow[];\n\n // ── Step 3: Sort overdue cards by urgency (most overdue first) ─────────\n const nowMs = now.getTime();\n const sortedDue = [...dueRows].sort((a, b) => {\n const overdueA = nowMs - new Date(a.due_at).getTime();\n const overdueB = nowMs - new Date(b.due_at).getTime();\n return overdueB - overdueA; // most overdue first\n });\n\n // ── Step 4: Apply cross-domain interleaving to due cards ───────────────\n const interleavedDue = interleave(\n sortedDue.map((row) => ({ ...rowToItem(row), domain: row.domain })),\n );\n\n // ── Step 5: Intersperse new cards at regular intervals ─────────────────\n const newItems = newRows.map(rowToItem);\n const merged = intersperseNew(interleavedDue, newItems, 5);\n\n // ── Step 6: Cap total at maxReviews ────────────────────────────────────\n const capped = merged.slice(0, maxReviews);\n\n // ── Compute metadata ──────────────────────────────────────────────────\n let newCount = 0;\n let reviewCount = 0;\n let relearnCount = 0;\n const domainSet = new Set<string>();\n\n for (const item of capped) {\n domainSet.add(item.domain);\n switch (item.state) {\n case \"new\":\n newCount++;\n break;\n case \"relearning\":\n relearnCount++;\n break;\n default:\n reviewCount++;\n break;\n }\n }\n\n return {\n items: capped,\n newCount,\n reviewCount,\n relearnCount,\n totalDomains: [...domainSet].sort(),\n };\n}\n\n// ── Helpers ──────────────────────────────────────────────────────────────────\n\n/** Convert a SQL row to a ReviewQueueItem. */\nfunction rowToItem(row: CardRow): ReviewQueueItem {\n return {\n cardId: row.card_id,\n tokenId: row.token_id,\n slug: row.slug,\n concept: row.concept,\n domain: row.domain,\n bloomLevel: row.bloom_level,\n state: row.state,\n dueAt: row.due_at,\n sourceLink: row.source_link,\n question: row.question,\n };\n}\n\n/**\n * Intersperse new cards into the review queue at regular intervals.\n *\n * Instead of front-loading or back-loading new cards, places one new card\n * every `interval` positions (e.g., positions 4, 9, 14, ...).\n * This gives the user a mix of familiar reviews and new material.\n *\n * @param reviews - The interleaved review cards\n * @param newCards - New cards to intersperse\n * @param interval - Place a new card every N positions (default 5)\n * @returns Merged array with new cards interspersed\n */\nfunction intersperseNew(\n reviews: ReviewQueueItem[],\n newCards: ReviewQueueItem[],\n interval: number,\n): ReviewQueueItem[] {\n if (newCards.length === 0) return [...reviews];\n if (reviews.length === 0) return [...newCards];\n\n const result: ReviewQueueItem[] = [];\n let reviewIdx = 0;\n let newIdx = 0;\n\n // Position counter tracks where we are in the final queue\n let position = 0;\n\n while (reviewIdx < reviews.length || newIdx < newCards.length) {\n // Insert a new card every `interval` positions (0-indexed: at 4, 9, 14, ...)\n if (\n newIdx < newCards.length &&\n position > 0 &&\n position % interval === interval - 1\n ) {\n result.push(newCards[newIdx]);\n newIdx++;\n } else if (reviewIdx < reviews.length) {\n result.push(reviews[reviewIdx]);\n reviewIdx++;\n } else if (newIdx < newCards.length) {\n // No more reviews — append remaining new cards\n result.push(newCards[newIdx]);\n newIdx++;\n }\n\n position++;\n }\n\n return result;\n}\n","import {\n appendFileSync,\n copyFileSync,\n existsSync,\n mkdirSync,\n readFileSync,\n writeFileSync,\n} from \"node:fs\";\nimport { homedir } from \"node:os\";\nimport { join } from \"node:path\";\nimport { fileURLToPath } from \"node:url\";\n\nconst HOME = homedir();\n\n/**\n * Get the path to ZAM's internal package SKILL.md.\n */\nexport function getPackageSkillPath(\n agent: \"default\" | \"claude\" | \"codex\" = \"default\",\n): string {\n // Support source modules plus the dist/index.js and dist/cli/index.js bundles.\n const packageRoot =\n [\n fileURLToPath(new URL(\"../../..\", import.meta.url)),\n fileURLToPath(new URL(\"../..\", import.meta.url)),\n fileURLToPath(new URL(\"..\", import.meta.url)),\n ].find((candidate) => existsSync(join(candidate, \"package.json\"))) ?? \"\";\n\n if (!packageRoot) return \"\";\n\n if (agent === \"codex\") {\n const codexPath = join(packageRoot, \".agents\", \"skills\", \"zam\", \"SKILL.md\");\n if (existsSync(codexPath)) return codexPath;\n return \"\";\n }\n\n if (agent === \"claude\") {\n const claudePath = join(\n packageRoot,\n \".claude\",\n \"skills\",\n \"zam\",\n \"SKILL.md\",\n );\n if (existsSync(claudePath)) return claudePath;\n return \"\";\n }\n\n // Try .agent first\n let path = join(packageRoot, \".agent\", \"skills\", \"zam\", \"SKILL.md\");\n if (existsSync(path)) return path;\n\n // Try .claude\n path = join(packageRoot, \".claude\", \"skills\", \"zam\", \"SKILL.md\");\n if (existsSync(path)) return path;\n\n return \"\";\n}\n\n/**\n * Distribute the ZAM active-recall training skill globally.\n * Copies SKILL.md into global directories for supported coding agents.\n */\nexport function distributeGlobalSkills(home: string = HOME): Array<{\n name: string;\n path: string;\n success: boolean;\n}> {\n const sourceSkill = getPackageSkillPath();\n const claudeSourceSkill = getPackageSkillPath(\"claude\");\n const codexSourceSkill = getPackageSkillPath(\"codex\");\n const results: Array<{ name: string; path: string; success: boolean }> = [];\n\n if (!sourceSkill) {\n console.warn(\"Could not find ZAM source SKILL.md in the package folder.\");\n return results;\n }\n\n // 1. Claude Code global directory\n const claudeSkillsDir = join(home, \".claude\", \"skills\", \"zam\");\n try {\n if (!claudeSourceSkill) {\n throw new Error(\"Claude skill source not found\");\n }\n mkdirSync(claudeSkillsDir, { recursive: true });\n copyFileSync(claudeSourceSkill, join(claudeSkillsDir, \"SKILL.md\"));\n results.push({\n name: \"Claude Code Global\",\n path: join(claudeSkillsDir, \"SKILL.md\"),\n success: true,\n });\n } catch (_err) {\n results.push({\n name: \"Claude Code Global\",\n path: claudeSkillsDir,\n success: false,\n });\n }\n\n // 2. Gemini/agy global directory\n const geminiSkillsDir = join(home, \".gemini\", \"skills\", \"zam\");\n try {\n mkdirSync(geminiSkillsDir, { recursive: true });\n copyFileSync(sourceSkill, join(geminiSkillsDir, \"SKILL.md\"));\n results.push({\n name: \"Gemini CLI Global\",\n path: join(geminiSkillsDir, \"SKILL.md\"),\n success: true,\n });\n } catch (_err) {\n results.push({\n name: \"Gemini CLI Global\",\n path: geminiSkillsDir,\n success: false,\n });\n }\n\n // 3. Codex global skills directory\n const codexSkillsDir = join(home, \".agents\", \"skills\", \"zam\");\n try {\n if (!codexSourceSkill) {\n throw new Error(\"Codex skill source not found\");\n }\n mkdirSync(codexSkillsDir, { recursive: true });\n copyFileSync(codexSourceSkill, join(codexSkillsDir, \"SKILL.md\"));\n results.push({\n name: \"Codex Global\",\n path: join(codexSkillsDir, \"SKILL.md\"),\n success: true,\n });\n } catch (_err) {\n results.push({\n name: \"Codex Global\",\n path: codexSkillsDir,\n success: false,\n });\n }\n\n // 4. Goose skills directory\n const gooseSkillsDir = join(home, \".goose\", \"skills\", \"zam\");\n try {\n mkdirSync(gooseSkillsDir, { recursive: true });\n copyFileSync(sourceSkill, join(gooseSkillsDir, \"SKILL.md\"));\n results.push({\n name: \"Goose Global\",\n path: join(gooseSkillsDir, \"SKILL.md\"),\n success: true,\n });\n } catch (_err) {\n results.push({\n name: \"Goose Global\",\n path: gooseSkillsDir,\n success: false,\n });\n }\n\n return results;\n}\n\nconst POSIX_OLD_HOOK = `\n# ZAM Shell Observation Hooks\nif (command -v zam >/dev/null 2>&1); then eval \"$(zam monitor start --quiet)\"; fi\n`;\nconst POWERSHELL_OLD_HOOK = `\n# ZAM Shell Observation Hooks\nif (Get-Command zam -ErrorAction SilentlyContinue) { Invoke-Expression (& zam monitor start --quiet pwsh) }\n`;\nconst HOOK_MARKER = \"# ZAM Monitor Session Helper\";\n\nfunction posixHook(shell: \"bash\" | \"zsh\"): string {\n return `\n${HOOK_MARKER}\nzam-monitor-session() {\n local session_id=\"\\${1:-}\"\n if [ -z \"$session_id\" ]; then\n printf 'Usage: zam-monitor-session <session-id>\\n' >&2\n return 2\n fi\n eval \"$(command zam monitor start --session \"$session_id\" --shell ${shell})\"\n}\n`;\n}\n\nconst POWERSHELL_HOOK = `\n${HOOK_MARKER}\nfunction Start-ZamMonitor {\n param([Parameter(Mandatory = $true)][string]$Session)\n Invoke-Expression (& zam monitor start --session $Session --shell pwsh)\n}\n`;\n\nfunction installHook(\n file: string,\n hook: string,\n oldHook: string,\n): { success: boolean; alreadyHooked: boolean } {\n try {\n const content = existsSync(file) ? readFileSync(file, \"utf8\") : \"\";\n if (content.includes(HOOK_MARKER)) {\n return { success: true, alreadyHooked: true };\n }\n\n if (content.includes(oldHook.trim())) {\n writeFileSync(file, content.replace(oldHook.trim(), hook.trim()), \"utf8\");\n } else {\n appendFileSync(file, hook);\n }\n return { success: true, alreadyHooked: false };\n } catch {\n return { success: false, alreadyHooked: false };\n }\n}\n\n/**\n * Add opt-in helpers for starting a monitored session to user shell profiles.\n */\nexport function injectShellHooks(home: string = HOME): Array<{\n shell: string;\n file: string;\n success: boolean;\n alreadyHooked: boolean;\n}> {\n const results: Array<{\n shell: string;\n file: string;\n success: boolean;\n alreadyHooked: boolean;\n }> = [];\n\n // 1. Zsh profile (~/.zshrc)\n const zshrc = join(home, \".zshrc\");\n if (existsSync(zshrc)) {\n const status = installHook(zshrc, posixHook(\"zsh\"), POSIX_OLD_HOOK);\n results.push({ shell: \"zsh\", file: zshrc, ...status });\n }\n\n // 2. Bash profile (~/.bashrc)\n const bashrc = join(home, \".bashrc\");\n if (existsSync(bashrc)) {\n const status = installHook(bashrc, posixHook(\"bash\"), POSIX_OLD_HOOK);\n results.push({ shell: \"bash\", file: bashrc, ...status });\n }\n\n // 3. PowerShell Profile ($HOME\\Documents\\PowerShell\\Microsoft.PowerShell_profile.ps1)\n // Check both PowerShell and WindowsPowerShell\n const pwshDirs = [\n join(home, \"Documents\", \"PowerShell\"),\n join(home, \"Documents\", \"WindowsPowerShell\"),\n ];\n\n for (const dir of pwshDirs) {\n const profileFile = join(dir, \"Microsoft.PowerShell_profile.ps1\");\n try {\n mkdirSync(dir, { recursive: true });\n const status = installHook(\n profileFile,\n POWERSHELL_HOOK,\n POWERSHELL_OLD_HOOK,\n );\n results.push({\n shell: \"powershell\",\n file: profileFile,\n ...status,\n });\n } catch {\n results.push({\n shell: \"powershell\",\n file: profileFile,\n success: false,\n alreadyHooked: false,\n });\n }\n }\n\n return results;\n}\n","import type { SupportedLocale } from \"./locale.js\";\n\nexport type TranslationKey =\n | \"welcome\"\n | \"new_review_relearn\"\n | \"domains\"\n | \"instruction\"\n | \"quit_hint\"\n | \"offline_warning\"\n | \"offline_instruction\"\n | \"nothing_due\"\n | \"evaluating\"\n | \"generating_question\"\n | \"translating\"\n | \"prompt_answer\"\n | \"session_ended\"\n | \"session_complete\"\n | \"cards_rated\"\n | \"avg_rating\"\n | \"forgot\"\n | \"feedback_title\"\n | \"answer_title\"\n | \"keep_waiting\"\n | \"local_ai_working\"\n | \"wait_warning\"\n | \"wait_info\"\n | \"keep_waiting_llm\"\n | \"proceeding_offline\"\n | \"eval_skipped\";\n\nexport const TRANSLATIONS: Record<\n SupportedLocale,\n Record<TranslationKey, string>\n> = {\n en: {\n welcome: \"Learning session: {count} card(s)\",\n new_review_relearn: \" New: {newC} Review: {reviewC} Relearn: {relearnC}\",\n domains: \" Domains: {domains}\",\n instruction:\n \"\\nRecall each answer first, reveal it, then rate yourself honestly.\",\n quit_hint:\n \"Type 'q' at the answer prompt (or press Ctrl+C) to stop anytime.\",\n offline_warning:\n \"\\n\\x1b[33m⚠ LLM-Feedback & automatic translation are disabled.\\x1b[0m\",\n offline_instruction:\n \" Enable with: \\x1b[36mnpm run dev -- settings llm on\\x1b[0m\\n\",\n nothing_due: \"Nothing due to learn. You're all caught up!\",\n evaluating: \"Evaluating answer via local AI...\",\n generating_question: \"Generating dynamic question...\",\n translating: \"Translating question dynamically...\",\n prompt_answer: \"Your answer (Enter to reveal · 'q' to stop):\",\n session_ended: \"Learning session ended.\",\n session_complete: \"Learning session complete!\",\n cards_rated: \" Cards rated: {count}\",\n avg_rating: \" Average rating: {avg}\",\n forgot: \" Forgot: {count} card(s)\",\n feedback_title: \"── ZAM Feedback {line}\",\n answer_title: \"── Answer {line}\",\n keep_waiting: \"Would you like to keep waiting?\",\n local_ai_working: \"The local AI is still generating the response.\",\n wait_warning: \"⚠ The LLM server is taking a while to load the model.\",\n wait_info:\n \"(This is expected when transitioning between models or starting up from cold.)\",\n keep_waiting_llm: \"Would you like to keep waiting for the model?\",\n proceeding_offline:\n \"⚠ Proceeding in offline-mode (without active LLM evaluations for this session).\",\n eval_skipped: \" [LLM Evaluation skipped: {reason}]\",\n },\n de: {\n welcome: \"Lern-Session: {count} Karte(n)\",\n new_review_relearn:\n \" Neu: {newC} Wiederholen: {reviewC} Lernen: {relearnC}\",\n domains: \" Domänen: {domains}\",\n instruction:\n \"\\nRufe jede Antwort zuerst ab, decke sie auf und bewerte dich dann ehrlich selbst.\",\n quit_hint:\n \"Gib 'q' bei der Antwortaufforderung ein (oder drücke Strg+C), um jederzeit zu beenden.\",\n offline_warning:\n \"\\n\\x1b[33m⚠ LLM-Feedback & automatische Übersetzung sind deaktiviert.\\x1b[0m\",\n offline_instruction:\n \" Aktivieren mit: \\x1b[36mnpm run dev -- settings llm on\\x1b[0m\\n\",\n nothing_due: \"Nichts fällig zu lernen. Du bist komplett auf dem Laufenden!\",\n evaluating: \"Bewerte Antwort via lokaler KI...\",\n generating_question: \"Generiere dynamische Frage...\",\n translating: \"Übersetze Frage dynamisch...\",\n prompt_answer: \"Deine Antwort (Eingabe zum Aufdecken · 'q' zum Beenden):\",\n session_ended: \"Lern-Session beendet.\",\n session_complete: \"Lern-Session abgeschlossen!\",\n cards_rated: \" Bewertete Karten: {count}\",\n avg_rating: \" Durchschnittliche Bewertung: {avg}\",\n forgot: \" Vergessen: {count} Karte(n)\",\n feedback_title: \"── ZAM Feedback {line}\",\n answer_title: \"── Antwort {line}\",\n keep_waiting: \"Möchtest du weiter auf die Bewertung warten?\",\n local_ai_working: \"Die lokale KI arbeitet noch an der Antwort.\",\n wait_warning:\n \"⚠ Der LLM-Server braucht ungewöhnlich lange, um das Modell zu laden.\",\n wait_info:\n \"(Das ist normal, wenn das Modell gewechselt wird oder kalt startet.)\",\n keep_waiting_llm: \"Möchtest du weiter auf das Modell warten?\",\n proceeding_offline:\n \"⚠ Fahre im Offline-Modus fort (ohne aktive LLM-Bewertungen in dieser Runde).\",\n eval_skipped: \" [LLM-Bewertung übersprungen: {reason}]\",\n },\n es: {\n welcome: \"Sesión de aprendizaje: {count} tarjeta(s)\",\n new_review_relearn:\n \" Nuevas: {newC} Repasar: {reviewC} Reaprender: {relearnC}\",\n domains: \" Dominios: {domains}\",\n instruction:\n \"\\nRecuerda cada respuesta primero, revélala y califícate honestamente.\",\n quit_hint:\n \"Escribe 'q' en la respuesta (o presiona Ctrl+C) para salir en cualquier momento.\",\n offline_warning:\n \"\\n\\x1b[33m⚠ Los comentarios de LLM y la traducción automática están desactivados.\\x1b[0m\",\n offline_instruction:\n \" Activar con: \\x1b[36mnpm run dev -- settings llm on\\x1b[0m\\n\",\n nothing_due: \"No hay nada pendiente para aprender. ¡Estás al día!\",\n evaluating: \"Evaluando respuesta con IA local...\",\n generating_question: \"Generando pregunta dinámica...\",\n translating: \"Traduciendo pregunta dinámicamente...\",\n prompt_answer: \"Tu respuesta (Intro para revelar · 'q' para salir):\",\n session_ended: \"Sesión de aprendizaje finalizada.\",\n session_complete: \"¡Sesión de aprendizaje completada!\",\n cards_rated: \" Tarjetas calificadas: {count}\",\n avg_rating: \" Calificación promedio: {avg}\",\n forgot: \" Olvidadas: {count} tarjeta(s)\",\n feedback_title: \"── Comentarios de ZAM {line}\",\n answer_title: \"── Respuesta {line}\",\n keep_waiting: \"¿Deseas seguir esperando la evaluación?\",\n local_ai_working: \"La IA local todavía está generando la respuesta.\",\n wait_warning: \"⚠ El servidor LLM está tardando en cargar el modelo.\",\n wait_info: \"(Esto es normal al cambiar de modelo o iniciar en frío.)\",\n keep_waiting_llm: \"¿Deseas seguir esperando el modelo?\",\n proceeding_offline:\n \"⚠ Continuando en modo fuera de línea (sin evaluaciones de LLM en esta sesión).\",\n eval_skipped: \" [Evaluación de LLM omitida: {reason}]\",\n },\n fr: {\n welcome: \"Session d'apprentissage : {count} carte(s)\",\n new_review_relearn:\n \" Nouveau: {newC} Révision: {reviewC} Relever: {relearnC}\",\n domains: \" Domaines: {domains}\",\n instruction:\n \"\\nRappelez-vous chaque réponse d'abord, révélez-la, puis évaluez-vous honnêtement.\",\n quit_hint: \"Tapez 'q' (ou Ctrl+C) pour quitter à tout moment.\",\n offline_warning:\n \"\\n\\x1b[33m⚠ Les commentaires LLM et la traduction automatique sont désactivés.\\x1b[0m\",\n offline_instruction:\n \" Activer avec : \\x1b[36mnpm run dev -- settings llm on\\x1b[0m\\n\",\n nothing_due: \"Rien à apprendre. Vous êtes à jour !\",\n evaluating: \"Évaluation de la réponse via l'IA locale...\",\n generating_question: \"Génération d'une question dynamique...\",\n translating: \"Traduction dynamique de la question...\",\n prompt_answer: \"Votre réponse (Entrée pour révéler · 'q' pour quitter) :\",\n session_ended: \"Session d'apprentissage arrêtée.\",\n session_complete: \"Session d'apprentissage terminée !\",\n cards_rated: \" Cartes évaluées: {count}\",\n avg_rating: \" Note moyenne: {avg}\",\n forgot: \" Oubliées: {count} carte(s)\",\n feedback_title: \"── Commentaires ZAM {line}\",\n answer_title: \"── Réponse {line}\",\n keep_waiting: \"Voulez-vous continuer à attendre l'évaluation ?\",\n local_ai_working:\n \"L'IA locale est toujours en train de générer la réponse.\",\n wait_warning: \"⚠ Le serveur LLM prend du temps pour charger le modèle.\",\n wait_info:\n \"(Ceci est normal lors de la transition entre modèles ou du démarrage à froid.)\",\n keep_waiting_llm: \"Voulez-vous continuer à attendre le modèle ?\",\n proceeding_offline:\n \"⚠ Poursuite en mode hors ligne (sans évaluation active de l'IA pour cette session).\",\n eval_skipped: \" [Évaluation LLM ignorée : {reason}]\",\n },\n pt: {\n welcome: \"Sessão de aprendizado: {count} cartão(ões)\",\n new_review_relearn:\n \" Novos: {newC} Revisar: {reviewC} Reaprender: {relearnC}\",\n domains: \" Domínios: {domains}\",\n instruction:\n \"\\nLembre-se de cada resposta primeiro, revele-a e avalie-se honestamente.\",\n quit_hint: \"Digite 'q' (ou Ctrl+C) para parar a qualquer momento.\",\n offline_warning:\n \"\\n\\x1b[33m⚠ O feedback do LLM e a tradução automática estão desativados.\\x1b[0m\",\n offline_instruction:\n \" Ativar com: \\x1b[36mnpm run dev -- settings llm on\\x1b[0m\\n\",\n nothing_due: \"Nada faturado para aprender. Você está atualizado!\",\n evaluating: \"Avaliando a resposta via IA local...\",\n generating_question: \"Gerando pergunta dinâmica...\",\n translating: \"Traduzindo pergunta dinamicamente...\",\n prompt_answer: \"Sua resposta (Enter para revelar · 'q' para parar):\",\n session_ended: \"Sessão de aprendizado encerrada.\",\n session_complete: \"Sessão de aprendizado concluída!\",\n cards_rated: \" Cartões avaliados: {count}\",\n avg_rating: \" Nota média: {avg}\",\n forgot: \" Esquecidos: {count} cartão(ões)\",\n feedback_title: \"── Feedback ZAM {line}\",\n answer_title: \"── Resposta {line}\",\n keep_waiting: \"Deseja continuar esperando pela avaliação?\",\n local_ai_working: \"A IA local ainda está gerando a resposta.\",\n wait_warning: \"⚠ O servidor LLM está demorando para carregar o modelo.\",\n wait_info: \"(Isso é esperado ao alternar modelos ou iniciar do zero.)\",\n keep_waiting_llm: \"Deseja continuar esperando o modelo?\",\n proceeding_offline:\n \"⚠ Continuando no modo offline (sem avaliações de LLM ativas nesta sessão).\",\n eval_skipped: \" [Avaliação LLM omitida: {reason}]\",\n },\n zh: {\n welcome: \"学习课: {count} 张卡片\",\n new_review_relearn: \" 新卡: {newC} 复习: {reviewC} 重学: {relearnC}\",\n domains: \" 知识领域: {domains}\",\n instruction: \"\\n首先在脑中回忆答案,然后揭晓并诚实自我评分。\",\n quit_hint: \"在回答提示处输入 'q' (或按 Ctrl+C) 可随时退出。\",\n offline_warning: \"\\n\\x1b[33m⚠ LLM 反馈与自动翻译已禁用。\\x1b[0m\",\n offline_instruction:\n \" 开启命令: \\x1b[36mnpm run dev -- settings llm on\\x1b[0m\\n\",\n nothing_due: \"目前没有需要学习的内容。您已全部掌握!\",\n evaluating: \"正在通过本地 AI 评估回答...\",\n generating_question: \"正在动态生成问题...\",\n translating: \"正在动态翻译问题...\",\n prompt_answer: \"您的回答 (按回车揭晓 · 输入 'q' 退出):\",\n session_ended: \"学习课已结束。\",\n session_complete: \"学习课已完成!\",\n cards_rated: \" 已评分卡片: {count}\",\n avg_rating: \" 平均分: {avg}\",\n forgot: \" 遗忘: {count} 张卡片\",\n feedback_title: \"── ZAM 反馈 {line}\",\n answer_title: \"── 参考答案 {line}\",\n keep_waiting: \"是否继续等待评分?\",\n local_ai_working: \"本地 AI 仍在生成回答。\",\n wait_warning: \"⚠ LLM 服务器正在加载模型,这可能需要一些时间。\",\n wait_info: \"(这在切换模型或冷启动时是正常现象。)\",\n keep_waiting_llm: \"是否继续等待模型加载?\",\n proceeding_offline:\n \"⚠ 正在以离线模式继续(本次学习课将不包含活跃的 AI 评估)。\",\n eval_skipped: \" [已跳过 LLM 评估: {reason}]\",\n },\n ja: {\n welcome: \"学習セッション: {count} 枚のカード\",\n new_review_relearn: \" 新規: {newC} 復習: {reviewC} 再学習: {relearnC}\",\n domains: \" ドメイン: {domains}\",\n instruction:\n \"\\n最初に回答を思い出し、次に回答を表示して、正直に自己評価してください。\",\n quit_hint:\n \"回答プロンプトで「q」を入力する(または Ctrl+C を押す)と、いつでも終了できます。\",\n offline_warning:\n \"\\n\\x1b[33m⚠ LLM フィードバックと自動翻訳は無効です。\\x1b[0m\",\n offline_instruction:\n \" 有効化するには: \\x1b[36mnpm run dev -- settings llm on\\x1b[0m\\n\",\n nothing_due: \"学習予定のカードはありません。すべて完了しています!\",\n evaluating: \"ローカルAIによる回答の評価中...\",\n generating_question: \"質問を動的に生成中...\",\n translating: \"質問を動的に翻訳中...\",\n prompt_answer: \"あなたの回答 (Enterで表示 · 'q'で終了):\",\n session_ended: \"学習セッションが終了しました。\",\n session_complete: \"学習セッションが完了しました!\",\n cards_rated: \" 評価済みカード数: {count}\",\n avg_rating: \" 平均評価: {avg}\",\n forgot: \" 忘れたカード数: {count} 枚\",\n feedback_title: \"── ZAM フィードバック {line}\",\n answer_title: \"── 解答 {line}\",\n keep_waiting: \"評価の生成を待ちますか?\",\n local_ai_working: \"ローカルAIが回答を生成しています。\",\n wait_warning:\n \"⚠ LLM サーバーがモデルをロードするのに時間がかかっています。\",\n wait_info: \"(モデルの移行中やコールドスタート時には、これが予想されます。)\",\n keep_waiting_llm: \"モデルのロードを待ち続けますか?\",\n proceeding_offline:\n \"⚠ オフラインモードで続行します(このセッションではアクティブな AI 評価は行われません)。\",\n eval_skipped: \" [LLM 評価がスキップされました: {reason}]\",\n },\n};\n\n/**\n * Format and interpolate a translation string with key-value params.\n */\nexport function t(\n locale: SupportedLocale,\n key: TranslationKey,\n params: Record<string, string | number> = {},\n): string {\n const dict = TRANSLATIONS[locale] || TRANSLATIONS.en;\n let str = dict[key] || TRANSLATIONS.en[key] || \"\";\n\n for (const [k, v] of Object.entries(params)) {\n str = str.replace(new RegExp(`{${k}}`, \"g\"), String(v));\n }\n\n return str;\n}\n","import { execFileSync, execSync } from \"node:child_process\";\nimport { existsSync } from \"node:fs\";\nimport { homedir } from \"node:os\";\nimport { join } from \"node:path\";\n\nexport interface InstallResult {\n success: boolean;\n message: string;\n}\n\nexport type LocalLLMRunner = \"fastflowlm\" | \"ollama\" | \"generic\";\n\n/**\n * Check if a command is executable on the system.\n */\nexport function hasCommand(cmd: string): boolean {\n try {\n const checkCmd =\n process.platform === \"win32\" ? `where ${cmd}` : `which ${cmd}`;\n execSync(checkCmd, { stdio: \"ignore\" });\n return true;\n } catch {\n return false;\n }\n}\n\n/**\n * Install FastFlowLM via winget on Windows.\n */\nexport function installFastFlowLM(): InstallResult {\n if (process.platform !== \"win32\") {\n return {\n success: false,\n message: \"FastFlowLM is only supported on Windows.\",\n };\n }\n\n // Check if already installed\n const hasFlm =\n hasCommand(\"flm\") || existsSync(\"C:\\\\Program Files\\\\flm\\\\flm.exe\");\n if (hasFlm) {\n return { success: true, message: \"FastFlowLM is already installed.\" };\n }\n\n if (!hasCommand(\"winget\")) {\n return {\n success: false,\n message: \"winget package manager was not found on this system.\",\n };\n }\n\n console.log(\"Installing FastFlowLM via winget...\");\n try {\n // -e option exact match, --accept-source-agreements --accept-package-agreements\n execSync(\n \"winget install -e --id FastFlowLM --accept-source-agreements --accept-package-agreements\",\n { stdio: \"inherit\" },\n );\n return { success: true, message: \"FastFlowLM installed successfully.\" };\n } catch (err) {\n return {\n success: false,\n message: `Failed to install FastFlowLM: ${(err as Error).message}`,\n };\n }\n}\n\n/**\n * Install Ollama via Homebrew on macOS.\n */\nexport function installOllama(): InstallResult {\n // Check if already installed\n const isMac = process.platform === \"darwin\";\n const isWin = process.platform === \"win32\";\n const hasOllama =\n hasCommand(\"ollama\") ||\n (isMac && existsSync(\"/Applications/Ollama.app\")) ||\n (isWin &&\n existsSync(\n join(homedir(), \"AppData\", \"Local\", \"Programs\", \"Ollama\", \"ollama.exe\"),\n ));\n\n if (hasOllama) {\n return { success: true, message: \"Ollama is already installed.\" };\n }\n\n if (process.platform === \"darwin\") {\n if (!hasCommand(\"brew\")) {\n return {\n success: false,\n message:\n \"Homebrew was not found. Please install Homebrew from brew.sh first.\",\n };\n }\n console.log(\"Installing Ollama via Homebrew Cask...\");\n try {\n execSync(\"brew install --cask ollama\", { stdio: \"inherit\" });\n return { success: true, message: \"Ollama installed successfully.\" };\n } catch (err) {\n return {\n success: false,\n message: `Failed to install Ollama: ${(err as Error).message}`,\n };\n }\n } else if (process.platform === \"win32\") {\n if (!hasCommand(\"winget\")) {\n return {\n success: false,\n message: \"winget was not found. Please install winget first.\",\n };\n }\n console.log(\"Installing Ollama via winget...\");\n try {\n execSync(\n \"winget install -e --id Ollama.Ollama --accept-source-agreements --accept-package-agreements\",\n { stdio: \"inherit\" },\n );\n return { success: true, message: \"Ollama installed successfully.\" };\n } catch (err) {\n return {\n success: false,\n message: `Failed to install Ollama: ${(err as Error).message}`,\n };\n }\n } else {\n // Linux installer script\n console.log(\"Installing Ollama via official installer script...\");\n try {\n execSync(\"curl -fsSL https://ollama.com/install.sh | sh\", {\n stdio: \"inherit\",\n });\n return { success: true, message: \"Ollama installed successfully.\" };\n } catch (err) {\n return {\n success: false,\n message: `Failed to install Ollama: ${(err as Error).message}`,\n };\n }\n }\n}\n\nfunction resolveOllamaCommand(): string | undefined {\n if (hasCommand(\"ollama\")) return \"ollama\";\n\n const candidates =\n process.platform === \"win32\"\n ? [\n join(\n homedir(),\n \"AppData\",\n \"Local\",\n \"Programs\",\n \"Ollama\",\n \"ollama.exe\",\n ),\n ]\n : process.platform === \"darwin\"\n ? [\"/Applications/Ollama.app/Contents/Resources/ollama\"]\n : [];\n\n return candidates.find((candidate) => existsSync(candidate));\n}\n\n/**\n * Prepare the recommended model after installing a local LLM runner.\n */\nexport function prepareLocalModel(\n runner: LocalLLMRunner,\n model: string,\n): InstallResult {\n if (runner === \"fastflowlm\") {\n return {\n success: true,\n message: `${model} will be downloaded by FastFlowLM on first use.`,\n };\n }\n\n if (runner !== \"ollama\") {\n return {\n success: false,\n message: \"No supported local LLM runner was selected.\",\n };\n }\n\n const ollamaCommand = resolveOllamaCommand();\n if (!ollamaCommand) {\n return {\n success: false,\n message:\n \"Ollama was installed but its command is not available yet. Restart the terminal, then run \" +\n `ollama pull ${model}.`,\n };\n }\n\n console.log(`Downloading ${model} with Ollama...`);\n try {\n execFileSync(ollamaCommand, [\"pull\", model], { stdio: \"inherit\" });\n return { success: true, message: `${model} is ready in Ollama.` };\n } catch (err) {\n return {\n success: false,\n message:\n `Could not prepare ${model}: ${(err as Error).message}. ` +\n `Start Ollama and run: ollama pull ${model}`,\n };\n }\n}\n","import { execSync } from \"node:child_process\";\n\nexport type SupportedLocale = \"en\" | \"de\" | \"es\" | \"fr\" | \"pt\" | \"zh\" | \"ja\";\n\nconst SUPPORTED_LOCALES: Set<SupportedLocale> = new Set([\n \"en\",\n \"de\",\n \"es\",\n \"fr\",\n \"pt\",\n \"zh\",\n \"ja\",\n]);\n\n/**\n * Clean and map raw locale string (e.g., \"de_DE.UTF-8\" or \"en-US\") to SupportedLocale.\n */\nexport function normalizeLocale(raw: string): SupportedLocale {\n const clean = raw.trim().toLowerCase().split(/[_-]/)[0];\n if (SUPPORTED_LOCALES.has(clean as SupportedLocale)) {\n return clean as SupportedLocale;\n }\n return \"en\";\n}\n\n/**\n * Detect the operating system's active language code dynamically.\n */\nexport function detectSystemLocale(): SupportedLocale {\n try {\n // 1. Check standard POSIX env vars (common on macOS/Linux/Git Bash/WSL)\n const envVars = [\n process.env.LANG,\n process.env.LANGUAGE,\n process.env.LC_ALL,\n process.env.LC_MESSAGES,\n ];\n\n for (const val of envVars) {\n if (val && val.trim().length > 0) {\n return normalizeLocale(val);\n }\n }\n\n // 2. On Windows, fallback to querying PowerShell Culture\n if (process.platform === \"win32\") {\n const output = execSync(\n 'powershell -NoProfile -Command \"[System.Globalization.CultureInfo]::CurrentCulture.Name\"',\n { stdio: \"pipe\", encoding: \"utf8\", timeout: 2000 },\n ).trim();\n if (output && output.length > 0) {\n return normalizeLocale(output);\n }\n }\n } catch {\n // Ignore errors and default to English\n }\n\n return \"en\";\n}\n","import { execSync } from \"node:child_process\";\n\nexport interface SystemProfile {\n os: \"windows\" | \"macos\" | \"linux\" | \"unknown\";\n arch: \"x64\" | \"arm64\" | \"unknown\";\n hasRyzenNPU: boolean;\n hasAppleSilicon: boolean;\n recommendedRunner: \"fastflowlm\" | \"ollama\" | \"generic\";\n recommendedModel: string;\n}\n\n/**\n * Run a shell command synchronously and return stdout.\n * Returns empty string on failure.\n */\nfunction runCommand(cmd: string): string {\n try {\n return execSync(cmd, { stdio: \"pipe\", encoding: \"utf8\" }).trim();\n } catch {\n return \"\";\n }\n}\n\nfunction detectWindowsAMDIPU(): boolean {\n if (process.platform !== \"win32\") return false;\n\n // WMI query for AMD IPU (Image Processing Unit), NPU, Ryzen AI CPUs, and modern NPU compute devices (DEV_1502, DEV_17F0)\n const cmd = `powershell -NoProfile -Command \"Get-CimInstance Win32_PnPEntity | Where-Object { $_.Name -like '*AMD IPU*' -or $_.Name -like '*AMD NPU*' -or $_.Name -like '*NPU Compute*' -or $_.Name -like '*Ryzen AI*' -or $_.HardwareID -like '*VEN_1022&DEV_1502*' -or $_.HardwareID -like '*VEN_1022&DEV_17F0*' } | Select-Object -First 1 -ExpandProperty Name\"`;\n const output = runCommand(cmd);\n\n return Boolean(\n output &&\n (output.toLowerCase().includes(\"amd\") ||\n output.toLowerCase().includes(\"ipu\") ||\n output.toLowerCase().includes(\"npu\") ||\n output.toLowerCase().includes(\"ryzen\")),\n );\n}\n\n/**\n * Profile the active system hardware and software capabilities.\n */\nexport function getSystemProfile(): SystemProfile {\n const platform = process.platform;\n const archStr = process.arch;\n\n let os: \"windows\" | \"macos\" | \"linux\" | \"unknown\" = \"unknown\";\n if (platform === \"win32\") os = \"windows\";\n else if (platform === \"darwin\") os = \"macos\";\n else if (platform === \"linux\") os = \"linux\";\n\n let arch: \"x64\" | \"arm64\" | \"unknown\" = \"unknown\";\n if (archStr === \"x64\") arch = \"x64\";\n else if (archStr === \"arm64\") arch = \"arm64\";\n\n const hasRyzenNPU = os === \"windows\" && detectWindowsAMDIPU();\n const hasAppleSilicon = os === \"macos\" && arch === \"arm64\";\n\n let recommendedRunner: \"fastflowlm\" | \"ollama\" | \"generic\" = \"generic\";\n let recommendedModel = \"qwen3.5:4b\";\n\n if (hasRyzenNPU) {\n recommendedRunner = \"fastflowlm\";\n recommendedModel = \"qwen3.5:4b\";\n } else if (hasAppleSilicon) {\n recommendedRunner = \"ollama\";\n recommendedModel = \"llama3.2:3b\";\n } else if (os === \"macos\" || os === \"linux\" || os === \"windows\") {\n // Standard PC / generic Mac\n recommendedRunner = \"ollama\";\n recommendedModel = \"llama3.2:3b\";\n }\n\n return {\n os,\n arch,\n hasRyzenNPU,\n hasAppleSilicon,\n recommendedRunner,\n recommendedModel,\n };\n}\n","import { existsSync } from \"node:fs\";\nimport { resolve } from \"node:path\";\nimport type { Database } from \"../db/types.js\";\nimport { getSetting } from \"../models/settings.js\";\n\nexport interface RepoPaths {\n personal: string | null;\n team: string | null;\n org: string | null;\n}\n\n/**\n * Resolve absolute paths for personal, team, and organization repositories.\n * Personal falls back to personal.workspace_dir if repo.personal is not set.\n */\nexport function getRepoPaths(db: Database): RepoPaths {\n const personalSetting =\n getSetting(db, \"repo.personal\") || getSetting(db, \"personal.workspace_dir\");\n const teamSetting = getSetting(db, \"repo.team\");\n const orgSetting = getSetting(db, \"repo.org\");\n\n return {\n personal: personalSetting ? resolve(personalSetting) : null,\n team: teamSetting ? resolve(teamSetting) : null,\n org: orgSetting ? resolve(orgSetting) : null,\n };\n}\n\n/**\n * Resolve a specific repo's path, or null if not configured.\n */\nexport function resolveRepoPath(\n db: Database,\n type: \"personal\" | \"team\" | \"org\",\n): string | null {\n const paths = getRepoPaths(db);\n return paths[type];\n}\n\n/**\n * Resolve paths to all existing \"/beliefs\" directories in the hierarchy,\n * sorted from most specific (personal) to most general (org).\n */\nexport function resolveAllBeliefPaths(db: Database): string[] {\n const paths = getRepoPaths(db);\n const dirs: string[] = [];\n\n if (paths.personal) {\n const personalDir = resolve(paths.personal, \"beliefs\");\n if (existsSync(personalDir)) dirs.push(personalDir);\n }\n if (paths.team) {\n const teamDir = resolve(paths.team, \"beliefs\");\n if (existsSync(teamDir)) dirs.push(teamDir);\n }\n if (paths.org) {\n const orgDir = resolve(paths.org, \"beliefs\");\n if (existsSync(orgDir)) dirs.push(orgDir);\n }\n\n return dirs;\n}\n\n/**\n * Resolve paths to all existing \"/goals\" directories in the hierarchy,\n * sorted from most specific (personal) to most general (org).\n */\nexport function resolveAllGoalPaths(db: Database): string[] {\n const paths = getRepoPaths(db);\n const dirs: string[] = [];\n\n if (paths.personal) {\n const personalDir = resolve(paths.personal, \"goals\");\n if (existsSync(personalDir)) dirs.push(personalDir);\n }\n if (paths.team) {\n const teamDir = resolve(paths.team, \"goals\");\n if (existsSync(teamDir)) dirs.push(teamDir);\n }\n if (paths.org) {\n const orgDir = resolve(paths.org, \"goals\");\n if (existsSync(orgDir)) dirs.push(orgDir);\n }\n\n return dirs;\n}\n"],"mappings":"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","names":["existsSync","mkdirSync","readFileSync","homedir","dirname","join","join","homedir","require","existsSync","readFileSync","dirname","mkdirSync","existsSync","readFileSync","writeFileSync","join","existsSync","join","readFileSync","t","writeFileSync","ulid","ulid","ulid","ulid","t","existsSync","mkdirSync","readFileSync","homedir","join","join","homedir","existsSync","mkdirSync","readFileSync","ulid","interval","dueAt","ulid","existsSync","readFileSync","dirname","join","anchorIndex","anchor","appendFileSync","existsSync","mkdirSync","readFileSync","writeFileSync","homedir","join","existsSync","homedir","join","execSync","execSync","existsSync","resolve","resolve","existsSync"]}
|
|
1
|
+
{"version":3,"sources":["../src/kernel/analytics/stats.ts","../src/kernel/credentials.ts","../src/kernel/connectors/azure-devops.ts","../src/kernel/db/connection.ts","../src/kernel/db/remote/hrana.ts","../src/kernel/db/remote/provider.ts","../src/kernel/db/schema.ts","../src/kernel/db/sync-adapter.ts","../src/kernel/db/snapshot.ts","../src/kernel/goals/engine.ts","../src/kernel/goals/parser.ts","../src/kernel/models/agent-skill.ts","../src/kernel/models/card.ts","../src/kernel/models/token.ts","../src/kernel/models/prerequisite.ts","../src/kernel/models/review.ts","../src/kernel/models/session.ts","../src/kernel/models/settings.ts","../src/kernel/observation/analyzer.ts","../src/kernel/observation/monitor-io.ts","../src/kernel/observation/session-synthesis.ts","../src/kernel/recall/evaluator.ts","../src/kernel/scheduler/fsrs.ts","../src/kernel/scheduler/blocker.ts","../src/kernel/observation/shell-hooks.ts","../src/kernel/observation/skill-discovery.ts","../src/kernel/recall/actions.ts","../src/kernel/recall/prompter.ts","../src/kernel/recall/reference-resolver.ts","../src/kernel/scheduler/interleaver.ts","../src/kernel/scheduler/queue.ts","../src/kernel/system/hooks.ts","../src/kernel/system/i18n.ts","../src/kernel/system/install-config.ts","../src/kernel/system/installer.ts","../src/kernel/system/locale.ts","../src/kernel/system/profiler.ts","../src/kernel/system/repos.ts","../src/kernel/system/update-check.ts"],"sourcesContent":["/**\n * Learning Analytics\n *\n * Progress statistics, competence tracking, and session summaries.\n * Ported from PoC's `stats` command with additions for FSRS and symbiosis modes.\n */\n\nimport type { Database } from \"../db/types.js\";\n\nexport interface UserStats {\n userId: string;\n totalTokens: number;\n cardsInDeck: number;\n dueToday: number;\n blocked: number;\n mature: number;\n avgStability: number | null;\n totalSessions: number;\n lastSession: string | null;\n}\n\nexport interface DomainCompetence {\n domain: string;\n totalCards: number;\n matureCards: number;\n avgStability: number;\n retentionRate: number;\n suggestedMode: \"shadowing\" | \"copilot\" | \"autonomy\";\n}\n\nasync function q(db: Database, sql: string, ...params: unknown[]) {\n return (await db.prepare(sql).get(...params)) as Record<string, unknown>;\n}\n\nasync function count(\n db: Database,\n sql: string,\n ...params: unknown[]\n): Promise<number> {\n return ((await q(db, sql, ...params)) as { n: number }).n;\n}\n\n/**\n * Get overall learning stats for a user (ported from PoC's `stats` command).\n */\nexport async function getUserStats(\n db: Database,\n userId: string,\n): Promise<UserStats> {\n const avgRow = (await q(\n db,\n \"SELECT AVG(stability) as v FROM cards WHERE user_id = ? AND reps > 0\",\n userId,\n )) as { v: number | null };\n\n const lastSessionRow = (await db\n .prepare(\n \"SELECT started_at FROM sessions WHERE user_id = ? ORDER BY started_at DESC LIMIT 1\",\n )\n .get(userId)) as { started_at: string } | undefined;\n\n return {\n userId,\n totalTokens: await count(db, \"SELECT COUNT(*) as n FROM tokens\"),\n cardsInDeck: await count(\n db,\n \"SELECT COUNT(*) as n FROM cards WHERE user_id = ?\",\n userId,\n ),\n dueToday: await count(\n db,\n \"SELECT COUNT(*) as n FROM cards WHERE user_id = ? AND blocked = 0 AND due_at <= datetime('now')\",\n userId,\n ),\n blocked: await count(\n db,\n \"SELECT COUNT(*) as n FROM cards WHERE user_id = ? AND blocked = 1\",\n userId,\n ),\n mature: await count(\n db,\n \"SELECT COUNT(*) as n FROM cards WHERE user_id = ? AND reps >= 3 AND stability >= 21\",\n userId,\n ),\n avgStability: avgRow.v ? Math.round(avgRow.v * 100) / 100 : null,\n totalSessions: await count(\n db,\n \"SELECT COUNT(*) as n FROM sessions WHERE user_id = ?\",\n userId,\n ),\n lastSession: lastSessionRow?.started_at ?? null,\n };\n}\n\n/**\n * Get competence per domain for a user.\n * Used to suggest symbiosis mode transitions.\n */\nexport async function getDomainCompetence(\n db: Database,\n userId: string,\n): Promise<DomainCompetence[]> {\n const domains = (await db\n .prepare(\n `SELECT DISTINCT t.domain FROM cards c\n JOIN tokens t ON t.id = c.token_id\n WHERE c.user_id = ? AND t.domain != ''`,\n )\n .all(userId)) as { domain: string }[];\n\n const competences: DomainCompetence[] = [];\n for (const d of domains) {\n const total = await count(\n db,\n `SELECT COUNT(*) as n FROM cards c\n JOIN tokens t ON t.id = c.token_id\n WHERE c.user_id = ? AND t.domain = ?`,\n userId,\n d.domain,\n );\n\n const mature = await count(\n db,\n `SELECT COUNT(*) as n FROM cards c\n JOIN tokens t ON t.id = c.token_id\n WHERE c.user_id = ? AND t.domain = ? AND c.reps >= 3 AND c.stability >= 21`,\n userId,\n d.domain,\n );\n\n const avgStab =\n (\n (await q(\n db,\n `SELECT AVG(c.stability) as v FROM cards c\n JOIN tokens t ON t.id = c.token_id\n WHERE c.user_id = ? AND t.domain = ? AND c.reps > 0`,\n userId,\n d.domain,\n )) as { v: number | null }\n ).v ?? 0;\n\n // Estimate retention from review history\n const reviews = (await q(\n db,\n `SELECT COUNT(*) as total,\n SUM(CASE WHEN rating >= 2 THEN 1 ELSE 0 END) as passed\n FROM review_logs\n WHERE user_id = ? AND token_id IN (SELECT id FROM tokens WHERE domain = ?)`,\n userId,\n d.domain,\n )) as { total: number; passed: number };\n\n const retentionRate =\n reviews.total > 0 ? reviews.passed / reviews.total : 0;\n\n let suggestedMode: DomainCompetence[\"suggestedMode\"];\n if (retentionRate > 0.9 && avgStab > 30) {\n suggestedMode = \"autonomy\";\n } else if (retentionRate > 0.7 && avgStab > 7) {\n suggestedMode = \"copilot\";\n } else {\n suggestedMode = \"shadowing\";\n }\n\n competences.push({\n domain: d.domain,\n totalCards: total,\n matureCards: mature,\n avgStability: Math.round(avgStab * 100) / 100,\n retentionRate: Math.round(retentionRate * 1000) / 1000,\n suggestedMode,\n });\n }\n return competences;\n}\n","/**\n * Credential store — reads/writes ~/.zam/credentials.json\n *\n * Connector secrets (Turso URL/token, ADO PAT, etc.) live here instead of\n * inside the SQLite database. This ensures credentials survive db deletion,\n * which is required when migrating from plain SQLite to a libsql embedded\n * replica (Turso cloud sync).\n */\n\nimport {\n chmodSync,\n existsSync,\n mkdirSync,\n readFileSync,\n writeFileSync,\n} from \"node:fs\";\nimport { homedir } from \"node:os\";\nimport { dirname, join } from \"node:path\";\n\nconst DEFAULT_CREDENTIALS_PATH = join(homedir(), \".zam\", \"credentials.json\");\n\nexport interface TursoCredentials {\n url: string;\n token: string;\n /**\n * Database access mode: \"native\" uses the legacy libsql driver, \"remote\"\n * uses the HTTP provider (no native bindings; required on Windows ARM64).\n */\n mode?: \"native\" | \"remote\";\n}\n\nexport interface ADOCredentials {\n org_url: string;\n project: string;\n pat: string;\n}\n\nexport interface Credentials {\n turso?: Partial<TursoCredentials>;\n ado?: Partial<ADOCredentials>;\n}\n\n/** Load credentials from ~/.zam/credentials.json. Returns empty object if missing. */\nexport function loadCredentials(path?: string): Credentials {\n const p = path ?? DEFAULT_CREDENTIALS_PATH;\n if (!existsSync(p)) return {};\n try {\n return JSON.parse(readFileSync(p, \"utf-8\")) as Credentials;\n } catch {\n return {};\n }\n}\n\n/** Save credentials to ~/.zam/credentials.json. */\nexport function saveCredentials(creds: Credentials, path?: string): void {\n const p = path ?? DEFAULT_CREDENTIALS_PATH;\n const dir = dirname(p);\n let createdDirectory = false;\n if (!existsSync(dir)) {\n mkdirSync(dir, { recursive: true, mode: 0o700 });\n createdDirectory = true;\n }\n if (\n process.platform !== \"win32\" &&\n (path === undefined || createdDirectory)\n ) {\n chmodSync(dir, 0o700);\n }\n writeFileSync(p, `${JSON.stringify(creds, null, 2)}\\n`, {\n encoding: \"utf-8\",\n mode: 0o600,\n });\n if (process.platform !== \"win32\") {\n chmodSync(p, 0o600);\n }\n}\n\n/** Get complete Turso credentials, or null if incomplete. */\nexport function getTursoCredentials(path?: string): TursoCredentials | null {\n const creds = loadCredentials(path);\n if (creds.turso?.url && creds.turso?.token) {\n return {\n url: creds.turso.url,\n token: creds.turso.token,\n ...(creds.turso.mode ? { mode: creds.turso.mode } : {}),\n };\n }\n return null;\n}\n\n/** Set Turso credentials. */\nexport function setTursoCredentials(\n url: string,\n token: string,\n path?: string,\n mode?: TursoCredentials[\"mode\"],\n): void {\n const creds = loadCredentials(path);\n creds.turso = { url, token, ...(mode ? { mode } : {}) };\n saveCredentials(creds, path);\n}\n\n/** Clear Turso credentials. */\nexport function clearTursoCredentials(path?: string): void {\n const creds = loadCredentials(path);\n delete creds.turso;\n saveCredentials(creds, path);\n}\n\n/** Get complete ADO credentials, or null if incomplete. */\nexport function getADOCredentials(path?: string): ADOCredentials | null {\n const creds = loadCredentials(path);\n if (creds.ado?.org_url && creds.ado?.project && creds.ado?.pat) {\n return {\n org_url: creds.ado.org_url,\n project: creds.ado.project,\n pat: creds.ado.pat,\n };\n }\n return null;\n}\n\n/** Set ADO credentials. */\nexport function setADOCredentials(\n orgUrl: string,\n project: string,\n pat: string,\n path?: string,\n): void {\n const creds = loadCredentials(path);\n creds.ado = { org_url: orgUrl, project, pat };\n saveCredentials(creds, path);\n}\n\n/** Clear ADO credentials. */\nexport function clearADOCredentials(path?: string): void {\n const creds = loadCredentials(path);\n delete creds.ado;\n saveCredentials(creds, path);\n}\n","/**\n * Azure DevOps connector — fetches work items from ADO boards.\n */\n\nimport { getADOCredentials } from \"../credentials.js\";\n\nexport interface ADOConfig {\n orgUrl: string;\n project: string;\n pat: string;\n}\n\nexport interface WorkItem {\n id: number;\n title: string;\n state: string;\n type: string;\n assignedTo: string;\n}\n\n/** Load ADO config from credentials file. Returns null if not configured. */\nexport function loadADOConfig(): ADOConfig | null {\n const creds = getADOCredentials();\n if (!creds) return null;\n return {\n orgUrl: creds.org_url.replace(/\\/+$/, \"\"),\n project: creds.project,\n pat: creds.pat,\n };\n}\n\nfunction authHeader(pat: string): string {\n return `Basic ${Buffer.from(`:${pat}`).toString(\"base64\")}`;\n}\n\n/**\n * Fetch active work items assigned to the current user.\n * Uses WIQL to query, then batch-fetches work item details.\n */\nexport async function fetchActiveWorkItems(\n config: ADOConfig,\n): Promise<WorkItem[]> {\n const { orgUrl, project, pat } = config;\n\n // Step 1: WIQL query for work item IDs\n const wiqlUrl = `${orgUrl}/${project}/_apis/wit/wiql?api-version=7.1`;\n const wiqlBody = {\n query: `SELECT [System.Id] FROM WorkItems WHERE [System.AssignedTo] = @me AND [System.State] NOT IN ('Closed', 'Completed', 'Done', 'Removed') ORDER BY [Microsoft.VSTS.Common.Priority] ASC, [System.ChangedDate] DESC`,\n };\n\n const wiqlRes = await fetch(wiqlUrl, {\n method: \"POST\",\n headers: {\n Authorization: authHeader(pat),\n \"Content-Type\": \"application/json\",\n },\n body: JSON.stringify(wiqlBody),\n });\n\n if (!wiqlRes.ok) {\n const text = await wiqlRes.text();\n throw new Error(`ADO WIQL query failed (${wiqlRes.status}): ${text}`);\n }\n\n const wiqlData = (await wiqlRes.json()) as { workItems: { id: number }[] };\n const ids = wiqlData.workItems.map((wi) => wi.id);\n\n if (ids.length === 0) return [];\n\n // Step 2: Batch fetch work item details (max 200 per request)\n const batchIds = ids.slice(0, 200);\n const fields =\n \"System.Id,System.Title,System.State,System.WorkItemType,System.AssignedTo\";\n const detailUrl = `${orgUrl}/${project}/_apis/wit/workitems?ids=${batchIds.join(\",\")}&fields=${fields}&api-version=7.1`;\n\n const detailRes = await fetch(detailUrl, {\n headers: { Authorization: authHeader(pat) },\n });\n\n if (!detailRes.ok) {\n const text = await detailRes.text();\n throw new Error(\n `ADO work items fetch failed (${detailRes.status}): ${text}`,\n );\n }\n\n const detailData = (await detailRes.json()) as {\n value: Array<{\n id: number;\n fields: {\n \"System.Title\": string;\n \"System.State\": string;\n \"System.WorkItemType\": string;\n \"System.AssignedTo\"?: { displayName: string };\n };\n }>;\n };\n\n return detailData.value.map((wi) => ({\n id: wi.id,\n title: wi.fields[\"System.Title\"],\n state: wi.fields[\"System.State\"],\n type: wi.fields[\"System.WorkItemType\"],\n assignedTo: wi.fields[\"System.AssignedTo\"]?.displayName ?? \"\",\n }));\n}\n","import { existsSync, mkdirSync, readFileSync, rmSync } from \"node:fs\";\nimport { createRequire } from \"node:module\";\nimport { homedir } from \"node:os\";\nimport { dirname, join } from \"node:path\";\nimport { getTursoCredentials } from \"../credentials.js\";\nimport { openRemoteDatabase } from \"./remote/provider.js\";\nimport { SCHEMA } from \"./schema.js\";\nimport { wrapSyncDatabase } from \"./sync-adapter.js\";\nimport type { Database, SyncDatabase } from \"./types.js\";\n\nconst DEFAULT_DB_DIR = join(homedir(), \".zam\");\nconst DEFAULT_DB_PATH = join(DEFAULT_DB_DIR, \"zam.db\");\nconst require = createRequire(import.meta.url);\n\ntype LibsqlConstructor = new (\n path: string,\n options?: Record<string, unknown>,\n) => SyncDatabase;\n\n/**\n * - `local`: better-sqlite3 file database (default without cloud credentials)\n * - `native`: legacy native libsql driver (remote URLs and embedded replicas)\n * - `remote`: Turso over HTTP, no native bindings (works on Windows ARM64)\n */\nexport type DatabaseProvider = \"local\" | \"native\" | \"remote\";\n\nexport interface ConnectionOptions {\n /** Path to the SQLite database file. Defaults to ~/.zam/zam.db */\n dbPath?: string;\n /** If true, run the schema even when the database already exists. */\n initialize?: boolean;\n /** Turso sync URL for embedded replica mode (e.g. libsql://db-name.turso.io) */\n syncUrl?: string;\n /** Turso auth token for direct remote or embedded replica access */\n authToken?: string;\n /** If false, ignore ~/.zam/credentials.json and force the local/default database. */\n useConfiguredCloud?: boolean;\n /** Explicit provider; overrides ZAM_DB_PROVIDER and the credentials mode. */\n provider?: DatabaseProvider;\n}\n\nfunction isRemoteDatabasePath(dbPath: string): boolean {\n return /^(libsql|https?|wss?):\\/\\//i.test(dbPath);\n}\n\nfunction isDatabaseProvider(value: unknown): value is DatabaseProvider {\n return value === \"local\" || value === \"native\" || value === \"remote\";\n}\n\nfunction openLocalSqlite(dbPath: string): SyncDatabase {\n // Loaded lazily (not as a top-level import) so that remote/HTTP Turso users\n // never trigger the native better-sqlite3 binding. A failure to load that\n // binding inside the packaged desktop app would otherwise crash the whole\n // CLI at startup — before any provider selection or error handling runs.\n const mod = require(\"better-sqlite3\") as\n | (new (\n path: string,\n ) => unknown)\n | { default: new (path: string) => unknown };\n const BetterSqlite3 = (\"default\" in mod ? mod.default : mod) as new (\n path: string,\n ) => unknown;\n return new BetterSqlite3(dbPath) as unknown as SyncDatabase;\n}\n\nfunction loadLibsql(): LibsqlConstructor {\n try {\n const module = require(\"libsql\") as\n | LibsqlConstructor\n | { default: LibsqlConstructor };\n return \"default\" in module ? module.default : module;\n } catch (err) {\n const detail = err instanceof Error ? ` ${err.message}` : \"\";\n throw new Error(\n \"Turso sync requires the optional native libsql backend, which is not \" +\n `available for ${process.platform}/${process.arch}. Switch to the ` +\n \"HTTP provider instead: zam connector setup turso --mode remote \" +\n `(or set ZAM_DB_PROVIDER=remote).${detail}`,\n );\n }\n}\n\n/**\n * Open (or create) the ZAM database.\n * Uses configured Turso credentials for the default database when present.\n * Falls back to local SQLite and WAL mode when no cloud credentials exist.\n * When syncUrl is provided explicitly, enables embedded replica sync with Turso.\n */\nexport async function openDatabase(\n options: ConnectionOptions = {},\n): Promise<Database> {\n const configuredCloud =\n options.useConfiguredCloud !== false && !options.dbPath && !options.syncUrl\n ? getTursoCredentials()\n : null;\n\n let requiresTurso = false;\n try {\n const configPath = join(process.cwd(), \".zam\", \"config.yaml\");\n if (existsSync(configPath)) {\n const configText = readFileSync(configPath, \"utf-8\");\n if (/[\\s\\S]*turso:[\\s\\S]*url:/m.test(configText)) {\n requiresTurso = true;\n }\n }\n } catch (_e) {}\n\n if (\n requiresTurso &&\n !configuredCloud &&\n options.useConfiguredCloud !== false &&\n !options.dbPath &&\n !options.syncUrl\n ) {\n throw new Error(\n \"Turso cloud database is configured in .zam/config.yaml but missing local credentials. Run: zam connector setup turso\",\n );\n }\n const dbPath = configuredCloud?.url ?? options.dbPath ?? DEFAULT_DB_PATH;\n const isRemote = isRemoteDatabasePath(dbPath);\n const isEmbeddedReplica = Boolean(options.syncUrl);\n const provider = resolveProvider(options, configuredCloud?.mode, isRemote);\n const shouldInitialize =\n options.initialize === true ||\n (!isRemote && !isEmbeddedReplica && !existsSync(dbPath));\n\n if (provider === \"remote\") {\n const url = isRemote ? dbPath : options.syncUrl;\n if (!url) {\n throw new Error(\n \"The remote database provider is selected but no Turso URL is \" +\n \"configured. Run: zam connector setup turso\",\n );\n }\n const db = openRemoteDatabase({\n url,\n authToken: configuredCloud?.token ?? options.authToken,\n });\n if (options.initialize) {\n await db.exec(SCHEMA);\n }\n await runMigrations(db);\n return db;\n }\n\n if (shouldInitialize && !isRemote) {\n const dir = dirname(dbPath);\n if (!existsSync(dir)) {\n mkdirSync(dir, { recursive: true });\n }\n }\n\n // Build constructor options for the optional libsql cloud/sync backend.\n const dbOpts: Record<string, unknown> = {};\n if (options.syncUrl) {\n dbOpts.syncUrl = options.syncUrl;\n\n // When syncUrl is provided, the db must be a libsql embedded replica (not\n // plain SQLite). The presence of a companion .meta (or -info) file proves\n // it was created by libsql.\n //\n // If the db exists WITHOUT metadata, it was created before Turso was\n // configured — delete it so libsql can sync fresh from cloud.\n //\n // If metadata exists WITHOUT the db, libsql throws InvalidLocalState —\n // delete the metadata so it can start fresh.\n const metaPath = `${dbPath}.meta`;\n const infoPath = `${dbPath}-info`;\n\n if (existsSync(dbPath) && !existsSync(metaPath) && !existsSync(infoPath)) {\n for (const suffix of [\"\", \"-wal\", \"-shm\"]) {\n const f = `${dbPath}${suffix}`;\n if (existsSync(f)) rmSync(f, { force: true });\n }\n } else if (\n !existsSync(dbPath) &&\n (existsSync(metaPath) || existsSync(infoPath))\n ) {\n if (existsSync(metaPath)) rmSync(metaPath);\n if (existsSync(infoPath)) rmSync(infoPath);\n }\n }\n const authToken = configuredCloud?.token ?? options.authToken;\n if (authToken) {\n dbOpts.authToken = authToken;\n }\n\n let driver: SyncDatabase;\n if (isRemote || isEmbeddedReplica) {\n try {\n const LibsqlDatabase = loadLibsql();\n try {\n driver = new LibsqlDatabase(dbPath, dbOpts);\n } catch (err) {\n const msg = (err as Error).message;\n if (msg.includes(\"InvalidLocalState\") && options.syncUrl) {\n // Last-ditch recovery: metadata is corrupt or mismatched\n const metaPath = `${dbPath}.meta`;\n const infoPath = `${dbPath}-info`;\n if (existsSync(metaPath)) rmSync(metaPath);\n if (existsSync(infoPath)) rmSync(infoPath);\n driver = new LibsqlDatabase(dbPath, dbOpts);\n } else {\n throw err;\n }\n }\n } catch (nativeErr) {\n // The native libsql driver is unavailable or failed to initialise — a\n // failure mode that can occur inside the packaged desktop app. For a pure\n // remote database we transparently fall back to the HTTP provider, which\n // needs no native bindings. Embedded replicas require the native driver,\n // so those still surface the original error.\n const fallbackUrl = isRemote ? dbPath : options.syncUrl;\n if (isRemote && !isEmbeddedReplica && fallbackUrl) {\n const db = openRemoteDatabase({\n url: fallbackUrl,\n authToken: configuredCloud?.token ?? options.authToken,\n });\n if (options.initialize) {\n await db.exec(SCHEMA);\n }\n await runMigrations(db);\n return db;\n }\n throw nativeErr;\n }\n } else {\n driver = openLocalSqlite(dbPath);\n }\n\n // Enable WAL mode and foreign keys for local SQLite.\n // Remote Turso databases and embedded replicas manage their own journaling.\n if (!isRemote && !isEmbeddedReplica) {\n driver.pragma(\"journal_mode = WAL\");\n }\n driver.pragma(\"foreign_keys = ON\");\n if (!isRemote) {\n driver.pragma(\"busy_timeout = 5000\");\n }\n\n const db = wrapSyncDatabase(driver);\n\n // For embedded replicas: sync from cloud FIRST so the local file has the\n // primary's schema before we try to run migrations or create tables.\n if (isEmbeddedReplica) {\n await db.sync?.();\n }\n\n if (shouldInitialize) {\n await db.exec(SCHEMA);\n }\n\n await runMigrations(db);\n\n return db;\n}\n\nfunction resolveProvider(\n options: ConnectionOptions,\n credentialsMode: string | undefined,\n isRemote: boolean,\n): DatabaseProvider {\n if (options.provider) return options.provider;\n const env = process.env.ZAM_DB_PROVIDER;\n if (isDatabaseProvider(env)) return env;\n if (isDatabaseProvider(credentialsMode) && (isRemote || options.syncUrl)) {\n return credentialsMode;\n }\n // Legacy default: cloud URLs and embedded replicas use the native driver.\n if (isRemote || options.syncUrl) return \"native\";\n return \"local\";\n}\n\n/**\n * Open the database with Turso cloud credentials auto-detected.\n * Credentials live in ~/.zam/credentials.json (NOT in the db), so a fresh\n * machine only has to collect missing secrets instead of bootstrapping local\n * state first.\n */\nexport async function openDatabaseWithSync(\n options: Omit<ConnectionOptions, \"syncUrl\" | \"authToken\"> = {},\n): Promise<Database> {\n return openDatabase(options);\n}\n\n/** Get the default database path */\nexport function getDefaultDbPath(): string {\n return DEFAULT_DB_PATH;\n}\n\n/**\n * Run incremental schema migrations on every open.\n * Each migration is idempotent — safe to run repeatedly.\n */\nasync function runMigrations(db: Database): Promise<void> {\n // M001: add execution_context to sessions\n const sessionCols = (await db.pragma(\"table_info(sessions)\")) as Array<{\n name: string;\n }>;\n if (\n sessionCols.length > 0 &&\n !sessionCols.some((c) => c.name === \"execution_context\")\n ) {\n await db.exec(\n `ALTER TABLE sessions ADD COLUMN execution_context TEXT NOT NULL DEFAULT 'shell'`,\n );\n }\n\n // M002: add deprecated_at to tokens\n const tokenCols = (await db.pragma(\"table_info(tokens)\")) as Array<{\n name: string;\n }>;\n if (\n tokenCols.length > 0 &&\n !tokenCols.some((c) => c.name === \"deprecated_at\")\n ) {\n await db.exec(`ALTER TABLE tokens ADD COLUMN deprecated_at TEXT`);\n }\n\n // M004: add source_link to tokens\n if (\n tokenCols.length > 0 &&\n !tokenCols.some((c) => c.name === \"source_link\")\n ) {\n await db.exec(`ALTER TABLE tokens ADD COLUMN source_link TEXT`);\n }\n\n // M005: add question to tokens\n if (tokenCols.length > 0 && !tokenCols.some((c) => c.name === \"question\")) {\n await db.exec(`ALTER TABLE tokens ADD COLUMN question TEXT`);\n }\n\n // M003: create agent_skills table (idempotent via IF NOT EXISTS in SCHEMA,\n // but also needed for databases that skipped the init path)\n await db.exec(`\n CREATE TABLE IF NOT EXISTS agent_skills (\n id TEXT PRIMARY KEY,\n slug TEXT NOT NULL UNIQUE,\n description TEXT NOT NULL,\n steps TEXT NOT NULL DEFAULT '[]',\n token_slugs TEXT NOT NULL DEFAULT '[]',\n source TEXT NOT NULL DEFAULT 'learned',\n created_at TEXT NOT NULL DEFAULT (datetime('now')),\n updated_at TEXT NOT NULL DEFAULT (datetime('now'))\n )\n `);\n\n // M006: persist confirmed monitor-derived ratings for audit and idempotence.\n await db.exec(`\n CREATE TABLE IF NOT EXISTS session_syntheses (\n session_id TEXT NOT NULL REFERENCES sessions(id) ON DELETE CASCADE,\n token_id TEXT NOT NULL REFERENCES tokens(id) ON DELETE CASCADE,\n card_id TEXT NOT NULL REFERENCES cards(id) ON DELETE CASCADE,\n inferred_rating INTEGER NOT NULL CHECK (inferred_rating BETWEEN 1 AND 4),\n confirmed_rating INTEGER NOT NULL CHECK (confirmed_rating BETWEEN 1 AND 4),\n confidence TEXT NOT NULL CHECK (confidence IN ('medium', 'high')),\n evidence TEXT NOT NULL DEFAULT '{}',\n review_log_id TEXT NOT NULL REFERENCES review_logs(id) ON DELETE CASCADE,\n session_step_id TEXT NOT NULL REFERENCES session_steps(id) ON DELETE CASCADE,\n created_at TEXT NOT NULL DEFAULT (datetime('now')),\n PRIMARY KEY (session_id, token_id)\n )\n `);\n}\n","/**\n * Minimal Hrana v3 over HTTP transport for Turso/libsql servers.\n *\n * Implements exactly the subset ZAM needs — execute, sequence, close, and\n * baton-scoped streams for transactions — over plain `fetch`, so it runs on\n * every architecture Node.js supports (including Windows ARM64, which has no\n * native libsql binding).\n *\n * Protocol reference: https://github.com/tursodatabase/libsql/blob/main/docs/HRANA_3_SPEC.md\n */\n\nexport interface HranaTransportOptions {\n /** Database URL (libsql://, https:// or http://). */\n url: string;\n /** Turso auth token; omitted for unauthenticated local servers. */\n authToken?: string;\n /** Per-request timeout in milliseconds. */\n timeoutMs?: number;\n /**\n * Total attempts for requests that failed at the transport level before a\n * response was received. Stateful stream requests (open batons) are never\n * retried.\n */\n maxAttempts?: number;\n}\n\nexport type HranaValue =\n | { type: \"null\" }\n | { type: \"integer\"; value: string }\n | { type: \"float\"; value: number }\n | { type: \"text\"; value: string }\n | { type: \"blob\"; base64: string };\n\nexport interface HranaStmt {\n sql: string;\n args?: HranaValue[];\n want_rows: boolean;\n}\n\nexport type HranaRequest =\n | { type: \"execute\"; stmt: HranaStmt }\n | { type: \"sequence\"; sql: string }\n | { type: \"close\" };\n\nexport interface HranaStmtResult {\n cols: Array<{ name: string | null }>;\n rows: HranaValue[][];\n affected_row_count: number;\n last_insert_rowid: string | null;\n}\n\ninterface HranaPipelineResponse {\n baton: string | null;\n base_url: string | null;\n results: Array<\n | { type: \"ok\"; response: { type: string; result?: HranaStmtResult } }\n | { type: \"error\"; error: { message: string; code?: string | null } }\n >;\n}\n\nconst DEFAULT_TIMEOUT_MS = 15_000;\nconst DEFAULT_MAX_ATTEMPTS = 2;\n\n/** Convert libsql:// and ws(s):// URLs to their HTTP equivalents. */\nexport function toHttpUrl(url: string): string {\n return url\n .replace(/^libsql:\\/\\//i, \"https://\")\n .replace(/^wss:\\/\\//i, \"https://\")\n .replace(/^ws:\\/\\//i, \"http://\")\n .replace(/\\/+$/, \"\");\n}\n\nexport function encodeValue(param: unknown): HranaValue {\n if (param === null) return { type: \"null\" };\n if (typeof param === \"string\") return { type: \"text\", value: param };\n if (typeof param === \"bigint\") {\n return { type: \"integer\", value: param.toString() };\n }\n if (typeof param === \"number\") {\n if (Number.isSafeInteger(param)) {\n return { type: \"integer\", value: param.toString() };\n }\n return { type: \"float\", value: param };\n }\n if (param instanceof Uint8Array) {\n return { type: \"blob\", base64: Buffer.from(param).toString(\"base64\") };\n }\n throw new TypeError(\n `Cannot bind a value of type ${typeof param} to a SQL parameter`,\n );\n}\n\nexport function decodeValue(value: HranaValue): unknown {\n switch (value.type) {\n case \"null\":\n return null;\n case \"integer\":\n return Number(value.value);\n case \"float\":\n return value.value;\n case \"text\":\n return value.value;\n case \"blob\":\n return new Uint8Array(Buffer.from(value.base64, \"base64\"));\n }\n}\n\nexport function rowsToObjects(\n result: HranaStmtResult,\n): Record<string, unknown>[] {\n return result.rows.map((row) => {\n const obj: Record<string, unknown> = {};\n row.forEach((value, i) => {\n obj[result.cols[i]?.name ?? `col${i}`] = decodeValue(value);\n });\n return obj;\n });\n}\n\n/**\n * Only failures that provably happened before the server could have seen the\n * request are retried; anything else (timeouts, resets mid-response) might\n * have executed the statement already, and retrying could duplicate a write.\n */\nfunction isRetryableTransportError(err: unknown): boolean {\n if (!(err instanceof Error) || err.name === \"HranaResponseError\") {\n return false;\n }\n const code = (err.cause as { code?: string } | undefined)?.code;\n return (\n code === \"ECONNREFUSED\" || code === \"ENOTFOUND\" || code === \"EAI_AGAIN\"\n );\n}\n\nclass HranaResponseError extends Error {\n override name = \"HranaResponseError\";\n}\n\n/**\n * A Hrana stream. Stateless requests use a fresh stream per call (the\n * pipeline ends with `close`); transactions keep the stream open via the\n * baton returned by the server.\n */\nexport class HranaTransport {\n private readonly pipelineUrl: string;\n private readonly authToken?: string;\n private readonly timeoutMs: number;\n private readonly maxAttempts: number;\n\n constructor(options: HranaTransportOptions) {\n this.pipelineUrl = `${toHttpUrl(options.url)}/v3/pipeline`;\n this.authToken = options.authToken;\n this.timeoutMs = options.timeoutMs ?? DEFAULT_TIMEOUT_MS;\n this.maxAttempts = options.maxAttempts ?? DEFAULT_MAX_ATTEMPTS;\n }\n\n /**\n * POST one pipeline of requests. Returns the server results plus the baton\n * for continuing an open stream. Retries transport-level failures only for\n * stateless pipelines (no baton involved on either side).\n */\n async pipeline(\n requests: HranaRequest[],\n baton?: string | null,\n baseUrl?: string | null,\n ): Promise<HranaPipelineResponse> {\n const url = baseUrl\n ? `${toHttpUrl(baseUrl)}/v3/pipeline`\n : this.pipelineUrl;\n const keepsState =\n baton != null || !requests.some((r) => r.type === \"close\");\n const attempts = keepsState ? 1 : this.maxAttempts;\n\n let lastError: unknown;\n for (let attempt = 1; attempt <= attempts; attempt++) {\n try {\n return await this.post(url, { baton: baton ?? null, requests });\n } catch (err) {\n lastError = err;\n if (!isRetryableTransportError(err) || attempt === attempts) {\n throw this.offline(err);\n }\n }\n }\n throw this.offline(lastError);\n }\n\n private async post(\n url: string,\n body: unknown,\n ): Promise<HranaPipelineResponse> {\n const controller = new AbortController();\n const timer = setTimeout(() => controller.abort(), this.timeoutMs);\n let response: Response;\n try {\n response = await fetch(url, {\n method: \"POST\",\n headers: {\n \"content-type\": \"application/json\",\n ...(this.authToken\n ? { authorization: `Bearer ${this.authToken}` }\n : {}),\n },\n body: JSON.stringify(body),\n signal: controller.signal,\n });\n } finally {\n clearTimeout(timer);\n }\n\n if (response.status === 401 || response.status === 403) {\n throw new HranaResponseError(\n `Turso rejected the configured credentials (HTTP ${response.status}). ` +\n \"Refresh the token with: zam connector setup turso\",\n );\n }\n if (!response.ok) {\n const detail = await response.text().catch(() => \"\");\n throw new HranaResponseError(\n `Turso request failed with HTTP ${response.status}${detail ? `: ${detail.slice(0, 200)}` : \"\"}`,\n );\n }\n return (await response.json()) as HranaPipelineResponse;\n }\n\n private offline(err: unknown): Error {\n if (err instanceof HranaResponseError) return err;\n const cause =\n err instanceof Error\n ? ((err.cause as Error | undefined)?.message ?? err.message)\n : String(err);\n return new HranaResponseError(\n `Cannot reach the Turso database at ${this.pipelineUrl}: ${cause}. ` +\n \"Check your network connection, or switch to the local provider \" +\n \"(ZAM_DB_PROVIDER=local).\",\n );\n }\n}\n\n/** Unwrap a single pipeline result entry, throwing on stream-level errors. */\nexport function unwrapResult(\n response: HranaPipelineResponse,\n index: number,\n): HranaStmtResult | undefined {\n const entry = response.results[index];\n if (!entry) {\n throw new HranaResponseError(`Turso response is missing result #${index}`);\n }\n if (entry.type === \"error\") {\n throw new Error(entry.error.message);\n }\n return entry.response.result;\n}\n","/**\n * RemoteTursoProvider — implements the async `Database` contract directly\n * against a Turso/libsql server over HTTP (Hrana v3).\n *\n * No native bindings and no extra runtime dependencies, so this provider is\n * the cloud path for architectures without a native libsql artifact (for\n * example Windows ARM64). Under ZAM's online-first assumption it is suitable\n * for interactive sessions: the per-review LLM call dominates latency.\n */\n\nimport type { Database, RunResult, Statement } from \"../types.js\";\nimport {\n encodeValue,\n type HranaRequest,\n type HranaStmtResult,\n HranaTransport,\n type HranaTransportOptions,\n rowsToObjects,\n unwrapResult,\n} from \"./hrana.js\";\n\nconst TABLE_INFO_PRAGMA = /^\\s*table_info\\s*\\(\\s*['\"]?(\\w+)['\"]?\\s*\\)\\s*$/i;\n\nfunction toRunResult(result: HranaStmtResult | undefined): RunResult {\n return {\n changes: result?.affected_row_count ?? 0,\n lastInsertRowid:\n result?.last_insert_rowid != null ? Number(result.last_insert_rowid) : 0,\n };\n}\n\n/** A pipeline executor; stateless by default, baton-scoped in transactions. */\ntype RunPipeline = (\n requests: HranaRequest[],\n) => Promise<{ results: Array<HranaStmtResult | undefined> }>;\n\nfunction makeStatement(sql: string, run: RunPipeline): Statement {\n const execute = async (params: unknown[], wantRows: boolean) => {\n const { results } = await run([\n {\n type: \"execute\",\n stmt: { sql, args: params.map(encodeValue), want_rows: wantRows },\n },\n ]);\n return results[0];\n };\n\n return {\n async run(...params: unknown[]) {\n return toRunResult(await execute(params, false));\n },\n async get(...params: unknown[]) {\n const result = await execute(params, true);\n return result ? rowsToObjects(result)[0] : undefined;\n },\n async all(...params: unknown[]) {\n const result = await execute(params, true);\n return result ? rowsToObjects(result) : [];\n },\n };\n}\n\nfunction makeDatabase(run: RunPipeline, transport: HranaTransport): Database {\n let txTail: Promise<unknown> = Promise.resolve();\n\n const db: Database = {\n prepare(sql: string) {\n return makeStatement(sql, run);\n },\n\n async exec(sql: string) {\n await run([{ type: \"sequence\", sql }]);\n },\n\n async pragma(source: string) {\n const tableInfo = TABLE_INFO_PRAGMA.exec(source);\n const sql = tableInfo\n ? `SELECT * FROM pragma_table_info('${tableInfo[1]}')`\n : `PRAGMA ${source}`;\n return db.prepare(sql).all();\n },\n\n transaction<T>(fn: (db: Database) => Promise<T>): Promise<T> {\n const next = txTail.then(async () => {\n const stream = openStream(transport);\n try {\n await stream.run([\n {\n type: \"execute\",\n stmt: { sql: \"BEGIN IMMEDIATE\", want_rows: false },\n },\n ]);\n const result = await fn(makeDatabase(stream.run, transport));\n await stream.run(\n [{ type: \"execute\", stmt: { sql: \"COMMIT\", want_rows: false } }],\n true,\n );\n return result;\n } catch (err) {\n await stream\n .run(\n [\n {\n type: \"execute\",\n stmt: { sql: \"ROLLBACK\", want_rows: false },\n },\n ],\n true,\n )\n .catch(() => {});\n throw err;\n }\n });\n txTail = next.catch(() => {});\n return next;\n },\n\n async close() {\n // Stateless transport: nothing to release.\n },\n };\n\n return db;\n}\n\ninterface Stream {\n run(requests: HranaRequest[], close?: boolean): ReturnType<RunPipeline>;\n}\n\n/** Open a baton-scoped stream that threads server state between pipelines. */\nfunction openStream(transport: HranaTransport): Stream {\n let baton: string | null = null;\n let baseUrl: string | null = null;\n\n return {\n async run(requests: HranaRequest[], close = false) {\n const sent: HranaRequest[] = close\n ? [...requests, { type: \"close\" }]\n : requests;\n const response = await transport.pipeline(sent, baton, baseUrl);\n baton = response.baton;\n baseUrl = response.base_url ?? baseUrl;\n return {\n results: requests.map((_, i) => unwrapResult(response, i)),\n };\n },\n };\n}\n\nexport type RemoteDatabaseOptions = HranaTransportOptions;\n\n/** Open a remote Turso database over HTTP. */\nexport function openRemoteDatabase(options: RemoteDatabaseOptions): Database {\n const transport = new HranaTransport(options);\n\n const statelessRun: RunPipeline = async (requests) => {\n const response = await transport.pipeline([...requests, { type: \"close\" }]);\n return { results: requests.map((_, i) => unwrapResult(response, i)) };\n };\n\n return makeDatabase(statelessRun, transport);\n}\n","/**\n * ZAM Learning Kernel — SQLite Schema\n *\n * Evolves the PoC's schema with:\n * - FSRS scheduling fields (replaces SM-2's ef/interval_days)\n * - Bloom taxonomy levels on tokens\n * - Symbiosis modes (shadowing/copilot/autonomy)\n * - ULID-based IDs\n * - Immutable review log\n */\n\nexport const SCHEMA = `\n-- PRAGMAs (WAL, foreign_keys) are set programmatically in connection.ts,\n-- not here, because libsql embedded replicas manage their own WAL.\n\n-- Knowledge tokens: atomic concepts/facts with Bloom levels\nCREATE TABLE IF NOT EXISTS tokens (\n id TEXT PRIMARY KEY,\n slug TEXT UNIQUE NOT NULL,\n concept TEXT NOT NULL,\n domain TEXT NOT NULL DEFAULT '',\n bloom_level INTEGER NOT NULL DEFAULT 1 CHECK (bloom_level BETWEEN 1 AND 5),\n context TEXT NOT NULL DEFAULT '',\n symbiosis_mode TEXT CHECK (symbiosis_mode IN ('shadowing', 'copilot', 'autonomy')),\n source_link TEXT,\n created_at TEXT NOT NULL DEFAULT (datetime('now')),\n updated_at TEXT NOT NULL DEFAULT (datetime('now')),\n deprecated_at TEXT,\n question TEXT\n);\n\n-- Prerequisite dependency graph: \"to learn A, first know B\"\nCREATE TABLE IF NOT EXISTS prerequisites (\n token_id TEXT NOT NULL REFERENCES tokens(id) ON DELETE CASCADE,\n requires_id TEXT NOT NULL REFERENCES tokens(id) ON DELETE CASCADE,\n PRIMARY KEY (token_id, requires_id)\n);\n\n-- Per-user scheduling state for each token (FSRS fields)\nCREATE TABLE IF NOT EXISTS cards (\n id TEXT PRIMARY KEY,\n token_id TEXT NOT NULL REFERENCES tokens(id) ON DELETE CASCADE,\n user_id TEXT NOT NULL,\n stability REAL NOT NULL DEFAULT 0.0,\n difficulty REAL NOT NULL DEFAULT 0.5,\n elapsed_days REAL NOT NULL DEFAULT 0.0,\n scheduled_days REAL NOT NULL DEFAULT 0.0,\n reps INTEGER NOT NULL DEFAULT 0,\n lapses INTEGER NOT NULL DEFAULT 0,\n state TEXT NOT NULL DEFAULT 'new' CHECK (state IN ('new', 'learning', 'review', 'relearning')),\n due_at TEXT NOT NULL DEFAULT (datetime('now')),\n last_review_at TEXT,\n blocked INTEGER NOT NULL DEFAULT 0,\n UNIQUE(token_id, user_id)\n);\n\n-- Immutable review log: every rating event\nCREATE TABLE IF NOT EXISTS review_logs (\n id TEXT PRIMARY KEY,\n card_id TEXT NOT NULL REFERENCES cards(id) ON DELETE CASCADE,\n token_id TEXT NOT NULL REFERENCES tokens(id) ON DELETE CASCADE,\n user_id TEXT NOT NULL,\n rating INTEGER NOT NULL CHECK (rating BETWEEN 1 AND 4),\n response_time_ms INTEGER,\n reviewed_at TEXT NOT NULL DEFAULT (datetime('now')),\n scheduled_at TEXT NOT NULL,\n session_id TEXT REFERENCES sessions(id)\n);\n\n-- Work+learning sessions\nCREATE TABLE IF NOT EXISTS sessions (\n id TEXT PRIMARY KEY,\n user_id TEXT NOT NULL,\n task TEXT NOT NULL,\n started_at TEXT NOT NULL DEFAULT (datetime('now')),\n completed_at TEXT\n);\n\n-- Steps within a session: who did what\nCREATE TABLE IF NOT EXISTS session_steps (\n id TEXT PRIMARY KEY,\n session_id TEXT NOT NULL REFERENCES sessions(id) ON DELETE CASCADE,\n token_id TEXT NOT NULL REFERENCES tokens(id) ON DELETE CASCADE,\n done_by TEXT NOT NULL CHECK (done_by IN ('user', 'agent')),\n rating INTEGER CHECK (rating BETWEEN 1 AND 4),\n notes TEXT,\n created_at TEXT NOT NULL DEFAULT (datetime('now'))\n);\n\n-- Confirmed ratings synthesized from monitor evidence.\n-- The composite primary key makes repeated synthesis idempotent per token.\nCREATE TABLE IF NOT EXISTS session_syntheses (\n session_id TEXT NOT NULL REFERENCES sessions(id) ON DELETE CASCADE,\n token_id TEXT NOT NULL REFERENCES tokens(id) ON DELETE CASCADE,\n card_id TEXT NOT NULL REFERENCES cards(id) ON DELETE CASCADE,\n inferred_rating INTEGER NOT NULL CHECK (inferred_rating BETWEEN 1 AND 4),\n confirmed_rating INTEGER NOT NULL CHECK (confirmed_rating BETWEEN 1 AND 4),\n confidence TEXT NOT NULL CHECK (confidence IN ('medium', 'high')),\n evidence TEXT NOT NULL DEFAULT '{}',\n review_log_id TEXT NOT NULL REFERENCES review_logs(id) ON DELETE CASCADE,\n session_step_id TEXT NOT NULL REFERENCES session_steps(id) ON DELETE CASCADE,\n created_at TEXT NOT NULL DEFAULT (datetime('now')),\n PRIMARY KEY (session_id, token_id)\n);\n\n-- User configuration\nCREATE TABLE IF NOT EXISTS user_config (\n key TEXT PRIMARY KEY,\n value TEXT NOT NULL,\n updated_at TEXT NOT NULL DEFAULT (datetime('now'))\n);\n\n-- Agent skills: task recipes the agent learns from user guidance\nCREATE TABLE IF NOT EXISTS agent_skills (\n id TEXT PRIMARY KEY,\n slug TEXT NOT NULL UNIQUE,\n description TEXT NOT NULL,\n steps TEXT NOT NULL DEFAULT '[]', -- JSON array of step strings\n token_slugs TEXT NOT NULL DEFAULT '[]', -- JSON array of related token slugs\n source TEXT NOT NULL DEFAULT 'learned'\n CHECK(source IN ('learned', 'builtin')),\n created_at TEXT NOT NULL DEFAULT (datetime('now')),\n updated_at TEXT NOT NULL DEFAULT (datetime('now'))\n);\n\n-- Performance indexes\nCREATE INDEX IF NOT EXISTS idx_tokens_domain ON tokens(domain);\nCREATE INDEX IF NOT EXISTS idx_tokens_slug ON tokens(slug);\nCREATE INDEX IF NOT EXISTS idx_prereqs_token ON prerequisites(token_id);\nCREATE INDEX IF NOT EXISTS idx_prereqs_requires ON prerequisites(requires_id);\nCREATE INDEX IF NOT EXISTS idx_cards_user_due ON cards(user_id, blocked, due_at);\nCREATE INDEX IF NOT EXISTS idx_cards_token_user ON cards(token_id, user_id);\nCREATE INDEX IF NOT EXISTS idx_review_logs_card ON review_logs(card_id);\nCREATE INDEX IF NOT EXISTS idx_review_logs_user ON review_logs(user_id, reviewed_at);\nCREATE INDEX IF NOT EXISTS idx_session_steps_session ON session_steps(session_id);\n`;\n","/**\n * Adapter that lifts a synchronous SQLite driver (better-sqlite3 or the\n * optional libsql embedded replica) into the async `Database` contract.\n *\n * Statements execute synchronously under the hood, so per-statement atomicity\n * is unchanged. Transactions are serialized through a promise queue because\n * an async callback can yield between statements, and two interleaved\n * BEGIN IMMEDIATE blocks on one connection would otherwise corrupt each\n * other's boundaries.\n */\n\nimport type { Database, SyncDatabase } from \"./types.js\";\n\nexport function wrapSyncDatabase(driver: SyncDatabase): Database {\n let txTail: Promise<unknown> = Promise.resolve();\n\n const db: Database = {\n prepare(sql: string) {\n return {\n async run(...params: unknown[]) {\n return driver.prepare(sql).run(...params);\n },\n async get(...params: unknown[]) {\n return driver.prepare(sql).get(...params);\n },\n async all(...params: unknown[]) {\n return driver.prepare(sql).all(...params);\n },\n };\n },\n\n async exec(sql: string) {\n driver.exec(sql);\n },\n\n async pragma(source: string) {\n return driver.pragma(source);\n },\n\n transaction<T>(fn: (db: Database) => Promise<T>): Promise<T> {\n const run = txTail.then(async () => {\n driver.exec(\"BEGIN IMMEDIATE\");\n try {\n const result = await fn(db);\n driver.exec(\"COMMIT\");\n return result;\n } catch (err) {\n driver.exec(\"ROLLBACK\");\n throw err;\n }\n });\n txTail = run.catch(() => {});\n return run;\n },\n\n ...(driver.sync\n ? {\n async sync() {\n driver.sync?.();\n },\n }\n : {}),\n\n async close() {\n driver.close();\n },\n };\n\n return db;\n}\n","/**\n * Portable database snapshots — Increment 12, Phase 4.\n *\n * A snapshot is portable SQL text: a one-line JSON manifest comment followed by\n * `INSERT` statements for every data row. It deliberately does NOT copy the\n * live WAL database file, so a user can move their learning history between\n * machines through a file-sync folder (Google Drive, OneDrive, iCloud, …)\n * without risking the corruption that comes from syncing an open SQLite/WAL\n * file directly.\n *\n * The schema is NOT embedded. Importing into a freshly initialized database —\n * which always runs the current SCHEMA + migrations on open — keeps snapshots\n * forward compatible across schema changes. Columns are written explicitly so a\n * later-added column never breaks an older snapshot.\n */\n\nimport { createHash } from \"node:crypto\";\nimport type { Database } from \"./types.js\";\n\nexport const SNAPSHOT_FORMAT = \"zam-snapshot\";\nexport const SNAPSHOT_VERSION = 1;\nconst MANIFEST_PREFIX = \"-- zam-snapshot: \";\n\n/**\n * Data tables in foreign-key-safe insertion order (parents before children).\n * Deletes for a `force` restore walk this list in reverse.\n */\nexport const SNAPSHOT_TABLES = [\n \"tokens\",\n \"sessions\",\n \"cards\",\n \"prerequisites\",\n \"session_steps\",\n \"review_logs\",\n \"session_syntheses\",\n \"user_config\",\n \"agent_skills\",\n] as const;\n\nexport interface SnapshotManifest {\n format: string;\n version: number;\n createdAt: string;\n /** Row count per table at export time. */\n tables: Record<string, number>;\n /** SHA-256 of the snapshot body (everything after the manifest line). */\n checksum: string;\n}\n\nexport interface ImportResult {\n /** Row count per table after the restore. */\n tables: Record<string, number>;\n total: number;\n}\n\n/** Render one SQL literal for a value read back from the database. */\nfunction quoteValue(value: unknown): string {\n if (value === null || value === undefined) return \"NULL\";\n if (typeof value === \"number\") {\n return Number.isFinite(value) ? String(value) : \"NULL\";\n }\n if (typeof value === \"bigint\") return value.toString();\n if (typeof value === \"string\") return `'${value.replace(/'/g, \"''\")}'`;\n if (value instanceof Uint8Array) {\n let hex = \"\";\n for (const byte of value) hex += byte.toString(16).padStart(2, \"0\");\n return `X'${hex}'`;\n }\n throw new Error(`Cannot serialize value of type ${typeof value} to SQL`);\n}\n\n/** Column names for a table, in definition order, adapting to migrations. */\nasync function getColumns(db: Database, table: string): Promise<string[]> {\n const cols = (await db.pragma(`table_info(${table})`)) as Array<{\n name: string;\n }>;\n return cols.map((c) => c.name);\n}\n\nasync function countRows(db: Database, table: string): Promise<number> {\n const row = (await db\n .prepare(`SELECT COUNT(*) AS n FROM ${table}`)\n .get()) as { n: number };\n return Number(row.n);\n}\n\n/**\n * Serialize the active database to a portable SQL-text snapshot.\n */\nexport async function exportSnapshot(\n db: Database,\n options: { createdAt?: string } = {},\n): Promise<string> {\n const createdAt = options.createdAt ?? new Date().toISOString();\n const tables: Record<string, number> = {};\n const sections: string[] = [];\n\n for (const table of SNAPSHOT_TABLES) {\n const columns = await getColumns(db, table);\n if (columns.length === 0) {\n // Table absent (older/partial database): record zero and skip.\n tables[table] = 0;\n continue;\n }\n\n const colList = columns.join(\", \");\n const rows = (await db\n .prepare(`SELECT ${colList} FROM ${table}`)\n .all()) as Array<Record<string, unknown>>;\n\n tables[table] = rows.length;\n if (rows.length === 0) continue;\n\n const lines = [`-- ${table} (${rows.length})`];\n for (const row of rows) {\n const values = columns.map((c) => quoteValue(row[c])).join(\", \");\n lines.push(`INSERT INTO ${table} (${colList}) VALUES (${values});`);\n }\n sections.push(lines.join(\"\\n\"));\n }\n\n const body = sections.length > 0 ? `${sections.join(\"\\n\\n\")}\\n` : \"\";\n const checksum = createHash(\"sha256\").update(body).digest(\"hex\");\n const manifest: SnapshotManifest = {\n format: SNAPSHOT_FORMAT,\n version: SNAPSHOT_VERSION,\n createdAt,\n tables,\n checksum,\n };\n\n return `${MANIFEST_PREFIX}${JSON.stringify(manifest)}\\n${body}`;\n}\n\n/** Split a snapshot into its manifest and body; validates the header only. */\nexport function parseSnapshot(snapshot: string): {\n manifest: SnapshotManifest;\n body: string;\n} {\n const newline = snapshot.indexOf(\"\\n\");\n const header = (\n newline === -1 ? snapshot : snapshot.slice(0, newline)\n ).trim();\n const body = newline === -1 ? \"\" : snapshot.slice(newline + 1);\n\n if (!header.startsWith(MANIFEST_PREFIX)) {\n throw new Error(\"Not a ZAM snapshot: missing manifest header.\");\n }\n\n let manifest: SnapshotManifest;\n try {\n manifest = JSON.parse(header.slice(MANIFEST_PREFIX.length));\n } catch {\n throw new Error(\"Snapshot manifest is not valid JSON.\");\n }\n\n if (manifest.format !== SNAPSHOT_FORMAT) {\n throw new Error(`Unsupported snapshot format: ${manifest.format}`);\n }\n if (manifest.version > SNAPSHOT_VERSION) {\n throw new Error(\n `Snapshot version ${manifest.version} is newer than supported ` +\n `(${SNAPSHOT_VERSION}). Upgrade ZAM to import it.`,\n );\n }\n\n return { manifest, body };\n}\n\n/** Parse a snapshot and verify its body checksum. Returns the manifest. */\nexport function verifySnapshot(snapshot: string): SnapshotManifest {\n const { manifest, body } = parseSnapshot(snapshot);\n const actual = createHash(\"sha256\").update(body).digest(\"hex\");\n if (actual !== manifest.checksum) {\n throw new Error(\"Snapshot is corrupted: checksum mismatch.\");\n }\n return manifest;\n}\n\n/**\n * Restore a snapshot into `db`. The database must already carry the current\n * schema (open it with `initialize: true`). Refuses to overwrite a non-empty\n * database unless `force` is set, and verifies row counts inside the\n * transaction so any mismatch rolls the whole restore back.\n */\nexport async function importSnapshot(\n db: Database,\n snapshot: string,\n options: { force?: boolean } = {},\n): Promise<ImportResult> {\n const { manifest, body } = parseSnapshot(snapshot);\n const actual = createHash(\"sha256\").update(body).digest(\"hex\");\n if (actual !== manifest.checksum) {\n throw new Error(\"Snapshot is corrupted: checksum mismatch.\");\n }\n\n return db.transaction(async (tx) => {\n let existing = 0;\n for (const table of SNAPSHOT_TABLES) {\n existing += await countRows(tx, table);\n }\n if (existing > 0 && !options.force) {\n throw new Error(\n `Target database already holds ${existing} row(s). ` +\n \"Pass force to overwrite it.\",\n );\n }\n\n if (options.force) {\n for (const table of [...SNAPSHOT_TABLES].reverse()) {\n await tx.exec(`DELETE FROM ${table};`);\n }\n }\n\n if (body.trim().length > 0) {\n await tx.exec(body);\n }\n\n const tables: Record<string, number> = {};\n let total = 0;\n for (const table of SNAPSHOT_TABLES) {\n const count = await countRows(tx, table);\n tables[table] = count;\n total += count;\n\n const expected = manifest.tables[table] ?? 0;\n if (count !== expected) {\n throw new Error(\n `Restore verification failed for ${table}: ` +\n `expected ${expected} row(s), found ${count}.`,\n );\n }\n }\n\n return { tables, total };\n });\n}\n","/**\n * Goal Engine — manages goal lifecycle via markdown files.\n *\n * Goals live as markdown files in a directory (typically the personal repo's\n * goals/ folder). The engine reads, creates, and updates these files.\n * It does not depend on the database — goals are git-tracked, not DB-tracked.\n */\n\nimport { existsSync, readdirSync, readFileSync, writeFileSync } from \"node:fs\";\nimport { basename, join } from \"node:path\";\nimport type { Goal, GoalStatus } from \"./parser.js\";\nimport {\n extractTasks,\n extractTokenRefs,\n parseGoalFile,\n serializeGoal,\n} from \"./parser.js\";\n\n// ── Types ────────────────────────────────────────────────────────────────────\n\nexport interface GoalSummary {\n slug: string;\n title: string;\n status: GoalStatus;\n parent: string | null;\n taskCount: number;\n tasksDone: number;\n tokenCount: number;\n}\n\nexport interface CreateGoalInput {\n slug: string;\n title: string;\n status?: GoalStatus;\n parent?: string;\n description?: string;\n}\n\n// ── Functions ────────────────────────────────────────────────────────────────\n\n/**\n * List all goals in the goals directory.\n * Returns summaries sorted by status (active first) then title.\n */\nexport function listGoals(goalsDir: string): GoalSummary[] {\n if (!existsSync(goalsDir)) return [];\n\n const files = readdirSync(goalsDir).filter(\n (f) => f.endsWith(\".md\") && f !== \"README.md\",\n );\n\n const summaries: GoalSummary[] = [];\n\n for (const file of files) {\n const filePath = join(goalsDir, file);\n const content = readFileSync(filePath, \"utf-8\");\n const slug = basename(file, \".md\");\n const goal = parseGoalFile(content, slug, filePath);\n const tasks = extractTasks(goal.body);\n const tokens = extractTokenRefs(goal.body);\n\n summaries.push({\n slug: goal.slug,\n title: goal.title,\n status: goal.status,\n parent: goal.parent,\n taskCount: tasks.length,\n tasksDone: tasks.filter((t) => t.done).length,\n tokenCount: tokens.length,\n });\n }\n\n const statusOrder: Record<GoalStatus, number> = {\n active: 0,\n paused: 1,\n completed: 2,\n abandoned: 3,\n };\n\n summaries.sort((a, b) => {\n const statusDiff = statusOrder[a.status] - statusOrder[b.status];\n if (statusDiff !== 0) return statusDiff;\n return a.title.localeCompare(b.title);\n });\n\n return summaries;\n}\n\n/**\n * Get a single goal by slug (filename without .md).\n * Returns undefined if the file doesn't exist.\n */\nexport function getGoal(goalsDir: string, slug: string): Goal | undefined {\n const filePath = join(goalsDir, `${slug}.md`);\n if (!existsSync(filePath)) return undefined;\n\n const content = readFileSync(filePath, \"utf-8\");\n return parseGoalFile(content, slug, filePath);\n}\n\n/**\n * Create a new goal file. Throws if a goal with this slug already exists.\n */\nexport function createGoal(goalsDir: string, input: CreateGoalInput): Goal {\n const filePath = join(goalsDir, `${input.slug}.md`);\n\n if (existsSync(filePath)) {\n throw new Error(`Goal already exists: ${input.slug}`);\n }\n\n const now = new Date().toISOString().slice(0, 10);\n\n const goal: Goal = {\n slug: input.slug,\n title: input.title,\n status: input.status ?? \"active\",\n parent: input.parent ?? null,\n created: now,\n updated: now,\n body: input.description\n ? `## Description\\n${input.description}\\n\\n## Tasks\\n\\n## Tokens`\n : \"## Description\\n\\n## Tasks\\n\\n## Tokens\",\n filePath,\n };\n\n writeFileSync(filePath, serializeGoal(goal), \"utf-8\");\n return goal;\n}\n\n/**\n * Update a goal's status. Writes the updated file back to disk.\n */\nexport function updateGoalStatus(\n goalsDir: string,\n slug: string,\n status: GoalStatus,\n): Goal {\n const goal = getGoal(goalsDir, slug);\n if (!goal) throw new Error(`Goal not found: ${slug}`);\n\n goal.status = status;\n goal.updated = new Date().toISOString().slice(0, 10);\n\n writeFileSync(goal.filePath, serializeGoal(goal), \"utf-8\");\n return goal;\n}\n\n/**\n * Get the goal tree — goals organized by parent relationships.\n * Returns root goals (no parent) with nested children.\n */\nexport function getGoalTree(\n goalsDir: string,\n): Array<GoalSummary & { children: GoalSummary[] }> {\n const all = listGoals(goalsDir);\n const bySlug = new Map(all.map((g) => [g.slug, g]));\n\n const roots: Array<GoalSummary & { children: GoalSummary[] }> = [];\n const children = new Map<string, GoalSummary[]>();\n\n for (const g of all) {\n if (g.parent && bySlug.has(g.parent)) {\n const list = children.get(g.parent) ?? [];\n list.push(g);\n children.set(g.parent, list);\n }\n }\n\n for (const g of all) {\n if (!g.parent || !bySlug.has(g.parent)) {\n roots.push({ ...g, children: children.get(g.slug) ?? [] });\n }\n }\n\n return roots;\n}\n","/**\n * Goal file parser — reads markdown files with YAML-style frontmatter.\n *\n * Goals are persisted as markdown files in the personal repo.\n * Each file has simple key: value frontmatter (no nested structures)\n * and a markdown body with description, tasks, and token references.\n */\n\n// ── Types ────────────────────────────────────────────────────────────────────\n\nexport type GoalStatus = \"active\" | \"completed\" | \"paused\" | \"abandoned\";\n\nexport interface Goal {\n slug: string; // derived from filename (e.g., \"learn-rust\" from \"learn-rust.md\")\n title: string;\n status: GoalStatus;\n parent: string | null; // slug of parent goal\n created: string; // ISO date\n updated: string; // ISO date\n body: string; // markdown body after frontmatter\n filePath: string; // absolute path to the file\n}\n\nexport interface GoalFrontmatter {\n title?: string;\n status?: string;\n parent?: string;\n created?: string;\n updated?: string;\n}\n\n// ── Parser ───────────────────────────────────────────────────────────────────\n\n/**\n * Parse a goal markdown file into a Goal object.\n *\n * Expected format:\n * ```\n * ---\n * title: Learn Rust fundamentals\n * status: active\n * parent: become-systems-programmer\n * created: 2026-03-28\n * updated: 2026-03-28\n * ---\n *\n * ## Description\n * ...\n * ```\n *\n * @param content - Raw file content\n * @param slug - Goal slug (derived from filename by caller)\n * @param filePath - Absolute path to the file\n */\nexport function parseGoalFile(\n content: string,\n slug: string,\n filePath: string,\n): Goal {\n const { frontmatter, body } = splitFrontmatter(content);\n\n const validStatuses: GoalStatus[] = [\n \"active\",\n \"completed\",\n \"paused\",\n \"abandoned\",\n ];\n const status = validStatuses.includes(frontmatter.status as GoalStatus)\n ? (frontmatter.status as GoalStatus)\n : \"active\";\n\n const now = new Date().toISOString().slice(0, 10);\n\n return {\n slug,\n title: frontmatter.title || slug,\n status,\n parent: frontmatter.parent || null,\n created: frontmatter.created || now,\n updated: frontmatter.updated || now,\n body,\n filePath,\n };\n}\n\n/**\n * Serialize a Goal back to markdown with frontmatter.\n */\nexport function serializeGoal(goal: Goal): string {\n const lines = [\"---\", `title: ${goal.title}`, `status: ${goal.status}`];\n\n if (goal.parent) {\n lines.push(`parent: ${goal.parent}`);\n }\n\n lines.push(`created: ${goal.created}`);\n lines.push(`updated: ${goal.updated}`);\n lines.push(\"---\");\n lines.push(\"\");\n\n if (goal.body.trim()) {\n lines.push(goal.body.trim());\n lines.push(\"\");\n }\n\n return lines.join(\"\\n\");\n}\n\n/**\n * Extract tasks (checklist items) from goal body.\n * Returns items like { text: \"Complete Rustlings\", done: false }.\n */\nexport function extractTasks(\n body: string,\n): Array<{ text: string; done: boolean }> {\n const tasks: Array<{ text: string; done: boolean }> = [];\n const taskRegex = /^[-*]\\s+\\[([ xX])\\]\\s+(.+)$/gm;\n let match: RegExpExecArray | null = taskRegex.exec(body);\n\n while (match !== null) {\n tasks.push({\n done: match[1] !== \" \",\n text: match[2].trim(),\n });\n match = taskRegex.exec(body);\n }\n\n return tasks;\n}\n\n/**\n * Extract token references from goal body.\n * Looks for lines like `- token/slug` under a \"## Tokens\" section.\n */\nexport function extractTokenRefs(body: string): string[] {\n const tokensSection = body.match(/## Tokens\\n([\\s\\S]*?)(?=\\n## |\\n*$)/);\n if (!tokensSection) return [];\n\n const refs: string[] = [];\n const lines = tokensSection[1].split(\"\\n\");\n\n for (const line of lines) {\n const match = line.match(/^[-*]\\s+(\\S+)/);\n if (match) {\n refs.push(match[1]);\n }\n }\n\n return refs;\n}\n\n// ── Internal helpers ─────────────────────────────────────────────────────────\n\nfunction splitFrontmatter(content: string): {\n frontmatter: GoalFrontmatter;\n body: string;\n} {\n const trimmed = content.trim();\n\n if (!trimmed.startsWith(\"---\")) {\n return { frontmatter: {}, body: trimmed };\n }\n\n const endIndex = trimmed.indexOf(\"---\", 3);\n if (endIndex === -1) {\n return { frontmatter: {}, body: trimmed };\n }\n\n const fmBlock = trimmed.slice(3, endIndex).trim();\n const body = trimmed.slice(endIndex + 3).trim();\n\n const frontmatter: GoalFrontmatter = {};\n for (const line of fmBlock.split(\"\\n\")) {\n const colonIndex = line.indexOf(\":\");\n if (colonIndex === -1) continue;\n\n const key = line.slice(0, colonIndex).trim();\n const value = line.slice(colonIndex + 1).trim();\n\n if (key && value) {\n (frontmatter as Record<string, string>)[key] = value;\n }\n }\n\n return { frontmatter, body };\n}\n","/**\n * Agent skills: task recipes the agent learns from user guidance.\n *\n * When the agent cannot execute a step, it admits it, asks for guidance,\n * and saves the successful approach here. Skills are linked to tokens so\n * FSRS decay naturally resurfaces them for review — automation ≠ retention.\n */\n\nimport { ulid } from \"ulid\";\nimport type { Database } from \"../db/types.js\";\n\n// ── Types ────────────────────────────────────────────────────────────────────\n\nexport type SkillSource = \"learned\" | \"builtin\";\n\nexport interface AgentSkill {\n id: string;\n slug: string;\n description: string;\n steps: string[]; // parsed from JSON\n token_slugs: string[]; // parsed from JSON\n source: SkillSource;\n created_at: string;\n updated_at: string;\n}\n\n/** Raw DB row — steps and token_slugs are stored as JSON strings */\ninterface AgentSkillRow {\n id: string;\n slug: string;\n description: string;\n steps: string;\n token_slugs: string;\n source: SkillSource;\n created_at: string;\n updated_at: string;\n}\n\nexport interface CreateAgentSkillInput {\n slug: string;\n description: string;\n steps: string[];\n token_slugs?: string[];\n source?: SkillSource;\n}\n\n// ── Helpers ──────────────────────────────────────────────────────────────────\n\nfunction parseRow(row: AgentSkillRow): AgentSkill {\n return {\n ...row,\n steps: JSON.parse(row.steps) as string[],\n token_slugs: JSON.parse(row.token_slugs) as string[],\n };\n}\n\n// ── Functions ────────────────────────────────────────────────────────────────\n\nexport async function createAgentSkill(\n db: Database,\n input: CreateAgentSkillInput,\n): Promise<AgentSkill> {\n const existing = (await db\n .prepare(\"SELECT * FROM agent_skills WHERE slug = ?\")\n .get(input.slug)) as AgentSkillRow | undefined;\n\n if (existing) {\n throw new Error(`Agent skill already exists: ${input.slug}`);\n }\n\n const id = ulid();\n const now = new Date().toISOString();\n\n await db\n .prepare(\n `INSERT INTO agent_skills (id, slug, description, steps, token_slugs, source, created_at, updated_at)\n VALUES (?, ?, ?, ?, ?, ?, ?, ?)`,\n )\n .run(\n id,\n input.slug,\n input.description,\n JSON.stringify(input.steps),\n JSON.stringify(input.token_slugs ?? []),\n input.source ?? \"learned\",\n now,\n now,\n );\n\n return parseRow(\n (await db\n .prepare(\"SELECT * FROM agent_skills WHERE id = ?\")\n .get(id)) as AgentSkillRow,\n );\n}\n\nexport async function getAgentSkill(\n db: Database,\n slug: string,\n): Promise<AgentSkill | undefined> {\n const row = (await db\n .prepare(\"SELECT * FROM agent_skills WHERE slug = ?\")\n .get(slug)) as AgentSkillRow | undefined;\n\n return row ? parseRow(row) : undefined;\n}\n\nexport async function listAgentSkills(db: Database): Promise<AgentSkill[]> {\n const rows = (await db\n .prepare(\"SELECT * FROM agent_skills ORDER BY created_at ASC\")\n .all()) as AgentSkillRow[];\n\n return rows.map(parseRow);\n}\n","/**\n * Card repository — typed wrappers around the cards table.\n *\n * Each card tracks one user's scheduling state for one token,\n * using FSRS fields (stability, difficulty, elapsed_days, etc.).\n */\n\nimport { ulid } from \"ulid\";\nimport type { Database } from \"../db/types.js\";\n\n// ── Types ────────────────────────────────────────────────────────────────────\n\nexport type CardState = \"new\" | \"learning\" | \"review\" | \"relearning\";\n\nexport interface Card {\n id: string;\n token_id: string;\n user_id: string;\n stability: number;\n difficulty: number;\n elapsed_days: number;\n scheduled_days: number;\n reps: number;\n lapses: number;\n state: CardState;\n due_at: string;\n last_review_at: string | null;\n blocked: number; // 0 or 1\n}\n\nexport interface UpdateCardInput {\n stability?: number;\n difficulty?: number;\n elapsed_days?: number;\n scheduled_days?: number;\n reps?: number;\n lapses?: number;\n state?: CardState;\n due_at?: string;\n last_review_at?: string | null;\n blocked?: number;\n}\n\nexport interface CardDeletionImpact {\n review_logs: number;\n}\n\nexport interface DeleteCardResult {\n card: Card;\n impact: CardDeletionImpact;\n}\n\n/** A due card joined with its token details. */\nexport interface DueCard extends Card {\n slug: string;\n concept: string;\n domain: string;\n bloom_level: number;\n}\n\n/** A blocked card joined with its token details. */\nexport interface BlockedCard extends Card {\n slug: string;\n concept: string;\n domain: string;\n bloom_level: number;\n}\n\n// ── Functions ────────────────────────────────────────────────────────────────\n\n/**\n * Ensure a card exists for the given token+user pair.\n *\n * If one already exists, return it. Otherwise create a new card with\n * default FSRS values (due immediately) and return it.\n *\n * Ported from the PoC's ensureCard helper.\n */\nexport async function ensureCard(\n db: Database,\n tokenId: string,\n userId: string,\n): Promise<Card> {\n const existing = (await db\n .prepare(\"SELECT * FROM cards WHERE token_id = ? AND user_id = ?\")\n .get(tokenId, userId)) as Card | undefined;\n\n if (existing) return existing;\n\n const id = ulid();\n const now = new Date().toISOString();\n\n await db\n .prepare(\n `INSERT INTO cards (id, token_id, user_id, due_at)\n VALUES (?, ?, ?, ?)`,\n )\n .run(id, tokenId, userId, now);\n\n return (await db.prepare(\"SELECT * FROM cards WHERE id = ?\").get(id)) as Card;\n}\n\n/**\n * Get a card by token+user. Returns undefined if no card exists.\n */\nexport async function getCard(\n db: Database,\n tokenId: string,\n userId: string,\n): Promise<Card | undefined> {\n return (await db\n .prepare(\"SELECT * FROM cards WHERE token_id = ? AND user_id = ?\")\n .get(tokenId, userId)) as Card | undefined;\n}\n\n/**\n * Get a card by its ULID.\n */\nexport async function getCardById(\n db: Database,\n cardId: string,\n): Promise<Card | undefined> {\n return (await db.prepare(\"SELECT * FROM cards WHERE id = ?\").get(cardId)) as\n | Card\n | undefined;\n}\n\n/**\n * Update a card's scheduling fields.\n *\n * Only the fields present in `updates` are changed. Throws if the card\n * does not exist.\n */\nexport async function updateCard(\n db: Database,\n cardId: string,\n updates: UpdateCardInput,\n): Promise<Card> {\n const fields: string[] = [];\n const values: unknown[] = [];\n\n if (updates.stability !== undefined) {\n fields.push(\"stability = ?\");\n values.push(updates.stability);\n }\n if (updates.difficulty !== undefined) {\n fields.push(\"difficulty = ?\");\n values.push(updates.difficulty);\n }\n if (updates.elapsed_days !== undefined) {\n fields.push(\"elapsed_days = ?\");\n values.push(updates.elapsed_days);\n }\n if (updates.scheduled_days !== undefined) {\n fields.push(\"scheduled_days = ?\");\n values.push(updates.scheduled_days);\n }\n if (updates.reps !== undefined) {\n fields.push(\"reps = ?\");\n values.push(updates.reps);\n }\n if (updates.lapses !== undefined) {\n fields.push(\"lapses = ?\");\n values.push(updates.lapses);\n }\n if (updates.state !== undefined) {\n fields.push(\"state = ?\");\n values.push(updates.state);\n }\n if (updates.due_at !== undefined) {\n fields.push(\"due_at = ?\");\n values.push(updates.due_at);\n }\n if (updates.last_review_at !== undefined) {\n fields.push(\"last_review_at = ?\");\n values.push(updates.last_review_at);\n }\n if (updates.blocked !== undefined) {\n fields.push(\"blocked = ?\");\n values.push(updates.blocked);\n }\n\n if (fields.length === 0) {\n throw new Error(\"updateCard called with no fields to update\");\n }\n\n values.push(cardId);\n\n const result = await db\n .prepare(`UPDATE cards SET ${fields.join(\", \")} WHERE id = ?`)\n .run(...values);\n\n if (result.changes === 0) {\n throw new Error(`Card not found: ${cardId}`);\n }\n\n return (await db\n .prepare(\"SELECT * FROM cards WHERE id = ?\")\n .get(cardId)) as Card;\n}\n\n/**\n * Preview the review-log rows that will be removed when deleting a user's card.\n */\nexport async function getCardDeletionImpact(\n db: Database,\n tokenId: string,\n userId: string,\n): Promise<CardDeletionImpact> {\n const card = await getCard(db, tokenId, userId);\n if (!card) {\n throw new Error(`Card not found for token ${tokenId} and user ${userId}`);\n }\n\n const reviewLogs = (await db\n .prepare(\"SELECT COUNT(*) AS n FROM review_logs WHERE card_id = ?\")\n .get(card.id)) as { n: number };\n\n return { review_logs: reviewLogs.n };\n}\n\n/**\n * Delete one user's card for a token. Review logs cascade via FK.\n */\nexport async function deleteCardForUser(\n db: Database,\n tokenId: string,\n userId: string,\n): Promise<DeleteCardResult> {\n const card = await getCard(db, tokenId, userId);\n if (!card) {\n throw new Error(`Card not found for token ${tokenId} and user ${userId}`);\n }\n\n const impact = await getCardDeletionImpact(db, tokenId, userId);\n await db.prepare(\"DELETE FROM cards WHERE id = ?\").run(card.id);\n\n return { card, impact };\n}\n\n/**\n * Get all cards that are due for review.\n *\n * A card is due when it is not blocked and due_at <= now.\n * Results are ordered by bloom_level ascending (fundamentals first),\n * then by due_at ascending (oldest first).\n *\n * Ported from the PoC's due-tokens command.\n */\nexport async function getDueCards(\n db: Database,\n userId: string,\n now?: string,\n): Promise<DueCard[]> {\n const cutoff = now ?? new Date().toISOString();\n\n return (await db\n .prepare(\n `SELECT c.*, t.slug, t.concept, t.domain, t.bloom_level\n FROM cards c\n JOIN tokens t ON t.id = c.token_id\n WHERE c.user_id = ? AND c.blocked = 0 AND c.due_at <= ?\n ORDER BY t.bloom_level ASC, c.due_at ASC`,\n )\n .all(userId, cutoff)) as DueCard[];\n}\n\n/**\n * Get all blocked cards for a user.\n *\n * Returns cards joined with their token details so the caller can\n * see what is waiting and why.\n */\nexport async function getBlockedCards(\n db: Database,\n userId: string,\n): Promise<BlockedCard[]> {\n return (await db\n .prepare(\n `SELECT c.*, t.slug, t.concept, t.domain, t.bloom_level\n FROM cards c\n JOIN tokens t ON t.id = c.token_id\n WHERE c.user_id = ? AND c.blocked = 1\n ORDER BY t.bloom_level ASC, t.slug ASC`,\n )\n .all(userId)) as BlockedCard[];\n}\n","/**\n * Token repository — typed wrappers around the tokens table.\n *\n * Tokens are atomic knowledge concepts with Bloom taxonomy levels\n * and optional symbiosis modes (shadowing / copilot / autonomy).\n */\n\nimport { ulid } from \"ulid\";\nimport type { Database } from \"../db/types.js\";\n\n// ── Types ────────────────────────────────────────────────────────────────────\n\nexport type BloomLevel = 1 | 2 | 3 | 4 | 5;\n\nexport type SymbiosisMode = \"shadowing\" | \"copilot\" | \"autonomy\";\n\nexport interface Token {\n id: string;\n slug: string;\n concept: string;\n domain: string;\n bloom_level: BloomLevel;\n context: string;\n symbiosis_mode: SymbiosisMode | null;\n source_link: string | null;\n question: string | null;\n created_at: string;\n updated_at: string;\n deprecated_at: string | null;\n}\n\nexport interface CreateTokenInput {\n slug: string;\n concept: string;\n domain?: string;\n bloom_level?: BloomLevel;\n context?: string;\n symbiosis_mode?: SymbiosisMode | null;\n source_link?: string | null;\n question?: string | null;\n}\n\nexport interface UpdateTokenInput {\n concept?: string;\n domain?: string;\n bloom_level?: BloomLevel;\n context?: string;\n symbiosis_mode?: SymbiosisMode | null;\n source_link?: string | null;\n question?: string | null;\n}\n\nexport interface ListTokensOptions {\n domain?: string;\n}\n\nexport interface TokenDeleteImpact {\n cards: number;\n review_logs: number;\n prerequisite_edges_from_token: number;\n prerequisite_edges_to_token: number;\n session_steps: number;\n sessions_touched: number;\n agent_skills: number;\n}\n\nexport interface DeleteTokenResult {\n token: Token;\n impact: TokenDeleteImpact;\n}\n\n// ── Scored result from fuzzy search ──────────────────────────────────────────\n\nexport interface ScoredToken extends Token {\n score: number;\n}\n\n// ── Functions ────────────────────────────────────────────────────────────────\n\n/**\n * Create a new knowledge token.\n * Throws if a token with the same slug already exists.\n */\nexport async function createToken(\n db: Database,\n input: CreateTokenInput,\n): Promise<Token> {\n const id = ulid();\n const now = new Date().toISOString();\n\n const bloom = input.bloom_level ?? 1;\n if (bloom < 1 || bloom > 5) {\n throw new Error(`bloom_level must be between 1 and 5, got ${bloom}`);\n }\n\n await db\n .prepare(`\n INSERT INTO tokens (id, slug, concept, domain, bloom_level, context, symbiosis_mode, source_link, question, created_at, updated_at)\n VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)\n `)\n .run(\n id,\n input.slug,\n input.concept,\n input.domain ?? \"\",\n bloom,\n input.context ?? \"\",\n input.symbiosis_mode ?? null,\n input.source_link ?? null,\n input.question ?? null,\n now,\n now,\n );\n\n return (await getTokenById(db, id)) as Token;\n}\n\n/**\n * Look up a token by its unique slug.\n * Returns undefined if not found.\n */\nexport async function getTokenBySlug(\n db: Database,\n slug: string,\n): Promise<Token | undefined> {\n return (await db.prepare(\"SELECT * FROM tokens WHERE slug = ?\").get(slug)) as\n | Token\n | undefined;\n}\n\n/**\n * Look up a token by its ULID.\n * Returns undefined if not found.\n */\nexport async function getTokenById(\n db: Database,\n id: string,\n): Promise<Token | undefined> {\n return (await db.prepare(\"SELECT * FROM tokens WHERE id = ?\").get(id)) as\n | Token\n | undefined;\n}\n\n/**\n * Update mutable fields on a token.\n *\n * Slug is intentionally immutable in v1 because it is referenced by other\n * parts of the system (for example agent skill metadata).\n */\nexport async function updateToken(\n db: Database,\n slug: string,\n updates: UpdateTokenInput,\n): Promise<Token> {\n const token = await getTokenBySlug(db, slug);\n if (!token) {\n throw new Error(`Token not found: ${slug}`);\n }\n\n const fields: string[] = [];\n const values: unknown[] = [];\n\n if (updates.concept !== undefined) {\n fields.push(\"concept = ?\");\n values.push(updates.concept);\n }\n if (updates.domain !== undefined) {\n fields.push(\"domain = ?\");\n values.push(updates.domain);\n }\n if (updates.bloom_level !== undefined) {\n if (updates.bloom_level < 1 || updates.bloom_level > 5) {\n throw new Error(\n `bloom_level must be between 1 and 5, got ${updates.bloom_level}`,\n );\n }\n fields.push(\"bloom_level = ?\");\n values.push(updates.bloom_level);\n }\n if (updates.context !== undefined) {\n fields.push(\"context = ?\");\n values.push(updates.context);\n }\n if (updates.symbiosis_mode !== undefined) {\n const validModes = [\"shadowing\", \"copilot\", \"autonomy\"];\n if (\n updates.symbiosis_mode !== null &&\n !validModes.includes(updates.symbiosis_mode)\n ) {\n throw new Error(`Invalid symbiosis_mode: ${updates.symbiosis_mode}`);\n }\n fields.push(\"symbiosis_mode = ?\");\n values.push(updates.symbiosis_mode);\n }\n if (updates.source_link !== undefined) {\n fields.push(\"source_link = ?\");\n values.push(updates.source_link);\n }\n if (updates.question !== undefined) {\n fields.push(\"question = ?\");\n values.push(updates.question);\n }\n\n if (fields.length === 0) {\n throw new Error(\"updateToken called with no fields to update\");\n }\n\n fields.push(\"updated_at = ?\");\n values.push(new Date().toISOString());\n values.push(slug);\n\n await db\n .prepare(`UPDATE tokens SET ${fields.join(\", \")} WHERE slug = ?`)\n .run(...values);\n return (await getTokenBySlug(db, slug)) as Token;\n}\n\n/**\n * Mark a token as deprecated. Deprecated tokens are excluded from review queues\n * and search results but are not deleted — they can still be consulted.\n *\n * Throws if the token does not exist or is already deprecated.\n */\nexport async function deprecateToken(\n db: Database,\n slug: string,\n): Promise<Token> {\n const token = await getTokenBySlug(db, slug);\n if (!token) {\n throw new Error(`Token not found: ${slug}`);\n }\n if (token.deprecated_at) {\n throw new Error(`Token already deprecated: ${slug}`);\n }\n\n const now = new Date().toISOString();\n await db\n .prepare(\n \"UPDATE tokens SET deprecated_at = ?, updated_at = ? WHERE slug = ?\",\n )\n .run(now, now, slug);\n\n return (await getTokenBySlug(db, slug)) as Token;\n}\n\n/**\n * Preview the rows that will be removed or updated when deleting a token.\n */\nexport async function getTokenDeleteImpact(\n db: Database,\n slug: string,\n): Promise<TokenDeleteImpact> {\n const token = await getTokenBySlug(db, slug);\n if (!token) {\n throw new Error(`Token not found: ${slug}`);\n }\n\n const cards = (await db\n .prepare(\"SELECT COUNT(*) AS n FROM cards WHERE token_id = ?\")\n .get(token.id)) as { n: number };\n const reviewLogs = (await db\n .prepare(\"SELECT COUNT(*) AS n FROM review_logs WHERE token_id = ?\")\n .get(token.id)) as { n: number };\n const prereqsFrom = (await db\n .prepare(\"SELECT COUNT(*) AS n FROM prerequisites WHERE token_id = ?\")\n .get(token.id)) as { n: number };\n const prereqsTo = (await db\n .prepare(\"SELECT COUNT(*) AS n FROM prerequisites WHERE requires_id = ?\")\n .get(token.id)) as { n: number };\n const sessionSteps = (await db\n .prepare(\"SELECT COUNT(*) AS n FROM session_steps WHERE token_id = ?\")\n .get(token.id)) as { n: number };\n const sessionsTouched = (await db\n .prepare(\n \"SELECT COUNT(DISTINCT session_id) AS n FROM session_steps WHERE token_id = ?\",\n )\n .get(token.id)) as { n: number };\n\n const skillRows = (await db\n .prepare(\"SELECT token_slugs FROM agent_skills\")\n .all()) as Array<{ token_slugs: string }>;\n const agentSkills = skillRows.filter((row) => {\n const tokenSlugs = JSON.parse(row.token_slugs) as string[];\n return tokenSlugs.includes(slug);\n }).length;\n\n return {\n cards: cards.n,\n review_logs: reviewLogs.n,\n prerequisite_edges_from_token: prereqsFrom.n,\n prerequisite_edges_to_token: prereqsTo.n,\n session_steps: sessionSteps.n,\n sessions_touched: sessionsTouched.n,\n agent_skills: agentSkills,\n };\n}\n\n/**\n * Hard-delete a token and clean up non-FK references that point at its slug.\n */\nexport async function deleteToken(\n db: Database,\n slug: string,\n): Promise<DeleteTokenResult> {\n const token = await getTokenBySlug(db, slug);\n if (!token) {\n throw new Error(`Token not found: ${slug}`);\n }\n\n const impact = await getTokenDeleteImpact(db, slug);\n\n await db.transaction(async (tx) => {\n const now = new Date().toISOString();\n const skillRows = (await tx\n .prepare(\"SELECT id, token_slugs FROM agent_skills\")\n .all()) as Array<{ id: string; token_slugs: string }>;\n\n for (const row of skillRows) {\n const tokenSlugs = JSON.parse(row.token_slugs) as string[];\n const filtered = tokenSlugs.filter((tokenSlug) => tokenSlug !== slug);\n if (filtered.length !== tokenSlugs.length) {\n await tx\n .prepare(\n \"UPDATE agent_skills SET token_slugs = ?, updated_at = ? WHERE id = ?\",\n )\n .run(JSON.stringify(filtered), now, row.id);\n }\n }\n\n await tx.prepare(\"DELETE FROM tokens WHERE id = ?\").run(token.id);\n });\n\n return { token, impact };\n}\n\n/**\n * Fuzzy search for tokens by keyword query.\n *\n * Uses SQLite LIKE queries on slug, concept, and domain to avoid loading\n * every non-deprecated token into memory. Each search term runs its own\n * LIKE query; results are aggregated in JS with a word-overlap score plus\n * a substring bonus on the concept field. Results are returned sorted by\n * relevance score descending.\n *\n * For very small search terms (< 3 chars) a light in-memory fallback is\n * used to avoid matching every token.\n */\nexport async function findTokens(\n db: Database,\n query: string,\n): Promise<ScoredToken[]> {\n const normalised = query.toLowerCase();\n const searchTokens = normalised\n .split(/[\\s,.\\-_/\\\\:;!?()[\\]{}]+/)\n .filter((t) => t.length > 0);\n\n if (searchTokens.length === 0) return [];\n\n // Short terms: fall back to in-memory scan (cheap — few tokens match anyway)\n const shortTerms = searchTokens.filter((t) => t.length <= 2);\n const longTerms = searchTokens.filter((t) => t.length > 2);\n\n const scoreMap = new Map<string, { token: Token; score: number }>();\n\n // Per-term SQL LIKE queries for each substantive search token.\n const likeSQL =\n `SELECT * FROM tokens WHERE deprecated_at IS NULL AND ` +\n `(lower(slug) LIKE ? OR lower(concept) LIKE ? OR lower(domain) LIKE ?)`;\n\n for (const term of longTerms) {\n const pattern = `%${term}%`;\n const rows = (await db\n .prepare(likeSQL)\n .all(pattern, pattern, pattern)) as Token[];\n for (const row of rows) {\n const entry = scoreMap.get(row.id);\n if (entry) {\n entry.score++;\n } else {\n scoreMap.set(row.id, { token: row, score: 1 });\n }\n }\n }\n\n // If there were short terms, or all terms were short, scan in-memory.\n if (shortTerms.length > 0 || longTerms.length === 0) {\n const allTokens = (await db\n .prepare(\"SELECT * FROM tokens WHERE deprecated_at IS NULL\")\n .all()) as Token[];\n\n for (const token of allTokens) {\n const words = `${token.slug} ${token.concept} ${token.domain}`\n .toLowerCase()\n .split(/[\\s,.\\-_/\\\\:;!?()[\\]{}]+/)\n .filter(Boolean);\n\n let matchCount = 0;\n for (const term of shortTerms.length > 0 ? shortTerms : searchTokens) {\n for (const w of words) {\n if (w === term) matchCount++;\n }\n }\n\n if (matchCount > 0) {\n const entry = scoreMap.get(token.id);\n if (entry) {\n entry.score += matchCount;\n } else {\n scoreMap.set(token.id, { token, score: matchCount });\n }\n }\n }\n }\n\n // Apply substring bonus and build result.\n const scored: ScoredToken[] = [];\n for (const { token, score } of scoreMap.values()) {\n let finalScore = score;\n if (token.concept.toLowerCase().includes(normalised.slice(0, 25))) {\n finalScore += 3;\n }\n scored.push({ score: finalScore, ...token });\n }\n\n scored.sort((a, b) => b.score - a.score);\n return scored;\n}\n\n/**\n * List all tokens, optionally filtered by domain.\n * Results are ordered by bloom_level then slug.\n */\nexport async function listTokens(\n db: Database,\n options?: ListTokensOptions,\n): Promise<Token[]> {\n if (options?.domain) {\n return (await db\n .prepare(\n \"SELECT * FROM tokens WHERE domain = ? AND deprecated_at IS NULL ORDER BY bloom_level, slug\",\n )\n .all(options.domain)) as Token[];\n }\n return (await db\n .prepare(\n \"SELECT * FROM tokens WHERE deprecated_at IS NULL ORDER BY bloom_level, domain, slug\",\n )\n .all()) as Token[];\n}\n","/**\n * Prerequisite repository — typed wrappers around the prerequisites table.\n *\n * Models the dependency graph: \"to learn token A, first know token B.\"\n * The graph must remain acyclic — cycles are rejected at insert time.\n */\n\nimport type { Database } from \"../db/types.js\";\nimport { type Card, type CardState, getCard } from \"./card.js\";\nimport { getTokenById } from \"./token.js\";\n\n// ── Types ────────────────────────────────────────────────────────────────────\n\nexport interface Prerequisite {\n token_id: string;\n requires_id: string;\n}\n\n/** A prerequisite row joined with the token it points to. */\nexport interface PrerequisiteWithToken extends Prerequisite {\n slug: string;\n concept: string;\n domain: string;\n bloom_level: number;\n}\n\n// ── Helpers ──────────────────────────────────────────────────────────────────\n\n/**\n * Collect all prerequisite edges as an adjacency map: child → parent set.\n * Only used for cycle detection; the full graph is loaded once per\n * addPrerequisite call (small N in practice).\n */\nasync function buildAncestorMap(\n db: Database,\n): Promise<Map<string, Set<string>>> {\n const rows = (await db\n .prepare(\"SELECT token_id, requires_id FROM prerequisites\")\n .all()) as Array<{ token_id: string; requires_id: string }>;\n const map = new Map<string, Set<string>>();\n for (const row of rows) {\n let ancestors = map.get(row.token_id);\n if (!ancestors) {\n ancestors = new Set();\n map.set(row.token_id, ancestors);\n }\n ancestors.add(row.requires_id);\n }\n return map;\n}\n\n/**\n * Returns true if adding edge (tokenId → requiresId) would create a cycle.\n * Uses BFS from requiresId: if tokenId is reachable, adding the edge closes\n * a loop.\n */\nexport async function wouldCreateCycle(\n db: Database,\n tokenId: string,\n requiresId: string,\n): Promise<boolean> {\n if (tokenId === requiresId) return true;\n\n const ancestors = await buildAncestorMap(db);\n const visited = new Set<string>();\n const queue = [requiresId];\n\n while (queue.length > 0) {\n const current = queue.shift() as string;\n if (current === tokenId) return true;\n if (visited.has(current)) continue;\n visited.add(current);\n\n const parents = ancestors.get(current);\n if (parents) {\n for (const parent of parents) {\n if (!visited.has(parent)) queue.push(parent);\n }\n }\n }\n return false;\n}\n\n// ── Functions ────────────────────────────────────────────────────────────────\n\n/**\n * Add a prerequisite edge: tokenId requires requiresId.\n *\n * Idempotent — silently ignores duplicate edges.\n * Throws if either token ID does not exist (FK constraint).\n * Throws if a token is declared as its own prerequisite.\n * Throws if the edge would create a cycle in the prerequisite graph.\n */\nexport async function addPrerequisite(\n db: Database,\n tokenId: string,\n requiresId: string,\n): Promise<void> {\n if (tokenId === requiresId) {\n throw new Error(\"A token cannot be a prerequisite of itself\");\n }\n\n if (await wouldCreateCycle(db, tokenId, requiresId)) {\n throw new Error(\n `Cannot add prerequisite: would create a cycle. ` +\n `${requiresId} already depends on ${tokenId} (directly or transitively).`,\n );\n }\n\n await db\n .prepare(\n \"INSERT OR IGNORE INTO prerequisites (token_id, requires_id) VALUES (?, ?)\",\n )\n .run(tokenId, requiresId);\n}\n\n/**\n * Get the direct prerequisites of a token — \"what does token X require?\"\n *\n * Returns prerequisite rows joined with the required token's details.\n */\nexport async function getPrerequisites(\n db: Database,\n tokenId: string,\n): Promise<PrerequisiteWithToken[]> {\n return (await db\n .prepare(\n `SELECT p.token_id, p.requires_id, t.slug, t.concept, t.domain, t.bloom_level\n FROM prerequisites p\n JOIN tokens t ON t.id = p.requires_id\n WHERE p.token_id = ?`,\n )\n .all(tokenId)) as PrerequisiteWithToken[];\n}\n\n/**\n * Get the direct dependents of a token — \"what depends on token X?\"\n *\n * Returns prerequisite rows joined with the dependent token's details.\n */\nexport async function getDependents(\n db: Database,\n tokenId: string,\n): Promise<PrerequisiteWithToken[]> {\n return (await db\n .prepare(\n `SELECT p.token_id, p.requires_id, t.slug, t.concept, t.domain, t.bloom_level\n FROM prerequisites p\n JOIN tokens t ON t.id = p.token_id\n WHERE p.requires_id = ?`,\n )\n .all(tokenId)) as PrerequisiteWithToken[];\n}\n\n// ── Visualization Neighborhood (for 3D focus graph) ──────────────────────────\n\n/** Token + optional per-user card snapshot, tailored for visual encoding (mastery, blocked state, bloom). */\nexport interface NeighborhoodToken {\n id: string;\n slug: string;\n concept: string;\n domain: string;\n bloom_level: number;\n card: {\n state: CardState;\n reps: number;\n stability: number;\n difficulty: number;\n blocked: boolean;\n due_at: string;\n last_review_at: string | null;\n } | null;\n}\n\n/**\n * The direct neighborhood for a focus-centric 3D view:\n * - center: the token in focus\n * - prerequisites: direct \"basis\" tokens required by the center (foundations, placed \"below\")\n * - dependents: direct tokens that require the center (higher-order abilities, placed \"above\")\n *\n * When userId is supplied, every node includes the user's Card state so the viz can\n * encode personal mastery (e.g. color by stability/reps, highlight blocked or due).\n */\nexport interface Neighborhood {\n center: NeighborhoodToken;\n prerequisites: NeighborhoodToken[];\n dependents: NeighborhoodToken[];\n}\n\n/**\n * Fetch the direct (depth-1) prerequisite neighborhood around one token.\n * This is the primary data source for the experimental 3D knowledge graph.\n */\nexport async function getTokenNeighborhood(\n db: Database,\n tokenId: string,\n userId?: string,\n): Promise<Neighborhood> {\n const token = await getTokenById(db, tokenId);\n if (!token) {\n throw new Error(`Token not found: ${tokenId}`);\n }\n\n const centerCard = userId ? await getCard(db, tokenId, userId) : undefined;\n\n const prereqRows = await getPrerequisites(db, tokenId);\n const depRows = await getDependents(db, tokenId);\n\n // Collect all related token ids for batched card lookup (when userId given)\n const relatedTokenIds = new Set<string>();\n for (const p of prereqRows) relatedTokenIds.add(p.requires_id);\n for (const d of depRows) relatedTokenIds.add(d.token_id);\n\n const cardMap = new Map<string, Card>();\n if (userId && relatedTokenIds.size > 0) {\n const ids = Array.from(relatedTokenIds);\n const placeholders = ids.map(() => \"?\").join(\",\");\n const rows = (await db\n .prepare(\n `SELECT * FROM cards WHERE token_id IN (${placeholders}) AND user_id = ?`,\n )\n .all(...ids, userId)) as Card[];\n for (const row of rows) {\n cardMap.set(row.token_id, row);\n }\n }\n\n const toNode = (\n t: {\n id: string;\n slug: string;\n concept: string;\n domain: string;\n bloom_level: number;\n },\n card?: Card,\n ): NeighborhoodToken => ({\n id: t.id,\n slug: t.slug,\n concept: t.concept,\n domain: t.domain,\n bloom_level: t.bloom_level,\n card: card\n ? {\n state: card.state,\n reps: card.reps,\n stability: card.stability,\n difficulty: card.difficulty,\n blocked: card.blocked === 1,\n due_at: card.due_at,\n last_review_at: card.last_review_at,\n }\n : null,\n });\n\n const center: NeighborhoodToken = toNode(token, centerCard);\n\n const prerequisites: NeighborhoodToken[] = prereqRows.map((p) =>\n toNode(\n {\n id: p.requires_id,\n slug: p.slug,\n concept: p.concept,\n domain: p.domain,\n bloom_level: p.bloom_level,\n },\n cardMap.get(p.requires_id),\n ),\n );\n\n const dependents: NeighborhoodToken[] = depRows.map((d) =>\n toNode(\n {\n id: d.token_id,\n slug: d.slug,\n concept: d.concept,\n domain: d.domain,\n bloom_level: d.bloom_level,\n },\n cardMap.get(d.token_id),\n ),\n );\n\n return { center, prerequisites, dependents };\n}\n","/**\n * Review log repository — typed wrappers around the review_logs table.\n *\n * The review log is immutable: every rating event is appended, never\n * updated or deleted. This provides a complete audit trail of a user's\n * learning history.\n */\n\nimport { ulid } from \"ulid\";\nimport type { Database } from \"../db/types.js\";\n\n// ── Types ────────────────────────────────────────────────────────────────────\n\nexport interface ReviewLog {\n id: string;\n card_id: string;\n token_id: string;\n user_id: string;\n rating: number; // 1-4\n response_time_ms: number | null;\n reviewed_at: string;\n scheduled_at: string;\n session_id: string | null;\n}\n\nexport interface CreateReviewInput {\n card_id: string;\n token_id: string;\n user_id: string;\n rating: number; // 1-4\n scheduled_at: string;\n response_time_ms?: number | null;\n session_id?: string | null;\n}\n\nexport interface ListReviewsOptions {\n /** Maximum number of reviews to return. */\n limit?: number;\n /** Return reviews after this ISO timestamp. */\n after?: string;\n /** Return reviews before this ISO timestamp. */\n before?: string;\n}\n\n// ── Functions ────────────────────────────────────────────────────────────────\n\n/**\n * Log an immutable review event.\n *\n * Validates that the rating is between 1 and 4 (matching the schema CHECK).\n * Returns the created review log entry.\n */\nexport async function logReview(\n db: Database,\n input: CreateReviewInput,\n): Promise<ReviewLog> {\n if (input.rating < 1 || input.rating > 4) {\n throw new Error(`Rating must be between 1 and 4, got ${input.rating}`);\n }\n\n const id = ulid();\n const now = new Date().toISOString();\n\n await db\n .prepare(\n `INSERT INTO review_logs (id, card_id, token_id, user_id, rating, response_time_ms, reviewed_at, scheduled_at, session_id)\n VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?)`,\n )\n .run(\n id,\n input.card_id,\n input.token_id,\n input.user_id,\n input.rating,\n input.response_time_ms ?? null,\n now,\n input.scheduled_at,\n input.session_id ?? null,\n );\n\n return (await db\n .prepare(\"SELECT * FROM review_logs WHERE id = ?\")\n .get(id)) as ReviewLog;\n}\n\n/**\n * Get all reviews for a specific card, ordered by reviewed_at ascending.\n */\nexport async function getReviewsForCard(\n db: Database,\n cardId: string,\n): Promise<ReviewLog[]> {\n return (await db\n .prepare(\n \"SELECT * FROM review_logs WHERE card_id = ? ORDER BY reviewed_at ASC\",\n )\n .all(cardId)) as ReviewLog[];\n}\n\n/**\n * Get reviews for a user, with optional filtering.\n *\n * Results are ordered by reviewed_at descending (most recent first).\n */\nexport async function getReviewsForUser(\n db: Database,\n userId: string,\n options?: ListReviewsOptions,\n): Promise<ReviewLog[]> {\n const conditions = [\"user_id = ?\"];\n const params: unknown[] = [userId];\n\n if (options?.after) {\n conditions.push(\"reviewed_at > ?\");\n params.push(options.after);\n }\n if (options?.before) {\n conditions.push(\"reviewed_at < ?\");\n params.push(options.before);\n }\n\n let sql = `SELECT * FROM review_logs WHERE ${conditions.join(\" AND \")} ORDER BY reviewed_at DESC`;\n\n if (options?.limit) {\n sql += \" LIMIT ?\";\n params.push(options.limit);\n }\n\n return (await db.prepare(sql).all(...params)) as ReviewLog[];\n}\n","/**\n * Session repository — typed wrappers around sessions and session_steps.\n *\n * A session represents a work+learning episode. Steps within a session\n * record which tokens were touched and by whom (user or agent).\n */\n\nimport { ulid } from \"ulid\";\nimport type { Database } from \"../db/types.js\";\n\n// ── Types ────────────────────────────────────────────────────────────────────\n\nexport type ExecutionContext = \"shell\" | \"ui\" | \"reallife\";\n\nexport interface Session {\n id: string;\n user_id: string;\n task: string;\n execution_context: ExecutionContext;\n started_at: string;\n completed_at: string | null;\n}\n\nexport interface SessionStep {\n id: string;\n session_id: string;\n token_id: string;\n done_by: \"user\" | \"agent\";\n rating: number | null; // 1-4 or null\n notes: string | null;\n created_at: string;\n}\n\nexport interface CreateSessionInput {\n user_id: string;\n task: string;\n execution_context?: ExecutionContext;\n}\n\nexport interface LogStepInput {\n session_id: string;\n token_id: string;\n done_by: \"user\" | \"agent\";\n rating?: number | null;\n notes?: string | null;\n}\n\n/** A step joined with its token details, returned by getSessionSummary. */\nexport interface StepWithToken extends SessionStep {\n slug: string;\n concept: string;\n domain: string;\n bloom_level: number;\n}\n\nexport interface SessionSummary {\n session: Session;\n steps: StepWithToken[];\n}\n\n// ── Functions ────────────────────────────────────────────────────────────────\n\n/**\n * Start a new session. Returns the created session.\n *\n * Ported from the PoC's start-session command.\n */\nexport async function startSession(\n db: Database,\n input: CreateSessionInput,\n): Promise<Session> {\n const id = ulid();\n const now = new Date().toISOString();\n const ctx = input.execution_context ?? \"shell\";\n\n await db\n .prepare(\n `INSERT INTO sessions (id, user_id, task, execution_context, started_at)\n VALUES (?, ?, ?, ?, ?)`,\n )\n .run(id, input.user_id, input.task, ctx, now);\n\n return (await db\n .prepare(\"SELECT * FROM sessions WHERE id = ?\")\n .get(id)) as Session;\n}\n\n/**\n * End a session by setting its completed_at timestamp.\n *\n * Throws if the session does not exist or is already completed.\n *\n * Ported from the PoC's end-session command.\n */\nexport async function endSession(\n db: Database,\n sessionId: string,\n): Promise<Session> {\n const session = (await db\n .prepare(\"SELECT * FROM sessions WHERE id = ?\")\n .get(sessionId)) as Session | undefined;\n\n if (!session) {\n throw new Error(`Session not found: ${sessionId}`);\n }\n if (session.completed_at) {\n throw new Error(`Session already completed: ${sessionId}`);\n }\n\n const now = new Date().toISOString();\n await db\n .prepare(\"UPDATE sessions SET completed_at = ? WHERE id = ?\")\n .run(now, sessionId);\n\n return (await db\n .prepare(\"SELECT * FROM sessions WHERE id = ?\")\n .get(sessionId)) as Session;\n}\n\n/**\n * Log a step within a session.\n *\n * Validates that done_by is 'user' or 'agent' and that the rating\n * (if provided) is between 1 and 4.\n *\n * Ported from the PoC's log-step command.\n */\nexport async function logStep(\n db: Database,\n input: LogStepInput,\n): Promise<SessionStep> {\n if (input.done_by !== \"user\" && input.done_by !== \"agent\") {\n throw new Error(\n `done_by must be 'user' or 'agent', got '${input.done_by}'`,\n );\n }\n if (input.rating != null && (input.rating < 1 || input.rating > 4)) {\n throw new Error(`Rating must be between 1 and 4, got ${input.rating}`);\n }\n\n // Verify the session exists\n const session = await db\n .prepare(\"SELECT id FROM sessions WHERE id = ?\")\n .get(input.session_id);\n if (!session) {\n throw new Error(`Session not found: ${input.session_id}`);\n }\n\n const id = ulid();\n const now = new Date().toISOString();\n\n await db\n .prepare(\n `INSERT INTO session_steps (id, session_id, token_id, done_by, rating, notes, created_at)\n VALUES (?, ?, ?, ?, ?, ?, ?)`,\n )\n .run(\n id,\n input.session_id,\n input.token_id,\n input.done_by,\n input.rating ?? null,\n input.notes ?? null,\n now,\n );\n\n return (await db\n .prepare(\"SELECT * FROM session_steps WHERE id = ?\")\n .get(id)) as SessionStep;\n}\n\n/**\n * Get a full session summary: the session record plus all steps\n * joined with their token details.\n *\n * Ported from the PoC's session-summary command.\n * Throws if the session does not exist.\n */\nexport async function getSessionSummary(\n db: Database,\n sessionId: string,\n): Promise<SessionSummary> {\n const session = (await db\n .prepare(\"SELECT * FROM sessions WHERE id = ?\")\n .get(sessionId)) as Session | undefined;\n\n if (!session) {\n throw new Error(`Session not found: ${sessionId}`);\n }\n\n const steps = (await db\n .prepare(\n `SELECT ss.*, t.slug, t.concept, t.domain, t.bloom_level\n FROM session_steps ss\n JOIN tokens t ON t.id = ss.token_id\n WHERE ss.session_id = ?\n ORDER BY ss.created_at ASC`,\n )\n .all(sessionId)) as StepWithToken[];\n\n return { session, steps };\n}\n","/**\n * User settings — key/value store backed by the user_config table.\n */\n\nimport type { Database } from \"../db/types.js\";\n\nexport interface UserSetting {\n key: string;\n value: string;\n updated_at: string;\n}\n\n/** Get a single setting by key. Returns undefined if not set. */\nexport async function getSetting(\n db: Database,\n key: string,\n): Promise<string | undefined> {\n const row = (await db\n .prepare(\"SELECT value FROM user_config WHERE key = ?\")\n .get(key)) as { value: string } | undefined;\n return row?.value;\n}\n\n/** Get all settings as a key-value map. */\nexport async function getAllSettings(\n db: Database,\n): Promise<Record<string, string>> {\n const rows = (await db\n .prepare(\"SELECT key, value FROM user_config ORDER BY key\")\n .all()) as { key: string; value: string }[];\n const map: Record<string, string> = {};\n for (const row of rows) {\n map[row.key] = row.value;\n }\n return map;\n}\n\n/** Get all settings with metadata. */\nexport async function getAllSettingsDetailed(\n db: Database,\n): Promise<UserSetting[]> {\n return (await db\n .prepare(\"SELECT key, value, updated_at FROM user_config ORDER BY key\")\n .all()) as UserSetting[];\n}\n\n/** Set a setting (insert or update). */\nexport async function setSetting(\n db: Database,\n key: string,\n value: string,\n): Promise<void> {\n await db\n .prepare(\n `INSERT INTO user_config (key, value, updated_at)\n VALUES (?, ?, datetime('now'))\n ON CONFLICT(key) DO UPDATE SET value = excluded.value, updated_at = excluded.updated_at`,\n )\n .run(key, value);\n}\n\n/** Delete a setting. Returns true if it existed. */\nexport async function deleteSetting(\n db: Database,\n key: string,\n): Promise<boolean> {\n const result = await db\n .prepare(\"DELETE FROM user_config WHERE key = ?\")\n .run(key);\n return result.changes > 0;\n}\n","/**\n * Monitor log analyzer — maps observed shell commands to token ratings.\n *\n * Pure functions, no DB or filesystem access. Takes parsed command records\n * and a token-to-pattern mapping, returns ratings with evidence.\n */\n\n// ── Types ────────────────────────────────────────────────────────────────────\n\nexport interface MonitorEvent {\n type: \"command_start\" | \"command_end\" | \"monitor_meta\";\n ts: string;\n seq?: number;\n pid?: number;\n command?: string;\n cwd?: string;\n exit_code?: number;\n event?: \"start\" | \"stop\";\n session_id?: string;\n shell?: string;\n}\n\nexport interface CommandRecord {\n seq: number;\n pid: number;\n command: string;\n cwd: string;\n startedAt: string;\n endedAt: string | null;\n durationMs: number | null;\n exitCode: number | null;\n}\n\nexport interface TokenPattern {\n slug: string;\n patterns: string[]; // command prefixes or regex strings\n}\n\nexport interface ObservationRating {\n tokenSlug: string;\n rating: 1 | 2 | 3 | 4 | null;\n confidence: \"high\" | \"medium\" | \"low\";\n evidence: {\n matchedCommands: number;\n helpSeeking: boolean;\n errorCount: number;\n selfCorrections: number;\n medianGapMs: number | null;\n thinkingGapMs: number | null;\n };\n matchedCommandTexts: string[];\n}\n\nexport interface AnalysisResult {\n ratings: ObservationRating[];\n unmatchedCommands: string[];\n timeSpan: { start: string; end: string; durationMs: number } | null;\n}\n\n// ── Parsing ──────────────────────────────────────────────────────────────────\n\n/**\n * Parse a JSONL string into MonitorEvent objects.\n * Skips malformed lines silently.\n */\nexport function parseMonitorLog(jsonl: string): MonitorEvent[] {\n const events: MonitorEvent[] = [];\n for (const line of jsonl.split(\"\\n\")) {\n const trimmed = line.trim();\n if (!trimmed) continue;\n try {\n events.push(JSON.parse(trimmed) as MonitorEvent);\n } catch {\n // skip malformed lines\n }\n }\n return events;\n}\n\n/**\n * Pair command_start and command_end events by (pid, seq) into CommandRecords.\n */\nexport function pairCommands(events: MonitorEvent[]): CommandRecord[] {\n const starts = new Map<string, MonitorEvent>();\n const records: CommandRecord[] = [];\n\n for (const e of events) {\n if (e.type === \"command_start\" && e.seq != null) {\n const key = `${e.pid ?? 0}:${e.seq}`;\n starts.set(key, e);\n } else if (e.type === \"command_end\" && e.seq != null) {\n const key = `${e.pid ?? 0}:${e.seq}`;\n const start = starts.get(key);\n if (start) {\n const startMs = new Date(start.ts).getTime();\n const endMs = new Date(e.ts).getTime();\n records.push({\n seq: e.seq,\n pid: e.pid ?? 0,\n command: start.command ?? \"\",\n cwd: start.cwd ?? \"\",\n startedAt: start.ts,\n endedAt: e.ts,\n durationMs: endMs - startMs,\n exitCode: e.exit_code ?? null,\n });\n starts.delete(key);\n }\n }\n }\n\n // Remaining unpaired starts (monitoring stopped mid-command)\n for (const [, start] of starts) {\n records.push({\n seq: start.seq ?? 0,\n pid: start.pid ?? 0,\n command: start.command ?? \"\",\n cwd: start.cwd ?? \"\",\n startedAt: start.ts,\n endedAt: null,\n durationMs: null,\n exitCode: null,\n });\n }\n\n records.sort(\n (a, b) => new Date(a.startedAt).getTime() - new Date(b.startedAt).getTime(),\n );\n return records;\n}\n\n// ── Analysis ─────────────────────────────────────────────────────────────────\n\nconst HELP_PATTERNS = [\"--help\", \"man \", \"tldr \", \"help \"];\nconst HELP_WINDOW_MS = 60_000;\nconst _RETRY_WINDOW_MS = 30_000;\n\nfunction matchesToken(command: string, patterns: string[]): boolean {\n const lower = command.toLowerCase();\n return patterns.some((p) => lower.includes(p.toLowerCase()));\n}\n\nfunction isHelpCommand(command: string): boolean {\n const lower = command.toLowerCase();\n return HELP_PATTERNS.some((p) => lower.includes(p));\n}\n\nfunction commandPrefix(command: string): string {\n return command.split(/\\s+/).slice(0, 2).join(\" \").toLowerCase();\n}\n\nfunction computeMedian(values: number[]): number | null {\n if (values.length === 0) return null;\n const sorted = [...values].sort((a, b) => a - b);\n const mid = Math.floor(sorted.length / 2);\n return sorted.length % 2 !== 0\n ? sorted[mid]\n : (sorted[mid - 1] + sorted[mid]) / 2;\n}\n\n/**\n * Analyze observed commands against token patterns and produce ratings.\n */\nexport function analyzeObservation(\n commands: CommandRecord[],\n tokenPatterns: TokenPattern[],\n): AnalysisResult {\n const matchedSet = new Set<number>(); // indices of commands matched to any token\n const ratings: ObservationRating[] = [];\n\n for (const tp of tokenPatterns) {\n const matchIndices: number[] = [];\n const matchedTexts: string[] = [];\n\n for (let i = 0; i < commands.length; i++) {\n if (matchesToken(commands[i].command, tp.patterns)) {\n matchIndices.push(i);\n matchedTexts.push(commands[i].command);\n matchedSet.add(i);\n }\n }\n\n if (matchIndices.length === 0) {\n ratings.push({\n tokenSlug: tp.slug,\n rating: null,\n confidence: \"low\",\n evidence: {\n matchedCommands: 0,\n helpSeeking: false,\n errorCount: 0,\n selfCorrections: 0,\n medianGapMs: null,\n thinkingGapMs: null,\n },\n matchedCommandTexts: [],\n });\n continue;\n }\n\n // Help-seeking: any help command within HELP_WINDOW_MS before a matched command\n let helpSeeking = false;\n for (const mi of matchIndices) {\n const matchTime = new Date(commands[mi].startedAt).getTime();\n for (let j = 0; j < commands.length; j++) {\n if (j === mi) continue;\n const cmdTime = new Date(commands[j].startedAt).getTime();\n if (cmdTime >= matchTime - HELP_WINDOW_MS && cmdTime < matchTime) {\n if (isHelpCommand(commands[j].command)) {\n helpSeeking = true;\n break;\n }\n }\n }\n if (helpSeeking) break;\n }\n\n // Error count: matched commands with non-zero exit code\n let errorCount = 0;\n for (const mi of matchIndices) {\n if (commands[mi].exitCode != null && commands[mi].exitCode !== 0) {\n errorCount++;\n }\n }\n\n // Self-corrections: same command prefix run multiple times with different args\n let selfCorrections = 0;\n const prefixGroups = new Map<string, number>();\n for (const mi of matchIndices) {\n const prefix = commandPrefix(commands[mi].command);\n prefixGroups.set(prefix, (prefixGroups.get(prefix) ?? 0) + 1);\n }\n for (const count of prefixGroups.values()) {\n if (count > 1) selfCorrections += count - 1;\n }\n\n // Speed: inter-command gaps between matched commands\n const gaps: number[] = [];\n for (let k = 1; k < matchIndices.length; k++) {\n const prev = commands[matchIndices[k - 1]];\n const curr = commands[matchIndices[k]];\n if (prev.endedAt) {\n const gap =\n new Date(curr.startedAt).getTime() - new Date(prev.endedAt).getTime();\n if (gap >= 0) gaps.push(gap);\n }\n }\n\n // Thinking gap: time before first matched command from previous command's end\n let thinkingGapMs: number | null = null;\n const firstMatchIdx = matchIndices[0];\n if (firstMatchIdx > 0) {\n const prev = commands[firstMatchIdx - 1];\n if (prev.endedAt) {\n thinkingGapMs =\n new Date(commands[firstMatchIdx].startedAt).getTime() -\n new Date(prev.endedAt).getTime();\n }\n }\n\n const medianGapMs = computeMedian(gaps);\n\n // Determine rating\n const rating = inferRating({\n helpSeeking,\n errorCount,\n selfCorrections,\n medianGapMs,\n thinkingGapMs,\n matchedCommands: matchIndices.length,\n });\n\n // Confidence: more matched commands = higher confidence\n const confidence =\n matchIndices.length >= 3\n ? \"high\"\n : matchIndices.length >= 2\n ? \"medium\"\n : \"low\";\n\n ratings.push({\n tokenSlug: tp.slug,\n rating,\n confidence,\n evidence: {\n matchedCommands: matchIndices.length,\n helpSeeking,\n errorCount,\n selfCorrections,\n medianGapMs,\n thinkingGapMs,\n },\n matchedCommandTexts: matchedTexts,\n });\n }\n\n // Unmatched commands\n const unmatchedCommands: string[] = [];\n for (let i = 0; i < commands.length; i++) {\n if (!matchedSet.has(i) && !isHelpCommand(commands[i].command)) {\n unmatchedCommands.push(commands[i].command);\n }\n }\n\n // Time span\n let timeSpan: AnalysisResult[\"timeSpan\"] = null;\n if (commands.length > 0) {\n const first = commands[0];\n const last = commands[commands.length - 1];\n const endTs = last.endedAt ?? last.startedAt;\n timeSpan = {\n start: first.startedAt,\n end: endTs,\n durationMs:\n new Date(endTs).getTime() - new Date(first.startedAt).getTime(),\n };\n }\n\n return { ratings, unmatchedCommands, timeSpan };\n}\n\n// ── Rating Inference ─────────────────────────────────────────────────────────\n\ninterface RatingSignals {\n helpSeeking: boolean;\n errorCount: number;\n selfCorrections: number;\n medianGapMs: number | null;\n thinkingGapMs: number | null;\n matchedCommands: number;\n}\n\nfunction inferRating(signals: RatingSignals): 1 | 2 | 3 | 4 {\n const {\n helpSeeking,\n errorCount,\n selfCorrections,\n medianGapMs,\n thinkingGapMs,\n } = signals;\n\n // Count negative signals\n let negatives = 0;\n if (helpSeeking) negatives += 2;\n if (errorCount >= 3) negatives += 3;\n else if (errorCount >= 1) negatives += 1;\n if (selfCorrections >= 2) negatives += 2;\n else if (selfCorrections >= 1) negatives += 1;\n if (medianGapMs != null && medianGapMs > 30_000) negatives += 2;\n else if (medianGapMs != null && medianGapMs > 10_000) negatives += 1;\n if (thinkingGapMs != null && thinkingGapMs > 30_000) negatives += 1;\n\n if (negatives >= 5) return 1;\n if (negatives >= 3) return 2;\n if (negatives >= 1) return 3;\n return 4;\n}\n","/**\n * Monitor I/O — read/write JSONL files for shell observation.\n *\n * Monitor logs live at ~/.zam/monitor/<session-id>.jsonl.\n * Separated from analyzer.ts so the analyzer remains pure-function testable.\n */\n\nimport {\n appendFileSync,\n existsSync,\n mkdirSync,\n readFileSync,\n statSync,\n} from \"node:fs\";\nimport { homedir } from \"node:os\";\nimport { join } from \"node:path\";\nimport type { MonitorEvent } from \"./analyzer.js\";\nimport { parseMonitorLog } from \"./analyzer.js\";\n\nconst MONITOR_DIR = join(homedir(), \".zam\", \"monitor\");\n\n/** Get the monitor directory path. */\nexport function getMonitorDir(): string {\n return MONITOR_DIR;\n}\n\n/** Get the JSONL file path for a session. */\nexport function getMonitorPath(sessionId: string): string {\n return join(MONITOR_DIR, `${sessionId}.jsonl`);\n}\n\n/** Ensure the monitor directory exists (mode 0700 for privacy). */\nexport function ensureMonitorDir(): void {\n if (!existsSync(MONITOR_DIR)) {\n mkdirSync(MONITOR_DIR, { recursive: true, mode: 0o700 });\n }\n}\n\n/** Append a single event to the session's JSONL file. */\nexport function writeMonitorEvent(\n sessionId: string,\n event: MonitorEvent,\n): void {\n ensureMonitorDir();\n const path = getMonitorPath(sessionId);\n appendFileSync(path, `${JSON.stringify(event)}\\n`);\n}\n\n/** Read and parse all events from a session's monitor log. */\nexport function readMonitorLog(sessionId: string): MonitorEvent[] {\n const path = getMonitorPath(sessionId);\n if (!existsSync(path)) {\n return [];\n }\n const content = readFileSync(path, \"utf-8\");\n return parseMonitorLog(content);\n}\n\n/** Check if a monitor log exists for a session. */\nexport function monitorLogExists(sessionId: string): boolean {\n return existsSync(getMonitorPath(sessionId));\n}\n\n/** Get basic stats about a monitor log without full parsing. */\nexport function getMonitorLogStats(sessionId: string): {\n exists: boolean;\n sizeBytes: number;\n lineCount: number;\n} {\n const path = getMonitorPath(sessionId);\n if (!existsSync(path)) {\n return { exists: false, sizeBytes: 0, lineCount: 0 };\n }\n const stat = statSync(path);\n const content = readFileSync(path, \"utf-8\");\n const lineCount = content.split(\"\\n\").filter((l) => l.trim()).length;\n return { exists: true, sizeBytes: stat.size, lineCount };\n}\n","/**\n * Session synthesis connects shell observation to durable learning state.\n *\n * A preview analyzes monitor commands without mutating the database. Applying\n * one confirmed candidate updates the card, review log, session step, blocking\n * state, and synthesis audit record in a single transaction.\n */\n\nimport { ulid } from \"ulid\";\nimport type { Database } from \"../db/types.js\";\nimport { listAgentSkills } from \"../models/agent-skill.js\";\nimport { ensureCard } from \"../models/card.js\";\nimport { getPrerequisites } from \"../models/prerequisite.js\";\nimport type { Session } from \"../models/session.js\";\nimport { logStep } from \"../models/session.js\";\nimport type { Token } from \"../models/token.js\";\nimport { getTokenBySlug } from \"../models/token.js\";\nimport { evaluateRatingWithinTransaction } from \"../recall/evaluator.js\";\nimport { cascadeBlock } from \"../scheduler/blocker.js\";\nimport type { Rating } from \"../scheduler/fsrs.js\";\nimport type {\n CommandRecord,\n ObservationRating,\n TokenPattern,\n} from \"./analyzer.js\";\nimport { analyzeObservation, pairCommands } from \"./analyzer.js\";\nimport { readMonitorLog } from \"./monitor-io.js\";\n\nexport type SynthesisConfidence = \"medium\" | \"high\";\n\nexport interface SessionSynthesisCandidate {\n tokenId: string;\n tokenSlug: string;\n concept: string;\n domain: string;\n inferredRating: Rating;\n confidence: SynthesisConfidence;\n evidence: ObservationRating[\"evidence\"];\n matchedCommandTexts: string[];\n}\n\nexport interface PrepareSessionSynthesisInput {\n sessionId: string;\n explicitPatterns?: TokenPattern[];\n minConfidence?: SynthesisConfidence;\n /** Test and integration hook; normal callers read the monitor log. */\n commands?: CommandRecord[];\n}\n\nexport interface SessionSynthesisPreview {\n sessionId: string;\n userId: string;\n patternCount: number;\n commandCount: number;\n alreadyApplied: number;\n skippedLowConfidence: number;\n candidates: SessionSynthesisCandidate[];\n unmatchedCommands: string[];\n timeSpan: {\n start: string;\n end: string;\n durationMs: number;\n } | null;\n}\n\nexport interface SessionSynthesisEvidence {\n signals: ObservationRating[\"evidence\"];\n matchedCommandTexts: string[];\n}\n\nexport interface SessionSynthesisRecord {\n session_id: string;\n token_id: string;\n card_id: string;\n inferred_rating: Rating;\n confirmed_rating: Rating;\n confidence: SynthesisConfidence;\n evidence: SessionSynthesisEvidence;\n review_log_id: string;\n session_step_id: string;\n created_at: string;\n}\n\ninterface SessionSynthesisRow extends Omit<SessionSynthesisRecord, \"evidence\"> {\n evidence: string;\n}\n\nexport interface ApplySessionSynthesisInput {\n sessionId: string;\n tokenSlug: string;\n inferredRating: Rating;\n confirmedRating: Rating;\n confidence: SynthesisConfidence;\n evidence: ObservationRating[\"evidence\"];\n matchedCommandTexts: string[];\n}\n\nexport interface ApplySessionSynthesisResult {\n applied: boolean;\n record: SessionSynthesisRecord;\n blocked?: Awaited<ReturnType<typeof cascadeBlock>>;\n}\n\nfunction parseSynthesisRow(row: SessionSynthesisRow): SessionSynthesisRecord {\n return {\n ...row,\n evidence: JSON.parse(row.evidence) as SessionSynthesisEvidence,\n };\n}\n\nfunction confidenceRank(confidence: ObservationRating[\"confidence\"]): number {\n return confidence === \"high\" ? 2 : confidence === \"medium\" ? 1 : 0;\n}\n\nfunction normalizeSkillStep(step: string): string[] {\n const codeSpans = [...step.matchAll(/`([^`]+)`/g)]\n .map((match) => match[1].trim())\n .filter(Boolean);\n if (codeSpans.length > 0) return codeSpans;\n\n const normalized = step\n .trim()\n .replace(/^(?:[-*]|\\d+[.)])\\s+/, \"\")\n .replace(/^(?:run|execute)\\s+/i, \"\")\n .replace(/^`|`$/g, \"\")\n .trim();\n return normalized ? [normalized] : [];\n}\n\nasync function buildSkillPatterns(db: Database): Promise<TokenPattern[]> {\n const byToken = new Map<string, Set<string>>();\n\n for (const skill of await listAgentSkills(db)) {\n if (skill.token_slugs.length !== 1) continue;\n const patterns = skill.steps.flatMap(normalizeSkillStep);\n for (const slug of skill.token_slugs) {\n const tokenPatterns = byToken.get(slug) ?? new Set<string>();\n for (const pattern of patterns) tokenPatterns.add(pattern);\n byToken.set(slug, tokenPatterns);\n }\n }\n\n return [...byToken.entries()].map(([slug, patterns]) => ({\n slug,\n patterns: [...patterns],\n }));\n}\n\nfunction mergePatterns(\n automatic: TokenPattern[],\n explicit: TokenPattern[],\n): TokenPattern[] {\n const merged = new Map<string, Set<string>>();\n for (const entry of [...automatic, ...explicit]) {\n const patterns = merged.get(entry.slug) ?? new Set<string>();\n for (const pattern of entry.patterns) {\n const trimmed = pattern.trim();\n if (trimmed) patterns.add(trimmed);\n }\n merged.set(entry.slug, patterns);\n }\n\n return [...merged.entries()]\n .filter(([, patterns]) => patterns.size > 0)\n .map(([slug, patterns]) => ({ slug, patterns: [...patterns] }));\n}\n\nasync function getSession(db: Database, sessionId: string): Promise<Session> {\n const session = (await db\n .prepare(\"SELECT * FROM sessions WHERE id = ?\")\n .get(sessionId)) as Session | undefined;\n if (!session) throw new Error(`Session not found: ${sessionId}`);\n return session;\n}\n\nexport async function getSessionSynthesisRecords(\n db: Database,\n sessionId: string,\n): Promise<SessionSynthesisRecord[]> {\n const rows = (await db\n .prepare(\n \"SELECT * FROM session_syntheses WHERE session_id = ? ORDER BY created_at\",\n )\n .all(sessionId)) as SessionSynthesisRow[];\n return rows.map(parseSynthesisRow);\n}\n\nexport async function prepareSessionSynthesis(\n db: Database,\n input: PrepareSessionSynthesisInput,\n): Promise<SessionSynthesisPreview> {\n const session = await getSession(db, input.sessionId);\n const patterns = mergePatterns(\n await buildSkillPatterns(db),\n input.explicitPatterns ?? [],\n );\n\n const validPatterns: TokenPattern[] = [];\n const tokens = new Map<string, Token>();\n for (const pattern of patterns) {\n const token = await getTokenBySlug(db, pattern.slug);\n if (!token || token.deprecated_at) continue;\n validPatterns.push(pattern);\n tokens.set(pattern.slug, token);\n }\n\n const commands =\n input.commands ?? pairCommands(readMonitorLog(input.sessionId));\n const analysis = analyzeObservation(commands, validPatterns);\n const applied = new Set(\n (await getSessionSynthesisRecords(db, input.sessionId)).map(\n (record) => record.token_id,\n ),\n );\n const minRank = confidenceRank(input.minConfidence ?? \"medium\");\n let skippedLowConfidence = 0;\n const candidates: SessionSynthesisCandidate[] = [];\n\n for (const rating of analysis.ratings) {\n const token = tokens.get(rating.tokenSlug);\n if (!token || rating.rating == null || applied.has(token.id)) continue;\n if (confidenceRank(rating.confidence) < minRank) {\n skippedLowConfidence++;\n continue;\n }\n\n candidates.push({\n tokenId: token.id,\n tokenSlug: token.slug,\n concept: token.concept,\n domain: token.domain,\n inferredRating: rating.rating,\n confidence: rating.confidence as SynthesisConfidence,\n evidence: rating.evidence,\n matchedCommandTexts: rating.matchedCommandTexts,\n });\n }\n\n return {\n sessionId: session.id,\n userId: session.user_id,\n patternCount: validPatterns.length,\n commandCount: commands.length,\n alreadyApplied: applied.size,\n skippedLowConfidence,\n candidates,\n unmatchedCommands: analysis.unmatchedCommands,\n timeSpan: analysis.timeSpan,\n };\n}\n\nexport async function applySessionSynthesis(\n db: Database,\n input: ApplySessionSynthesisInput,\n): Promise<ApplySessionSynthesisResult> {\n return db.transaction(async (tx) => {\n const session = await getSession(tx, input.sessionId);\n const token = await getTokenBySlug(tx, input.tokenSlug);\n if (!token || token.deprecated_at) {\n throw new Error(`Active token not found: ${input.tokenSlug}`);\n }\n\n const existing = (await tx\n .prepare(\n \"SELECT * FROM session_syntheses WHERE session_id = ? AND token_id = ?\",\n )\n .get(session.id, token.id)) as SessionSynthesisRow | undefined;\n if (existing) {\n return { applied: false, record: parseSynthesisRow(existing) };\n }\n\n const card = await ensureCard(tx, token.id, session.user_id);\n const reviewLogId = ulid();\n await evaluateRatingWithinTransaction(tx, {\n cardId: card.id,\n tokenId: token.id,\n userId: session.user_id,\n rating: input.confirmedRating,\n sessionId: session.id,\n reviewLogId,\n });\n\n let blocked: Awaited<ReturnType<typeof cascadeBlock>> | undefined;\n if (input.confirmedRating === 1) {\n const prerequisites = await getPrerequisites(tx, token.id);\n if (prerequisites.length > 0) {\n blocked = await cascadeBlock(tx, session.user_id, token.slug);\n }\n }\n\n const notes = `Observation synthesis (${input.confidence}, inferred ${input.inferredRating}): ${input.matchedCommandTexts.slice(0, 3).join(\" | \")}`;\n const step = await logStep(tx, {\n session_id: session.id,\n token_id: token.id,\n done_by: \"user\",\n rating: input.confirmedRating,\n notes,\n });\n\n const evidence: SessionSynthesisEvidence = {\n signals: input.evidence,\n matchedCommandTexts: input.matchedCommandTexts,\n };\n const now = new Date().toISOString();\n await tx\n .prepare(\n `INSERT INTO session_syntheses (\n session_id, token_id, card_id, inferred_rating, confirmed_rating,\n confidence, evidence, review_log_id, session_step_id, created_at\n ) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?)`,\n )\n .run(\n session.id,\n token.id,\n card.id,\n input.inferredRating,\n input.confirmedRating,\n input.confidence,\n JSON.stringify(evidence),\n reviewLogId,\n step.id,\n now,\n );\n\n const record = (await tx\n .prepare(\n \"SELECT * FROM session_syntheses WHERE session_id = ? AND token_id = ?\",\n )\n .get(session.id, token.id)) as SessionSynthesisRow;\n\n return {\n applied: true,\n record: parseSynthesisRow(record),\n blocked,\n };\n });\n}\n","/**\n * Rating Evaluator\n *\n * Processes a user's self-assessment rating after a recall attempt.\n * Coordinates between FSRS scheduling, review logging, and blocking.\n */\n\nimport { ulid } from \"ulid\";\nimport type { Database } from \"../db/types.js\";\nimport { updateCard } from \"../models/card.js\";\nimport type { Rating, SchedulingCard } from \"../scheduler/fsrs.js\";\nimport { createFSRS } from \"../scheduler/fsrs.js\";\n\nexport interface EvaluateInput {\n cardId: string;\n tokenId: string;\n userId: string;\n rating: Rating;\n sessionId?: string;\n responseTimeMs?: number;\n reviewLogId?: string;\n}\n\nexport interface EvaluateResult {\n nextDueAt: string;\n stability: number;\n difficulty: number;\n state: string;\n scheduledDays: number;\n reps: number;\n lapses: number;\n}\n\n/**\n * Process a rating: update the card via FSRS, log the review.\n * Returns the updated scheduling state.\n *\n * Note: blocking logic (cascade-block) is handled separately by the caller\n * when rating === 1 and the token has prerequisites.\n */\nexport async function evaluateRating(\n db: Database,\n input: EvaluateInput,\n): Promise<EvaluateResult> {\n return db.transaction((tx) => evaluateRatingWithinTransaction(tx, input));\n}\n\n/**\n * Apply a rating using a transaction already owned by the caller.\n * This is used when prerequisite blocking must commit with the review.\n */\nexport async function evaluateRatingWithinTransaction(\n db: Database,\n input: EvaluateInput,\n): Promise<EvaluateResult> {\n // Get current card state\n const card = (await db\n .prepare(\"SELECT * FROM cards WHERE id = ?\")\n .get(input.cardId)) as\n | {\n stability: number;\n difficulty: number;\n elapsed_days: number;\n scheduled_days: number;\n reps: number;\n lapses: number;\n state: string;\n due_at: string;\n last_review_at: string | null;\n }\n | undefined;\n\n if (!card) {\n throw new Error(`Card not found: ${input.cardId}`);\n }\n\n const now = new Date();\n const fsrs = createFSRS();\n\n // Build scheduling card from DB state\n const schedulingCard: SchedulingCard = {\n stability: card.stability,\n difficulty: card.difficulty,\n elapsedDays: card.elapsed_days,\n scheduledDays: card.scheduled_days,\n reps: card.reps,\n lapses: card.lapses,\n state: card.state as SchedulingCard[\"state\"],\n dueAt: new Date(card.due_at),\n lastReviewAt: card.last_review_at ? new Date(card.last_review_at) : null,\n };\n\n // Run FSRS\n const updated = fsrs.schedule(schedulingCard, input.rating, now);\n\n // Update the card in the DB\n await updateCard(db, input.cardId, {\n stability: updated.stability,\n difficulty: updated.difficulty,\n elapsed_days: updated.elapsedDays,\n scheduled_days: updated.scheduledDays,\n reps: updated.reps,\n lapses: updated.lapses,\n state: updated.state,\n due_at: updated.dueAt.toISOString(),\n last_review_at: now.toISOString(),\n });\n\n // Log the review (immutable)\n const reviewLogId = input.reviewLogId ?? ulid();\n await db\n .prepare(\n `INSERT INTO review_logs (id, card_id, token_id, user_id, rating, response_time_ms, reviewed_at, scheduled_at, session_id)\n VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?)`,\n )\n .run(\n reviewLogId,\n input.cardId,\n input.tokenId,\n input.userId,\n input.rating,\n input.responseTimeMs ?? null,\n now.toISOString(),\n card.due_at,\n input.sessionId ?? null,\n );\n\n return {\n nextDueAt: updated.dueAt.toISOString(),\n stability: updated.stability,\n difficulty: updated.difficulty,\n state: updated.state,\n scheduledDays: updated.scheduledDays,\n reps: updated.reps,\n lapses: updated.lapses,\n };\n}\n","/**\n * FSRS-5 — Free Spaced Repetition Scheduler (v5)\n *\n * Pure-function implementation of the FSRS algorithm that replaces\n * the PoC's SM-2 scheduler. This is the mathematical heart of ZAM's\n * spaced-repetition engine.\n *\n * Reference: https://github.com/open-spaced-repetition/fsrs4anki/wiki/The-Algorithm\n */\n\n// ── Rating scale ────────────────────────────────────────────────────────────\n\n/** 1 = Again (forgot), 2 = Hard, 3 = Good, 4 = Easy */\nexport type Rating = 1 | 2 | 3 | 4;\n\n// ── Card states ─────────────────────────────────────────────────────────────\n\nexport type CardState = \"new\" | \"learning\" | \"review\" | \"relearning\";\n\n// ── Scheduling card ─────────────────────────────────────────────────────────\n\nexport interface SchedulingCard {\n /** Memory stability in days — expected half-life of recall probability. */\n stability: number;\n /** Intrinsic difficulty on a 1–10 scale. */\n difficulty: number;\n /** Days elapsed since the last review. */\n elapsedDays: number;\n /** Currently scheduled interval in days. */\n scheduledDays: number;\n /** Count of successful consecutive reviews. */\n reps: number;\n /** Times the card was forgotten (rated Again). */\n lapses: number;\n /** Current learning state. */\n state: CardState;\n /** When the card is next due. */\n dueAt: Date;\n /** When the card was last reviewed (null for new cards). */\n lastReviewAt: Date | null;\n}\n\n// ── Parameters ──────────────────────────────────────────────────────────────\n\nexport interface FSRSParameters {\n /** 19 optimised weight parameters (w0 – w18). */\n w: number[];\n /** Target retention rate, e.g. 0.9 means we aim for 90% recall. */\n requestRetention: number;\n}\n\n// ── FSRS-5 default weights ──────────────────────────────────────────────────\n\nconst DEFAULT_W: number[] = [\n 0.4072,\n 1.1829,\n 3.1262,\n 15.4722, // w0–w3: initial stability per rating\n 7.2102,\n 0.5316,\n 1.0651, // w4–w6: difficulty\n 0.0092,\n 1.5988,\n 0.1176,\n 1.0014, // w7–w10: stability after forgetting\n 2.0032,\n 0.0266,\n 0.3077,\n 0.15, // w11–w14: stability increase\n 0.0,\n 2.7849,\n 0.3477,\n 0.6831, // w15–w18: additional parameters\n];\n\nconst DEFAULT_REQUEST_RETENTION = 0.9;\n\n// ── FSRS object returned by the factory ─────────────────────────────────────\n\nexport interface FSRS {\n /** Return a fully updated card after applying a rating. Pure function. */\n schedule(card: SchedulingCard, rating: Rating, now?: Date): SchedulingCard;\n /** The parameters baked into this instance. */\n readonly params: Readonly<FSRSParameters>;\n}\n\n// ── Helpers ─────────────────────────────────────────────────────────────────\n\n/** Clamp a number to [lo, hi]. */\nfunction clamp(value: number, lo: number, hi: number): number {\n return Math.min(hi, Math.max(lo, value));\n}\n\n/** Days between two Date objects (may be fractional). */\nfunction daysBetween(a: Date, b: Date): number {\n return (b.getTime() - a.getTime()) / (1000 * 60 * 60 * 24);\n}\n\n// ── Core FSRS-5 formulas ────────────────────────────────────────────────────\n\n/**\n * Initial stability for a brand-new card.\n * S₀ = w[rating - 1]\n */\nfunction initialStability(w: number[], rating: Rating): number {\n return w[rating - 1];\n}\n\n/**\n * Initial difficulty for a brand-new card.\n * D₀(rating) = w4 - exp(w5 * (rating - 1)) + 1\n * Clamped to [1, 10].\n */\nfunction initialDifficulty(w: number[], rating: Rating): number {\n return clamp(w[4] - Math.exp(w[5] * (rating - 1)) + 1, 1, 10);\n}\n\n/**\n * Updated difficulty after a review.\n * D' = w7 * D₀(3) + (1 - w7) * (D - w6 * (rating - 3))\n * Clamped to [1, 10].\n */\nfunction nextDifficulty(w: number[], d: number, rating: Rating): number {\n const d0ForGood = initialDifficulty(w, 3);\n const updated = w[7] * d0ForGood + (1 - w[7]) * (d - w[6] * (rating - 3));\n return clamp(updated, 1, 10);\n}\n\n/**\n * Retrievability — probability of recall.\n * R = (1 + elapsed / (9 * S))^(-1)\n */\nfunction retrievability(elapsed: number, stability: number): number {\n if (stability <= 0) return 0;\n return (1 + elapsed / (9 * stability)) ** -1;\n}\n\n/**\n * Stability after a **successful** recall (rating >= 2).\n * S' = S * (exp(w8) * (11 - D) * S^(-w9) * (exp(w10 * (1 - R)) - 1) * hard_penalty * easy_bonus + 1)\n */\nfunction stabilityAfterSuccess(\n w: number[],\n s: number,\n d: number,\n r: number,\n rating: Rating,\n): number {\n const hardPenalty = rating === 2 ? w[15] : 1;\n const easyBonus = rating === 4 ? w[16] : 1;\n\n const inner =\n Math.exp(w[8]) *\n (11 - d) *\n s ** -w[9] *\n (Math.exp(w[10] * (1 - r)) - 1) *\n hardPenalty *\n easyBonus;\n\n return s * (inner + 1);\n}\n\n/**\n * Stability after **forgetting** (rating = 1).\n * S' = w11 * D^(-w12) * ((S+1)^w13 - 1) * exp(w14 * (1 - R))\n */\nfunction stabilityAfterForgetting(\n w: number[],\n s: number,\n d: number,\n r: number,\n): number {\n return (\n w[11] * d ** -w[12] * ((s + 1) ** w[13] - 1) * Math.exp(w[14] * (1 - r))\n );\n}\n\n/**\n * Optimal interval given new stability and target retention.\n * I = 9 * S' * (1/requestRetention - 1)\n * Minimum 1 day.\n */\nfunction nextInterval(stability: number, requestRetention: number): number {\n const interval = 9 * stability * (1 / requestRetention - 1);\n return Math.max(1, Math.round(interval));\n}\n\n// ── Factory ─────────────────────────────────────────────────────────────────\n\n/**\n * Create a new card with default initial values.\n */\nexport function createEmptyCard(now?: Date): SchedulingCard {\n const ts = now ?? new Date();\n return {\n stability: 0,\n difficulty: 0,\n elapsedDays: 0,\n scheduledDays: 0,\n reps: 0,\n lapses: 0,\n state: \"new\",\n dueAt: ts,\n lastReviewAt: null,\n };\n}\n\n/**\n * Create an FSRS scheduler instance.\n *\n * All scheduling is done through pure functions — no side effects,\n * no database access, no mutation of the input card.\n */\nexport function createFSRS(params?: Partial<FSRSParameters>): FSRS {\n const resolvedParams: FSRSParameters = {\n w: params?.w ?? [...DEFAULT_W],\n requestRetention: params?.requestRetention ?? DEFAULT_REQUEST_RETENTION,\n };\n\n function schedule(\n card: SchedulingCard,\n rating: Rating,\n now?: Date,\n ): SchedulingCard {\n const reviewTime = now ?? new Date();\n const w = resolvedParams.w;\n\n // Compute elapsed days since last review (or 0 for new cards).\n const elapsed =\n card.lastReviewAt !== null\n ? Math.max(0, daysBetween(card.lastReviewAt, reviewTime))\n : 0;\n\n // ── New card ──────────────────────────────────────────────────────\n if (card.state === \"new\") {\n const s = initialStability(w, rating);\n const d = initialDifficulty(w, rating);\n const interval = nextInterval(s, resolvedParams.requestRetention);\n\n const dueAt = new Date(reviewTime);\n dueAt.setDate(dueAt.getDate() + interval);\n\n // New cards always move to \"learning\" after first rating.\n\n return {\n stability: s,\n difficulty: d,\n elapsedDays: 0,\n scheduledDays: interval,\n reps: rating >= 2 ? 1 : 0,\n lapses: rating === 1 ? 1 : 0,\n state: \"learning\",\n dueAt,\n lastReviewAt: reviewTime,\n };\n }\n\n // ── Existing card (learning / review / relearning) ───────────────\n\n const r = retrievability(elapsed, card.stability);\n\n let newStability: number;\n let newDifficulty: number;\n let newReps: number;\n let newLapses: number;\n let newState: CardState;\n\n if (rating === 1) {\n // Forgot — apply forgetting formula\n newStability = stabilityAfterForgetting(\n w,\n card.stability,\n card.difficulty,\n r,\n );\n newDifficulty = nextDifficulty(w, card.difficulty, rating);\n newReps = 0;\n newLapses = card.lapses + 1;\n newState = \"relearning\";\n } else {\n // Recalled — apply success formula\n newStability = stabilityAfterSuccess(\n w,\n card.stability,\n card.difficulty,\n r,\n rating,\n );\n newDifficulty = nextDifficulty(w, card.difficulty, rating);\n newReps = card.reps + 1;\n newLapses = card.lapses;\n // Successful recall transitions any state to review.\n newState = \"review\";\n }\n\n const interval = nextInterval(\n newStability,\n resolvedParams.requestRetention,\n );\n\n const dueAt = new Date(reviewTime);\n dueAt.setDate(dueAt.getDate() + interval);\n\n return {\n stability: newStability,\n difficulty: newDifficulty,\n elapsedDays: elapsed,\n scheduledDays: interval,\n reps: newReps,\n lapses: newLapses,\n state: newState,\n dueAt,\n lastReviewAt: reviewTime,\n };\n }\n\n return {\n schedule,\n params: Object.freeze(resolvedParams),\n };\n}\n","/**\n * Cascade Block & Unblock — prerequisite-aware blocking logic.\n *\n * Ported from the PoC's cascade-block and unblock-ready commands.\n *\n * When a user rates a token as \"forgot\" (rating 1) and that token has\n * prerequisites, we block the token and surface its prerequisites into\n * the active deck. When all prerequisites are met, we unblock.\n */\n\nimport type { Database } from \"../db/types.js\";\nimport { ensureCard } from \"../models/card.js\";\nimport { getPrerequisites } from \"../models/prerequisite.js\";\nimport { getTokenBySlug } from \"../models/token.js\";\n\n// ── Types ────────────────────────────────────────────────────────────────────\n\nexport interface CascadeBlockResult {\n blockedSlug: string;\n prerequisites: Array<{ slug: string; concept: string; bloomLevel: number }>;\n}\n\nexport interface UnblockResult {\n unblocked: Array<{ slug: string; concept: string }>;\n}\n\n// ── Functions ────────────────────────────────────────────────────────────────\n\n/**\n * Block a token and surface its prerequisites.\n *\n * Called when a user rates a token as \"forgot\" (rating 1). The token is\n * marked as blocked so it won't appear in review queues. All direct\n * prerequisites are ensured to have cards (unblocked, due now) so they\n * appear in the user's next review session.\n *\n * @param db - Database connection\n * @param userId - The user whose card to block\n * @param tokenSlug - Slug of the token the user forgot\n * @returns Info about what was blocked and which prerequisites were surfaced\n */\nexport async function cascadeBlock(\n db: Database,\n userId: string,\n tokenSlug: string,\n): Promise<CascadeBlockResult> {\n const token = await getTokenBySlug(db, tokenSlug);\n if (!token) {\n throw new Error(`Unknown token slug: ${tokenSlug}`);\n }\n\n const prereqs = await getPrerequisites(db, token.id);\n if (prereqs.length === 0) {\n throw new Error(`Cannot block ${tokenSlug}: token has no prerequisites`);\n }\n\n // Ensure a card exists, then block it\n await ensureCard(db, token.id, userId);\n await db\n .prepare(\"UPDATE cards SET blocked = 1 WHERE token_id = ? AND user_id = ?\")\n .run(token.id, userId);\n\n // Surface all direct prerequisites — ensure cards exist (unblocked, due now)\n const surfaced: Array<{ slug: string; concept: string; bloomLevel: number }> =\n [];\n\n for (const prereq of prereqs) {\n // ensureCard creates a new card if missing (defaults: blocked=0, due_at=now)\n const card = await ensureCard(db, prereq.requires_id, userId);\n\n // If the prerequisite card was somehow blocked with no prereqs of its own,\n // make sure it's unblocked and due now so it surfaces\n if (card.blocked === 1) {\n const prereqOfPrereq = (await db\n .prepare(\"SELECT COUNT(*) as n FROM prerequisites WHERE token_id = ?\")\n .get(prereq.requires_id)) as { n: number };\n\n // Only force-unblock if it has no prerequisites of its own\n if (prereqOfPrereq.n === 0) {\n const now = new Date().toISOString();\n await db\n .prepare(\n \"UPDATE cards SET blocked = 0, due_at = ? WHERE token_id = ? AND user_id = ?\",\n )\n .run(now, prereq.requires_id, userId);\n }\n }\n\n surfaced.push({\n slug: prereq.slug,\n concept: prereq.concept,\n bloomLevel: prereq.bloom_level,\n });\n }\n\n return {\n blockedSlug: tokenSlug,\n prerequisites: surfaced,\n };\n}\n\n/**\n * Scan all blocked cards for a user and unblock any whose prerequisites are met.\n *\n * A blocked card is ready to unblock when ALL of its direct prerequisites have:\n * - reps >= 1 (the user has successfully recalled it at least once)\n * - blocked = 0 (the prerequisite itself is not blocked)\n *\n * If a blocked card has no prerequisites at all, it is unblocked immediately\n * (it was likely blocked in error or its prerequisites were removed).\n *\n * @param db - Database connection\n * @param userId - The user whose blocked cards to check\n * @returns List of cards that were unblocked\n */\nexport async function unblockReady(\n db: Database,\n userId: string,\n): Promise<UnblockResult> {\n const blockedCards = (await db\n .prepare(\n `SELECT c.token_id, t.slug, t.concept\n FROM cards c\n JOIN tokens t ON t.id = c.token_id\n WHERE c.user_id = ? AND c.blocked = 1`,\n )\n .all(userId)) as Array<{\n token_id: string;\n slug: string;\n concept: string;\n }>;\n\n const unblocked: Array<{ slug: string; concept: string }> = [];\n\n for (const card of blockedCards) {\n const totalPrereqs = (await db\n .prepare(\"SELECT COUNT(*) as n FROM prerequisites WHERE token_id = ?\")\n .get(card.token_id)) as { n: number };\n\n const metPrereqs = (await db\n .prepare(\n `SELECT COUNT(*) as n FROM cards c\n JOIN prerequisites p ON p.requires_id = c.token_id\n WHERE p.token_id = ? AND c.user_id = ? AND c.reps >= 1 AND c.blocked = 0`,\n )\n .get(card.token_id, userId)) as { n: number };\n\n if (totalPrereqs.n === 0 || metPrereqs.n === totalPrereqs.n) {\n const now = new Date().toISOString();\n await db\n .prepare(\n \"UPDATE cards SET blocked = 0, due_at = ? WHERE token_id = ? AND user_id = ?\",\n )\n .run(now, card.token_id, userId);\n\n unblocked.push({ slug: card.slug, concept: card.concept });\n }\n }\n\n return { unblocked };\n}\n","/**\n * Shell hook code generation for zsh, bash, and PowerShell.\n *\n * Pure functions that return shell code strings. The CLI command\n * `zam monitor start/stop` calls these and prints to stdout.\n */\n\nfunction psSingleQuoted(value: string): string {\n return `'${value.replace(/'/g, \"''\")}'`;\n}\n\n/**\n * Generate zsh hooks that capture commands to a JSONL file.\n * Uses $EPOCHREALTIME for sub-second timestamp precision.\n */\nexport function generateZshHooks(\n monitorFile: string,\n sessionId: string,\n): string {\n return `\n# ZAM monitor hooks for session ${sessionId}\nexport __ZAM_MONITOR_FILE=\"${monitorFile}\"\nexport __ZAM_MONITOR_SEQ=0\nexport __ZAM_MONITOR_SESSION=\"${sessionId}\"\n\n__zam_ts() {\n if [[ -n \"\\${EPOCHREALTIME:-}\" ]]; then\n local sec=\"\\${EPOCHREALTIME%%.*}\"\n local frac=\"\\${EPOCHREALTIME##*.}\"\n frac=\"\\${frac:0:3}\"\n printf '%s.%sZ' \"$(date -u -r \"$sec\" '+%Y-%m-%dT%H:%M:%S' 2>/dev/null || date -u '+%Y-%m-%dT%H:%M:%S')\" \"$frac\"\n else\n date -u '+%Y-%m-%dT%H:%M:%SZ'\n fi\n}\n\n__zam_preexec() {\n (( __ZAM_MONITOR_SEQ++ ))\n local cmd=\"\\${1//\\\\\"/\\\\\\\\\\\\\"}\"\n local cwd=\"\\${PWD//\\\\\"/\\\\\\\\\\\\\"}\"\n local ts=\"$(__zam_ts)\"\n printf '{\"type\":\"command_start\",\"ts\":\"%s\",\"command\":\"%s\",\"cwd\":\"%s\",\"seq\":%d,\"pid\":%d}\\\\n' \\\\\n \"$ts\" \"$cmd\" \"$cwd\" \"$__ZAM_MONITOR_SEQ\" \"$$\" \\\\\n >> \"$__ZAM_MONITOR_FILE\"\n}\n\n__zam_precmd() {\n local exit_code=$?\n [[ $__ZAM_MONITOR_SEQ -eq 0 ]] && return\n local ts=\"$(__zam_ts)\"\n printf '{\"type\":\"command_end\",\"ts\":\"%s\",\"exit_code\":%d,\"seq\":%d,\"pid\":%d}\\\\n' \\\\\n \"$ts\" \"$exit_code\" \"$__ZAM_MONITOR_SEQ\" \"$$\" \\\\\n >> \"$__ZAM_MONITOR_FILE\"\n}\n\nautoload -Uz add-zsh-hook\nadd-zsh-hook preexec __zam_preexec\nadd-zsh-hook precmd __zam_precmd\n\necho \"ZAM monitor active for session $__ZAM_MONITOR_SESSION\"\n`.trim();\n}\n\n/**\n * Generate bash hooks that capture commands to a JSONL file.\n * Uses DEBUG trap for preexec, PROMPT_COMMAND for precmd.\n */\nexport function generateBashHooks(\n monitorFile: string,\n sessionId: string,\n): string {\n return `\n# ZAM monitor hooks for session ${sessionId}\nexport __ZAM_MONITOR_FILE=\"${monitorFile}\"\nexport __ZAM_MONITOR_SEQ=0\nexport __ZAM_MONITOR_SESSION=\"${sessionId}\"\nexport __ZAM_MONITOR_CMD_ACTIVE=0\n\n__zam_ts() {\n date -u '+%Y-%m-%dT%H:%M:%SZ'\n}\n\n__zam_debug_trap() {\n [[ \"$__ZAM_MONITOR_CMD_ACTIVE\" -eq 1 ]] && return\n __ZAM_MONITOR_CMD_ACTIVE=1\n (( __ZAM_MONITOR_SEQ++ ))\n local cmd=\"\\${BASH_COMMAND//\\\\\"/\\\\\\\\\\\\\"}\"\n local cwd=\"\\${PWD//\\\\\"/\\\\\\\\\\\\\"}\"\n local ts=\"$(__zam_ts)\"\n printf '{\"type\":\"command_start\",\"ts\":\"%s\",\"command\":\"%s\",\"cwd\":\"%s\",\"seq\":%d,\"pid\":%d}\\\\n' \\\\\n \"$ts\" \"$cmd\" \"$cwd\" \"$__ZAM_MONITOR_SEQ\" \"$$\" \\\\\n >> \"$__ZAM_MONITOR_FILE\"\n}\n\n__zam_prompt_cmd() {\n local exit_code=$?\n if [[ \"$__ZAM_MONITOR_CMD_ACTIVE\" -eq 1 ]]; then\n __ZAM_MONITOR_CMD_ACTIVE=0\n local ts=\"$(__zam_ts)\"\n printf '{\"type\":\"command_end\",\"ts\":\"%s\",\"exit_code\":%d,\"seq\":%d,\"pid\":%d}\\\\n' \\\\\n \"$ts\" \"$exit_code\" \"$__ZAM_MONITOR_SEQ\" \"$$\" \\\\\n >> \"$__ZAM_MONITOR_FILE\"\n fi\n}\n\ntrap '__zam_debug_trap' DEBUG\nPROMPT_COMMAND=\"__zam_prompt_cmd;\\${PROMPT_COMMAND:-}\"\n\necho \"ZAM monitor active for session $__ZAM_MONITOR_SESSION\"\n`.trim();\n}\n\n/**\n * Generate PowerShell hooks that capture completed commands to a JSONL file.\n * PowerShell has no zsh-style preexec hook, so this records the most recent\n * history item from the prompt function after each command completes.\n */\nexport function generatePowerShellHooks(\n monitorFile: string,\n sessionId: string,\n): string {\n return `\n# ZAM monitor hooks for session ${sessionId}\n$global:__ZAM_MONITOR_FILE = ${psSingleQuoted(monitorFile)}\n$global:__ZAM_MONITOR_SEQ = 0\n$global:__ZAM_MONITOR_SESSION = ${psSingleQuoted(sessionId)}\n$global:__ZAM_MONITOR_SKIP_NEXT_PROMPT = $true\n\nfunction global:__zam_write_monitor_event {\n param([hashtable]$Event)\n $json = $Event | ConvertTo-Json -Compress -Depth 4\n $utf8NoBom = New-Object System.Text.UTF8Encoding $false\n [System.IO.File]::AppendAllText($global:__ZAM_MONITOR_FILE, $json + [Environment]::NewLine, $utf8NoBom)\n}\n\nfunction global:__zam_iso_utc {\n param([datetime]$Date)\n if ($Date -eq [datetime]::MinValue) {\n return (Get-Date).ToUniversalTime().ToString(\"o\")\n }\n return $Date.ToUniversalTime().ToString(\"o\")\n}\n\nfunction global:__zam_update_last_history_id {\n $history = Get-History -Count 1\n if ($null -ne $history) {\n $global:__ZAM_MONITOR_LAST_HISTORY_ID = $history.Id\n } elseif ($null -eq $global:__ZAM_MONITOR_LAST_HISTORY_ID) {\n $global:__ZAM_MONITOR_LAST_HISTORY_ID = 0\n }\n}\n\nfunction global:__zam_record_last_history {\n param(\n [bool]$Success,\n [object]$NativeExitCode\n )\n\n $history = Get-History -Count 1\n if ($null -eq $history) { return }\n if ($history.Id -le $global:__ZAM_MONITOR_LAST_HISTORY_ID) { return }\n\n $global:__ZAM_MONITOR_LAST_HISTORY_ID = $history.Id\n $global:__ZAM_MONITOR_SEQ += 1\n\n $exitCode = 0\n if (-not $Success) {\n if ($NativeExitCode -is [int] -and $NativeExitCode -ne 0) {\n $exitCode = $NativeExitCode\n } else {\n $exitCode = 1\n }\n }\n\n $cwd = (Get-Location).Path\n __zam_write_monitor_event @{\n type = \"command_start\"\n ts = (__zam_iso_utc $history.StartExecutionTime)\n command = $history.CommandLine\n cwd = $cwd\n seq = $global:__ZAM_MONITOR_SEQ\n pid = $PID\n }\n __zam_write_monitor_event @{\n type = \"command_end\"\n ts = (__zam_iso_utc $history.EndExecutionTime)\n exit_code = $exitCode\n seq = $global:__ZAM_MONITOR_SEQ\n pid = $PID\n }\n}\n\nif (-not (Test-Path function:\\\\__zam_previous_prompt) -and (Test-Path function:\\\\prompt)) {\n Set-Item -Path function:\\\\__zam_previous_prompt -Value (Get-Item function:\\\\prompt).ScriptBlock\n}\n__zam_update_last_history_id\n\nfunction global:prompt {\n $zamSuccess = $?\n $zamNativeExitCode = $global:LASTEXITCODE\n\n if ($global:__ZAM_MONITOR_SKIP_NEXT_PROMPT) {\n __zam_update_last_history_id\n $global:__ZAM_MONITOR_SKIP_NEXT_PROMPT = $false\n } else {\n __zam_record_last_history -Success $zamSuccess -NativeExitCode $zamNativeExitCode\n }\n\n if (Test-Path function:\\\\__zam_previous_prompt) {\n & (Get-Item function:\\\\__zam_previous_prompt).ScriptBlock\n } else {\n \"PS $($executionContext.SessionState.Path.CurrentLocation)$('>' * ($nestedPromptLevel + 1)) \"\n }\n}\n\nWrite-Host \"ZAM monitor active for session $global:__ZAM_MONITOR_SESSION\"\n`.trim();\n}\n\n/** Generate zsh code to remove monitor hooks. */\nexport function generateZshUnhooks(): string {\n return `\n# Remove ZAM monitor hooks\nadd-zsh-hook -d preexec __zam_preexec 2>/dev/null\nadd-zsh-hook -d precmd __zam_precmd 2>/dev/null\nunset -f __zam_preexec __zam_precmd __zam_ts 2>/dev/null\nunset __ZAM_MONITOR_FILE __ZAM_MONITOR_SEQ __ZAM_MONITOR_SESSION 2>/dev/null\necho \"ZAM monitor stopped.\"\n`.trim();\n}\n\n/** Generate bash code to remove monitor hooks. */\nexport function generateBashUnhooks(): string {\n return `\n# Remove ZAM monitor hooks\ntrap - DEBUG\nPROMPT_COMMAND=\"\\${PROMPT_COMMAND/__zam_prompt_cmd;/}\"\nunset -f __zam_debug_trap __zam_prompt_cmd __zam_ts 2>/dev/null\nunset __ZAM_MONITOR_FILE __ZAM_MONITOR_SEQ __ZAM_MONITOR_SESSION __ZAM_MONITOR_CMD_ACTIVE 2>/dev/null\necho \"ZAM monitor stopped.\"\n`.trim();\n}\n\n/** Generate PowerShell code to remove monitor hooks. */\nexport function generatePowerShellUnhooks(): string {\n return `\n# Remove ZAM monitor hooks\nif (Test-Path function:\\\\__zam_previous_prompt) {\n Set-Item -Path function:\\\\prompt -Value (Get-Item function:\\\\__zam_previous_prompt).ScriptBlock\n Remove-Item function:\\\\__zam_previous_prompt -Force -ErrorAction SilentlyContinue\n}\nRemove-Item function:\\\\__zam_write_monitor_event,function:\\\\__zam_iso_utc,function:\\\\__zam_update_last_history_id,function:\\\\__zam_record_last_history -ErrorAction SilentlyContinue\nRemove-Variable -Name __ZAM_MONITOR_FILE,__ZAM_MONITOR_SEQ,__ZAM_MONITOR_SESSION,__ZAM_MONITOR_LAST_HISTORY_ID,__ZAM_MONITOR_SKIP_NEXT_PROMPT -Scope Global -ErrorAction SilentlyContinue\nWrite-Host \"ZAM monitor stopped.\"\n`.trim();\n}\n","/**\n * Skill Discovery — identifies recurring non-standard command patterns\n * across multiple sessions and proposes them as minimal reusable skills.\n *\n * The key insight from Increment 2: \"The human's demonstrated competence\n * is the gate for automation — not the other way around.\" A pattern must\n * appear consistently across sessions before being proposed as a skill.\n *\n * Pure functions — no DB access. Callers provide command records and\n * existing skills; this module returns proposed skills.\n */\n\nimport type { CommandRecord } from \"./analyzer.js\";\n\n// ── Types ────────────────────────────────────────────────────────────────────\n\nexport interface CommandSequence {\n /** The ordered command prefixes forming the pattern (e.g., [\"git checkout\", \"npm install\", \"npm run build\"]) */\n steps: string[];\n /** How many sessions contained this sequence */\n sessionCount: number;\n /** Total occurrences across all sessions */\n totalOccurrences: number;\n /** Example full commands from the most recent occurrence */\n examples: string[];\n}\n\nexport interface SkillProposal {\n /** Suggested slug for the skill */\n slug: string;\n /** Human-readable description of what the pattern does */\n description: string;\n /** The command steps forming the skill */\n steps: string[];\n /** How many sessions demonstrated this pattern */\n sessionCount: number;\n /** Confidence that this is a real, repeatable skill */\n confidence: \"high\" | \"medium\" | \"low\";\n /** Example commands from actual usage */\n examples: string[];\n}\n\nexport interface DiscoveryOptions {\n /** Minimum number of sessions a pattern must appear in (default: 2) */\n minSessions?: number;\n /** Minimum sequence length to consider (default: 2) */\n minSequenceLength?: number;\n /** Maximum sequence length to consider (default: 5) */\n maxSequenceLength?: number;\n /** Existing skill slugs to exclude from proposals */\n existingSkillSlugs?: string[];\n}\n\n// ── Discovery ────────────────────────────────────────────────────────────────\n\n/**\n * Discover recurring command patterns across multiple sessions.\n *\n * Takes a map of session ID → command records, finds command sequences\n * that appear in multiple sessions, and proposes them as skills.\n *\n * @param sessionCommands - Map of session ID to that session's commands\n * @param options - Discovery configuration\n * @returns Array of skill proposals, sorted by confidence then session count\n */\nexport function discoverSkills(\n sessionCommands: Map<string, CommandRecord[]>,\n options: DiscoveryOptions = {},\n): SkillProposal[] {\n const minSessions = options.minSessions ?? 2;\n const minLen = options.minSequenceLength ?? 2;\n const maxLen = options.maxSequenceLength ?? 5;\n const existing = new Set(options.existingSkillSlugs ?? []);\n\n // Step 1: Extract normalized command sequences from each session\n const sessionSequences = new Map<string, string[][]>();\n for (const [sessionId, commands] of sessionCommands) {\n const sequences = extractSequences(commands, minLen, maxLen);\n if (sequences.length > 0) {\n sessionSequences.set(sessionId, sequences);\n }\n }\n\n // Step 2: Count how many sessions contain each unique sequence\n const sequenceIndex = new Map<string, CommandSequence>();\n\n for (const [, sequences] of sessionSequences) {\n // Deduplicate within a session — count each sequence once per session\n const seen = new Set<string>();\n for (const seq of sequences) {\n const key = seq.join(\" → \");\n if (seen.has(key)) continue;\n seen.add(key);\n\n const entry = sequenceIndex.get(key);\n if (entry) {\n entry.sessionCount++;\n entry.totalOccurrences++;\n } else {\n sequenceIndex.set(key, {\n steps: seq,\n sessionCount: 1,\n totalOccurrences: 1,\n examples: [],\n });\n }\n }\n }\n\n // Step 3: Collect examples from the most recent session for qualifying sequences\n const lastSessionId = [...sessionCommands.keys()].pop();\n if (lastSessionId) {\n const lastCommands = sessionCommands.get(lastSessionId)!;\n for (const [_key, entry] of sequenceIndex) {\n if (entry.sessionCount >= minSessions) {\n entry.examples = findExamplesForSequence(lastCommands, entry.steps);\n }\n }\n }\n\n // Step 4: Filter to patterns that appear in enough sessions\n const candidates = [...sequenceIndex.values()].filter(\n (s) => s.sessionCount >= minSessions,\n );\n\n // Step 5: Remove subsequences of longer patterns (prefer maximal patterns)\n const pruned = removeSubsequences(candidates);\n\n // Step 6: Convert to skill proposals\n const proposals: SkillProposal[] = [];\n for (const seq of pruned) {\n const slug = generateSlug(seq.steps);\n\n // Skip if this skill already exists\n if (existing.has(slug)) continue;\n\n proposals.push({\n slug,\n description: describeSequence(seq.steps),\n steps: seq.steps,\n sessionCount: seq.sessionCount,\n confidence:\n seq.sessionCount >= 4\n ? \"high\"\n : seq.sessionCount >= 3\n ? \"medium\"\n : \"low\",\n examples: seq.examples,\n });\n }\n\n // Sort: high confidence first, then by session count\n const confidenceOrder = { high: 0, medium: 1, low: 2 };\n proposals.sort((a, b) => {\n const confDiff =\n confidenceOrder[a.confidence] - confidenceOrder[b.confidence];\n if (confDiff !== 0) return confDiff;\n return b.sessionCount - a.sessionCount;\n });\n\n return proposals;\n}\n\n// ── Helpers ──────────────────────────────────────────────────────────────────\n\n/**\n * Normalize a command to its tool + subcommand prefix.\n * \"git checkout -b feat/foo\" → \"git checkout\"\n * \"npm run build\" → \"npm run build\"\n * \"docker compose up -d\" → \"docker compose up\"\n */\nfunction normalizeCommand(command: string): string {\n const parts = command.trim().split(/\\s+/);\n\n // Known multi-word tools\n const multiWord = [\"docker compose\", \"npm run\", \"npx\", \"git\"];\n const lower = command.toLowerCase();\n\n for (const mw of multiWord) {\n if (lower.startsWith(mw) && parts.length >= mw.split(\" \").length + 1) {\n return parts\n .slice(0, mw.split(\" \").length + 1)\n .join(\" \")\n .toLowerCase();\n }\n }\n\n // Default: first two words\n return parts.slice(0, Math.min(2, parts.length)).join(\" \").toLowerCase();\n}\n\n/**\n * Extract all command subsequences of length minLen..maxLen from a session.\n * Uses normalized command prefixes for comparison.\n */\nfunction extractSequences(\n commands: CommandRecord[],\n minLen: number,\n maxLen: number,\n): string[][] {\n // Filter out trivial commands\n const filtered = commands.filter((c) => {\n const lower = c.command.toLowerCase().trim();\n return (\n lower.length > 0 &&\n !lower.startsWith(\"cd \") &&\n lower !== \"cd\" &&\n lower !== \"ls\" &&\n lower !== \"pwd\" &&\n lower !== \"clear\" &&\n lower !== \"exit\" &&\n !lower.startsWith(\"echo \")\n );\n });\n\n const normalized = filtered.map((c) => normalizeCommand(c.command));\n const sequences: string[][] = [];\n\n for (let len = minLen; len <= maxLen; len++) {\n for (let i = 0; i <= normalized.length - len; i++) {\n const seq = normalized.slice(i, i + len);\n // Only include if at least 2 distinct commands (not just \"git commit\" repeated)\n if (new Set(seq).size >= 2) {\n sequences.push(seq);\n }\n }\n }\n\n return sequences;\n}\n\n/**\n * Find example full commands for a given normalized sequence in a command list.\n */\nfunction findExamplesForSequence(\n commands: CommandRecord[],\n steps: string[],\n): string[] {\n const normalized = commands.map((c) => ({\n norm: normalizeCommand(c.command),\n full: c.command,\n }));\n\n for (let i = 0; i <= normalized.length - steps.length; i++) {\n let match = true;\n for (let j = 0; j < steps.length; j++) {\n if (normalized[i + j].norm !== steps[j]) {\n match = false;\n break;\n }\n }\n if (match) {\n return normalized.slice(i, i + steps.length).map((n) => n.full);\n }\n }\n\n return [];\n}\n\n/**\n * Remove sequences that are strict subsequences of longer qualifying sequences.\n */\nfunction removeSubsequences(candidates: CommandSequence[]): CommandSequence[] {\n // Sort by length descending so we check long sequences first\n const sorted = [...candidates].sort(\n (a, b) => b.steps.length - a.steps.length,\n );\n const result: CommandSequence[] = [];\n\n for (const candidate of sorted) {\n const key = candidate.steps.join(\" → \");\n const isSubsequence = result.some((longer) => {\n const longerKey = longer.steps.join(\" → \");\n return longerKey.includes(key) && longerKey !== key;\n });\n\n if (!isSubsequence) {\n result.push(candidate);\n }\n }\n\n return result;\n}\n\n/**\n * Generate a slug from command steps.\n * [\"git checkout\", \"npm install\", \"npm run build\"] → \"checkout-install-build\"\n */\nfunction generateSlug(steps: string[]): string {\n return steps\n .map((s) => {\n const parts = s.split(/\\s+/);\n return parts[parts.length - 1]; // last word of each step\n })\n .join(\"-\");\n}\n\n/**\n * Generate a human-readable description of what a command sequence does.\n */\nfunction describeSequence(steps: string[]): string {\n return `Recurring pattern: ${steps.join(\" → \")}`;\n}\n","import type { Database } from \"../db/types.js\";\nimport type { DeleteCardResult } from \"../models/card.js\";\nimport { deleteCardForUser, getCardById } from \"../models/card.js\";\nimport { getPrerequisites } from \"../models/prerequisite.js\";\nimport type {\n DeleteTokenResult,\n Token,\n UpdateTokenInput,\n} from \"../models/token.js\";\nimport {\n deleteToken,\n deprecateToken,\n getTokenById,\n updateToken,\n} from \"../models/token.js\";\nimport type { CascadeBlockResult } from \"../scheduler/blocker.js\";\nimport { cascadeBlock } from \"../scheduler/blocker.js\";\nimport type { Rating } from \"../scheduler/fsrs.js\";\nimport type { EvaluateResult } from \"./evaluator.js\";\nimport { evaluateRatingWithinTransaction } from \"./evaluator.js\";\n\nexport type ReviewActionType =\n | \"rate\"\n | \"skip\"\n | \"edit-token\"\n | \"deprecate-token\"\n | \"delete-token\"\n | \"delete-card\"\n | \"stop\";\n\nexport interface ExecuteReviewActionInput {\n cardId: string;\n userId: string;\n action: ReviewActionType;\n rating?: Rating;\n tokenUpdates?: UpdateTokenInput;\n}\n\nexport interface ReviewActionResult {\n action: ReviewActionType;\n token: Token;\n evaluation?: EvaluateResult;\n blocked?: CascadeBlockResult;\n updatedToken?: Token;\n deletedToken?: DeleteTokenResult;\n deletedCard?: DeleteCardResult;\n skipped?: boolean;\n stopped?: boolean;\n}\n\nasync function getReviewTarget(\n db: Database,\n cardId: string,\n userId: string,\n): Promise<{ cardId: string; token: Token }> {\n const card = await getCardById(db, cardId);\n if (!card) {\n throw new Error(`Card not found: ${cardId}`);\n }\n if (card.user_id !== userId) {\n throw new Error(`Card ${cardId} does not belong to user ${userId}`);\n }\n\n const token = await getTokenById(db, card.token_id);\n if (!token) {\n throw new Error(`Token not found for card ${cardId}`);\n }\n\n return { cardId: card.id, token };\n}\n\nexport async function executeReviewAction(\n db: Database,\n input: ExecuteReviewActionInput,\n): Promise<ReviewActionResult> {\n if (input.action === \"rate\") {\n if (input.rating == null) {\n throw new Error(\"rating is required for action=rate\");\n }\n const rating = input.rating;\n\n return db.transaction(async (tx) => {\n const target = await getReviewTarget(tx, input.cardId, input.userId);\n\n const evaluation = await evaluateRatingWithinTransaction(tx, {\n cardId: target.cardId,\n tokenId: target.token.id,\n userId: input.userId,\n rating,\n });\n\n let blocked: CascadeBlockResult | undefined;\n if (rating === 1) {\n const prereqs = await getPrerequisites(tx, target.token.id);\n if (prereqs.length > 0) {\n blocked = await cascadeBlock(tx, input.userId, target.token.slug);\n }\n }\n\n return {\n action: input.action,\n token: target.token,\n evaluation,\n blocked,\n };\n });\n }\n\n const target = await getReviewTarget(db, input.cardId, input.userId);\n\n switch (input.action) {\n case \"skip\":\n return { action: input.action, token: target.token, skipped: true };\n\n case \"stop\":\n return { action: input.action, token: target.token, stopped: true };\n\n case \"edit-token\": {\n const updatedToken = await updateToken(\n db,\n target.token.slug,\n input.tokenUpdates ?? {},\n );\n return {\n action: input.action,\n token: target.token,\n updatedToken,\n };\n }\n\n case \"deprecate-token\": {\n const updatedToken = await deprecateToken(db, target.token.slug);\n return {\n action: input.action,\n token: target.token,\n updatedToken,\n };\n }\n\n case \"delete-token\": {\n const deletedToken = await deleteToken(db, target.token.slug);\n return {\n action: input.action,\n token: target.token,\n deletedToken,\n };\n }\n\n case \"delete-card\": {\n const deletedCard = await deleteCardForUser(\n db,\n target.token.id,\n input.userId,\n );\n return {\n action: input.action,\n token: target.token,\n deletedCard,\n };\n }\n\n default: {\n const exhaustive: never = input.action;\n throw new Error(`Unsupported review action: ${exhaustive}`);\n }\n }\n}\n","/**\n * Active Recall Prompt Generation\n *\n * Generates review prompts from tokens, adapting the question style\n * to the token's Bloom taxonomy level. This is NOT an LLM call —\n * it's template-based prompt assembly for the CLI and bridge.\n */\n\nexport type BloomLevel = 1 | 2 | 3 | 4 | 5;\n\nexport interface RecallPrompt {\n cardId: string;\n tokenId: string;\n slug: string;\n question: string;\n concept: string;\n domain: string;\n bloomLevel: BloomLevel;\n bloomVerb: string;\n hints: string[];\n sourceLink?: string | null;\n}\n\nconst BLOOM_VERBS: Record<BloomLevel, string> = {\n 1: \"Remember\",\n 2: \"Understand\",\n 3: \"Apply\",\n 4: \"Analyze\",\n 5: \"Synthesize\",\n};\n\nfunction formatSlugForCue(slug: string): string {\n return slug.replace(/[-_]/g, \" \");\n}\n\nconst BLOOM_CUES: Record<BloomLevel, (slug: string) => string> = {\n 1: (slug) =>\n `Recall the definition and core concept of: ${formatSlugForCue(slug)}`,\n 2: (slug) => `Explain the concept and how ${formatSlugForCue(slug)} works.`,\n 3: (slug) =>\n `Describe how or where you would apply the concept of ${formatSlugForCue(slug)}.`,\n 4: (slug) =>\n `Analyze the trade-offs, advantages, or alternatives of ${formatSlugForCue(slug)}.`,\n 5: (slug) =>\n `How would you design a solution using the concept of ${formatSlugForCue(slug)}?`,\n};\n\nexport interface PromptInput {\n cardId: string;\n tokenId: string;\n slug: string;\n concept: string;\n domain: string;\n bloomLevel: BloomLevel;\n sourceLink?: string | null;\n question?: string | null;\n}\n\n/**\n * Generate a template-based concept-free recall cue using the slug and domain.\n */\nexport function generateConceptFreeCue(\n bloomLevel: BloomLevel,\n slug: string,\n _domain: string,\n): string {\n const bloom = (\n bloomLevel >= 1 && bloomLevel <= 5 ? bloomLevel : 1\n ) as BloomLevel;\n return BLOOM_CUES[bloom](slug);\n}\n\n/**\n * Generate a recall prompt for a token at its Bloom level.\n * When called from the CLI, the prompt is rendered in the terminal.\n * When called from the AI bridge, the JSON is returned for the AI to present conversationally.\n */\nexport function generatePrompt(input: PromptInput): RecallPrompt {\n const bloom = (\n input.bloomLevel >= 1 && input.bloomLevel <= 5 ? input.bloomLevel : 1\n ) as BloomLevel;\n\n const question = input.question?.trim()\n ? input.question.trim()\n : generateConceptFreeCue(bloom, input.slug, input.domain);\n\n return {\n cardId: input.cardId,\n tokenId: input.tokenId,\n slug: input.slug,\n question,\n concept: input.concept,\n domain: input.domain,\n bloomLevel: bloom,\n bloomVerb: BLOOM_VERBS[bloom],\n hints: [],\n sourceLink: input.sourceLink ?? null,\n };\n}\n","import { existsSync, readFileSync } from \"node:fs\";\nimport { dirname, join, resolve } from \"node:path\";\n\nexport interface ResolvedReference {\n sourceType: \"local\" | \"remote_web\" | \"dynamic_search\";\n content: string;\n filePath?: string;\n url?: string;\n}\n\n/**\n * A source reference resolved and bounded for inclusion in a review payload.\n * Same shape as ResolvedReference plus the originating link and a truncation flag.\n */\nexport interface ReviewContext {\n sourceLink: string;\n sourceType: ResolvedReference[\"sourceType\"];\n content: string;\n filePath?: string;\n url?: string;\n truncated: boolean;\n}\n\n/** Default cap on resolved content length, so bridge JSON / terminal output stays bounded. */\nexport const DEFAULT_REVIEW_CONTEXT_MAX_CHARS = 6000;\n\n/**\n * Strips HTML tags and attempts to convert basic structure to readable text/markdown.\n */\nfunction htmlToText(html: string): string {\n // Extract body if present\n let content = html;\n const bodyMatch = /<body[^>]*>([\\s\\S]*?)<\\/body>/i.exec(html);\n if (bodyMatch) {\n content = bodyMatch[1];\n }\n\n // Strip script, style, and head tags completely\n content = content.replace(/<(script|style|head)[^>]*>([\\s\\S]*?)<\\/\\1>/gi, \"\");\n // Replace headings\n content = content.replace(\n /<h[1-6][^>]*>([\\s\\S]*?)<\\/h[1-6]>/gi,\n \"\\n\\n# $1\\n\",\n );\n // Replace paragraph/div/li tags with line breaks\n content = content.replace(/<(p|div|li)[^>]*>/gi, \"\\n\");\n content = content.replace(/<\\/(p|div|li)>/gi, \"\\n\");\n content = content.replace(/<br\\s*\\/?>/gi, \"\\n\");\n // Strip all other HTML tags\n content = content.replace(/<[^>]+>/g, \"\");\n // Decode basic HTML entities\n content = content\n .replace(/ /g, \" \")\n .replace(/</g, \"<\")\n .replace(/>/g, \">\")\n .replace(/&/g, \"&\")\n .replace(/"/g, '\"')\n .replace(/'/g, \"'\");\n\n // Collapse consecutive newlines\n content = content.replace(/\\n{3,}/g, \"\\n\\n\").trim();\n return content;\n}\n\n/**\n * Parse line anchors like #L10-L25 or #L10 and extract the lines from file content.\n */\nfunction extractLines(content: string, anchor: string): string {\n const lines = content.split(/\\r?\\n/);\n const match = /#L(\\d+)(?:-L(\\d+))?$/i.exec(anchor);\n if (!match) return content;\n\n const start = Number.parseInt(match[1], 10) - 1; // 0-indexed\n const end = match[2] ? Number.parseInt(match[2], 10) - 1 : start;\n\n if (start < 0 || start >= lines.length) return content;\n\n const slice = lines.slice(start, Math.min(end + 1, lines.length));\n return slice.join(\"\\n\");\n}\n\n/**\n * Resolves a given token's source_link into readable textual content.\n */\nexport async function resolveReference(\n sourceLink: string,\n): Promise<ResolvedReference> {\n const cleaned = sourceLink.trim();\n\n // 1. Dynamic Web Search\n if (cleaned.startsWith(\"search://\")) {\n try {\n const url = new URL(cleaned);\n const query = url.searchParams.get(\"q\") || \"\";\n return {\n sourceType: \"dynamic_search\",\n content: `QUERY_DIRECTIVE: Run web search for \"${query}\"`,\n url: cleaned,\n };\n } catch {\n // Fallback if URL parsing fails\n const query = cleaned.replace(/^search:\\/\\/(\\??q=)?/, \"\");\n return {\n sourceType: \"dynamic_search\",\n content: `QUERY_DIRECTIVE: Run web search for \"${decodeURIComponent(query)}\"`,\n url: cleaned,\n };\n }\n }\n\n // 2. HTTP/HTTPS URLs\n if (cleaned.startsWith(\"http://\") || cleaned.startsWith(\"https://\")) {\n // 2.a GitHub URIs\n const gitHubMatch =\n /^https?:\\/\\/(?:www\\.)?github\\.com\\/([^/]+)\\/([^/]+)\\/blob\\/([^/]+)\\/(.+)$/i.exec(\n cleaned,\n );\n if (gitHubMatch) {\n const [_, owner, repo, branch, fullPathWithAnchor] = gitHubMatch;\n const anchorIndex = fullPathWithAnchor.indexOf(\"#\");\n const filePath =\n anchorIndex !== -1\n ? fullPathWithAnchor.slice(0, anchorIndex)\n : fullPathWithAnchor;\n const anchor =\n anchorIndex !== -1 ? fullPathWithAnchor.slice(anchorIndex) : \"\";\n\n // Try local resolution: check if repo folder exists in sibling directories\n const parentDir = dirname(process.cwd());\n const localRepoPath = join(parentDir, repo);\n const localFilePath = join(localRepoPath, filePath);\n\n if (existsSync(localFilePath)) {\n try {\n let fileContent = readFileSync(localFilePath, \"utf-8\");\n if (anchor) {\n fileContent = extractLines(fileContent, anchor);\n }\n return {\n sourceType: \"local\",\n content: fileContent,\n filePath: localFilePath,\n };\n } catch (_e) {\n // Fallback to fetch if file read fails\n }\n }\n\n // Remote fallback: fetch raw content from githubusercontent\n const rawUrl = `https://raw.githubusercontent.com/${owner}/${repo}/${branch}/${filePath}`;\n try {\n const response = await fetch(rawUrl);\n if (response.ok) {\n let rawText = await response.text();\n if (anchor) {\n rawText = extractLines(rawText, anchor);\n }\n return {\n sourceType: \"remote_web\",\n content: rawText,\n url: cleaned,\n };\n }\n } catch (_e) {\n // Fallback to generic URL loading\n }\n }\n\n // 2.b Generic HTTPS/HTTP URLs\n try {\n const response = await fetch(cleaned);\n if (response.ok) {\n const text = await response.text();\n const cleanText = htmlToText(text);\n return {\n sourceType: \"remote_web\",\n content: cleanText,\n url: cleaned,\n };\n }\n throw new Error(`HTTP error ${response.status}: ${response.statusText}`);\n } catch (err) {\n return {\n sourceType: \"remote_web\",\n content: `Error fetching URL reference: ${(err as Error).message}\\nLink: ${cleaned}`,\n url: cleaned,\n };\n }\n }\n\n // 3. Local Workspace Path (relative to process.cwd)\n const anchorIndex = cleaned.indexOf(\"#\");\n const relativePath =\n anchorIndex !== -1 ? cleaned.slice(0, anchorIndex) : cleaned;\n const anchor = anchorIndex !== -1 ? cleaned.slice(anchorIndex) : \"\";\n const absolutePath = resolve(process.cwd(), relativePath);\n\n if (existsSync(absolutePath)) {\n try {\n let fileContent = readFileSync(absolutePath, \"utf-8\");\n if (anchor) {\n fileContent = extractLines(fileContent, anchor);\n }\n return {\n sourceType: \"local\",\n content: fileContent,\n filePath: absolutePath,\n };\n } catch (_e) {\n // Fallback\n }\n }\n\n // Final fallback: return the path/link description as string\n return {\n sourceType: \"local\",\n content: `Local reference file not found or unreadable.\\nReference: ${cleaned}`,\n filePath: absolutePath,\n };\n}\n\n/**\n * Resolve a token's source_link into bounded, review-ready context.\n *\n * Wraps {@link resolveReference} for the review/bridge flow: returns `null`\n * for empty links and caps content length so the surrounding payload (bridge\n * JSON or terminal output) stays manageable, flagging when truncation occurred.\n */\nexport async function resolveReviewContext(\n sourceLink: string | null | undefined,\n opts: { maxChars?: number } = {},\n): Promise<ReviewContext | null> {\n const cleaned = sourceLink?.trim();\n if (!cleaned) return null;\n\n const maxChars = opts.maxChars ?? DEFAULT_REVIEW_CONTEXT_MAX_CHARS;\n const resolved = await resolveReference(cleaned);\n\n let content = resolved.content;\n let truncated = false;\n if (content.length > maxChars) {\n content = content.slice(0, maxChars);\n truncated = true;\n }\n\n return {\n sourceLink: cleaned,\n sourceType: resolved.sourceType,\n content,\n filePath: resolved.filePath,\n url: resolved.url,\n truncated,\n };\n}\n\n/**\n * Normalizes a path, stripping anchors and converting separators.\n */\nexport function normalizePath(p: string): string {\n const base = p.split(\"#\")[0].trim();\n return base.replace(/\\\\/g, \"/\").toLowerCase();\n}\n\n/**\n * Checks if a token's source_link references a changed file.\n */\nexport function matchesFilePath(\n sourceLink: string | null,\n changedFile: string,\n): boolean {\n if (!sourceLink) return false;\n\n const normSource = normalizePath(sourceLink);\n const normChanged = normalizePath(changedFile);\n\n if (!normSource || !normChanged) return false;\n\n // 1. GitHub URI matching\n const gitHubMatch =\n /^https?:\\/\\/(?:www\\.)?github\\.com\\/([^/]+)\\/([^/]+)\\/blob\\/([^/]+)\\/(.+)$/i.exec(\n normSource,\n );\n if (gitHubMatch) {\n const filePath = gitHubMatch[4];\n return filePath === normChanged;\n }\n\n // Generic URL check (don't match web references against local paths)\n if (normSource.startsWith(\"http://\") || normSource.startsWith(\"https://\")) {\n return false;\n }\n\n // 2. Relative/Absolute path matching\n return normSource.endsWith(normChanged) || normChanged.endsWith(normSource);\n}\n","/**\n * Cross-domain interleaving — prevents same-domain streaks in review queues.\n *\n * Given an array of items with a `domain` field, reorders them using a\n * round-robin strategy so that consecutive items come from different domains.\n * This leverages the interleaving effect: mixing topics during practice\n * strengthens discrimination and long-term retention.\n */\n\n// ── Types ────────────────────────────────────────────────────────────────────\n\nexport interface QueueItem {\n cardId: string;\n tokenId: string;\n domain: string;\n}\n\n// ── Functions ────────────────────────────────────────────────────────────────\n\n/**\n * Reorder items so no domain appears more than `maxConsecutive` times in a row.\n *\n * Algorithm: group items by domain, then round-robin across domain groups.\n * Each round picks one item from each non-exhausted domain. Within a domain,\n * the original order is preserved (so urgency sorting survives).\n *\n * If a domain has more items than others, its extras will appear after all\n * other domains are exhausted — but the `maxConsecutive` cap is still\n * respected by inserting items from the largest remaining domains first.\n *\n * @param items - Array of items to interleave. Not mutated.\n * @param maxConsecutive - Max consecutive items from the same domain. Defaults to 2.\n * @returns A new array with the same items in interleaved order.\n */\nexport function interleave<T extends { domain: string }>(\n items: T[],\n maxConsecutive: number = 2,\n): T[] {\n if (items.length <= 1) return [...items];\n\n // Group items by domain, preserving original order within each group\n const byDomain = new Map<string, T[]>();\n for (const item of items) {\n const group = byDomain.get(item.domain);\n if (group) {\n group.push(item);\n } else {\n byDomain.set(item.domain, [item]);\n }\n }\n\n // If there's only one domain, no interleaving possible\n if (byDomain.size === 1) return [...items];\n\n const result: T[] = [];\n let consecutiveCount = 0;\n let lastDomain: string | null = null;\n\n // Track how many items we've consumed from each domain\n const cursors = new Map<string, number>();\n for (const domain of byDomain.keys()) {\n cursors.set(domain, 0);\n }\n\n // Round-robin: sort domains by remaining count (largest first) each round\n while (result.length < items.length) {\n // Get domains that still have items, sorted by remaining count descending\n const activeDomains = [...byDomain.entries()]\n .filter(\n ([domain]) =>\n (cursors.get(domain) ?? 0) < (byDomain.get(domain)?.length ?? 0),\n )\n .sort((a, b) => {\n const remainA = a[1].length - (cursors.get(a[0]) ?? 0);\n const remainB = b[1].length - (cursors.get(b[0]) ?? 0);\n return remainB - remainA;\n });\n\n if (activeDomains.length === 0) break;\n\n let pickedThisRound = false;\n\n for (const [domain, group] of activeDomains) {\n const cursor = cursors.get(domain) ?? 0;\n if (cursor >= group.length) continue;\n\n // Check if adding from this domain would exceed maxConsecutive\n if (domain === lastDomain && consecutiveCount >= maxConsecutive) {\n // Try to find another domain first\n continue;\n }\n\n // Pick one item from this domain\n result.push(group[cursor]);\n cursors.set(domain, cursor + 1);\n pickedThisRound = true;\n\n if (domain === lastDomain) {\n consecutiveCount++;\n } else {\n lastDomain = domain;\n consecutiveCount = 1;\n }\n\n break;\n }\n\n // If we couldn't pick without exceeding maxConsecutive, we must accept\n // a streak from whatever domain has items left\n if (!pickedThisRound) {\n for (const [domain, group] of activeDomains) {\n const cursor = cursors.get(domain) ?? 0;\n if (cursor >= group.length) continue;\n\n result.push(group[cursor]);\n cursors.set(domain, cursor + 1);\n\n if (domain === lastDomain) {\n consecutiveCount++;\n } else {\n lastDomain = domain;\n consecutiveCount = 1;\n }\n\n break;\n }\n }\n }\n\n return result;\n}\n","/**\n * Review Queue Builder — assembles a session's review queue.\n *\n * Combines due-card fetching, new-card selection, urgency sorting,\n * and cross-domain interleaving into a single ready-to-review queue.\n */\n\nimport type { Database } from \"../db/types.js\";\nimport { interleave } from \"./interleaver.js\";\n\n// ── Types ────────────────────────────────────────────────────────────────────\n\nexport interface ReviewQueueOptions {\n userId: string;\n maxNew?: number; // default 10\n maxReviews?: number; // default 50\n now?: Date;\n}\n\nexport interface ReviewQueueItem {\n cardId: string;\n tokenId: string;\n slug: string;\n concept: string;\n domain: string;\n bloomLevel: number;\n state: string; // 'new' | 'learning' | 'review' | 'relearning'\n dueAt: string;\n sourceLink: string | null;\n question: string | null;\n}\n\nexport interface ReviewQueue {\n items: ReviewQueueItem[];\n newCount: number;\n reviewCount: number;\n relearnCount: number;\n totalDomains: string[];\n}\n\n// ── Internal row type from SQL queries ───────────────────────────────────────\n\ninterface CardRow {\n card_id: string;\n token_id: string;\n slug: string;\n concept: string;\n domain: string;\n bloom_level: number;\n state: string;\n due_at: string;\n source_link: string | null;\n question: string | null;\n}\n\n// ── Functions ────────────────────────────────────────────────────────────────\n\n/**\n * Build a review queue for a user's study session.\n *\n * The queue is assembled in stages:\n * 1. Fetch all due cards (not blocked, due_at <= now, state in review/relearning/learning)\n * 2. Fetch new cards (state = 'new', not blocked) up to maxNew\n * 3. Sort overdue cards by urgency — most overdue first\n * 4. Apply cross-domain interleaving to prevent same-domain streaks\n * 5. Intersperse new cards at regular intervals (every 5th position)\n * 6. Cap total at maxReviews\n *\n * @param db - Database connection\n * @param options - Queue building options\n * @returns The assembled review queue with metadata\n */\nexport async function buildReviewQueue(\n db: Database,\n options: ReviewQueueOptions,\n): Promise<ReviewQueue> {\n const maxNew = options.maxNew ?? 10;\n const maxReviews = options.maxReviews ?? 50;\n const now = options.now ?? new Date();\n const nowISO = now.toISOString();\n\n // ── Step 1: Fetch due cards (review, relearning, learning — not new) ───\n const dueRows = (await db\n .prepare(\n `SELECT\n c.id AS card_id,\n c.token_id AS token_id,\n t.slug AS slug,\n t.concept AS concept,\n t.domain AS domain,\n t.bloom_level AS bloom_level,\n c.state AS state,\n c.due_at AS due_at,\n t.source_link AS source_link,\n t.question AS question\n FROM cards c\n JOIN tokens t ON t.id = c.token_id\n WHERE c.user_id = ?\n AND c.blocked = 0\n AND c.due_at <= ?\n AND c.state IN ('review', 'relearning', 'learning')\n AND t.deprecated_at IS NULL\n ORDER BY c.due_at ASC`,\n )\n .all(options.userId, nowISO)) as CardRow[];\n\n // ── Step 2: Fetch new cards ────────────────────────────────────────────\n const newRows = (await db\n .prepare(\n `SELECT\n c.id AS card_id,\n c.token_id AS token_id,\n t.slug AS slug,\n t.concept AS concept,\n t.domain AS domain,\n t.bloom_level AS bloom_level,\n c.state AS state,\n c.due_at AS due_at,\n t.source_link AS source_link,\n t.question AS question\n FROM cards c\n JOIN tokens t ON t.id = c.token_id\n WHERE c.user_id = ?\n AND c.blocked = 0\n AND c.state = 'new'\n AND t.deprecated_at IS NULL\n ORDER BY t.bloom_level ASC, t.slug ASC\n LIMIT ?`,\n )\n .all(options.userId, maxNew)) as CardRow[];\n\n // ── Step 3: Sort overdue cards by urgency (most overdue first) ─────────\n const nowMs = now.getTime();\n const sortedDue = [...dueRows].sort((a, b) => {\n const overdueA = nowMs - new Date(a.due_at).getTime();\n const overdueB = nowMs - new Date(b.due_at).getTime();\n return overdueB - overdueA; // most overdue first\n });\n\n // ── Step 4: Apply cross-domain interleaving to due cards ───────────────\n const interleavedDue = interleave(\n sortedDue.map((row) => ({ ...rowToItem(row), domain: row.domain })),\n );\n\n // ── Step 5: Intersperse new cards at regular intervals ─────────────────\n const newItems = newRows.map(rowToItem);\n const merged = intersperseNew(interleavedDue, newItems, 5);\n\n // ── Step 6: Cap total at maxReviews ────────────────────────────────────\n const capped = merged.slice(0, maxReviews);\n\n // ── Compute metadata ──────────────────────────────────────────────────\n let newCount = 0;\n let reviewCount = 0;\n let relearnCount = 0;\n const domainSet = new Set<string>();\n\n for (const item of capped) {\n domainSet.add(item.domain);\n switch (item.state) {\n case \"new\":\n newCount++;\n break;\n case \"relearning\":\n relearnCount++;\n break;\n default:\n reviewCount++;\n break;\n }\n }\n\n return {\n items: capped,\n newCount,\n reviewCount,\n relearnCount,\n totalDomains: [...domainSet].sort(),\n };\n}\n\n// ── Helpers ──────────────────────────────────────────────────────────────────\n\n/** Convert a SQL row to a ReviewQueueItem. */\nfunction rowToItem(row: CardRow): ReviewQueueItem {\n return {\n cardId: row.card_id,\n tokenId: row.token_id,\n slug: row.slug,\n concept: row.concept,\n domain: row.domain,\n bloomLevel: row.bloom_level,\n state: row.state,\n dueAt: row.due_at,\n sourceLink: row.source_link,\n question: row.question,\n };\n}\n\n/**\n * Intersperse new cards into the review queue at regular intervals.\n *\n * Instead of front-loading or back-loading new cards, places one new card\n * every `interval` positions (e.g., positions 4, 9, 14, ...).\n * This gives the user a mix of familiar reviews and new material.\n *\n * @param reviews - The interleaved review cards\n * @param newCards - New cards to intersperse\n * @param interval - Place a new card every N positions (default 5)\n * @returns Merged array with new cards interspersed\n */\nfunction intersperseNew(\n reviews: ReviewQueueItem[],\n newCards: ReviewQueueItem[],\n interval: number,\n): ReviewQueueItem[] {\n if (newCards.length === 0) return [...reviews];\n if (reviews.length === 0) return [...newCards];\n\n const result: ReviewQueueItem[] = [];\n let reviewIdx = 0;\n let newIdx = 0;\n\n // Position counter tracks where we are in the final queue\n let position = 0;\n\n while (reviewIdx < reviews.length || newIdx < newCards.length) {\n // Insert a new card every `interval` positions (0-indexed: at 4, 9, 14, ...)\n if (\n newIdx < newCards.length &&\n position > 0 &&\n position % interval === interval - 1\n ) {\n result.push(newCards[newIdx]);\n newIdx++;\n } else if (reviewIdx < reviews.length) {\n result.push(reviews[reviewIdx]);\n reviewIdx++;\n } else if (newIdx < newCards.length) {\n // No more reviews — append remaining new cards\n result.push(newCards[newIdx]);\n newIdx++;\n }\n\n position++;\n }\n\n return result;\n}\n","import {\n appendFileSync,\n copyFileSync,\n existsSync,\n mkdirSync,\n readFileSync,\n writeFileSync,\n} from \"node:fs\";\nimport { homedir } from \"node:os\";\nimport { join } from \"node:path\";\nimport { fileURLToPath } from \"node:url\";\n\nconst HOME = homedir();\n\n/**\n * Get the path to ZAM's internal package SKILL.md.\n */\nexport function getPackageSkillPath(\n agent: \"default\" | \"claude\" | \"codex\" = \"default\",\n): string {\n // Support source modules plus the dist/index.js and dist/cli/index.js bundles.\n const packageRoot =\n [\n fileURLToPath(new URL(\"../../..\", import.meta.url)),\n fileURLToPath(new URL(\"../..\", import.meta.url)),\n fileURLToPath(new URL(\"..\", import.meta.url)),\n ].find((candidate) => existsSync(join(candidate, \"package.json\"))) ?? \"\";\n\n if (!packageRoot) return \"\";\n\n if (agent === \"codex\") {\n const codexPath = join(packageRoot, \".agents\", \"skills\", \"zam\", \"SKILL.md\");\n if (existsSync(codexPath)) return codexPath;\n return \"\";\n }\n\n if (agent === \"claude\") {\n const claudePath = join(\n packageRoot,\n \".claude\",\n \"skills\",\n \"zam\",\n \"SKILL.md\",\n );\n if (existsSync(claudePath)) return claudePath;\n return \"\";\n }\n\n // Try .agent first\n let path = join(packageRoot, \".agent\", \"skills\", \"zam\", \"SKILL.md\");\n if (existsSync(path)) return path;\n\n // Try .claude\n path = join(packageRoot, \".claude\", \"skills\", \"zam\", \"SKILL.md\");\n if (existsSync(path)) return path;\n\n return \"\";\n}\n\n/**\n * Distribute the ZAM active-recall training skill globally.\n * Copies SKILL.md into global directories for supported coding agents.\n */\nexport function distributeGlobalSkills(home: string = HOME): Array<{\n name: string;\n path: string;\n success: boolean;\n}> {\n const sourceSkill = getPackageSkillPath();\n const claudeSourceSkill = getPackageSkillPath(\"claude\");\n const codexSourceSkill = getPackageSkillPath(\"codex\");\n const results: Array<{ name: string; path: string; success: boolean }> = [];\n\n if (!sourceSkill) {\n console.warn(\"Could not find ZAM source SKILL.md in the package folder.\");\n return results;\n }\n\n // 1. Claude Code global directory\n const claudeSkillsDir = join(home, \".claude\", \"skills\", \"zam\");\n try {\n if (!claudeSourceSkill) {\n throw new Error(\"Claude skill source not found\");\n }\n mkdirSync(claudeSkillsDir, { recursive: true });\n copyFileSync(claudeSourceSkill, join(claudeSkillsDir, \"SKILL.md\"));\n results.push({\n name: \"Claude Code Global\",\n path: join(claudeSkillsDir, \"SKILL.md\"),\n success: true,\n });\n } catch (_err) {\n results.push({\n name: \"Claude Code Global\",\n path: claudeSkillsDir,\n success: false,\n });\n }\n\n // 2. Gemini/agy global directory\n const geminiSkillsDir = join(home, \".gemini\", \"skills\", \"zam\");\n try {\n mkdirSync(geminiSkillsDir, { recursive: true });\n copyFileSync(sourceSkill, join(geminiSkillsDir, \"SKILL.md\"));\n results.push({\n name: \"Gemini CLI Global\",\n path: join(geminiSkillsDir, \"SKILL.md\"),\n success: true,\n });\n } catch (_err) {\n results.push({\n name: \"Gemini CLI Global\",\n path: geminiSkillsDir,\n success: false,\n });\n }\n\n // 3. Codex global skills directory\n const codexSkillsDir = join(home, \".agents\", \"skills\", \"zam\");\n try {\n if (!codexSourceSkill) {\n throw new Error(\"Codex skill source not found\");\n }\n mkdirSync(codexSkillsDir, { recursive: true });\n copyFileSync(codexSourceSkill, join(codexSkillsDir, \"SKILL.md\"));\n results.push({\n name: \"Codex Global\",\n path: join(codexSkillsDir, \"SKILL.md\"),\n success: true,\n });\n } catch (_err) {\n results.push({\n name: \"Codex Global\",\n path: codexSkillsDir,\n success: false,\n });\n }\n\n // 4. Goose skills directory\n const gooseSkillsDir = join(home, \".goose\", \"skills\", \"zam\");\n try {\n mkdirSync(gooseSkillsDir, { recursive: true });\n copyFileSync(sourceSkill, join(gooseSkillsDir, \"SKILL.md\"));\n results.push({\n name: \"Goose Global\",\n path: join(gooseSkillsDir, \"SKILL.md\"),\n success: true,\n });\n } catch (_err) {\n results.push({\n name: \"Goose Global\",\n path: gooseSkillsDir,\n success: false,\n });\n }\n\n return results;\n}\n\nconst POSIX_OLD_HOOK = `\n# ZAM Shell Observation Hooks\nif (command -v zam >/dev/null 2>&1); then eval \"$(zam monitor start --quiet)\"; fi\n`;\nconst POWERSHELL_OLD_HOOK = `\n# ZAM Shell Observation Hooks\nif (Get-Command zam -ErrorAction SilentlyContinue) { Invoke-Expression (& zam monitor start --quiet pwsh) }\n`;\nconst HOOK_MARKER = \"# ZAM Monitor Session Helper\";\n\nfunction posixHook(shell: \"bash\" | \"zsh\"): string {\n return `\n${HOOK_MARKER}\nzam-monitor-session() {\n local session_id=\"\\${1:-}\"\n if [ -z \"$session_id\" ]; then\n printf 'Usage: zam-monitor-session <session-id>\\n' >&2\n return 2\n fi\n eval \"$(command zam monitor start --session \"$session_id\" --shell ${shell})\"\n}\n`;\n}\n\nconst POWERSHELL_HOOK = `\n${HOOK_MARKER}\nfunction Start-ZamMonitor {\n param([Parameter(Mandatory = $true)][string]$Session)\n Invoke-Expression (& zam monitor start --session $Session --shell pwsh)\n}\n`;\n\nfunction installHook(\n file: string,\n hook: string,\n oldHook: string,\n): { success: boolean; alreadyHooked: boolean } {\n try {\n const content = existsSync(file) ? readFileSync(file, \"utf8\") : \"\";\n if (content.includes(HOOK_MARKER)) {\n return { success: true, alreadyHooked: true };\n }\n\n if (content.includes(oldHook.trim())) {\n writeFileSync(file, content.replace(oldHook.trim(), hook.trim()), \"utf8\");\n } else {\n appendFileSync(file, hook);\n }\n return { success: true, alreadyHooked: false };\n } catch {\n return { success: false, alreadyHooked: false };\n }\n}\n\n/**\n * Add opt-in helpers for starting a monitored session to user shell profiles.\n */\nexport function injectShellHooks(home: string = HOME): Array<{\n shell: string;\n file: string;\n success: boolean;\n alreadyHooked: boolean;\n}> {\n const results: Array<{\n shell: string;\n file: string;\n success: boolean;\n alreadyHooked: boolean;\n }> = [];\n\n // 1. Zsh profile (~/.zshrc)\n const zshrc = join(home, \".zshrc\");\n if (existsSync(zshrc)) {\n const status = installHook(zshrc, posixHook(\"zsh\"), POSIX_OLD_HOOK);\n results.push({ shell: \"zsh\", file: zshrc, ...status });\n }\n\n // 2. Bash profile (~/.bashrc)\n const bashrc = join(home, \".bashrc\");\n if (existsSync(bashrc)) {\n const status = installHook(bashrc, posixHook(\"bash\"), POSIX_OLD_HOOK);\n results.push({ shell: \"bash\", file: bashrc, ...status });\n }\n\n // 3. PowerShell Profile ($HOME\\Documents\\PowerShell\\Microsoft.PowerShell_profile.ps1)\n // Check both PowerShell and WindowsPowerShell\n const pwshDirs = [\n join(home, \"Documents\", \"PowerShell\"),\n join(home, \"Documents\", \"WindowsPowerShell\"),\n ];\n\n for (const dir of pwshDirs) {\n const profileFile = join(dir, \"Microsoft.PowerShell_profile.ps1\");\n try {\n mkdirSync(dir, { recursive: true });\n const status = installHook(\n profileFile,\n POWERSHELL_HOOK,\n POWERSHELL_OLD_HOOK,\n );\n results.push({\n shell: \"powershell\",\n file: profileFile,\n ...status,\n });\n } catch {\n results.push({\n shell: \"powershell\",\n file: profileFile,\n success: false,\n alreadyHooked: false,\n });\n }\n }\n\n return results;\n}\n","import type { SupportedLocale } from \"./locale.js\";\n\nexport type TranslationKey =\n | \"welcome\"\n | \"new_review_relearn\"\n | \"domains\"\n | \"instruction\"\n | \"quit_hint\"\n | \"offline_warning\"\n | \"offline_instruction\"\n | \"nothing_due\"\n | \"evaluating\"\n | \"generating_question\"\n | \"translating\"\n | \"prompt_answer\"\n | \"session_ended\"\n | \"session_complete\"\n | \"cards_rated\"\n | \"avg_rating\"\n | \"forgot\"\n | \"feedback_title\"\n | \"answer_title\"\n | \"keep_waiting\"\n | \"local_ai_working\"\n | \"wait_warning\"\n | \"wait_info\"\n | \"keep_waiting_llm\"\n | \"proceeding_offline\"\n | \"eval_skipped\";\n\nexport const TRANSLATIONS: Record<\n SupportedLocale,\n Record<TranslationKey, string>\n> = {\n en: {\n welcome: \"Learning session: {count} card(s)\",\n new_review_relearn: \" New: {newC} Review: {reviewC} Relearn: {relearnC}\",\n domains: \" Domains: {domains}\",\n instruction:\n \"\\nRecall each answer first, reveal it, then rate yourself honestly.\",\n quit_hint:\n \"Type 'q' at the answer prompt (or press Ctrl+C) to stop anytime.\",\n offline_warning:\n \"\\n\\x1b[33m⚠ LLM-Feedback & automatic translation are disabled.\\x1b[0m\",\n offline_instruction:\n \" Enable with: \\x1b[36mnpm run dev -- settings llm on\\x1b[0m\\n\",\n nothing_due: \"Nothing due to learn. You're all caught up!\",\n evaluating: \"Evaluating answer via local AI...\",\n generating_question: \"Generating dynamic question...\",\n translating: \"Translating question dynamically...\",\n prompt_answer: \"Your answer (Enter to reveal · 'q' to stop):\",\n session_ended: \"Learning session ended.\",\n session_complete: \"Learning session complete!\",\n cards_rated: \" Cards rated: {count}\",\n avg_rating: \" Average rating: {avg}\",\n forgot: \" Forgot: {count} card(s)\",\n feedback_title: \"── ZAM Feedback {line}\",\n answer_title: \"── Answer {line}\",\n keep_waiting: \"Would you like to keep waiting?\",\n local_ai_working: \"The local AI is still generating the response.\",\n wait_warning: \"⚠ The LLM server is taking a while to load the model.\",\n wait_info:\n \"(This is expected when transitioning between models or starting up from cold.)\",\n keep_waiting_llm: \"Would you like to keep waiting for the model?\",\n proceeding_offline:\n \"⚠ Proceeding in offline-mode (without active LLM evaluations for this session).\",\n eval_skipped: \" [LLM Evaluation skipped: {reason}]\",\n },\n de: {\n welcome: \"Lern-Session: {count} Karte(n)\",\n new_review_relearn:\n \" Neu: {newC} Wiederholen: {reviewC} Lernen: {relearnC}\",\n domains: \" Domänen: {domains}\",\n instruction:\n \"\\nRufe jede Antwort zuerst ab, decke sie auf und bewerte dich dann ehrlich selbst.\",\n quit_hint:\n \"Gib 'q' bei der Antwortaufforderung ein (oder drücke Strg+C), um jederzeit zu beenden.\",\n offline_warning:\n \"\\n\\x1b[33m⚠ LLM-Feedback & automatische Übersetzung sind deaktiviert.\\x1b[0m\",\n offline_instruction:\n \" Aktivieren mit: \\x1b[36mnpm run dev -- settings llm on\\x1b[0m\\n\",\n nothing_due: \"Nichts fällig zu lernen. Du bist komplett auf dem Laufenden!\",\n evaluating: \"Bewerte Antwort via lokaler KI...\",\n generating_question: \"Generiere dynamische Frage...\",\n translating: \"Übersetze Frage dynamisch...\",\n prompt_answer: \"Deine Antwort (Eingabe zum Aufdecken · 'q' zum Beenden):\",\n session_ended: \"Lern-Session beendet.\",\n session_complete: \"Lern-Session abgeschlossen!\",\n cards_rated: \" Bewertete Karten: {count}\",\n avg_rating: \" Durchschnittliche Bewertung: {avg}\",\n forgot: \" Vergessen: {count} Karte(n)\",\n feedback_title: \"── ZAM Feedback {line}\",\n answer_title: \"── Antwort {line}\",\n keep_waiting: \"Möchtest du weiter auf die Bewertung warten?\",\n local_ai_working: \"Die lokale KI arbeitet noch an der Antwort.\",\n wait_warning:\n \"⚠ Der LLM-Server braucht ungewöhnlich lange, um das Modell zu laden.\",\n wait_info:\n \"(Das ist normal, wenn das Modell gewechselt wird oder kalt startet.)\",\n keep_waiting_llm: \"Möchtest du weiter auf das Modell warten?\",\n proceeding_offline:\n \"⚠ Fahre im Offline-Modus fort (ohne aktive LLM-Bewertungen in dieser Runde).\",\n eval_skipped: \" [LLM-Bewertung übersprungen: {reason}]\",\n },\n es: {\n welcome: \"Sesión de aprendizaje: {count} tarjeta(s)\",\n new_review_relearn:\n \" Nuevas: {newC} Repasar: {reviewC} Reaprender: {relearnC}\",\n domains: \" Dominios: {domains}\",\n instruction:\n \"\\nRecuerda cada respuesta primero, revélala y califícate honestamente.\",\n quit_hint:\n \"Escribe 'q' en la respuesta (o presiona Ctrl+C) para salir en cualquier momento.\",\n offline_warning:\n \"\\n\\x1b[33m⚠ Los comentarios de LLM y la traducción automática están desactivados.\\x1b[0m\",\n offline_instruction:\n \" Activar con: \\x1b[36mnpm run dev -- settings llm on\\x1b[0m\\n\",\n nothing_due: \"No hay nada pendiente para aprender. ¡Estás al día!\",\n evaluating: \"Evaluando respuesta con IA local...\",\n generating_question: \"Generando pregunta dinámica...\",\n translating: \"Traduciendo pregunta dinámicamente...\",\n prompt_answer: \"Tu respuesta (Intro para revelar · 'q' para salir):\",\n session_ended: \"Sesión de aprendizaje finalizada.\",\n session_complete: \"¡Sesión de aprendizaje completada!\",\n cards_rated: \" Tarjetas calificadas: {count}\",\n avg_rating: \" Calificación promedio: {avg}\",\n forgot: \" Olvidadas: {count} tarjeta(s)\",\n feedback_title: \"── Comentarios de ZAM {line}\",\n answer_title: \"── Respuesta {line}\",\n keep_waiting: \"¿Deseas seguir esperando la evaluación?\",\n local_ai_working: \"La IA local todavía está generando la respuesta.\",\n wait_warning: \"⚠ El servidor LLM está tardando en cargar el modelo.\",\n wait_info: \"(Esto es normal al cambiar de modelo o iniciar en frío.)\",\n keep_waiting_llm: \"¿Deseas seguir esperando el modelo?\",\n proceeding_offline:\n \"⚠ Continuando en modo fuera de línea (sin evaluaciones de LLM en esta sesión).\",\n eval_skipped: \" [Evaluación de LLM omitida: {reason}]\",\n },\n fr: {\n welcome: \"Session d'apprentissage : {count} carte(s)\",\n new_review_relearn:\n \" Nouveau: {newC} Révision: {reviewC} Relever: {relearnC}\",\n domains: \" Domaines: {domains}\",\n instruction:\n \"\\nRappelez-vous chaque réponse d'abord, révélez-la, puis évaluez-vous honnêtement.\",\n quit_hint: \"Tapez 'q' (ou Ctrl+C) pour quitter à tout moment.\",\n offline_warning:\n \"\\n\\x1b[33m⚠ Les commentaires LLM et la traduction automatique sont désactivés.\\x1b[0m\",\n offline_instruction:\n \" Activer avec : \\x1b[36mnpm run dev -- settings llm on\\x1b[0m\\n\",\n nothing_due: \"Rien à apprendre. Vous êtes à jour !\",\n evaluating: \"Évaluation de la réponse via l'IA locale...\",\n generating_question: \"Génération d'une question dynamique...\",\n translating: \"Traduction dynamique de la question...\",\n prompt_answer: \"Votre réponse (Entrée pour révéler · 'q' pour quitter) :\",\n session_ended: \"Session d'apprentissage arrêtée.\",\n session_complete: \"Session d'apprentissage terminée !\",\n cards_rated: \" Cartes évaluées: {count}\",\n avg_rating: \" Note moyenne: {avg}\",\n forgot: \" Oubliées: {count} carte(s)\",\n feedback_title: \"── Commentaires ZAM {line}\",\n answer_title: \"── Réponse {line}\",\n keep_waiting: \"Voulez-vous continuer à attendre l'évaluation ?\",\n local_ai_working:\n \"L'IA locale est toujours en train de générer la réponse.\",\n wait_warning: \"⚠ Le serveur LLM prend du temps pour charger le modèle.\",\n wait_info:\n \"(Ceci est normal lors de la transition entre modèles ou du démarrage à froid.)\",\n keep_waiting_llm: \"Voulez-vous continuer à attendre le modèle ?\",\n proceeding_offline:\n \"⚠ Poursuite en mode hors ligne (sans évaluation active de l'IA pour cette session).\",\n eval_skipped: \" [Évaluation LLM ignorée : {reason}]\",\n },\n pt: {\n welcome: \"Sessão de aprendizado: {count} cartão(ões)\",\n new_review_relearn:\n \" Novos: {newC} Revisar: {reviewC} Reaprender: {relearnC}\",\n domains: \" Domínios: {domains}\",\n instruction:\n \"\\nLembre-se de cada resposta primeiro, revele-a e avalie-se honestamente.\",\n quit_hint: \"Digite 'q' (ou Ctrl+C) para parar a qualquer momento.\",\n offline_warning:\n \"\\n\\x1b[33m⚠ O feedback do LLM e a tradução automática estão desativados.\\x1b[0m\",\n offline_instruction:\n \" Ativar com: \\x1b[36mnpm run dev -- settings llm on\\x1b[0m\\n\",\n nothing_due: \"Nada faturado para aprender. Você está atualizado!\",\n evaluating: \"Avaliando a resposta via IA local...\",\n generating_question: \"Gerando pergunta dinâmica...\",\n translating: \"Traduzindo pergunta dinamicamente...\",\n prompt_answer: \"Sua resposta (Enter para revelar · 'q' para parar):\",\n session_ended: \"Sessão de aprendizado encerrada.\",\n session_complete: \"Sessão de aprendizado concluída!\",\n cards_rated: \" Cartões avaliados: {count}\",\n avg_rating: \" Nota média: {avg}\",\n forgot: \" Esquecidos: {count} cartão(ões)\",\n feedback_title: \"── Feedback ZAM {line}\",\n answer_title: \"── Resposta {line}\",\n keep_waiting: \"Deseja continuar esperando pela avaliação?\",\n local_ai_working: \"A IA local ainda está gerando a resposta.\",\n wait_warning: \"⚠ O servidor LLM está demorando para carregar o modelo.\",\n wait_info: \"(Isso é esperado ao alternar modelos ou iniciar do zero.)\",\n keep_waiting_llm: \"Deseja continuar esperando o modelo?\",\n proceeding_offline:\n \"⚠ Continuando no modo offline (sem avaliações de LLM ativas nesta sessão).\",\n eval_skipped: \" [Avaliação LLM omitida: {reason}]\",\n },\n zh: {\n welcome: \"学习课: {count} 张卡片\",\n new_review_relearn: \" 新卡: {newC} 复习: {reviewC} 重学: {relearnC}\",\n domains: \" 知识领域: {domains}\",\n instruction: \"\\n首先在脑中回忆答案,然后揭晓并诚实自我评分。\",\n quit_hint: \"在回答提示处输入 'q' (或按 Ctrl+C) 可随时退出。\",\n offline_warning: \"\\n\\x1b[33m⚠ LLM 反馈与自动翻译已禁用。\\x1b[0m\",\n offline_instruction:\n \" 开启命令: \\x1b[36mnpm run dev -- settings llm on\\x1b[0m\\n\",\n nothing_due: \"目前没有需要学习的内容。您已全部掌握!\",\n evaluating: \"正在通过本地 AI 评估回答...\",\n generating_question: \"正在动态生成问题...\",\n translating: \"正在动态翻译问题...\",\n prompt_answer: \"您的回答 (按回车揭晓 · 输入 'q' 退出):\",\n session_ended: \"学习课已结束。\",\n session_complete: \"学习课已完成!\",\n cards_rated: \" 已评分卡片: {count}\",\n avg_rating: \" 平均分: {avg}\",\n forgot: \" 遗忘: {count} 张卡片\",\n feedback_title: \"── ZAM 反馈 {line}\",\n answer_title: \"── 参考答案 {line}\",\n keep_waiting: \"是否继续等待评分?\",\n local_ai_working: \"本地 AI 仍在生成回答。\",\n wait_warning: \"⚠ LLM 服务器正在加载模型,这可能需要一些时间。\",\n wait_info: \"(这在切换模型或冷启动时是正常现象。)\",\n keep_waiting_llm: \"是否继续等待模型加载?\",\n proceeding_offline:\n \"⚠ 正在以离线模式继续(本次学习课将不包含活跃的 AI 评估)。\",\n eval_skipped: \" [已跳过 LLM 评估: {reason}]\",\n },\n ja: {\n welcome: \"学習セッション: {count} 枚のカード\",\n new_review_relearn: \" 新規: {newC} 復習: {reviewC} 再学習: {relearnC}\",\n domains: \" ドメイン: {domains}\",\n instruction:\n \"\\n最初に回答を思い出し、次に回答を表示して、正直に自己評価してください。\",\n quit_hint:\n \"回答プロンプトで「q」を入力する(または Ctrl+C を押す)と、いつでも終了できます。\",\n offline_warning:\n \"\\n\\x1b[33m⚠ LLM フィードバックと自動翻訳は無効です。\\x1b[0m\",\n offline_instruction:\n \" 有効化するには: \\x1b[36mnpm run dev -- settings llm on\\x1b[0m\\n\",\n nothing_due: \"学習予定のカードはありません。すべて完了しています!\",\n evaluating: \"ローカルAIによる回答の評価中...\",\n generating_question: \"質問を動的に生成中...\",\n translating: \"質問を動的に翻訳中...\",\n prompt_answer: \"あなたの回答 (Enterで表示 · 'q'で終了):\",\n session_ended: \"学習セッションが終了しました。\",\n session_complete: \"学習セッションが完了しました!\",\n cards_rated: \" 評価済みカード数: {count}\",\n avg_rating: \" 平均評価: {avg}\",\n forgot: \" 忘れたカード数: {count} 枚\",\n feedback_title: \"── ZAM フィードバック {line}\",\n answer_title: \"── 解答 {line}\",\n keep_waiting: \"評価の生成を待ちますか?\",\n local_ai_working: \"ローカルAIが回答を生成しています。\",\n wait_warning:\n \"⚠ LLM サーバーがモデルをロードするのに時間がかかっています。\",\n wait_info: \"(モデルの移行中やコールドスタート時には、これが予想されます。)\",\n keep_waiting_llm: \"モデルのロードを待ち続けますか?\",\n proceeding_offline:\n \"⚠ オフラインモードで続行します(このセッションではアクティブな AI 評価は行われません)。\",\n eval_skipped: \" [LLM 評価がスキップされました: {reason}]\",\n },\n};\n\n/**\n * Format and interpolate a translation string with key-value params.\n */\nexport function t(\n locale: SupportedLocale,\n key: TranslationKey,\n params: Record<string, string | number> = {},\n): string {\n const dict = TRANSLATIONS[locale] || TRANSLATIONS.en;\n let str = dict[key] || TRANSLATIONS.en[key] || \"\";\n\n for (const [k, v] of Object.entries(params)) {\n str = str.replace(new RegExp(`{${k}}`, \"g\"), String(v));\n }\n\n return str;\n}\n","/**\n * Per-machine install configuration — Increment 12, Phase 3.\n *\n * Records whether this machine runs ZAM in \"developer\" mode (source checkout,\n * git-backed workspace, manual `git`/`npm` updates) or \"default\" mode (an\n * installed application updated through a package manager or the in-app\n * updater). Stored in ~/.zam/config.json — a per-machine file, NOT the database\n * and NOT the personal folder, so the mode never travels through a shared Turso\n * database or a synced folder, where it would be wrong for the other machine.\n *\n * The personal-content folder itself is unchanged: it stays the existing\n * `personal.workspace_dir` setting, and can already point at any local or\n * file-synced directory (Drive, OneDrive, Dropbox, iCloud) — no GitHub required.\n */\n\nimport { existsSync, mkdirSync, readFileSync, writeFileSync } from \"node:fs\";\nimport { homedir } from \"node:os\";\nimport { dirname, join } from \"node:path\";\nimport type { InstallChannel } from \"./update-check.js\";\n\nexport type InstallMode = \"developer\" | \"default\";\n\nexport interface InstallConfig {\n mode?: InstallMode;\n /** How this copy was installed; drives the self-update mechanism. */\n channel?: InstallChannel;\n}\n\nconst DEFAULT_CONFIG_PATH = join(homedir(), \".zam\", \"config.json\");\n\n/** Load ~/.zam/config.json. Returns an empty config if missing or unreadable. */\nexport function loadInstallConfig(path = DEFAULT_CONFIG_PATH): InstallConfig {\n if (!existsSync(path)) return {};\n try {\n return JSON.parse(readFileSync(path, \"utf-8\")) as InstallConfig;\n } catch {\n return {};\n }\n}\n\n/** Persist the install config, preserving any unrelated keys already on disk. */\nexport function saveInstallConfig(\n config: InstallConfig,\n path = DEFAULT_CONFIG_PATH,\n): void {\n const dir = dirname(path);\n if (!existsSync(dir)) mkdirSync(dir, { recursive: true });\n writeFileSync(path, `${JSON.stringify(config, null, 2)}\\n`, \"utf-8\");\n}\n\n/**\n * This machine's install mode. Defaults to \"developer\" — the only historical\n * mode — so existing source/CLI installs keep their behavior. A packaged\n * \"default\" install writes mode explicitly at install time.\n */\nexport function getInstallMode(path = DEFAULT_CONFIG_PATH): InstallMode {\n return loadInstallConfig(path).mode ?? \"developer\";\n}\n\nexport function setInstallMode(\n mode: InstallMode,\n path = DEFAULT_CONFIG_PATH,\n): void {\n const config = loadInstallConfig(path);\n config.mode = mode;\n saveInstallConfig(config, path);\n}\n\n/**\n * How this copy was installed, used to pick the self-update mechanism. Falls\n * back to \"developer\" for developer mode and \"direct\" for an installed app\n * whose channel was not recorded.\n */\nexport function getInstallChannel(path = DEFAULT_CONFIG_PATH): InstallChannel {\n const config = loadInstallConfig(path);\n if (config.channel) return config.channel;\n return (config.mode ?? \"developer\") === \"developer\" ? \"developer\" : \"direct\";\n}\n\nexport function setInstallChannel(\n channel: InstallChannel,\n path = DEFAULT_CONFIG_PATH,\n): void {\n const config = loadInstallConfig(path);\n config.channel = channel;\n saveInstallConfig(config, path);\n}\n\n/**\n * Best-effort detection of the file-sync provider a folder lives in, from its\n * path. Used only for friendly messaging (\"this folder syncs via OneDrive —\n * good for moving snapshots between machines\"), never for behavior.\n */\nexport function detectSyncProvider(dir: string): string | null {\n const p = dir.toLowerCase();\n if (p.includes(\"onedrive\")) return \"OneDrive\";\n if (p.includes(\"dropbox\")) return \"Dropbox\";\n if (\n p.includes(\"google drive\") ||\n p.includes(\"googledrive\") ||\n p.includes(\"/my drive\")\n ) {\n return \"Google Drive\";\n }\n if (p.includes(\"icloud\") || p.includes(\"mobile documents\")) {\n return \"iCloud Drive\";\n }\n return null;\n}\n","import { execFileSync, execSync } from \"node:child_process\";\nimport { existsSync } from \"node:fs\";\nimport { homedir } from \"node:os\";\nimport { join } from \"node:path\";\n\nexport interface InstallResult {\n success: boolean;\n message: string;\n}\n\nexport type LocalLLMRunner = \"fastflowlm\" | \"ollama\" | \"generic\";\n\n/** A resolved way to install a tool: a human label and the command to run. */\nexport interface InstallPlan {\n method: string;\n command: string;\n}\n\n/**\n * Check if a command is executable on the system.\n */\nexport function hasCommand(cmd: string): boolean {\n try {\n const checkCmd =\n process.platform === \"win32\" ? `where ${cmd}` : `which ${cmd}`;\n execSync(checkCmd, { stdio: \"ignore\" });\n return true;\n } catch {\n return false;\n }\n}\n\n/**\n * Install FastFlowLM via winget on Windows.\n */\nexport function installFastFlowLM(): InstallResult {\n if (process.platform !== \"win32\") {\n return {\n success: false,\n message: \"FastFlowLM is only supported on Windows.\",\n };\n }\n\n // Check if already installed\n const hasFlm =\n hasCommand(\"flm\") || existsSync(\"C:\\\\Program Files\\\\flm\\\\flm.exe\");\n if (hasFlm) {\n return { success: true, message: \"FastFlowLM is already installed.\" };\n }\n\n if (!hasCommand(\"winget\")) {\n return {\n success: false,\n message: \"winget package manager was not found on this system.\",\n };\n }\n\n console.log(\"Installing FastFlowLM via winget...\");\n try {\n // -e option exact match, --accept-source-agreements --accept-package-agreements\n execSync(\n \"winget install -e --id FastFlowLM --accept-source-agreements --accept-package-agreements\",\n { stdio: \"inherit\" },\n );\n return { success: true, message: \"FastFlowLM installed successfully.\" };\n } catch (err) {\n return {\n success: false,\n message: `Failed to install FastFlowLM: ${(err as Error).message}`,\n };\n }\n}\n\n/**\n * Install Ollama via Homebrew on macOS.\n */\nexport function installOllama(): InstallResult {\n // Check if already installed\n const isMac = process.platform === \"darwin\";\n const isWin = process.platform === \"win32\";\n const hasOllama =\n hasCommand(\"ollama\") ||\n (isMac && existsSync(\"/Applications/Ollama.app\")) ||\n (isWin &&\n existsSync(\n join(homedir(), \"AppData\", \"Local\", \"Programs\", \"Ollama\", \"ollama.exe\"),\n ));\n\n if (hasOllama) {\n return { success: true, message: \"Ollama is already installed.\" };\n }\n\n if (process.platform === \"darwin\") {\n if (!hasCommand(\"brew\")) {\n return {\n success: false,\n message:\n \"Homebrew was not found. Please install Homebrew from brew.sh first.\",\n };\n }\n console.log(\"Installing Ollama via Homebrew Cask...\");\n try {\n execSync(\"brew install --cask ollama\", { stdio: \"inherit\" });\n return { success: true, message: \"Ollama installed successfully.\" };\n } catch (err) {\n return {\n success: false,\n message: `Failed to install Ollama: ${(err as Error).message}`,\n };\n }\n } else if (process.platform === \"win32\") {\n if (!hasCommand(\"winget\")) {\n return {\n success: false,\n message: \"winget was not found. Please install winget first.\",\n };\n }\n console.log(\"Installing Ollama via winget...\");\n try {\n execSync(\n \"winget install -e --id Ollama.Ollama --accept-source-agreements --accept-package-agreements\",\n { stdio: \"inherit\" },\n );\n return { success: true, message: \"Ollama installed successfully.\" };\n } catch (err) {\n return {\n success: false,\n message: `Failed to install Ollama: ${(err as Error).message}`,\n };\n }\n } else {\n // Linux installer script\n console.log(\"Installing Ollama via official installer script...\");\n try {\n execSync(\"curl -fsSL https://ollama.com/install.sh | sh\", {\n stdio: \"inherit\",\n });\n return { success: true, message: \"Ollama installed successfully.\" };\n } catch (err) {\n return {\n success: false,\n message: `Failed to install Ollama: ${(err as Error).message}`,\n };\n }\n }\n}\n\nfunction resolveOllamaCommand(): string | undefined {\n if (hasCommand(\"ollama\")) return \"ollama\";\n\n const candidates =\n process.platform === \"win32\"\n ? [\n join(\n homedir(),\n \"AppData\",\n \"Local\",\n \"Programs\",\n \"Ollama\",\n \"ollama.exe\",\n ),\n ]\n : process.platform === \"darwin\"\n ? [\"/Applications/Ollama.app/Contents/Resources/ollama\"]\n : [];\n\n return candidates.find((candidate) => existsSync(candidate));\n}\n\n/**\n * Prepare the recommended model after installing a local LLM runner.\n */\nexport function prepareLocalModel(\n runner: LocalLLMRunner,\n model: string,\n): InstallResult {\n if (runner === \"fastflowlm\") {\n return {\n success: true,\n message: `${model} will be downloaded by FastFlowLM on first use.`,\n };\n }\n\n if (runner !== \"ollama\") {\n return {\n success: false,\n message: \"No supported local LLM runner was selected.\",\n };\n }\n\n const ollamaCommand = resolveOllamaCommand();\n if (!ollamaCommand) {\n return {\n success: false,\n message:\n \"Ollama was installed but its command is not available yet. Restart the terminal, then run \" +\n `ollama pull ${model}.`,\n };\n }\n\n console.log(`Downloading ${model} with Ollama...`);\n try {\n execFileSync(ollamaCommand, [\"pull\", model], { stdio: \"inherit\" });\n return { success: true, message: `${model} is ready in Ollama.` };\n } catch (err) {\n return {\n success: false,\n message:\n `Could not prepare ${model}: ${(err as Error).message}. ` +\n `Start Ollama and run: ollama pull ${model}`,\n };\n }\n}\n\n/**\n * Pick how to install the opencode agent for the current machine.\n *\n * npm is preferred on every platform: ZAM already requires Node, and the\n * `opencode-ai` package pulls the correct native binary for Apple Silicon and\n * Windows on ARM — avoiding the bash-on-Windows and Homebrew-tap caveats.\n * Returns null when no automatic method is available (e.g. Windows without npm,\n * Scoop, or Chocolatey).\n */\nexport function planOpenCodeInstall(env: {\n platform: NodeJS.Platform;\n hasNpm: boolean;\n hasBrew: boolean;\n hasScoop: boolean;\n hasChoco: boolean;\n}): InstallPlan | null {\n if (env.hasNpm) {\n return { method: \"npm\", command: \"npm install -g opencode-ai\" };\n }\n if (env.platform === \"darwin\") {\n if (env.hasBrew) {\n return {\n method: \"homebrew\",\n command: \"brew install anomalyco/tap/opencode\",\n };\n }\n return {\n method: \"script\",\n command: \"curl -fsSL https://opencode.ai/install | bash\",\n };\n }\n if (env.platform === \"win32\") {\n if (env.hasScoop)\n return { method: \"scoop\", command: \"scoop install opencode\" };\n if (env.hasChoco) {\n return { method: \"chocolatey\", command: \"choco install opencode\" };\n }\n return null;\n }\n // Linux and other Unix-likes.\n return {\n method: \"script\",\n command: \"curl -fsSL https://opencode.ai/install | bash\",\n };\n}\n\n/**\n * Install the opencode agent (the default agent ZAM provisions). opencode reads\n * the AGENTS.md that `zam setup` writes, so it picks up the ZAM skill once both\n * are present.\n */\nexport function installOpenCode(): InstallResult {\n if (hasCommand(\"opencode\")) {\n return { success: true, message: \"opencode is already installed.\" };\n }\n\n const plan = planOpenCodeInstall({\n platform: process.platform,\n hasNpm: hasCommand(\"npm\"),\n hasBrew: hasCommand(\"brew\"),\n hasScoop: hasCommand(\"scoop\"),\n hasChoco: hasCommand(\"choco\"),\n });\n\n if (!plan) {\n return {\n success: false,\n message:\n \"Could not find a way to install opencode automatically. Install npm, \" +\n \"Scoop, or Chocolatey, or follow https://opencode.ai/docs (native \" +\n \"Apple Silicon and Windows on ARM builds are available).\",\n };\n }\n\n console.log(`Installing opencode via ${plan.method}...`);\n try {\n execSync(plan.command, { stdio: \"inherit\" });\n return { success: true, message: `opencode installed via ${plan.method}.` };\n } catch (err) {\n return {\n success: false,\n message:\n `Failed to install opencode: ${(err as Error).message}. ` +\n `Try manually: ${plan.command}`,\n };\n }\n}\n","import { execSync } from \"node:child_process\";\n\nexport type SupportedLocale = \"en\" | \"de\" | \"es\" | \"fr\" | \"pt\" | \"zh\" | \"ja\";\n\nconst SUPPORTED_LOCALES: Set<SupportedLocale> = new Set([\n \"en\",\n \"de\",\n \"es\",\n \"fr\",\n \"pt\",\n \"zh\",\n \"ja\",\n]);\n\n/**\n * Clean and map raw locale string (e.g., \"de_DE.UTF-8\" or \"en-US\") to SupportedLocale.\n */\nexport function normalizeLocale(raw: string): SupportedLocale {\n const clean = raw.trim().toLowerCase().split(/[_-]/)[0];\n if (SUPPORTED_LOCALES.has(clean as SupportedLocale)) {\n return clean as SupportedLocale;\n }\n return \"en\";\n}\n\n/**\n * Detect the operating system's active language code dynamically.\n */\nexport function detectSystemLocale(): SupportedLocale {\n try {\n // 1. Check standard POSIX env vars (common on macOS/Linux/Git Bash/WSL)\n const envVars = [\n process.env.LANG,\n process.env.LANGUAGE,\n process.env.LC_ALL,\n process.env.LC_MESSAGES,\n ];\n\n for (const val of envVars) {\n if (val && val.trim().length > 0) {\n return normalizeLocale(val);\n }\n }\n\n // 2. On Windows, fallback to querying PowerShell Culture\n if (process.platform === \"win32\") {\n const output = execSync(\n 'powershell -NoProfile -Command \"[System.Globalization.CultureInfo]::CurrentCulture.Name\"',\n { stdio: \"pipe\", encoding: \"utf8\", timeout: 2000 },\n ).trim();\n if (output && output.length > 0) {\n return normalizeLocale(output);\n }\n }\n } catch {\n // Ignore errors and default to English\n }\n\n return \"en\";\n}\n","import { execSync } from \"node:child_process\";\n\nexport interface SystemProfile {\n os: \"windows\" | \"macos\" | \"linux\" | \"unknown\";\n arch: \"x64\" | \"arm64\" | \"unknown\";\n hasRyzenNPU: boolean;\n hasAppleSilicon: boolean;\n recommendedRunner: \"fastflowlm\" | \"ollama\" | \"generic\";\n recommendedModel: string;\n}\n\n/**\n * Run a shell command synchronously and return stdout.\n * Returns empty string on failure.\n */\nfunction runCommand(cmd: string): string {\n try {\n return execSync(cmd, { stdio: \"pipe\", encoding: \"utf8\" }).trim();\n } catch {\n return \"\";\n }\n}\n\nfunction detectWindowsAMDIPU(): boolean {\n if (process.platform !== \"win32\") return false;\n\n // WMI query for AMD IPU (Image Processing Unit), NPU, Ryzen AI CPUs, and modern NPU compute devices (DEV_1502, DEV_17F0)\n const cmd = `powershell -NoProfile -Command \"Get-CimInstance Win32_PnPEntity | Where-Object { $_.Name -like '*AMD IPU*' -or $_.Name -like '*AMD NPU*' -or $_.Name -like '*NPU Compute*' -or $_.Name -like '*Ryzen AI*' -or $_.HardwareID -like '*VEN_1022&DEV_1502*' -or $_.HardwareID -like '*VEN_1022&DEV_17F0*' } | Select-Object -First 1 -ExpandProperty Name\"`;\n const output = runCommand(cmd);\n\n return Boolean(\n output &&\n (output.toLowerCase().includes(\"amd\") ||\n output.toLowerCase().includes(\"ipu\") ||\n output.toLowerCase().includes(\"npu\") ||\n output.toLowerCase().includes(\"ryzen\")),\n );\n}\n\n/**\n * Profile the active system hardware and software capabilities.\n */\nexport function getSystemProfile(): SystemProfile {\n const platform = process.platform;\n const archStr = process.arch;\n\n let os: \"windows\" | \"macos\" | \"linux\" | \"unknown\" = \"unknown\";\n if (platform === \"win32\") os = \"windows\";\n else if (platform === \"darwin\") os = \"macos\";\n else if (platform === \"linux\") os = \"linux\";\n\n let arch: \"x64\" | \"arm64\" | \"unknown\" = \"unknown\";\n if (archStr === \"x64\") arch = \"x64\";\n else if (archStr === \"arm64\") arch = \"arm64\";\n\n const hasRyzenNPU = os === \"windows\" && detectWindowsAMDIPU();\n const hasAppleSilicon = os === \"macos\" && arch === \"arm64\";\n\n let recommendedRunner: \"fastflowlm\" | \"ollama\" | \"generic\" = \"generic\";\n let recommendedModel = \"qwen3.5:4b\";\n\n if (hasRyzenNPU) {\n recommendedRunner = \"fastflowlm\";\n recommendedModel = \"qwen3.5:4b\";\n } else if (hasAppleSilicon) {\n recommendedRunner = \"ollama\";\n recommendedModel = \"llama3.2:3b\";\n } else if (os === \"macos\" || os === \"linux\" || os === \"windows\") {\n // Standard PC / generic Mac\n recommendedRunner = \"ollama\";\n recommendedModel = \"llama3.2:3b\";\n }\n\n return {\n os,\n arch,\n hasRyzenNPU,\n hasAppleSilicon,\n recommendedRunner,\n recommendedModel,\n };\n}\n","import { existsSync } from \"node:fs\";\nimport { resolve } from \"node:path\";\nimport type { Database } from \"../db/types.js\";\nimport { getSetting } from \"../models/settings.js\";\n\nexport interface RepoPaths {\n personal: string | null;\n team: string | null;\n org: string | null;\n}\n\n/**\n * Resolve absolute paths for personal, team, and organization repositories.\n * Personal falls back to personal.workspace_dir if repo.personal is not set.\n */\nexport async function getRepoPaths(db: Database): Promise<RepoPaths> {\n const personalSetting =\n (await getSetting(db, \"repo.personal\")) ||\n (await getSetting(db, \"personal.workspace_dir\"));\n const teamSetting = await getSetting(db, \"repo.team\");\n const orgSetting = await getSetting(db, \"repo.org\");\n\n return {\n personal: personalSetting ? resolve(personalSetting) : null,\n team: teamSetting ? resolve(teamSetting) : null,\n org: orgSetting ? resolve(orgSetting) : null,\n };\n}\n\n/**\n * Resolve a specific repo's path, or null if not configured.\n */\nexport async function resolveRepoPath(\n db: Database,\n type: \"personal\" | \"team\" | \"org\",\n): Promise<string | null> {\n const paths = await getRepoPaths(db);\n return paths[type];\n}\n\n/**\n * Resolve paths to all existing \"/beliefs\" directories in the hierarchy,\n * sorted from most specific (personal) to most general (org).\n */\nexport async function resolveAllBeliefPaths(db: Database): Promise<string[]> {\n const paths = await getRepoPaths(db);\n const dirs: string[] = [];\n\n if (paths.personal) {\n const personalDir = resolve(paths.personal, \"beliefs\");\n if (existsSync(personalDir)) dirs.push(personalDir);\n }\n if (paths.team) {\n const teamDir = resolve(paths.team, \"beliefs\");\n if (existsSync(teamDir)) dirs.push(teamDir);\n }\n if (paths.org) {\n const orgDir = resolve(paths.org, \"beliefs\");\n if (existsSync(orgDir)) dirs.push(orgDir);\n }\n\n return dirs;\n}\n\n/**\n * Resolve paths to all existing \"/goals\" directories in the hierarchy,\n * sorted from most specific (personal) to most general (org).\n */\nexport async function resolveAllGoalPaths(db: Database): Promise<string[]> {\n const paths = await getRepoPaths(db);\n const dirs: string[] = [];\n\n if (paths.personal) {\n const personalDir = resolve(paths.personal, \"goals\");\n if (existsSync(personalDir)) dirs.push(personalDir);\n }\n if (paths.team) {\n const teamDir = resolve(paths.team, \"goals\");\n if (existsSync(teamDir)) dirs.push(teamDir);\n }\n if (paths.org) {\n const orgDir = resolve(paths.org, \"goals\");\n if (existsSync(orgDir)) dirs.push(orgDir);\n }\n\n return dirs;\n}\n","/**\n * Update-decision logic — Increment 12, Phase 5.\n *\n * The brain behind \"the app noticed a newer version\": given the current and\n * latest versions and how this copy was installed, decide what the UI should\n * do. Deliberately network-free and pure, so it is fully unit-tested; the\n * actual version fetch (e.g. GitHub releases) and the Tauri self-update live in\n * the CLI/desktop layers that call this.\n */\n\nexport type InstallChannel = \"developer\" | \"direct\" | \"winget\" | \"homebrew\";\n\n/** Provisional package identifiers; finalized when channels ship (Phase 2). */\nexport const WINGET_PACKAGE_ID = \"ZAM.ZAM\";\nexport const HOMEBREW_CASK = \"zam\";\n\nexport type UpdateActionKind =\n | \"none\"\n | \"self-update\"\n | \"run-command\"\n | \"inform\";\n\nexport interface UpdateDecision {\n updateAvailable: boolean;\n currentVersion: string;\n latestVersion: string;\n channel: InstallChannel;\n /** What the UI should do about the update. */\n action: UpdateActionKind;\n /** For \"run-command\"/\"inform\": the command to surface to the user. */\n command?: string;\n /** Locale-agnostic explanation; the UI provides its own localized copy. */\n reason: string;\n}\n\ninterface ParsedVersion {\n core: number[];\n pre: string[];\n}\n\nfunction parseVersion(version: string): ParsedVersion {\n const clean = version.trim().replace(/^v/i, \"\");\n const [main, pre = \"\"] = clean.split(\"-\", 2);\n const core = main.split(\".\").map((n) => Number.parseInt(n, 10) || 0);\n while (core.length < 3) core.push(0);\n return { core, pre: pre ? pre.split(\".\") : [] };\n}\n\n/**\n * Compare two semver-ish versions. Returns 1 if `a` is newer than `b`, -1 if\n * older, 0 if equal. A version with a prerelease tag (1.0.0-beta) sorts below\n * the same core release (1.0.0), per semver.\n */\nexport function compareVersions(a: string, b: string): -1 | 0 | 1 {\n const pa = parseVersion(a);\n const pb = parseVersion(b);\n\n for (let i = 0; i < 3; i++) {\n if (pa.core[i] !== pb.core[i]) return pa.core[i] > pb.core[i] ? 1 : -1;\n }\n\n if (pa.pre.length === 0 && pb.pre.length === 0) return 0;\n if (pa.pre.length === 0) return 1; // release > prerelease\n if (pb.pre.length === 0) return -1;\n\n const len = Math.max(pa.pre.length, pb.pre.length);\n for (let i = 0; i < len; i++) {\n const x = pa.pre[i];\n const y = pb.pre[i];\n if (x === undefined) return -1; // shorter prerelease set has lower precedence\n if (y === undefined) return 1;\n\n const xNum = /^\\d+$/.test(x);\n const yNum = /^\\d+$/.test(y);\n if (xNum && yNum) {\n const dx = Number(x);\n const dy = Number(y);\n if (dx !== dy) return dx > dy ? 1 : -1;\n } else if (xNum !== yNum) {\n return xNum ? -1 : 1; // numeric identifiers rank below alphanumeric\n } else if (x !== y) {\n return x > y ? 1 : -1;\n }\n }\n return 0;\n}\n\n/**\n * Decide what to do given current/latest versions and the install channel.\n * The mechanism follows how the copy was installed so we never self-replace a\n * package-managed install or a source checkout.\n */\nexport function decideUpdate(input: {\n currentVersion: string;\n latestVersion: string;\n channel: InstallChannel;\n}): UpdateDecision {\n const { currentVersion, latestVersion, channel } = input;\n const base = { currentVersion, latestVersion, channel };\n\n if (compareVersions(latestVersion, currentVersion) <= 0) {\n return {\n ...base,\n updateAvailable: false,\n action: \"none\",\n reason: \"Already on the latest version.\",\n };\n }\n\n switch (channel) {\n case \"direct\":\n return {\n ...base,\n updateAvailable: true,\n action: \"self-update\",\n reason: \"A signed update can be installed in place.\",\n };\n case \"winget\":\n return {\n ...base,\n updateAvailable: true,\n action: \"run-command\",\n command: `winget upgrade --id ${WINGET_PACKAGE_ID}`,\n reason: \"Update available through winget.\",\n };\n case \"homebrew\":\n return {\n ...base,\n updateAvailable: true,\n action: \"run-command\",\n command: `brew upgrade --cask ${HOMEBREW_CASK}`,\n reason: \"Update available through Homebrew.\",\n };\n default:\n return {\n ...base,\n updateAvailable: true,\n action: \"inform\",\n command: \"git pull && npm install && npm run build\",\n reason: \"Developer install — update from source.\",\n };\n }\n}\n"],"mappings":"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","names":["existsSync","mkdirSync","readFileSync","homedir","dirname","join","join","homedir","require","existsSync","readFileSync","db","dirname","mkdirSync","count","existsSync","readFileSync","writeFileSync","join","existsSync","join","readFileSync","t","writeFileSync","ulid","ulid","t","t","ulid","ulid","count","existsSync","mkdirSync","readFileSync","homedir","join","join","homedir","existsSync","mkdirSync","readFileSync","ulid","ulid","interval","dueAt","ulid","ulid","target","existsSync","readFileSync","dirname","join","anchorIndex","anchor","appendFileSync","existsSync","mkdirSync","readFileSync","writeFileSync","homedir","join","existsSync","mkdirSync","readFileSync","writeFileSync","homedir","dirname","join","existsSync","homedir","join","execSync","execSync","existsSync","resolve","resolve","existsSync"]}
|