moflo 4.8.17 → 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 (244) 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/bin/generate-code-map.mjs +91 -12
  5. package/package.json +2 -2
  6. package/src/@claude-flow/cli/bin/cli.js +5 -0
  7. package/src/@claude-flow/cli/dist/src/init/moflo-init.js +9 -0
  8. package/src/@claude-flow/cli/dist/src/init/statusline-generator.js +1 -1
  9. package/src/@claude-flow/cli/dist/src/services/agentic-flow-bridge.js +8 -0
  10. package/src/@claude-flow/cli/package.json +1 -1
  11. package/src/@claude-flow/memory/dist/agent-memory-scope.d.ts +131 -0
  12. package/src/@claude-flow/memory/dist/agent-memory-scope.js +223 -0
  13. package/src/@claude-flow/memory/dist/agent-memory-scope.test.d.ts +8 -0
  14. package/src/@claude-flow/memory/dist/agent-memory-scope.test.js +466 -0
  15. package/src/@claude-flow/memory/dist/agentdb-adapter.d.ts +165 -0
  16. package/src/@claude-flow/memory/dist/agentdb-adapter.js +806 -0
  17. package/src/@claude-flow/memory/dist/agentdb-backend.d.ts +212 -0
  18. package/src/@claude-flow/memory/dist/agentdb-backend.js +842 -0
  19. package/src/@claude-flow/memory/dist/agentdb-backend.test.d.ts +7 -0
  20. package/src/@claude-flow/memory/dist/agentdb-backend.test.js +258 -0
  21. package/src/@claude-flow/memory/dist/application/commands/delete-memory.command.d.ts +65 -0
  22. package/src/@claude-flow/memory/dist/application/commands/delete-memory.command.js +129 -0
  23. package/src/@claude-flow/memory/dist/application/commands/store-memory.command.d.ts +48 -0
  24. package/src/@claude-flow/memory/dist/application/commands/store-memory.command.js +72 -0
  25. package/src/@claude-flow/memory/dist/application/index.d.ts +12 -0
  26. package/src/@claude-flow/memory/dist/application/index.js +15 -0
  27. package/src/@claude-flow/memory/dist/application/queries/search-memory.query.d.ts +72 -0
  28. package/src/@claude-flow/memory/dist/application/queries/search-memory.query.js +143 -0
  29. package/src/@claude-flow/memory/dist/application/services/memory-application-service.d.ts +121 -0
  30. package/src/@claude-flow/memory/dist/application/services/memory-application-service.js +190 -0
  31. package/src/@claude-flow/memory/dist/auto-memory-bridge.d.ts +226 -0
  32. package/src/@claude-flow/memory/dist/auto-memory-bridge.js +709 -0
  33. package/src/@claude-flow/memory/dist/auto-memory-bridge.test.d.ts +8 -0
  34. package/src/@claude-flow/memory/dist/auto-memory-bridge.test.js +757 -0
  35. package/src/@claude-flow/memory/dist/benchmark.test.d.ts +2 -0
  36. package/src/@claude-flow/memory/dist/benchmark.test.js +277 -0
  37. package/src/@claude-flow/memory/dist/cache-manager.d.ts +134 -0
  38. package/src/@claude-flow/memory/dist/cache-manager.js +407 -0
  39. package/src/@claude-flow/memory/dist/controller-registry.d.ts +216 -0
  40. package/src/@claude-flow/memory/dist/controller-registry.js +893 -0
  41. package/src/@claude-flow/memory/dist/controller-registry.test.d.ts +14 -0
  42. package/src/@claude-flow/memory/dist/controller-registry.test.js +593 -0
  43. package/src/@claude-flow/memory/dist/database-provider.d.ts +87 -0
  44. package/src/@claude-flow/memory/dist/database-provider.js +372 -0
  45. package/src/@claude-flow/memory/dist/database-provider.test.d.ts +7 -0
  46. package/src/@claude-flow/memory/dist/database-provider.test.js +287 -0
  47. package/src/@claude-flow/memory/dist/domain/entities/memory-entry.d.ts +143 -0
  48. package/src/@claude-flow/memory/dist/domain/entities/memory-entry.js +226 -0
  49. package/src/@claude-flow/memory/dist/domain/index.d.ts +11 -0
  50. package/src/@claude-flow/memory/dist/domain/index.js +12 -0
  51. package/src/@claude-flow/memory/dist/domain/repositories/memory-repository.interface.d.ts +102 -0
  52. package/src/@claude-flow/memory/dist/domain/repositories/memory-repository.interface.js +11 -0
  53. package/src/@claude-flow/memory/dist/domain/services/memory-domain-service.d.ts +105 -0
  54. package/src/@claude-flow/memory/dist/domain/services/memory-domain-service.js +297 -0
  55. package/src/@claude-flow/memory/dist/hnsw-index.d.ts +111 -0
  56. package/src/@claude-flow/memory/dist/hnsw-index.js +781 -0
  57. package/src/@claude-flow/memory/dist/hnsw-lite.d.ts +23 -0
  58. package/src/@claude-flow/memory/dist/hnsw-lite.js +168 -0
  59. package/src/@claude-flow/memory/dist/index.d.ts +204 -0
  60. package/src/@claude-flow/memory/dist/index.js +358 -0
  61. package/src/@claude-flow/memory/dist/infrastructure/index.d.ts +17 -0
  62. package/src/@claude-flow/memory/dist/infrastructure/index.js +16 -0
  63. package/src/@claude-flow/memory/dist/infrastructure/repositories/hybrid-memory-repository.d.ts +66 -0
  64. package/src/@claude-flow/memory/dist/infrastructure/repositories/hybrid-memory-repository.js +409 -0
  65. package/src/@claude-flow/memory/dist/learning-bridge.d.ts +137 -0
  66. package/src/@claude-flow/memory/dist/learning-bridge.js +335 -0
  67. package/src/@claude-flow/memory/dist/learning-bridge.test.d.ts +8 -0
  68. package/src/@claude-flow/memory/dist/learning-bridge.test.js +578 -0
  69. package/src/@claude-flow/memory/dist/memory-graph.d.ts +100 -0
  70. package/src/@claude-flow/memory/dist/memory-graph.js +333 -0
  71. package/src/@claude-flow/memory/dist/memory-graph.test.d.ts +8 -0
  72. package/src/@claude-flow/memory/dist/memory-graph.test.js +609 -0
  73. package/src/@claude-flow/memory/dist/migration.d.ts +68 -0
  74. package/src/@claude-flow/memory/dist/migration.js +513 -0
  75. package/src/@claude-flow/memory/dist/persistent-sona.d.ts +144 -0
  76. package/src/@claude-flow/memory/dist/persistent-sona.js +332 -0
  77. package/src/@claude-flow/memory/dist/query-builder.d.ts +211 -0
  78. package/src/@claude-flow/memory/dist/query-builder.js +438 -0
  79. package/src/@claude-flow/memory/dist/rvf-backend.d.ts +51 -0
  80. package/src/@claude-flow/memory/dist/rvf-backend.js +481 -0
  81. package/src/@claude-flow/memory/dist/rvf-learning-store.d.ts +139 -0
  82. package/src/@claude-flow/memory/dist/rvf-learning-store.js +295 -0
  83. package/src/@claude-flow/memory/dist/rvf-migration.d.ts +45 -0
  84. package/src/@claude-flow/memory/dist/rvf-migration.js +234 -0
  85. package/src/@claude-flow/memory/dist/sqljs-backend.d.ts +127 -0
  86. package/src/@claude-flow/memory/dist/sqljs-backend.js +600 -0
  87. package/src/@claude-flow/memory/dist/types.d.ts +484 -0
  88. package/src/@claude-flow/memory/dist/types.js +58 -0
  89. package/src/@claude-flow/shared/dist/core/config/defaults.d.ts +41 -0
  90. package/src/@claude-flow/shared/dist/core/config/defaults.js +186 -0
  91. package/src/@claude-flow/shared/dist/core/config/index.d.ts +8 -0
  92. package/src/@claude-flow/shared/dist/core/config/index.js +12 -0
  93. package/src/@claude-flow/shared/dist/core/config/loader.d.ts +45 -0
  94. package/src/@claude-flow/shared/dist/core/config/loader.js +222 -0
  95. package/src/@claude-flow/shared/dist/core/config/schema.d.ts +1134 -0
  96. package/src/@claude-flow/shared/dist/core/config/schema.js +158 -0
  97. package/src/@claude-flow/shared/dist/core/config/validator.d.ts +92 -0
  98. package/src/@claude-flow/shared/dist/core/config/validator.js +147 -0
  99. package/src/@claude-flow/shared/dist/core/event-bus.d.ts +31 -0
  100. package/src/@claude-flow/shared/dist/core/event-bus.js +197 -0
  101. package/src/@claude-flow/shared/dist/core/index.d.ts +15 -0
  102. package/src/@claude-flow/shared/dist/core/index.js +19 -0
  103. package/src/@claude-flow/shared/dist/core/interfaces/agent.interface.d.ts +200 -0
  104. package/src/@claude-flow/shared/dist/core/interfaces/agent.interface.js +6 -0
  105. package/src/@claude-flow/shared/dist/core/interfaces/coordinator.interface.d.ts +310 -0
  106. package/src/@claude-flow/shared/dist/core/interfaces/coordinator.interface.js +7 -0
  107. package/src/@claude-flow/shared/dist/core/interfaces/event.interface.d.ts +224 -0
  108. package/src/@claude-flow/shared/dist/core/interfaces/event.interface.js +46 -0
  109. package/src/@claude-flow/shared/dist/core/interfaces/index.d.ts +10 -0
  110. package/src/@claude-flow/shared/dist/core/interfaces/index.js +15 -0
  111. package/src/@claude-flow/shared/dist/core/interfaces/memory.interface.d.ts +298 -0
  112. package/src/@claude-flow/shared/dist/core/interfaces/memory.interface.js +7 -0
  113. package/src/@claude-flow/shared/dist/core/interfaces/task.interface.d.ts +185 -0
  114. package/src/@claude-flow/shared/dist/core/interfaces/task.interface.js +6 -0
  115. package/src/@claude-flow/shared/dist/core/orchestrator/event-coordinator.d.ts +35 -0
  116. package/src/@claude-flow/shared/dist/core/orchestrator/event-coordinator.js +101 -0
  117. package/src/@claude-flow/shared/dist/core/orchestrator/health-monitor.d.ts +60 -0
  118. package/src/@claude-flow/shared/dist/core/orchestrator/health-monitor.js +166 -0
  119. package/src/@claude-flow/shared/dist/core/orchestrator/index.d.ts +46 -0
  120. package/src/@claude-flow/shared/dist/core/orchestrator/index.js +64 -0
  121. package/src/@claude-flow/shared/dist/core/orchestrator/lifecycle-manager.d.ts +56 -0
  122. package/src/@claude-flow/shared/dist/core/orchestrator/lifecycle-manager.js +195 -0
  123. package/src/@claude-flow/shared/dist/core/orchestrator/session-manager.d.ts +83 -0
  124. package/src/@claude-flow/shared/dist/core/orchestrator/session-manager.js +193 -0
  125. package/src/@claude-flow/shared/dist/core/orchestrator/task-manager.d.ts +49 -0
  126. package/src/@claude-flow/shared/dist/core/orchestrator/task-manager.js +253 -0
  127. package/src/@claude-flow/shared/dist/events/domain-events.d.ts +282 -0
  128. package/src/@claude-flow/shared/dist/events/domain-events.js +165 -0
  129. package/src/@claude-flow/shared/dist/events/event-store.d.ts +126 -0
  130. package/src/@claude-flow/shared/dist/events/event-store.js +432 -0
  131. package/src/@claude-flow/shared/dist/events/event-store.test.d.ts +8 -0
  132. package/src/@claude-flow/shared/dist/events/event-store.test.js +297 -0
  133. package/src/@claude-flow/shared/dist/events/example-usage.d.ts +10 -0
  134. package/src/@claude-flow/shared/dist/events/example-usage.js +193 -0
  135. package/src/@claude-flow/shared/dist/events/index.d.ts +21 -0
  136. package/src/@claude-flow/shared/dist/events/index.js +22 -0
  137. package/src/@claude-flow/shared/dist/events/projections.d.ts +177 -0
  138. package/src/@claude-flow/shared/dist/events/projections.js +421 -0
  139. package/src/@claude-flow/shared/dist/events/rvf-event-log.d.ts +82 -0
  140. package/src/@claude-flow/shared/dist/events/rvf-event-log.js +340 -0
  141. package/src/@claude-flow/shared/dist/events/state-reconstructor.d.ts +101 -0
  142. package/src/@claude-flow/shared/dist/events/state-reconstructor.js +263 -0
  143. package/src/@claude-flow/shared/dist/events.d.ts +80 -0
  144. package/src/@claude-flow/shared/dist/events.js +249 -0
  145. package/src/@claude-flow/shared/dist/hooks/example-usage.d.ts +42 -0
  146. package/src/@claude-flow/shared/dist/hooks/example-usage.js +351 -0
  147. package/src/@claude-flow/shared/dist/hooks/executor.d.ts +100 -0
  148. package/src/@claude-flow/shared/dist/hooks/executor.js +267 -0
  149. package/src/@claude-flow/shared/dist/hooks/hooks.test.d.ts +9 -0
  150. package/src/@claude-flow/shared/dist/hooks/hooks.test.js +322 -0
  151. package/src/@claude-flow/shared/dist/hooks/index.d.ts +52 -0
  152. package/src/@claude-flow/shared/dist/hooks/index.js +51 -0
  153. package/src/@claude-flow/shared/dist/hooks/registry.d.ts +133 -0
  154. package/src/@claude-flow/shared/dist/hooks/registry.js +277 -0
  155. package/src/@claude-flow/shared/dist/hooks/safety/bash-safety.d.ts +105 -0
  156. package/src/@claude-flow/shared/dist/hooks/safety/bash-safety.js +481 -0
  157. package/src/@claude-flow/shared/dist/hooks/safety/file-organization.d.ts +144 -0
  158. package/src/@claude-flow/shared/dist/hooks/safety/file-organization.js +328 -0
  159. package/src/@claude-flow/shared/dist/hooks/safety/git-commit.d.ts +158 -0
  160. package/src/@claude-flow/shared/dist/hooks/safety/git-commit.js +450 -0
  161. package/src/@claude-flow/shared/dist/hooks/safety/index.d.ts +17 -0
  162. package/src/@claude-flow/shared/dist/hooks/safety/index.js +17 -0
  163. package/src/@claude-flow/shared/dist/hooks/session-hooks.d.ts +234 -0
  164. package/src/@claude-flow/shared/dist/hooks/session-hooks.js +334 -0
  165. package/src/@claude-flow/shared/dist/hooks/task-hooks.d.ts +163 -0
  166. package/src/@claude-flow/shared/dist/hooks/task-hooks.js +326 -0
  167. package/src/@claude-flow/shared/dist/hooks/types.d.ts +267 -0
  168. package/src/@claude-flow/shared/dist/hooks/types.js +62 -0
  169. package/src/@claude-flow/shared/dist/hooks/verify-exports.test.d.ts +9 -0
  170. package/src/@claude-flow/shared/dist/hooks/verify-exports.test.js +93 -0
  171. package/src/@claude-flow/shared/dist/index.d.ts +20 -0
  172. package/src/@claude-flow/shared/dist/index.js +50 -0
  173. package/src/@claude-flow/shared/dist/mcp/connection-pool.d.ts +98 -0
  174. package/src/@claude-flow/shared/dist/mcp/connection-pool.js +364 -0
  175. package/src/@claude-flow/shared/dist/mcp/index.d.ts +69 -0
  176. package/src/@claude-flow/shared/dist/mcp/index.js +84 -0
  177. package/src/@claude-flow/shared/dist/mcp/server.d.ts +166 -0
  178. package/src/@claude-flow/shared/dist/mcp/server.js +593 -0
  179. package/src/@claude-flow/shared/dist/mcp/session-manager.d.ts +136 -0
  180. package/src/@claude-flow/shared/dist/mcp/session-manager.js +335 -0
  181. package/src/@claude-flow/shared/dist/mcp/tool-registry.d.ts +178 -0
  182. package/src/@claude-flow/shared/dist/mcp/tool-registry.js +439 -0
  183. package/src/@claude-flow/shared/dist/mcp/transport/http.d.ts +104 -0
  184. package/src/@claude-flow/shared/dist/mcp/transport/http.js +476 -0
  185. package/src/@claude-flow/shared/dist/mcp/transport/index.d.ts +102 -0
  186. package/src/@claude-flow/shared/dist/mcp/transport/index.js +238 -0
  187. package/src/@claude-flow/shared/dist/mcp/transport/stdio.d.ts +104 -0
  188. package/src/@claude-flow/shared/dist/mcp/transport/stdio.js +263 -0
  189. package/src/@claude-flow/shared/dist/mcp/transport/websocket.d.ts +133 -0
  190. package/src/@claude-flow/shared/dist/mcp/transport/websocket.js +396 -0
  191. package/src/@claude-flow/shared/dist/mcp/types.d.ts +438 -0
  192. package/src/@claude-flow/shared/dist/mcp/types.js +54 -0
  193. package/src/@claude-flow/shared/dist/plugin-interface.d.ts +544 -0
  194. package/src/@claude-flow/shared/dist/plugin-interface.js +23 -0
  195. package/src/@claude-flow/shared/dist/plugin-loader.d.ts +139 -0
  196. package/src/@claude-flow/shared/dist/plugin-loader.js +434 -0
  197. package/src/@claude-flow/shared/dist/plugin-registry.d.ts +183 -0
  198. package/src/@claude-flow/shared/dist/plugin-registry.js +457 -0
  199. package/src/@claude-flow/shared/dist/plugins/index.d.ts +10 -0
  200. package/src/@claude-flow/shared/dist/plugins/index.js +10 -0
  201. package/src/@claude-flow/shared/dist/plugins/official/hive-mind-plugin.d.ts +106 -0
  202. package/src/@claude-flow/shared/dist/plugins/official/hive-mind-plugin.js +241 -0
  203. package/src/@claude-flow/shared/dist/plugins/official/index.d.ts +10 -0
  204. package/src/@claude-flow/shared/dist/plugins/official/index.js +10 -0
  205. package/src/@claude-flow/shared/dist/plugins/official/maestro-plugin.d.ts +121 -0
  206. package/src/@claude-flow/shared/dist/plugins/official/maestro-plugin.js +355 -0
  207. package/src/@claude-flow/shared/dist/plugins/types.d.ts +93 -0
  208. package/src/@claude-flow/shared/dist/plugins/types.js +9 -0
  209. package/src/@claude-flow/shared/dist/resilience/bulkhead.d.ts +105 -0
  210. package/src/@claude-flow/shared/dist/resilience/bulkhead.js +206 -0
  211. package/src/@claude-flow/shared/dist/resilience/circuit-breaker.d.ts +132 -0
  212. package/src/@claude-flow/shared/dist/resilience/circuit-breaker.js +233 -0
  213. package/src/@claude-flow/shared/dist/resilience/index.d.ts +19 -0
  214. package/src/@claude-flow/shared/dist/resilience/index.js +19 -0
  215. package/src/@claude-flow/shared/dist/resilience/rate-limiter.d.ts +168 -0
  216. package/src/@claude-flow/shared/dist/resilience/rate-limiter.js +314 -0
  217. package/src/@claude-flow/shared/dist/resilience/retry.d.ts +91 -0
  218. package/src/@claude-flow/shared/dist/resilience/retry.js +159 -0
  219. package/src/@claude-flow/shared/dist/security/index.d.ts +10 -0
  220. package/src/@claude-flow/shared/dist/security/index.js +12 -0
  221. package/src/@claude-flow/shared/dist/security/input-validation.d.ts +73 -0
  222. package/src/@claude-flow/shared/dist/security/input-validation.js +201 -0
  223. package/src/@claude-flow/shared/dist/security/secure-random.d.ts +92 -0
  224. package/src/@claude-flow/shared/dist/security/secure-random.js +142 -0
  225. package/src/@claude-flow/shared/dist/services/index.d.ts +7 -0
  226. package/src/@claude-flow/shared/dist/services/index.js +7 -0
  227. package/src/@claude-flow/shared/dist/services/v3-progress.service.d.ts +124 -0
  228. package/src/@claude-flow/shared/dist/services/v3-progress.service.js +402 -0
  229. package/src/@claude-flow/shared/dist/types/agent.types.d.ts +137 -0
  230. package/src/@claude-flow/shared/dist/types/agent.types.js +6 -0
  231. package/src/@claude-flow/shared/dist/types/index.d.ts +11 -0
  232. package/src/@claude-flow/shared/dist/types/index.js +17 -0
  233. package/src/@claude-flow/shared/dist/types/mcp.types.d.ts +266 -0
  234. package/src/@claude-flow/shared/dist/types/mcp.types.js +7 -0
  235. package/src/@claude-flow/shared/dist/types/memory.types.d.ts +236 -0
  236. package/src/@claude-flow/shared/dist/types/memory.types.js +7 -0
  237. package/src/@claude-flow/shared/dist/types/swarm.types.d.ts +186 -0
  238. package/src/@claude-flow/shared/dist/types/swarm.types.js +65 -0
  239. package/src/@claude-flow/shared/dist/types/task.types.d.ts +178 -0
  240. package/src/@claude-flow/shared/dist/types/task.types.js +32 -0
  241. package/src/@claude-flow/shared/dist/types.d.ts +197 -0
  242. package/src/@claude-flow/shared/dist/types.js +21 -0
  243. package/src/@claude-flow/shared/dist/utils/secure-logger.d.ts +69 -0
  244. package/src/@claude-flow/shared/dist/utils/secure-logger.js +208 -0
