@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.
Files changed (156) hide show
  1. package/README.md +356 -18
  2. package/dist/agent-memory-scope.d.ts +131 -0
  3. package/dist/agent-memory-scope.d.ts.map +1 -0
  4. package/dist/agent-memory-scope.js +215 -0
  5. package/dist/agent-memory-scope.js.map +1 -0
  6. package/dist/agent-memory-scope.test.d.ts +8 -0
  7. package/dist/agent-memory-scope.test.d.ts.map +1 -0
  8. package/dist/agent-memory-scope.test.js +463 -0
  9. package/dist/agent-memory-scope.test.js.map +1 -0
  10. package/dist/agentdb-adapter.d.ts +22 -3
  11. package/dist/agentdb-adapter.d.ts.map +1 -1
  12. package/dist/agentdb-adapter.js +135 -8
  13. package/dist/agentdb-adapter.js.map +1 -1
  14. package/dist/auto-memory-bridge.d.ts +226 -0
  15. package/dist/auto-memory-bridge.d.ts.map +1 -0
  16. package/dist/auto-memory-bridge.js +709 -0
  17. package/dist/auto-memory-bridge.js.map +1 -0
  18. package/dist/auto-memory-bridge.test.d.ts +8 -0
  19. package/dist/auto-memory-bridge.test.d.ts.map +1 -0
  20. package/dist/auto-memory-bridge.test.js +754 -0
  21. package/dist/auto-memory-bridge.test.js.map +1 -0
  22. package/dist/benchmark.test.d.ts +2 -0
  23. package/dist/benchmark.test.d.ts.map +1 -0
  24. package/dist/benchmark.test.js +277 -0
  25. package/dist/benchmark.test.js.map +1 -0
  26. package/dist/index.d.ts +8 -0
  27. package/dist/index.d.ts.map +1 -1
  28. package/dist/index.js +8 -0
  29. package/dist/index.js.map +1 -1
  30. package/dist/learning-bridge.d.ts +137 -0
  31. package/dist/learning-bridge.d.ts.map +1 -0
  32. package/dist/learning-bridge.js +335 -0
  33. package/dist/learning-bridge.js.map +1 -0
  34. package/dist/learning-bridge.test.d.ts +8 -0
  35. package/dist/learning-bridge.test.d.ts.map +1 -0
  36. package/dist/learning-bridge.test.js +578 -0
  37. package/dist/learning-bridge.test.js.map +1 -0
  38. package/dist/memory-graph.d.ts +100 -0
  39. package/dist/memory-graph.d.ts.map +1 -0
  40. package/dist/memory-graph.js +333 -0
  41. package/dist/memory-graph.js.map +1 -0
  42. package/dist/memory-graph.test.d.ts +8 -0
  43. package/dist/memory-graph.test.d.ts.map +1 -0
  44. package/dist/memory-graph.test.js +609 -0
  45. package/dist/memory-graph.test.js.map +1 -0
  46. package/dist/types.d.ts +3 -0
  47. package/dist/types.d.ts.map +1 -1
  48. package/package.json +19 -4
  49. package/.agentic-flow/intelligence.json +0 -16
  50. package/__tests__/coverage/base.css +0 -224
  51. package/__tests__/coverage/block-navigation.js +0 -87
  52. package/__tests__/coverage/coverage-final.json +0 -19
  53. package/__tests__/coverage/favicon.png +0 -0
  54. package/__tests__/coverage/index.html +0 -206
  55. package/__tests__/coverage/lcov-report/base.css +0 -224
  56. package/__tests__/coverage/lcov-report/block-navigation.js +0 -87
  57. package/__tests__/coverage/lcov-report/favicon.png +0 -0
  58. package/__tests__/coverage/lcov-report/index.html +0 -206
  59. package/__tests__/coverage/lcov-report/prettify.css +0 -1
  60. package/__tests__/coverage/lcov-report/prettify.js +0 -2
  61. package/__tests__/coverage/lcov-report/sort-arrow-sprite.png +0 -0
  62. package/__tests__/coverage/lcov-report/sorter.js +0 -210
  63. package/__tests__/coverage/lcov-report/src/agentdb-adapter.ts.html +0 -2737
  64. package/__tests__/coverage/lcov-report/src/agentdb-backend.ts.html +0 -3130
  65. package/__tests__/coverage/lcov-report/src/application/commands/delete-memory.command.ts.html +0 -601
  66. package/__tests__/coverage/lcov-report/src/application/commands/index.html +0 -131
  67. package/__tests__/coverage/lcov-report/src/application/commands/store-memory.command.ts.html +0 -394
  68. package/__tests__/coverage/lcov-report/src/application/queries/index.html +0 -116
  69. package/__tests__/coverage/lcov-report/src/application/queries/search-memory.query.ts.html +0 -796
  70. package/__tests__/coverage/lcov-report/src/application/services/index.html +0 -116
  71. package/__tests__/coverage/lcov-report/src/application/services/memory-application-service.ts.html +0 -793
  72. package/__tests__/coverage/lcov-report/src/cache-manager.ts.html +0 -1633
  73. package/__tests__/coverage/lcov-report/src/database-provider.ts.html +0 -1618
  74. package/__tests__/coverage/lcov-report/src/domain/entities/index.html +0 -116
  75. package/__tests__/coverage/lcov-report/src/domain/entities/memory-entry.ts.html +0 -952
  76. package/__tests__/coverage/lcov-report/src/domain/services/index.html +0 -116
  77. package/__tests__/coverage/lcov-report/src/domain/services/memory-domain-service.ts.html +0 -1294
  78. package/__tests__/coverage/lcov-report/src/hnsw-index.ts.html +0 -3124
  79. package/__tests__/coverage/lcov-report/src/hybrid-backend.ts.html +0 -2167
  80. package/__tests__/coverage/lcov-report/src/index.html +0 -266
  81. package/__tests__/coverage/lcov-report/src/infrastructure/repositories/hybrid-memory-repository.ts.html +0 -1633
  82. package/__tests__/coverage/lcov-report/src/infrastructure/repositories/index.html +0 -116
  83. package/__tests__/coverage/lcov-report/src/migration.ts.html +0 -2092
  84. package/__tests__/coverage/lcov-report/src/query-builder.ts.html +0 -1711
  85. package/__tests__/coverage/lcov-report/src/sqlite-backend.ts.html +0 -2281
  86. package/__tests__/coverage/lcov-report/src/sqljs-backend.ts.html +0 -2374
  87. package/__tests__/coverage/lcov-report/src/types.ts.html +0 -2266
  88. package/__tests__/coverage/lcov.info +0 -10238
  89. package/__tests__/coverage/prettify.css +0 -1
  90. package/__tests__/coverage/prettify.js +0 -2
  91. package/__tests__/coverage/sort-arrow-sprite.png +0 -0
  92. package/__tests__/coverage/sorter.js +0 -210
  93. package/__tests__/coverage/src/agentdb-adapter.ts.html +0 -2737
  94. package/__tests__/coverage/src/agentdb-backend.ts.html +0 -3130
  95. package/__tests__/coverage/src/application/commands/delete-memory.command.ts.html +0 -601
  96. package/__tests__/coverage/src/application/commands/index.html +0 -131
  97. package/__tests__/coverage/src/application/commands/store-memory.command.ts.html +0 -394
  98. package/__tests__/coverage/src/application/queries/index.html +0 -116
  99. package/__tests__/coverage/src/application/queries/search-memory.query.ts.html +0 -796
  100. package/__tests__/coverage/src/application/services/index.html +0 -116
  101. package/__tests__/coverage/src/application/services/memory-application-service.ts.html +0 -793
  102. package/__tests__/coverage/src/cache-manager.ts.html +0 -1633
  103. package/__tests__/coverage/src/database-provider.ts.html +0 -1618
  104. package/__tests__/coverage/src/domain/entities/index.html +0 -116
  105. package/__tests__/coverage/src/domain/entities/memory-entry.ts.html +0 -952
  106. package/__tests__/coverage/src/domain/services/index.html +0 -116
  107. package/__tests__/coverage/src/domain/services/memory-domain-service.ts.html +0 -1294
  108. package/__tests__/coverage/src/hnsw-index.ts.html +0 -3124
  109. package/__tests__/coverage/src/hybrid-backend.ts.html +0 -2167
  110. package/__tests__/coverage/src/index.html +0 -266
  111. package/__tests__/coverage/src/infrastructure/repositories/hybrid-memory-repository.ts.html +0 -1633
  112. package/__tests__/coverage/src/infrastructure/repositories/index.html +0 -116
  113. package/__tests__/coverage/src/migration.ts.html +0 -2092
  114. package/__tests__/coverage/src/query-builder.ts.html +0 -1711
  115. package/__tests__/coverage/src/sqlite-backend.ts.html +0 -2281
  116. package/__tests__/coverage/src/sqljs-backend.ts.html +0 -2374
  117. package/__tests__/coverage/src/types.ts.html +0 -2266
  118. package/benchmarks/cache-hit-rate.bench.ts +0 -535
  119. package/benchmarks/hnsw-indexing.bench.ts +0 -552
  120. package/benchmarks/memory-write.bench.ts +0 -469
  121. package/benchmarks/vector-search.bench.ts +0 -449
  122. package/docs/AGENTDB-INTEGRATION.md +0 -388
  123. package/docs/CROSS_PLATFORM.md +0 -505
  124. package/docs/WINDOWS_SUPPORT.md +0 -422
  125. package/examples/agentdb-example.ts +0 -345
  126. package/examples/cross-platform-usage.ts +0 -326
  127. package/framework/benchmark.ts +0 -112
  128. package/src/agentdb-adapter.ts +0 -884
  129. package/src/agentdb-backend.test.ts +0 -339
  130. package/src/agentdb-backend.ts +0 -1016
  131. package/src/application/commands/delete-memory.command.ts +0 -172
  132. package/src/application/commands/store-memory.command.ts +0 -103
  133. package/src/application/index.ts +0 -36
  134. package/src/application/queries/search-memory.query.ts +0 -237
  135. package/src/application/services/memory-application-service.ts +0 -236
  136. package/src/cache-manager.ts +0 -516
  137. package/src/database-provider.test.ts +0 -364
  138. package/src/database-provider.ts +0 -511
  139. package/src/domain/entities/memory-entry.ts +0 -289
  140. package/src/domain/index.ts +0 -35
  141. package/src/domain/repositories/memory-repository.interface.ts +0 -120
  142. package/src/domain/services/memory-domain-service.ts +0 -403
  143. package/src/hnsw-index.ts +0 -1013
  144. package/src/hybrid-backend.test.ts +0 -399
  145. package/src/hybrid-backend.ts +0 -694
  146. package/src/index.ts +0 -515
  147. package/src/infrastructure/index.ts +0 -23
  148. package/src/infrastructure/repositories/hybrid-memory-repository.ts +0 -516
  149. package/src/migration.ts +0 -669
  150. package/src/query-builder.ts +0 -542
  151. package/src/sqlite-backend.ts +0 -732
  152. package/src/sqljs-backend.ts +0 -763
  153. package/src/types.ts +0 -727
  154. package/tsconfig.json +0 -9
  155. package/tsconfig.tsbuildinfo +0 -1
  156. 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