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.
Files changed (241) hide show
  1. package/.claude/guidance/shipped/moflo.md +45 -0
  2. package/.claude/helpers/statusline.cjs +1 -1
  3. package/.claude/workflow-state.json +9 -0
  4. package/package.json +2 -2
  5. package/src/@claude-flow/cli/dist/src/init/statusline-generator.js +1 -1
  6. package/src/@claude-flow/cli/dist/src/services/agentic-flow-bridge.js +5 -3
  7. package/src/@claude-flow/cli/package.json +1 -1
  8. package/src/@claude-flow/memory/dist/agent-memory-scope.d.ts +131 -0
  9. package/src/@claude-flow/memory/dist/agent-memory-scope.js +223 -0
  10. package/src/@claude-flow/memory/dist/agent-memory-scope.test.d.ts +8 -0
  11. package/src/@claude-flow/memory/dist/agent-memory-scope.test.js +466 -0
  12. package/src/@claude-flow/memory/dist/agentdb-adapter.d.ts +165 -0
  13. package/src/@claude-flow/memory/dist/agentdb-adapter.js +806 -0
  14. package/src/@claude-flow/memory/dist/agentdb-backend.d.ts +212 -0
  15. package/src/@claude-flow/memory/dist/agentdb-backend.js +842 -0
  16. package/src/@claude-flow/memory/dist/agentdb-backend.test.d.ts +7 -0
  17. package/src/@claude-flow/memory/dist/agentdb-backend.test.js +258 -0
  18. package/src/@claude-flow/memory/dist/application/commands/delete-memory.command.d.ts +65 -0
  19. package/src/@claude-flow/memory/dist/application/commands/delete-memory.command.js +129 -0
  20. package/src/@claude-flow/memory/dist/application/commands/store-memory.command.d.ts +48 -0
  21. package/src/@claude-flow/memory/dist/application/commands/store-memory.command.js +72 -0
  22. package/src/@claude-flow/memory/dist/application/index.d.ts +12 -0
  23. package/src/@claude-flow/memory/dist/application/index.js +15 -0
  24. package/src/@claude-flow/memory/dist/application/queries/search-memory.query.d.ts +72 -0
  25. package/src/@claude-flow/memory/dist/application/queries/search-memory.query.js +143 -0
  26. package/src/@claude-flow/memory/dist/application/services/memory-application-service.d.ts +121 -0
  27. package/src/@claude-flow/memory/dist/application/services/memory-application-service.js +190 -0
  28. package/src/@claude-flow/memory/dist/auto-memory-bridge.d.ts +226 -0
  29. package/src/@claude-flow/memory/dist/auto-memory-bridge.js +709 -0
  30. package/src/@claude-flow/memory/dist/auto-memory-bridge.test.d.ts +8 -0
  31. package/src/@claude-flow/memory/dist/auto-memory-bridge.test.js +757 -0
  32. package/src/@claude-flow/memory/dist/benchmark.test.d.ts +2 -0
  33. package/src/@claude-flow/memory/dist/benchmark.test.js +277 -0
  34. package/src/@claude-flow/memory/dist/cache-manager.d.ts +134 -0
  35. package/src/@claude-flow/memory/dist/cache-manager.js +407 -0
  36. package/src/@claude-flow/memory/dist/controller-registry.d.ts +216 -0
  37. package/src/@claude-flow/memory/dist/controller-registry.js +893 -0
  38. package/src/@claude-flow/memory/dist/controller-registry.test.d.ts +14 -0
  39. package/src/@claude-flow/memory/dist/controller-registry.test.js +593 -0
  40. package/src/@claude-flow/memory/dist/database-provider.d.ts +87 -0
  41. package/src/@claude-flow/memory/dist/database-provider.js +372 -0
  42. package/src/@claude-flow/memory/dist/database-provider.test.d.ts +7 -0
  43. package/src/@claude-flow/memory/dist/database-provider.test.js +287 -0
  44. package/src/@claude-flow/memory/dist/domain/entities/memory-entry.d.ts +143 -0
  45. package/src/@claude-flow/memory/dist/domain/entities/memory-entry.js +226 -0
  46. package/src/@claude-flow/memory/dist/domain/index.d.ts +11 -0
  47. package/src/@claude-flow/memory/dist/domain/index.js +12 -0
  48. package/src/@claude-flow/memory/dist/domain/repositories/memory-repository.interface.d.ts +102 -0
  49. package/src/@claude-flow/memory/dist/domain/repositories/memory-repository.interface.js +11 -0
  50. package/src/@claude-flow/memory/dist/domain/services/memory-domain-service.d.ts +105 -0
  51. package/src/@claude-flow/memory/dist/domain/services/memory-domain-service.js +297 -0
  52. package/src/@claude-flow/memory/dist/hnsw-index.d.ts +111 -0
  53. package/src/@claude-flow/memory/dist/hnsw-index.js +781 -0
  54. package/src/@claude-flow/memory/dist/hnsw-lite.d.ts +23 -0
  55. package/src/@claude-flow/memory/dist/hnsw-lite.js +168 -0
  56. package/src/@claude-flow/memory/dist/index.d.ts +204 -0
  57. package/src/@claude-flow/memory/dist/index.js +358 -0
  58. package/src/@claude-flow/memory/dist/infrastructure/index.d.ts +17 -0
  59. package/src/@claude-flow/memory/dist/infrastructure/index.js +16 -0
  60. package/src/@claude-flow/memory/dist/infrastructure/repositories/hybrid-memory-repository.d.ts +66 -0
  61. package/src/@claude-flow/memory/dist/infrastructure/repositories/hybrid-memory-repository.js +409 -0
  62. package/src/@claude-flow/memory/dist/learning-bridge.d.ts +137 -0
  63. package/src/@claude-flow/memory/dist/learning-bridge.js +335 -0
  64. package/src/@claude-flow/memory/dist/learning-bridge.test.d.ts +8 -0
  65. package/src/@claude-flow/memory/dist/learning-bridge.test.js +578 -0
  66. package/src/@claude-flow/memory/dist/memory-graph.d.ts +100 -0
  67. package/src/@claude-flow/memory/dist/memory-graph.js +333 -0
  68. package/src/@claude-flow/memory/dist/memory-graph.test.d.ts +8 -0
  69. package/src/@claude-flow/memory/dist/memory-graph.test.js +609 -0
  70. package/src/@claude-flow/memory/dist/migration.d.ts +68 -0
  71. package/src/@claude-flow/memory/dist/migration.js +513 -0
  72. package/src/@claude-flow/memory/dist/persistent-sona.d.ts +144 -0
  73. package/src/@claude-flow/memory/dist/persistent-sona.js +332 -0
  74. package/src/@claude-flow/memory/dist/query-builder.d.ts +211 -0
  75. package/src/@claude-flow/memory/dist/query-builder.js +438 -0
  76. package/src/@claude-flow/memory/dist/rvf-backend.d.ts +51 -0
  77. package/src/@claude-flow/memory/dist/rvf-backend.js +481 -0
  78. package/src/@claude-flow/memory/dist/rvf-learning-store.d.ts +139 -0
  79. package/src/@claude-flow/memory/dist/rvf-learning-store.js +295 -0
  80. package/src/@claude-flow/memory/dist/rvf-migration.d.ts +45 -0
  81. package/src/@claude-flow/memory/dist/rvf-migration.js +234 -0
  82. package/src/@claude-flow/memory/dist/sqljs-backend.d.ts +127 -0
  83. package/src/@claude-flow/memory/dist/sqljs-backend.js +600 -0
  84. package/src/@claude-flow/memory/dist/types.d.ts +484 -0
  85. package/src/@claude-flow/memory/dist/types.js +58 -0
  86. package/src/@claude-flow/shared/dist/core/config/defaults.d.ts +41 -0
  87. package/src/@claude-flow/shared/dist/core/config/defaults.js +186 -0
  88. package/src/@claude-flow/shared/dist/core/config/index.d.ts +8 -0
  89. package/src/@claude-flow/shared/dist/core/config/index.js +12 -0
  90. package/src/@claude-flow/shared/dist/core/config/loader.d.ts +45 -0
  91. package/src/@claude-flow/shared/dist/core/config/loader.js +222 -0
  92. package/src/@claude-flow/shared/dist/core/config/schema.d.ts +1134 -0
  93. package/src/@claude-flow/shared/dist/core/config/schema.js +158 -0
  94. package/src/@claude-flow/shared/dist/core/config/validator.d.ts +92 -0
  95. package/src/@claude-flow/shared/dist/core/config/validator.js +147 -0
  96. package/src/@claude-flow/shared/dist/core/event-bus.d.ts +31 -0
  97. package/src/@claude-flow/shared/dist/core/event-bus.js +197 -0
  98. package/src/@claude-flow/shared/dist/core/index.d.ts +15 -0
  99. package/src/@claude-flow/shared/dist/core/index.js +19 -0
  100. package/src/@claude-flow/shared/dist/core/interfaces/agent.interface.d.ts +200 -0
  101. package/src/@claude-flow/shared/dist/core/interfaces/agent.interface.js +6 -0
  102. package/src/@claude-flow/shared/dist/core/interfaces/coordinator.interface.d.ts +310 -0
  103. package/src/@claude-flow/shared/dist/core/interfaces/coordinator.interface.js +7 -0
  104. package/src/@claude-flow/shared/dist/core/interfaces/event.interface.d.ts +224 -0
  105. package/src/@claude-flow/shared/dist/core/interfaces/event.interface.js +46 -0
  106. package/src/@claude-flow/shared/dist/core/interfaces/index.d.ts +10 -0
  107. package/src/@claude-flow/shared/dist/core/interfaces/index.js +15 -0
  108. package/src/@claude-flow/shared/dist/core/interfaces/memory.interface.d.ts +298 -0
  109. package/src/@claude-flow/shared/dist/core/interfaces/memory.interface.js +7 -0
  110. package/src/@claude-flow/shared/dist/core/interfaces/task.interface.d.ts +185 -0
  111. package/src/@claude-flow/shared/dist/core/interfaces/task.interface.js +6 -0
  112. package/src/@claude-flow/shared/dist/core/orchestrator/event-coordinator.d.ts +35 -0
  113. package/src/@claude-flow/shared/dist/core/orchestrator/event-coordinator.js +101 -0
  114. package/src/@claude-flow/shared/dist/core/orchestrator/health-monitor.d.ts +60 -0
  115. package/src/@claude-flow/shared/dist/core/orchestrator/health-monitor.js +166 -0
  116. package/src/@claude-flow/shared/dist/core/orchestrator/index.d.ts +46 -0
  117. package/src/@claude-flow/shared/dist/core/orchestrator/index.js +64 -0
  118. package/src/@claude-flow/shared/dist/core/orchestrator/lifecycle-manager.d.ts +56 -0
  119. package/src/@claude-flow/shared/dist/core/orchestrator/lifecycle-manager.js +195 -0
  120. package/src/@claude-flow/shared/dist/core/orchestrator/session-manager.d.ts +83 -0
  121. package/src/@claude-flow/shared/dist/core/orchestrator/session-manager.js +193 -0
  122. package/src/@claude-flow/shared/dist/core/orchestrator/task-manager.d.ts +49 -0
  123. package/src/@claude-flow/shared/dist/core/orchestrator/task-manager.js +253 -0
  124. package/src/@claude-flow/shared/dist/events/domain-events.d.ts +282 -0
  125. package/src/@claude-flow/shared/dist/events/domain-events.js +165 -0
  126. package/src/@claude-flow/shared/dist/events/event-store.d.ts +126 -0
  127. package/src/@claude-flow/shared/dist/events/event-store.js +432 -0
  128. package/src/@claude-flow/shared/dist/events/event-store.test.d.ts +8 -0
  129. package/src/@claude-flow/shared/dist/events/event-store.test.js +297 -0
  130. package/src/@claude-flow/shared/dist/events/example-usage.d.ts +10 -0
  131. package/src/@claude-flow/shared/dist/events/example-usage.js +193 -0
  132. package/src/@claude-flow/shared/dist/events/index.d.ts +21 -0
  133. package/src/@claude-flow/shared/dist/events/index.js +22 -0
  134. package/src/@claude-flow/shared/dist/events/projections.d.ts +177 -0
  135. package/src/@claude-flow/shared/dist/events/projections.js +421 -0
  136. package/src/@claude-flow/shared/dist/events/rvf-event-log.d.ts +82 -0
  137. package/src/@claude-flow/shared/dist/events/rvf-event-log.js +340 -0
  138. package/src/@claude-flow/shared/dist/events/state-reconstructor.d.ts +101 -0
  139. package/src/@claude-flow/shared/dist/events/state-reconstructor.js +263 -0
  140. package/src/@claude-flow/shared/dist/events.d.ts +80 -0
  141. package/src/@claude-flow/shared/dist/events.js +249 -0
  142. package/src/@claude-flow/shared/dist/hooks/example-usage.d.ts +42 -0
  143. package/src/@claude-flow/shared/dist/hooks/example-usage.js +351 -0
  144. package/src/@claude-flow/shared/dist/hooks/executor.d.ts +100 -0
  145. package/src/@claude-flow/shared/dist/hooks/executor.js +267 -0
  146. package/src/@claude-flow/shared/dist/hooks/hooks.test.d.ts +9 -0
  147. package/src/@claude-flow/shared/dist/hooks/hooks.test.js +322 -0
  148. package/src/@claude-flow/shared/dist/hooks/index.d.ts +52 -0
  149. package/src/@claude-flow/shared/dist/hooks/index.js +51 -0
  150. package/src/@claude-flow/shared/dist/hooks/registry.d.ts +133 -0
  151. package/src/@claude-flow/shared/dist/hooks/registry.js +277 -0
  152. package/src/@claude-flow/shared/dist/hooks/safety/bash-safety.d.ts +105 -0
  153. package/src/@claude-flow/shared/dist/hooks/safety/bash-safety.js +481 -0
  154. package/src/@claude-flow/shared/dist/hooks/safety/file-organization.d.ts +144 -0
  155. package/src/@claude-flow/shared/dist/hooks/safety/file-organization.js +328 -0
  156. package/src/@claude-flow/shared/dist/hooks/safety/git-commit.d.ts +158 -0
  157. package/src/@claude-flow/shared/dist/hooks/safety/git-commit.js +450 -0
  158. package/src/@claude-flow/shared/dist/hooks/safety/index.d.ts +17 -0
  159. package/src/@claude-flow/shared/dist/hooks/safety/index.js +17 -0
  160. package/src/@claude-flow/shared/dist/hooks/session-hooks.d.ts +234 -0
  161. package/src/@claude-flow/shared/dist/hooks/session-hooks.js +334 -0
  162. package/src/@claude-flow/shared/dist/hooks/task-hooks.d.ts +163 -0
  163. package/src/@claude-flow/shared/dist/hooks/task-hooks.js +326 -0
  164. package/src/@claude-flow/shared/dist/hooks/types.d.ts +267 -0
  165. package/src/@claude-flow/shared/dist/hooks/types.js +62 -0
  166. package/src/@claude-flow/shared/dist/hooks/verify-exports.test.d.ts +9 -0
  167. package/src/@claude-flow/shared/dist/hooks/verify-exports.test.js +93 -0
  168. package/src/@claude-flow/shared/dist/index.d.ts +20 -0
  169. package/src/@claude-flow/shared/dist/index.js +50 -0
  170. package/src/@claude-flow/shared/dist/mcp/connection-pool.d.ts +98 -0
  171. package/src/@claude-flow/shared/dist/mcp/connection-pool.js +364 -0
  172. package/src/@claude-flow/shared/dist/mcp/index.d.ts +69 -0
  173. package/src/@claude-flow/shared/dist/mcp/index.js +84 -0
  174. package/src/@claude-flow/shared/dist/mcp/server.d.ts +166 -0
  175. package/src/@claude-flow/shared/dist/mcp/server.js +593 -0
  176. package/src/@claude-flow/shared/dist/mcp/session-manager.d.ts +136 -0
  177. package/src/@claude-flow/shared/dist/mcp/session-manager.js +335 -0
  178. package/src/@claude-flow/shared/dist/mcp/tool-registry.d.ts +178 -0
  179. package/src/@claude-flow/shared/dist/mcp/tool-registry.js +439 -0
  180. package/src/@claude-flow/shared/dist/mcp/transport/http.d.ts +104 -0
  181. package/src/@claude-flow/shared/dist/mcp/transport/http.js +476 -0
  182. package/src/@claude-flow/shared/dist/mcp/transport/index.d.ts +102 -0
  183. package/src/@claude-flow/shared/dist/mcp/transport/index.js +238 -0
  184. package/src/@claude-flow/shared/dist/mcp/transport/stdio.d.ts +104 -0
  185. package/src/@claude-flow/shared/dist/mcp/transport/stdio.js +263 -0
  186. package/src/@claude-flow/shared/dist/mcp/transport/websocket.d.ts +133 -0
  187. package/src/@claude-flow/shared/dist/mcp/transport/websocket.js +396 -0
  188. package/src/@claude-flow/shared/dist/mcp/types.d.ts +438 -0
  189. package/src/@claude-flow/shared/dist/mcp/types.js +54 -0
  190. package/src/@claude-flow/shared/dist/plugin-interface.d.ts +544 -0
  191. package/src/@claude-flow/shared/dist/plugin-interface.js +23 -0
  192. package/src/@claude-flow/shared/dist/plugin-loader.d.ts +139 -0
  193. package/src/@claude-flow/shared/dist/plugin-loader.js +434 -0
  194. package/src/@claude-flow/shared/dist/plugin-registry.d.ts +183 -0
  195. package/src/@claude-flow/shared/dist/plugin-registry.js +457 -0
  196. package/src/@claude-flow/shared/dist/plugins/index.d.ts +10 -0
  197. package/src/@claude-flow/shared/dist/plugins/index.js +10 -0
  198. package/src/@claude-flow/shared/dist/plugins/official/hive-mind-plugin.d.ts +106 -0
  199. package/src/@claude-flow/shared/dist/plugins/official/hive-mind-plugin.js +241 -0
  200. package/src/@claude-flow/shared/dist/plugins/official/index.d.ts +10 -0
  201. package/src/@claude-flow/shared/dist/plugins/official/index.js +10 -0
  202. package/src/@claude-flow/shared/dist/plugins/official/maestro-plugin.d.ts +121 -0
  203. package/src/@claude-flow/shared/dist/plugins/official/maestro-plugin.js +355 -0
  204. package/src/@claude-flow/shared/dist/plugins/types.d.ts +93 -0
  205. package/src/@claude-flow/shared/dist/plugins/types.js +9 -0
  206. package/src/@claude-flow/shared/dist/resilience/bulkhead.d.ts +105 -0
  207. package/src/@claude-flow/shared/dist/resilience/bulkhead.js +206 -0
  208. package/src/@claude-flow/shared/dist/resilience/circuit-breaker.d.ts +132 -0
  209. package/src/@claude-flow/shared/dist/resilience/circuit-breaker.js +233 -0
  210. package/src/@claude-flow/shared/dist/resilience/index.d.ts +19 -0
  211. package/src/@claude-flow/shared/dist/resilience/index.js +19 -0
  212. package/src/@claude-flow/shared/dist/resilience/rate-limiter.d.ts +168 -0
  213. package/src/@claude-flow/shared/dist/resilience/rate-limiter.js +314 -0
  214. package/src/@claude-flow/shared/dist/resilience/retry.d.ts +91 -0
  215. package/src/@claude-flow/shared/dist/resilience/retry.js +159 -0
  216. package/src/@claude-flow/shared/dist/security/index.d.ts +10 -0
  217. package/src/@claude-flow/shared/dist/security/index.js +12 -0
  218. package/src/@claude-flow/shared/dist/security/input-validation.d.ts +73 -0
  219. package/src/@claude-flow/shared/dist/security/input-validation.js +201 -0
  220. package/src/@claude-flow/shared/dist/security/secure-random.d.ts +92 -0
  221. package/src/@claude-flow/shared/dist/security/secure-random.js +142 -0
  222. package/src/@claude-flow/shared/dist/services/index.d.ts +7 -0
  223. package/src/@claude-flow/shared/dist/services/index.js +7 -0
  224. package/src/@claude-flow/shared/dist/services/v3-progress.service.d.ts +124 -0
  225. package/src/@claude-flow/shared/dist/services/v3-progress.service.js +402 -0
  226. package/src/@claude-flow/shared/dist/types/agent.types.d.ts +137 -0
  227. package/src/@claude-flow/shared/dist/types/agent.types.js +6 -0
  228. package/src/@claude-flow/shared/dist/types/index.d.ts +11 -0
  229. package/src/@claude-flow/shared/dist/types/index.js +17 -0
  230. package/src/@claude-flow/shared/dist/types/mcp.types.d.ts +266 -0
  231. package/src/@claude-flow/shared/dist/types/mcp.types.js +7 -0
  232. package/src/@claude-flow/shared/dist/types/memory.types.d.ts +236 -0
  233. package/src/@claude-flow/shared/dist/types/memory.types.js +7 -0
  234. package/src/@claude-flow/shared/dist/types/swarm.types.d.ts +186 -0
  235. package/src/@claude-flow/shared/dist/types/swarm.types.js +65 -0
  236. package/src/@claude-flow/shared/dist/types/task.types.d.ts +178 -0
  237. package/src/@claude-flow/shared/dist/types/task.types.js +32 -0
  238. package/src/@claude-flow/shared/dist/types.d.ts +197 -0
  239. package/src/@claude-flow/shared/dist/types.js +21 -0
  240. package/src/@claude-flow/shared/dist/utils/secure-logger.d.ts +69 -0
  241. 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