sandstream-kit 1.0.1 → 1.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (84) hide show
  1. package/README.md +18 -0
  2. package/dist/cli.js +458 -2
  3. package/dist/cli.js.map +1 -1
  4. package/dist/database.d.ts +2 -2
  5. package/dist/database.js +9 -14
  6. package/dist/database.js.map +1 -1
  7. package/dist/lock.js +5 -3
  8. package/dist/lock.js.map +1 -1
  9. package/dist/memory/backup 2.d.ts +6 -0
  10. package/dist/memory/backup 2.js +80 -0
  11. package/dist/memory/backup 2.js.map +1 -0
  12. package/dist/memory/backup.d.ts +6 -0
  13. package/dist/memory/backup.js +80 -0
  14. package/dist/memory/backup.js.map +1 -0
  15. package/dist/memory/db 2.d.ts +40 -0
  16. package/dist/memory/db 2.js +233 -0
  17. package/dist/memory/db 2.js.map +1 -0
  18. package/dist/memory/db.d.ts +40 -0
  19. package/dist/memory/db.js +233 -0
  20. package/dist/memory/db.js.map +1 -0
  21. package/dist/memory/hook 2.d.ts +6 -0
  22. package/dist/memory/hook 2.js +51 -0
  23. package/dist/memory/hook 2.js.map +1 -0
  24. package/dist/memory/hook.d.ts +6 -0
  25. package/dist/memory/hook.js +51 -0
  26. package/dist/memory/hook.js.map +1 -0
  27. package/dist/memory/hook.test 2.d.ts +1 -0
  28. package/dist/memory/hook.test 2.js +35 -0
  29. package/dist/memory/hook.test 2.js.map +1 -0
  30. package/dist/memory/install 2.d.ts +8 -0
  31. package/dist/memory/install 2.js +72 -0
  32. package/dist/memory/install 2.js.map +1 -0
  33. package/dist/memory/install.d.ts +8 -0
  34. package/dist/memory/install.js +72 -0
  35. package/dist/memory/install.js.map +1 -0
  36. package/dist/memory/install.test 2.d.ts +1 -0
  37. package/dist/memory/install.test 2.js +59 -0
  38. package/dist/memory/install.test 2.js.map +1 -0
  39. package/dist/memory/pal 2.d.ts +47 -0
  40. package/dist/memory/pal 2.js +154 -0
  41. package/dist/memory/pal 2.js.map +1 -0
  42. package/dist/memory/pal.d.ts +47 -0
  43. package/dist/memory/pal.js +154 -0
  44. package/dist/memory/pal.js.map +1 -0
  45. package/dist/memory/parser.d.ts +25 -0
  46. package/dist/memory/parser.js +164 -0
  47. package/dist/memory/parser.js.map +1 -0
  48. package/dist/memory/project 2.d.ts +1 -0
  49. package/dist/memory/project 2.js +24 -0
  50. package/dist/memory/project 2.js.map +1 -0
  51. package/dist/memory/project.d.ts +1 -0
  52. package/dist/memory/project.js +24 -0
  53. package/dist/memory/project.js.map +1 -0
  54. package/dist/memory/scan 2.d.ts +15 -0
  55. package/dist/memory/scan 2.js +94 -0
  56. package/dist/memory/scan 2.js.map +1 -0
  57. package/dist/memory/scan.d.ts +15 -0
  58. package/dist/memory/scan.js +94 -0
  59. package/dist/memory/scan.js.map +1 -0
  60. package/dist/memory/scan.test 2.d.ts +1 -0
  61. package/dist/memory/scan.test 2.js +55 -0
  62. package/dist/memory/scan.test 2.js.map +1 -0
  63. package/dist/memory/shared 2.d.ts +33 -0
  64. package/dist/memory/shared 2.js +120 -0
  65. package/dist/memory/shared 2.js.map +1 -0
  66. package/dist/memory/shared.d.ts +33 -0
  67. package/dist/memory/shared.js +120 -0
  68. package/dist/memory/shared.js.map +1 -0
  69. package/dist/memory/threads 2.d.ts +37 -0
  70. package/dist/memory/threads 2.js +50 -0
  71. package/dist/memory/threads 2.js.map +1 -0
  72. package/dist/memory/threads.d.ts +37 -0
  73. package/dist/memory/threads.js +50 -0
  74. package/dist/memory/threads.js.map +1 -0
  75. package/dist/memory/threads.test 2.d.ts +1 -0
  76. package/dist/memory/threads.test 2.js +66 -0
  77. package/dist/memory/threads.test 2.js.map +1 -0
  78. package/dist/memory/types 2.d.ts +52 -0
  79. package/dist/memory/types 2.js +5 -0
  80. package/dist/memory/types 2.js.map +1 -0
  81. package/dist/memory/types.d.ts +52 -0
  82. package/dist/memory/types.js +5 -0
  83. package/dist/memory/types.js.map +1 -0
  84. package/package.json +1 -1