@@ -0,0 +1,206 @@
1
+ /**
2
+ * Bulkhead Pattern
3
+ *
4
+ * Isolates failures by limiting concurrent executions.
5
+ *
6
+ * @module v3/shared/resilience/bulkhead
7
+ */
8
+ import { EventEmitter } from 'events';
9
+ /**
10
+ * Default options
11
+ */
12
+ const DEFAULT_OPTIONS = {
13
+ maxConcurrent: 10,
14
+ maxQueue: 100,
15
+ queueTimeout: 30000,
16
+ };
17
+ /**
18
+ * Bulkhead
19
+ *
20
+ * Limits concurrent executions to prevent resource exhaustion.
21
+ *
22
+ * @example
23
+ * const bulkhead = new Bulkhead({
24
+ * name: 'database',
25
+ * maxConcurrent: 10,
26
+ * maxQueue: 50,
27
+ * });
28
+ *
29
+ * try {
30
+ * const result = await bulkhead.execute(() => dbQuery());
31
+ * } catch (error) {
32
+ * if (error.message.includes('Bulkhead full')) {
33
+ * // Handle capacity exceeded
34
+ * }
35
+ * }
36
+ */
37
+ export class Bulkhead extends EventEmitter {
38
+ options;
39
+ active = 0;
40
+ queue = [];
41
+ completed = 0;
42
+ rejected = 0;
43
+ timedOut = 0;
44
+ constructor(options) {
45
+ super();
46
+ this.options = { ...DEFAULT_OPTIONS, ...options };
47
+ }
48
+ /**
49
+ * Execute a function within the bulkhead
50
+ */
51
+ async execute(fn) {
52
+ // If there's room for execution, run immediately
53
+ if (this.active < this.options.maxConcurrent) {
54
+ return this.runNow(fn);
55
+ }
56
+ // Check if queue is full
57
+ if (this.queue.length >= this.options.maxQueue) {
58
+ this.rejected++;
59
+ this.options.onRejected?.('full');
60
+ throw new Error(`Bulkhead '${this.options.name}' is full. Max concurrent: ${this.options.maxConcurrent}, queue: ${this.options.maxQueue}`);
61
+ }
62
+ // Add to queue
63
+ return this.addToQueue(fn);
64
+ }
65
+ /**
66
+ * Get current statistics
67
+ */
68
+ getStats() {
69
+ return {
70
+ active: this.active,
71
+ queued: this.queue.length,
72
+ maxConcurrent: this.options.maxConcurrent,
73
+ maxQueue: this.options.maxQueue,
74
+ completed: this.completed,
75
+ rejected: this.rejected,
76
+ timedOut: this.timedOut,
77
+ };
78
+ }
79
+ /**
80
+ * Check if there's capacity available
81
+ */
82
+ hasCapacity() {
83
+ return this.active < this.options.maxConcurrent || this.queue.length < this.options.maxQueue;
84
+ }
85
+ /**
86
+ * Get available capacity (concurrent + queue)
87
+ */
88
+ availableCapacity() {
89
+ const concurrentAvailable = this.options.maxConcurrent - this.active;
90
+ const queueAvailable = this.options.maxQueue - this.queue.length;
91
+ return concurrentAvailable + queueAvailable;
92
+ }
93
+ /**
94
+ * Reset statistics
95
+ */
96
+ resetStats() {
97
+ this.completed = 0;
98
+ this.rejected = 0;
99
+ this.timedOut = 0;
100
+ }
101
+ /**
102
+ * Run function immediately
103
+ */
104
+ async runNow(fn) {
105
+ this.active++;
106
+ this.emit('acquire');
107
+ try {
108
+ const result = await fn();
109
+ this.completed++;
110
+ return result;
111
+ }
112
+ finally {
113
+ this.active--;
114
+ this.emit('release');
115
+ this.processQueue();
116
+ }
117
+ }
118
+ /**
119
+ * Add function to queue
120
+ */
121
+ addToQueue(fn) {
122
+ return new Promise((resolve, reject) => {
123
+ const item = {
124
+ fn,
125
+ resolve,
126
+ reject,
127
+ queuedAt: Date.now(),
128
+ };
129
+ // Set timeout for queued item
130
+ item.timeoutId = setTimeout(() => {
131
+ const index = this.queue.indexOf(item);
132
+ if (index !== -1) {
133
+ this.queue.splice(index, 1);
134
+ this.timedOut++;
135
+ this.options.onRejected?.('timeout');
136
+ reject(new Error(`Bulkhead '${this.options.name}' queue timeout after ${this.options.queueTimeout}ms`));
137
+ }
138
+ }, this.options.queueTimeout);
139
+ this.queue.push(item);
140
+ this.emit('queued', { queueLength: this.queue.length });
141
+ });
142
+ }
143
+ /**
144
+ * Process next item in queue
145
+ */
146
+ processQueue() {
147
+ if (this.active >= this.options.maxConcurrent) {
148
+ return;
149
+ }
150
+ const item = this.queue.shift();
151
+ if (!item) {
152
+ return;
153
+ }
154
+ // Clear timeout
155
+ if (item.timeoutId) {
156
+ clearTimeout(item.timeoutId);
157
+ }
158
+ // Execute the queued function
159
+ this.active++;
160
+ this.emit('acquire');
161
+ item.fn()
162
+ .then((result) => {
163
+ this.completed++;
164
+ item.resolve(result);
165
+ })
166
+ .catch((error) => {
167
+ item.reject(error);
168
+ })
169
+ .finally(() => {
170
+ this.active--;
171
+ this.emit('release');
172
+ this.processQueue();
173
+ });
174
+ }
175
+ }
176
+ /**
177
+ * Create a semaphore for limiting concurrent access
178
+ */
179
+ export function createSemaphore(maxConcurrent) {
180
+ let current = 0;
181
+ const waiting = [];
182
+ return {
183
+ async acquire() {
184
+ if (current < maxConcurrent) {
185
+ current++;
186
+ return;
187
+ }
188
+ return new Promise((resolve) => {
189
+ waiting.push(resolve);
190
+ });
191
+ },
192
+ release() {
193
+ const next = waiting.shift();
194
+ if (next) {
195
+ next();
196
+ }
197
+ else {
198
+ current = Math.max(0, current - 1);
199
+ }
200
+ },
201
+ available() {
202
+ return maxConcurrent - current;
203
+ },
204
+ };
205
+ }
206
+ //# sourceMappingURL=bulkhead.js.map
@@ -0,0 +1,132 @@
1
+ /**
2
+ * Circuit Breaker Pattern
3
+ *
4
+ * Prevents cascading failures by breaking the circuit after failures.
5
+ *
6
+ * @module v3/shared/resilience/circuit-breaker
7
+ */
8
+ import { EventEmitter } from 'events';
9
+ /**
10
+ * Circuit breaker states
11
+ */
12
+ export declare enum CircuitBreakerState {
13
+ /** Circuit is closed, requests flow normally */
14
+ CLOSED = "CLOSED",
15
+ /** Circuit is open, requests are rejected immediately */
16
+ OPEN = "OPEN",
17
+ /** Circuit is testing if service recovered */
18
+ HALF_OPEN = "HALF_OPEN"
19
+ }
20
+ /**
21
+ * Circuit breaker options
22
+ */
23
+ export interface CircuitBreakerOptions {
24
+ /** Name for identification */
25
+ name: string;
26
+ /** Failure threshold before opening circuit (default: 5) */
27
+ failureThreshold: number;
28
+ /** Success threshold in half-open state to close circuit (default: 3) */
29
+ successThreshold: number;
30
+ /** Time to wait before testing again in ms (default: 30000) */
31
+ timeout: number;
32
+ /** Time window to track failures in ms (default: 60000) */
33
+ rollingWindow: number;
34
+ /** Volume threshold - minimum requests before tripping (default: 10) */
35
+ volumeThreshold: number;
36
+ /** Custom failure detection */
37
+ isFailure?: (error: Error) => boolean;
38
+ /** Fallback function when circuit is open */
39
+ fallback?: <T>(error: Error) => T | Promise<T>;
40
+ /** Callback when state changes */
41
+ onStateChange?: (from: CircuitBreakerState, to: CircuitBreakerState) => void;
42
+ }
43
+ /**
44
+ * Circuit breaker statistics
45
+ */
46
+ export interface CircuitBreakerStats {
47
+ state: CircuitBreakerState;
48
+ failures: number;
49
+ successes: number;
50
+ totalRequests: number;
51
+ rejectedRequests: number;
52
+ lastFailure: Date | null;
53
+ lastSuccess: Date | null;
54
+ openSince: Date | null;
55
+ }
56
+ /**
57
+ * Circuit Breaker
58
+ *
59
+ * Implements the circuit breaker pattern to prevent cascading failures.
60
+ *
61
+ * @example
62
+ * const breaker = new CircuitBreaker({
63
+ * name: 'external-api',
64
+ * failureThreshold: 5,
65
+ * timeout: 30000,
66
+ * });
67
+ *
68
+ * try {
69
+ * const result = await breaker.execute(() => fetchExternalAPI());
70
+ * } catch (error) {
71
+ * if (error.message === 'Circuit is open') {
72
+ * // Handle circuit open case
73
+ * }
74
+ * }
75
+ */
76
+ export declare class CircuitBreaker extends EventEmitter {
77
+ private readonly options;
78
+ private state;
79
+ private requests;
80
+ private halfOpenSuccesses;
81
+ private openedAt;
82
+ private lastFailure;
83
+ private lastSuccess;
84
+ private rejectedCount;
85
+ private timeoutId?;
86
+ constructor(options: CircuitBreakerOptions);
87
+ /**
88
+ * Execute a function through the circuit breaker
89
+ */
90
+ execute<T>(fn: () => Promise<T>): Promise<T>;
91
+ /**
92
+ * Get current state
93
+ */
94
+ getState(): CircuitBreakerState;
95
+ /**
96
+ * Get statistics
97
+ */
98
+ getStats(): CircuitBreakerStats;
99
+ /**
100
+ * Force reset the circuit breaker
101
+ */
102
+ reset(): void;
103
+ /**
104
+ * Handle successful request
105
+ */
106
+ private onSuccess;
107
+ /**
108
+ * Handle failed request
109
+ */
110
+ private onFailure;
111
+ /**
112
+ * Check if state should change based on timeout
113
+ */
114
+ private checkState;
115
+ /**
116
+ * Transition to new state
117
+ */
118
+ private transitionTo;
119
+ /**
120
+ * Schedule transition to half-open
121
+ */
122
+ private scheduleHalfOpen;
123
+ /**
124
+ * Notify state change
125
+ */
126
+ private notifyStateChange;
127
+ /**
128
+ * Clean old requests outside rolling window
129
+ */
130
+ private cleanOldRequests;
131
+ }
132
+ //# sourceMappingURL=circuit-breaker.d.ts.map
@@ -0,0 +1,233 @@
1
+ /**
2
+ * Circuit Breaker Pattern
3
+ *
4
+ * Prevents cascading failures by breaking the circuit after failures.
5
+ *
6
+ * @module v3/shared/resilience/circuit-breaker
7
+ */
8
+ import { EventEmitter } from 'events';
9
+ /**
10
+ * Circuit breaker states
11
+ */
12
+ export var CircuitBreakerState;
13
+ (function (CircuitBreakerState) {
14
+ /** Circuit is closed, requests flow normally */
15
+ CircuitBreakerState["CLOSED"] = "CLOSED";
16
+ /** Circuit is open, requests are rejected immediately */
17
+ CircuitBreakerState["OPEN"] = "OPEN";
18
+ /** Circuit is testing if service recovered */
19
+ CircuitBreakerState["HALF_OPEN"] = "HALF_OPEN";
20
+ })(CircuitBreakerState || (CircuitBreakerState = {}));
21
+ /**
22
+ * Default options
23
+ */
24
+ const DEFAULT_OPTIONS = {
25
+ failureThreshold: 5,
26
+ successThreshold: 3,
27
+ timeout: 30000,
28
+ rollingWindow: 60000,
29
+ volumeThreshold: 10,
30
+ };
31
+ /**
32
+ * Circuit Breaker
33
+ *
34
+ * Implements the circuit breaker pattern to prevent cascading failures.
35
+ *
36
+ * @example
37
+ * const breaker = new CircuitBreaker({
38
+ * name: 'external-api',
39
+ * failureThreshold: 5,
40
+ * timeout: 30000,
41
+ * });
42
+ *
43
+ * try {
44
+ * const result = await breaker.execute(() => fetchExternalAPI());
45
+ * } catch (error) {
46
+ * if (error.message === 'Circuit is open') {
47
+ * // Handle circuit open case
48
+ * }
49
+ * }
50
+ */
51
+ export class CircuitBreaker extends EventEmitter {
52
+ options;
53
+ state = CircuitBreakerState.CLOSED;
54
+ requests = [];
55
+ halfOpenSuccesses = 0;
56
+ openedAt = null;
57
+ lastFailure = null;
58
+ lastSuccess = null;
59
+ rejectedCount = 0;
60
+ timeoutId;
61
+ constructor(options) {
62
+ super();
63
+ this.options = { ...DEFAULT_OPTIONS, ...options };
64
+ }
65
+ /**
66
+ * Execute a function through the circuit breaker
67
+ */
68
+ async execute(fn) {
69
+ // Clean up old requests
70
+ this.cleanOldRequests();
71
+ // Check if circuit should be tested
72
+ this.checkState();
73
+ // If open, reject immediately or use fallback
74
+ if (this.state === CircuitBreakerState.OPEN) {
75
+ this.rejectedCount++;
76
+ const error = new Error(`Circuit breaker '${this.options.name}' is open`);
77
+ if (this.options.fallback) {
78
+ return this.options.fallback(error);
79
+ }
80
+ throw error;
81
+ }
82
+ try {
83
+ const result = await fn();
84
+ this.onSuccess();
85
+ return result;
86
+ }
87
+ catch (error) {
88
+ const err = error instanceof Error ? error : new Error(String(error));
89
+ // Check if this should be counted as failure
90
+ const isFailure = this.options.isFailure?.(err) ?? true;
91
+ if (isFailure) {
92
+ this.onFailure(err);
93
+ }
94
+ throw error;
95
+ }
96
+ }
97
+ /**
98
+ * Get current state
99
+ */
100
+ getState() {
101
+ this.checkState();
102
+ return this.state;
103
+ }
104
+ /**
105
+ * Get statistics
106
+ */
107
+ getStats() {
108
+ this.cleanOldRequests();
109
+ return {
110
+ state: this.state,
111
+ failures: this.requests.filter((r) => !r.success).length,
112
+ successes: this.requests.filter((r) => r.success).length,
113
+ totalRequests: this.requests.length,
114
+ rejectedRequests: this.rejectedCount,
115
+ lastFailure: this.lastFailure,
116
+ lastSuccess: this.lastSuccess,
117
+ openSince: this.openedAt,
118
+ };
119
+ }
120
+ /**
121
+ * Force reset the circuit breaker
122
+ */
123
+ reset() {
124
+ const previousState = this.state;
125
+ this.state = CircuitBreakerState.CLOSED;
126
+ this.requests = [];
127
+ this.halfOpenSuccesses = 0;
128
+ this.openedAt = null;
129
+ if (this.timeoutId) {
130
+ clearTimeout(this.timeoutId);
131
+ this.timeoutId = undefined;
132
+ }
133
+ if (previousState !== this.state) {
134
+ this.notifyStateChange(previousState, this.state);
135
+ }
136
+ }
137
+ /**
138
+ * Handle successful request
139
+ */
140
+ onSuccess() {
141
+ this.lastSuccess = new Date();
142
+ this.requests.push({ timestamp: Date.now(), success: true });
143
+ if (this.state === CircuitBreakerState.HALF_OPEN) {
144
+ this.halfOpenSuccesses++;
145
+ if (this.halfOpenSuccesses >= this.options.successThreshold) {
146
+ this.transitionTo(CircuitBreakerState.CLOSED);
147
+ this.halfOpenSuccesses = 0;
148
+ }
149
+ }
150
+ }
151
+ /**
152
+ * Handle failed request
153
+ */
154
+ onFailure(error) {
155
+ this.lastFailure = new Date();
156
+ this.requests.push({ timestamp: Date.now(), success: false });
157
+ if (this.state === CircuitBreakerState.HALF_OPEN) {
158
+ // Failed during half-open, go back to open
159
+ this.transitionTo(CircuitBreakerState.OPEN);
160
+ this.halfOpenSuccesses = 0;
161
+ return;
162
+ }
163
+ // Check if we should open the circuit
164
+ const failures = this.requests.filter((r) => !r.success).length;
165
+ const totalRequests = this.requests.length;
166
+ if (totalRequests >= this.options.volumeThreshold &&
167
+ failures >= this.options.failureThreshold) {
168
+ this.transitionTo(CircuitBreakerState.OPEN);
169
+ }
170
+ }
171
+ /**
172
+ * Check if state should change based on timeout
173
+ */
174
+ checkState() {
175
+ if (this.state === CircuitBreakerState.OPEN && this.openedAt) {
176
+ const elapsed = Date.now() - this.openedAt.getTime();
177
+ if (elapsed >= this.options.timeout) {
178
+ this.transitionTo(CircuitBreakerState.HALF_OPEN);
179
+ }
180
+ }
181
+ }
182
+ /**
183
+ * Transition to new state
184
+ */
185
+ transitionTo(newState) {
186
+ const previousState = this.state;
187
+ if (previousState === newState) {
188
+ return;
189
+ }
190
+ this.state = newState;
191
+ if (newState === CircuitBreakerState.OPEN) {
192
+ this.openedAt = new Date();
193
+ this.scheduleHalfOpen();
194
+ }
195
+ else if (newState === CircuitBreakerState.CLOSED) {
196
+ this.openedAt = null;
197
+ this.requests = [];
198
+ if (this.timeoutId) {
199
+ clearTimeout(this.timeoutId);
200
+ this.timeoutId = undefined;
201
+ }
202
+ }
203
+ this.notifyStateChange(previousState, newState);
204
+ }
205
+ /**
206
+ * Schedule transition to half-open
207
+ */
208
+ scheduleHalfOpen() {
209
+ if (this.timeoutId) {
210
+ clearTimeout(this.timeoutId);
211
+ }
212
+ this.timeoutId = setTimeout(() => {
213
+ if (this.state === CircuitBreakerState.OPEN) {
214
+ this.transitionTo(CircuitBreakerState.HALF_OPEN);
215
+ }
216
+ }, this.options.timeout);
217
+ }
218
+ /**
219
+ * Notify state change
220
+ */
221
+ notifyStateChange(from, to) {
222
+ this.emit('stateChange', { from, to });
223
+ this.options.onStateChange?.(from, to);
224
+ }
225
+ /**
226
+ * Clean old requests outside rolling window
227
+ */
228
+ cleanOldRequests() {
229
+ const cutoff = Date.now() - this.options.rollingWindow;
230
+ this.requests = this.requests.filter((r) => r.timestamp >= cutoff);
231
+ }
232
+ }
233
+ //# sourceMappingURL=circuit-breaker.js.map
@@ -0,0 +1,19 @@
1
+ /**
2
+ * Resilience Patterns
3
+ *
4
+ * Production-ready resilience utilities:
5
+ * - Retry with exponential backoff
6
+ * - Circuit breaker pattern
7
+ * - Rate limiting
8
+ *
9
+ * @module v3/shared/resilience
10
+ */
11
+ export { retry, RetryError } from './retry.js';
12
+ export type { RetryOptions, RetryResult } from './retry.js';
13
+ export { CircuitBreaker } from './circuit-breaker.js';
14
+ export type { CircuitBreakerOptions, CircuitBreakerStats } from './circuit-breaker.js';
15
+ export { SlidingWindowRateLimiter, TokenBucketRateLimiter } from './rate-limiter.js';
16
+ export type { RateLimiter, RateLimiterOptions, RateLimitResult } from './rate-limiter.js';
17
+ export { Bulkhead } from './bulkhead.js';
18
+ export type { BulkheadOptions, BulkheadStats } from './bulkhead.js';
19
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1,19 @@
1
+ /**
2
+ * Resilience Patterns
3
+ *
4
+ * Production-ready resilience utilities:
5
+ * - Retry with exponential backoff
6
+ * - Circuit breaker pattern
7
+ * - Rate limiting
8
+ *
9
+ * @module v3/shared/resilience
10
+ */
11
+ // Retry
12
+ export { retry, RetryError } from './retry.js';
13
+ // Circuit Breaker
14
+ export { CircuitBreaker } from './circuit-breaker.js';
15
+ // Rate Limiter
16
+ export { SlidingWindowRateLimiter, TokenBucketRateLimiter } from './rate-limiter.js';
17
+ // Bulkhead
18
+ export { Bulkhead } from './bulkhead.js';
19
+ //# sourceMappingURL=index.js.map