universal-agent-memory 2.9.0 → 3.0.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/bin/cli.js +27 -2
- package/dist/bin/cli.js.map +1 -1
- package/dist/cli/hooks.d.ts +7 -0
- package/dist/cli/hooks.d.ts.map +1 -0
- package/dist/cli/hooks.js +127 -0
- package/dist/cli/hooks.js.map +1 -0
- package/dist/cli/memory.d.ts +4 -1
- package/dist/cli/memory.d.ts.map +1 -1
- package/dist/cli/memory.js +248 -5
- package/dist/cli/memory.js.map +1 -1
- package/dist/coordination/service.d.ts +2 -0
- package/dist/coordination/service.d.ts.map +1 -1
- package/dist/coordination/service.js +13 -1
- package/dist/coordination/service.js.map +1 -1
- package/dist/index.d.ts +10 -0
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +10 -0
- package/dist/index.js.map +1 -1
- package/dist/memory/agent-scoped-memory.d.ts +67 -0
- package/dist/memory/agent-scoped-memory.d.ts.map +1 -0
- package/dist/memory/agent-scoped-memory.js +126 -0
- package/dist/memory/agent-scoped-memory.js.map +1 -0
- package/dist/memory/correction-propagator.d.ts +44 -0
- package/dist/memory/correction-propagator.d.ts.map +1 -0
- package/dist/memory/correction-propagator.js +156 -0
- package/dist/memory/correction-propagator.js.map +1 -0
- package/dist/memory/daily-log.d.ts +67 -0
- package/dist/memory/daily-log.d.ts.map +1 -0
- package/dist/memory/daily-log.js +143 -0
- package/dist/memory/daily-log.js.map +1 -0
- package/dist/memory/hierarchical-memory.d.ts +12 -0
- package/dist/memory/hierarchical-memory.d.ts.map +1 -1
- package/dist/memory/hierarchical-memory.js +68 -0
- package/dist/memory/hierarchical-memory.js.map +1 -1
- package/dist/memory/memory-maintenance.d.ts +39 -0
- package/dist/memory/memory-maintenance.d.ts.map +1 -0
- package/dist/memory/memory-maintenance.js +305 -0
- package/dist/memory/memory-maintenance.js.map +1 -0
- package/dist/memory/write-gate.d.ts +39 -0
- package/dist/memory/write-gate.d.ts.map +1 -0
- package/dist/memory/write-gate.js +190 -0
- package/dist/memory/write-gate.js.map +1 -0
- package/package.json +1 -1
- package/templates/CLAUDE.template.md +29 -29
- package/templates/SCHEMA.md +57 -0
- package/templates/hooks/pre-compact.sh +40 -0
- package/templates/hooks/session-start.sh +66 -0
|
@@ -0,0 +1,305 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Automated Memory Maintenance
|
|
3
|
+
*
|
|
4
|
+
* Periodic maintenance tasks: verify stale claims, prune decayed entries,
|
|
5
|
+
* consolidate daily logs, and surface maintenance recommendations.
|
|
6
|
+
*
|
|
7
|
+
* Inspired by Total Recall's maintenance cadences and Clawe's heartbeat system.
|
|
8
|
+
*/
|
|
9
|
+
import Database from 'better-sqlite3';
|
|
10
|
+
import { existsSync, readdirSync, statSync } from 'fs';
|
|
11
|
+
import { join } from 'path';
|
|
12
|
+
import { ensureDailyLogSchema } from './daily-log.js';
|
|
13
|
+
const DEFAULT_CONFIG = {
|
|
14
|
+
staleDaysThreshold: 14,
|
|
15
|
+
decayRate: 0.95,
|
|
16
|
+
minImportanceAfterDecay: 1,
|
|
17
|
+
archiveDaysOld: 30,
|
|
18
|
+
duplicateSimilarityThreshold: 0.92,
|
|
19
|
+
};
|
|
20
|
+
/**
|
|
21
|
+
* Run full maintenance cycle on the memory database.
|
|
22
|
+
*/
|
|
23
|
+
export function runMaintenance(dbPath, config = {}) {
|
|
24
|
+
const cfg = { ...DEFAULT_CONFIG, ...config };
|
|
25
|
+
const result = {
|
|
26
|
+
staleEntriesPruned: 0,
|
|
27
|
+
decayedEntriesUpdated: 0,
|
|
28
|
+
dailyLogsArchived: 0,
|
|
29
|
+
duplicatesRemoved: 0,
|
|
30
|
+
staleWorktrees: [],
|
|
31
|
+
recommendations: [],
|
|
32
|
+
};
|
|
33
|
+
if (!existsSync(dbPath)) {
|
|
34
|
+
result.recommendations.push('Database not found. Run `uam init` first.');
|
|
35
|
+
return result;
|
|
36
|
+
}
|
|
37
|
+
const db = new Database(dbPath);
|
|
38
|
+
try {
|
|
39
|
+
// 1. Apply importance decay to old memories
|
|
40
|
+
result.decayedEntriesUpdated = applyDecay(db, cfg);
|
|
41
|
+
// 2. Prune entries that decayed below minimum importance
|
|
42
|
+
result.staleEntriesPruned = pruneStale(db, cfg);
|
|
43
|
+
// 3. Archive old daily log entries
|
|
44
|
+
result.dailyLogsArchived = archiveDailyLogs(db, cfg);
|
|
45
|
+
// 4. Remove duplicates
|
|
46
|
+
result.duplicatesRemoved = removeDuplicates(db);
|
|
47
|
+
// 5. Detect stale worktrees
|
|
48
|
+
result.staleWorktrees = detectStaleWorktrees(dbPath, cfg);
|
|
49
|
+
// 6. Generate recommendations
|
|
50
|
+
result.recommendations = generateRecommendations(db, cfg, result.staleWorktrees);
|
|
51
|
+
}
|
|
52
|
+
finally {
|
|
53
|
+
db.close();
|
|
54
|
+
}
|
|
55
|
+
return result;
|
|
56
|
+
}
|
|
57
|
+
function applyDecay(db, cfg) {
|
|
58
|
+
const now = Date.now();
|
|
59
|
+
let updated = 0;
|
|
60
|
+
try {
|
|
61
|
+
const rows = db.prepare(`
|
|
62
|
+
SELECT id, importance, timestamp
|
|
63
|
+
FROM memories
|
|
64
|
+
WHERE importance > ?
|
|
65
|
+
`).all(cfg.minImportanceAfterDecay);
|
|
66
|
+
const updateStmt = db.prepare('UPDATE memories SET importance = ? WHERE id = ?');
|
|
67
|
+
for (const row of rows) {
|
|
68
|
+
const daysSince = (now - new Date(row.timestamp).getTime()) / (1000 * 60 * 60 * 24);
|
|
69
|
+
if (daysSince < 1)
|
|
70
|
+
continue; // Skip recent entries
|
|
71
|
+
const decayed = Math.round(row.importance * Math.pow(cfg.decayRate, daysSince));
|
|
72
|
+
if (decayed !== row.importance && decayed >= cfg.minImportanceAfterDecay) {
|
|
73
|
+
updateStmt.run(decayed, row.id);
|
|
74
|
+
updated++;
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
}
|
|
78
|
+
catch {
|
|
79
|
+
// Table might not exist
|
|
80
|
+
}
|
|
81
|
+
try {
|
|
82
|
+
const rows = db.prepare(`
|
|
83
|
+
SELECT id, importance, timestamp
|
|
84
|
+
FROM session_memories
|
|
85
|
+
WHERE importance > ?
|
|
86
|
+
`).all(cfg.minImportanceAfterDecay);
|
|
87
|
+
const updateStmt = db.prepare('UPDATE session_memories SET importance = ? WHERE id = ?');
|
|
88
|
+
for (const row of rows) {
|
|
89
|
+
const daysSince = (now - new Date(row.timestamp).getTime()) / (1000 * 60 * 60 * 24);
|
|
90
|
+
if (daysSince < 1)
|
|
91
|
+
continue;
|
|
92
|
+
const decayed = Math.round(row.importance * Math.pow(cfg.decayRate, daysSince));
|
|
93
|
+
if (decayed !== row.importance && decayed >= cfg.minImportanceAfterDecay) {
|
|
94
|
+
updateStmt.run(decayed, row.id);
|
|
95
|
+
updated++;
|
|
96
|
+
}
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
catch {
|
|
100
|
+
// Table might not exist
|
|
101
|
+
}
|
|
102
|
+
return updated;
|
|
103
|
+
}
|
|
104
|
+
function pruneStale(db, cfg) {
|
|
105
|
+
let pruned = 0;
|
|
106
|
+
const cutoffDate = new Date();
|
|
107
|
+
cutoffDate.setDate(cutoffDate.getDate() - cfg.staleDaysThreshold);
|
|
108
|
+
const cutoff = cutoffDate.toISOString();
|
|
109
|
+
try {
|
|
110
|
+
const result = db.prepare(`
|
|
111
|
+
DELETE FROM memories
|
|
112
|
+
WHERE importance <= ? AND timestamp < ?
|
|
113
|
+
`).run(cfg.minImportanceAfterDecay, cutoff);
|
|
114
|
+
pruned += result.changes;
|
|
115
|
+
}
|
|
116
|
+
catch {
|
|
117
|
+
// Table might not exist
|
|
118
|
+
}
|
|
119
|
+
try {
|
|
120
|
+
const result = db.prepare(`
|
|
121
|
+
DELETE FROM session_memories
|
|
122
|
+
WHERE importance <= ? AND timestamp < ?
|
|
123
|
+
`).run(cfg.minImportanceAfterDecay, cutoff);
|
|
124
|
+
pruned += result.changes;
|
|
125
|
+
}
|
|
126
|
+
catch {
|
|
127
|
+
// Table might not exist
|
|
128
|
+
}
|
|
129
|
+
return pruned;
|
|
130
|
+
}
|
|
131
|
+
function archiveDailyLogs(db, cfg) {
|
|
132
|
+
try {
|
|
133
|
+
ensureDailyLogSchema(db);
|
|
134
|
+
const cutoff = new Date();
|
|
135
|
+
cutoff.setDate(cutoff.getDate() - cfg.archiveDaysOld);
|
|
136
|
+
const cutoffDate = cutoff.toISOString().split('T')[0];
|
|
137
|
+
// Delete promoted entries older than threshold
|
|
138
|
+
const result = db.prepare(`
|
|
139
|
+
DELETE FROM daily_log WHERE date < ? AND promoted = 1
|
|
140
|
+
`).run(cutoffDate);
|
|
141
|
+
return result.changes;
|
|
142
|
+
}
|
|
143
|
+
catch {
|
|
144
|
+
return 0;
|
|
145
|
+
}
|
|
146
|
+
}
|
|
147
|
+
function removeDuplicates(db) {
|
|
148
|
+
let removed = 0;
|
|
149
|
+
try {
|
|
150
|
+
// Remove exact duplicates in memories (keep lowest id)
|
|
151
|
+
const result = db.prepare(`
|
|
152
|
+
DELETE FROM memories
|
|
153
|
+
WHERE id NOT IN (
|
|
154
|
+
SELECT MIN(id) FROM memories GROUP BY content, project_id
|
|
155
|
+
)
|
|
156
|
+
`).run();
|
|
157
|
+
removed += result.changes;
|
|
158
|
+
}
|
|
159
|
+
catch {
|
|
160
|
+
// Table might not exist
|
|
161
|
+
}
|
|
162
|
+
try {
|
|
163
|
+
// Remove exact duplicates in session_memories
|
|
164
|
+
const result = db.prepare(`
|
|
165
|
+
DELETE FROM session_memories
|
|
166
|
+
WHERE id NOT IN (
|
|
167
|
+
SELECT MIN(id) FROM session_memories GROUP BY content, session_id
|
|
168
|
+
)
|
|
169
|
+
`).run();
|
|
170
|
+
removed += result.changes;
|
|
171
|
+
}
|
|
172
|
+
catch {
|
|
173
|
+
// Table might not exist
|
|
174
|
+
}
|
|
175
|
+
return removed;
|
|
176
|
+
}
|
|
177
|
+
function detectStaleWorktrees(dbPath, cfg) {
|
|
178
|
+
// Look for .worktrees directory relative to the DB path
|
|
179
|
+
// DB is typically at agents/data/memory/short_term.db, project root is 3 levels up
|
|
180
|
+
const projectRoot = join(dbPath, '..', '..', '..', '..');
|
|
181
|
+
const worktreesDir = join(projectRoot, '.worktrees');
|
|
182
|
+
if (!existsSync(worktreesDir))
|
|
183
|
+
return [];
|
|
184
|
+
const stale = [];
|
|
185
|
+
const staleDays = cfg.archiveDaysOld; // reuse same threshold
|
|
186
|
+
const cutoff = Date.now() - staleDays * 24 * 60 * 60 * 1000;
|
|
187
|
+
try {
|
|
188
|
+
const entries = readdirSync(worktreesDir);
|
|
189
|
+
for (const entry of entries) {
|
|
190
|
+
const entryPath = join(worktreesDir, entry);
|
|
191
|
+
try {
|
|
192
|
+
const stats = statSync(entryPath);
|
|
193
|
+
if (stats.isDirectory() && stats.mtimeMs < cutoff) {
|
|
194
|
+
stale.push(entry);
|
|
195
|
+
}
|
|
196
|
+
}
|
|
197
|
+
catch {
|
|
198
|
+
// Skip entries we can't stat
|
|
199
|
+
}
|
|
200
|
+
}
|
|
201
|
+
}
|
|
202
|
+
catch {
|
|
203
|
+
// Can't read directory
|
|
204
|
+
}
|
|
205
|
+
return stale;
|
|
206
|
+
}
|
|
207
|
+
function generateRecommendations(db, cfg, staleWorktrees = []) {
|
|
208
|
+
const recs = [];
|
|
209
|
+
try {
|
|
210
|
+
// Check total memory count
|
|
211
|
+
const memCount = db.prepare('SELECT COUNT(*) as c FROM memories').get().c;
|
|
212
|
+
if (memCount > 45) {
|
|
213
|
+
recs.push(`Working memory near capacity (${memCount}/50). Consider promoting important entries and pruning old ones.`);
|
|
214
|
+
}
|
|
215
|
+
// Check for stale entries
|
|
216
|
+
const staleDate = new Date();
|
|
217
|
+
staleDate.setDate(staleDate.getDate() - cfg.staleDaysThreshold);
|
|
218
|
+
const staleCount = db.prepare(`
|
|
219
|
+
SELECT COUNT(*) as c FROM memories WHERE timestamp < ? AND importance <= 3
|
|
220
|
+
`).get(staleDate.toISOString()).c;
|
|
221
|
+
if (staleCount > 0) {
|
|
222
|
+
recs.push(`${staleCount} stale entries with low importance found. Run maintenance again or review manually.`);
|
|
223
|
+
}
|
|
224
|
+
}
|
|
225
|
+
catch {
|
|
226
|
+
// Table might not exist
|
|
227
|
+
}
|
|
228
|
+
try {
|
|
229
|
+
// Check unpromoted daily log entries
|
|
230
|
+
ensureDailyLogSchema(db);
|
|
231
|
+
const unpromoted = db.prepare(`
|
|
232
|
+
SELECT COUNT(*) as c FROM daily_log WHERE promoted = 0
|
|
233
|
+
`).get().c;
|
|
234
|
+
if (unpromoted > 10) {
|
|
235
|
+
recs.push(`${unpromoted} unpromoted daily log entries. Run \`uam memory promote\` to review.`);
|
|
236
|
+
}
|
|
237
|
+
}
|
|
238
|
+
catch {
|
|
239
|
+
// Table might not exist
|
|
240
|
+
}
|
|
241
|
+
try {
|
|
242
|
+
// Check for entries with very low importance
|
|
243
|
+
const lowImp = db.prepare(`
|
|
244
|
+
SELECT COUNT(*) as c FROM session_memories WHERE importance <= 2
|
|
245
|
+
`).get().c;
|
|
246
|
+
if (lowImp > 20) {
|
|
247
|
+
recs.push(`${lowImp} session memories with very low importance. These may be pruned on next maintenance.`);
|
|
248
|
+
}
|
|
249
|
+
}
|
|
250
|
+
catch {
|
|
251
|
+
// Table might not exist
|
|
252
|
+
}
|
|
253
|
+
if (staleWorktrees.length > 0) {
|
|
254
|
+
recs.push(`${staleWorktrees.length} stale worktrees found (>${cfg.archiveDaysOld} days old): ${staleWorktrees.slice(0, 5).join(', ')}${staleWorktrees.length > 5 ? '...' : ''}. Run \`uam worktree cleanup <id>\` to remove.`);
|
|
255
|
+
}
|
|
256
|
+
if (recs.length === 0) {
|
|
257
|
+
recs.push('Memory system is healthy. No action needed.');
|
|
258
|
+
}
|
|
259
|
+
return recs;
|
|
260
|
+
}
|
|
261
|
+
/**
|
|
262
|
+
* Get a quick health summary without running full maintenance.
|
|
263
|
+
*/
|
|
264
|
+
export function getHealthSummary(dbPath) {
|
|
265
|
+
const summary = {
|
|
266
|
+
memoriesCount: 0,
|
|
267
|
+
sessionCount: 0,
|
|
268
|
+
dailyLogCount: 0,
|
|
269
|
+
staleCount: 0,
|
|
270
|
+
unpromotedCount: 0,
|
|
271
|
+
healthy: true,
|
|
272
|
+
};
|
|
273
|
+
if (!existsSync(dbPath))
|
|
274
|
+
return summary;
|
|
275
|
+
const db = new Database(dbPath, { readonly: true });
|
|
276
|
+
try {
|
|
277
|
+
try {
|
|
278
|
+
summary.memoriesCount = db.prepare('SELECT COUNT(*) as c FROM memories').get().c;
|
|
279
|
+
}
|
|
280
|
+
catch { /* table doesn't exist */ }
|
|
281
|
+
try {
|
|
282
|
+
summary.sessionCount = db.prepare('SELECT COUNT(*) as c FROM session_memories').get().c;
|
|
283
|
+
}
|
|
284
|
+
catch { /* table doesn't exist */ }
|
|
285
|
+
try {
|
|
286
|
+
summary.dailyLogCount = db.prepare('SELECT COUNT(*) as c FROM daily_log').get().c;
|
|
287
|
+
summary.unpromotedCount = db.prepare('SELECT COUNT(*) as c FROM daily_log WHERE promoted = 0').get().c;
|
|
288
|
+
}
|
|
289
|
+
catch { /* table doesn't exist */ }
|
|
290
|
+
const staleDate = new Date();
|
|
291
|
+
staleDate.setDate(staleDate.getDate() - 14);
|
|
292
|
+
try {
|
|
293
|
+
summary.staleCount = db.prepare(`
|
|
294
|
+
SELECT COUNT(*) as c FROM memories WHERE timestamp < ? AND importance <= 3
|
|
295
|
+
`).get(staleDate.toISOString()).c;
|
|
296
|
+
}
|
|
297
|
+
catch { /* table doesn't exist */ }
|
|
298
|
+
summary.healthy = summary.memoriesCount <= 45 && summary.staleCount < 10 && summary.unpromotedCount < 20;
|
|
299
|
+
}
|
|
300
|
+
finally {
|
|
301
|
+
db.close();
|
|
302
|
+
}
|
|
303
|
+
return summary;
|
|
304
|
+
}
|
|
305
|
+
//# sourceMappingURL=memory-maintenance.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"memory-maintenance.js","sourceRoot":"","sources":["../../src/memory/memory-maintenance.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,39 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Write Gate for UAM Memory System
|
|
3
|
+
*
|
|
4
|
+
* Evaluates candidate memories against 5 criteria before persisting.
|
|
5
|
+
* Inspired by Total Recall's write gate: "Does this change future behavior?"
|
|
6
|
+
*
|
|
7
|
+
* Gate Criteria:
|
|
8
|
+
* 1. Behavioral change - changes how the agent operates in future
|
|
9
|
+
* 2. Commitment with consequences - deadline, deliverable, follow-up
|
|
10
|
+
* 3. Decision with rationale - why X was chosen over Y
|
|
11
|
+
* 4. Stable recurring fact - not transient, will matter again
|
|
12
|
+
* 5. Explicit user request - user said "remember this"
|
|
13
|
+
*/
|
|
14
|
+
export interface WriteGateResult {
|
|
15
|
+
passed: boolean;
|
|
16
|
+
score: number;
|
|
17
|
+
criteria: GateCriteria[];
|
|
18
|
+
rejectionReason?: string;
|
|
19
|
+
}
|
|
20
|
+
export interface GateCriteria {
|
|
21
|
+
name: string;
|
|
22
|
+
matched: boolean;
|
|
23
|
+
confidence: number;
|
|
24
|
+
evidence?: string;
|
|
25
|
+
}
|
|
26
|
+
export interface WriteGateConfig {
|
|
27
|
+
minScore: number;
|
|
28
|
+
enableFuzzyMatching: boolean;
|
|
29
|
+
}
|
|
30
|
+
/**
|
|
31
|
+
* Evaluate whether a candidate memory should be persisted.
|
|
32
|
+
* Returns a WriteGateResult with score and matched criteria.
|
|
33
|
+
*/
|
|
34
|
+
export declare function evaluateWriteGate(content: string, config?: WriteGateConfig): WriteGateResult;
|
|
35
|
+
/**
|
|
36
|
+
* Format a human-readable summary of the write gate evaluation.
|
|
37
|
+
*/
|
|
38
|
+
export declare function formatGateResult(result: WriteGateResult): string;
|
|
39
|
+
//# sourceMappingURL=write-gate.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"write-gate.d.ts","sourceRoot":"","sources":["../../src/memory/write-gate.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;GAYG;AAEH,MAAM,WAAW,eAAe;IAC9B,MAAM,EAAE,OAAO,CAAC;IAChB,KAAK,EAAE,MAAM,CAAC;IACd,QAAQ,EAAE,YAAY,EAAE,CAAC;IACzB,eAAe,CAAC,EAAE,MAAM,CAAC;CAC1B;AAED,MAAM,WAAW,YAAY;IAC3B,IAAI,EAAE,MAAM,CAAC;IACb,OAAO,EAAE,OAAO,CAAC;IACjB,UAAU,EAAE,MAAM,CAAC;IACnB,QAAQ,CAAC,EAAE,MAAM,CAAC;CACnB;AAED,MAAM,WAAW,eAAe;IAC9B,QAAQ,EAAE,MAAM,CAAC;IACjB,mBAAmB,EAAE,OAAO,CAAC;CAC9B;AA0DD;;;GAGG;AACH,wBAAgB,iBAAiB,CAC/B,OAAO,EAAE,MAAM,EACf,MAAM,GAAE,eAAgC,GACvC,eAAe,CAqGjB;AAkBD;;GAEG;AACH,wBAAgB,gBAAgB,CAAC,MAAM,EAAE,eAAe,GAAG,MAAM,CAiBhE"}
|
|
@@ -0,0 +1,190 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Write Gate for UAM Memory System
|
|
3
|
+
*
|
|
4
|
+
* Evaluates candidate memories against 5 criteria before persisting.
|
|
5
|
+
* Inspired by Total Recall's write gate: "Does this change future behavior?"
|
|
6
|
+
*
|
|
7
|
+
* Gate Criteria:
|
|
8
|
+
* 1. Behavioral change - changes how the agent operates in future
|
|
9
|
+
* 2. Commitment with consequences - deadline, deliverable, follow-up
|
|
10
|
+
* 3. Decision with rationale - why X was chosen over Y
|
|
11
|
+
* 4. Stable recurring fact - not transient, will matter again
|
|
12
|
+
* 5. Explicit user request - user said "remember this"
|
|
13
|
+
*/
|
|
14
|
+
const DEFAULT_CONFIG = {
|
|
15
|
+
minScore: 0.3,
|
|
16
|
+
enableFuzzyMatching: true,
|
|
17
|
+
};
|
|
18
|
+
const BEHAVIORAL_PATTERNS = [
|
|
19
|
+
/\b(prefer|always|never|don'?t|avoid|stop|instead)\b/i,
|
|
20
|
+
/\b(use|switch to|migrate to|adopt)\b.*\b(over|instead of|not)\b/i,
|
|
21
|
+
/\b(default|convention|standard|rule|policy)\b/i,
|
|
22
|
+
/\b(format|style|indent|naming|casing)\b/i,
|
|
23
|
+
/\b(timezone|locale|language|encoding)\b/i,
|
|
24
|
+
/\b(before|after|when|every time)\b.*\b(always|must|should)\b/i,
|
|
25
|
+
];
|
|
26
|
+
const COMMITMENT_PATTERNS = [
|
|
27
|
+
/\b(deadline|due|by|until|before)\b.*\b(\d{4}|\d{1,2}[\/\-]\d{1,2}|monday|tuesday|wednesday|thursday|friday|saturday|sunday|tomorrow|next week|end of)\b/i,
|
|
28
|
+
/\b(deliver|ship|release|deploy|submit|send)\b.*\b(by|before|on)\b/i,
|
|
29
|
+
/\b(follow up|check back|revisit|circle back|get back)\b/i,
|
|
30
|
+
/\b(waiting on|blocked by|depends on|need from)\b/i,
|
|
31
|
+
/\b(committed to|promised|agreed to|will do)\b/i,
|
|
32
|
+
/\b(TODO|FIXME|HACK)\b/,
|
|
33
|
+
];
|
|
34
|
+
const DECISION_PATTERNS = [
|
|
35
|
+
/\b(decided|chose|picked|selected|went with|opted for)\b/i,
|
|
36
|
+
/\b(because|reason|rationale|trade-?off|pros? and cons?)\b/i,
|
|
37
|
+
/\b(over|instead of|rather than|as opposed to)\b/i,
|
|
38
|
+
/\b(alternative|option|approach|strategy|architecture)\b/i,
|
|
39
|
+
/\b(evaluated|compared|benchmarked|tested)\b.*\b(and|vs\.?|versus)\b/i,
|
|
40
|
+
];
|
|
41
|
+
const STABLE_FACT_PATTERNS = [
|
|
42
|
+
/\b(api|endpoint|url|port|host|domain)\b.*\b(is|at|on)\b/i,
|
|
43
|
+
/\b(version|release|v\d+)\b/i,
|
|
44
|
+
/\b(password|secret|key|token|credential)\b.*\b(stored|located|in|at)\b/i,
|
|
45
|
+
/\b(schema|table|column|field|index)\b.*\b(is|has|named)\b/i,
|
|
46
|
+
/\b(environment|staging|production|dev)\b.*\b(uses?|runs?|on|at)\b/i,
|
|
47
|
+
/\b(rotates?|expires?|renews?)\b.*\b(every|monthly|weekly|daily)\b/i,
|
|
48
|
+
/\b(contact|email|phone|slack)\b.*\b(is|at)\b/i,
|
|
49
|
+
];
|
|
50
|
+
const EXPLICIT_REMEMBER_PATTERNS = [
|
|
51
|
+
/\b(remember|memorize|note|save|store|record|keep in mind)\b.*\b(this|that)\b/i,
|
|
52
|
+
/\b(important|critical|crucial|key|essential)\b.*\b(to (know|note|remember))\b/i,
|
|
53
|
+
/\b(don'?t forget|make sure to remember|for future reference)\b/i,
|
|
54
|
+
/\bremember\s*:/i,
|
|
55
|
+
];
|
|
56
|
+
const NOISE_PATTERNS = [
|
|
57
|
+
/^(thanks|thank you|ok|okay|got it|sounds good|great|perfect|nice|cool|lgtm)/i,
|
|
58
|
+
/^(yes|no|sure|right|correct|exactly|indeed|absolutely)$/i,
|
|
59
|
+
/\b(looks? good|works? for me|makes? sense)\b/i,
|
|
60
|
+
/^(can you|could you|please|would you)\b/i,
|
|
61
|
+
/\b(just ran|just tested|just checked|running now)\b/i,
|
|
62
|
+
];
|
|
63
|
+
/**
|
|
64
|
+
* Evaluate whether a candidate memory should be persisted.
|
|
65
|
+
* Returns a WriteGateResult with score and matched criteria.
|
|
66
|
+
*/
|
|
67
|
+
export function evaluateWriteGate(content, config = DEFAULT_CONFIG) {
|
|
68
|
+
if (!content || content.trim().length === 0) {
|
|
69
|
+
return {
|
|
70
|
+
passed: false,
|
|
71
|
+
score: 0,
|
|
72
|
+
criteria: [],
|
|
73
|
+
rejectionReason: 'Empty content',
|
|
74
|
+
};
|
|
75
|
+
}
|
|
76
|
+
const trimmed = content.trim();
|
|
77
|
+
// Short content is likely noise
|
|
78
|
+
if (trimmed.length < 10) {
|
|
79
|
+
return {
|
|
80
|
+
passed: false,
|
|
81
|
+
score: 0,
|
|
82
|
+
criteria: [],
|
|
83
|
+
rejectionReason: 'Content too short to be a meaningful memory',
|
|
84
|
+
};
|
|
85
|
+
}
|
|
86
|
+
// Check for noise patterns first
|
|
87
|
+
for (const pattern of NOISE_PATTERNS) {
|
|
88
|
+
if (pattern.test(trimmed)) {
|
|
89
|
+
return {
|
|
90
|
+
passed: false,
|
|
91
|
+
score: 0,
|
|
92
|
+
criteria: [],
|
|
93
|
+
rejectionReason: 'Content matches noise pattern (acknowledgment, transient request)',
|
|
94
|
+
};
|
|
95
|
+
}
|
|
96
|
+
}
|
|
97
|
+
const criteria = [];
|
|
98
|
+
// Criterion 1: Behavioral change
|
|
99
|
+
const behavioralScore = matchPatterns(trimmed, BEHAVIORAL_PATTERNS);
|
|
100
|
+
criteria.push({
|
|
101
|
+
name: 'behavioral_change',
|
|
102
|
+
matched: behavioralScore > 0,
|
|
103
|
+
confidence: behavioralScore,
|
|
104
|
+
evidence: behavioralScore > 0 ? 'Changes how the agent should operate' : undefined,
|
|
105
|
+
});
|
|
106
|
+
// Criterion 2: Commitment with consequences
|
|
107
|
+
const commitmentScore = matchPatterns(trimmed, COMMITMENT_PATTERNS);
|
|
108
|
+
criteria.push({
|
|
109
|
+
name: 'commitment',
|
|
110
|
+
matched: commitmentScore > 0,
|
|
111
|
+
confidence: commitmentScore,
|
|
112
|
+
evidence: commitmentScore > 0 ? 'Contains deadline, deliverable, or follow-up' : undefined,
|
|
113
|
+
});
|
|
114
|
+
// Criterion 3: Decision with rationale
|
|
115
|
+
const decisionScore = matchPatterns(trimmed, DECISION_PATTERNS);
|
|
116
|
+
criteria.push({
|
|
117
|
+
name: 'decision_rationale',
|
|
118
|
+
matched: decisionScore > 0,
|
|
119
|
+
confidence: decisionScore,
|
|
120
|
+
evidence: decisionScore > 0 ? 'Records a decision and its reasoning' : undefined,
|
|
121
|
+
});
|
|
122
|
+
// Criterion 4: Stable recurring fact
|
|
123
|
+
const factScore = matchPatterns(trimmed, STABLE_FACT_PATTERNS);
|
|
124
|
+
criteria.push({
|
|
125
|
+
name: 'stable_fact',
|
|
126
|
+
matched: factScore > 0,
|
|
127
|
+
confidence: factScore,
|
|
128
|
+
evidence: factScore > 0 ? 'Durable fact that will be referenced again' : undefined,
|
|
129
|
+
});
|
|
130
|
+
// Criterion 5: Explicit user request
|
|
131
|
+
const explicitScore = matchPatterns(trimmed, EXPLICIT_REMEMBER_PATTERNS);
|
|
132
|
+
criteria.push({
|
|
133
|
+
name: 'explicit_request',
|
|
134
|
+
matched: explicitScore > 0,
|
|
135
|
+
confidence: explicitScore,
|
|
136
|
+
evidence: explicitScore > 0 ? 'User explicitly requested to remember' : undefined,
|
|
137
|
+
});
|
|
138
|
+
// Aggregate score: highest matching criterion wins, with bonus for multiple matches
|
|
139
|
+
const matchedCriteria = criteria.filter(c => c.matched);
|
|
140
|
+
const maxConfidence = Math.max(...criteria.map(c => c.confidence), 0);
|
|
141
|
+
const multiMatchBonus = matchedCriteria.length > 1 ? 0.1 * (matchedCriteria.length - 1) : 0;
|
|
142
|
+
const score = Math.min(1.0, maxConfidence + multiMatchBonus);
|
|
143
|
+
// High importance heuristic: long, structured content with technical terms
|
|
144
|
+
const lengthBonus = trimmed.length > 200 ? 0.15 : trimmed.length > 100 ? 0.05 : 0;
|
|
145
|
+
const finalScore = Math.min(1.0, score + lengthBonus);
|
|
146
|
+
const passed = finalScore >= config.minScore;
|
|
147
|
+
return {
|
|
148
|
+
passed,
|
|
149
|
+
score: finalScore,
|
|
150
|
+
criteria,
|
|
151
|
+
rejectionReason: passed
|
|
152
|
+
? undefined
|
|
153
|
+
: 'Content does not match any write gate criteria (behavioral change, commitment, decision, stable fact, or explicit request)',
|
|
154
|
+
};
|
|
155
|
+
}
|
|
156
|
+
/**
|
|
157
|
+
* Match content against a set of regex patterns.
|
|
158
|
+
* Returns a confidence score 0-1 based on number and quality of matches.
|
|
159
|
+
*/
|
|
160
|
+
function matchPatterns(content, patterns) {
|
|
161
|
+
let matches = 0;
|
|
162
|
+
for (const pattern of patterns) {
|
|
163
|
+
if (pattern.test(content)) {
|
|
164
|
+
matches++;
|
|
165
|
+
}
|
|
166
|
+
}
|
|
167
|
+
if (matches === 0)
|
|
168
|
+
return 0;
|
|
169
|
+
// Score: 0.4 for first match, +0.15 for each additional, cap at 1.0
|
|
170
|
+
return Math.min(1.0, 0.4 + (matches - 1) * 0.15);
|
|
171
|
+
}
|
|
172
|
+
/**
|
|
173
|
+
* Format a human-readable summary of the write gate evaluation.
|
|
174
|
+
*/
|
|
175
|
+
export function formatGateResult(result) {
|
|
176
|
+
const status = result.passed ? 'PASSED' : 'REJECTED';
|
|
177
|
+
const lines = [`Write Gate: ${status} (score: ${result.score.toFixed(2)})`];
|
|
178
|
+
for (const criterion of result.criteria) {
|
|
179
|
+
const icon = criterion.matched ? '+' : '-';
|
|
180
|
+
lines.push(` ${icon} ${criterion.name}: ${criterion.matched ? `yes (${criterion.confidence.toFixed(2)})` : 'no'}`);
|
|
181
|
+
if (criterion.evidence) {
|
|
182
|
+
lines.push(` ${criterion.evidence}`);
|
|
183
|
+
}
|
|
184
|
+
}
|
|
185
|
+
if (result.rejectionReason) {
|
|
186
|
+
lines.push(` Reason: ${result.rejectionReason}`);
|
|
187
|
+
}
|
|
188
|
+
return lines.join('\n');
|
|
189
|
+
}
|
|
190
|
+
//# sourceMappingURL=write-gate.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"write-gate.js","sourceRoot":"","sources":["../../src/memory/write-gate.ts"],"names":[],"mappings":"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"}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "universal-agent-memory",
|
|
3
|
-
"version": "
|
|
3
|
+
"version": "3.0.1",
|
|
4
4
|
"description": "Universal AI agent memory system - CLAUDE.md templates, memory, worktrees for Claude Code, Factory.AI, VSCode, OpenCode",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"main": "dist/index.js",
|