@@ -0,0 +1,40 @@
1
+ /**
2
+ * kit memory — local SQLite store (node:sqlite + FTS5).
3
+ *
4
+ * Schema is derived from cloudctx (MIT — github.com/chadptk1238/cloudctx): we reuse
5
+ * its proven table layout, PRAGMAs and FTS5 design. Differences: kit is a Node/TS
6
+ * project, so we use the built-in `node:sqlite` driver instead of `bun:sqlite`; we
7
+ * add a `harness` column (kit is harness-agnostic) and a `pending_actions` table
8
+ * (PAL — the structured, actionable layer on top of raw conversation memory).
9
+ *
10
+ * Memory is RAW and append-only: one row per message, no summarisation. Retrieval
11
+ * (FTS5) happens at time of work — store everything raw, search on demand. This DB
12
+ * is secret-dense (it indexes private transcripts): it lives only under ~/.kit/ with
13
+ * 0600 perms and is never committed. Redaction / encryption is stage B7.
14
+ */
15
+ import { DatabaseSync } from "node:sqlite";
16
+ import type { MemoryStats, MessageInput, SearchHit, SessionInput, ToolUseInput } from "./types.js";
17
+ export declare const SCHEMA_VERSION = 2;
18
+ export declare function getMemoryDir(): string;
19
+ export declare function getMemoryDbPath(): string;
20
+ /**
21
+ * Open (creating + migrating if needed) the memory database. Pass ":memory:" for
22
+ * an ephemeral in-process DB (tests). Otherwise defaults to ~/.kit/memory.db.
23
+ */
24
+ export declare function openMemoryDb(path?: string): DatabaseSync;
25
+ export declare function upsertSession(db: DatabaseSync, s: SessionInput): void;
26
+ /** Insert a message idempotently (by uuid). Returns true if a new row was added. */
27
+ export declare function insertMessage(db: DatabaseSync, m: MessageInput): boolean;
28
+ export declare function insertToolUse(db: DatabaseSync, t: ToolUseInput): void;
29
+ export interface SearchOptions {
30
+ limit?: number;
31
+ /** Restrict to messages whose cwd is this repo root (or a subdirectory). */
32
+ projectPath?: string;
33
+ }
34
+ /**
35
+ * Full-text search over raw message content (FTS5 MATCH, ranked by `rank`).
36
+ * Pass opts.projectPath to scope to one repo (relevance + blast-radius); omit it
37
+ * for cross-project ("--global") recall over the personal store.
38
+ */
39
+ export declare function searchMessages(db: DatabaseSync, query: string, opts?: SearchOptions): SearchHit[];
40
+ export declare function getStats(db: DatabaseSync): MemoryStats;
@@ -0,0 +1,233 @@
1
+ /**
2
+ * kit memory — local SQLite store (node:sqlite + FTS5).
3
+ *
4
+ * Schema is derived from cloudctx (MIT — github.com/chadptk1238/cloudctx): we reuse
5
+ * its proven table layout, PRAGMAs and FTS5 design. Differences: kit is a Node/TS
6
+ * project, so we use the built-in `node:sqlite` driver instead of `bun:sqlite`; we
7
+ * add a `harness` column (kit is harness-agnostic) and a `pending_actions` table
8
+ * (PAL — the structured, actionable layer on top of raw conversation memory).
9
+ *
10
+ * Memory is RAW and append-only: one row per message, no summarisation. Retrieval
11
+ * (FTS5) happens at time of work — store everything raw, search on demand. This DB
12
+ * is secret-dense (it indexes private transcripts): it lives only under ~/.kit/ with
13
+ * 0600 perms and is never committed. Redaction / encryption is stage B7.
14
+ */
15
+ import { DatabaseSync } from "node:sqlite";
16
+ import { homedir } from "node:os";
17
+ import { join } from "node:path";
18
+ import { existsSync, mkdirSync, chmodSync, statSync } from "node:fs";
19
+ export const SCHEMA_VERSION = 2;
20
+ export function getMemoryDir() {
21
+ return process.env.KIT_MEMORY_DIR ?? join(homedir(), ".kit");
22
+ }
23
+ export function getMemoryDbPath() {
24
+ return process.env.KIT_MEMORY_DB ?? join(getMemoryDir(), "memory.db");
25
+ }
26
+ function ensureMemoryDir() {
27
+ const dir = getMemoryDir();
28
+ if (!existsSync(dir))
29
+ mkdirSync(dir, { recursive: true, mode: 0o700 });
30
+ }
31
+ const SCHEMA_SQL = `
32
+ CREATE TABLE IF NOT EXISTS sessions (
33
+ id INTEGER PRIMARY KEY AUTOINCREMENT,
34
+ session_id TEXT UNIQUE NOT NULL,
35
+ harness TEXT NOT NULL DEFAULT 'claude-code',
36
+ project TEXT,
37
+ first_message_at TEXT,
38
+ last_message_at TEXT,
39
+ message_count INTEGER DEFAULT 0,
40
+ is_agent_sidechain INTEGER DEFAULT 0
41
+ );
42
+
43
+ CREATE TABLE IF NOT EXISTS messages (
44
+ id INTEGER PRIMARY KEY AUTOINCREMENT,
45
+ uuid TEXT UNIQUE,
46
+ session_id TEXT NOT NULL,
47
+ parent_uuid TEXT,
48
+ type TEXT NOT NULL,
49
+ role TEXT,
50
+ content TEXT,
51
+ model TEXT,
52
+ input_tokens INTEGER,
53
+ output_tokens INTEGER,
54
+ timestamp TEXT,
55
+ cwd TEXT,
56
+ git_branch TEXT,
57
+ version TEXT
58
+ );
59
+
60
+ CREATE TABLE IF NOT EXISTS tool_uses (
61
+ id INTEGER PRIMARY KEY AUTOINCREMENT,
62
+ message_uuid TEXT,
63
+ session_id TEXT,
64
+ tool_name TEXT,
65
+ tool_input TEXT,
66
+ timestamp TEXT
67
+ );
68
+
69
+ CREATE TABLE IF NOT EXISTS saved_threads (
70
+ name TEXT PRIMARY KEY,
71
+ session_id TEXT NOT NULL,
72
+ summary TEXT,
73
+ project_path TEXT,
74
+ saved_at TEXT DEFAULT CURRENT_TIMESTAMP
75
+ );
76
+
77
+ CREATE TABLE IF NOT EXISTS pending_actions (
78
+ id TEXT PRIMARY KEY,
79
+ status TEXT NOT NULL DEFAULT 'open',
80
+ title TEXT NOT NULL,
81
+ detail TEXT,
82
+ scope TEXT,
83
+ kind TEXT NOT NULL DEFAULT 'manual',
84
+ verify_cmd TEXT,
85
+ created_at TEXT DEFAULT CURRENT_TIMESTAMP,
86
+ next_check TEXT,
87
+ snooze_until TEXT,
88
+ closed_at TEXT,
89
+ verify_passes INTEGER NOT NULL DEFAULT 0
90
+ );
91
+
92
+ CREATE INDEX IF NOT EXISTS idx_messages_session ON messages(session_id);
93
+ CREATE INDEX IF NOT EXISTS idx_messages_type ON messages(type);
94
+ CREATE INDEX IF NOT EXISTS idx_messages_timestamp ON messages(timestamp);
95
+ CREATE INDEX IF NOT EXISTS idx_tool_uses_tool ON tool_uses(tool_name);
96
+ CREATE INDEX IF NOT EXISTS idx_pending_status ON pending_actions(status);
97
+
98
+ CREATE VIRTUAL TABLE IF NOT EXISTS messages_fts USING fts5(
99
+ content,
100
+ content=messages,
101
+ content_rowid=id
102
+ );
103
+
104
+ CREATE TRIGGER IF NOT EXISTS messages_ai AFTER INSERT ON messages BEGIN
105
+ INSERT INTO messages_fts(rowid, content) VALUES (new.id, new.content);
106
+ END;
107
+ CREATE TRIGGER IF NOT EXISTS messages_ad AFTER DELETE ON messages BEGIN
108
+ INSERT INTO messages_fts(messages_fts, rowid, content) VALUES ('delete', old.id, old.content);
109
+ END;
110
+ CREATE TRIGGER IF NOT EXISTS messages_au AFTER UPDATE ON messages BEGIN
111
+ INSERT INTO messages_fts(messages_fts, rowid, content) VALUES ('delete', old.id, old.content);
112
+ INSERT INTO messages_fts(rowid, content) VALUES (new.id, new.content);
113
+ END;
114
+ `;
115
+ function ensureColumn(db, table, column, decl) {
116
+ const cols = db.prepare(`PRAGMA table_info(${table})`).all();
117
+ if (!cols.some((col) => col.name === column)) {
118
+ db.exec(`ALTER TABLE ${table} ADD COLUMN ${column} ${decl}`);
119
+ }
120
+ }
121
+ function migrate(db) {
122
+ db.exec("CREATE TABLE IF NOT EXISTS schema_meta (version INTEGER NOT NULL)");
123
+ db.exec(SCHEMA_SQL);
124
+ // v2: pending_actions.verify_passes (N=2 auto-verify confirmation). Add to
125
+ // tables created before this column existed.
126
+ ensureColumn(db, "pending_actions", "verify_passes", "INTEGER NOT NULL DEFAULT 0");
127
+ const row = db.prepare("SELECT version FROM schema_meta LIMIT 1").get();
128
+ if (!row) {
129
+ db.prepare("INSERT INTO schema_meta(version) VALUES (?)").run(SCHEMA_VERSION);
130
+ }
131
+ else if (row.version < SCHEMA_VERSION) {
132
+ db.prepare("UPDATE schema_meta SET version = ?").run(SCHEMA_VERSION);
133
+ }
134
+ }
135
+ /**
136
+ * Open (creating + migrating if needed) the memory database. Pass ":memory:" for
137
+ * an ephemeral in-process DB (tests). Otherwise defaults to ~/.kit/memory.db.
138
+ */
139
+ export function openMemoryDb(path) {
140
+ const dbPath = path ?? getMemoryDbPath();
141
+ if (dbPath !== ":memory:")
142
+ ensureMemoryDir();
143
+ const db = new DatabaseSync(dbPath);
144
+ db.exec("PRAGMA journal_mode = WAL");
145
+ db.exec("PRAGMA busy_timeout = 5000");
146
+ db.exec("PRAGMA foreign_keys = OFF");
147
+ migrate(db);
148
+ if (dbPath !== ":memory:" && existsSync(dbPath)) {
149
+ try {
150
+ chmodSync(dbPath, 0o600);
151
+ }
152
+ catch {
153
+ // best-effort: non-POSIX filesystems may not support chmod
154
+ }
155
+ }
156
+ return db;
157
+ }
158
+ export function upsertSession(db, s) {
159
+ db.prepare(`INSERT INTO sessions (session_id, harness, project, first_message_at, last_message_at, is_agent_sidechain)
160
+ VALUES (?, ?, ?, ?, ?, ?)
161
+ ON CONFLICT(session_id) DO UPDATE SET
162
+ last_message_at = COALESCE(excluded.last_message_at, sessions.last_message_at),
163
+ first_message_at = COALESCE(sessions.first_message_at, excluded.first_message_at),
164
+ project = COALESCE(excluded.project, sessions.project),
165
+ harness = excluded.harness,
166
+ is_agent_sidechain = MAX(sessions.is_agent_sidechain, excluded.is_agent_sidechain)`).run(s.sessionId, s.harness, s.project ?? null, s.firstMessageAt ?? null, s.lastMessageAt ?? null, s.isAgentSidechain ? 1 : 0);
167
+ }
168
+ /** Insert a message idempotently (by uuid). Returns true if a new row was added. */
169
+ export function insertMessage(db, m) {
170
+ const res = db
171
+ .prepare(`INSERT OR IGNORE INTO messages
172
+ (uuid, session_id, parent_uuid, type, role, content, model, input_tokens, output_tokens, timestamp, cwd, git_branch, version)
173
+ VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)`)
174
+ .run(m.uuid, m.sessionId, m.parentUuid ?? null, m.type, m.role ?? null, m.content ?? null, m.model ?? null, m.inputTokens ?? null, m.outputTokens ?? null, m.timestamp ?? null, m.cwd ?? null, m.gitBranch ?? null, m.version ?? null);
175
+ if (Number(res.changes) > 0) {
176
+ db.prepare("UPDATE sessions SET message_count = message_count + 1 WHERE session_id = ?").run(m.sessionId);
177
+ return true;
178
+ }
179
+ return false;
180
+ }
181
+ export function insertToolUse(db, t) {
182
+ db.prepare(`INSERT INTO tool_uses (message_uuid, session_id, tool_name, tool_input, timestamp)
183
+ VALUES (?, ?, ?, ?, ?)`).run(t.messageUuid ?? null, t.sessionId ?? null, t.toolName, t.toolInput ?? null, t.timestamp ?? null);
184
+ }
185
+ /**
186
+ * Full-text search over raw message content (FTS5 MATCH, ranked by `rank`).
187
+ * Pass opts.projectPath to scope to one repo (relevance + blast-radius); omit it
188
+ * for cross-project ("--global") recall over the personal store.
189
+ */
190
+ export function searchMessages(db, query, opts = {}) {
191
+ const limit = opts.limit ?? 20;
192
+ const params = [query];
193
+ let where = "messages_fts MATCH ?";
194
+ if (opts.projectPath) {
195
+ where += " AND (m.cwd = ? OR m.cwd LIKE ?)";
196
+ params.push(opts.projectPath, `${opts.projectPath}/%`);
197
+ }
198
+ params.push(limit);
199
+ return db
200
+ .prepare(`SELECT m.id AS id, m.uuid AS uuid, m.session_id AS sessionId, m.role AS role,
201
+ m.content AS content, m.timestamp AS timestamp
202
+ FROM messages_fts f
203
+ JOIN messages m ON m.id = f.rowid
204
+ WHERE ${where}
205
+ ORDER BY rank
206
+ LIMIT ?`)
207
+ .all(...params);
208
+ }
209
+ export function getStats(db) {
210
+ const count = (sql) => {
211
+ const r = db.prepare(sql).get();
212
+ return r ? Number(r.n) : 0;
213
+ };
214
+ const dbPath = getMemoryDbPath();
215
+ let sizeBytes = 0;
216
+ if (dbPath !== ":memory:" && existsSync(dbPath)) {
217
+ try {
218
+ sizeBytes = statSync(dbPath).size;
219
+ }
220
+ catch {
221
+ // best-effort: size is informational only
222
+ }
223
+ }
224
+ return {
225
+ sessions: count("SELECT COUNT(*) AS n FROM sessions"),
226
+ messages: count("SELECT COUNT(*) AS n FROM messages"),
227
+ toolUses: count("SELECT COUNT(*) AS n FROM tool_uses"),
228
+ pendingOpen: count("SELECT COUNT(*) AS n FROM pending_actions WHERE status = 'open'"),
229
+ dbPath,
230
+ sizeBytes,
231
+ };
232
+ }
233
+ //# sourceMappingURL=db.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"db.js","sourceRoot":"","sources":["../../src/memory/db.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;GAaG;AACH,OAAO,EAAE,YAAY,EAAE,MAAM,aAAa,CAAC;AAC3C,OAAO,EAAE,OAAO,EAAE,MAAM,SAAS,CAAC;AAClC,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AACjC,OAAO,EAAE,UAAU,EAAE,SAAS,EAAE,SAAS,EAAE,QAAQ,EAAE,MAAM,SAAS,CAAC;AASrE,MAAM,CAAC,MAAM,cAAc,GAAG,CAAC,CAAC;AAEhC,MAAM,UAAU,YAAY;IAC1B,OAAO,OAAO,CAAC,GAAG,CAAC,cAAc,IAAI,IAAI,CAAC,OAAO,EAAE,EAAE,MAAM,CAAC,CAAC;AAC/D,CAAC;AAED,MAAM,UAAU,eAAe;IAC7B,OAAO,OAAO,CAAC,GAAG,CAAC,aAAa,IAAI,IAAI,CAAC,YAAY,EAAE,EAAE,WAAW,CAAC,CAAC;AACxE,CAAC;AAED,SAAS,eAAe;IACtB,MAAM,GAAG,GAAG,YAAY,EAAE,CAAC;IAC3B,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC;QAAE,SAAS,CAAC,GAAG,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,IAAI,EAAE,KAAK,EAAE,CAAC,CAAC;AACzE,CAAC;AAED,MAAM,UAAU,GAAG;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;CAmFlB,CAAC;AAEF,SAAS,YAAY,CACnB,EAAgB,EAChB,KAAa,EACb,MAAc,EACd,IAAY;IAEZ,MAAM,IAAI,GAAG,EAAE,CAAC,OAAO,CAAC,qBAAqB,KAAK,GAAG,CAAC,CAAC,GAAG,EAAwB,CAAC;IACnF,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,GAAG,CAAC,IAAI,KAAK,MAAM,CAAC,EAAE,CAAC;QAC7C,EAAE,CAAC,IAAI,CAAC,eAAe,KAAK,eAAe,MAAM,IAAI,IAAI,EAAE,CAAC,CAAC;IAC/D,CAAC;AACH,CAAC;AAED,SAAS,OAAO,CAAC,EAAgB;IAC/B,EAAE,CAAC,IAAI,CAAC,mEAAmE,CAAC,CAAC;IAC7E,EAAE,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;IACpB,2EAA2E;IAC3E,6CAA6C;IAC7C,YAAY,CAAC,EAAE,EAAE,iBAAiB,EAAE,eAAe,EAAE,4BAA4B,CAAC,CAAC;IACnF,MAAM,GAAG,GAAG,EAAE,CAAC,OAAO,CAAC,yCAAyC,CAAC,CAAC,GAAG,EAExD,CAAC;IACd,IAAI,CAAC,GAAG,EAAE,CAAC;QACT,EAAE,CAAC,OAAO,CAAC,6CAA6C,CAAC,CAAC,GAAG,CAAC,cAAc,CAAC,CAAC;IAChF,CAAC;SAAM,IAAI,GAAG,CAAC,OAAO,GAAG,cAAc,EAAE,CAAC;QACxC,EAAE,CAAC,OAAO,CAAC,oCAAoC,CAAC,CAAC,GAAG,CAAC,cAAc,CAAC,CAAC;IACvE,CAAC;AACH,CAAC;AAED;;;GAGG;AACH,MAAM,UAAU,YAAY,CAAC,IAAa;IACxC,MAAM,MAAM,GAAG,IAAI,IAAI,eAAe,EAAE,CAAC;IACzC,IAAI,MAAM,KAAK,UAAU;QAAE,eAAe,EAAE,CAAC;IAC7C,MAAM,EAAE,GAAG,IAAI,YAAY,CAAC,MAAM,CAAC,CAAC;IACpC,EAAE,CAAC,IAAI,CAAC,2BAA2B,CAAC,CAAC;IACrC,EAAE,CAAC,IAAI,CAAC,4BAA4B,CAAC,CAAC;IACtC,EAAE,CAAC,IAAI,CAAC,2BAA2B,CAAC,CAAC;IACrC,OAAO,CAAC,EAAE,CAAC,CAAC;IACZ,IAAI,MAAM,KAAK,UAAU,IAAI,UAAU,CAAC,MAAM,CAAC,EAAE,CAAC;QAChD,IAAI,CAAC;YACH,SAAS,CAAC,MAAM,EAAE,KAAK,CAAC,CAAC;QAC3B,CAAC;QAAC,MAAM,CAAC;YACP,2DAA2D;QAC7D,CAAC;IACH,CAAC;IACD,OAAO,EAAE,CAAC;AACZ,CAAC;AAED,MAAM,UAAU,aAAa,CAAC,EAAgB,EAAE,CAAe;IAC7D,EAAE,CAAC,OAAO,CACR;;;;;;;0FAOsF,CACvF,CAAC,GAAG,CACH,CAAC,CAAC,SAAS,EACX,CAAC,CAAC,OAAO,EACT,CAAC,CAAC,OAAO,IAAI,IAAI,EACjB,CAAC,CAAC,cAAc,IAAI,IAAI,EACxB,CAAC,CAAC,aAAa,IAAI,IAAI,EACvB,CAAC,CAAC,gBAAgB,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAC3B,CAAC;AACJ,CAAC;AAED,oFAAoF;AACpF,MAAM,UAAU,aAAa,CAAC,EAAgB,EAAE,CAAe;IAC7D,MAAM,GAAG,GAAG,EAAE;SACX,OAAO,CACN;;sDAEgD,CACjD;SACA,GAAG,CACF,CAAC,CAAC,IAAI,EACN,CAAC,CAAC,SAAS,EACX,CAAC,CAAC,UAAU,IAAI,IAAI,EACpB,CAAC,CAAC,IAAI,EACN,CAAC,CAAC,IAAI,IAAI,IAAI,EACd,CAAC,CAAC,OAAO,IAAI,IAAI,EACjB,CAAC,CAAC,KAAK,IAAI,IAAI,EACf,CAAC,CAAC,WAAW,IAAI,IAAI,EACrB,CAAC,CAAC,YAAY,IAAI,IAAI,EACtB,CAAC,CAAC,SAAS,IAAI,IAAI,EACnB,CAAC,CAAC,GAAG,IAAI,IAAI,EACb,CAAC,CAAC,SAAS,IAAI,IAAI,EACnB,CAAC,CAAC,OAAO,IAAI,IAAI,CAClB,CAAC;IACJ,IAAI,MAAM,CAAC,GAAG,CAAC,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC;QAC5B,EAAE,CAAC,OAAO,CACR,4EAA4E,CAC7E,CAAC,GAAG,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC;QACnB,OAAO,IAAI,CAAC;IACd,CAAC;IACD,OAAO,KAAK,CAAC;AACf,CAAC;AAED,MAAM,UAAU,aAAa,CAAC,EAAgB,EAAE,CAAe;IAC7D,EAAE,CAAC,OAAO,CACR;4BACwB,CACzB,CAAC,GAAG,CACH,CAAC,CAAC,WAAW,IAAI,IAAI,EACrB,CAAC,CAAC,SAAS,IAAI,IAAI,EACnB,CAAC,CAAC,QAAQ,EACV,CAAC,CAAC,SAAS,IAAI,IAAI,EACnB,CAAC,CAAC,SAAS,IAAI,IAAI,CACpB,CAAC;AACJ,CAAC;AAQD;;;;GAIG;AACH,MAAM,UAAU,cAAc,CAC5B,EAAgB,EAChB,KAAa,EACb,OAAsB,EAAE;IAExB,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,IAAI,EAAE,CAAC;IAC/B,MAAM,MAAM,GAAwB,CAAC,KAAK,CAAC,CAAC;IAC5C,IAAI,KAAK,GAAG,sBAAsB,CAAC;IACnC,IAAI,IAAI,CAAC,WAAW,EAAE,CAAC;QACrB,KAAK,IAAI,kCAAkC,CAAC;QAC5C,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,WAAW,EAAE,GAAG,IAAI,CAAC,WAAW,IAAI,CAAC,CAAC;IACzD,CAAC;IACD,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IACnB,OAAO,EAAE;SACN,OAAO,CACN;;;;eAIS,KAAK;;eAEL,CACV;SACA,GAAG,CAAC,GAAG,MAAM,CAA2B,CAAC;AAC9C,CAAC;AAED,MAAM,UAAU,QAAQ,CAAC,EAAgB;IACvC,MAAM,KAAK,GAAG,CAAC,GAAW,EAAU,EAAE;QACpC,MAAM,CAAC,GAAG,EAAE,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC,GAAG,EAA+B,CAAC;QAC7D,OAAO,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;IAC7B,CAAC,CAAC;IACF,MAAM,MAAM,GAAG,eAAe,EAAE,CAAC;IACjC,IAAI,SAAS,GAAG,CAAC,CAAC;IAClB,IAAI,MAAM,KAAK,UAAU,IAAI,UAAU,CAAC,MAAM,CAAC,EAAE,CAAC;QAChD,IAAI,CAAC;YACH,SAAS,GAAG,QAAQ,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC;QACpC,CAAC;QAAC,MAAM,CAAC;YACP,0CAA0C;QAC5C,CAAC;IACH,CAAC;IACD,OAAO;QACL,QAAQ,EAAE,KAAK,CAAC,oCAAoC,CAAC;QACrD,QAAQ,EAAE,KAAK,CAAC,oCAAoC,CAAC;QACrD,QAAQ,EAAE,KAAK,CAAC,qCAAqC,CAAC;QACtD,WAAW,EAAE,KAAK,CAChB,iEAAiE,CAClE;QACD,MAAM;QACN,SAAS;KACV,CAAC;AACJ,CAAC"}
@@ -0,0 +1,6 @@
1
+ /** Reminder injected before every prompt. Empty string on any error (fail-open). */
2
+ export declare function userPromptSubmitReminder(): string;
3
+ /** Index the just-ended session. Returns count of newly indexed messages (fail-open). */
4
+ export declare function runSessionEndIndex(): {
5
+ messages: number;
6
+ };
@@ -0,0 +1,51 @@
1
+ /**
2
+ * kit memory — Claude Code hook entry points (the "whole system is two hooks").
3
+ *
4
+ * - UserPromptSubmit → a short reminder that searchable memory exists. The agent
5
+ * pulls on demand (`kit memory search`) instead of pre-loading everything.
6
+ * - SessionEnd → index the just-ended session into the store (incremental sync).
7
+ *
8
+ * Both are FAIL-OPEN: any error yields an empty/no-op result so a hook can never
9
+ * block a prompt or break a session. Deterministic, zero model calls.
10
+ */
11
+ import { basename } from "node:path";
12
+ import { openMemoryDb, getStats } from "./db.js";
13
+ import { indexClaudeTranscripts } from "./parser.js";
14
+ import { palList } from "./pal.js";
15
+ import { getCurrentProjectRoot } from "./project.js";
16
+ /** Reminder injected before every prompt. Empty string on any error (fail-open). */
17
+ export function userPromptSubmitReminder() {
18
+ try {
19
+ const db = openMemoryDb();
20
+ const s = getStats(db);
21
+ // Only surface THIS project's open items (plus globally-scoped) — no cross-project noise.
22
+ const openItems = palList(db, { scope: basename(getCurrentProjectRoot()) });
23
+ db.close();
24
+ let pending = "";
25
+ if (openItems.length > 0) {
26
+ const shown = openItems.slice(0, 3);
27
+ const titles = shown.map((p) => `${p.id} ${p.title}`).join("; ");
28
+ const more = openItems.length > shown.length ? " …" : "";
29
+ pending = ` ${openItems.length} open action item(s) blocked on you: ${titles}${more}.`;
30
+ }
31
+ return (`You have local conversation memory: ${s.messages} messages indexed. ` +
32
+ "Before answering anything project-specific, run `kit memory search <terms>` " +
33
+ `to retrieve what was actually said instead of reconstructing it.${pending}`);
34
+ }
35
+ catch {
36
+ return ""; // fail-open: never block a prompt
37
+ }
38
+ }
39
+ /** Index the just-ended session. Returns count of newly indexed messages (fail-open). */
40
+ export function runSessionEndIndex() {
41
+ try {
42
+ const db = openMemoryDb();
43
+ const res = indexClaudeTranscripts(db);
44
+ db.close();
45
+ return { messages: res.messages };
46
+ }
47
+ catch {
48
+ return { messages: 0 }; // fail-open
49
+ }
50
+ }
51
+ //# sourceMappingURL=hook%202.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"hook 2.js","sourceRoot":"","sources":["../../src/memory/hook 2.ts"],"names":[],"mappings":"AAAA;;;;;;;;;GASG;AACH,OAAO,EAAE,QAAQ,EAAE,MAAM,WAAW,CAAC;AACrC,OAAO,EAAE,YAAY,EAAE,QAAQ,EAAE,MAAM,SAAS,CAAC;AACjD,OAAO,EAAE,sBAAsB,EAAE,MAAM,aAAa,CAAC;AACrD,OAAO,EAAE,OAAO,EAAE,MAAM,UAAU,CAAC;AACnC,OAAO,EAAE,qBAAqB,EAAE,MAAM,cAAc,CAAC;AAErD,oFAAoF;AACpF,MAAM,UAAU,wBAAwB;IACtC,IAAI,CAAC;QACH,MAAM,EAAE,GAAG,YAAY,EAAE,CAAC;QAC1B,MAAM,CAAC,GAAG,QAAQ,CAAC,EAAE,CAAC,CAAC;QACvB,0FAA0F;QAC1F,MAAM,SAAS,GAAG,OAAO,CAAC,EAAE,EAAE,EAAE,KAAK,EAAE,QAAQ,CAAC,qBAAqB,EAAE,CAAC,EAAE,CAAC,CAAC;QAC5E,EAAE,CAAC,KAAK,EAAE,CAAC;QACX,IAAI,OAAO,GAAG,EAAE,CAAC;QACjB,IAAI,SAAS,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACzB,MAAM,KAAK,GAAG,SAAS,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;YACpC,MAAM,MAAM,GAAG,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,GAAG,CAAC,CAAC,EAAE,IAAI,CAAC,CAAC,KAAK,EAAE,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YACjE,MAAM,IAAI,GAAG,SAAS,CAAC,MAAM,GAAG,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC;YACzD,OAAO,GAAG,IAAI,SAAS,CAAC,MAAM,wCAAwC,MAAM,GAAG,IAAI,GAAG,CAAC;QACzF,CAAC;QACD,OAAO,CACL,uCAAuC,CAAC,CAAC,QAAQ,qBAAqB;YACtE,8EAA8E;YAC9E,mEAAmE,OAAO,EAAE,CAC7E,CAAC;IACJ,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,EAAE,CAAC,CAAC,kCAAkC;IAC/C,CAAC;AACH,CAAC;AAED,yFAAyF;AACzF,MAAM,UAAU,kBAAkB;IAChC,IAAI,CAAC;QACH,MAAM,EAAE,GAAG,YAAY,EAAE,CAAC;QAC1B,MAAM,GAAG,GAAG,sBAAsB,CAAC,EAAE,CAAC,CAAC;QACvC,EAAE,CAAC,KAAK,EAAE,CAAC;QACX,OAAO,EAAE,QAAQ,EAAE,GAAG,CAAC,QAAQ,EAAE,CAAC;IACpC,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,EAAE,QAAQ,EAAE,CAAC,EAAE,CAAC,CAAC,YAAY;IACtC,CAAC;AACH,CAAC"}
@@ -0,0 +1,6 @@
1
+ /** Reminder injected before every prompt. Empty string on any error (fail-open). */
2
+ export declare function userPromptSubmitReminder(): string;
3
+ /** Index the just-ended session. Returns count of newly indexed messages (fail-open). */
4
+ export declare function runSessionEndIndex(): {
5
+ messages: number;
6
+ };
@@ -0,0 +1,51 @@
1
+ /**
2
+ * kit memory — Claude Code hook entry points (the "whole system is two hooks").
3
+ *
4
+ * - UserPromptSubmit → a short reminder that searchable memory exists. The agent
5
+ * pulls on demand (`kit memory search`) instead of pre-loading everything.
6
+ * - SessionEnd → index the just-ended session into the store (incremental sync).
7
+ *
8
+ * Both are FAIL-OPEN: any error yields an empty/no-op result so a hook can never
9
+ * block a prompt or break a session. Deterministic, zero model calls.
10
+ */
11
+ import { basename } from "node:path";
12
+ import { openMemoryDb, getStats } from "./db.js";
13
+ import { indexClaudeTranscripts } from "./parser.js";
14
+ import { palList } from "./pal.js";
15
+ import { getCurrentProjectRoot } from "./project.js";
16
+ /** Reminder injected before every prompt. Empty string on any error (fail-open). */
17
+ export function userPromptSubmitReminder() {
18
+ try {
19
+ const db = openMemoryDb();
20
+ const s = getStats(db);
21
+ // Only surface THIS project's open items (plus globally-scoped) — no cross-project noise.
22
+ const openItems = palList(db, { scope: basename(getCurrentProjectRoot()) });
23
+ db.close();
24
+ let pending = "";
25
+ if (openItems.length > 0) {
26
+ const shown = openItems.slice(0, 3);
27
+ const titles = shown.map((p) => `${p.id} ${p.title}`).join("; ");
28
+ const more = openItems.length > shown.length ? " …" : "";
29
+ pending = ` ${openItems.length} open action item(s) blocked on you: ${titles}${more}.`;
30
+ }
31
+ return (`You have local conversation memory: ${s.messages} messages indexed. ` +
32
+ "Before answering anything project-specific, run `kit memory search <terms>` " +
33
+ `to retrieve what was actually said instead of reconstructing it.${pending}`);
34
+ }
35
+ catch {
36
+ return ""; // fail-open: never block a prompt
37
+ }
38
+ }
39
+ /** Index the just-ended session. Returns count of newly indexed messages (fail-open). */
40
+ export function runSessionEndIndex() {
41
+ try {
42
+ const db = openMemoryDb();
43
+ const res = indexClaudeTranscripts(db);
44
+ db.close();
45
+ return { messages: res.messages };
46
+ }
47
+ catch {
48
+ return { messages: 0 }; // fail-open
49
+ }
50
+ }
51
+ //# sourceMappingURL=hook.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"hook.js","sourceRoot":"","sources":["../../src/memory/hook.ts"],"names":[],"mappings":"AAAA;;;;;;;;;GASG;AACH,OAAO,EAAE,QAAQ,EAAE,MAAM,WAAW,CAAC;AACrC,OAAO,EAAE,YAAY,EAAE,QAAQ,EAAE,MAAM,SAAS,CAAC;AACjD,OAAO,EAAE,sBAAsB,EAAE,MAAM,aAAa,CAAC;AACrD,OAAO,EAAE,OAAO,EAAE,MAAM,UAAU,CAAC;AACnC,OAAO,EAAE,qBAAqB,EAAE,MAAM,cAAc,CAAC;AAErD,oFAAoF;AACpF,MAAM,UAAU,wBAAwB;IACtC,IAAI,CAAC;QACH,MAAM,EAAE,GAAG,YAAY,EAAE,CAAC;QAC1B,MAAM,CAAC,GAAG,QAAQ,CAAC,EAAE,CAAC,CAAC;QACvB,0FAA0F;QAC1F,MAAM,SAAS,GAAG,OAAO,CAAC,EAAE,EAAE,EAAE,KAAK,EAAE,QAAQ,CAAC,qBAAqB,EAAE,CAAC,EAAE,CAAC,CAAC;QAC5E,EAAE,CAAC,KAAK,EAAE,CAAC;QACX,IAAI,OAAO,GAAG,EAAE,CAAC;QACjB,IAAI,SAAS,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACzB,MAAM,KAAK,GAAG,SAAS,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;YACpC,MAAM,MAAM,GAAG,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,GAAG,CAAC,CAAC,EAAE,IAAI,CAAC,CAAC,KAAK,EAAE,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YACjE,MAAM,IAAI,GAAG,SAAS,CAAC,MAAM,GAAG,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC;YACzD,OAAO,GAAG,IAAI,SAAS,CAAC,MAAM,wCAAwC,MAAM,GAAG,IAAI,GAAG,CAAC;QACzF,CAAC;QACD,OAAO,CACL,uCAAuC,CAAC,CAAC,QAAQ,qBAAqB;YACtE,8EAA8E;YAC9E,mEAAmE,OAAO,EAAE,CAC7E,CAAC;IACJ,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,EAAE,CAAC,CAAC,kCAAkC;IAC/C,CAAC;AACH,CAAC;AAED,yFAAyF;AACzF,MAAM,UAAU,kBAAkB;IAChC,IAAI,CAAC;QACH,MAAM,EAAE,GAAG,YAAY,EAAE,CAAC;QAC1B,MAAM,GAAG,GAAG,sBAAsB,CAAC,EAAE,CAAC,CAAC;QACvC,EAAE,CAAC,KAAK,EAAE,CAAC;QACX,OAAO,EAAE,QAAQ,EAAE,GAAG,CAAC,QAAQ,EAAE,CAAC;IACpC,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,EAAE,QAAQ,EAAE,CAAC,EAAE,CAAC,CAAC,YAAY;IACtC,CAAC;AACH,CAAC"}
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,35 @@
1
+ import { describe, it, before, after } from "node:test";
2
+ import assert from "node:assert/strict";
3
+ import { mkdtempSync, rmSync } from "node:fs";
4
+ import { tmpdir } from "node:os";
5
+ import { join } from "node:path";
6
+ import { openMemoryDb, upsertSession, insertMessage } from "./db.js";
7
+ import { userPromptSubmitReminder } from "./hook.js";
8
+ describe("memory hook — UserPromptSubmit reminder", () => {
9
+ let tmp;
10
+ const prev = process.env.KIT_MEMORY_DB;
11
+ before(() => {
12
+ tmp = mkdtempSync(join(tmpdir(), "kit-hook-"));
13
+ process.env.KIT_MEMORY_DB = join(tmp, "memory.db");
14
+ const db = openMemoryDb();
15
+ upsertSession(db, { sessionId: "s1", harness: "claude-code" });
16
+ insertMessage(db, { uuid: "u1", sessionId: "s1", type: "user", content: "hi" });
17
+ db.close();
18
+ });
19
+ after(() => {
20
+ if (prev === undefined)
21
+ delete process.env.KIT_MEMORY_DB;
22
+ else
23
+ process.env.KIT_MEMORY_DB = prev;
24
+ rmSync(tmp, { recursive: true, force: true });
25
+ });
26
+ it("nudges the agent to search and reports the message count", () => {
27
+ const text = userPromptSubmitReminder();
28
+ assert.match(text, /kit memory search/);
29
+ assert.match(text, /1 messages/);
30
+ });
31
+ it("never throws (fail-open)", () => {
32
+ assert.doesNotThrow(() => userPromptSubmitReminder());
33
+ });
34
+ });
35
+ //# sourceMappingURL=hook.test%202.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"hook.test 2.js","sourceRoot":"","sources":["../../src/memory/hook.test 2.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,EAAE,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,WAAW,CAAC;AACxD,OAAO,MAAM,MAAM,oBAAoB,CAAC;AACxC,OAAO,EAAE,WAAW,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AAC9C,OAAO,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AACjC,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AACjC,OAAO,EAAE,YAAY,EAAE,aAAa,EAAE,aAAa,EAAE,MAAM,SAAS,CAAC;AACrE,OAAO,EAAE,wBAAwB,EAAE,MAAM,WAAW,CAAC;AAErD,QAAQ,CAAC,yCAAyC,EAAE,GAAG,EAAE;IACvD,IAAI,GAAW,CAAC;IAChB,MAAM,IAAI,GAAG,OAAO,CAAC,GAAG,CAAC,aAAa,CAAC;IAEvC,MAAM,CAAC,GAAG,EAAE;QACV,GAAG,GAAG,WAAW,CAAC,IAAI,CAAC,MAAM,EAAE,EAAE,WAAW,CAAC,CAAC,CAAC;QAC/C,OAAO,CAAC,GAAG,CAAC,aAAa,GAAG,IAAI,CAAC,GAAG,EAAE,WAAW,CAAC,CAAC;QACnD,MAAM,EAAE,GAAG,YAAY,EAAE,CAAC;QAC1B,aAAa,CAAC,EAAE,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,OAAO,EAAE,aAAa,EAAE,CAAC,CAAC;QAC/D,aAAa,CAAC,EAAE,EAAE,EAAE,IAAI,EAAE,IAAI,EAAE,SAAS,EAAE,IAAI,EAAE,IAAI,EAAE,MAAM,EAAE,OAAO,EAAE,IAAI,EAAE,CAAC,CAAC;QAChF,EAAE,CAAC,KAAK,EAAE,CAAC;IACb,CAAC,CAAC,CAAC;IAEH,KAAK,CAAC,GAAG,EAAE;QACT,IAAI,IAAI,KAAK,SAAS;YAAE,OAAO,OAAO,CAAC,GAAG,CAAC,aAAa,CAAC;;YACpD,OAAO,CAAC,GAAG,CAAC,aAAa,GAAG,IAAI,CAAC;QACtC,MAAM,CAAC,GAAG,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC,CAAC;IAChD,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,0DAA0D,EAAE,GAAG,EAAE;QAClE,MAAM,IAAI,GAAG,wBAAwB,EAAE,CAAC;QACxC,MAAM,CAAC,KAAK,CAAC,IAAI,EAAE,mBAAmB,CAAC,CAAC;QACxC,MAAM,CAAC,KAAK,CAAC,IAAI,EAAE,YAAY,CAAC,CAAC;IACnC,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,0BAA0B,EAAE,GAAG,EAAE;QAClC,MAAM,CAAC,YAAY,CAAC,GAAG,EAAE,CAAC,wBAAwB,EAAE,CAAC,CAAC;IACxD,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}
@@ -0,0 +1,8 @@
1
+ export declare function getClaudeSettingsPath(): string;
2
+ export declare function installMemoryHooks(path?: string): {
3
+ added: string[];
4
+ alreadyPresent: string[];
5
+ };
6
+ export declare function uninstallMemoryHooks(path?: string): {
7
+ removed: string[];
8
+ };
@@ -0,0 +1,72 @@
1
+ /**
2
+ * kit memory — install/remove the two Claude Code hooks in ~/.claude/settings.json.
3
+ *
4
+ * Idempotent and non-destructive: merges our hook entries into the existing
5
+ * settings without touching the user's other hooks. Re-running adds nothing.
6
+ * Honors KIT_CLAUDE_SETTINGS for tests.
7
+ */
8
+ import { existsSync, readFileSync, writeFileSync, mkdirSync } from "node:fs";
9
+ import { homedir } from "node:os";
10
+ import { join, dirname } from "node:path";
11
+ const MEMORY_HOOKS = [
12
+ { event: "UserPromptSubmit", command: "kit memory hook user-prompt-submit" },
13
+ { event: "SessionEnd", command: "kit memory hook session-end" },
14
+ ];
15
+ export function getClaudeSettingsPath() {
16
+ return process.env.KIT_CLAUDE_SETTINGS ?? join(homedir(), ".claude", "settings.json");
17
+ }
18
+ function readSettings(path) {
19
+ if (!existsSync(path))
20
+ return {};
21
+ try {
22
+ return JSON.parse(readFileSync(path, "utf8"));
23
+ }
24
+ catch {
25
+ return {}; // corrupt/unreadable → start fresh rather than crash
26
+ }
27
+ }
28
+ function writeSettings(path, s) {
29
+ mkdirSync(dirname(path), { recursive: true });
30
+ writeFileSync(path, JSON.stringify(s, null, 2) + "\n");
31
+ }
32
+ function groupsHaveCommand(groups, command) {
33
+ return groups.some((g) => g.hooks?.some((h) => h.command === command));
34
+ }
35
+ export function installMemoryHooks(path = getClaudeSettingsPath()) {
36
+ const s = readSettings(path);
37
+ const hooks = (s.hooks ??= {});
38
+ const added = [];
39
+ const alreadyPresent = [];
40
+ for (const { event, command } of MEMORY_HOOKS) {
41
+ const groups = (hooks[event] ??= []);
42
+ if (groupsHaveCommand(groups, command)) {
43
+ alreadyPresent.push(event);
44
+ continue;
45
+ }
46
+ groups.push({ hooks: [{ type: "command", command }] });
47
+ added.push(event);
48
+ }
49
+ if (added.length)
50
+ writeSettings(path, s);
51
+ return { added, alreadyPresent };
52
+ }
53
+ export function uninstallMemoryHooks(path = getClaudeSettingsPath()) {
54
+ const s = readSettings(path);
55
+ if (!s.hooks)
56
+ return { removed: [] };
57
+ const removed = [];
58
+ for (const { event, command } of MEMORY_HOOKS) {
59
+ const groups = s.hooks[event];
60
+ if (!Array.isArray(groups))
61
+ continue;
62
+ const filtered = groups.filter((g) => !g.hooks?.some((h) => h.command === command));
63
+ if (filtered.length !== groups.length) {
64
+ s.hooks[event] = filtered;
65
+ removed.push(event);
66
+ }
67
+ }
68
+ if (removed.length)
69
+ writeSettings(path, s);
70
+ return { removed };
71
+ }
72
+ //# sourceMappingURL=install%202.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"install 2.js","sourceRoot":"","sources":["../../src/memory/install 2.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,8 @@
1
+ export declare function getClaudeSettingsPath(): string;
2
+ export declare function installMemoryHooks(path?: string): {
3
+ added: string[];
4
+ alreadyPresent: string[];
5
+ };
6
+ export declare function uninstallMemoryHooks(path?: string): {
7
+ removed: string[];
8
+ };
@@ -0,0 +1,72 @@
1
+ /**
2
+ * kit memory — install/remove the two Claude Code hooks in ~/.claude/settings.json.
3
+ *
4
+ * Idempotent and non-destructive: merges our hook entries into the existing
5
+ * settings without touching the user's other hooks. Re-running adds nothing.
6
+ * Honors KIT_CLAUDE_SETTINGS for tests.
7
+ */
8
+ import { existsSync, readFileSync, writeFileSync, mkdirSync } from "node:fs";
9
+ import { homedir } from "node:os";
10
+ import { join, dirname } from "node:path";
11
+ const MEMORY_HOOKS = [
12
+ { event: "UserPromptSubmit", command: "kit memory hook user-prompt-submit" },
13
+ { event: "SessionEnd", command: "kit memory hook session-end" },
14
+ ];
15
+ export function getClaudeSettingsPath() {
16
+ return process.env.KIT_CLAUDE_SETTINGS ?? join(homedir(), ".claude", "settings.json");
17
+ }
18
+ function readSettings(path) {
19
+ if (!existsSync(path))
20
+ return {};
21
+ try {
22
+ return JSON.parse(readFileSync(path, "utf8"));
23
+ }
24
+ catch {
25
+ return {}; // corrupt/unreadable → start fresh rather than crash
26
+ }
27
+ }
28
+ function writeSettings(path, s) {
29
+ mkdirSync(dirname(path), { recursive: true });
30
+ writeFileSync(path, JSON.stringify(s, null, 2) + "\n");
31
+ }
32
+ function groupsHaveCommand(groups, command) {
33
+ return groups.some((g) => g.hooks?.some((h) => h.command === command));
34
+ }
35
+ export function installMemoryHooks(path = getClaudeSettingsPath()) {
36
+ const s = readSettings(path);
37
+ const hooks = (s.hooks ??= {});
38
+ const added = [];
39
+ const alreadyPresent = [];
40
+ for (const { event, command } of MEMORY_HOOKS) {
41
+ const groups = (hooks[event] ??= []);
42
+ if (groupsHaveCommand(groups, command)) {
43
+ alreadyPresent.push(event);
44
+ continue;
45
+ }
46
+ groups.push({ hooks: [{ type: "command", command }] });
47
+ added.push(event);
48
+ }
49
+ if (added.length)
50
+ writeSettings(path, s);
51
+ return { added, alreadyPresent };
52
+ }
53
+ export function uninstallMemoryHooks(path = getClaudeSettingsPath()) {
54
+ const s = readSettings(path);
55
+ if (!s.hooks)
56
+ return { removed: [] };
57
+ const removed = [];
58
+ for (const { event, command } of MEMORY_HOOKS) {
59
+ const groups = s.hooks[event];
60
+ if (!Array.isArray(groups))
61
+ continue;
62
+ const filtered = groups.filter((g) => !g.hooks?.some((h) => h.command === command));
63
+ if (filtered.length !== groups.length) {
64
+ s.hooks[event] = filtered;
65
+ removed.push(event);
66
+ }
67
+ }
68
+ if (removed.length)
69
+ writeSettings(path, s);
70
+ return { removed };
71
+ }
72
+ //# sourceMappingURL=install.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"install.js","sourceRoot":"","sources":["../../src/memory/install.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1 @@
1
+ export {};