moflo 4.0.2 → 4.0.3
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/package.json +114 -110
- package/v3/@claude-flow/cli/dist/src/memory/memory-bridge.js +194 -81
- package/v3/@claude-flow/cli/dist/src/memory/memory-initializer.js +1892 -1841
- package/v3/@claude-flow/memory/README.md +587 -0
- package/v3/@claude-flow/memory/dist/agent-memory-scope.d.ts +131 -0
- package/v3/@claude-flow/memory/dist/agent-memory-scope.js +223 -0
- package/v3/@claude-flow/memory/dist/agent-memory-scope.test.d.ts +8 -0
- package/v3/@claude-flow/memory/dist/agent-memory-scope.test.js +463 -0
- package/v3/@claude-flow/memory/dist/agentdb-adapter.d.ts +165 -0
- package/v3/@claude-flow/memory/dist/agentdb-adapter.js +806 -0
- package/v3/@claude-flow/memory/dist/agentdb-backend.d.ts +214 -0
- package/v3/@claude-flow/memory/dist/agentdb-backend.js +844 -0
- package/v3/@claude-flow/memory/dist/agentdb-backend.test.d.ts +7 -0
- package/v3/@claude-flow/memory/dist/agentdb-backend.test.js +258 -0
- package/v3/@claude-flow/memory/dist/application/commands/delete-memory.command.d.ts +65 -0
- package/v3/@claude-flow/memory/dist/application/commands/delete-memory.command.js +129 -0
- package/v3/@claude-flow/memory/dist/application/commands/store-memory.command.d.ts +48 -0
- package/v3/@claude-flow/memory/dist/application/commands/store-memory.command.js +72 -0
- package/v3/@claude-flow/memory/dist/application/index.d.ts +12 -0
- package/v3/@claude-flow/memory/dist/application/index.js +15 -0
- package/v3/@claude-flow/memory/dist/application/queries/search-memory.query.d.ts +72 -0
- package/v3/@claude-flow/memory/dist/application/queries/search-memory.query.js +143 -0
- package/v3/@claude-flow/memory/dist/application/services/memory-application-service.d.ts +121 -0
- package/v3/@claude-flow/memory/dist/application/services/memory-application-service.js +190 -0
- package/v3/@claude-flow/memory/dist/auto-memory-bridge.d.ts +226 -0
- package/v3/@claude-flow/memory/dist/auto-memory-bridge.js +709 -0
- package/v3/@claude-flow/memory/dist/auto-memory-bridge.test.d.ts +8 -0
- package/v3/@claude-flow/memory/dist/auto-memory-bridge.test.js +754 -0
- package/v3/@claude-flow/memory/dist/benchmark.test.d.ts +2 -0
- package/v3/@claude-flow/memory/dist/benchmark.test.js +277 -0
- package/v3/@claude-flow/memory/dist/cache-manager.d.ts +134 -0
- package/v3/@claude-flow/memory/dist/cache-manager.js +407 -0
- package/v3/@claude-flow/memory/dist/controller-registry.d.ts +216 -0
- package/v3/@claude-flow/memory/dist/controller-registry.js +893 -0
- package/v3/@claude-flow/memory/dist/controller-registry.test.d.ts +14 -0
- package/v3/@claude-flow/memory/dist/controller-registry.test.js +636 -0
- package/v3/@claude-flow/memory/dist/database-provider.d.ts +87 -0
- package/v3/@claude-flow/memory/dist/database-provider.js +410 -0
- package/v3/@claude-flow/memory/dist/database-provider.test.d.ts +7 -0
- package/v3/@claude-flow/memory/dist/database-provider.test.js +285 -0
- package/v3/@claude-flow/memory/dist/domain/entities/memory-entry.d.ts +143 -0
- package/v3/@claude-flow/memory/dist/domain/entities/memory-entry.js +226 -0
- package/v3/@claude-flow/memory/dist/domain/index.d.ts +11 -0
- package/v3/@claude-flow/memory/dist/domain/index.js +12 -0
- package/v3/@claude-flow/memory/dist/domain/repositories/memory-repository.interface.d.ts +102 -0
- package/v3/@claude-flow/memory/dist/domain/repositories/memory-repository.interface.js +11 -0
- package/v3/@claude-flow/memory/dist/domain/services/memory-domain-service.d.ts +105 -0
- package/v3/@claude-flow/memory/dist/domain/services/memory-domain-service.js +297 -0
- package/v3/@claude-flow/memory/dist/hnsw-index.d.ts +111 -0
- package/v3/@claude-flow/memory/dist/hnsw-index.js +781 -0
- package/v3/@claude-flow/memory/dist/hnsw-lite.d.ts +23 -0
- package/v3/@claude-flow/memory/dist/hnsw-lite.js +168 -0
- package/v3/@claude-flow/memory/dist/hybrid-backend.d.ts +245 -0
- package/v3/@claude-flow/memory/dist/hybrid-backend.js +569 -0
- package/v3/@claude-flow/memory/dist/hybrid-backend.test.d.ts +8 -0
- package/v3/@claude-flow/memory/dist/hybrid-backend.test.js +320 -0
- package/v3/@claude-flow/memory/dist/index.d.ts +208 -0
- package/v3/@claude-flow/memory/dist/index.js +362 -0
- package/v3/@claude-flow/memory/dist/infrastructure/index.d.ts +17 -0
- package/v3/@claude-flow/memory/dist/infrastructure/index.js +16 -0
- package/v3/@claude-flow/memory/dist/infrastructure/repositories/hybrid-memory-repository.d.ts +66 -0
- package/v3/@claude-flow/memory/dist/infrastructure/repositories/hybrid-memory-repository.js +409 -0
- package/v3/@claude-flow/memory/dist/learning-bridge.d.ts +137 -0
- package/v3/@claude-flow/memory/dist/learning-bridge.js +335 -0
- package/v3/@claude-flow/memory/dist/learning-bridge.test.d.ts +8 -0
- package/v3/@claude-flow/memory/dist/learning-bridge.test.js +578 -0
- package/v3/@claude-flow/memory/dist/memory-graph.d.ts +100 -0
- package/v3/@claude-flow/memory/dist/memory-graph.js +333 -0
- package/v3/@claude-flow/memory/dist/memory-graph.test.d.ts +8 -0
- package/v3/@claude-flow/memory/dist/memory-graph.test.js +609 -0
- package/v3/@claude-flow/memory/dist/migration.d.ts +68 -0
- package/v3/@claude-flow/memory/dist/migration.js +513 -0
- package/v3/@claude-flow/memory/dist/persistent-sona.d.ts +144 -0
- package/v3/@claude-flow/memory/dist/persistent-sona.js +332 -0
- package/v3/@claude-flow/memory/dist/query-builder.d.ts +211 -0
- package/v3/@claude-flow/memory/dist/query-builder.js +438 -0
- package/v3/@claude-flow/memory/dist/rvf-backend.d.ts +51 -0
- package/v3/@claude-flow/memory/dist/rvf-backend.js +481 -0
- package/v3/@claude-flow/memory/dist/rvf-learning-store.d.ts +139 -0
- package/v3/@claude-flow/memory/dist/rvf-learning-store.js +295 -0
- package/v3/@claude-flow/memory/dist/rvf-migration.d.ts +45 -0
- package/v3/@claude-flow/memory/dist/rvf-migration.js +254 -0
- package/v3/@claude-flow/memory/dist/sqlite-backend.d.ts +121 -0
- package/v3/@claude-flow/memory/dist/sqlite-backend.js +564 -0
- package/v3/@claude-flow/memory/dist/sqljs-backend.d.ts +128 -0
- package/v3/@claude-flow/memory/dist/sqljs-backend.js +601 -0
- package/v3/@claude-flow/memory/dist/types.d.ts +484 -0
- package/v3/@claude-flow/memory/dist/types.js +58 -0
- package/v3/@claude-flow/memory/package.json +46 -0
|
@@ -0,0 +1,409 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Hybrid Memory Repository - Infrastructure Layer
|
|
3
|
+
*
|
|
4
|
+
* Implements IMemoryRepository using SQLite + AgentDB hybrid backend.
|
|
5
|
+
* Per ADR-009, this is the default memory backend.
|
|
6
|
+
*
|
|
7
|
+
* @module v3/memory/infrastructure/repositories
|
|
8
|
+
*/
|
|
9
|
+
/**
|
|
10
|
+
* Hybrid Memory Repository
|
|
11
|
+
*
|
|
12
|
+
* Uses SQLite for metadata and AgentDB for vectors.
|
|
13
|
+
* Implements hot caching for frequently accessed entries.
|
|
14
|
+
*/
|
|
15
|
+
export class HybridMemoryRepository {
|
|
16
|
+
config;
|
|
17
|
+
entries = new Map();
|
|
18
|
+
namespaceIndex = new Map();
|
|
19
|
+
vectorIndex = new Map();
|
|
20
|
+
cache = new Map();
|
|
21
|
+
initialized = false;
|
|
22
|
+
constructor(config) {
|
|
23
|
+
this.config = config;
|
|
24
|
+
}
|
|
25
|
+
// ============================================================================
|
|
26
|
+
// Lifecycle
|
|
27
|
+
// ============================================================================
|
|
28
|
+
async initialize() {
|
|
29
|
+
if (this.initialized)
|
|
30
|
+
return;
|
|
31
|
+
// In production, would initialize SQLite and AgentDB connections
|
|
32
|
+
// For now, using in-memory implementation
|
|
33
|
+
this.entries = new Map();
|
|
34
|
+
this.namespaceIndex = new Map();
|
|
35
|
+
this.vectorIndex = new Map();
|
|
36
|
+
this.cache = new Map();
|
|
37
|
+
this.initialized = true;
|
|
38
|
+
}
|
|
39
|
+
async shutdown() {
|
|
40
|
+
// Clear all data
|
|
41
|
+
this.cache.clear();
|
|
42
|
+
this.initialized = false;
|
|
43
|
+
}
|
|
44
|
+
async clear() {
|
|
45
|
+
this.entries.clear();
|
|
46
|
+
this.namespaceIndex.clear();
|
|
47
|
+
this.vectorIndex.clear();
|
|
48
|
+
this.cache.clear();
|
|
49
|
+
}
|
|
50
|
+
// ============================================================================
|
|
51
|
+
// Basic CRUD
|
|
52
|
+
// ============================================================================
|
|
53
|
+
async save(entry) {
|
|
54
|
+
this.ensureInitialized();
|
|
55
|
+
// Store entry
|
|
56
|
+
this.entries.set(entry.id, entry);
|
|
57
|
+
// Update namespace index
|
|
58
|
+
if (!this.namespaceIndex.has(entry.namespace)) {
|
|
59
|
+
this.namespaceIndex.set(entry.namespace, new Set());
|
|
60
|
+
}
|
|
61
|
+
this.namespaceIndex.get(entry.namespace).add(entry.id);
|
|
62
|
+
// Store vector if present
|
|
63
|
+
if (entry.vector) {
|
|
64
|
+
this.vectorIndex.set(entry.id, entry.vector);
|
|
65
|
+
}
|
|
66
|
+
// Update cache
|
|
67
|
+
this.updateCache(entry);
|
|
68
|
+
}
|
|
69
|
+
async findById(id) {
|
|
70
|
+
this.ensureInitialized();
|
|
71
|
+
// Check cache first
|
|
72
|
+
const cached = this.cache.get(id);
|
|
73
|
+
if (cached) {
|
|
74
|
+
return cached.entry;
|
|
75
|
+
}
|
|
76
|
+
return this.entries.get(id) ?? null;
|
|
77
|
+
}
|
|
78
|
+
async findByKey(namespace, key) {
|
|
79
|
+
this.ensureInitialized();
|
|
80
|
+
for (const entry of this.entries.values()) {
|
|
81
|
+
if (entry.namespace === namespace && entry.key === key) {
|
|
82
|
+
return entry;
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
return null;
|
|
86
|
+
}
|
|
87
|
+
async findByCompositeKey(compositeKey) {
|
|
88
|
+
const [namespace, key] = compositeKey.split(':');
|
|
89
|
+
return this.findByKey(namespace, key);
|
|
90
|
+
}
|
|
91
|
+
async delete(id) {
|
|
92
|
+
this.ensureInitialized();
|
|
93
|
+
const entry = this.entries.get(id);
|
|
94
|
+
if (!entry)
|
|
95
|
+
return false;
|
|
96
|
+
// Remove from all indexes
|
|
97
|
+
this.entries.delete(id);
|
|
98
|
+
this.namespaceIndex.get(entry.namespace)?.delete(id);
|
|
99
|
+
this.vectorIndex.delete(id);
|
|
100
|
+
this.cache.delete(id);
|
|
101
|
+
return true;
|
|
102
|
+
}
|
|
103
|
+
async exists(id) {
|
|
104
|
+
return this.entries.has(id);
|
|
105
|
+
}
|
|
106
|
+
// ============================================================================
|
|
107
|
+
// Bulk Operations
|
|
108
|
+
// ============================================================================
|
|
109
|
+
async saveMany(entries) {
|
|
110
|
+
const errors = [];
|
|
111
|
+
let success = 0;
|
|
112
|
+
for (const entry of entries) {
|
|
113
|
+
try {
|
|
114
|
+
await this.save(entry);
|
|
115
|
+
success++;
|
|
116
|
+
}
|
|
117
|
+
catch (error) {
|
|
118
|
+
errors.push({
|
|
119
|
+
id: entry.id,
|
|
120
|
+
error: error instanceof Error ? error.message : 'Unknown error',
|
|
121
|
+
});
|
|
122
|
+
}
|
|
123
|
+
}
|
|
124
|
+
return {
|
|
125
|
+
success,
|
|
126
|
+
failed: errors.length,
|
|
127
|
+
errors,
|
|
128
|
+
};
|
|
129
|
+
}
|
|
130
|
+
async findByIds(ids) {
|
|
131
|
+
return ids
|
|
132
|
+
.map((id) => this.entries.get(id))
|
|
133
|
+
.filter((e) => e !== undefined);
|
|
134
|
+
}
|
|
135
|
+
async deleteMany(ids) {
|
|
136
|
+
const errors = [];
|
|
137
|
+
let success = 0;
|
|
138
|
+
for (const id of ids) {
|
|
139
|
+
try {
|
|
140
|
+
if (await this.delete(id)) {
|
|
141
|
+
success++;
|
|
142
|
+
}
|
|
143
|
+
else {
|
|
144
|
+
errors.push({ id, error: 'Entry not found' });
|
|
145
|
+
}
|
|
146
|
+
}
|
|
147
|
+
catch (error) {
|
|
148
|
+
errors.push({
|
|
149
|
+
id,
|
|
150
|
+
error: error instanceof Error ? error.message : 'Unknown error',
|
|
151
|
+
});
|
|
152
|
+
}
|
|
153
|
+
}
|
|
154
|
+
return {
|
|
155
|
+
success,
|
|
156
|
+
failed: errors.length,
|
|
157
|
+
errors,
|
|
158
|
+
};
|
|
159
|
+
}
|
|
160
|
+
// ============================================================================
|
|
161
|
+
// Query Operations
|
|
162
|
+
// ============================================================================
|
|
163
|
+
async findAll(options) {
|
|
164
|
+
this.ensureInitialized();
|
|
165
|
+
let results = Array.from(this.entries.values());
|
|
166
|
+
// Apply filters
|
|
167
|
+
if (options?.namespace) {
|
|
168
|
+
results = results.filter((e) => e.namespace === options.namespace);
|
|
169
|
+
}
|
|
170
|
+
if (options?.type) {
|
|
171
|
+
results = results.filter((e) => e.type === options.type);
|
|
172
|
+
}
|
|
173
|
+
if (options?.status) {
|
|
174
|
+
results = results.filter((e) => e.status === options.status);
|
|
175
|
+
}
|
|
176
|
+
// Apply sorting
|
|
177
|
+
const orderBy = options?.orderBy ?? 'createdAt';
|
|
178
|
+
const orderDir = options?.orderDirection ?? 'desc';
|
|
179
|
+
results.sort((a, b) => {
|
|
180
|
+
let aVal, bVal;
|
|
181
|
+
switch (orderBy) {
|
|
182
|
+
case 'accessCount':
|
|
183
|
+
aVal = a.accessCount;
|
|
184
|
+
bVal = b.accessCount;
|
|
185
|
+
break;
|
|
186
|
+
case 'lastAccessedAt':
|
|
187
|
+
aVal = a.lastAccessedAt.getTime();
|
|
188
|
+
bVal = b.lastAccessedAt.getTime();
|
|
189
|
+
break;
|
|
190
|
+
case 'updatedAt':
|
|
191
|
+
aVal = a.updatedAt.getTime();
|
|
192
|
+
bVal = b.updatedAt.getTime();
|
|
193
|
+
break;
|
|
194
|
+
default:
|
|
195
|
+
aVal = a.createdAt.getTime();
|
|
196
|
+
bVal = b.createdAt.getTime();
|
|
197
|
+
}
|
|
198
|
+
return orderDir === 'asc' ? aVal - bVal : bVal - aVal;
|
|
199
|
+
});
|
|
200
|
+
// Apply pagination
|
|
201
|
+
if (options?.offset) {
|
|
202
|
+
results = results.slice(options.offset);
|
|
203
|
+
}
|
|
204
|
+
if (options?.limit) {
|
|
205
|
+
results = results.slice(0, options.limit);
|
|
206
|
+
}
|
|
207
|
+
return results;
|
|
208
|
+
}
|
|
209
|
+
async findByNamespace(namespace, options) {
|
|
210
|
+
return this.findAll({ ...options, namespace });
|
|
211
|
+
}
|
|
212
|
+
async findByType(type, options) {
|
|
213
|
+
return this.findAll({ ...options, type });
|
|
214
|
+
}
|
|
215
|
+
async findByStatus(status, options) {
|
|
216
|
+
return this.findAll({ ...options, status });
|
|
217
|
+
}
|
|
218
|
+
// ============================================================================
|
|
219
|
+
// Vector Search
|
|
220
|
+
// ============================================================================
|
|
221
|
+
async searchByVector(options) {
|
|
222
|
+
this.ensureInitialized();
|
|
223
|
+
const results = [];
|
|
224
|
+
for (const [id, vector] of this.vectorIndex) {
|
|
225
|
+
const entry = this.entries.get(id);
|
|
226
|
+
if (!entry)
|
|
227
|
+
continue;
|
|
228
|
+
if (options.namespace && entry.namespace !== options.namespace)
|
|
229
|
+
continue;
|
|
230
|
+
if (options.type && entry.type !== options.type)
|
|
231
|
+
continue;
|
|
232
|
+
const similarity = this.cosineSimilarity(options.vector, vector);
|
|
233
|
+
const threshold = options.threshold ?? 0.5;
|
|
234
|
+
if (similarity >= threshold) {
|
|
235
|
+
results.push({
|
|
236
|
+
entry,
|
|
237
|
+
similarity,
|
|
238
|
+
distance: 1 - similarity,
|
|
239
|
+
});
|
|
240
|
+
}
|
|
241
|
+
}
|
|
242
|
+
// Sort by similarity descending
|
|
243
|
+
results.sort((a, b) => b.similarity - a.similarity);
|
|
244
|
+
// Limit results
|
|
245
|
+
return results.slice(0, options.limit ?? 10);
|
|
246
|
+
}
|
|
247
|
+
async findSimilar(entryId, limit = 10) {
|
|
248
|
+
const entry = this.entries.get(entryId);
|
|
249
|
+
if (!entry || !entry.vector)
|
|
250
|
+
return [];
|
|
251
|
+
return this.searchByVector({
|
|
252
|
+
vector: entry.vector,
|
|
253
|
+
namespace: entry.namespace,
|
|
254
|
+
limit: limit + 1, // Include self
|
|
255
|
+
}).then((results) => results.filter((r) => r.entry.id !== entryId).slice(0, limit));
|
|
256
|
+
}
|
|
257
|
+
// ============================================================================
|
|
258
|
+
// Maintenance Operations
|
|
259
|
+
// ============================================================================
|
|
260
|
+
async findExpired() {
|
|
261
|
+
return Array.from(this.entries.values()).filter((e) => e.isExpired());
|
|
262
|
+
}
|
|
263
|
+
async deleteExpired() {
|
|
264
|
+
const expired = await this.findExpired();
|
|
265
|
+
for (const entry of expired) {
|
|
266
|
+
await this.delete(entry.id);
|
|
267
|
+
}
|
|
268
|
+
return expired.length;
|
|
269
|
+
}
|
|
270
|
+
async findCold(milliseconds) {
|
|
271
|
+
return Array.from(this.entries.values()).filter((e) => e.isCold(milliseconds));
|
|
272
|
+
}
|
|
273
|
+
async archiveCold(milliseconds) {
|
|
274
|
+
const cold = await this.findCold(milliseconds);
|
|
275
|
+
for (const entry of cold) {
|
|
276
|
+
entry.archive();
|
|
277
|
+
await this.save(entry);
|
|
278
|
+
}
|
|
279
|
+
return cold.length;
|
|
280
|
+
}
|
|
281
|
+
// ============================================================================
|
|
282
|
+
// Statistics
|
|
283
|
+
// ============================================================================
|
|
284
|
+
async getStatistics() {
|
|
285
|
+
const entries = Array.from(this.entries.values());
|
|
286
|
+
const entriesByNamespace = {};
|
|
287
|
+
const entriesByType = {
|
|
288
|
+
semantic: 0,
|
|
289
|
+
episodic: 0,
|
|
290
|
+
procedural: 0,
|
|
291
|
+
working: 0,
|
|
292
|
+
};
|
|
293
|
+
let totalAccessCount = 0;
|
|
294
|
+
let totalSize = 0;
|
|
295
|
+
let activeCount = 0;
|
|
296
|
+
let archivedCount = 0;
|
|
297
|
+
let deletedCount = 0;
|
|
298
|
+
for (const entry of entries) {
|
|
299
|
+
// Count by namespace
|
|
300
|
+
entriesByNamespace[entry.namespace] = (entriesByNamespace[entry.namespace] ?? 0) + 1;
|
|
301
|
+
// Count by type
|
|
302
|
+
entriesByType[entry.type]++;
|
|
303
|
+
// Accumulate stats
|
|
304
|
+
totalAccessCount += entry.accessCount;
|
|
305
|
+
totalSize += JSON.stringify(entry.value).length;
|
|
306
|
+
// Count by status
|
|
307
|
+
switch (entry.status) {
|
|
308
|
+
case 'active':
|
|
309
|
+
activeCount++;
|
|
310
|
+
break;
|
|
311
|
+
case 'archived':
|
|
312
|
+
archivedCount++;
|
|
313
|
+
break;
|
|
314
|
+
case 'deleted':
|
|
315
|
+
deletedCount++;
|
|
316
|
+
break;
|
|
317
|
+
}
|
|
318
|
+
}
|
|
319
|
+
// Find hottest and coldest
|
|
320
|
+
const sorted = [...entries].sort((a, b) => b.accessCount - a.accessCount);
|
|
321
|
+
const hottestEntries = sorted.slice(0, 5).map((e) => e.id);
|
|
322
|
+
const coldestEntries = sorted.slice(-5).reverse().map((e) => e.id);
|
|
323
|
+
return {
|
|
324
|
+
totalEntries: entries.length,
|
|
325
|
+
activeEntries: activeCount,
|
|
326
|
+
archivedEntries: archivedCount,
|
|
327
|
+
deletedEntries: deletedCount,
|
|
328
|
+
totalSize,
|
|
329
|
+
entriesByNamespace,
|
|
330
|
+
entriesByType,
|
|
331
|
+
averageAccessCount: entries.length > 0 ? totalAccessCount / entries.length : 0,
|
|
332
|
+
hottestEntries,
|
|
333
|
+
coldestEntries,
|
|
334
|
+
};
|
|
335
|
+
}
|
|
336
|
+
async count(options) {
|
|
337
|
+
const entries = await this.findAll(options);
|
|
338
|
+
return entries.length;
|
|
339
|
+
}
|
|
340
|
+
// ============================================================================
|
|
341
|
+
// Namespace Operations
|
|
342
|
+
// ============================================================================
|
|
343
|
+
async listNamespaces() {
|
|
344
|
+
return Array.from(this.namespaceIndex.keys());
|
|
345
|
+
}
|
|
346
|
+
async deleteNamespace(namespace) {
|
|
347
|
+
const ids = this.namespaceIndex.get(namespace);
|
|
348
|
+
if (!ids)
|
|
349
|
+
return 0;
|
|
350
|
+
const count = ids.size;
|
|
351
|
+
for (const id of ids) {
|
|
352
|
+
await this.delete(id);
|
|
353
|
+
}
|
|
354
|
+
this.namespaceIndex.delete(namespace);
|
|
355
|
+
return count;
|
|
356
|
+
}
|
|
357
|
+
async getNamespaceSize(namespace) {
|
|
358
|
+
const ids = this.namespaceIndex.get(namespace);
|
|
359
|
+
if (!ids)
|
|
360
|
+
return 0;
|
|
361
|
+
let size = 0;
|
|
362
|
+
for (const id of ids) {
|
|
363
|
+
const entry = this.entries.get(id);
|
|
364
|
+
if (entry) {
|
|
365
|
+
size += JSON.stringify(entry.value).length;
|
|
366
|
+
}
|
|
367
|
+
}
|
|
368
|
+
return size;
|
|
369
|
+
}
|
|
370
|
+
// ============================================================================
|
|
371
|
+
// Private Methods
|
|
372
|
+
// ============================================================================
|
|
373
|
+
ensureInitialized() {
|
|
374
|
+
if (!this.initialized) {
|
|
375
|
+
throw new Error('Repository not initialized. Call initialize() first.');
|
|
376
|
+
}
|
|
377
|
+
}
|
|
378
|
+
updateCache(entry) {
|
|
379
|
+
const maxCacheSize = this.config.cacheSize ?? 100;
|
|
380
|
+
// Evict oldest entries if cache is full
|
|
381
|
+
if (this.cache.size >= maxCacheSize) {
|
|
382
|
+
const oldest = Array.from(this.cache.entries())
|
|
383
|
+
.sort(([, a], [, b]) => a.timestamp - b.timestamp)
|
|
384
|
+
.slice(0, Math.floor(maxCacheSize * 0.2));
|
|
385
|
+
for (const [id] of oldest) {
|
|
386
|
+
this.cache.delete(id);
|
|
387
|
+
}
|
|
388
|
+
}
|
|
389
|
+
this.cache.set(entry.id, {
|
|
390
|
+
entry,
|
|
391
|
+
timestamp: Date.now(),
|
|
392
|
+
});
|
|
393
|
+
}
|
|
394
|
+
cosineSimilarity(a, b) {
|
|
395
|
+
if (a.length !== b.length)
|
|
396
|
+
return 0;
|
|
397
|
+
let dotProduct = 0;
|
|
398
|
+
let normA = 0;
|
|
399
|
+
let normB = 0;
|
|
400
|
+
for (let i = 0; i < a.length; i++) {
|
|
401
|
+
dotProduct += a[i] * b[i];
|
|
402
|
+
normA += a[i] * a[i];
|
|
403
|
+
normB += b[i] * b[i];
|
|
404
|
+
}
|
|
405
|
+
const denominator = Math.sqrt(normA) * Math.sqrt(normB);
|
|
406
|
+
return denominator === 0 ? 0 : dotProduct / denominator;
|
|
407
|
+
}
|
|
408
|
+
}
|
|
409
|
+
//# sourceMappingURL=hybrid-memory-repository.js.map
|
|
@@ -0,0 +1,137 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* LearningBridge - Connects AutoMemoryBridge to NeuralLearningSystem
|
|
3
|
+
*
|
|
4
|
+
* When insights are recorded via AutoMemoryBridge, this module triggers
|
|
5
|
+
* neural learning trajectories so the system continuously improves from
|
|
6
|
+
* its own discoveries. The NeuralLearningSystem dependency is optional:
|
|
7
|
+
* when unavailable, all operations degrade gracefully to no-ops.
|
|
8
|
+
*
|
|
9
|
+
* @module @claude-flow/memory/learning-bridge
|
|
10
|
+
*/
|
|
11
|
+
import { EventEmitter } from 'node:events';
|
|
12
|
+
import type { IMemoryBackend, SONAMode } from './types.js';
|
|
13
|
+
import type { MemoryInsight } from './auto-memory-bridge.js';
|
|
14
|
+
/**
|
|
15
|
+
* Factory function that returns a neural system instance.
|
|
16
|
+
* Used for dependency injection so tests can supply a mock.
|
|
17
|
+
*/
|
|
18
|
+
export type NeuralLoader = () => Promise<any>;
|
|
19
|
+
/** Configuration for the LearningBridge */
|
|
20
|
+
export interface LearningBridgeConfig {
|
|
21
|
+
/** SONA operating mode (default: 'balanced') */
|
|
22
|
+
sonaMode?: SONAMode;
|
|
23
|
+
/** Per-hour confidence decay rate (default: 0.005) */
|
|
24
|
+
confidenceDecayRate?: number;
|
|
25
|
+
/** Confidence boost per access (default: 0.03) */
|
|
26
|
+
accessBoostAmount?: number;
|
|
27
|
+
/** Maximum confidence value (default: 1.0) */
|
|
28
|
+
maxConfidence?: number;
|
|
29
|
+
/** Minimum confidence floor (default: 0.1) */
|
|
30
|
+
minConfidence?: number;
|
|
31
|
+
/** EWC regularization strength (default: 2000) */
|
|
32
|
+
ewcLambda?: number;
|
|
33
|
+
/** Min active trajectories before consolidation runs (default: 10) */
|
|
34
|
+
consolidationThreshold?: number;
|
|
35
|
+
/** Enable the bridge (default: true). When false all methods are no-ops */
|
|
36
|
+
enabled?: boolean;
|
|
37
|
+
/**
|
|
38
|
+
* Optional factory for the neural learning system.
|
|
39
|
+
* When provided, this replaces the default dynamic import of @claude-flow/neural.
|
|
40
|
+
* Primarily used for testing.
|
|
41
|
+
*/
|
|
42
|
+
neuralLoader?: NeuralLoader;
|
|
43
|
+
}
|
|
44
|
+
/** Aggregated learning statistics */
|
|
45
|
+
export interface LearningStats {
|
|
46
|
+
totalTrajectories: number;
|
|
47
|
+
completedTrajectories: number;
|
|
48
|
+
activeTrajectories: number;
|
|
49
|
+
totalConsolidations: number;
|
|
50
|
+
totalDecays: number;
|
|
51
|
+
avgConfidenceBoost: number;
|
|
52
|
+
neuralAvailable: boolean;
|
|
53
|
+
}
|
|
54
|
+
/** Result returned by consolidate() */
|
|
55
|
+
export interface ConsolidateResult {
|
|
56
|
+
trajectoriesCompleted: number;
|
|
57
|
+
patternsLearned: number;
|
|
58
|
+
entriesUpdated: number;
|
|
59
|
+
durationMs: number;
|
|
60
|
+
}
|
|
61
|
+
/** A single pattern match returned by findSimilarPatterns() */
|
|
62
|
+
export interface PatternMatch {
|
|
63
|
+
content: string;
|
|
64
|
+
similarity: number;
|
|
65
|
+
category: string;
|
|
66
|
+
confidence: number;
|
|
67
|
+
}
|
|
68
|
+
/**
|
|
69
|
+
* Connects AutoMemoryBridge insights to the NeuralLearningSystem.
|
|
70
|
+
*
|
|
71
|
+
* @example
|
|
72
|
+
* ```typescript
|
|
73
|
+
* const bridge = new LearningBridge(memoryBackend);
|
|
74
|
+
* await bridge.onInsightRecorded(insight, entryId);
|
|
75
|
+
* await bridge.onInsightAccessed(entryId);
|
|
76
|
+
* const result = await bridge.consolidate();
|
|
77
|
+
* ```
|
|
78
|
+
*/
|
|
79
|
+
export declare class LearningBridge extends EventEmitter {
|
|
80
|
+
private neural;
|
|
81
|
+
private backend;
|
|
82
|
+
private config;
|
|
83
|
+
private activeTrajectories;
|
|
84
|
+
private stats;
|
|
85
|
+
private destroyed;
|
|
86
|
+
private neuralInitPromise;
|
|
87
|
+
constructor(backend: IMemoryBackend, config?: LearningBridgeConfig);
|
|
88
|
+
/**
|
|
89
|
+
* Notify the bridge that an insight has been recorded in AgentDB.
|
|
90
|
+
* Creates a learning trajectory so the neural system can track the
|
|
91
|
+
* insight's lifecycle.
|
|
92
|
+
*/
|
|
93
|
+
onInsightRecorded(insight: MemoryInsight, entryId: string): Promise<void>;
|
|
94
|
+
/**
|
|
95
|
+
* Notify the bridge that an insight entry was accessed.
|
|
96
|
+
* Boosts confidence in the backend and records a step in the
|
|
97
|
+
* trajectory if one exists.
|
|
98
|
+
*/
|
|
99
|
+
onInsightAccessed(entryId: string): Promise<void>;
|
|
100
|
+
/**
|
|
101
|
+
* Consolidate active trajectories by completing them in the neural system.
|
|
102
|
+
* Only runs when there are enough active trajectories to justify the cost.
|
|
103
|
+
*/
|
|
104
|
+
consolidate(): Promise<ConsolidateResult>;
|
|
105
|
+
/**
|
|
106
|
+
* Apply time-based confidence decay to entries in the given namespace.
|
|
107
|
+
* Entries not accessed for more than one hour see their confidence reduced
|
|
108
|
+
* proportionally to the hours elapsed, down to minConfidence.
|
|
109
|
+
*
|
|
110
|
+
* @returns number of entries whose confidence was lowered
|
|
111
|
+
*/
|
|
112
|
+
decayConfidences(namespace: string): Promise<number>;
|
|
113
|
+
/**
|
|
114
|
+
* Find patterns similar to the given content using the neural system.
|
|
115
|
+
* Returns an empty array when the neural system is unavailable.
|
|
116
|
+
*/
|
|
117
|
+
findSimilarPatterns(content: string, k?: number): Promise<PatternMatch[]>;
|
|
118
|
+
/** Return aggregated learning statistics */
|
|
119
|
+
getStats(): LearningStats;
|
|
120
|
+
/** Tear down the bridge. Subsequent method calls become no-ops. */
|
|
121
|
+
destroy(): void;
|
|
122
|
+
/**
|
|
123
|
+
* Lazily attempt to load and initialize the NeuralLearningSystem.
|
|
124
|
+
* The promise is cached so that repeated calls do not re-attempt
|
|
125
|
+
* after a failure.
|
|
126
|
+
*/
|
|
127
|
+
private initNeural;
|
|
128
|
+
private loadNeural;
|
|
129
|
+
/**
|
|
130
|
+
* Create a deterministic hash-based embedding for content.
|
|
131
|
+
* This is a lightweight stand-in for a real embedding model,
|
|
132
|
+
* suitable for pattern matching within the neural trajectory system.
|
|
133
|
+
*/
|
|
134
|
+
private createHashEmbedding;
|
|
135
|
+
}
|
|
136
|
+
export default LearningBridge;
|
|
137
|
+
//# sourceMappingURL=learning-bridge.d.ts.map
|