moflo 4.8.19 → 4.8.20
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/.claude/guidance/shipped/moflo.md +45 -0
- package/.claude/helpers/statusline.cjs +1 -1
- package/.claude/workflow-state.json +9 -0
- package/package.json +2 -2
- package/src/@claude-flow/cli/dist/src/init/statusline-generator.js +1 -1
- package/src/@claude-flow/cli/dist/src/services/agentic-flow-bridge.js +5 -3
- package/src/@claude-flow/cli/package.json +1 -1
- package/src/@claude-flow/memory/dist/agent-memory-scope.d.ts +131 -0
- package/src/@claude-flow/memory/dist/agent-memory-scope.js +223 -0
- package/src/@claude-flow/memory/dist/agent-memory-scope.test.d.ts +8 -0
- package/src/@claude-flow/memory/dist/agent-memory-scope.test.js +466 -0
- package/src/@claude-flow/memory/dist/agentdb-adapter.d.ts +165 -0
- package/src/@claude-flow/memory/dist/agentdb-adapter.js +806 -0
- package/src/@claude-flow/memory/dist/agentdb-backend.d.ts +212 -0
- package/src/@claude-flow/memory/dist/agentdb-backend.js +842 -0
- package/src/@claude-flow/memory/dist/agentdb-backend.test.d.ts +7 -0
- package/src/@claude-flow/memory/dist/agentdb-backend.test.js +258 -0
- package/src/@claude-flow/memory/dist/application/commands/delete-memory.command.d.ts +65 -0
- package/src/@claude-flow/memory/dist/application/commands/delete-memory.command.js +129 -0
- package/src/@claude-flow/memory/dist/application/commands/store-memory.command.d.ts +48 -0
- package/src/@claude-flow/memory/dist/application/commands/store-memory.command.js +72 -0
- package/src/@claude-flow/memory/dist/application/index.d.ts +12 -0
- package/src/@claude-flow/memory/dist/application/index.js +15 -0
- package/src/@claude-flow/memory/dist/application/queries/search-memory.query.d.ts +72 -0
- package/src/@claude-flow/memory/dist/application/queries/search-memory.query.js +143 -0
- package/src/@claude-flow/memory/dist/application/services/memory-application-service.d.ts +121 -0
- package/src/@claude-flow/memory/dist/application/services/memory-application-service.js +190 -0
- package/src/@claude-flow/memory/dist/auto-memory-bridge.d.ts +226 -0
- package/src/@claude-flow/memory/dist/auto-memory-bridge.js +709 -0
- package/src/@claude-flow/memory/dist/auto-memory-bridge.test.d.ts +8 -0
- package/src/@claude-flow/memory/dist/auto-memory-bridge.test.js +757 -0
- package/src/@claude-flow/memory/dist/benchmark.test.d.ts +2 -0
- package/src/@claude-flow/memory/dist/benchmark.test.js +277 -0
- package/src/@claude-flow/memory/dist/cache-manager.d.ts +134 -0
- package/src/@claude-flow/memory/dist/cache-manager.js +407 -0
- package/src/@claude-flow/memory/dist/controller-registry.d.ts +216 -0
- package/src/@claude-flow/memory/dist/controller-registry.js +893 -0
- package/src/@claude-flow/memory/dist/controller-registry.test.d.ts +14 -0
- package/src/@claude-flow/memory/dist/controller-registry.test.js +593 -0
- package/src/@claude-flow/memory/dist/database-provider.d.ts +87 -0
- package/src/@claude-flow/memory/dist/database-provider.js +372 -0
- package/src/@claude-flow/memory/dist/database-provider.test.d.ts +7 -0
- package/src/@claude-flow/memory/dist/database-provider.test.js +287 -0
- package/src/@claude-flow/memory/dist/domain/entities/memory-entry.d.ts +143 -0
- package/src/@claude-flow/memory/dist/domain/entities/memory-entry.js +226 -0
- package/src/@claude-flow/memory/dist/domain/index.d.ts +11 -0
- package/src/@claude-flow/memory/dist/domain/index.js +12 -0
- package/src/@claude-flow/memory/dist/domain/repositories/memory-repository.interface.d.ts +102 -0
- package/src/@claude-flow/memory/dist/domain/repositories/memory-repository.interface.js +11 -0
- package/src/@claude-flow/memory/dist/domain/services/memory-domain-service.d.ts +105 -0
- package/src/@claude-flow/memory/dist/domain/services/memory-domain-service.js +297 -0
- package/src/@claude-flow/memory/dist/hnsw-index.d.ts +111 -0
- package/src/@claude-flow/memory/dist/hnsw-index.js +781 -0
- package/src/@claude-flow/memory/dist/hnsw-lite.d.ts +23 -0
- package/src/@claude-flow/memory/dist/hnsw-lite.js +168 -0
- package/src/@claude-flow/memory/dist/index.d.ts +204 -0
- package/src/@claude-flow/memory/dist/index.js +358 -0
- package/src/@claude-flow/memory/dist/infrastructure/index.d.ts +17 -0
- package/src/@claude-flow/memory/dist/infrastructure/index.js +16 -0
- package/src/@claude-flow/memory/dist/infrastructure/repositories/hybrid-memory-repository.d.ts +66 -0
- package/src/@claude-flow/memory/dist/infrastructure/repositories/hybrid-memory-repository.js +409 -0
- package/src/@claude-flow/memory/dist/learning-bridge.d.ts +137 -0
- package/src/@claude-flow/memory/dist/learning-bridge.js +335 -0
- package/src/@claude-flow/memory/dist/learning-bridge.test.d.ts +8 -0
- package/src/@claude-flow/memory/dist/learning-bridge.test.js +578 -0
- package/src/@claude-flow/memory/dist/memory-graph.d.ts +100 -0
- package/src/@claude-flow/memory/dist/memory-graph.js +333 -0
- package/src/@claude-flow/memory/dist/memory-graph.test.d.ts +8 -0
- package/src/@claude-flow/memory/dist/memory-graph.test.js +609 -0
- package/src/@claude-flow/memory/dist/migration.d.ts +68 -0
- package/src/@claude-flow/memory/dist/migration.js +513 -0
- package/src/@claude-flow/memory/dist/persistent-sona.d.ts +144 -0
- package/src/@claude-flow/memory/dist/persistent-sona.js +332 -0
- package/src/@claude-flow/memory/dist/query-builder.d.ts +211 -0
- package/src/@claude-flow/memory/dist/query-builder.js +438 -0
- package/src/@claude-flow/memory/dist/rvf-backend.d.ts +51 -0
- package/src/@claude-flow/memory/dist/rvf-backend.js +481 -0
- package/src/@claude-flow/memory/dist/rvf-learning-store.d.ts +139 -0
- package/src/@claude-flow/memory/dist/rvf-learning-store.js +295 -0
- package/src/@claude-flow/memory/dist/rvf-migration.d.ts +45 -0
- package/src/@claude-flow/memory/dist/rvf-migration.js +234 -0
- package/src/@claude-flow/memory/dist/sqljs-backend.d.ts +127 -0
- package/src/@claude-flow/memory/dist/sqljs-backend.js +600 -0
- package/src/@claude-flow/memory/dist/types.d.ts +484 -0
- package/src/@claude-flow/memory/dist/types.js +58 -0
- package/src/@claude-flow/shared/dist/core/config/defaults.d.ts +41 -0
- package/src/@claude-flow/shared/dist/core/config/defaults.js +186 -0
- package/src/@claude-flow/shared/dist/core/config/index.d.ts +8 -0
- package/src/@claude-flow/shared/dist/core/config/index.js +12 -0
- package/src/@claude-flow/shared/dist/core/config/loader.d.ts +45 -0
- package/src/@claude-flow/shared/dist/core/config/loader.js +222 -0
- package/src/@claude-flow/shared/dist/core/config/schema.d.ts +1134 -0
- package/src/@claude-flow/shared/dist/core/config/schema.js +158 -0
- package/src/@claude-flow/shared/dist/core/config/validator.d.ts +92 -0
- package/src/@claude-flow/shared/dist/core/config/validator.js +147 -0
- package/src/@claude-flow/shared/dist/core/event-bus.d.ts +31 -0
- package/src/@claude-flow/shared/dist/core/event-bus.js +197 -0
- package/src/@claude-flow/shared/dist/core/index.d.ts +15 -0
- package/src/@claude-flow/shared/dist/core/index.js +19 -0
- package/src/@claude-flow/shared/dist/core/interfaces/agent.interface.d.ts +200 -0
- package/src/@claude-flow/shared/dist/core/interfaces/agent.interface.js +6 -0
- package/src/@claude-flow/shared/dist/core/interfaces/coordinator.interface.d.ts +310 -0
- package/src/@claude-flow/shared/dist/core/interfaces/coordinator.interface.js +7 -0
- package/src/@claude-flow/shared/dist/core/interfaces/event.interface.d.ts +224 -0
- package/src/@claude-flow/shared/dist/core/interfaces/event.interface.js +46 -0
- package/src/@claude-flow/shared/dist/core/interfaces/index.d.ts +10 -0
- package/src/@claude-flow/shared/dist/core/interfaces/index.js +15 -0
- package/src/@claude-flow/shared/dist/core/interfaces/memory.interface.d.ts +298 -0
- package/src/@claude-flow/shared/dist/core/interfaces/memory.interface.js +7 -0
- package/src/@claude-flow/shared/dist/core/interfaces/task.interface.d.ts +185 -0
- package/src/@claude-flow/shared/dist/core/interfaces/task.interface.js +6 -0
- package/src/@claude-flow/shared/dist/core/orchestrator/event-coordinator.d.ts +35 -0
- package/src/@claude-flow/shared/dist/core/orchestrator/event-coordinator.js +101 -0
- package/src/@claude-flow/shared/dist/core/orchestrator/health-monitor.d.ts +60 -0
- package/src/@claude-flow/shared/dist/core/orchestrator/health-monitor.js +166 -0
- package/src/@claude-flow/shared/dist/core/orchestrator/index.d.ts +46 -0
- package/src/@claude-flow/shared/dist/core/orchestrator/index.js +64 -0
- package/src/@claude-flow/shared/dist/core/orchestrator/lifecycle-manager.d.ts +56 -0
- package/src/@claude-flow/shared/dist/core/orchestrator/lifecycle-manager.js +195 -0
- package/src/@claude-flow/shared/dist/core/orchestrator/session-manager.d.ts +83 -0
- package/src/@claude-flow/shared/dist/core/orchestrator/session-manager.js +193 -0
- package/src/@claude-flow/shared/dist/core/orchestrator/task-manager.d.ts +49 -0
- package/src/@claude-flow/shared/dist/core/orchestrator/task-manager.js +253 -0
- package/src/@claude-flow/shared/dist/events/domain-events.d.ts +282 -0
- package/src/@claude-flow/shared/dist/events/domain-events.js +165 -0
- package/src/@claude-flow/shared/dist/events/event-store.d.ts +126 -0
- package/src/@claude-flow/shared/dist/events/event-store.js +432 -0
- package/src/@claude-flow/shared/dist/events/event-store.test.d.ts +8 -0
- package/src/@claude-flow/shared/dist/events/event-store.test.js +297 -0
- package/src/@claude-flow/shared/dist/events/example-usage.d.ts +10 -0
- package/src/@claude-flow/shared/dist/events/example-usage.js +193 -0
- package/src/@claude-flow/shared/dist/events/index.d.ts +21 -0
- package/src/@claude-flow/shared/dist/events/index.js +22 -0
- package/src/@claude-flow/shared/dist/events/projections.d.ts +177 -0
- package/src/@claude-flow/shared/dist/events/projections.js +421 -0
- package/src/@claude-flow/shared/dist/events/rvf-event-log.d.ts +82 -0
- package/src/@claude-flow/shared/dist/events/rvf-event-log.js +340 -0
- package/src/@claude-flow/shared/dist/events/state-reconstructor.d.ts +101 -0
- package/src/@claude-flow/shared/dist/events/state-reconstructor.js +263 -0
- package/src/@claude-flow/shared/dist/events.d.ts +80 -0
- package/src/@claude-flow/shared/dist/events.js +249 -0
- package/src/@claude-flow/shared/dist/hooks/example-usage.d.ts +42 -0
- package/src/@claude-flow/shared/dist/hooks/example-usage.js +351 -0
- package/src/@claude-flow/shared/dist/hooks/executor.d.ts +100 -0
- package/src/@claude-flow/shared/dist/hooks/executor.js +267 -0
- package/src/@claude-flow/shared/dist/hooks/hooks.test.d.ts +9 -0
- package/src/@claude-flow/shared/dist/hooks/hooks.test.js +322 -0
- package/src/@claude-flow/shared/dist/hooks/index.d.ts +52 -0
- package/src/@claude-flow/shared/dist/hooks/index.js +51 -0
- package/src/@claude-flow/shared/dist/hooks/registry.d.ts +133 -0
- package/src/@claude-flow/shared/dist/hooks/registry.js +277 -0
- package/src/@claude-flow/shared/dist/hooks/safety/bash-safety.d.ts +105 -0
- package/src/@claude-flow/shared/dist/hooks/safety/bash-safety.js +481 -0
- package/src/@claude-flow/shared/dist/hooks/safety/file-organization.d.ts +144 -0
- package/src/@claude-flow/shared/dist/hooks/safety/file-organization.js +328 -0
- package/src/@claude-flow/shared/dist/hooks/safety/git-commit.d.ts +158 -0
- package/src/@claude-flow/shared/dist/hooks/safety/git-commit.js +450 -0
- package/src/@claude-flow/shared/dist/hooks/safety/index.d.ts +17 -0
- package/src/@claude-flow/shared/dist/hooks/safety/index.js +17 -0
- package/src/@claude-flow/shared/dist/hooks/session-hooks.d.ts +234 -0
- package/src/@claude-flow/shared/dist/hooks/session-hooks.js +334 -0
- package/src/@claude-flow/shared/dist/hooks/task-hooks.d.ts +163 -0
- package/src/@claude-flow/shared/dist/hooks/task-hooks.js +326 -0
- package/src/@claude-flow/shared/dist/hooks/types.d.ts +267 -0
- package/src/@claude-flow/shared/dist/hooks/types.js +62 -0
- package/src/@claude-flow/shared/dist/hooks/verify-exports.test.d.ts +9 -0
- package/src/@claude-flow/shared/dist/hooks/verify-exports.test.js +93 -0
- package/src/@claude-flow/shared/dist/index.d.ts +20 -0
- package/src/@claude-flow/shared/dist/index.js +50 -0
- package/src/@claude-flow/shared/dist/mcp/connection-pool.d.ts +98 -0
- package/src/@claude-flow/shared/dist/mcp/connection-pool.js +364 -0
- package/src/@claude-flow/shared/dist/mcp/index.d.ts +69 -0
- package/src/@claude-flow/shared/dist/mcp/index.js +84 -0
- package/src/@claude-flow/shared/dist/mcp/server.d.ts +166 -0
- package/src/@claude-flow/shared/dist/mcp/server.js +593 -0
- package/src/@claude-flow/shared/dist/mcp/session-manager.d.ts +136 -0
- package/src/@claude-flow/shared/dist/mcp/session-manager.js +335 -0
- package/src/@claude-flow/shared/dist/mcp/tool-registry.d.ts +178 -0
- package/src/@claude-flow/shared/dist/mcp/tool-registry.js +439 -0
- package/src/@claude-flow/shared/dist/mcp/transport/http.d.ts +104 -0
- package/src/@claude-flow/shared/dist/mcp/transport/http.js +476 -0
- package/src/@claude-flow/shared/dist/mcp/transport/index.d.ts +102 -0
- package/src/@claude-flow/shared/dist/mcp/transport/index.js +238 -0
- package/src/@claude-flow/shared/dist/mcp/transport/stdio.d.ts +104 -0
- package/src/@claude-flow/shared/dist/mcp/transport/stdio.js +263 -0
- package/src/@claude-flow/shared/dist/mcp/transport/websocket.d.ts +133 -0
- package/src/@claude-flow/shared/dist/mcp/transport/websocket.js +396 -0
- package/src/@claude-flow/shared/dist/mcp/types.d.ts +438 -0
- package/src/@claude-flow/shared/dist/mcp/types.js +54 -0
- package/src/@claude-flow/shared/dist/plugin-interface.d.ts +544 -0
- package/src/@claude-flow/shared/dist/plugin-interface.js +23 -0
- package/src/@claude-flow/shared/dist/plugin-loader.d.ts +139 -0
- package/src/@claude-flow/shared/dist/plugin-loader.js +434 -0
- package/src/@claude-flow/shared/dist/plugin-registry.d.ts +183 -0
- package/src/@claude-flow/shared/dist/plugin-registry.js +457 -0
- package/src/@claude-flow/shared/dist/plugins/index.d.ts +10 -0
- package/src/@claude-flow/shared/dist/plugins/index.js +10 -0
- package/src/@claude-flow/shared/dist/plugins/official/hive-mind-plugin.d.ts +106 -0
- package/src/@claude-flow/shared/dist/plugins/official/hive-mind-plugin.js +241 -0
- package/src/@claude-flow/shared/dist/plugins/official/index.d.ts +10 -0
- package/src/@claude-flow/shared/dist/plugins/official/index.js +10 -0
- package/src/@claude-flow/shared/dist/plugins/official/maestro-plugin.d.ts +121 -0
- package/src/@claude-flow/shared/dist/plugins/official/maestro-plugin.js +355 -0
- package/src/@claude-flow/shared/dist/plugins/types.d.ts +93 -0
- package/src/@claude-flow/shared/dist/plugins/types.js +9 -0
- package/src/@claude-flow/shared/dist/resilience/bulkhead.d.ts +105 -0
- package/src/@claude-flow/shared/dist/resilience/bulkhead.js +206 -0
- package/src/@claude-flow/shared/dist/resilience/circuit-breaker.d.ts +132 -0
- package/src/@claude-flow/shared/dist/resilience/circuit-breaker.js +233 -0
- package/src/@claude-flow/shared/dist/resilience/index.d.ts +19 -0
- package/src/@claude-flow/shared/dist/resilience/index.js +19 -0
- package/src/@claude-flow/shared/dist/resilience/rate-limiter.d.ts +168 -0
- package/src/@claude-flow/shared/dist/resilience/rate-limiter.js +314 -0
- package/src/@claude-flow/shared/dist/resilience/retry.d.ts +91 -0
- package/src/@claude-flow/shared/dist/resilience/retry.js +159 -0
- package/src/@claude-flow/shared/dist/security/index.d.ts +10 -0
- package/src/@claude-flow/shared/dist/security/index.js +12 -0
- package/src/@claude-flow/shared/dist/security/input-validation.d.ts +73 -0
- package/src/@claude-flow/shared/dist/security/input-validation.js +201 -0
- package/src/@claude-flow/shared/dist/security/secure-random.d.ts +92 -0
- package/src/@claude-flow/shared/dist/security/secure-random.js +142 -0
- package/src/@claude-flow/shared/dist/services/index.d.ts +7 -0
- package/src/@claude-flow/shared/dist/services/index.js +7 -0
- package/src/@claude-flow/shared/dist/services/v3-progress.service.d.ts +124 -0
- package/src/@claude-flow/shared/dist/services/v3-progress.service.js +402 -0
- package/src/@claude-flow/shared/dist/types/agent.types.d.ts +137 -0
- package/src/@claude-flow/shared/dist/types/agent.types.js +6 -0
- package/src/@claude-flow/shared/dist/types/index.d.ts +11 -0
- package/src/@claude-flow/shared/dist/types/index.js +17 -0
- package/src/@claude-flow/shared/dist/types/mcp.types.d.ts +266 -0
- package/src/@claude-flow/shared/dist/types/mcp.types.js +7 -0
- package/src/@claude-flow/shared/dist/types/memory.types.d.ts +236 -0
- package/src/@claude-flow/shared/dist/types/memory.types.js +7 -0
- package/src/@claude-flow/shared/dist/types/swarm.types.d.ts +186 -0
- package/src/@claude-flow/shared/dist/types/swarm.types.js +65 -0
- package/src/@claude-flow/shared/dist/types/task.types.d.ts +178 -0
- package/src/@claude-flow/shared/dist/types/task.types.js +32 -0
- package/src/@claude-flow/shared/dist/types.d.ts +197 -0
- package/src/@claude-flow/shared/dist/types.js +21 -0
- package/src/@claude-flow/shared/dist/utils/secure-logger.d.ts +69 -0
- package/src/@claude-flow/shared/dist/utils/secure-logger.js +208 -0
|
@@ -0,0 +1,297 @@
|
|
|
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
|
+
import { describe, it, expect, beforeEach, afterEach } from 'vitest';
|
|
8
|
+
import { EventStore } from './event-store.js';
|
|
9
|
+
import { AgentStateProjection, TaskHistoryProjection, MemoryIndexProjection } from './projections.js';
|
|
10
|
+
import { createAgentSpawnedEvent, createAgentStartedEvent, createAgentTaskAssignedEvent, createAgentTaskCompletedEvent, createTaskCreatedEvent, createTaskStartedEvent, createTaskCompletedEvent, createMemoryStoredEvent, createMemoryRetrievedEvent, createMemoryDeletedEvent, } from './domain-events.js';
|
|
11
|
+
describe('EventStore', () => {
|
|
12
|
+
let eventStore;
|
|
13
|
+
beforeEach(async () => {
|
|
14
|
+
eventStore = new EventStore({ databasePath: ':memory:', verbose: false });
|
|
15
|
+
await eventStore.initialize();
|
|
16
|
+
});
|
|
17
|
+
afterEach(async () => {
|
|
18
|
+
await eventStore.shutdown();
|
|
19
|
+
});
|
|
20
|
+
describe('Event Appending', () => {
|
|
21
|
+
it('should append events to the store', async () => {
|
|
22
|
+
const event = createAgentSpawnedEvent('agent-1', 'coder', 'core', ['coding', 'testing']);
|
|
23
|
+
await eventStore.append(event);
|
|
24
|
+
const events = await eventStore.getEvents('agent-1');
|
|
25
|
+
expect(events).toHaveLength(1);
|
|
26
|
+
expect(events[0].type).toBe('agent:spawned');
|
|
27
|
+
expect(events[0].version).toBe(1);
|
|
28
|
+
});
|
|
29
|
+
it('should increment version for each event', async () => {
|
|
30
|
+
const event1 = createAgentSpawnedEvent('agent-1', 'coder', 'core', []);
|
|
31
|
+
const event2 = createAgentStartedEvent('agent-1');
|
|
32
|
+
await eventStore.append(event1);
|
|
33
|
+
await eventStore.append(event2);
|
|
34
|
+
const events = await eventStore.getEvents('agent-1');
|
|
35
|
+
expect(events[0].version).toBe(1);
|
|
36
|
+
expect(events[1].version).toBe(2);
|
|
37
|
+
});
|
|
38
|
+
it('should maintain version per aggregate', async () => {
|
|
39
|
+
const agent1Event = createAgentSpawnedEvent('agent-1', 'coder', 'core', []);
|
|
40
|
+
const agent2Event = createAgentSpawnedEvent('agent-2', 'tester', 'quality', []);
|
|
41
|
+
await eventStore.append(agent1Event);
|
|
42
|
+
await eventStore.append(agent2Event);
|
|
43
|
+
const agent1Events = await eventStore.getEvents('agent-1');
|
|
44
|
+
const agent2Events = await eventStore.getEvents('agent-2');
|
|
45
|
+
expect(agent1Events[0].version).toBe(1);
|
|
46
|
+
expect(agent2Events[0].version).toBe(1);
|
|
47
|
+
});
|
|
48
|
+
});
|
|
49
|
+
describe('Event Retrieval', () => {
|
|
50
|
+
beforeEach(async () => {
|
|
51
|
+
// Setup: Create multiple events
|
|
52
|
+
await eventStore.append(createAgentSpawnedEvent('agent-1', 'coder', 'core', []));
|
|
53
|
+
await eventStore.append(createAgentStartedEvent('agent-1'));
|
|
54
|
+
await eventStore.append(createTaskCreatedEvent('task-1', 'implementation', 'Build feature', 'Description', 'high', []));
|
|
55
|
+
await eventStore.append(createTaskStartedEvent('task-1', 'agent-1'));
|
|
56
|
+
});
|
|
57
|
+
it('should retrieve events by aggregate ID', async () => {
|
|
58
|
+
const events = await eventStore.getEvents('agent-1');
|
|
59
|
+
expect(events).toHaveLength(2);
|
|
60
|
+
expect(events[0].type).toBe('agent:spawned');
|
|
61
|
+
expect(events[1].type).toBe('agent:started');
|
|
62
|
+
});
|
|
63
|
+
it('should retrieve events from specific version', async () => {
|
|
64
|
+
const events = await eventStore.getEvents('agent-1', 2);
|
|
65
|
+
expect(events).toHaveLength(1);
|
|
66
|
+
expect(events[0].type).toBe('agent:started');
|
|
67
|
+
});
|
|
68
|
+
it('should retrieve events by type', async () => {
|
|
69
|
+
const events = await eventStore.getEventsByType('task:created');
|
|
70
|
+
expect(events).toHaveLength(1);
|
|
71
|
+
expect(events[0].aggregateId).toBe('task-1');
|
|
72
|
+
});
|
|
73
|
+
});
|
|
74
|
+
describe('Event Filtering', () => {
|
|
75
|
+
beforeEach(async () => {
|
|
76
|
+
await eventStore.append(createAgentSpawnedEvent('agent-1', 'coder', 'core', []));
|
|
77
|
+
await eventStore.append(createAgentSpawnedEvent('agent-2', 'tester', 'quality', []));
|
|
78
|
+
await eventStore.append(createTaskCreatedEvent('task-1', 'implementation', 'Task 1', 'Desc', 'high', []));
|
|
79
|
+
await eventStore.append(createMemoryStoredEvent('mem-1', 'default', 'key1', 'semantic', 1024));
|
|
80
|
+
});
|
|
81
|
+
it('should filter by aggregate types', async () => {
|
|
82
|
+
const events = await eventStore.query({ aggregateTypes: ['agent'] });
|
|
83
|
+
expect(events).toHaveLength(2);
|
|
84
|
+
expect(events.every((e) => e.aggregateType === 'agent')).toBe(true);
|
|
85
|
+
});
|
|
86
|
+
it('should filter by event types', async () => {
|
|
87
|
+
const events = await eventStore.query({ eventTypes: ['agent:spawned'] });
|
|
88
|
+
expect(events).toHaveLength(2);
|
|
89
|
+
expect(events.every((e) => e.type === 'agent:spawned')).toBe(true);
|
|
90
|
+
});
|
|
91
|
+
it('should filter by aggregate IDs', async () => {
|
|
92
|
+
const events = await eventStore.query({ aggregateIds: ['agent-1', 'task-1'] });
|
|
93
|
+
expect(events).toHaveLength(2);
|
|
94
|
+
});
|
|
95
|
+
it('should support pagination', async () => {
|
|
96
|
+
const page1 = await eventStore.query({ limit: 2, offset: 0 });
|
|
97
|
+
const page2 = await eventStore.query({ limit: 2, offset: 2 });
|
|
98
|
+
expect(page1).toHaveLength(2);
|
|
99
|
+
expect(page2).toHaveLength(2);
|
|
100
|
+
expect(page1[0].id).not.toBe(page2[0].id);
|
|
101
|
+
});
|
|
102
|
+
});
|
|
103
|
+
describe('Event Replay', () => {
|
|
104
|
+
it('should replay all events', async () => {
|
|
105
|
+
await eventStore.append(createAgentSpawnedEvent('agent-1', 'coder', 'core', []));
|
|
106
|
+
await eventStore.append(createAgentStartedEvent('agent-1'));
|
|
107
|
+
await eventStore.append(createTaskCreatedEvent('task-1', 'implementation', 'Task', 'Desc', 'high', []));
|
|
108
|
+
const events = [];
|
|
109
|
+
for await (const event of eventStore.replay()) {
|
|
110
|
+
events.push(event);
|
|
111
|
+
}
|
|
112
|
+
expect(events).toHaveLength(3);
|
|
113
|
+
});
|
|
114
|
+
it('should replay from specific version', async () => {
|
|
115
|
+
// Create multiple events on the same aggregate so versions increment past 2
|
|
116
|
+
await eventStore.append(createAgentSpawnedEvent('agent-1', 'coder', 'core', [])); // v1
|
|
117
|
+
await eventStore.append(createAgentStartedEvent('agent-1')); // v2
|
|
118
|
+
await eventStore.append(createTaskCreatedEvent('task-1', 'implementation', 'Task', 'Desc', 'high', [])); // v1
|
|
119
|
+
const events = [];
|
|
120
|
+
for await (const event of eventStore.replay(2)) {
|
|
121
|
+
events.push(event);
|
|
122
|
+
}
|
|
123
|
+
// replay(2) returns events where per-aggregate version >= 2
|
|
124
|
+
// Only agent-1's second event (v2) qualifies
|
|
125
|
+
expect(events.length).toBeGreaterThanOrEqual(1);
|
|
126
|
+
expect(events[0].version).toBeGreaterThanOrEqual(2);
|
|
127
|
+
});
|
|
128
|
+
});
|
|
129
|
+
describe('Snapshots', () => {
|
|
130
|
+
it('should save snapshots', async () => {
|
|
131
|
+
const snapshot = {
|
|
132
|
+
aggregateId: 'agent-1',
|
|
133
|
+
aggregateType: 'agent',
|
|
134
|
+
version: 5,
|
|
135
|
+
state: { status: 'active', tasks: ['task-1'] },
|
|
136
|
+
timestamp: Date.now(),
|
|
137
|
+
};
|
|
138
|
+
await eventStore.saveSnapshot(snapshot);
|
|
139
|
+
const retrieved = await eventStore.getSnapshot('agent-1');
|
|
140
|
+
expect(retrieved).not.toBeNull();
|
|
141
|
+
expect(retrieved?.version).toBe(5);
|
|
142
|
+
expect(retrieved?.state).toEqual({ status: 'active', tasks: ['task-1'] });
|
|
143
|
+
});
|
|
144
|
+
it('should return latest snapshot', async () => {
|
|
145
|
+
const snapshot1 = {
|
|
146
|
+
aggregateId: 'agent-1',
|
|
147
|
+
aggregateType: 'agent',
|
|
148
|
+
version: 5,
|
|
149
|
+
state: { status: 'active' },
|
|
150
|
+
timestamp: Date.now(),
|
|
151
|
+
};
|
|
152
|
+
const snapshot2 = {
|
|
153
|
+
aggregateId: 'agent-1',
|
|
154
|
+
aggregateType: 'agent',
|
|
155
|
+
version: 10,
|
|
156
|
+
state: { status: 'completed' },
|
|
157
|
+
timestamp: Date.now(),
|
|
158
|
+
};
|
|
159
|
+
await eventStore.saveSnapshot(snapshot1);
|
|
160
|
+
await eventStore.saveSnapshot(snapshot2);
|
|
161
|
+
const retrieved = await eventStore.getSnapshot('agent-1');
|
|
162
|
+
expect(retrieved?.version).toBe(10);
|
|
163
|
+
});
|
|
164
|
+
});
|
|
165
|
+
describe('Statistics', () => {
|
|
166
|
+
it('should provide event store statistics', async () => {
|
|
167
|
+
await eventStore.append(createAgentSpawnedEvent('agent-1', 'coder', 'core', []));
|
|
168
|
+
await eventStore.append(createAgentSpawnedEvent('agent-2', 'tester', 'quality', []));
|
|
169
|
+
await eventStore.append(createTaskCreatedEvent('task-1', 'implementation', 'Task', 'Desc', 'high', []));
|
|
170
|
+
const stats = await eventStore.getStats();
|
|
171
|
+
expect(stats.totalEvents).toBe(3);
|
|
172
|
+
expect(stats.eventsByType['agent:spawned']).toBe(2);
|
|
173
|
+
expect(stats.eventsByType['task:created']).toBe(1);
|
|
174
|
+
expect(stats.eventsByAggregate['agent-1']).toBe(1);
|
|
175
|
+
expect(stats.eventsByAggregate['agent-2']).toBe(1);
|
|
176
|
+
});
|
|
177
|
+
});
|
|
178
|
+
});
|
|
179
|
+
describe('AgentStateProjection', () => {
|
|
180
|
+
let eventStore;
|
|
181
|
+
let projection;
|
|
182
|
+
beforeEach(async () => {
|
|
183
|
+
eventStore = new EventStore({ databasePath: ':memory:' });
|
|
184
|
+
await eventStore.initialize();
|
|
185
|
+
projection = new AgentStateProjection(eventStore);
|
|
186
|
+
});
|
|
187
|
+
afterEach(async () => {
|
|
188
|
+
await eventStore.shutdown();
|
|
189
|
+
});
|
|
190
|
+
it('should build agent state from events', async () => {
|
|
191
|
+
await eventStore.append(createAgentSpawnedEvent('agent-1', 'coder', 'core', ['coding']));
|
|
192
|
+
await eventStore.append(createAgentStartedEvent('agent-1'));
|
|
193
|
+
await projection.initialize();
|
|
194
|
+
const agent = projection.getAgent('agent-1');
|
|
195
|
+
expect(agent).not.toBeNull();
|
|
196
|
+
expect(agent?.role).toBe('coder');
|
|
197
|
+
expect(agent?.status).toBe('active');
|
|
198
|
+
});
|
|
199
|
+
it('should track task assignments', async () => {
|
|
200
|
+
await eventStore.append(createAgentSpawnedEvent('agent-1', 'coder', 'core', []));
|
|
201
|
+
await eventStore.append(createAgentTaskAssignedEvent('agent-1', 'task-1', Date.now()));
|
|
202
|
+
await projection.initialize();
|
|
203
|
+
const agent = projection.getAgent('agent-1');
|
|
204
|
+
expect(agent?.currentTask).toBe('task-1');
|
|
205
|
+
});
|
|
206
|
+
it('should track completed tasks', async () => {
|
|
207
|
+
await eventStore.append(createAgentSpawnedEvent('agent-1', 'coder', 'core', []));
|
|
208
|
+
await eventStore.append(createAgentTaskCompletedEvent('agent-1', 'task-1', { success: true }, Date.now(), 5000));
|
|
209
|
+
await projection.initialize();
|
|
210
|
+
const agent = projection.getAgent('agent-1');
|
|
211
|
+
expect(agent?.completedTasks).toContain('task-1');
|
|
212
|
+
expect(agent?.taskCount).toBe(1);
|
|
213
|
+
});
|
|
214
|
+
it('should filter agents by status', async () => {
|
|
215
|
+
await eventStore.append(createAgentSpawnedEvent('agent-1', 'coder', 'core', []));
|
|
216
|
+
await eventStore.append(createAgentSpawnedEvent('agent-2', 'tester', 'quality', []));
|
|
217
|
+
await eventStore.append(createAgentStartedEvent('agent-1'));
|
|
218
|
+
await projection.initialize();
|
|
219
|
+
const activeAgents = projection.getAgentsByStatus('active');
|
|
220
|
+
const idleAgents = projection.getAgentsByStatus('idle');
|
|
221
|
+
expect(activeAgents).toHaveLength(1);
|
|
222
|
+
expect(idleAgents).toHaveLength(1);
|
|
223
|
+
});
|
|
224
|
+
});
|
|
225
|
+
describe('TaskHistoryProjection', () => {
|
|
226
|
+
let eventStore;
|
|
227
|
+
let projection;
|
|
228
|
+
beforeEach(async () => {
|
|
229
|
+
eventStore = new EventStore({ databasePath: ':memory:' });
|
|
230
|
+
await eventStore.initialize();
|
|
231
|
+
projection = new TaskHistoryProjection(eventStore);
|
|
232
|
+
});
|
|
233
|
+
afterEach(async () => {
|
|
234
|
+
await eventStore.shutdown();
|
|
235
|
+
});
|
|
236
|
+
it('should build task history from events', async () => {
|
|
237
|
+
await eventStore.append(createTaskCreatedEvent('task-1', 'implementation', 'Build Feature', 'Description', 'high', []));
|
|
238
|
+
await eventStore.append(createTaskStartedEvent('task-1', 'agent-1'));
|
|
239
|
+
await eventStore.append(createTaskCompletedEvent('task-1', { success: true }, 5000));
|
|
240
|
+
await projection.initialize();
|
|
241
|
+
const task = projection.getTask('task-1');
|
|
242
|
+
expect(task).not.toBeNull();
|
|
243
|
+
expect(task?.title).toBe('Build Feature');
|
|
244
|
+
expect(task?.status).toBe('completed');
|
|
245
|
+
expect(task?.assignedAgent).toBe('agent-1');
|
|
246
|
+
});
|
|
247
|
+
it('should calculate average task duration', async () => {
|
|
248
|
+
await eventStore.append(createTaskCreatedEvent('task-1', 'implementation', 'Task 1', 'Desc', 'high', []));
|
|
249
|
+
await eventStore.append(createTaskCompletedEvent('task-1', { success: true }, 5000));
|
|
250
|
+
await eventStore.append(createTaskCreatedEvent('task-2', 'implementation', 'Task 2', 'Desc', 'high', []));
|
|
251
|
+
await eventStore.append(createTaskCompletedEvent('task-2', { success: true }, 3000));
|
|
252
|
+
await projection.initialize();
|
|
253
|
+
const avgDuration = projection.getAverageTaskDuration();
|
|
254
|
+
expect(avgDuration).toBe(4000);
|
|
255
|
+
});
|
|
256
|
+
});
|
|
257
|
+
describe('MemoryIndexProjection', () => {
|
|
258
|
+
let eventStore;
|
|
259
|
+
let projection;
|
|
260
|
+
beforeEach(async () => {
|
|
261
|
+
eventStore = new EventStore({ databasePath: ':memory:' });
|
|
262
|
+
await eventStore.initialize();
|
|
263
|
+
projection = new MemoryIndexProjection(eventStore);
|
|
264
|
+
});
|
|
265
|
+
afterEach(async () => {
|
|
266
|
+
await eventStore.shutdown();
|
|
267
|
+
});
|
|
268
|
+
it('should build memory index from events', async () => {
|
|
269
|
+
await eventStore.append(createMemoryStoredEvent('mem-1', 'default', 'key1', 'semantic', 1024));
|
|
270
|
+
await eventStore.append(createMemoryRetrievedEvent('mem-1', 'default', 'key1', 1));
|
|
271
|
+
await projection.initialize();
|
|
272
|
+
const memory = projection.getMemory('mem-1');
|
|
273
|
+
expect(memory).not.toBeNull();
|
|
274
|
+
expect(memory?.namespace).toBe('default');
|
|
275
|
+
expect(memory?.accessCount).toBe(1);
|
|
276
|
+
});
|
|
277
|
+
it('should track memory deletions', async () => {
|
|
278
|
+
await eventStore.append(createMemoryStoredEvent('mem-1', 'default', 'key1', 'semantic', 1024));
|
|
279
|
+
await eventStore.append(createMemoryDeletedEvent('mem-1', 'default', 'key1'));
|
|
280
|
+
await projection.initialize();
|
|
281
|
+
const memory = projection.getMemory('mem-1');
|
|
282
|
+
expect(memory?.isDeleted).toBe(true);
|
|
283
|
+
const activeMemories = projection.getActiveMemories();
|
|
284
|
+
expect(activeMemories).toHaveLength(0);
|
|
285
|
+
});
|
|
286
|
+
it('should calculate total size by namespace', async () => {
|
|
287
|
+
await eventStore.append(createMemoryStoredEvent('mem-1', 'ns1', 'key1', 'semantic', 1024));
|
|
288
|
+
await eventStore.append(createMemoryStoredEvent('mem-2', 'ns1', 'key2', 'semantic', 2048));
|
|
289
|
+
await eventStore.append(createMemoryStoredEvent('mem-3', 'ns2', 'key3', 'semantic', 512));
|
|
290
|
+
await projection.initialize();
|
|
291
|
+
const ns1Size = projection.getTotalSizeByNamespace('ns1');
|
|
292
|
+
const ns2Size = projection.getTotalSizeByNamespace('ns2');
|
|
293
|
+
expect(ns1Size).toBe(3072);
|
|
294
|
+
expect(ns2Size).toBe(512);
|
|
295
|
+
});
|
|
296
|
+
});
|
|
297
|
+
//# sourceMappingURL=event-store.test.js.map
|
|
@@ -0,0 +1,193 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Event Store Usage Example
|
|
3
|
+
*
|
|
4
|
+
* Demonstrates how to use the Event Sourcing system in V3 Claude Flow.
|
|
5
|
+
*
|
|
6
|
+
* Usage:
|
|
7
|
+
* npx tsx v3/@claude-flow/shared/src/events/example-usage.ts
|
|
8
|
+
*/
|
|
9
|
+
import { EventStore } from './event-store.js';
|
|
10
|
+
import { AgentStateProjection, TaskHistoryProjection, MemoryIndexProjection } from './projections.js';
|
|
11
|
+
import { createAgentSpawnedEvent, createAgentStartedEvent, createAgentTaskAssignedEvent, createAgentTaskCompletedEvent, createTaskCreatedEvent, createTaskStartedEvent, createTaskCompletedEvent, createMemoryStoredEvent, createMemoryRetrievedEvent, createSwarmInitializedEvent, createSwarmScaledEvent, } from './domain-events.js';
|
|
12
|
+
async function main() {
|
|
13
|
+
console.log('Event Sourcing Example - V3 Claude Flow\n');
|
|
14
|
+
// =========================================================================
|
|
15
|
+
// 1. Initialize Event Store
|
|
16
|
+
// =========================================================================
|
|
17
|
+
console.log('1. Initializing Event Store...');
|
|
18
|
+
const eventStore = new EventStore({
|
|
19
|
+
databasePath: './event-store-example.db',
|
|
20
|
+
verbose: true,
|
|
21
|
+
autoPersistInterval: 0, // Manual persist for demo
|
|
22
|
+
});
|
|
23
|
+
await eventStore.initialize();
|
|
24
|
+
console.log(' Event Store initialized\n');
|
|
25
|
+
// =========================================================================
|
|
26
|
+
// 2. Record Swarm Initialization Events
|
|
27
|
+
// =========================================================================
|
|
28
|
+
console.log('2. Recording Swarm Events...');
|
|
29
|
+
await eventStore.append(createSwarmInitializedEvent('hierarchical-mesh', 15, {
|
|
30
|
+
security: { strict: true },
|
|
31
|
+
memory: { backend: 'agentdb' },
|
|
32
|
+
}));
|
|
33
|
+
await eventStore.append(createSwarmScaledEvent(0, 5, 'Initial agent spawn'));
|
|
34
|
+
console.log(' Swarm events recorded\n');
|
|
35
|
+
// =========================================================================
|
|
36
|
+
// 3. Record Agent Lifecycle Events
|
|
37
|
+
// =========================================================================
|
|
38
|
+
console.log('3. Recording Agent Lifecycle Events...');
|
|
39
|
+
// Agent 1: Queen Coordinator
|
|
40
|
+
await eventStore.append(createAgentSpawnedEvent('agent-1', 'queen-coordinator', 'coordination', [
|
|
41
|
+
'orchestration',
|
|
42
|
+
'task-assignment',
|
|
43
|
+
]));
|
|
44
|
+
await eventStore.append(createAgentStartedEvent('agent-1'));
|
|
45
|
+
// Agent 2: Security Architect
|
|
46
|
+
await eventStore.append(createAgentSpawnedEvent('agent-2', 'security-architect', 'security', [
|
|
47
|
+
'threat-modeling',
|
|
48
|
+
'security-design',
|
|
49
|
+
]));
|
|
50
|
+
await eventStore.append(createAgentStartedEvent('agent-2'));
|
|
51
|
+
// Agent 3: Core Architect
|
|
52
|
+
await eventStore.append(createAgentSpawnedEvent('agent-3', 'core-architect', 'core', ['ddd-design', 'architecture']));
|
|
53
|
+
await eventStore.append(createAgentStartedEvent('agent-3'));
|
|
54
|
+
console.log(' Agent events recorded\n');
|
|
55
|
+
// =========================================================================
|
|
56
|
+
// 4. Record Task Execution Events
|
|
57
|
+
// =========================================================================
|
|
58
|
+
console.log('4. Recording Task Execution Events...');
|
|
59
|
+
// Task 1: Security Audit
|
|
60
|
+
await eventStore.append(createTaskCreatedEvent('task-1', 'security-audit', 'Audit CVE-1 (Command Injection)', 'Review and fix command injection vulnerability', 'critical', []));
|
|
61
|
+
await eventStore.append(createTaskStartedEvent('task-1', 'agent-2'));
|
|
62
|
+
await eventStore.append(createAgentTaskAssignedEvent('agent-2', 'task-1', Date.now()));
|
|
63
|
+
// Example: Async work gap between task start and completion
|
|
64
|
+
await new Promise((resolve) => setTimeout(resolve, 100));
|
|
65
|
+
await eventStore.append(createTaskCompletedEvent('task-1', { vulnerabilitiesFixed: 3, severity: 'high' }, 5234));
|
|
66
|
+
await eventStore.append(createAgentTaskCompletedEvent('agent-2', 'task-1', { success: true }, Date.now(), 5234));
|
|
67
|
+
// Task 2: DDD Architecture Design
|
|
68
|
+
await eventStore.append(createTaskCreatedEvent('task-2', 'architecture-design', 'Design V3 Domain Structure', 'Implement bounded contexts for V3', 'high', []));
|
|
69
|
+
await eventStore.append(createTaskStartedEvent('task-2', 'agent-3'));
|
|
70
|
+
await eventStore.append(createAgentTaskAssignedEvent('agent-3', 'task-2', Date.now()));
|
|
71
|
+
console.log(' Task events recorded\n');
|
|
72
|
+
// =========================================================================
|
|
73
|
+
// 5. Record Memory Operations Events
|
|
74
|
+
// =========================================================================
|
|
75
|
+
console.log('5. Recording Memory Operations Events...');
|
|
76
|
+
await eventStore.append(createMemoryStoredEvent('mem-1', 'agent-context', 'agent-2-state', 'episodic', 2048));
|
|
77
|
+
await eventStore.append(createMemoryStoredEvent('mem-2', 'task-results', 'task-1-result', 'semantic', 1024));
|
|
78
|
+
await eventStore.append(createMemoryRetrievedEvent('mem-1', 'agent-context', 'agent-2-state', 1));
|
|
79
|
+
await eventStore.append(createMemoryRetrievedEvent('mem-1', 'agent-context', 'agent-2-state', 2));
|
|
80
|
+
console.log(' Memory events recorded\n');
|
|
81
|
+
// =========================================================================
|
|
82
|
+
// 6. Query Events
|
|
83
|
+
// =========================================================================
|
|
84
|
+
console.log('6. Querying Events...\n');
|
|
85
|
+
const allAgentEvents = await eventStore.query({ aggregateTypes: ['agent'] });
|
|
86
|
+
console.log(` Total agent events: ${allAgentEvents.length}`);
|
|
87
|
+
const allTaskEvents = await eventStore.query({ aggregateTypes: ['task'] });
|
|
88
|
+
console.log(` Total task events: ${allTaskEvents.length}`);
|
|
89
|
+
const completedTaskEvents = await eventStore.getEventsByType('task:completed');
|
|
90
|
+
console.log(` Completed task events: ${completedTaskEvents.length}`);
|
|
91
|
+
console.log();
|
|
92
|
+
// =========================================================================
|
|
93
|
+
// 7. Build Projections
|
|
94
|
+
// =========================================================================
|
|
95
|
+
console.log('7. Building Projections...\n');
|
|
96
|
+
// Agent State Projection
|
|
97
|
+
const agentProjection = new AgentStateProjection(eventStore);
|
|
98
|
+
await agentProjection.initialize();
|
|
99
|
+
console.log(' Agent State Projection:');
|
|
100
|
+
const allAgents = agentProjection.getAllAgents();
|
|
101
|
+
for (const agent of allAgents) {
|
|
102
|
+
console.log(` - ${agent.id}: ${agent.role} (${agent.status})`);
|
|
103
|
+
console.log(` Tasks completed: ${agent.completedTasks.length}`);
|
|
104
|
+
console.log(` Avg duration: ${agent.taskCount > 0 ? agent.totalTaskDuration / agent.taskCount : 0}ms`);
|
|
105
|
+
}
|
|
106
|
+
const activeAgents = agentProjection.getActiveAgentCount();
|
|
107
|
+
console.log(`\n Active agents: ${activeAgents}`);
|
|
108
|
+
// Task History Projection
|
|
109
|
+
const taskProjection = new TaskHistoryProjection(eventStore);
|
|
110
|
+
await taskProjection.initialize();
|
|
111
|
+
console.log('\n Task History Projection:');
|
|
112
|
+
const allTasks = taskProjection.getAllTasks();
|
|
113
|
+
for (const task of allTasks) {
|
|
114
|
+
console.log(` - ${task.id}: ${task.title} (${task.status})`);
|
|
115
|
+
console.log(` Assigned to: ${task.assignedAgent || 'unassigned'}`);
|
|
116
|
+
console.log(` Duration: ${task.duration || 'N/A'}ms`);
|
|
117
|
+
}
|
|
118
|
+
const avgTaskDuration = taskProjection.getAverageTaskDuration();
|
|
119
|
+
console.log(`\n Average task duration: ${avgTaskDuration.toFixed(2)}ms`);
|
|
120
|
+
// Memory Index Projection
|
|
121
|
+
const memoryProjection = new MemoryIndexProjection(eventStore);
|
|
122
|
+
await memoryProjection.initialize();
|
|
123
|
+
console.log('\n Memory Index Projection:');
|
|
124
|
+
const activeMemories = memoryProjection.getActiveMemories();
|
|
125
|
+
for (const memory of activeMemories) {
|
|
126
|
+
console.log(` - ${memory.id}: ${memory.namespace}/${memory.key}`);
|
|
127
|
+
console.log(` Size: ${memory.size} bytes, Accessed: ${memory.accessCount} times`);
|
|
128
|
+
}
|
|
129
|
+
const totalSize = activeMemories.reduce((sum, m) => sum + m.size, 0);
|
|
130
|
+
console.log(`\n Total memory used: ${totalSize} bytes`);
|
|
131
|
+
// =========================================================================
|
|
132
|
+
// 8. Event Replay
|
|
133
|
+
// =========================================================================
|
|
134
|
+
console.log('\n8. Event Replay Example...\n');
|
|
135
|
+
let replayCount = 0;
|
|
136
|
+
for await (const event of eventStore.replay(0)) {
|
|
137
|
+
replayCount++;
|
|
138
|
+
if (replayCount <= 3) {
|
|
139
|
+
console.log(` Replaying: ${event.type} (v${event.version})`);
|
|
140
|
+
}
|
|
141
|
+
}
|
|
142
|
+
console.log(` Total events replayed: ${replayCount}`);
|
|
143
|
+
// =========================================================================
|
|
144
|
+
// 9. Snapshots
|
|
145
|
+
// =========================================================================
|
|
146
|
+
console.log('\n9. Snapshot Example...\n');
|
|
147
|
+
const agent2State = agentProjection.getAgent('agent-2');
|
|
148
|
+
if (agent2State) {
|
|
149
|
+
await eventStore.saveSnapshot({
|
|
150
|
+
aggregateId: 'agent-2',
|
|
151
|
+
aggregateType: 'agent',
|
|
152
|
+
version: 5,
|
|
153
|
+
state: agent2State,
|
|
154
|
+
timestamp: Date.now(),
|
|
155
|
+
});
|
|
156
|
+
console.log(' Snapshot saved for agent-2');
|
|
157
|
+
const snapshot = await eventStore.getSnapshot('agent-2');
|
|
158
|
+
console.log(` Snapshot retrieved: version ${snapshot?.version}`);
|
|
159
|
+
}
|
|
160
|
+
// =========================================================================
|
|
161
|
+
// 10. Statistics
|
|
162
|
+
// =========================================================================
|
|
163
|
+
console.log('\n10. Event Store Statistics...\n');
|
|
164
|
+
const stats = await eventStore.getStats();
|
|
165
|
+
console.log(` Total events: ${stats.totalEvents}`);
|
|
166
|
+
console.log(` Events by type:`);
|
|
167
|
+
for (const [type, count] of Object.entries(stats.eventsByType)) {
|
|
168
|
+
console.log(` - ${type}: ${count}`);
|
|
169
|
+
}
|
|
170
|
+
console.log(`\n Events by aggregate:`);
|
|
171
|
+
const topAggregates = Object.entries(stats.eventsByAggregate)
|
|
172
|
+
.sort(([, a], [, b]) => b - a)
|
|
173
|
+
.slice(0, 5);
|
|
174
|
+
for (const [aggregate, count] of topAggregates) {
|
|
175
|
+
console.log(` - ${aggregate}: ${count}`);
|
|
176
|
+
}
|
|
177
|
+
console.log(`\n Snapshots: ${stats.snapshotCount}`);
|
|
178
|
+
// =========================================================================
|
|
179
|
+
// Cleanup
|
|
180
|
+
// =========================================================================
|
|
181
|
+
console.log('\n11. Persisting and Shutting Down...\n');
|
|
182
|
+
await eventStore.persist();
|
|
183
|
+
await eventStore.shutdown();
|
|
184
|
+
console.log(' Event Store shutdown complete');
|
|
185
|
+
console.log('\nExample completed successfully!');
|
|
186
|
+
console.log('Database saved to: ./event-store-example.db');
|
|
187
|
+
}
|
|
188
|
+
// Run the example
|
|
189
|
+
main().catch((error) => {
|
|
190
|
+
console.error('Error running example:', error);
|
|
191
|
+
process.exit(1);
|
|
192
|
+
});
|
|
193
|
+
//# sourceMappingURL=example-usage.js.map
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Event Sourcing System (ADR-007)
|
|
3
|
+
*
|
|
4
|
+
* Complete event sourcing implementation for V3 Claude Flow:
|
|
5
|
+
* - Domain events for all aggregates (agent, task, memory, swarm)
|
|
6
|
+
* - Persistent event store with SQLite backend
|
|
7
|
+
* - Projections for building read models
|
|
8
|
+
* - Event replay and snapshots
|
|
9
|
+
*
|
|
10
|
+
* @module v3/shared/events
|
|
11
|
+
*/
|
|
12
|
+
export type { DomainEvent, AllDomainEvents, AgentSpawnedEvent, AgentStartedEvent, AgentStoppedEvent, AgentFailedEvent, AgentStatusChangedEvent, AgentTaskAssignedEvent, AgentTaskCompletedEvent, TaskCreatedEvent, TaskStartedEvent, TaskCompletedEvent, TaskFailedEvent, TaskBlockedEvent, TaskQueuedEvent, MemoryStoredEvent, MemoryRetrievedEvent, MemoryDeletedEvent, MemoryExpiredEvent, SwarmInitializedEvent, SwarmScaledEvent, SwarmTerminatedEvent, SwarmPhaseChangedEvent, SwarmMilestoneReachedEvent, SwarmErrorEvent, } from './domain-events.js';
|
|
13
|
+
export { createAgentSpawnedEvent, createAgentStartedEvent, createAgentStoppedEvent, createAgentFailedEvent, createTaskCreatedEvent, createTaskStartedEvent, createTaskCompletedEvent, createTaskFailedEvent, createMemoryStoredEvent, createMemoryRetrievedEvent, createMemoryDeletedEvent, createSwarmInitializedEvent, createSwarmScaledEvent, createSwarmTerminatedEvent, } from './domain-events.js';
|
|
14
|
+
export { EventStore } from './event-store.js';
|
|
15
|
+
export type { EventStoreConfig, EventFilter, EventSnapshot, EventStoreStats, } from './event-store.js';
|
|
16
|
+
export { Projection, AgentStateProjection, TaskHistoryProjection, MemoryIndexProjection, } from './projections.js';
|
|
17
|
+
export type { AgentProjectionState, TaskProjectionState, MemoryProjectionState, } from './projections.js';
|
|
18
|
+
export { RvfEventLog } from './rvf-event-log.js';
|
|
19
|
+
export type { RvfEventLogConfig } from './rvf-event-log.js';
|
|
20
|
+
export { StateReconstructor, createStateReconstructor, AgentAggregate, TaskAggregate, type AggregateRoot, type ReconstructorOptions, } from './state-reconstructor.js';
|
|
21
|
+
//# sourceMappingURL=index.d.ts.map
|
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Event Sourcing System (ADR-007)
|
|
3
|
+
*
|
|
4
|
+
* Complete event sourcing implementation for V3 Claude Flow:
|
|
5
|
+
* - Domain events for all aggregates (agent, task, memory, swarm)
|
|
6
|
+
* - Persistent event store with SQLite backend
|
|
7
|
+
* - Projections for building read models
|
|
8
|
+
* - Event replay and snapshots
|
|
9
|
+
*
|
|
10
|
+
* @module v3/shared/events
|
|
11
|
+
*/
|
|
12
|
+
// Domain Event Factory Functions
|
|
13
|
+
export { createAgentSpawnedEvent, createAgentStartedEvent, createAgentStoppedEvent, createAgentFailedEvent, createTaskCreatedEvent, createTaskStartedEvent, createTaskCompletedEvent, createTaskFailedEvent, createMemoryStoredEvent, createMemoryRetrievedEvent, createMemoryDeletedEvent, createSwarmInitializedEvent, createSwarmScaledEvent, createSwarmTerminatedEvent, } from './domain-events.js';
|
|
14
|
+
// Event Store
|
|
15
|
+
export { EventStore } from './event-store.js';
|
|
16
|
+
// Projections
|
|
17
|
+
export { Projection, AgentStateProjection, TaskHistoryProjection, MemoryIndexProjection, } from './projections.js';
|
|
18
|
+
// RVF Event Log (ADR-057 — zero-dependency alternative to EventStore)
|
|
19
|
+
export { RvfEventLog } from './rvf-event-log.js';
|
|
20
|
+
// State Reconstruction (ADR-007)
|
|
21
|
+
export { StateReconstructor, createStateReconstructor, AgentAggregate, TaskAggregate, } from './state-reconstructor.js';
|
|
22
|
+
//# sourceMappingURL=index.js.map
|