@claude-flow/memory 3.0.0-alpha.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.
Files changed (214) hide show
  1. package/.agentic-flow/intelligence.json +16 -0
  2. package/README.md +249 -0
  3. package/__tests__/coverage/base.css +224 -0
  4. package/__tests__/coverage/block-navigation.js +87 -0
  5. package/__tests__/coverage/coverage-final.json +19 -0
  6. package/__tests__/coverage/favicon.png +0 -0
  7. package/__tests__/coverage/index.html +206 -0
  8. package/__tests__/coverage/lcov-report/base.css +224 -0
  9. package/__tests__/coverage/lcov-report/block-navigation.js +87 -0
  10. package/__tests__/coverage/lcov-report/favicon.png +0 -0
  11. package/__tests__/coverage/lcov-report/index.html +206 -0
  12. package/__tests__/coverage/lcov-report/prettify.css +1 -0
  13. package/__tests__/coverage/lcov-report/prettify.js +2 -0
  14. package/__tests__/coverage/lcov-report/sort-arrow-sprite.png +0 -0
  15. package/__tests__/coverage/lcov-report/sorter.js +210 -0
  16. package/__tests__/coverage/lcov-report/src/agentdb-adapter.ts.html +2737 -0
  17. package/__tests__/coverage/lcov-report/src/agentdb-backend.ts.html +3130 -0
  18. package/__tests__/coverage/lcov-report/src/application/commands/delete-memory.command.ts.html +601 -0
  19. package/__tests__/coverage/lcov-report/src/application/commands/index.html +131 -0
  20. package/__tests__/coverage/lcov-report/src/application/commands/store-memory.command.ts.html +394 -0
  21. package/__tests__/coverage/lcov-report/src/application/queries/index.html +116 -0
  22. package/__tests__/coverage/lcov-report/src/application/queries/search-memory.query.ts.html +796 -0
  23. package/__tests__/coverage/lcov-report/src/application/services/index.html +116 -0
  24. package/__tests__/coverage/lcov-report/src/application/services/memory-application-service.ts.html +793 -0
  25. package/__tests__/coverage/lcov-report/src/cache-manager.ts.html +1633 -0
  26. package/__tests__/coverage/lcov-report/src/database-provider.ts.html +1618 -0
  27. package/__tests__/coverage/lcov-report/src/domain/entities/index.html +116 -0
  28. package/__tests__/coverage/lcov-report/src/domain/entities/memory-entry.ts.html +952 -0
  29. package/__tests__/coverage/lcov-report/src/domain/services/index.html +116 -0
  30. package/__tests__/coverage/lcov-report/src/domain/services/memory-domain-service.ts.html +1294 -0
  31. package/__tests__/coverage/lcov-report/src/hnsw-index.ts.html +3124 -0
  32. package/__tests__/coverage/lcov-report/src/hybrid-backend.ts.html +2167 -0
  33. package/__tests__/coverage/lcov-report/src/index.html +266 -0
  34. package/__tests__/coverage/lcov-report/src/infrastructure/repositories/hybrid-memory-repository.ts.html +1633 -0
  35. package/__tests__/coverage/lcov-report/src/infrastructure/repositories/index.html +116 -0
  36. package/__tests__/coverage/lcov-report/src/migration.ts.html +2092 -0
  37. package/__tests__/coverage/lcov-report/src/query-builder.ts.html +1711 -0
  38. package/__tests__/coverage/lcov-report/src/sqlite-backend.ts.html +2281 -0
  39. package/__tests__/coverage/lcov-report/src/sqljs-backend.ts.html +2374 -0
  40. package/__tests__/coverage/lcov-report/src/types.ts.html +2266 -0
  41. package/__tests__/coverage/lcov.info +10238 -0
  42. package/__tests__/coverage/prettify.css +1 -0
  43. package/__tests__/coverage/prettify.js +2 -0
  44. package/__tests__/coverage/sort-arrow-sprite.png +0 -0
  45. package/__tests__/coverage/sorter.js +210 -0
  46. package/__tests__/coverage/src/agentdb-adapter.ts.html +2737 -0
  47. package/__tests__/coverage/src/agentdb-backend.ts.html +3130 -0
  48. package/__tests__/coverage/src/application/commands/delete-memory.command.ts.html +601 -0
  49. package/__tests__/coverage/src/application/commands/index.html +131 -0
  50. package/__tests__/coverage/src/application/commands/store-memory.command.ts.html +394 -0
  51. package/__tests__/coverage/src/application/queries/index.html +116 -0
  52. package/__tests__/coverage/src/application/queries/search-memory.query.ts.html +796 -0
  53. package/__tests__/coverage/src/application/services/index.html +116 -0
  54. package/__tests__/coverage/src/application/services/memory-application-service.ts.html +793 -0
  55. package/__tests__/coverage/src/cache-manager.ts.html +1633 -0
  56. package/__tests__/coverage/src/database-provider.ts.html +1618 -0
  57. package/__tests__/coverage/src/domain/entities/index.html +116 -0
  58. package/__tests__/coverage/src/domain/entities/memory-entry.ts.html +952 -0
  59. package/__tests__/coverage/src/domain/services/index.html +116 -0
  60. package/__tests__/coverage/src/domain/services/memory-domain-service.ts.html +1294 -0
  61. package/__tests__/coverage/src/hnsw-index.ts.html +3124 -0
  62. package/__tests__/coverage/src/hybrid-backend.ts.html +2167 -0
  63. package/__tests__/coverage/src/index.html +266 -0
  64. package/__tests__/coverage/src/infrastructure/repositories/hybrid-memory-repository.ts.html +1633 -0
  65. package/__tests__/coverage/src/infrastructure/repositories/index.html +116 -0
  66. package/__tests__/coverage/src/migration.ts.html +2092 -0
  67. package/__tests__/coverage/src/query-builder.ts.html +1711 -0
  68. package/__tests__/coverage/src/sqlite-backend.ts.html +2281 -0
  69. package/__tests__/coverage/src/sqljs-backend.ts.html +2374 -0
  70. package/__tests__/coverage/src/types.ts.html +2266 -0
  71. package/benchmarks/cache-hit-rate.bench.ts +535 -0
  72. package/benchmarks/hnsw-indexing.bench.ts +552 -0
  73. package/benchmarks/memory-write.bench.ts +469 -0
  74. package/benchmarks/vector-search.bench.ts +449 -0
  75. package/dist/agentdb-adapter.d.ts +146 -0
  76. package/dist/agentdb-adapter.d.ts.map +1 -0
  77. package/dist/agentdb-adapter.js +679 -0
  78. package/dist/agentdb-adapter.js.map +1 -0
  79. package/dist/agentdb-backend.d.ts +214 -0
  80. package/dist/agentdb-backend.d.ts.map +1 -0
  81. package/dist/agentdb-backend.js +827 -0
  82. package/dist/agentdb-backend.js.map +1 -0
  83. package/dist/agentdb-backend.test.d.ts +7 -0
  84. package/dist/agentdb-backend.test.d.ts.map +1 -0
  85. package/dist/agentdb-backend.test.js +258 -0
  86. package/dist/agentdb-backend.test.js.map +1 -0
  87. package/dist/application/commands/delete-memory.command.d.ts +65 -0
  88. package/dist/application/commands/delete-memory.command.d.ts.map +1 -0
  89. package/dist/application/commands/delete-memory.command.js +129 -0
  90. package/dist/application/commands/delete-memory.command.js.map +1 -0
  91. package/dist/application/commands/store-memory.command.d.ts +48 -0
  92. package/dist/application/commands/store-memory.command.d.ts.map +1 -0
  93. package/dist/application/commands/store-memory.command.js +72 -0
  94. package/dist/application/commands/store-memory.command.js.map +1 -0
  95. package/dist/application/index.d.ts +12 -0
  96. package/dist/application/index.d.ts.map +1 -0
  97. package/dist/application/index.js +15 -0
  98. package/dist/application/index.js.map +1 -0
  99. package/dist/application/queries/search-memory.query.d.ts +72 -0
  100. package/dist/application/queries/search-memory.query.d.ts.map +1 -0
  101. package/dist/application/queries/search-memory.query.js +143 -0
  102. package/dist/application/queries/search-memory.query.js.map +1 -0
  103. package/dist/application/services/memory-application-service.d.ts +121 -0
  104. package/dist/application/services/memory-application-service.d.ts.map +1 -0
  105. package/dist/application/services/memory-application-service.js +190 -0
  106. package/dist/application/services/memory-application-service.js.map +1 -0
  107. package/dist/cache-manager.d.ts +134 -0
  108. package/dist/cache-manager.d.ts.map +1 -0
  109. package/dist/cache-manager.js +407 -0
  110. package/dist/cache-manager.js.map +1 -0
  111. package/dist/database-provider.d.ts +86 -0
  112. package/dist/database-provider.d.ts.map +1 -0
  113. package/dist/database-provider.js +385 -0
  114. package/dist/database-provider.js.map +1 -0
  115. package/dist/database-provider.test.d.ts +7 -0
  116. package/dist/database-provider.test.d.ts.map +1 -0
  117. package/dist/database-provider.test.js +285 -0
  118. package/dist/database-provider.test.js.map +1 -0
  119. package/dist/domain/entities/memory-entry.d.ts +143 -0
  120. package/dist/domain/entities/memory-entry.d.ts.map +1 -0
  121. package/dist/domain/entities/memory-entry.js +226 -0
  122. package/dist/domain/entities/memory-entry.js.map +1 -0
  123. package/dist/domain/index.d.ts +11 -0
  124. package/dist/domain/index.d.ts.map +1 -0
  125. package/dist/domain/index.js +12 -0
  126. package/dist/domain/index.js.map +1 -0
  127. package/dist/domain/repositories/memory-repository.interface.d.ts +102 -0
  128. package/dist/domain/repositories/memory-repository.interface.d.ts.map +1 -0
  129. package/dist/domain/repositories/memory-repository.interface.js +11 -0
  130. package/dist/domain/repositories/memory-repository.interface.js.map +1 -0
  131. package/dist/domain/services/memory-domain-service.d.ts +105 -0
  132. package/dist/domain/services/memory-domain-service.d.ts.map +1 -0
  133. package/dist/domain/services/memory-domain-service.js +297 -0
  134. package/dist/domain/services/memory-domain-service.js.map +1 -0
  135. package/dist/hnsw-index.d.ts +111 -0
  136. package/dist/hnsw-index.d.ts.map +1 -0
  137. package/dist/hnsw-index.js +781 -0
  138. package/dist/hnsw-index.js.map +1 -0
  139. package/dist/hybrid-backend.d.ts +217 -0
  140. package/dist/hybrid-backend.d.ts.map +1 -0
  141. package/dist/hybrid-backend.js +491 -0
  142. package/dist/hybrid-backend.js.map +1 -0
  143. package/dist/hybrid-backend.test.d.ts +8 -0
  144. package/dist/hybrid-backend.test.d.ts.map +1 -0
  145. package/dist/hybrid-backend.test.js +320 -0
  146. package/dist/hybrid-backend.test.js.map +1 -0
  147. package/dist/index.d.ts +188 -0
  148. package/dist/index.d.ts.map +1 -0
  149. package/dist/index.js +345 -0
  150. package/dist/index.js.map +1 -0
  151. package/dist/infrastructure/index.d.ts +17 -0
  152. package/dist/infrastructure/index.d.ts.map +1 -0
  153. package/dist/infrastructure/index.js +16 -0
  154. package/dist/infrastructure/index.js.map +1 -0
  155. package/dist/infrastructure/repositories/hybrid-memory-repository.d.ts +66 -0
  156. package/dist/infrastructure/repositories/hybrid-memory-repository.d.ts.map +1 -0
  157. package/dist/infrastructure/repositories/hybrid-memory-repository.js +409 -0
  158. package/dist/infrastructure/repositories/hybrid-memory-repository.js.map +1 -0
  159. package/dist/migration.d.ts +68 -0
  160. package/dist/migration.d.ts.map +1 -0
  161. package/dist/migration.js +513 -0
  162. package/dist/migration.js.map +1 -0
  163. package/dist/query-builder.d.ts +211 -0
  164. package/dist/query-builder.d.ts.map +1 -0
  165. package/dist/query-builder.js +438 -0
  166. package/dist/query-builder.js.map +1 -0
  167. package/dist/sqlite-backend.d.ts +121 -0
  168. package/dist/sqlite-backend.d.ts.map +1 -0
  169. package/dist/sqlite-backend.js +564 -0
  170. package/dist/sqlite-backend.js.map +1 -0
  171. package/dist/sqljs-backend.d.ts +128 -0
  172. package/dist/sqljs-backend.d.ts.map +1 -0
  173. package/dist/sqljs-backend.js +598 -0
  174. package/dist/sqljs-backend.js.map +1 -0
  175. package/dist/types.d.ts +481 -0
  176. package/dist/types.d.ts.map +1 -0
  177. package/dist/types.js +58 -0
  178. package/dist/types.js.map +1 -0
  179. package/docs/AGENTDB-INTEGRATION.md +388 -0
  180. package/docs/CROSS_PLATFORM.md +505 -0
  181. package/docs/WINDOWS_SUPPORT.md +422 -0
  182. package/examples/agentdb-example.ts +345 -0
  183. package/examples/cross-platform-usage.ts +326 -0
  184. package/framework/benchmark.ts +112 -0
  185. package/package.json +31 -0
  186. package/src/agentdb-adapter.ts +884 -0
  187. package/src/agentdb-backend.test.ts +339 -0
  188. package/src/agentdb-backend.ts +1016 -0
  189. package/src/application/commands/delete-memory.command.ts +172 -0
  190. package/src/application/commands/store-memory.command.ts +103 -0
  191. package/src/application/index.ts +36 -0
  192. package/src/application/queries/search-memory.query.ts +237 -0
  193. package/src/application/services/memory-application-service.ts +236 -0
  194. package/src/cache-manager.ts +516 -0
  195. package/src/database-provider.test.ts +364 -0
  196. package/src/database-provider.ts +511 -0
  197. package/src/domain/entities/memory-entry.ts +289 -0
  198. package/src/domain/index.ts +35 -0
  199. package/src/domain/repositories/memory-repository.interface.ts +120 -0
  200. package/src/domain/services/memory-domain-service.ts +403 -0
  201. package/src/hnsw-index.ts +1013 -0
  202. package/src/hybrid-backend.test.ts +399 -0
  203. package/src/hybrid-backend.ts +694 -0
  204. package/src/index.ts +515 -0
  205. package/src/infrastructure/index.ts +23 -0
  206. package/src/infrastructure/repositories/hybrid-memory-repository.ts +516 -0
  207. package/src/migration.ts +669 -0
  208. package/src/query-builder.ts +542 -0
  209. package/src/sqlite-backend.ts +732 -0
  210. package/src/sqljs-backend.ts +763 -0
  211. package/src/types.ts +727 -0
  212. package/tsconfig.json +9 -0
  213. package/tsconfig.tsbuildinfo +1 -0
  214. package/verify-cross-platform.ts +170 -0
