sandstream-kit 1.0.0 → 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.
- package/README.md +18 -13
- package/dist/cli.js +458 -2
- package/dist/cli.js.map +1 -1
- package/dist/database.d.ts +2 -2
- package/dist/database.js +9 -14
- package/dist/database.js.map +1 -1
- package/dist/lock.js +5 -3
- package/dist/lock.js.map +1 -1
- package/dist/memory/backup 2.d.ts +6 -0
- package/dist/memory/backup 2.js +80 -0
- package/dist/memory/backup 2.js.map +1 -0
- package/dist/memory/backup.d.ts +6 -0
- package/dist/memory/backup.js +80 -0
- package/dist/memory/backup.js.map +1 -0
- package/dist/memory/db 2.d.ts +40 -0
- package/dist/memory/db 2.js +233 -0
- package/dist/memory/db 2.js.map +1 -0
- package/dist/memory/db.d.ts +40 -0
- package/dist/memory/db.js +233 -0
- package/dist/memory/db.js.map +1 -0
- package/dist/memory/hook 2.d.ts +6 -0
- package/dist/memory/hook 2.js +51 -0
- package/dist/memory/hook 2.js.map +1 -0
- package/dist/memory/hook.d.ts +6 -0
- package/dist/memory/hook.js +51 -0
- package/dist/memory/hook.js.map +1 -0
- package/dist/memory/hook.test 2.d.ts +1 -0
- package/dist/memory/hook.test 2.js +35 -0
- package/dist/memory/hook.test 2.js.map +1 -0
- package/dist/memory/install 2.d.ts +8 -0
- package/dist/memory/install 2.js +72 -0
- package/dist/memory/install 2.js.map +1 -0
- package/dist/memory/install.d.ts +8 -0
- package/dist/memory/install.js +72 -0
- package/dist/memory/install.js.map +1 -0
- package/dist/memory/install.test 2.d.ts +1 -0
- package/dist/memory/install.test 2.js +59 -0
- package/dist/memory/install.test 2.js.map +1 -0
- package/dist/memory/pal 2.d.ts +47 -0
- package/dist/memory/pal 2.js +154 -0
- package/dist/memory/pal 2.js.map +1 -0
- package/dist/memory/pal.d.ts +47 -0
- package/dist/memory/pal.js +154 -0
- package/dist/memory/pal.js.map +1 -0
- package/dist/memory/parser.d.ts +25 -0
- package/dist/memory/parser.js +164 -0
- package/dist/memory/parser.js.map +1 -0
- package/dist/memory/project 2.d.ts +1 -0
- package/dist/memory/project 2.js +24 -0
- package/dist/memory/project 2.js.map +1 -0
- package/dist/memory/project.d.ts +1 -0
- package/dist/memory/project.js +24 -0
- package/dist/memory/project.js.map +1 -0
- package/dist/memory/scan 2.d.ts +15 -0
- package/dist/memory/scan 2.js +94 -0
- package/dist/memory/scan 2.js.map +1 -0
- package/dist/memory/scan.d.ts +15 -0
- package/dist/memory/scan.js +94 -0
- package/dist/memory/scan.js.map +1 -0
- package/dist/memory/scan.test 2.d.ts +1 -0
- package/dist/memory/scan.test 2.js +55 -0
- package/dist/memory/scan.test 2.js.map +1 -0
- package/dist/memory/shared 2.d.ts +33 -0
- package/dist/memory/shared 2.js +120 -0
- package/dist/memory/shared 2.js.map +1 -0
- package/dist/memory/shared.d.ts +33 -0
- package/dist/memory/shared.js +120 -0
- package/dist/memory/shared.js.map +1 -0
- package/dist/memory/threads 2.d.ts +37 -0
- package/dist/memory/threads 2.js +50 -0
- package/dist/memory/threads 2.js.map +1 -0
- package/dist/memory/threads.d.ts +37 -0
- package/dist/memory/threads.js +50 -0
- package/dist/memory/threads.js.map +1 -0
- package/dist/memory/threads.test 2.d.ts +1 -0
- package/dist/memory/threads.test 2.js +66 -0
- package/dist/memory/threads.test 2.js.map +1 -0
- package/dist/memory/types 2.d.ts +52 -0
- package/dist/memory/types 2.js +5 -0
- package/dist/memory/types 2.js.map +1 -0
- package/dist/memory/types.d.ts +52 -0
- package/dist/memory/types.js +5 -0
- package/dist/memory/types.js.map +1 -0
- package/dist/secrets-pull.d.ts +1 -1
- package/dist/secrets-pull.js +1 -1
- 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":"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"}
|
|
@@ -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,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,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 {};
|