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,340 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* RVF Event Log (ADR-057 Phase 2)
|
|
3
|
+
*
|
|
4
|
+
* Pure-TypeScript append-only event log that stores events in a binary
|
|
5
|
+
* file format. Replaces the sql.js-dependent EventStore with a zero-
|
|
6
|
+
* dependency alternative.
|
|
7
|
+
*
|
|
8
|
+
* Binary format:
|
|
9
|
+
* File header: 4 bytes — magic "RVFL"
|
|
10
|
+
* Record: 4 bytes (uint32 BE payload length) + N bytes (JSON payload)
|
|
11
|
+
*
|
|
12
|
+
* In-memory indexes are rebuilt on initialize() by replaying the file.
|
|
13
|
+
* Snapshots are stored in a separate `.snap.rvf` file using the same format.
|
|
14
|
+
*
|
|
15
|
+
* @module v3/shared/events/rvf-event-log
|
|
16
|
+
*/
|
|
17
|
+
import { EventEmitter } from 'node:events';
|
|
18
|
+
import { existsSync, mkdirSync, readFileSync, writeFileSync, appendFileSync, renameSync } from 'node:fs';
|
|
19
|
+
import { dirname } from 'node:path';
|
|
20
|
+
/** Validate a file path is safe */
|
|
21
|
+
function validatePath(p) {
|
|
22
|
+
if (p.includes('\0'))
|
|
23
|
+
throw new Error('Event log path contains null bytes');
|
|
24
|
+
}
|
|
25
|
+
const DEFAULT_CONFIG = {
|
|
26
|
+
logPath: 'events.rvf',
|
|
27
|
+
verbose: false,
|
|
28
|
+
snapshotThreshold: 100,
|
|
29
|
+
};
|
|
30
|
+
// =============================================================================
|
|
31
|
+
// Constants
|
|
32
|
+
// =============================================================================
|
|
33
|
+
/** Magic bytes that identify an RVF event log file */
|
|
34
|
+
const MAGIC = Buffer.from('RVFL');
|
|
35
|
+
const MAGIC_LENGTH = 4;
|
|
36
|
+
const LENGTH_PREFIX_BYTES = 4;
|
|
37
|
+
// =============================================================================
|
|
38
|
+
// RvfEventLog Implementation
|
|
39
|
+
// =============================================================================
|
|
40
|
+
export class RvfEventLog extends EventEmitter {
|
|
41
|
+
config;
|
|
42
|
+
initialized = false;
|
|
43
|
+
/**
|
|
44
|
+
* All events kept in insertion order.
|
|
45
|
+
* Rebuilt from the file on initialize().
|
|
46
|
+
*/
|
|
47
|
+
events = [];
|
|
48
|
+
/** Fast lookup: aggregateId -> indices into this.events */
|
|
49
|
+
aggregateIndex = new Map();
|
|
50
|
+
/** Version tracking per aggregate */
|
|
51
|
+
aggregateVersions = new Map();
|
|
52
|
+
/** Snapshots keyed by aggregateId (latest wins) */
|
|
53
|
+
snapshots = new Map();
|
|
54
|
+
/** Path to the companion snapshot file */
|
|
55
|
+
snapshotPath;
|
|
56
|
+
constructor(config = {}) {
|
|
57
|
+
super();
|
|
58
|
+
this.config = { ...DEFAULT_CONFIG, ...config };
|
|
59
|
+
this.snapshotPath = this.config.logPath.replace(/\.rvf$/, '.snap.rvf');
|
|
60
|
+
if (this.snapshotPath === this.config.logPath) {
|
|
61
|
+
this.snapshotPath = this.config.logPath + '.snap.rvf';
|
|
62
|
+
}
|
|
63
|
+
validatePath(this.config.logPath);
|
|
64
|
+
}
|
|
65
|
+
// ===========================================================================
|
|
66
|
+
// Lifecycle
|
|
67
|
+
// ===========================================================================
|
|
68
|
+
/** Create / open the log file and rebuild in-memory indexes. */
|
|
69
|
+
async initialize() {
|
|
70
|
+
if (this.initialized)
|
|
71
|
+
return;
|
|
72
|
+
this.ensureDirectory(this.config.logPath);
|
|
73
|
+
// --- events file ---
|
|
74
|
+
if (existsSync(this.config.logPath)) {
|
|
75
|
+
this.replayFile(this.config.logPath, (event) => {
|
|
76
|
+
this.indexEvent(event);
|
|
77
|
+
});
|
|
78
|
+
}
|
|
79
|
+
else {
|
|
80
|
+
const tmpLog = this.config.logPath + '.tmp';
|
|
81
|
+
writeFileSync(tmpLog, MAGIC);
|
|
82
|
+
renameSync(tmpLog, this.config.logPath);
|
|
83
|
+
}
|
|
84
|
+
// --- snapshots file ---
|
|
85
|
+
if (existsSync(this.snapshotPath)) {
|
|
86
|
+
this.replayFile(this.snapshotPath, (_raw) => {
|
|
87
|
+
const snap = _raw;
|
|
88
|
+
this.snapshots.set(snap.aggregateId, snap);
|
|
89
|
+
});
|
|
90
|
+
}
|
|
91
|
+
else {
|
|
92
|
+
const tmpSnap = this.snapshotPath + '.tmp';
|
|
93
|
+
writeFileSync(tmpSnap, MAGIC);
|
|
94
|
+
renameSync(tmpSnap, this.snapshotPath);
|
|
95
|
+
}
|
|
96
|
+
this.initialized = true;
|
|
97
|
+
if (this.config.verbose) {
|
|
98
|
+
console.log(`[RvfEventLog] Initialized – ${this.events.length} events, ` +
|
|
99
|
+
`${this.snapshots.size} snapshots`);
|
|
100
|
+
}
|
|
101
|
+
this.emit('initialized');
|
|
102
|
+
}
|
|
103
|
+
/** Flush to disk and release resources. */
|
|
104
|
+
async close() {
|
|
105
|
+
if (!this.initialized)
|
|
106
|
+
return;
|
|
107
|
+
// All data is already on disk (append-only), so just clear memory.
|
|
108
|
+
this.events = [];
|
|
109
|
+
this.aggregateIndex.clear();
|
|
110
|
+
this.aggregateVersions.clear();
|
|
111
|
+
this.snapshots.clear();
|
|
112
|
+
this.initialized = false;
|
|
113
|
+
this.emit('shutdown');
|
|
114
|
+
}
|
|
115
|
+
// ===========================================================================
|
|
116
|
+
// Write Operations
|
|
117
|
+
// ===========================================================================
|
|
118
|
+
/** Append a domain event to the log. */
|
|
119
|
+
async append(event) {
|
|
120
|
+
this.ensureInitialized();
|
|
121
|
+
if (!event.aggregateId || typeof event.aggregateId !== 'string') {
|
|
122
|
+
throw new Error('Event must have a valid aggregateId string');
|
|
123
|
+
}
|
|
124
|
+
if (!event.type || typeof event.type !== 'string') {
|
|
125
|
+
throw new Error('Event must have a valid type string');
|
|
126
|
+
}
|
|
127
|
+
// Assign next version for aggregate
|
|
128
|
+
const currentVersion = this.aggregateVersions.get(event.aggregateId) ?? 0;
|
|
129
|
+
const nextVersion = currentVersion + 1;
|
|
130
|
+
event.version = nextVersion;
|
|
131
|
+
// Persist to disk first (crash-safe ordering)
|
|
132
|
+
this.appendRecord(this.config.logPath, event);
|
|
133
|
+
// Update in-memory state
|
|
134
|
+
this.indexEvent(event);
|
|
135
|
+
this.emit('event:appended', event);
|
|
136
|
+
if (nextVersion % this.config.snapshotThreshold === 0) {
|
|
137
|
+
this.emit('snapshot:recommended', {
|
|
138
|
+
aggregateId: event.aggregateId,
|
|
139
|
+
version: nextVersion,
|
|
140
|
+
});
|
|
141
|
+
}
|
|
142
|
+
}
|
|
143
|
+
/** Save a snapshot for an aggregate. */
|
|
144
|
+
async saveSnapshot(snapshot) {
|
|
145
|
+
this.ensureInitialized();
|
|
146
|
+
this.appendRecord(this.snapshotPath, snapshot);
|
|
147
|
+
this.snapshots.set(snapshot.aggregateId, snapshot);
|
|
148
|
+
this.emit('snapshot:saved', snapshot);
|
|
149
|
+
}
|
|
150
|
+
// ===========================================================================
|
|
151
|
+
// Read Operations
|
|
152
|
+
// ===========================================================================
|
|
153
|
+
/** Get events for a specific aggregate, optionally from a version. */
|
|
154
|
+
async getEvents(aggregateId, fromVersion) {
|
|
155
|
+
this.ensureInitialized();
|
|
156
|
+
const indices = this.aggregateIndex.get(aggregateId);
|
|
157
|
+
if (!indices || indices.length === 0)
|
|
158
|
+
return [];
|
|
159
|
+
let result = indices.map((i) => this.events[i]);
|
|
160
|
+
if (fromVersion !== undefined) {
|
|
161
|
+
result = result.filter((e) => e.version >= fromVersion);
|
|
162
|
+
}
|
|
163
|
+
// Events within an aggregate are already version-ordered because we
|
|
164
|
+
// append in order, but sort defensively.
|
|
165
|
+
return result.sort((a, b) => a.version - b.version);
|
|
166
|
+
}
|
|
167
|
+
/** Query events with an optional filter (matches EventStore.query API). */
|
|
168
|
+
async getAllEvents(filter) {
|
|
169
|
+
this.ensureInitialized();
|
|
170
|
+
if (!filter) {
|
|
171
|
+
return [...this.events].sort((a, b) => a.timestamp - b.timestamp);
|
|
172
|
+
}
|
|
173
|
+
let result = [...this.events];
|
|
174
|
+
// Aggregate ID filter
|
|
175
|
+
if (filter.aggregateIds && filter.aggregateIds.length > 0) {
|
|
176
|
+
const set = new Set(filter.aggregateIds);
|
|
177
|
+
result = result.filter((e) => set.has(e.aggregateId));
|
|
178
|
+
}
|
|
179
|
+
// Aggregate type filter
|
|
180
|
+
if (filter.aggregateTypes && filter.aggregateTypes.length > 0) {
|
|
181
|
+
const set = new Set(filter.aggregateTypes);
|
|
182
|
+
result = result.filter((e) => set.has(e.aggregateType));
|
|
183
|
+
}
|
|
184
|
+
// Event type filter
|
|
185
|
+
if (filter.eventTypes && filter.eventTypes.length > 0) {
|
|
186
|
+
const set = new Set(filter.eventTypes);
|
|
187
|
+
result = result.filter((e) => set.has(e.type));
|
|
188
|
+
}
|
|
189
|
+
// Timestamp filters
|
|
190
|
+
if (filter.afterTimestamp !== undefined) {
|
|
191
|
+
result = result.filter((e) => e.timestamp > filter.afterTimestamp);
|
|
192
|
+
}
|
|
193
|
+
if (filter.beforeTimestamp !== undefined) {
|
|
194
|
+
result = result.filter((e) => e.timestamp < filter.beforeTimestamp);
|
|
195
|
+
}
|
|
196
|
+
// Version filter
|
|
197
|
+
if (filter.fromVersion !== undefined) {
|
|
198
|
+
result = result.filter((e) => e.version >= filter.fromVersion);
|
|
199
|
+
}
|
|
200
|
+
// Sort by timestamp ascending (matches EventStore behaviour)
|
|
201
|
+
result.sort((a, b) => a.timestamp - b.timestamp);
|
|
202
|
+
// Pagination
|
|
203
|
+
if (filter.offset) {
|
|
204
|
+
result = result.slice(filter.offset);
|
|
205
|
+
}
|
|
206
|
+
if (filter.limit) {
|
|
207
|
+
result = result.slice(0, filter.limit);
|
|
208
|
+
}
|
|
209
|
+
return result;
|
|
210
|
+
}
|
|
211
|
+
/** Get latest snapshot for an aggregate. */
|
|
212
|
+
async getSnapshot(aggregateId) {
|
|
213
|
+
this.ensureInitialized();
|
|
214
|
+
return this.snapshots.get(aggregateId) ?? null;
|
|
215
|
+
}
|
|
216
|
+
/** Return event store statistics. */
|
|
217
|
+
async getStats() {
|
|
218
|
+
this.ensureInitialized();
|
|
219
|
+
const eventsByType = {};
|
|
220
|
+
const eventsByAggregate = {};
|
|
221
|
+
let oldest = null;
|
|
222
|
+
let newest = null;
|
|
223
|
+
for (const event of this.events) {
|
|
224
|
+
// by type
|
|
225
|
+
eventsByType[event.type] = (eventsByType[event.type] ?? 0) + 1;
|
|
226
|
+
// by aggregate
|
|
227
|
+
eventsByAggregate[event.aggregateId] =
|
|
228
|
+
(eventsByAggregate[event.aggregateId] ?? 0) + 1;
|
|
229
|
+
// timestamp range
|
|
230
|
+
if (oldest === null || event.timestamp < oldest)
|
|
231
|
+
oldest = event.timestamp;
|
|
232
|
+
if (newest === null || event.timestamp > newest)
|
|
233
|
+
newest = event.timestamp;
|
|
234
|
+
}
|
|
235
|
+
return {
|
|
236
|
+
totalEvents: this.events.length,
|
|
237
|
+
eventsByType,
|
|
238
|
+
eventsByAggregate,
|
|
239
|
+
oldestEvent: oldest,
|
|
240
|
+
newestEvent: newest,
|
|
241
|
+
snapshotCount: this.snapshots.size,
|
|
242
|
+
};
|
|
243
|
+
}
|
|
244
|
+
/**
|
|
245
|
+
* Flush to disk.
|
|
246
|
+
* For the append-only log this is a no-op because every append() call
|
|
247
|
+
* writes to disk synchronously. Provided for API compatibility with
|
|
248
|
+
* EventStore.
|
|
249
|
+
*/
|
|
250
|
+
async persist() {
|
|
251
|
+
// All records are already flushed on append. Nothing to do.
|
|
252
|
+
if (this.config.verbose) {
|
|
253
|
+
console.log('[RvfEventLog] persist() called — all data already on disk');
|
|
254
|
+
}
|
|
255
|
+
}
|
|
256
|
+
// ===========================================================================
|
|
257
|
+
// Private Helpers
|
|
258
|
+
// ===========================================================================
|
|
259
|
+
/**
|
|
260
|
+
* Replay an RVF file and invoke `handler` for every decoded record.
|
|
261
|
+
* Used both for events and snapshots.
|
|
262
|
+
*/
|
|
263
|
+
replayFile(filePath, handler) {
|
|
264
|
+
const buf = readFileSync(filePath);
|
|
265
|
+
// Validate magic
|
|
266
|
+
if (buf.length < MAGIC_LENGTH || buf.subarray(0, MAGIC_LENGTH).compare(MAGIC) !== 0) {
|
|
267
|
+
throw new Error(`[RvfEventLog] Invalid file header in ${filePath}`);
|
|
268
|
+
}
|
|
269
|
+
let offset = MAGIC_LENGTH;
|
|
270
|
+
const MAX_PAYLOAD_SIZE = 100 * 1024 * 1024; // 100MB safety limit
|
|
271
|
+
while (offset + LENGTH_PREFIX_BYTES <= buf.length) {
|
|
272
|
+
const payloadLength = buf.readUInt32BE(offset);
|
|
273
|
+
offset += LENGTH_PREFIX_BYTES;
|
|
274
|
+
if (payloadLength > MAX_PAYLOAD_SIZE) {
|
|
275
|
+
if (this.config.verbose) {
|
|
276
|
+
console.warn(`[RvfEventLog] Payload size ${payloadLength} exceeds safety limit`);
|
|
277
|
+
}
|
|
278
|
+
break;
|
|
279
|
+
}
|
|
280
|
+
if (offset + payloadLength > buf.length) {
|
|
281
|
+
// Truncated record — stop reading (crash recovery).
|
|
282
|
+
if (this.config.verbose) {
|
|
283
|
+
console.warn(`[RvfEventLog] Truncated record at offset ${offset - LENGTH_PREFIX_BYTES} — ` +
|
|
284
|
+
`expected ${payloadLength} bytes, have ${buf.length - offset}`);
|
|
285
|
+
}
|
|
286
|
+
break;
|
|
287
|
+
}
|
|
288
|
+
const json = buf.subarray(offset, offset + payloadLength).toString('utf8');
|
|
289
|
+
offset += payloadLength;
|
|
290
|
+
try {
|
|
291
|
+
const record = JSON.parse(json);
|
|
292
|
+
handler(record);
|
|
293
|
+
}
|
|
294
|
+
catch {
|
|
295
|
+
if (this.config.verbose) {
|
|
296
|
+
console.warn(`[RvfEventLog] Corrupt JSON record skipped`);
|
|
297
|
+
}
|
|
298
|
+
}
|
|
299
|
+
}
|
|
300
|
+
}
|
|
301
|
+
/** Append a single record to an RVF file. */
|
|
302
|
+
appendRecord(filePath, record) {
|
|
303
|
+
const json = JSON.stringify(record);
|
|
304
|
+
const payload = Buffer.from(json, 'utf8');
|
|
305
|
+
const lengthBuf = Buffer.allocUnsafe(LENGTH_PREFIX_BYTES);
|
|
306
|
+
lengthBuf.writeUInt32BE(payload.length, 0);
|
|
307
|
+
appendFileSync(filePath, Buffer.concat([lengthBuf, payload]));
|
|
308
|
+
}
|
|
309
|
+
/** Add an event to the in-memory indexes. */
|
|
310
|
+
indexEvent(event) {
|
|
311
|
+
const idx = this.events.length;
|
|
312
|
+
this.events.push(event);
|
|
313
|
+
// aggregateIndex
|
|
314
|
+
let indices = this.aggregateIndex.get(event.aggregateId);
|
|
315
|
+
if (!indices) {
|
|
316
|
+
indices = [];
|
|
317
|
+
this.aggregateIndex.set(event.aggregateId, indices);
|
|
318
|
+
}
|
|
319
|
+
indices.push(idx);
|
|
320
|
+
// version tracker
|
|
321
|
+
const current = this.aggregateVersions.get(event.aggregateId) ?? 0;
|
|
322
|
+
if (event.version > current) {
|
|
323
|
+
this.aggregateVersions.set(event.aggregateId, event.version);
|
|
324
|
+
}
|
|
325
|
+
}
|
|
326
|
+
/** Ensure parent directory exists for a file path. */
|
|
327
|
+
ensureDirectory(filePath) {
|
|
328
|
+
const dir = dirname(filePath);
|
|
329
|
+
if (!existsSync(dir)) {
|
|
330
|
+
mkdirSync(dir, { recursive: true });
|
|
331
|
+
}
|
|
332
|
+
}
|
|
333
|
+
/** Guard that throws if initialize() has not been called. */
|
|
334
|
+
ensureInitialized() {
|
|
335
|
+
if (!this.initialized) {
|
|
336
|
+
throw new Error('RvfEventLog not initialized. Call initialize() first.');
|
|
337
|
+
}
|
|
338
|
+
}
|
|
339
|
+
}
|
|
340
|
+
//# sourceMappingURL=rvf-event-log.js.map
|
|
@@ -0,0 +1,101 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* State Reconstructor - ADR-007 Implementation
|
|
3
|
+
*
|
|
4
|
+
* Reconstructs aggregate state from event streams.
|
|
5
|
+
* Implements event sourcing patterns for V3.
|
|
6
|
+
*
|
|
7
|
+
* @module v3/shared/events/state-reconstructor
|
|
8
|
+
*/
|
|
9
|
+
import { EventStore } from './event-store.js';
|
|
10
|
+
import type { DomainEvent } from './domain-events.js';
|
|
11
|
+
/**
|
|
12
|
+
* Aggregate root interface
|
|
13
|
+
*/
|
|
14
|
+
export interface AggregateRoot {
|
|
15
|
+
id: string;
|
|
16
|
+
version: number;
|
|
17
|
+
apply(event: DomainEvent): void;
|
|
18
|
+
getState(): Record<string, unknown>;
|
|
19
|
+
}
|
|
20
|
+
/**
|
|
21
|
+
* Reconstructor options
|
|
22
|
+
*/
|
|
23
|
+
export interface ReconstructorOptions {
|
|
24
|
+
useSnapshots: boolean;
|
|
25
|
+
snapshotInterval: number;
|
|
26
|
+
maxEventsToReplay: number;
|
|
27
|
+
}
|
|
28
|
+
/**
|
|
29
|
+
* State Reconstructor
|
|
30
|
+
*
|
|
31
|
+
* Reconstructs aggregate state from event history.
|
|
32
|
+
* Supports snapshots for performance optimization.
|
|
33
|
+
*/
|
|
34
|
+
export declare class StateReconstructor {
|
|
35
|
+
private readonly eventStore;
|
|
36
|
+
private readonly options;
|
|
37
|
+
constructor(eventStore: EventStore, options?: Partial<ReconstructorOptions>);
|
|
38
|
+
/**
|
|
39
|
+
* Reconstruct aggregate state from events
|
|
40
|
+
*/
|
|
41
|
+
reconstruct<T extends AggregateRoot>(aggregateId: string, factory: (id: string) => T): Promise<T>;
|
|
42
|
+
/**
|
|
43
|
+
* Reconstruct state at a specific point in time
|
|
44
|
+
*/
|
|
45
|
+
reconstructAtTime<T extends AggregateRoot>(aggregateId: string, factory: (id: string) => T, timestamp: Date): Promise<T>;
|
|
46
|
+
/**
|
|
47
|
+
* Reconstruct state at a specific version
|
|
48
|
+
*/
|
|
49
|
+
reconstructAtVersion<T extends AggregateRoot>(aggregateId: string, factory: (id: string) => T, targetVersion: number): Promise<T>;
|
|
50
|
+
/**
|
|
51
|
+
* Apply snapshot to aggregate
|
|
52
|
+
*/
|
|
53
|
+
private applySnapshot;
|
|
54
|
+
/**
|
|
55
|
+
* Create snapshot for aggregate
|
|
56
|
+
*/
|
|
57
|
+
private createSnapshot;
|
|
58
|
+
/**
|
|
59
|
+
* Get aggregate type from instance
|
|
60
|
+
*/
|
|
61
|
+
private getAggregateType;
|
|
62
|
+
}
|
|
63
|
+
/**
|
|
64
|
+
* Agent Aggregate - Example implementation
|
|
65
|
+
*/
|
|
66
|
+
export declare class AgentAggregate implements AggregateRoot {
|
|
67
|
+
id: string;
|
|
68
|
+
version: number;
|
|
69
|
+
private state;
|
|
70
|
+
constructor(id: string);
|
|
71
|
+
apply(event: DomainEvent): void;
|
|
72
|
+
getState(): Record<string, unknown>;
|
|
73
|
+
restoreFromSnapshot(snapshotState: unknown): void;
|
|
74
|
+
get name(): string;
|
|
75
|
+
get role(): string;
|
|
76
|
+
get status(): string;
|
|
77
|
+
get currentTask(): string | null;
|
|
78
|
+
get completedTasks(): string[];
|
|
79
|
+
get capabilities(): string[];
|
|
80
|
+
}
|
|
81
|
+
/**
|
|
82
|
+
* Task Aggregate - Example implementation
|
|
83
|
+
*/
|
|
84
|
+
export declare class TaskAggregate implements AggregateRoot {
|
|
85
|
+
id: string;
|
|
86
|
+
version: number;
|
|
87
|
+
private state;
|
|
88
|
+
constructor(id: string);
|
|
89
|
+
apply(event: DomainEvent): void;
|
|
90
|
+
getState(): Record<string, unknown>;
|
|
91
|
+
restoreFromSnapshot(snapshotState: unknown): void;
|
|
92
|
+
get title(): string;
|
|
93
|
+
get status(): string;
|
|
94
|
+
get assignedAgent(): string | null;
|
|
95
|
+
get result(): unknown;
|
|
96
|
+
}
|
|
97
|
+
/**
|
|
98
|
+
* Factory function
|
|
99
|
+
*/
|
|
100
|
+
export declare function createStateReconstructor(eventStore: EventStore, options?: Partial<ReconstructorOptions>): StateReconstructor;
|
|
101
|
+
//# sourceMappingURL=state-reconstructor.d.ts.map
|
|
@@ -0,0 +1,263 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* State Reconstructor - ADR-007 Implementation
|
|
3
|
+
*
|
|
4
|
+
* Reconstructs aggregate state from event streams.
|
|
5
|
+
* Implements event sourcing patterns for V3.
|
|
6
|
+
*
|
|
7
|
+
* @module v3/shared/events/state-reconstructor
|
|
8
|
+
*/
|
|
9
|
+
/**
|
|
10
|
+
* State Reconstructor
|
|
11
|
+
*
|
|
12
|
+
* Reconstructs aggregate state from event history.
|
|
13
|
+
* Supports snapshots for performance optimization.
|
|
14
|
+
*/
|
|
15
|
+
export class StateReconstructor {
|
|
16
|
+
eventStore;
|
|
17
|
+
options;
|
|
18
|
+
constructor(eventStore, options) {
|
|
19
|
+
this.eventStore = eventStore;
|
|
20
|
+
this.options = {
|
|
21
|
+
useSnapshots: true,
|
|
22
|
+
snapshotInterval: 100,
|
|
23
|
+
maxEventsToReplay: 10000,
|
|
24
|
+
...options,
|
|
25
|
+
};
|
|
26
|
+
}
|
|
27
|
+
/**
|
|
28
|
+
* Reconstruct aggregate state from events
|
|
29
|
+
*/
|
|
30
|
+
async reconstruct(aggregateId, factory) {
|
|
31
|
+
const aggregate = factory(aggregateId);
|
|
32
|
+
// Try to load from snapshot first
|
|
33
|
+
if (this.options.useSnapshots) {
|
|
34
|
+
const snapshot = await this.eventStore.getSnapshot(aggregateId);
|
|
35
|
+
if (snapshot) {
|
|
36
|
+
this.applySnapshot(aggregate, snapshot);
|
|
37
|
+
}
|
|
38
|
+
}
|
|
39
|
+
// Get events after snapshot version (or all if no snapshot)
|
|
40
|
+
const events = await this.eventStore.getEvents(aggregateId, aggregate.version + 1);
|
|
41
|
+
// Apply events
|
|
42
|
+
for (const event of events) {
|
|
43
|
+
if (events.length > this.options.maxEventsToReplay) {
|
|
44
|
+
throw new Error(`Too many events to replay (${events.length}). Consider creating a snapshot.`);
|
|
45
|
+
}
|
|
46
|
+
aggregate.apply(event);
|
|
47
|
+
}
|
|
48
|
+
// Create snapshot if interval reached
|
|
49
|
+
if (this.options.useSnapshots && aggregate.version % this.options.snapshotInterval === 0) {
|
|
50
|
+
await this.createSnapshot(aggregate);
|
|
51
|
+
}
|
|
52
|
+
return aggregate;
|
|
53
|
+
}
|
|
54
|
+
/**
|
|
55
|
+
* Reconstruct state at a specific point in time
|
|
56
|
+
*/
|
|
57
|
+
async reconstructAtTime(aggregateId, factory, timestamp) {
|
|
58
|
+
const aggregate = factory(aggregateId);
|
|
59
|
+
// Get all events up to timestamp
|
|
60
|
+
const allEvents = await this.eventStore.getEvents(aggregateId);
|
|
61
|
+
const events = allEvents.filter((e) => e.timestamp <= timestamp.getTime());
|
|
62
|
+
// Apply events
|
|
63
|
+
for (const event of events) {
|
|
64
|
+
aggregate.apply(event);
|
|
65
|
+
}
|
|
66
|
+
return aggregate;
|
|
67
|
+
}
|
|
68
|
+
/**
|
|
69
|
+
* Reconstruct state at a specific version
|
|
70
|
+
*/
|
|
71
|
+
async reconstructAtVersion(aggregateId, factory, targetVersion) {
|
|
72
|
+
const aggregate = factory(aggregateId);
|
|
73
|
+
// Get events up to target version
|
|
74
|
+
const events = await this.eventStore.getEvents(aggregateId);
|
|
75
|
+
const limitedEvents = events.filter((e) => e.version <= targetVersion);
|
|
76
|
+
// Apply events
|
|
77
|
+
for (const event of limitedEvents) {
|
|
78
|
+
aggregate.apply(event);
|
|
79
|
+
}
|
|
80
|
+
return aggregate;
|
|
81
|
+
}
|
|
82
|
+
/**
|
|
83
|
+
* Apply snapshot to aggregate
|
|
84
|
+
*/
|
|
85
|
+
applySnapshot(aggregate, snapshot) {
|
|
86
|
+
// Type assertion for aggregate that has restoreFromSnapshot
|
|
87
|
+
const restorable = aggregate;
|
|
88
|
+
if (typeof restorable.restoreFromSnapshot === 'function') {
|
|
89
|
+
restorable.restoreFromSnapshot(snapshot.state);
|
|
90
|
+
}
|
|
91
|
+
// Update version
|
|
92
|
+
aggregate.version = snapshot.version;
|
|
93
|
+
}
|
|
94
|
+
/**
|
|
95
|
+
* Create snapshot for aggregate
|
|
96
|
+
*/
|
|
97
|
+
async createSnapshot(aggregate) {
|
|
98
|
+
const snapshot = {
|
|
99
|
+
aggregateId: aggregate.id,
|
|
100
|
+
aggregateType: this.getAggregateType(aggregate),
|
|
101
|
+
version: aggregate.version,
|
|
102
|
+
state: aggregate.getState(),
|
|
103
|
+
timestamp: Date.now(),
|
|
104
|
+
};
|
|
105
|
+
await this.eventStore.saveSnapshot(snapshot);
|
|
106
|
+
}
|
|
107
|
+
/**
|
|
108
|
+
* Get aggregate type from instance
|
|
109
|
+
*/
|
|
110
|
+
getAggregateType(aggregate) {
|
|
111
|
+
const typeName = aggregate.constructor.name.toLowerCase().replace('aggregate', '');
|
|
112
|
+
// Map to valid aggregate types
|
|
113
|
+
if (typeName === 'agent' || typeName === 'task' || typeName === 'memory' || typeName === 'swarm') {
|
|
114
|
+
return typeName;
|
|
115
|
+
}
|
|
116
|
+
return 'agent'; // Default fallback
|
|
117
|
+
}
|
|
118
|
+
}
|
|
119
|
+
/**
|
|
120
|
+
* Agent Aggregate - Example implementation
|
|
121
|
+
*/
|
|
122
|
+
export class AgentAggregate {
|
|
123
|
+
id;
|
|
124
|
+
version = 0;
|
|
125
|
+
state = {
|
|
126
|
+
name: '',
|
|
127
|
+
role: '',
|
|
128
|
+
status: 'idle',
|
|
129
|
+
currentTask: null,
|
|
130
|
+
completedTasks: [],
|
|
131
|
+
capabilities: [],
|
|
132
|
+
createdAt: null,
|
|
133
|
+
lastActiveAt: null,
|
|
134
|
+
};
|
|
135
|
+
constructor(id) {
|
|
136
|
+
this.id = id;
|
|
137
|
+
}
|
|
138
|
+
apply(event) {
|
|
139
|
+
this.version = event.version;
|
|
140
|
+
switch (event.type) {
|
|
141
|
+
case 'agent:spawned':
|
|
142
|
+
this.state.name = event.payload.name;
|
|
143
|
+
this.state.role = event.payload.role;
|
|
144
|
+
this.state.capabilities = event.payload.capabilities ?? [];
|
|
145
|
+
this.state.status = 'idle';
|
|
146
|
+
this.state.createdAt = new Date(event.timestamp);
|
|
147
|
+
break;
|
|
148
|
+
case 'agent:started':
|
|
149
|
+
this.state.status = 'active';
|
|
150
|
+
this.state.lastActiveAt = new Date(event.timestamp);
|
|
151
|
+
break;
|
|
152
|
+
case 'agent:task-assigned':
|
|
153
|
+
this.state.currentTask = event.payload.taskId;
|
|
154
|
+
this.state.status = 'busy';
|
|
155
|
+
this.state.lastActiveAt = new Date(event.timestamp);
|
|
156
|
+
break;
|
|
157
|
+
case 'agent:task-completed':
|
|
158
|
+
this.state.completedTasks.push(event.payload.taskId);
|
|
159
|
+
this.state.currentTask = null;
|
|
160
|
+
this.state.status = 'active';
|
|
161
|
+
this.state.lastActiveAt = new Date(event.timestamp);
|
|
162
|
+
break;
|
|
163
|
+
case 'agent:terminated':
|
|
164
|
+
this.state.status = 'terminated';
|
|
165
|
+
break;
|
|
166
|
+
}
|
|
167
|
+
}
|
|
168
|
+
getState() {
|
|
169
|
+
return { ...this.state };
|
|
170
|
+
}
|
|
171
|
+
restoreFromSnapshot(snapshotState) {
|
|
172
|
+
const state = snapshotState;
|
|
173
|
+
this.state = {
|
|
174
|
+
...state,
|
|
175
|
+
createdAt: state.createdAt ? new Date(state.createdAt) : null,
|
|
176
|
+
lastActiveAt: state.lastActiveAt ? new Date(state.lastActiveAt) : null,
|
|
177
|
+
};
|
|
178
|
+
}
|
|
179
|
+
// Getters for type safety
|
|
180
|
+
get name() { return this.state.name; }
|
|
181
|
+
get role() { return this.state.role; }
|
|
182
|
+
get status() { return this.state.status; }
|
|
183
|
+
get currentTask() { return this.state.currentTask; }
|
|
184
|
+
get completedTasks() { return [...this.state.completedTasks]; }
|
|
185
|
+
get capabilities() { return [...this.state.capabilities]; }
|
|
186
|
+
}
|
|
187
|
+
/**
|
|
188
|
+
* Task Aggregate - Example implementation
|
|
189
|
+
*/
|
|
190
|
+
export class TaskAggregate {
|
|
191
|
+
id;
|
|
192
|
+
version = 0;
|
|
193
|
+
state = {
|
|
194
|
+
title: '',
|
|
195
|
+
description: '',
|
|
196
|
+
type: '',
|
|
197
|
+
priority: 'normal',
|
|
198
|
+
status: 'pending',
|
|
199
|
+
assignedAgent: null,
|
|
200
|
+
result: null,
|
|
201
|
+
createdAt: null,
|
|
202
|
+
startedAt: null,
|
|
203
|
+
completedAt: null,
|
|
204
|
+
};
|
|
205
|
+
constructor(id) {
|
|
206
|
+
this.id = id;
|
|
207
|
+
}
|
|
208
|
+
apply(event) {
|
|
209
|
+
this.version = event.version;
|
|
210
|
+
switch (event.type) {
|
|
211
|
+
case 'task:created':
|
|
212
|
+
this.state.title = event.payload.title;
|
|
213
|
+
this.state.description = event.payload.description;
|
|
214
|
+
this.state.type = event.payload.taskType;
|
|
215
|
+
this.state.priority = event.payload.priority ?? 'normal';
|
|
216
|
+
this.state.status = 'pending';
|
|
217
|
+
this.state.createdAt = new Date(event.timestamp);
|
|
218
|
+
break;
|
|
219
|
+
case 'task:started':
|
|
220
|
+
this.state.assignedAgent = event.payload.agentId;
|
|
221
|
+
this.state.status = 'running';
|
|
222
|
+
this.state.startedAt = new Date(event.timestamp);
|
|
223
|
+
break;
|
|
224
|
+
case 'task:completed':
|
|
225
|
+
this.state.result = event.payload.result;
|
|
226
|
+
this.state.status = 'completed';
|
|
227
|
+
this.state.completedAt = new Date(event.timestamp);
|
|
228
|
+
break;
|
|
229
|
+
case 'task:failed':
|
|
230
|
+
this.state.status = 'failed';
|
|
231
|
+
this.state.completedAt = new Date(event.timestamp);
|
|
232
|
+
break;
|
|
233
|
+
case 'task:cancelled':
|
|
234
|
+
this.state.status = 'cancelled';
|
|
235
|
+
this.state.completedAt = new Date(event.timestamp);
|
|
236
|
+
break;
|
|
237
|
+
}
|
|
238
|
+
}
|
|
239
|
+
getState() {
|
|
240
|
+
return { ...this.state };
|
|
241
|
+
}
|
|
242
|
+
restoreFromSnapshot(snapshotState) {
|
|
243
|
+
const state = snapshotState;
|
|
244
|
+
this.state = {
|
|
245
|
+
...state,
|
|
246
|
+
createdAt: state.createdAt ? new Date(state.createdAt) : null,
|
|
247
|
+
startedAt: state.startedAt ? new Date(state.startedAt) : null,
|
|
248
|
+
completedAt: state.completedAt ? new Date(state.completedAt) : null,
|
|
249
|
+
};
|
|
250
|
+
}
|
|
251
|
+
// Getters
|
|
252
|
+
get title() { return this.state.title; }
|
|
253
|
+
get status() { return this.state.status; }
|
|
254
|
+
get assignedAgent() { return this.state.assignedAgent; }
|
|
255
|
+
get result() { return this.state.result; }
|
|
256
|
+
}
|
|
257
|
+
/**
|
|
258
|
+
* Factory function
|
|
259
|
+
*/
|
|
260
|
+
export function createStateReconstructor(eventStore, options) {
|
|
261
|
+
return new StateReconstructor(eventStore, options);
|
|
262
|
+
}
|
|
263
|
+
//# sourceMappingURL=state-reconstructor.js.map
|