@@ -0,0 +1,827 @@
1
+ /**
2
+ * AgentDB Backend - Integration with agentdb@2.0.0-alpha.3.4
3
+ *
4
+ * Provides IMemoryBackend implementation using AgentDB with:
5
+ * - HNSW vector search (150x-12,500x faster than brute-force)
6
+ * - Native or WASM backend support with graceful fallback
7
+ * - Optional dependency handling (works without hnswlib-node)
8
+ * - Seamless integration with HybridBackend
9
+ *
10
+ * @module v3/memory/agentdb-backend
11
+ */
12
+ import { EventEmitter } from 'node:events';
13
+ // ===== AgentDB Optional Import =====
14
+ let AgentDB;
15
+ let HNSWIndex;
16
+ let isHnswlibAvailable;
17
+ // Dynamically import agentdb (handled at runtime)
18
+ let agentdbImportPromise;
19
+ function ensureAgentDBImport() {
20
+ if (!agentdbImportPromise) {
21
+ agentdbImportPromise = (async () => {
22
+ try {
23
+ const agentdbModule = await import('agentdb');
24
+ AgentDB = agentdbModule.AgentDB || agentdbModule.default;
25
+ HNSWIndex = agentdbModule.HNSWIndex;
26
+ isHnswlibAvailable = agentdbModule.isHnswlibAvailable;
27
+ }
28
+ catch (error) {
29
+ // AgentDB not available - will use fallback
30
+ }
31
+ })();
32
+ }
33
+ return agentdbImportPromise;
34
+ }
35
+ /**
36
+ * Default configuration
37
+ */
38
+ const DEFAULT_CONFIG = {
39
+ namespace: 'default',
40
+ forceWasm: false,
41
+ vectorBackend: 'auto',
42
+ vectorDimension: 1536,
43
+ hnswM: 16,
44
+ hnswEfConstruction: 200,
45
+ hnswEfSearch: 100,
46
+ cacheEnabled: true,
47
+ maxEntries: 1000000,
48
+ };
49
+ // ===== AgentDB Backend Implementation =====
50
+ /**
51
+ * AgentDB Backend
52
+ *
53
+ * Integrates AgentDB for vector search with the V3 memory system.
54
+ * Provides 150x-12,500x faster search compared to brute-force approaches.
55
+ *
56
+ * Features:
57
+ * - HNSW indexing for fast approximate nearest neighbor search
58
+ * - Automatic fallback: native hnswlib → ruvector → WASM
59
+ * - Graceful handling of optional native dependencies
60
+ * - Semantic search with filtering
61
+ * - Compatible with HybridBackend for combined SQLite+AgentDB queries
62
+ */
63
+ export class AgentDBBackend extends EventEmitter {
64
+ config;
65
+ agentdb;
66
+ initialized = false;
67
+ available = false;
68
+ // In-memory storage for compatibility
69
+ entries = new Map();
70
+ namespaceIndex = new Map();
71
+ keyIndex = new Map();
72
+ // O(1) reverse lookup for numeric ID -> string ID (fixes O(n) linear scan)
73
+ numericToStringIdMap = new Map();
74
+ // Performance tracking
75
+ stats = {
76
+ queryCount: 0,
77
+ totalQueryTime: 0,
78
+ searchCount: 0,
79
+ totalSearchTime: 0,
80
+ };
81
+ constructor(config = {}) {
82
+ super();
83
+ this.config = { ...DEFAULT_CONFIG, ...config };
84
+ this.available = false; // Will be set during initialization
85
+ }
86
+ /**
87
+ * Initialize AgentDB
88
+ */
89
+ async initialize() {
90
+ if (this.initialized)
91
+ return;
92
+ // Try to import AgentDB
93
+ await ensureAgentDBImport();
94
+ this.available = AgentDB !== undefined;
95
+ if (!this.available) {
96
+ console.warn('AgentDB not available, using fallback in-memory storage');
97
+ this.initialized = true;
98
+ return;
99
+ }
100
+ try {
101
+ // Initialize AgentDB with config
102
+ this.agentdb = new AgentDB({
103
+ dbPath: this.config.dbPath || ':memory:',
104
+ namespace: this.config.namespace,
105
+ forceWasm: this.config.forceWasm,
106
+ vectorBackend: this.config.vectorBackend,
107
+ vectorDimension: this.config.vectorDimension,
108
+ });
109
+ await this.agentdb.initialize();
110
+ // Create memory_entries table if it doesn't exist
111
+ await this.createSchema();
112
+ this.initialized = true;
113
+ this.emit('initialized', {
114
+ backend: this.agentdb.vectorBackendName,
115
+ isWasm: this.agentdb.isWasm,
116
+ });
117
+ }
118
+ catch (error) {
119
+ console.error('Failed to initialize AgentDB:', error);
120
+ this.available = false;
121
+ this.initialized = true;
122
+ this.emit('initialization:failed', { error });
123
+ }
124
+ }
125
+ /**
126
+ * Shutdown AgentDB
127
+ */
128
+ async shutdown() {
129
+ if (!this.initialized)
130
+ return;
131
+ if (this.agentdb) {
132
+ await this.agentdb.close();
133
+ }
134
+ this.initialized = false;
135
+ this.emit('shutdown');
136
+ }
137
+ /**
138
+ * Store a memory entry
139
+ */
140
+ async store(entry) {
141
+ // Generate embedding if needed
142
+ if (entry.content && !entry.embedding && this.config.embeddingGenerator) {
143
+ entry.embedding = await this.config.embeddingGenerator(entry.content);
144
+ }
145
+ // Store in-memory for quick access
146
+ this.entries.set(entry.id, entry);
147
+ // Register ID mapping for O(1) reverse lookup
148
+ this.registerIdMapping(entry.id);
149
+ // Update indexes
150
+ this.updateIndexes(entry);
151
+ // Store in AgentDB if available
152
+ if (this.agentdb) {
153
+ await this.storeInAgentDB(entry);
154
+ }
155
+ this.emit('entry:stored', { id: entry.id });
156
+ }
157
+ /**
158
+ * Get entry by ID
159
+ */
160
+ async get(id) {
161
+ // Check in-memory first
162
+ const cached = this.entries.get(id);
163
+ if (cached)
164
+ return cached;
165
+ // Query AgentDB if available
166
+ if (this.agentdb) {
167
+ return this.getFromAgentDB(id);
168
+ }
169
+ return null;
170
+ }
171
+ /**
172
+ * Get entry by key
173
+ */
174
+ async getByKey(namespace, key) {
175
+ const keyIndexKey = `${namespace}:${key}`;
176
+ const id = this.keyIndex.get(keyIndexKey);
177
+ if (!id)
178
+ return null;
179
+ return this.get(id);
180
+ }
181
+ /**
182
+ * Update entry
183
+ */
184
+ async update(id, update) {
185
+ const entry = this.entries.get(id);
186
+ if (!entry)
187
+ return null;
188
+ // Apply updates
189
+ if (update.content !== undefined) {
190
+ entry.content = update.content;
191
+ // Regenerate embedding if needed
192
+ if (this.config.embeddingGenerator) {
193
+ entry.embedding = await this.config.embeddingGenerator(entry.content);
194
+ }
195
+ }
196
+ if (update.tags !== undefined) {
197
+ entry.tags = update.tags;
198
+ }
199
+ if (update.metadata !== undefined) {
200
+ entry.metadata = { ...entry.metadata, ...update.metadata };
201
+ }
202
+ if (update.accessLevel !== undefined) {
203
+ entry.accessLevel = update.accessLevel;
204
+ }
205
+ if (update.expiresAt !== undefined) {
206
+ entry.expiresAt = update.expiresAt;
207
+ }
208
+ if (update.references !== undefined) {
209
+ entry.references = update.references;
210
+ }
211
+ entry.updatedAt = Date.now();
212
+ entry.version++;
213
+ // Update in AgentDB
214
+ if (this.agentdb) {
215
+ await this.updateInAgentDB(entry);
216
+ }
217
+ this.emit('entry:updated', { id });
218
+ return entry;
219
+ }
220
+ /**
221
+ * Delete entry
222
+ */
223
+ async delete(id) {
224
+ const entry = this.entries.get(id);
225
+ if (!entry)
226
+ return false;
227
+ // Remove from indexes
228
+ this.entries.delete(id);
229
+ this.unregisterIdMapping(id); // Clean up reverse lookup map
230
+ this.namespaceIndex.get(entry.namespace)?.delete(id);
231
+ const keyIndexKey = `${entry.namespace}:${entry.key}`;
232
+ this.keyIndex.delete(keyIndexKey);
233
+ // Delete from AgentDB
234
+ if (this.agentdb) {
235
+ await this.deleteFromAgentDB(id);
236
+ }
237
+ this.emit('entry:deleted', { id });
238
+ return true;
239
+ }
240
+ /**
241
+ * Query entries
242
+ */
243
+ async query(query) {
244
+ const startTime = performance.now();
245
+ let results = [];
246
+ if (query.type === 'semantic' && (query.embedding || query.content)) {
247
+ // Use semantic search
248
+ const searchResults = await this.semanticSearch(query);
249
+ results = searchResults.map((r) => r.entry);
250
+ }
251
+ else {
252
+ // Fallback to in-memory filtering
253
+ results = this.queryInMemory(query);
254
+ }
255
+ const duration = performance.now() - startTime;
256
+ this.stats.queryCount++;
257
+ this.stats.totalQueryTime += duration;
258
+ return results;
259
+ }
260
+ /**
261
+ * Semantic vector search
262
+ */
263
+ async search(embedding, options) {
264
+ const startTime = performance.now();
265
+ if (!this.agentdb) {
266
+ // Fallback to brute-force search
267
+ return this.bruteForceSearch(embedding, options);
268
+ }
269
+ try {
270
+ // Use AgentDB HNSW search
271
+ const results = await this.searchWithAgentDB(embedding, options);
272
+ const duration = performance.now() - startTime;
273
+ this.stats.searchCount++;
274
+ this.stats.totalSearchTime += duration;
275
+ return results;
276
+ }
277
+ catch (error) {
278
+ console.error('AgentDB search failed, falling back to brute-force:', error);
279
+ return this.bruteForceSearch(embedding, options);
280
+ }
281
+ }
282
+ /**
283
+ * Bulk insert
284
+ */
285
+ async bulkInsert(entries) {
286
+ for (const entry of entries) {
287
+ await this.store(entry);
288
+ }
289
+ }
290
+ /**
291
+ * Bulk delete
292
+ */
293
+ async bulkDelete(ids) {
294
+ let deleted = 0;
295
+ for (const id of ids) {
296
+ if (await this.delete(id)) {
297
+ deleted++;
298
+ }
299
+ }
300
+ return deleted;
301
+ }
302
+ /**
303
+ * Count entries
304
+ */
305
+ async count(namespace) {
306
+ if (namespace) {
307
+ return this.namespaceIndex.get(namespace)?.size || 0;
308
+ }
309
+ return this.entries.size;
310
+ }
311
+ /**
312
+ * List namespaces
313
+ */
314
+ async listNamespaces() {
315
+ return Array.from(this.namespaceIndex.keys());
316
+ }
317
+ /**
318
+ * Clear namespace
319
+ */
320
+ async clearNamespace(namespace) {
321
+ const ids = this.namespaceIndex.get(namespace);
322
+ if (!ids)
323
+ return 0;
324
+ let deleted = 0;
325
+ for (const id of ids) {
326
+ if (await this.delete(id)) {
327
+ deleted++;
328
+ }
329
+ }
330
+ return deleted;
331
+ }
332
+ /**
333
+ * Get statistics
334
+ */
335
+ async getStats() {
336
+ const entriesByNamespace = {};
337
+ for (const [namespace, ids] of this.namespaceIndex) {
338
+ entriesByNamespace[namespace] = ids.size;
339
+ }
340
+ const entriesByType = {
341
+ episodic: 0,
342
+ semantic: 0,
343
+ procedural: 0,
344
+ working: 0,
345
+ cache: 0,
346
+ };
347
+ for (const entry of this.entries.values()) {
348
+ entriesByType[entry.type]++;
349
+ }
350
+ // Get HNSW stats if available
351
+ let hnswStats;
352
+ if (this.agentdb && HNSWIndex) {
353
+ try {
354
+ const hnsw = this.agentdb.getController('hnsw');
355
+ if (hnsw) {
356
+ const stats = hnsw.getStats();
357
+ hnswStats = {
358
+ vectorCount: stats.numElements || 0,
359
+ memoryUsage: 0,
360
+ avgSearchTime: stats.avgSearchTimeMs || 0,
361
+ buildTime: stats.lastBuildTime || 0,
362
+ compressionRatio: 1.0,
363
+ };
364
+ }
365
+ }
366
+ catch {
367
+ // HNSW not available
368
+ }
369
+ }
370
+ return {
371
+ totalEntries: this.entries.size,
372
+ entriesByNamespace,
373
+ entriesByType,
374
+ memoryUsage: this.estimateMemoryUsage(),
375
+ hnswStats,
376
+ avgQueryTime: this.stats.queryCount > 0
377
+ ? this.stats.totalQueryTime / this.stats.queryCount
378
+ : 0,
379
+ avgSearchTime: this.stats.searchCount > 0
380
+ ? this.stats.totalSearchTime / this.stats.searchCount
381
+ : 0,
382
+ };
383
+ }
384
+ /**
385
+ * Health check
386
+ */
387
+ async healthCheck() {
388
+ const issues = [];
389
+ const recommendations = [];
390
+ // Check AgentDB availability
391
+ const storageHealth = this.agentdb
392
+ ? { status: 'healthy', latency: 0 }
393
+ : {
394
+ status: 'degraded',
395
+ latency: 0,
396
+ message: 'AgentDB not available, using fallback',
397
+ };
398
+ // Check index health
399
+ const indexHealth = { status: 'healthy', latency: 0 };
400
+ if (!this.agentdb) {
401
+ indexHealth.status = 'degraded';
402
+ indexHealth.message = 'HNSW index not available';
403
+ recommendations.push('Install agentdb for 150x-12,500x faster vector search');
404
+ }
405
+ // Check cache health
406
+ const cacheHealth = { status: 'healthy', latency: 0 };
407
+ const status = storageHealth.status === 'unhealthy' || indexHealth.status === 'unhealthy'
408
+ ? 'unhealthy'
409
+ : storageHealth.status === 'degraded' || indexHealth.status === 'degraded'
410
+ ? 'degraded'
411
+ : 'healthy';
412
+ return {
413
+ status,
414
+ components: {
415
+ storage: storageHealth,
416
+ index: indexHealth,
417
+ cache: cacheHealth,
418
+ },
419
+ timestamp: Date.now(),
420
+ issues,
421
+ recommendations,
422
+ };
423
+ }
424
+ // ===== Private Methods =====
425
+ /**
426
+ * Create database schema
427
+ */
428
+ async createSchema() {
429
+ if (!this.agentdb)
430
+ return;
431
+ const db = this.agentdb.database;
432
+ if (!db || typeof db.run !== 'function') {
433
+ // AgentDB doesn't expose raw database - using native API
434
+ return;
435
+ }
436
+ try {
437
+ // Create memory_entries table
438
+ await db.run(`
439
+ CREATE TABLE IF NOT EXISTS memory_entries (
440
+ id TEXT PRIMARY KEY,
441
+ key TEXT NOT NULL,
442
+ content TEXT NOT NULL,
443
+ embedding BLOB,
444
+ type TEXT NOT NULL,
445
+ namespace TEXT NOT NULL,
446
+ tags TEXT,
447
+ metadata TEXT,
448
+ owner_id TEXT,
449
+ access_level TEXT NOT NULL,
450
+ created_at INTEGER NOT NULL,
451
+ updated_at INTEGER NOT NULL,
452
+ expires_at INTEGER,
453
+ version INTEGER NOT NULL,
454
+ references TEXT,
455
+ access_count INTEGER DEFAULT 0,
456
+ last_accessed_at INTEGER
457
+ )
458
+ `);
459
+ // Create indexes
460
+ await db.run('CREATE INDEX IF NOT EXISTS idx_namespace ON memory_entries(namespace)');
461
+ await db.run('CREATE INDEX IF NOT EXISTS idx_key ON memory_entries(key)');
462
+ await db.run('CREATE INDEX IF NOT EXISTS idx_type ON memory_entries(type)');
463
+ }
464
+ catch {
465
+ // Schema creation failed - using in-memory only
466
+ }
467
+ }
468
+ /**
469
+ * Store entry in AgentDB
470
+ */
471
+ async storeInAgentDB(entry) {
472
+ if (!this.agentdb)
473
+ return;
474
+ // Try to use agentdb's native store method if available
475
+ try {
476
+ if (typeof this.agentdb.store === 'function') {
477
+ await this.agentdb.store(entry.id, {
478
+ key: entry.key,
479
+ content: entry.content,
480
+ embedding: entry.embedding,
481
+ type: entry.type,
482
+ namespace: entry.namespace,
483
+ tags: entry.tags,
484
+ metadata: entry.metadata,
485
+ });
486
+ return;
487
+ }
488
+ // Fallback: use database directly if available
489
+ const db = this.agentdb.database;
490
+ if (!db || typeof db.run !== 'function') {
491
+ // No compatible database interface - skip agentdb storage
492
+ // Entry is already stored in-memory
493
+ return;
494
+ }
495
+ await db.run(`
496
+ INSERT OR REPLACE INTO memory_entries
497
+ (id, key, content, embedding, type, namespace, tags, metadata, owner_id,
498
+ access_level, created_at, updated_at, expires_at, version, references,
499
+ access_count, last_accessed_at)
500
+ VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)
501
+ `, [
502
+ entry.id,
503
+ entry.key,
504
+ entry.content,
505
+ entry.embedding ? Buffer.from(entry.embedding.buffer) : null,
506
+ entry.type,
507
+ entry.namespace,
508
+ JSON.stringify(entry.tags),
509
+ JSON.stringify(entry.metadata),
510
+ entry.ownerId || null,
511
+ entry.accessLevel,
512
+ entry.createdAt,
513
+ entry.updatedAt,
514
+ entry.expiresAt || null,
515
+ entry.version,
516
+ JSON.stringify(entry.references),
517
+ entry.accessCount,
518
+ entry.lastAccessedAt,
519
+ ]);
520
+ }
521
+ catch {
522
+ // AgentDB storage failed - entry is already in-memory
523
+ }
524
+ // Add to vector index if HNSW is available
525
+ if (entry.embedding && HNSWIndex) {
526
+ try {
527
+ const hnsw = this.agentdb.getController('hnsw');
528
+ if (hnsw) {
529
+ // Convert string ID to number for HNSW (use hash)
530
+ const numericId = this.stringIdToNumeric(entry.id);
531
+ hnsw.addVector(numericId, entry.embedding);
532
+ }
533
+ }
534
+ catch {
535
+ // HNSW not available
536
+ }
537
+ }
538
+ }
539
+ /**
540
+ * Get entry from AgentDB
541
+ */
542
+ async getFromAgentDB(id) {
543
+ if (!this.agentdb)
544
+ return null;
545
+ try {
546
+ // Try native get method first
547
+ if (typeof this.agentdb.get === 'function') {
548
+ const data = await this.agentdb.get(id);
549
+ if (data)
550
+ return this.dataToEntry(id, data);
551
+ }
552
+ // Fallback to database
553
+ const db = this.agentdb.database;
554
+ if (!db || typeof db.get !== 'function')
555
+ return null;
556
+ const row = await db.get('SELECT * FROM memory_entries WHERE id = ?', [id]);
557
+ if (!row)
558
+ return null;
559
+ return this.rowToEntry(row);
560
+ }
561
+ catch {
562
+ return null;
563
+ }
564
+ }
565
+ /**
566
+ * Convert agentdb data to MemoryEntry
567
+ */
568
+ dataToEntry(id, data) {
569
+ const now = Date.now();
570
+ return {
571
+ id,
572
+ key: data.key || id,
573
+ content: data.content || '',
574
+ embedding: data.embedding,
575
+ type: data.type || 'semantic',
576
+ namespace: data.namespace || this.config.namespace,
577
+ tags: data.tags || [],
578
+ metadata: data.metadata || {},
579
+ ownerId: data.ownerId,
580
+ accessLevel: data.accessLevel || 'private',
581
+ createdAt: data.createdAt || now,
582
+ updatedAt: data.updatedAt || now,
583
+ expiresAt: data.expiresAt,
584
+ version: data.version || 1,
585
+ references: data.references || [],
586
+ accessCount: data.accessCount || 0,
587
+ lastAccessedAt: data.lastAccessedAt || now,
588
+ };
589
+ }
590
+ /**
591
+ * Update entry in AgentDB
592
+ */
593
+ async updateInAgentDB(entry) {
594
+ await this.storeInAgentDB(entry);
595
+ }
596
+ /**
597
+ * Delete entry from AgentDB
598
+ */
599
+ async deleteFromAgentDB(id) {
600
+ if (!this.agentdb)
601
+ return;
602
+ try {
603
+ // Try native delete method first
604
+ if (typeof this.agentdb.delete === 'function') {
605
+ await this.agentdb.delete(id);
606
+ return;
607
+ }
608
+ // Fallback to database
609
+ const db = this.agentdb.database;
610
+ if (!db || typeof db.run !== 'function')
611
+ return;
612
+ await db.run('DELETE FROM memory_entries WHERE id = ?', [id]);
613
+ }
614
+ catch {
615
+ // Delete failed - entry removed from in-memory
616
+ }
617
+ }
618
+ /**
619
+ * Search with AgentDB HNSW
620
+ */
621
+ async searchWithAgentDB(embedding, options) {
622
+ if (!this.agentdb || !HNSWIndex) {
623
+ return [];
624
+ }
625
+ try {
626
+ const hnsw = this.agentdb.getController('hnsw');
627
+ if (!hnsw) {
628
+ return this.bruteForceSearch(embedding, options);
629
+ }
630
+ const results = await hnsw.search(embedding, options.k, {
631
+ threshold: options.threshold,
632
+ });
633
+ const searchResults = [];
634
+ for (const result of results) {
635
+ const id = this.numericIdToString(result.id);
636
+ const entry = await this.get(id);
637
+ if (!entry)
638
+ continue;
639
+ searchResults.push({
640
+ entry,
641
+ score: result.similarity,
642
+ distance: result.distance,
643
+ });
644
+ }
645
+ return searchResults;
646
+ }
647
+ catch (error) {
648
+ console.error('HNSW search failed:', error);
649
+ return this.bruteForceSearch(embedding, options);
650
+ }
651
+ }
652
+ /**
653
+ * Brute-force vector search fallback
654
+ */
655
+ bruteForceSearch(embedding, options) {
656
+ const results = [];
657
+ for (const entry of this.entries.values()) {
658
+ if (!entry.embedding)
659
+ continue;
660
+ const score = this.cosineSimilarity(embedding, entry.embedding);
661
+ const distance = 1 - score;
662
+ if (options.threshold && score < options.threshold)
663
+ continue;
664
+ results.push({ entry, score, distance });
665
+ }
666
+ // Sort by score descending
667
+ results.sort((a, b) => b.score - a.score);
668
+ return results.slice(0, options.k);
669
+ }
670
+ /**
671
+ * Semantic search helper
672
+ */
673
+ async semanticSearch(query) {
674
+ let embedding = query.embedding;
675
+ if (!embedding && query.content && this.config.embeddingGenerator) {
676
+ embedding = await this.config.embeddingGenerator(query.content);
677
+ }
678
+ if (!embedding) {
679
+ return [];
680
+ }
681
+ return this.search(embedding, {
682
+ k: query.limit,
683
+ threshold: query.threshold,
684
+ filters: query,
685
+ });
686
+ }
687
+ /**
688
+ * In-memory query fallback
689
+ */
690
+ queryInMemory(query) {
691
+ let results = Array.from(this.entries.values());
692
+ // Apply filters
693
+ if (query.namespace) {
694
+ results = results.filter((e) => e.namespace === query.namespace);
695
+ }
696
+ if (query.key) {
697
+ results = results.filter((e) => e.key === query.key);
698
+ }
699
+ if (query.keyPrefix) {
700
+ results = results.filter((e) => e.key.startsWith(query.keyPrefix));
701
+ }
702
+ if (query.tags && query.tags.length > 0) {
703
+ results = results.filter((e) => query.tags.every((tag) => e.tags.includes(tag)));
704
+ }
705
+ return results.slice(0, query.limit);
706
+ }
707
+ /**
708
+ * Update in-memory indexes
709
+ */
710
+ updateIndexes(entry) {
711
+ const namespace = entry.namespace;
712
+ if (!this.namespaceIndex.has(namespace)) {
713
+ this.namespaceIndex.set(namespace, new Set());
714
+ }
715
+ this.namespaceIndex.get(namespace).add(entry.id);
716
+ const keyIndexKey = `${namespace}:${entry.key}`;
717
+ this.keyIndex.set(keyIndexKey, entry.id);
718
+ }
719
+ /**
720
+ * Convert DB row to MemoryEntry
721
+ */
722
+ rowToEntry(row) {
723
+ return {
724
+ id: row.id,
725
+ key: row.key,
726
+ content: row.content,
727
+ embedding: row.embedding
728
+ ? new Float32Array(new Uint8Array(row.embedding).buffer)
729
+ : undefined,
730
+ type: row.type,
731
+ namespace: row.namespace,
732
+ tags: JSON.parse(row.tags || '[]'),
733
+ metadata: JSON.parse(row.metadata || '{}'),
734
+ ownerId: row.owner_id,
735
+ accessLevel: row.access_level,
736
+ createdAt: row.created_at,
737
+ updatedAt: row.updated_at,
738
+ expiresAt: row.expires_at,
739
+ version: row.version,
740
+ references: JSON.parse(row.references || '[]'),
741
+ accessCount: row.access_count || 0,
742
+ lastAccessedAt: row.last_accessed_at || row.created_at,
743
+ };
744
+ }
745
+ /**
746
+ * Convert string ID to numeric for HNSW
747
+ */
748
+ stringIdToNumeric(id) {
749
+ let hash = 0;
750
+ for (let i = 0; i < id.length; i++) {
751
+ hash = (hash << 5) - hash + id.charCodeAt(i);
752
+ hash |= 0;
753
+ }
754
+ return Math.abs(hash);
755
+ }
756
+ /**
757
+ * Convert numeric ID back to string using O(1) reverse lookup
758
+ * PERFORMANCE FIX: Uses pre-built reverse map instead of O(n) linear scan
759
+ */
760
+ numericIdToString(numericId) {
761
+ // Use O(1) reverse lookup map
762
+ const stringId = this.numericToStringIdMap.get(numericId);
763
+ if (stringId) {
764
+ return stringId;
765
+ }
766
+ // Fallback for unmapped IDs
767
+ return String(numericId);
768
+ }
769
+ /**
770
+ * Register string ID in reverse lookup map
771
+ * Called when storing entries to maintain bidirectional mapping
772
+ */
773
+ registerIdMapping(stringId) {
774
+ const numericId = this.stringIdToNumeric(stringId);
775
+ this.numericToStringIdMap.set(numericId, stringId);
776
+ }
777
+ /**
778
+ * Unregister string ID from reverse lookup map
779
+ * Called when deleting entries
780
+ */
781
+ unregisterIdMapping(stringId) {
782
+ const numericId = this.stringIdToNumeric(stringId);
783
+ this.numericToStringIdMap.delete(numericId);
784
+ }
785
+ /**
786
+ * Cosine similarity (returns value in range [0, 1] where 1 = identical)
787
+ */
788
+ cosineSimilarity(a, b) {
789
+ let dotProduct = 0;
790
+ let normA = 0;
791
+ let normB = 0;
792
+ for (let i = 0; i < a.length; i++) {
793
+ dotProduct += a[i] * b[i];
794
+ normA += a[i] * a[i];
795
+ normB += b[i] * b[i];
796
+ }
797
+ const magnitude = Math.sqrt(normA) * Math.sqrt(normB);
798
+ return magnitude === 0 ? 0 : dotProduct / magnitude;
799
+ }
800
+ /**
801
+ * Estimate memory usage
802
+ */
803
+ estimateMemoryUsage() {
804
+ let total = 0;
805
+ for (const entry of this.entries.values()) {
806
+ total += entry.content.length * 2;
807
+ if (entry.embedding) {
808
+ total += entry.embedding.length * 4;
809
+ }
810
+ }
811
+ return total;
812
+ }
813
+ /**
814
+ * Check if AgentDB is available
815
+ */
816
+ isAvailable() {
817
+ return this.available;
818
+ }
819
+ /**
820
+ * Get underlying AgentDB instance
821
+ */
822
+ getAgentDB() {
823
+ return this.agentdb;
824
+ }
825
+ }
826
+ export default AgentDBBackend;
827
+ //# sourceMappingURL=agentdb-backend.js.map