@sparkleideas/shared 3.0.0-alpha.7
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +323 -0
- package/__tests__/hooks/bash-safety.test.ts +289 -0
- package/__tests__/hooks/file-organization.test.ts +335 -0
- package/__tests__/hooks/git-commit.test.ts +336 -0
- package/__tests__/hooks/index.ts +23 -0
- package/__tests__/hooks/session-hooks.test.ts +357 -0
- package/__tests__/hooks/task-hooks.test.ts +193 -0
- package/docs/EVENTS_IMPLEMENTATION_SUMMARY.md +388 -0
- package/docs/EVENTS_QUICK_REFERENCE.md +470 -0
- package/docs/EVENTS_README.md +352 -0
- package/package.json +39 -0
- package/src/core/config/defaults.ts +207 -0
- package/src/core/config/index.ts +15 -0
- package/src/core/config/loader.ts +271 -0
- package/src/core/config/schema.ts +188 -0
- package/src/core/config/validator.ts +209 -0
- package/src/core/event-bus.ts +236 -0
- package/src/core/index.ts +22 -0
- package/src/core/interfaces/agent.interface.ts +251 -0
- package/src/core/interfaces/coordinator.interface.ts +363 -0
- package/src/core/interfaces/event.interface.ts +267 -0
- package/src/core/interfaces/index.ts +19 -0
- package/src/core/interfaces/memory.interface.ts +332 -0
- package/src/core/interfaces/task.interface.ts +223 -0
- package/src/core/orchestrator/event-coordinator.ts +122 -0
- package/src/core/orchestrator/health-monitor.ts +214 -0
- package/src/core/orchestrator/index.ts +89 -0
- package/src/core/orchestrator/lifecycle-manager.ts +263 -0
- package/src/core/orchestrator/session-manager.ts +279 -0
- package/src/core/orchestrator/task-manager.ts +317 -0
- package/src/events/domain-events.ts +584 -0
- package/src/events/event-store.test.ts +387 -0
- package/src/events/event-store.ts +588 -0
- package/src/events/example-usage.ts +293 -0
- package/src/events/index.ts +90 -0
- package/src/events/projections.ts +561 -0
- package/src/events/state-reconstructor.ts +349 -0
- package/src/events.ts +367 -0
- package/src/hooks/INTEGRATION.md +658 -0
- package/src/hooks/README.md +532 -0
- package/src/hooks/example-usage.ts +499 -0
- package/src/hooks/executor.ts +379 -0
- package/src/hooks/hooks.test.ts +421 -0
- package/src/hooks/index.ts +131 -0
- package/src/hooks/registry.ts +333 -0
- package/src/hooks/safety/bash-safety.ts +604 -0
- package/src/hooks/safety/file-organization.ts +473 -0
- package/src/hooks/safety/git-commit.ts +623 -0
- package/src/hooks/safety/index.ts +46 -0
- package/src/hooks/session-hooks.ts +559 -0
- package/src/hooks/task-hooks.ts +513 -0
- package/src/hooks/types.ts +357 -0
- package/src/hooks/verify-exports.test.ts +125 -0
- package/src/index.ts +195 -0
- package/src/mcp/connection-pool.ts +438 -0
- package/src/mcp/index.ts +183 -0
- package/src/mcp/server.ts +774 -0
- package/src/mcp/session-manager.ts +428 -0
- package/src/mcp/tool-registry.ts +566 -0
- package/src/mcp/transport/http.ts +557 -0
- package/src/mcp/transport/index.ts +294 -0
- package/src/mcp/transport/stdio.ts +324 -0
- package/src/mcp/transport/websocket.ts +484 -0
- package/src/mcp/types.ts +565 -0
- package/src/plugin-interface.ts +663 -0
- package/src/plugin-loader.ts +638 -0
- package/src/plugin-registry.ts +604 -0
- package/src/plugins/index.ts +34 -0
- package/src/plugins/official/hive-mind-plugin.ts +330 -0
- package/src/plugins/official/index.ts +24 -0
- package/src/plugins/official/maestro-plugin.ts +508 -0
- package/src/plugins/types.ts +108 -0
- package/src/resilience/bulkhead.ts +277 -0
- package/src/resilience/circuit-breaker.ts +326 -0
- package/src/resilience/index.ts +26 -0
- package/src/resilience/rate-limiter.ts +420 -0
- package/src/resilience/retry.ts +224 -0
- package/src/security/index.ts +39 -0
- package/src/security/input-validation.ts +265 -0
- package/src/security/secure-random.ts +159 -0
- package/src/services/index.ts +16 -0
- package/src/services/v3-progress.service.ts +505 -0
- package/src/types/agent.types.ts +144 -0
- package/src/types/index.ts +22 -0
- package/src/types/mcp.types.ts +300 -0
- package/src/types/memory.types.ts +263 -0
- package/src/types/swarm.types.ts +255 -0
- package/src/types/task.types.ts +205 -0
- package/src/types.ts +367 -0
- package/src/utils/secure-logger.d.ts +69 -0
- package/src/utils/secure-logger.d.ts.map +1 -0
- package/src/utils/secure-logger.js +208 -0
- package/src/utils/secure-logger.js.map +1 -0
- package/src/utils/secure-logger.ts +257 -0
- package/tmp.json +0 -0
- package/tsconfig.json +9 -0
|
@@ -0,0 +1,387 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Event Store Tests
|
|
3
|
+
*
|
|
4
|
+
* Comprehensive tests for the event sourcing implementation.
|
|
5
|
+
* Tests event storage, retrieval, filtering, snapshots, and projections.
|
|
6
|
+
*/
|
|
7
|
+
|
|
8
|
+
import { describe, it, expect, beforeEach, afterEach } from 'vitest';
|
|
9
|
+
import { EventStore } from './event-store.js';
|
|
10
|
+
import { AgentStateProjection, TaskHistoryProjection, MemoryIndexProjection } from './projections.js';
|
|
11
|
+
import {
|
|
12
|
+
createAgentSpawnedEvent,
|
|
13
|
+
createAgentStartedEvent,
|
|
14
|
+
createAgentTaskAssignedEvent,
|
|
15
|
+
createAgentTaskCompletedEvent,
|
|
16
|
+
createTaskCreatedEvent,
|
|
17
|
+
createTaskStartedEvent,
|
|
18
|
+
createTaskCompletedEvent,
|
|
19
|
+
createMemoryStoredEvent,
|
|
20
|
+
createMemoryRetrievedEvent,
|
|
21
|
+
createMemoryDeletedEvent,
|
|
22
|
+
createSwarmInitializedEvent,
|
|
23
|
+
} from './domain-events.js';
|
|
24
|
+
|
|
25
|
+
describe('EventStore', () => {
|
|
26
|
+
let eventStore: EventStore;
|
|
27
|
+
|
|
28
|
+
beforeEach(async () => {
|
|
29
|
+
eventStore = new EventStore({ databasePath: ':memory:', verbose: false });
|
|
30
|
+
await eventStore.initialize();
|
|
31
|
+
});
|
|
32
|
+
|
|
33
|
+
afterEach(async () => {
|
|
34
|
+
await eventStore.shutdown();
|
|
35
|
+
});
|
|
36
|
+
|
|
37
|
+
describe('Event Appending', () => {
|
|
38
|
+
it('should append events to the store', async () => {
|
|
39
|
+
const event = createAgentSpawnedEvent('agent-1', 'coder', 'core', ['coding', 'testing']);
|
|
40
|
+
|
|
41
|
+
await eventStore.append(event);
|
|
42
|
+
|
|
43
|
+
const events = await eventStore.getEvents('agent-1');
|
|
44
|
+
expect(events).toHaveLength(1);
|
|
45
|
+
expect(events[0].type).toBe('agent:spawned');
|
|
46
|
+
expect(events[0].version).toBe(1);
|
|
47
|
+
});
|
|
48
|
+
|
|
49
|
+
it('should increment version for each event', async () => {
|
|
50
|
+
const event1 = createAgentSpawnedEvent('agent-1', 'coder', 'core', []);
|
|
51
|
+
const event2 = createAgentStartedEvent('agent-1');
|
|
52
|
+
|
|
53
|
+
await eventStore.append(event1);
|
|
54
|
+
await eventStore.append(event2);
|
|
55
|
+
|
|
56
|
+
const events = await eventStore.getEvents('agent-1');
|
|
57
|
+
expect(events[0].version).toBe(1);
|
|
58
|
+
expect(events[1].version).toBe(2);
|
|
59
|
+
});
|
|
60
|
+
|
|
61
|
+
it('should maintain version per aggregate', async () => {
|
|
62
|
+
const agent1Event = createAgentSpawnedEvent('agent-1', 'coder', 'core', []);
|
|
63
|
+
const agent2Event = createAgentSpawnedEvent('agent-2', 'tester', 'quality', []);
|
|
64
|
+
|
|
65
|
+
await eventStore.append(agent1Event);
|
|
66
|
+
await eventStore.append(agent2Event);
|
|
67
|
+
|
|
68
|
+
const agent1Events = await eventStore.getEvents('agent-1');
|
|
69
|
+
const agent2Events = await eventStore.getEvents('agent-2');
|
|
70
|
+
|
|
71
|
+
expect(agent1Events[0].version).toBe(1);
|
|
72
|
+
expect(agent2Events[0].version).toBe(1);
|
|
73
|
+
});
|
|
74
|
+
});
|
|
75
|
+
|
|
76
|
+
describe('Event Retrieval', () => {
|
|
77
|
+
beforeEach(async () => {
|
|
78
|
+
// Setup: Create multiple events
|
|
79
|
+
await eventStore.append(createAgentSpawnedEvent('agent-1', 'coder', 'core', []));
|
|
80
|
+
await eventStore.append(createAgentStartedEvent('agent-1'));
|
|
81
|
+
await eventStore.append(createTaskCreatedEvent('task-1', 'implementation', 'Build feature', 'Description', 'high', []));
|
|
82
|
+
await eventStore.append(createTaskStartedEvent('task-1', 'agent-1'));
|
|
83
|
+
});
|
|
84
|
+
|
|
85
|
+
it('should retrieve events by aggregate ID', async () => {
|
|
86
|
+
const events = await eventStore.getEvents('agent-1');
|
|
87
|
+
expect(events).toHaveLength(2);
|
|
88
|
+
expect(events[0].type).toBe('agent:spawned');
|
|
89
|
+
expect(events[1].type).toBe('agent:started');
|
|
90
|
+
});
|
|
91
|
+
|
|
92
|
+
it('should retrieve events from specific version', async () => {
|
|
93
|
+
const events = await eventStore.getEvents('agent-1', 2);
|
|
94
|
+
expect(events).toHaveLength(1);
|
|
95
|
+
expect(events[0].type).toBe('agent:started');
|
|
96
|
+
});
|
|
97
|
+
|
|
98
|
+
it('should retrieve events by type', async () => {
|
|
99
|
+
const events = await eventStore.getEventsByType('task:created');
|
|
100
|
+
expect(events).toHaveLength(1);
|
|
101
|
+
expect(events[0].aggregateId).toBe('task-1');
|
|
102
|
+
});
|
|
103
|
+
});
|
|
104
|
+
|
|
105
|
+
describe('Event Filtering', () => {
|
|
106
|
+
beforeEach(async () => {
|
|
107
|
+
await eventStore.append(createAgentSpawnedEvent('agent-1', 'coder', 'core', []));
|
|
108
|
+
await eventStore.append(createAgentSpawnedEvent('agent-2', 'tester', 'quality', []));
|
|
109
|
+
await eventStore.append(createTaskCreatedEvent('task-1', 'implementation', 'Task 1', 'Desc', 'high', []));
|
|
110
|
+
await eventStore.append(createMemoryStoredEvent('mem-1', 'default', 'key1', 'semantic', 1024));
|
|
111
|
+
});
|
|
112
|
+
|
|
113
|
+
it('should filter by aggregate types', async () => {
|
|
114
|
+
const events = await eventStore.query({ aggregateTypes: ['agent'] });
|
|
115
|
+
expect(events).toHaveLength(2);
|
|
116
|
+
expect(events.every((e) => e.aggregateType === 'agent')).toBe(true);
|
|
117
|
+
});
|
|
118
|
+
|
|
119
|
+
it('should filter by event types', async () => {
|
|
120
|
+
const events = await eventStore.query({ eventTypes: ['agent:spawned'] });
|
|
121
|
+
expect(events).toHaveLength(2);
|
|
122
|
+
expect(events.every((e) => e.type === 'agent:spawned')).toBe(true);
|
|
123
|
+
});
|
|
124
|
+
|
|
125
|
+
it('should filter by aggregate IDs', async () => {
|
|
126
|
+
const events = await eventStore.query({ aggregateIds: ['agent-1', 'task-1'] });
|
|
127
|
+
expect(events).toHaveLength(2);
|
|
128
|
+
});
|
|
129
|
+
|
|
130
|
+
it('should support pagination', async () => {
|
|
131
|
+
const page1 = await eventStore.query({ limit: 2, offset: 0 });
|
|
132
|
+
const page2 = await eventStore.query({ limit: 2, offset: 2 });
|
|
133
|
+
|
|
134
|
+
expect(page1).toHaveLength(2);
|
|
135
|
+
expect(page2).toHaveLength(2);
|
|
136
|
+
expect(page1[0].id).not.toBe(page2[0].id);
|
|
137
|
+
});
|
|
138
|
+
});
|
|
139
|
+
|
|
140
|
+
describe('Event Replay', () => {
|
|
141
|
+
it('should replay all events', async () => {
|
|
142
|
+
await eventStore.append(createAgentSpawnedEvent('agent-1', 'coder', 'core', []));
|
|
143
|
+
await eventStore.append(createAgentStartedEvent('agent-1'));
|
|
144
|
+
await eventStore.append(createTaskCreatedEvent('task-1', 'implementation', 'Task', 'Desc', 'high', []));
|
|
145
|
+
|
|
146
|
+
const events: any[] = [];
|
|
147
|
+
for await (const event of eventStore.replay()) {
|
|
148
|
+
events.push(event);
|
|
149
|
+
}
|
|
150
|
+
|
|
151
|
+
expect(events).toHaveLength(3);
|
|
152
|
+
});
|
|
153
|
+
|
|
154
|
+
it('should replay from specific version', async () => {
|
|
155
|
+
await eventStore.append(createAgentSpawnedEvent('agent-1', 'coder', 'core', []));
|
|
156
|
+
await eventStore.append(createAgentStartedEvent('agent-1'));
|
|
157
|
+
await eventStore.append(createTaskCreatedEvent('task-1', 'implementation', 'Task', 'Desc', 'high', []));
|
|
158
|
+
|
|
159
|
+
const events: any[] = [];
|
|
160
|
+
for await (const event of eventStore.replay(2)) {
|
|
161
|
+
events.push(event);
|
|
162
|
+
}
|
|
163
|
+
|
|
164
|
+
expect(events.length).toBeGreaterThanOrEqual(2);
|
|
165
|
+
});
|
|
166
|
+
});
|
|
167
|
+
|
|
168
|
+
describe('Snapshots', () => {
|
|
169
|
+
it('should save snapshots', async () => {
|
|
170
|
+
const snapshot = {
|
|
171
|
+
aggregateId: 'agent-1',
|
|
172
|
+
aggregateType: 'agent' as const,
|
|
173
|
+
version: 5,
|
|
174
|
+
state: { status: 'active', tasks: ['task-1'] },
|
|
175
|
+
timestamp: Date.now(),
|
|
176
|
+
};
|
|
177
|
+
|
|
178
|
+
await eventStore.saveSnapshot(snapshot);
|
|
179
|
+
|
|
180
|
+
const retrieved = await eventStore.getSnapshot('agent-1');
|
|
181
|
+
expect(retrieved).not.toBeNull();
|
|
182
|
+
expect(retrieved?.version).toBe(5);
|
|
183
|
+
expect(retrieved?.state).toEqual({ status: 'active', tasks: ['task-1'] });
|
|
184
|
+
});
|
|
185
|
+
|
|
186
|
+
it('should return latest snapshot', async () => {
|
|
187
|
+
const snapshot1 = {
|
|
188
|
+
aggregateId: 'agent-1',
|
|
189
|
+
aggregateType: 'agent' as const,
|
|
190
|
+
version: 5,
|
|
191
|
+
state: { status: 'active' },
|
|
192
|
+
timestamp: Date.now(),
|
|
193
|
+
};
|
|
194
|
+
|
|
195
|
+
const snapshot2 = {
|
|
196
|
+
aggregateId: 'agent-1',
|
|
197
|
+
aggregateType: 'agent' as const,
|
|
198
|
+
version: 10,
|
|
199
|
+
state: { status: 'completed' },
|
|
200
|
+
timestamp: Date.now(),
|
|
201
|
+
};
|
|
202
|
+
|
|
203
|
+
await eventStore.saveSnapshot(snapshot1);
|
|
204
|
+
await eventStore.saveSnapshot(snapshot2);
|
|
205
|
+
|
|
206
|
+
const retrieved = await eventStore.getSnapshot('agent-1');
|
|
207
|
+
expect(retrieved?.version).toBe(10);
|
|
208
|
+
});
|
|
209
|
+
});
|
|
210
|
+
|
|
211
|
+
describe('Statistics', () => {
|
|
212
|
+
it('should provide event store statistics', async () => {
|
|
213
|
+
await eventStore.append(createAgentSpawnedEvent('agent-1', 'coder', 'core', []));
|
|
214
|
+
await eventStore.append(createAgentSpawnedEvent('agent-2', 'tester', 'quality', []));
|
|
215
|
+
await eventStore.append(createTaskCreatedEvent('task-1', 'implementation', 'Task', 'Desc', 'high', []));
|
|
216
|
+
|
|
217
|
+
const stats = await eventStore.getStats();
|
|
218
|
+
|
|
219
|
+
expect(stats.totalEvents).toBe(3);
|
|
220
|
+
expect(stats.eventsByType['agent:spawned']).toBe(2);
|
|
221
|
+
expect(stats.eventsByType['task:created']).toBe(1);
|
|
222
|
+
expect(stats.eventsByAggregate['agent-1']).toBe(1);
|
|
223
|
+
expect(stats.eventsByAggregate['agent-2']).toBe(1);
|
|
224
|
+
});
|
|
225
|
+
});
|
|
226
|
+
});
|
|
227
|
+
|
|
228
|
+
describe('AgentStateProjection', () => {
|
|
229
|
+
let eventStore: EventStore;
|
|
230
|
+
let projection: AgentStateProjection;
|
|
231
|
+
|
|
232
|
+
beforeEach(async () => {
|
|
233
|
+
eventStore = new EventStore({ databasePath: ':memory:' });
|
|
234
|
+
await eventStore.initialize();
|
|
235
|
+
|
|
236
|
+
projection = new AgentStateProjection(eventStore);
|
|
237
|
+
});
|
|
238
|
+
|
|
239
|
+
afterEach(async () => {
|
|
240
|
+
await eventStore.shutdown();
|
|
241
|
+
});
|
|
242
|
+
|
|
243
|
+
it('should build agent state from events', async () => {
|
|
244
|
+
await eventStore.append(createAgentSpawnedEvent('agent-1', 'coder', 'core', ['coding']));
|
|
245
|
+
await eventStore.append(createAgentStartedEvent('agent-1'));
|
|
246
|
+
|
|
247
|
+
await projection.initialize();
|
|
248
|
+
|
|
249
|
+
const agent = projection.getAgent('agent-1');
|
|
250
|
+
expect(agent).not.toBeNull();
|
|
251
|
+
expect(agent?.role).toBe('coder');
|
|
252
|
+
expect(agent?.status).toBe('active');
|
|
253
|
+
});
|
|
254
|
+
|
|
255
|
+
it('should track task assignments', async () => {
|
|
256
|
+
await eventStore.append(createAgentSpawnedEvent('agent-1', 'coder', 'core', []));
|
|
257
|
+
await eventStore.append(createAgentTaskAssignedEvent('agent-1', 'task-1', Date.now()));
|
|
258
|
+
|
|
259
|
+
await projection.initialize();
|
|
260
|
+
|
|
261
|
+
const agent = projection.getAgent('agent-1');
|
|
262
|
+
expect(agent?.currentTask).toBe('task-1');
|
|
263
|
+
});
|
|
264
|
+
|
|
265
|
+
it('should track completed tasks', async () => {
|
|
266
|
+
await eventStore.append(createAgentSpawnedEvent('agent-1', 'coder', 'core', []));
|
|
267
|
+
await eventStore.append(createAgentTaskCompletedEvent('agent-1', 'task-1', { success: true }, Date.now(), 5000));
|
|
268
|
+
|
|
269
|
+
await projection.initialize();
|
|
270
|
+
|
|
271
|
+
const agent = projection.getAgent('agent-1');
|
|
272
|
+
expect(agent?.completedTasks).toContain('task-1');
|
|
273
|
+
expect(agent?.taskCount).toBe(1);
|
|
274
|
+
});
|
|
275
|
+
|
|
276
|
+
it('should filter agents by status', async () => {
|
|
277
|
+
await eventStore.append(createAgentSpawnedEvent('agent-1', 'coder', 'core', []));
|
|
278
|
+
await eventStore.append(createAgentSpawnedEvent('agent-2', 'tester', 'quality', []));
|
|
279
|
+
await eventStore.append(createAgentStartedEvent('agent-1'));
|
|
280
|
+
|
|
281
|
+
await projection.initialize();
|
|
282
|
+
|
|
283
|
+
const activeAgents = projection.getAgentsByStatus('active');
|
|
284
|
+
const idleAgents = projection.getAgentsByStatus('idle');
|
|
285
|
+
|
|
286
|
+
expect(activeAgents).toHaveLength(1);
|
|
287
|
+
expect(idleAgents).toHaveLength(1);
|
|
288
|
+
});
|
|
289
|
+
});
|
|
290
|
+
|
|
291
|
+
describe('TaskHistoryProjection', () => {
|
|
292
|
+
let eventStore: EventStore;
|
|
293
|
+
let projection: TaskHistoryProjection;
|
|
294
|
+
|
|
295
|
+
beforeEach(async () => {
|
|
296
|
+
eventStore = new EventStore({ databasePath: ':memory:' });
|
|
297
|
+
await eventStore.initialize();
|
|
298
|
+
|
|
299
|
+
projection = new TaskHistoryProjection(eventStore);
|
|
300
|
+
});
|
|
301
|
+
|
|
302
|
+
afterEach(async () => {
|
|
303
|
+
await eventStore.shutdown();
|
|
304
|
+
});
|
|
305
|
+
|
|
306
|
+
it('should build task history from events', async () => {
|
|
307
|
+
await eventStore.append(createTaskCreatedEvent('task-1', 'implementation', 'Build Feature', 'Description', 'high', []));
|
|
308
|
+
await eventStore.append(createTaskStartedEvent('task-1', 'agent-1'));
|
|
309
|
+
await eventStore.append(createTaskCompletedEvent('task-1', { success: true }, 5000));
|
|
310
|
+
|
|
311
|
+
await projection.initialize();
|
|
312
|
+
|
|
313
|
+
const task = projection.getTask('task-1');
|
|
314
|
+
expect(task).not.toBeNull();
|
|
315
|
+
expect(task?.title).toBe('Build Feature');
|
|
316
|
+
expect(task?.status).toBe('completed');
|
|
317
|
+
expect(task?.assignedAgent).toBe('agent-1');
|
|
318
|
+
});
|
|
319
|
+
|
|
320
|
+
it('should calculate average task duration', async () => {
|
|
321
|
+
await eventStore.append(createTaskCreatedEvent('task-1', 'implementation', 'Task 1', 'Desc', 'high', []));
|
|
322
|
+
await eventStore.append(createTaskCompletedEvent('task-1', { success: true }, 5000));
|
|
323
|
+
|
|
324
|
+
await eventStore.append(createTaskCreatedEvent('task-2', 'implementation', 'Task 2', 'Desc', 'high', []));
|
|
325
|
+
await eventStore.append(createTaskCompletedEvent('task-2', { success: true }, 3000));
|
|
326
|
+
|
|
327
|
+
await projection.initialize();
|
|
328
|
+
|
|
329
|
+
const avgDuration = projection.getAverageTaskDuration();
|
|
330
|
+
expect(avgDuration).toBe(4000);
|
|
331
|
+
});
|
|
332
|
+
});
|
|
333
|
+
|
|
334
|
+
describe('MemoryIndexProjection', () => {
|
|
335
|
+
let eventStore: EventStore;
|
|
336
|
+
let projection: MemoryIndexProjection;
|
|
337
|
+
|
|
338
|
+
beforeEach(async () => {
|
|
339
|
+
eventStore = new EventStore({ databasePath: ':memory:' });
|
|
340
|
+
await eventStore.initialize();
|
|
341
|
+
|
|
342
|
+
projection = new MemoryIndexProjection(eventStore);
|
|
343
|
+
});
|
|
344
|
+
|
|
345
|
+
afterEach(async () => {
|
|
346
|
+
await eventStore.shutdown();
|
|
347
|
+
});
|
|
348
|
+
|
|
349
|
+
it('should build memory index from events', async () => {
|
|
350
|
+
await eventStore.append(createMemoryStoredEvent('mem-1', 'default', 'key1', 'semantic', 1024));
|
|
351
|
+
await eventStore.append(createMemoryRetrievedEvent('mem-1', 'default', 'key1', 1));
|
|
352
|
+
|
|
353
|
+
await projection.initialize();
|
|
354
|
+
|
|
355
|
+
const memory = projection.getMemory('mem-1');
|
|
356
|
+
expect(memory).not.toBeNull();
|
|
357
|
+
expect(memory?.namespace).toBe('default');
|
|
358
|
+
expect(memory?.accessCount).toBe(1);
|
|
359
|
+
});
|
|
360
|
+
|
|
361
|
+
it('should track memory deletions', async () => {
|
|
362
|
+
await eventStore.append(createMemoryStoredEvent('mem-1', 'default', 'key1', 'semantic', 1024));
|
|
363
|
+
await eventStore.append(createMemoryDeletedEvent('mem-1', 'default', 'key1'));
|
|
364
|
+
|
|
365
|
+
await projection.initialize();
|
|
366
|
+
|
|
367
|
+
const memory = projection.getMemory('mem-1');
|
|
368
|
+
expect(memory?.isDeleted).toBe(true);
|
|
369
|
+
|
|
370
|
+
const activeMemories = projection.getActiveMemories();
|
|
371
|
+
expect(activeMemories).toHaveLength(0);
|
|
372
|
+
});
|
|
373
|
+
|
|
374
|
+
it('should calculate total size by namespace', async () => {
|
|
375
|
+
await eventStore.append(createMemoryStoredEvent('mem-1', 'ns1', 'key1', 'semantic', 1024));
|
|
376
|
+
await eventStore.append(createMemoryStoredEvent('mem-2', 'ns1', 'key2', 'semantic', 2048));
|
|
377
|
+
await eventStore.append(createMemoryStoredEvent('mem-3', 'ns2', 'key3', 'semantic', 512));
|
|
378
|
+
|
|
379
|
+
await projection.initialize();
|
|
380
|
+
|
|
381
|
+
const ns1Size = projection.getTotalSizeByNamespace('ns1');
|
|
382
|
+
const ns2Size = projection.getTotalSizeByNamespace('ns2');
|
|
383
|
+
|
|
384
|
+
expect(ns1Size).toBe(3072);
|
|
385
|
+
expect(ns2Size).toBe(512);
|
|
386
|
+
});
|
|
387
|
+
});
|