@claude-flow/memory 3.0.0-alpha.1 → 3.0.0-alpha.7
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 +356 -18
- package/dist/agent-memory-scope.d.ts +131 -0
- package/dist/agent-memory-scope.d.ts.map +1 -0
- package/dist/agent-memory-scope.js +215 -0
- package/dist/agent-memory-scope.js.map +1 -0
- package/dist/agent-memory-scope.test.d.ts +8 -0
- package/dist/agent-memory-scope.test.d.ts.map +1 -0
- package/dist/agent-memory-scope.test.js +463 -0
- package/dist/agent-memory-scope.test.js.map +1 -0
- package/dist/agentdb-adapter.d.ts +22 -3
- package/dist/agentdb-adapter.d.ts.map +1 -1
- package/dist/agentdb-adapter.js +135 -8
- package/dist/agentdb-adapter.js.map +1 -1
- package/dist/auto-memory-bridge.d.ts +226 -0
- package/dist/auto-memory-bridge.d.ts.map +1 -0
- package/dist/auto-memory-bridge.js +709 -0
- package/dist/auto-memory-bridge.js.map +1 -0
- package/dist/auto-memory-bridge.test.d.ts +8 -0
- package/dist/auto-memory-bridge.test.d.ts.map +1 -0
- package/dist/auto-memory-bridge.test.js +754 -0
- package/dist/auto-memory-bridge.test.js.map +1 -0
- package/dist/benchmark.test.d.ts +2 -0
- package/dist/benchmark.test.d.ts.map +1 -0
- package/dist/benchmark.test.js +277 -0
- package/dist/benchmark.test.js.map +1 -0
- package/dist/index.d.ts +8 -0
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +8 -0
- package/dist/index.js.map +1 -1
- package/dist/learning-bridge.d.ts +137 -0
- package/dist/learning-bridge.d.ts.map +1 -0
- package/dist/learning-bridge.js +335 -0
- package/dist/learning-bridge.js.map +1 -0
- package/dist/learning-bridge.test.d.ts +8 -0
- package/dist/learning-bridge.test.d.ts.map +1 -0
- package/dist/learning-bridge.test.js +578 -0
- package/dist/learning-bridge.test.js.map +1 -0
- package/dist/memory-graph.d.ts +100 -0
- package/dist/memory-graph.d.ts.map +1 -0
- package/dist/memory-graph.js +333 -0
- package/dist/memory-graph.js.map +1 -0
- package/dist/memory-graph.test.d.ts +8 -0
- package/dist/memory-graph.test.d.ts.map +1 -0
- package/dist/memory-graph.test.js +609 -0
- package/dist/memory-graph.test.js.map +1 -0
- package/dist/types.d.ts +3 -0
- package/dist/types.d.ts.map +1 -1
- package/package.json +19 -4
- package/.agentic-flow/intelligence.json +0 -16
- package/__tests__/coverage/base.css +0 -224
- package/__tests__/coverage/block-navigation.js +0 -87
- package/__tests__/coverage/coverage-final.json +0 -19
- package/__tests__/coverage/favicon.png +0 -0
- package/__tests__/coverage/index.html +0 -206
- package/__tests__/coverage/lcov-report/base.css +0 -224
- package/__tests__/coverage/lcov-report/block-navigation.js +0 -87
- package/__tests__/coverage/lcov-report/favicon.png +0 -0
- package/__tests__/coverage/lcov-report/index.html +0 -206
- package/__tests__/coverage/lcov-report/prettify.css +0 -1
- package/__tests__/coverage/lcov-report/prettify.js +0 -2
- package/__tests__/coverage/lcov-report/sort-arrow-sprite.png +0 -0
- package/__tests__/coverage/lcov-report/sorter.js +0 -210
- package/__tests__/coverage/lcov-report/src/agentdb-adapter.ts.html +0 -2737
- package/__tests__/coverage/lcov-report/src/agentdb-backend.ts.html +0 -3130
- package/__tests__/coverage/lcov-report/src/application/commands/delete-memory.command.ts.html +0 -601
- package/__tests__/coverage/lcov-report/src/application/commands/index.html +0 -131
- package/__tests__/coverage/lcov-report/src/application/commands/store-memory.command.ts.html +0 -394
- package/__tests__/coverage/lcov-report/src/application/queries/index.html +0 -116
- package/__tests__/coverage/lcov-report/src/application/queries/search-memory.query.ts.html +0 -796
- package/__tests__/coverage/lcov-report/src/application/services/index.html +0 -116
- package/__tests__/coverage/lcov-report/src/application/services/memory-application-service.ts.html +0 -793
- package/__tests__/coverage/lcov-report/src/cache-manager.ts.html +0 -1633
- package/__tests__/coverage/lcov-report/src/database-provider.ts.html +0 -1618
- package/__tests__/coverage/lcov-report/src/domain/entities/index.html +0 -116
- package/__tests__/coverage/lcov-report/src/domain/entities/memory-entry.ts.html +0 -952
- package/__tests__/coverage/lcov-report/src/domain/services/index.html +0 -116
- package/__tests__/coverage/lcov-report/src/domain/services/memory-domain-service.ts.html +0 -1294
- package/__tests__/coverage/lcov-report/src/hnsw-index.ts.html +0 -3124
- package/__tests__/coverage/lcov-report/src/hybrid-backend.ts.html +0 -2167
- package/__tests__/coverage/lcov-report/src/index.html +0 -266
- package/__tests__/coverage/lcov-report/src/infrastructure/repositories/hybrid-memory-repository.ts.html +0 -1633
- package/__tests__/coverage/lcov-report/src/infrastructure/repositories/index.html +0 -116
- package/__tests__/coverage/lcov-report/src/migration.ts.html +0 -2092
- package/__tests__/coverage/lcov-report/src/query-builder.ts.html +0 -1711
- package/__tests__/coverage/lcov-report/src/sqlite-backend.ts.html +0 -2281
- package/__tests__/coverage/lcov-report/src/sqljs-backend.ts.html +0 -2374
- package/__tests__/coverage/lcov-report/src/types.ts.html +0 -2266
- package/__tests__/coverage/lcov.info +0 -10238
- package/__tests__/coverage/prettify.css +0 -1
- package/__tests__/coverage/prettify.js +0 -2
- package/__tests__/coverage/sort-arrow-sprite.png +0 -0
- package/__tests__/coverage/sorter.js +0 -210
- package/__tests__/coverage/src/agentdb-adapter.ts.html +0 -2737
- package/__tests__/coverage/src/agentdb-backend.ts.html +0 -3130
- package/__tests__/coverage/src/application/commands/delete-memory.command.ts.html +0 -601
- package/__tests__/coverage/src/application/commands/index.html +0 -131
- package/__tests__/coverage/src/application/commands/store-memory.command.ts.html +0 -394
- package/__tests__/coverage/src/application/queries/index.html +0 -116
- package/__tests__/coverage/src/application/queries/search-memory.query.ts.html +0 -796
- package/__tests__/coverage/src/application/services/index.html +0 -116
- package/__tests__/coverage/src/application/services/memory-application-service.ts.html +0 -793
- package/__tests__/coverage/src/cache-manager.ts.html +0 -1633
- package/__tests__/coverage/src/database-provider.ts.html +0 -1618
- package/__tests__/coverage/src/domain/entities/index.html +0 -116
- package/__tests__/coverage/src/domain/entities/memory-entry.ts.html +0 -952
- package/__tests__/coverage/src/domain/services/index.html +0 -116
- package/__tests__/coverage/src/domain/services/memory-domain-service.ts.html +0 -1294
- package/__tests__/coverage/src/hnsw-index.ts.html +0 -3124
- package/__tests__/coverage/src/hybrid-backend.ts.html +0 -2167
- package/__tests__/coverage/src/index.html +0 -266
- package/__tests__/coverage/src/infrastructure/repositories/hybrid-memory-repository.ts.html +0 -1633
- package/__tests__/coverage/src/infrastructure/repositories/index.html +0 -116
- package/__tests__/coverage/src/migration.ts.html +0 -2092
- package/__tests__/coverage/src/query-builder.ts.html +0 -1711
- package/__tests__/coverage/src/sqlite-backend.ts.html +0 -2281
- package/__tests__/coverage/src/sqljs-backend.ts.html +0 -2374
- package/__tests__/coverage/src/types.ts.html +0 -2266
- package/benchmarks/cache-hit-rate.bench.ts +0 -535
- package/benchmarks/hnsw-indexing.bench.ts +0 -552
- package/benchmarks/memory-write.bench.ts +0 -469
- package/benchmarks/vector-search.bench.ts +0 -449
- package/docs/AGENTDB-INTEGRATION.md +0 -388
- package/docs/CROSS_PLATFORM.md +0 -505
- package/docs/WINDOWS_SUPPORT.md +0 -422
- package/examples/agentdb-example.ts +0 -345
- package/examples/cross-platform-usage.ts +0 -326
- package/framework/benchmark.ts +0 -112
- package/src/agentdb-adapter.ts +0 -884
- package/src/agentdb-backend.test.ts +0 -339
- package/src/agentdb-backend.ts +0 -1016
- package/src/application/commands/delete-memory.command.ts +0 -172
- package/src/application/commands/store-memory.command.ts +0 -103
- package/src/application/index.ts +0 -36
- package/src/application/queries/search-memory.query.ts +0 -237
- package/src/application/services/memory-application-service.ts +0 -236
- package/src/cache-manager.ts +0 -516
- package/src/database-provider.test.ts +0 -364
- package/src/database-provider.ts +0 -511
- package/src/domain/entities/memory-entry.ts +0 -289
- package/src/domain/index.ts +0 -35
- package/src/domain/repositories/memory-repository.interface.ts +0 -120
- package/src/domain/services/memory-domain-service.ts +0 -403
- package/src/hnsw-index.ts +0 -1013
- package/src/hybrid-backend.test.ts +0 -399
- package/src/hybrid-backend.ts +0 -694
- package/src/index.ts +0 -515
- package/src/infrastructure/index.ts +0 -23
- package/src/infrastructure/repositories/hybrid-memory-repository.ts +0 -516
- package/src/migration.ts +0 -669
- package/src/query-builder.ts +0 -542
- package/src/sqlite-backend.ts +0 -732
- package/src/sqljs-backend.ts +0 -763
- package/src/types.ts +0 -727
- package/tsconfig.json +0 -9
- package/tsconfig.tsbuildinfo +0 -1
- package/verify-cross-platform.ts +0 -170
|
@@ -0,0 +1,709 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* AutoMemoryBridge - Bidirectional sync between Claude Code Auto Memory and AgentDB
|
|
3
|
+
*
|
|
4
|
+
* Per ADR-048: Bridges Claude Code's auto memory (markdown files at
|
|
5
|
+
* ~/.claude/projects/<project>/memory/) with claude-flow's unified memory
|
|
6
|
+
* system (AgentDB + HNSW).
|
|
7
|
+
*
|
|
8
|
+
* Auto memory files are human-readable markdown that Claude loads into its
|
|
9
|
+
* system prompt. MEMORY.md (first 200 lines) is the entrypoint; topic files
|
|
10
|
+
* store detailed notes and are read on demand.
|
|
11
|
+
*
|
|
12
|
+
* @module @claude-flow/memory/auto-memory-bridge
|
|
13
|
+
*/
|
|
14
|
+
import { createHash } from 'node:crypto';
|
|
15
|
+
import { EventEmitter } from 'node:events';
|
|
16
|
+
import * as fs from 'node:fs/promises';
|
|
17
|
+
import { existsSync, readFileSync, readdirSync } from 'node:fs';
|
|
18
|
+
import * as path from 'node:path';
|
|
19
|
+
import { createDefaultEntry, } from './types.js';
|
|
20
|
+
import { LearningBridge } from './learning-bridge.js';
|
|
21
|
+
import { MemoryGraph } from './memory-graph.js';
|
|
22
|
+
// ===== Constants =====
|
|
23
|
+
const DEFAULT_TOPIC_MAPPING = {
|
|
24
|
+
'project-patterns': 'patterns.md',
|
|
25
|
+
'debugging': 'debugging.md',
|
|
26
|
+
'architecture': 'architecture.md',
|
|
27
|
+
'performance': 'performance.md',
|
|
28
|
+
'security': 'security.md',
|
|
29
|
+
'preferences': 'preferences.md',
|
|
30
|
+
'swarm-results': 'swarm-results.md',
|
|
31
|
+
};
|
|
32
|
+
const CATEGORY_LABELS = {
|
|
33
|
+
'project-patterns': 'Project Patterns',
|
|
34
|
+
'debugging': 'Debugging',
|
|
35
|
+
'architecture': 'Architecture',
|
|
36
|
+
'performance': 'Performance',
|
|
37
|
+
'security': 'Security',
|
|
38
|
+
'preferences': 'Preferences',
|
|
39
|
+
'swarm-results': 'Swarm Results',
|
|
40
|
+
};
|
|
41
|
+
const DEFAULT_CONFIG = {
|
|
42
|
+
memoryDir: '',
|
|
43
|
+
workingDir: process.cwd(),
|
|
44
|
+
maxIndexLines: 180,
|
|
45
|
+
topicMapping: DEFAULT_TOPIC_MAPPING,
|
|
46
|
+
syncMode: 'on-session-end',
|
|
47
|
+
syncIntervalMs: 60_000,
|
|
48
|
+
minConfidence: 0.7,
|
|
49
|
+
maxTopicFileLines: 500,
|
|
50
|
+
pruneStrategy: 'confidence-weighted',
|
|
51
|
+
};
|
|
52
|
+
// ===== AutoMemoryBridge =====
|
|
53
|
+
/**
|
|
54
|
+
* Bidirectional bridge between Claude Code auto memory and AgentDB.
|
|
55
|
+
*
|
|
56
|
+
* @example
|
|
57
|
+
* ```typescript
|
|
58
|
+
* const bridge = new AutoMemoryBridge(memoryBackend, {
|
|
59
|
+
* workingDir: '/workspaces/my-project',
|
|
60
|
+
* });
|
|
61
|
+
*
|
|
62
|
+
* // Record an insight
|
|
63
|
+
* await bridge.recordInsight({
|
|
64
|
+
* category: 'debugging',
|
|
65
|
+
* summary: 'HNSW index requires initialization before search',
|
|
66
|
+
* source: 'agent:tester',
|
|
67
|
+
* confidence: 0.95,
|
|
68
|
+
* });
|
|
69
|
+
*
|
|
70
|
+
* // Sync to auto memory files
|
|
71
|
+
* await bridge.syncToAutoMemory();
|
|
72
|
+
*
|
|
73
|
+
* // Import auto memory into AgentDB
|
|
74
|
+
* await bridge.importFromAutoMemory();
|
|
75
|
+
* ```
|
|
76
|
+
*/
|
|
77
|
+
export class AutoMemoryBridge extends EventEmitter {
|
|
78
|
+
config;
|
|
79
|
+
backend;
|
|
80
|
+
lastSyncTime = 0;
|
|
81
|
+
syncTimer = null;
|
|
82
|
+
insights = [];
|
|
83
|
+
/** Track AgentDB keys of insights already written to files during this session */
|
|
84
|
+
syncedInsightKeys = new Set();
|
|
85
|
+
/** Monotonic counter to prevent key collisions within the same ms */
|
|
86
|
+
insightCounter = 0;
|
|
87
|
+
/** Optional learning bridge (ADR-049) */
|
|
88
|
+
learningBridge;
|
|
89
|
+
/** Optional knowledge graph (ADR-049) */
|
|
90
|
+
memoryGraph;
|
|
91
|
+
constructor(backend, config = {}) {
|
|
92
|
+
super();
|
|
93
|
+
this.backend = backend;
|
|
94
|
+
this.config = {
|
|
95
|
+
...DEFAULT_CONFIG,
|
|
96
|
+
...config,
|
|
97
|
+
topicMapping: {
|
|
98
|
+
...DEFAULT_TOPIC_MAPPING,
|
|
99
|
+
...(config.topicMapping || {}),
|
|
100
|
+
},
|
|
101
|
+
};
|
|
102
|
+
if (!this.config.memoryDir) {
|
|
103
|
+
this.config.memoryDir = resolveAutoMemoryDir(this.config.workingDir);
|
|
104
|
+
}
|
|
105
|
+
if (this.config.syncMode === 'periodic' && this.config.syncIntervalMs > 0) {
|
|
106
|
+
this.startPeriodicSync();
|
|
107
|
+
}
|
|
108
|
+
// ADR-049: Initialize optional learning bridge and knowledge graph
|
|
109
|
+
if (config.learning) {
|
|
110
|
+
this.learningBridge = new LearningBridge(backend, config.learning);
|
|
111
|
+
}
|
|
112
|
+
if (config.graph) {
|
|
113
|
+
this.memoryGraph = new MemoryGraph(config.graph);
|
|
114
|
+
}
|
|
115
|
+
}
|
|
116
|
+
/** Get the resolved auto memory directory path */
|
|
117
|
+
getMemoryDir() {
|
|
118
|
+
return this.config.memoryDir;
|
|
119
|
+
}
|
|
120
|
+
/** Get the path to MEMORY.md */
|
|
121
|
+
getIndexPath() {
|
|
122
|
+
return path.join(this.config.memoryDir, 'MEMORY.md');
|
|
123
|
+
}
|
|
124
|
+
/** Get the path to a topic file */
|
|
125
|
+
getTopicPath(category) {
|
|
126
|
+
const filename = this.config.topicMapping[category] || `${category}.md`;
|
|
127
|
+
return path.join(this.config.memoryDir, filename);
|
|
128
|
+
}
|
|
129
|
+
/**
|
|
130
|
+
* Record a memory insight.
|
|
131
|
+
* Stores in the in-memory buffer and optionally writes immediately.
|
|
132
|
+
*/
|
|
133
|
+
async recordInsight(insight) {
|
|
134
|
+
this.insights.push(insight);
|
|
135
|
+
// Store in AgentDB
|
|
136
|
+
const key = await this.storeInsightInAgentDB(insight);
|
|
137
|
+
this.syncedInsightKeys.add(key);
|
|
138
|
+
// If sync-on-write, write immediately to files
|
|
139
|
+
if (this.config.syncMode === 'on-write') {
|
|
140
|
+
await this.writeInsightToFiles(insight);
|
|
141
|
+
}
|
|
142
|
+
// ADR-049: Notify learning bridge
|
|
143
|
+
if (this.learningBridge) {
|
|
144
|
+
await this.learningBridge.onInsightRecorded(insight, key);
|
|
145
|
+
}
|
|
146
|
+
this.emit('insight:recorded', insight);
|
|
147
|
+
}
|
|
148
|
+
/**
|
|
149
|
+
* Sync high-confidence AgentDB entries to auto memory files.
|
|
150
|
+
* Called on session-end or periodically.
|
|
151
|
+
*/
|
|
152
|
+
async syncToAutoMemory() {
|
|
153
|
+
const startTime = Date.now();
|
|
154
|
+
const errors = [];
|
|
155
|
+
const updatedCategories = new Set();
|
|
156
|
+
try {
|
|
157
|
+
// ADR-049: Consolidate learning trajectories before syncing
|
|
158
|
+
if (this.learningBridge) {
|
|
159
|
+
await this.learningBridge.consolidate();
|
|
160
|
+
}
|
|
161
|
+
// Ensure directory exists
|
|
162
|
+
await this.ensureMemoryDir();
|
|
163
|
+
// Snapshot and clear the buffer atomically to avoid race conditions
|
|
164
|
+
const buffered = this.insights.splice(0, this.insights.length);
|
|
165
|
+
// Flush buffered insights to files
|
|
166
|
+
for (const insight of buffered) {
|
|
167
|
+
try {
|
|
168
|
+
await this.writeInsightToFiles(insight);
|
|
169
|
+
updatedCategories.add(insight.category);
|
|
170
|
+
}
|
|
171
|
+
catch (err) {
|
|
172
|
+
errors.push(`Failed to write insight: ${err.message}`);
|
|
173
|
+
}
|
|
174
|
+
}
|
|
175
|
+
// Query AgentDB for high-confidence entries since last sync,
|
|
176
|
+
// skipping entries we already wrote from the buffer above
|
|
177
|
+
const entries = await this.queryRecentInsights();
|
|
178
|
+
for (const entry of entries) {
|
|
179
|
+
const entryKey = entry.key;
|
|
180
|
+
if (this.syncedInsightKeys.has(entryKey))
|
|
181
|
+
continue;
|
|
182
|
+
try {
|
|
183
|
+
const category = this.classifyEntry(entry);
|
|
184
|
+
await this.appendToTopicFile(category, entry);
|
|
185
|
+
updatedCategories.add(category);
|
|
186
|
+
this.syncedInsightKeys.add(entryKey);
|
|
187
|
+
}
|
|
188
|
+
catch (err) {
|
|
189
|
+
errors.push(`Failed to sync entry ${entry.id}: ${err.message}`);
|
|
190
|
+
}
|
|
191
|
+
}
|
|
192
|
+
// Curate MEMORY.md index
|
|
193
|
+
await this.curateIndex();
|
|
194
|
+
const synced = buffered.length + entries.length;
|
|
195
|
+
this.lastSyncTime = Date.now();
|
|
196
|
+
// Prevent unbounded growth of syncedInsightKeys
|
|
197
|
+
if (this.syncedInsightKeys.size > 10_000) {
|
|
198
|
+
const keys = [...this.syncedInsightKeys];
|
|
199
|
+
this.syncedInsightKeys = new Set(keys.slice(keys.length - 5_000));
|
|
200
|
+
}
|
|
201
|
+
const result = {
|
|
202
|
+
synced,
|
|
203
|
+
categories: [...updatedCategories],
|
|
204
|
+
durationMs: Date.now() - startTime,
|
|
205
|
+
errors,
|
|
206
|
+
};
|
|
207
|
+
this.emit('sync:completed', result);
|
|
208
|
+
return result;
|
|
209
|
+
}
|
|
210
|
+
catch (err) {
|
|
211
|
+
errors.push(`Sync failed: ${err.message}`);
|
|
212
|
+
this.emit('sync:failed', { error: err });
|
|
213
|
+
return {
|
|
214
|
+
synced: 0,
|
|
215
|
+
categories: [],
|
|
216
|
+
durationMs: Date.now() - startTime,
|
|
217
|
+
errors,
|
|
218
|
+
};
|
|
219
|
+
}
|
|
220
|
+
}
|
|
221
|
+
/**
|
|
222
|
+
* Import auto memory files into AgentDB.
|
|
223
|
+
* Called on session-start to hydrate AgentDB with previous learnings.
|
|
224
|
+
* Uses bulk insert for efficiency.
|
|
225
|
+
*/
|
|
226
|
+
async importFromAutoMemory() {
|
|
227
|
+
const startTime = Date.now();
|
|
228
|
+
const memoryDir = this.config.memoryDir;
|
|
229
|
+
if (!existsSync(memoryDir)) {
|
|
230
|
+
return { imported: 0, skipped: 0, files: [], durationMs: 0 };
|
|
231
|
+
}
|
|
232
|
+
let imported = 0;
|
|
233
|
+
let skipped = 0;
|
|
234
|
+
const processedFiles = [];
|
|
235
|
+
const files = readdirSync(memoryDir).filter(f => f.endsWith('.md'));
|
|
236
|
+
// Pre-fetch existing content hashes to avoid N queries
|
|
237
|
+
const existingHashes = await this.fetchExistingContentHashes();
|
|
238
|
+
// Batch entries for bulk insert
|
|
239
|
+
const batch = [];
|
|
240
|
+
for (const file of files) {
|
|
241
|
+
const filePath = path.join(memoryDir, file);
|
|
242
|
+
const content = await fs.readFile(filePath, 'utf-8');
|
|
243
|
+
const entries = parseMarkdownEntries(content);
|
|
244
|
+
for (const entry of entries) {
|
|
245
|
+
const contentHash = hashContent(entry.content);
|
|
246
|
+
if (existingHashes.has(contentHash)) {
|
|
247
|
+
skipped++;
|
|
248
|
+
continue;
|
|
249
|
+
}
|
|
250
|
+
const input = {
|
|
251
|
+
key: `auto-memory:${file}:${entry.heading}`,
|
|
252
|
+
content: entry.content,
|
|
253
|
+
namespace: 'auto-memory',
|
|
254
|
+
type: 'semantic',
|
|
255
|
+
tags: ['auto-memory', file.replace('.md', '')],
|
|
256
|
+
metadata: {
|
|
257
|
+
sourceFile: file,
|
|
258
|
+
heading: entry.heading,
|
|
259
|
+
importedAt: new Date().toISOString(),
|
|
260
|
+
contentHash,
|
|
261
|
+
},
|
|
262
|
+
};
|
|
263
|
+
batch.push(createDefaultEntry(input));
|
|
264
|
+
existingHashes.add(contentHash);
|
|
265
|
+
imported++;
|
|
266
|
+
}
|
|
267
|
+
processedFiles.push(file);
|
|
268
|
+
}
|
|
269
|
+
// Bulk insert all at once
|
|
270
|
+
if (batch.length > 0) {
|
|
271
|
+
await this.backend.bulkInsert(batch);
|
|
272
|
+
}
|
|
273
|
+
// ADR-049: Build knowledge graph from imported entries
|
|
274
|
+
if (this.memoryGraph && batch.length > 0) {
|
|
275
|
+
await this.memoryGraph.buildFromBackend(this.backend, 'auto-memory');
|
|
276
|
+
}
|
|
277
|
+
const result = {
|
|
278
|
+
imported,
|
|
279
|
+
skipped,
|
|
280
|
+
files: processedFiles,
|
|
281
|
+
durationMs: Date.now() - startTime,
|
|
282
|
+
};
|
|
283
|
+
this.emit('import:completed', result);
|
|
284
|
+
return result;
|
|
285
|
+
}
|
|
286
|
+
/**
|
|
287
|
+
* Curate MEMORY.md to stay under the line limit.
|
|
288
|
+
* Groups entries by category and prunes low-confidence items.
|
|
289
|
+
*/
|
|
290
|
+
async curateIndex() {
|
|
291
|
+
await this.ensureMemoryDir();
|
|
292
|
+
// Collect summaries from all topic files
|
|
293
|
+
const sections = {};
|
|
294
|
+
for (const [category, filename] of Object.entries(this.config.topicMapping)) {
|
|
295
|
+
const topicPath = path.join(this.config.memoryDir, filename);
|
|
296
|
+
if (existsSync(topicPath)) {
|
|
297
|
+
const content = await fs.readFile(topicPath, 'utf-8');
|
|
298
|
+
const summaries = extractSummaries(content);
|
|
299
|
+
if (summaries.length > 0) {
|
|
300
|
+
sections[category] = summaries;
|
|
301
|
+
}
|
|
302
|
+
}
|
|
303
|
+
}
|
|
304
|
+
// ADR-049: Use graph PageRank to prioritize sections
|
|
305
|
+
let sectionOrder;
|
|
306
|
+
if (this.memoryGraph) {
|
|
307
|
+
const topNodes = this.memoryGraph.getTopNodes(20);
|
|
308
|
+
const categoryCounts = new Map();
|
|
309
|
+
for (const node of topNodes) {
|
|
310
|
+
const cat = node.community || 'general';
|
|
311
|
+
categoryCounts.set(cat, (categoryCounts.get(cat) || 0) + 1);
|
|
312
|
+
}
|
|
313
|
+
sectionOrder = [...categoryCounts.entries()]
|
|
314
|
+
.sort((a, b) => b[1] - a[1])
|
|
315
|
+
.map(([cat]) => cat)
|
|
316
|
+
.filter((cat) => sections[cat]);
|
|
317
|
+
}
|
|
318
|
+
// Prune sections before building the index to avoid O(n^2) rebuild loop
|
|
319
|
+
const budget = this.config.maxIndexLines;
|
|
320
|
+
pruneSectionsToFit(sections, budget, this.config.pruneStrategy);
|
|
321
|
+
// Build the final index (with optional graph-aware ordering)
|
|
322
|
+
const lines = buildIndexLines(sections, this.config.topicMapping, sectionOrder);
|
|
323
|
+
await fs.writeFile(this.getIndexPath(), lines.join('\n'), 'utf-8');
|
|
324
|
+
this.emit('index:curated', { lines: lines.length });
|
|
325
|
+
}
|
|
326
|
+
/**
|
|
327
|
+
* Get auto memory status: directory info, file count, line counts.
|
|
328
|
+
*/
|
|
329
|
+
getStatus() {
|
|
330
|
+
const memoryDir = this.config.memoryDir;
|
|
331
|
+
if (!existsSync(memoryDir)) {
|
|
332
|
+
return {
|
|
333
|
+
memoryDir,
|
|
334
|
+
exists: false,
|
|
335
|
+
files: [],
|
|
336
|
+
totalLines: 0,
|
|
337
|
+
indexLines: 0,
|
|
338
|
+
lastSyncTime: this.lastSyncTime,
|
|
339
|
+
bufferedInsights: this.insights.length,
|
|
340
|
+
};
|
|
341
|
+
}
|
|
342
|
+
const fileStats = [];
|
|
343
|
+
let totalLines = 0;
|
|
344
|
+
let indexLines = 0;
|
|
345
|
+
let mdFiles;
|
|
346
|
+
try {
|
|
347
|
+
mdFiles = readdirSync(memoryDir).filter(f => f.endsWith('.md'));
|
|
348
|
+
}
|
|
349
|
+
catch {
|
|
350
|
+
return {
|
|
351
|
+
memoryDir,
|
|
352
|
+
exists: true,
|
|
353
|
+
files: [],
|
|
354
|
+
totalLines: 0,
|
|
355
|
+
indexLines: 0,
|
|
356
|
+
lastSyncTime: this.lastSyncTime,
|
|
357
|
+
bufferedInsights: this.insights.length,
|
|
358
|
+
};
|
|
359
|
+
}
|
|
360
|
+
for (const file of mdFiles) {
|
|
361
|
+
try {
|
|
362
|
+
const content = readFileSync(path.join(memoryDir, file), 'utf-8');
|
|
363
|
+
const lineCount = content.split('\n').length;
|
|
364
|
+
fileStats.push({ name: file, lines: lineCount });
|
|
365
|
+
totalLines += lineCount;
|
|
366
|
+
if (file === 'MEMORY.md') {
|
|
367
|
+
indexLines = lineCount;
|
|
368
|
+
}
|
|
369
|
+
}
|
|
370
|
+
catch {
|
|
371
|
+
// Skip unreadable files
|
|
372
|
+
}
|
|
373
|
+
}
|
|
374
|
+
return {
|
|
375
|
+
memoryDir,
|
|
376
|
+
exists: true,
|
|
377
|
+
files: fileStats,
|
|
378
|
+
totalLines,
|
|
379
|
+
indexLines,
|
|
380
|
+
lastSyncTime: this.lastSyncTime,
|
|
381
|
+
bufferedInsights: this.insights.length,
|
|
382
|
+
};
|
|
383
|
+
}
|
|
384
|
+
/** Stop periodic sync and clean up */
|
|
385
|
+
destroy() {
|
|
386
|
+
if (this.syncTimer) {
|
|
387
|
+
clearInterval(this.syncTimer);
|
|
388
|
+
this.syncTimer = null;
|
|
389
|
+
}
|
|
390
|
+
// ADR-049: Clean up learning bridge
|
|
391
|
+
if (this.learningBridge) {
|
|
392
|
+
this.learningBridge.destroy();
|
|
393
|
+
}
|
|
394
|
+
this.removeAllListeners();
|
|
395
|
+
}
|
|
396
|
+
// ===== Private Methods =====
|
|
397
|
+
async ensureMemoryDir() {
|
|
398
|
+
const dir = this.config.memoryDir;
|
|
399
|
+
if (!existsSync(dir)) {
|
|
400
|
+
await fs.mkdir(dir, { recursive: true });
|
|
401
|
+
}
|
|
402
|
+
}
|
|
403
|
+
async storeInsightInAgentDB(insight) {
|
|
404
|
+
const content = insight.detail
|
|
405
|
+
? `${insight.summary}\n\n${insight.detail}`
|
|
406
|
+
: insight.summary;
|
|
407
|
+
const key = `insight:${insight.category}:${Date.now()}:${this.insightCounter++}`;
|
|
408
|
+
const input = {
|
|
409
|
+
key,
|
|
410
|
+
content,
|
|
411
|
+
namespace: 'learnings',
|
|
412
|
+
type: 'semantic',
|
|
413
|
+
tags: ['insight', insight.category, `source:${insight.source}`],
|
|
414
|
+
metadata: {
|
|
415
|
+
category: insight.category,
|
|
416
|
+
summary: insight.summary,
|
|
417
|
+
source: insight.source,
|
|
418
|
+
confidence: insight.confidence,
|
|
419
|
+
contentHash: hashContent(content),
|
|
420
|
+
...(insight.agentDbId ? { linkedEntryId: insight.agentDbId } : {}),
|
|
421
|
+
},
|
|
422
|
+
};
|
|
423
|
+
const entry = createDefaultEntry(input);
|
|
424
|
+
await this.backend.store(entry);
|
|
425
|
+
return key;
|
|
426
|
+
}
|
|
427
|
+
async writeInsightToFiles(insight) {
|
|
428
|
+
await this.ensureMemoryDir();
|
|
429
|
+
const topicPath = this.getTopicPath(insight.category);
|
|
430
|
+
const line = formatInsightLine(insight);
|
|
431
|
+
if (existsSync(topicPath)) {
|
|
432
|
+
const existing = await fs.readFile(topicPath, 'utf-8');
|
|
433
|
+
// Exact line-based dedup: check if the summary already appears as a bullet
|
|
434
|
+
if (hasSummaryLine(existing, insight.summary))
|
|
435
|
+
return;
|
|
436
|
+
const lineCount = existing.split('\n').length;
|
|
437
|
+
if (lineCount >= this.config.maxTopicFileLines) {
|
|
438
|
+
const pruned = pruneTopicFile(existing, this.config.maxTopicFileLines - 10);
|
|
439
|
+
await fs.writeFile(topicPath, pruned + '\n' + line, 'utf-8');
|
|
440
|
+
}
|
|
441
|
+
else {
|
|
442
|
+
await fs.appendFile(topicPath, '\n' + line, 'utf-8');
|
|
443
|
+
}
|
|
444
|
+
}
|
|
445
|
+
else {
|
|
446
|
+
const label = CATEGORY_LABELS[insight.category] || insight.category;
|
|
447
|
+
const header = `# ${label}\n\n`;
|
|
448
|
+
await fs.writeFile(topicPath, header + line, 'utf-8');
|
|
449
|
+
}
|
|
450
|
+
}
|
|
451
|
+
async queryRecentInsights() {
|
|
452
|
+
const query = {
|
|
453
|
+
type: 'hybrid',
|
|
454
|
+
namespace: 'learnings',
|
|
455
|
+
tags: ['insight'],
|
|
456
|
+
updatedAfter: this.lastSyncTime || 0,
|
|
457
|
+
limit: 50,
|
|
458
|
+
};
|
|
459
|
+
try {
|
|
460
|
+
const entries = await this.backend.query(query);
|
|
461
|
+
return entries.filter(e => {
|
|
462
|
+
const confidence = e.metadata?.confidence || 0;
|
|
463
|
+
return confidence >= this.config.minConfidence;
|
|
464
|
+
});
|
|
465
|
+
}
|
|
466
|
+
catch {
|
|
467
|
+
return [];
|
|
468
|
+
}
|
|
469
|
+
}
|
|
470
|
+
classifyEntry(entry) {
|
|
471
|
+
const category = entry.metadata?.category;
|
|
472
|
+
if (category && category in DEFAULT_TOPIC_MAPPING) {
|
|
473
|
+
return category;
|
|
474
|
+
}
|
|
475
|
+
const tags = entry.tags || [];
|
|
476
|
+
if (tags.includes('debugging') || tags.includes('bug') || tags.includes('fix')) {
|
|
477
|
+
return 'debugging';
|
|
478
|
+
}
|
|
479
|
+
if (tags.includes('architecture') || tags.includes('design')) {
|
|
480
|
+
return 'architecture';
|
|
481
|
+
}
|
|
482
|
+
if (tags.includes('performance') || tags.includes('benchmark')) {
|
|
483
|
+
return 'performance';
|
|
484
|
+
}
|
|
485
|
+
if (tags.includes('security') || tags.includes('cve')) {
|
|
486
|
+
return 'security';
|
|
487
|
+
}
|
|
488
|
+
if (tags.includes('swarm') || tags.includes('agent')) {
|
|
489
|
+
return 'swarm-results';
|
|
490
|
+
}
|
|
491
|
+
return 'project-patterns';
|
|
492
|
+
}
|
|
493
|
+
async appendToTopicFile(category, entry) {
|
|
494
|
+
const insight = {
|
|
495
|
+
category,
|
|
496
|
+
summary: entry.metadata?.summary || entry.content.split('\n')[0],
|
|
497
|
+
detail: entry.content,
|
|
498
|
+
source: entry.metadata?.source || 'agentdb',
|
|
499
|
+
confidence: entry.metadata?.confidence || 0.5,
|
|
500
|
+
agentDbId: entry.id,
|
|
501
|
+
};
|
|
502
|
+
await this.writeInsightToFiles(insight);
|
|
503
|
+
}
|
|
504
|
+
/** Fetch all existing content hashes from the auto-memory namespace in one query */
|
|
505
|
+
async fetchExistingContentHashes() {
|
|
506
|
+
try {
|
|
507
|
+
const entries = await this.backend.query({
|
|
508
|
+
type: 'hybrid',
|
|
509
|
+
namespace: 'auto-memory',
|
|
510
|
+
limit: 10_000,
|
|
511
|
+
});
|
|
512
|
+
const hashes = new Set();
|
|
513
|
+
for (const entry of entries) {
|
|
514
|
+
const hash = entry.metadata?.contentHash;
|
|
515
|
+
if (hash)
|
|
516
|
+
hashes.add(hash);
|
|
517
|
+
}
|
|
518
|
+
return hashes;
|
|
519
|
+
}
|
|
520
|
+
catch {
|
|
521
|
+
return new Set();
|
|
522
|
+
}
|
|
523
|
+
}
|
|
524
|
+
startPeriodicSync() {
|
|
525
|
+
this.syncTimer = setInterval(async () => {
|
|
526
|
+
try {
|
|
527
|
+
await this.syncToAutoMemory();
|
|
528
|
+
}
|
|
529
|
+
catch (err) {
|
|
530
|
+
this.emit('sync:error', err);
|
|
531
|
+
}
|
|
532
|
+
}, this.config.syncIntervalMs);
|
|
533
|
+
if (this.syncTimer.unref) {
|
|
534
|
+
this.syncTimer.unref();
|
|
535
|
+
}
|
|
536
|
+
}
|
|
537
|
+
}
|
|
538
|
+
// ===== Utility Functions =====
|
|
539
|
+
/**
|
|
540
|
+
* Resolve the auto memory directory for a given working directory.
|
|
541
|
+
* Mirrors Claude Code's path derivation from git root.
|
|
542
|
+
*/
|
|
543
|
+
export function resolveAutoMemoryDir(workingDir) {
|
|
544
|
+
const gitRoot = findGitRoot(workingDir);
|
|
545
|
+
const basePath = gitRoot || workingDir;
|
|
546
|
+
// Claude Code normalizes to forward slashes then replaces with dashes
|
|
547
|
+
// The leading dash IS preserved (e.g. /workspaces/foo -> -workspaces-foo)
|
|
548
|
+
const normalized = basePath.split(path.sep).join('/');
|
|
549
|
+
const projectKey = normalized.replace(/\//g, '-');
|
|
550
|
+
return path.join(process.env.HOME || process.env.USERPROFILE || '~', '.claude', 'projects', projectKey, 'memory');
|
|
551
|
+
}
|
|
552
|
+
/**
|
|
553
|
+
* Find the git root directory by walking up from workingDir.
|
|
554
|
+
*/
|
|
555
|
+
export function findGitRoot(dir) {
|
|
556
|
+
let current = path.resolve(dir);
|
|
557
|
+
const root = path.parse(current).root;
|
|
558
|
+
while (current !== root) {
|
|
559
|
+
if (existsSync(path.join(current, '.git'))) {
|
|
560
|
+
return current;
|
|
561
|
+
}
|
|
562
|
+
current = path.dirname(current);
|
|
563
|
+
}
|
|
564
|
+
return null;
|
|
565
|
+
}
|
|
566
|
+
/**
|
|
567
|
+
* Parse markdown content into structured entries.
|
|
568
|
+
* Splits on ## headings and extracts content under each.
|
|
569
|
+
*/
|
|
570
|
+
export function parseMarkdownEntries(content) {
|
|
571
|
+
const entries = [];
|
|
572
|
+
const lines = content.split('\n');
|
|
573
|
+
let currentHeading = '';
|
|
574
|
+
let currentLines = [];
|
|
575
|
+
for (const line of lines) {
|
|
576
|
+
const headingMatch = line.match(/^##\s+(.+)/);
|
|
577
|
+
if (headingMatch) {
|
|
578
|
+
if (currentHeading && currentLines.length > 0) {
|
|
579
|
+
entries.push({
|
|
580
|
+
heading: currentHeading,
|
|
581
|
+
content: currentLines.join('\n').trim(),
|
|
582
|
+
metadata: {},
|
|
583
|
+
});
|
|
584
|
+
}
|
|
585
|
+
currentHeading = headingMatch[1];
|
|
586
|
+
currentLines = [];
|
|
587
|
+
}
|
|
588
|
+
else if (currentHeading) {
|
|
589
|
+
currentLines.push(line);
|
|
590
|
+
}
|
|
591
|
+
}
|
|
592
|
+
if (currentHeading && currentLines.length > 0) {
|
|
593
|
+
entries.push({
|
|
594
|
+
heading: currentHeading,
|
|
595
|
+
content: currentLines.join('\n').trim(),
|
|
596
|
+
metadata: {},
|
|
597
|
+
});
|
|
598
|
+
}
|
|
599
|
+
return entries;
|
|
600
|
+
}
|
|
601
|
+
/**
|
|
602
|
+
* Extract clean one-line summaries from a topic file.
|
|
603
|
+
* Returns bullet-point items (lines starting with '- '), stripping
|
|
604
|
+
* metadata annotations like _(source, date, conf: 0.95)_.
|
|
605
|
+
*/
|
|
606
|
+
export function extractSummaries(content) {
|
|
607
|
+
return content
|
|
608
|
+
.split('\n')
|
|
609
|
+
.filter(line => line.startsWith('- '))
|
|
610
|
+
.map(line => line.slice(2).trim())
|
|
611
|
+
.filter(line => !line.startsWith('See `'))
|
|
612
|
+
.map(line => line.replace(/\s*_\(.*?\)_\s*$/, '').trim())
|
|
613
|
+
.filter(Boolean);
|
|
614
|
+
}
|
|
615
|
+
/**
|
|
616
|
+
* Format an insight as a markdown line for topic files.
|
|
617
|
+
*/
|
|
618
|
+
export function formatInsightLine(insight) {
|
|
619
|
+
const timestamp = new Date().toISOString().split('T')[0];
|
|
620
|
+
const prefix = `- ${insight.summary}`;
|
|
621
|
+
const suffix = ` _(${insight.source}, ${timestamp}, conf: ${insight.confidence.toFixed(2)})_`;
|
|
622
|
+
if (insight.detail && insight.detail.split('\n').length > 2) {
|
|
623
|
+
return `${prefix}${suffix}\n ${insight.detail.split('\n').join('\n ')}`;
|
|
624
|
+
}
|
|
625
|
+
return `${prefix}${suffix}`;
|
|
626
|
+
}
|
|
627
|
+
/**
|
|
628
|
+
* Hash content for deduplication.
|
|
629
|
+
*/
|
|
630
|
+
export function hashContent(content) {
|
|
631
|
+
return createHash('sha256').update(content).digest('hex').slice(0, 16);
|
|
632
|
+
}
|
|
633
|
+
/**
|
|
634
|
+
* Prune a topic file to stay under the line limit.
|
|
635
|
+
* Removes oldest entries (those closest to the top after the header).
|
|
636
|
+
*/
|
|
637
|
+
export function pruneTopicFile(content, maxLines) {
|
|
638
|
+
const lines = content.split('\n');
|
|
639
|
+
if (lines.length <= maxLines)
|
|
640
|
+
return content;
|
|
641
|
+
const header = lines.slice(0, 3);
|
|
642
|
+
const entries = lines.slice(3);
|
|
643
|
+
const kept = entries.slice(entries.length - (maxLines - 3));
|
|
644
|
+
return [...header, ...kept].join('\n');
|
|
645
|
+
}
|
|
646
|
+
/**
|
|
647
|
+
* Check if a summary already exists as a bullet line in topic file content.
|
|
648
|
+
* Uses exact bullet prefix matching (not substring) to avoid false positives.
|
|
649
|
+
*/
|
|
650
|
+
export function hasSummaryLine(content, summary) {
|
|
651
|
+
// Match lines that start with "- <summary>" (possibly followed by metadata)
|
|
652
|
+
return content.split('\n').some(line => line.startsWith(`- ${summary}`));
|
|
653
|
+
}
|
|
654
|
+
/**
|
|
655
|
+
* Prune sections to fit within a line budget.
|
|
656
|
+
* Removes entries from the largest sections first.
|
|
657
|
+
*/
|
|
658
|
+
function pruneSectionsToFit(sections, budget, strategy) {
|
|
659
|
+
// Pre-compute total line count: title(1) + blank(1) + per-section(heading + items + "See..." + blank)
|
|
660
|
+
let totalLines = 2;
|
|
661
|
+
for (const summaries of Object.values(sections)) {
|
|
662
|
+
totalLines += 1 + summaries.length + 1 + 1;
|
|
663
|
+
}
|
|
664
|
+
while (totalLines > budget) {
|
|
665
|
+
const sorted = Object.entries(sections)
|
|
666
|
+
.filter(([, items]) => items.length > 1)
|
|
667
|
+
.sort((a, b) => b[1].length - a[1].length);
|
|
668
|
+
if (sorted.length === 0)
|
|
669
|
+
break;
|
|
670
|
+
const [targetCat, targetItems] = sorted[0];
|
|
671
|
+
if (strategy === 'lru' || strategy === 'fifo') {
|
|
672
|
+
targetItems.shift();
|
|
673
|
+
}
|
|
674
|
+
else {
|
|
675
|
+
targetItems.pop();
|
|
676
|
+
}
|
|
677
|
+
totalLines--; // one fewer bullet line
|
|
678
|
+
if (targetItems.length === 0) {
|
|
679
|
+
delete sections[targetCat];
|
|
680
|
+
totalLines -= 3; // heading + "See..." + blank removed
|
|
681
|
+
}
|
|
682
|
+
}
|
|
683
|
+
}
|
|
684
|
+
/**
|
|
685
|
+
* Build MEMORY.md index lines from curated sections.
|
|
686
|
+
*/
|
|
687
|
+
function buildIndexLines(sections, topicMapping, sectionOrder) {
|
|
688
|
+
const lines = ['# Claude Flow V3 Project Memory', ''];
|
|
689
|
+
// Use provided order, then append any remaining sections
|
|
690
|
+
const orderedCategories = sectionOrder
|
|
691
|
+
? [...sectionOrder, ...Object.keys(sections).filter((k) => !sectionOrder.includes(k))]
|
|
692
|
+
: Object.keys(sections);
|
|
693
|
+
for (const category of orderedCategories) {
|
|
694
|
+
const summaries = sections[category];
|
|
695
|
+
if (!summaries || summaries.length === 0)
|
|
696
|
+
continue;
|
|
697
|
+
const label = CATEGORY_LABELS[category] || category;
|
|
698
|
+
const filename = topicMapping[category] || `${category}.md`;
|
|
699
|
+
lines.push(`## ${label}`);
|
|
700
|
+
for (const summary of summaries) {
|
|
701
|
+
lines.push(`- ${summary}`);
|
|
702
|
+
}
|
|
703
|
+
lines.push(`- See \`${filename}\` for details`);
|
|
704
|
+
lines.push('');
|
|
705
|
+
}
|
|
706
|
+
return lines;
|
|
707
|
+
}
|
|
708
|
+
export default AutoMemoryBridge;
|
|
709
|
+
//# sourceMappingURL=auto-memory-bridge.js.map
|