forge-server 0.1.0

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 (412) hide show
  1. package/.claude/hooks/worktree-create.sh +64 -0
  2. package/.claude/hooks/worktree-remove.sh +57 -0
  3. package/.claude/settings.local.json +29 -0
  4. package/.forge/knowledge/conventions.yaml +1 -0
  5. package/.forge/knowledge/decisions.yaml +1 -0
  6. package/.forge/knowledge/gotchas.yaml +1 -0
  7. package/.forge/knowledge/patterns.yaml +1 -0
  8. package/.forge/manifest.yaml +6 -0
  9. package/CLAUDE.md +144 -0
  10. package/bin/setup-forge.sh +132 -0
  11. package/dist/cli.d.ts +3 -0
  12. package/dist/cli.d.ts.map +1 -0
  13. package/dist/cli.js +553 -0
  14. package/dist/cli.js.map +1 -0
  15. package/dist/context/codebase.d.ts +57 -0
  16. package/dist/context/codebase.d.ts.map +1 -0
  17. package/dist/context/codebase.js +301 -0
  18. package/dist/context/codebase.js.map +1 -0
  19. package/dist/context/injector.d.ts +147 -0
  20. package/dist/context/injector.d.ts.map +1 -0
  21. package/dist/context/injector.js +533 -0
  22. package/dist/context/injector.js.map +1 -0
  23. package/dist/context/memory.d.ts +32 -0
  24. package/dist/context/memory.d.ts.map +1 -0
  25. package/dist/context/memory.js +140 -0
  26. package/dist/context/memory.js.map +1 -0
  27. package/dist/context/session-index.d.ts +54 -0
  28. package/dist/context/session-index.d.ts.map +1 -0
  29. package/dist/context/session-index.js +265 -0
  30. package/dist/context/session-index.js.map +1 -0
  31. package/dist/context/session.d.ts +42 -0
  32. package/dist/context/session.d.ts.map +1 -0
  33. package/dist/context/session.js +121 -0
  34. package/dist/context/session.js.map +1 -0
  35. package/dist/index.d.ts +3 -0
  36. package/dist/index.d.ts.map +1 -0
  37. package/dist/index.js +37 -0
  38. package/dist/index.js.map +1 -0
  39. package/dist/ingestion/chunker.d.ts +19 -0
  40. package/dist/ingestion/chunker.d.ts.map +1 -0
  41. package/dist/ingestion/chunker.js +189 -0
  42. package/dist/ingestion/chunker.js.map +1 -0
  43. package/dist/ingestion/embedder.d.ts +45 -0
  44. package/dist/ingestion/embedder.d.ts.map +1 -0
  45. package/dist/ingestion/embedder.js +152 -0
  46. package/dist/ingestion/embedder.js.map +1 -0
  47. package/dist/ingestion/git-analyzer.d.ts +77 -0
  48. package/dist/ingestion/git-analyzer.d.ts.map +1 -0
  49. package/dist/ingestion/git-analyzer.js +437 -0
  50. package/dist/ingestion/git-analyzer.js.map +1 -0
  51. package/dist/ingestion/indexer.d.ts +79 -0
  52. package/dist/ingestion/indexer.d.ts.map +1 -0
  53. package/dist/ingestion/indexer.js +766 -0
  54. package/dist/ingestion/indexer.js.map +1 -0
  55. package/dist/ingestion/markdown-chunker.d.ts +19 -0
  56. package/dist/ingestion/markdown-chunker.d.ts.map +1 -0
  57. package/dist/ingestion/markdown-chunker.js +243 -0
  58. package/dist/ingestion/markdown-chunker.js.map +1 -0
  59. package/dist/ingestion/markdown-knowledge.d.ts +21 -0
  60. package/dist/ingestion/markdown-knowledge.d.ts.map +1 -0
  61. package/dist/ingestion/markdown-knowledge.js +129 -0
  62. package/dist/ingestion/markdown-knowledge.js.map +1 -0
  63. package/dist/ingestion/parser.d.ts +20 -0
  64. package/dist/ingestion/parser.d.ts.map +1 -0
  65. package/dist/ingestion/parser.js +429 -0
  66. package/dist/ingestion/parser.js.map +1 -0
  67. package/dist/ingestion/watcher.d.ts +28 -0
  68. package/dist/ingestion/watcher.d.ts.map +1 -0
  69. package/dist/ingestion/watcher.js +147 -0
  70. package/dist/ingestion/watcher.js.map +1 -0
  71. package/dist/knowledge/hydrator.d.ts +37 -0
  72. package/dist/knowledge/hydrator.d.ts.map +1 -0
  73. package/dist/knowledge/hydrator.js +220 -0
  74. package/dist/knowledge/hydrator.js.map +1 -0
  75. package/dist/knowledge/registry.d.ts +129 -0
  76. package/dist/knowledge/registry.d.ts.map +1 -0
  77. package/dist/knowledge/registry.js +361 -0
  78. package/dist/knowledge/registry.js.map +1 -0
  79. package/dist/knowledge/search.d.ts +114 -0
  80. package/dist/knowledge/search.d.ts.map +1 -0
  81. package/dist/knowledge/search.js +428 -0
  82. package/dist/knowledge/search.js.map +1 -0
  83. package/dist/knowledge/store.d.ts +76 -0
  84. package/dist/knowledge/store.d.ts.map +1 -0
  85. package/dist/knowledge/store.js +230 -0
  86. package/dist/knowledge/store.js.map +1 -0
  87. package/dist/learning/confidence.d.ts +30 -0
  88. package/dist/learning/confidence.d.ts.map +1 -0
  89. package/dist/learning/confidence.js +165 -0
  90. package/dist/learning/confidence.js.map +1 -0
  91. package/dist/learning/patterns.d.ts +52 -0
  92. package/dist/learning/patterns.d.ts.map +1 -0
  93. package/dist/learning/patterns.js +290 -0
  94. package/dist/learning/patterns.js.map +1 -0
  95. package/dist/learning/trajectory.d.ts +55 -0
  96. package/dist/learning/trajectory.d.ts.map +1 -0
  97. package/dist/learning/trajectory.js +200 -0
  98. package/dist/learning/trajectory.js.map +1 -0
  99. package/dist/memory/memory-compat.d.ts +100 -0
  100. package/dist/memory/memory-compat.d.ts.map +1 -0
  101. package/dist/memory/memory-compat.js +146 -0
  102. package/dist/memory/memory-compat.js.map +1 -0
  103. package/dist/memory/observation-store.d.ts +57 -0
  104. package/dist/memory/observation-store.d.ts.map +1 -0
  105. package/dist/memory/observation-store.js +154 -0
  106. package/dist/memory/observation-store.js.map +1 -0
  107. package/dist/memory/session-tracker.d.ts +81 -0
  108. package/dist/memory/session-tracker.d.ts.map +1 -0
  109. package/dist/memory/session-tracker.js +262 -0
  110. package/dist/memory/session-tracker.js.map +1 -0
  111. package/dist/pipeline/engine.d.ts +179 -0
  112. package/dist/pipeline/engine.d.ts.map +1 -0
  113. package/dist/pipeline/engine.js +691 -0
  114. package/dist/pipeline/engine.js.map +1 -0
  115. package/dist/pipeline/events.d.ts +54 -0
  116. package/dist/pipeline/events.d.ts.map +1 -0
  117. package/dist/pipeline/events.js +157 -0
  118. package/dist/pipeline/events.js.map +1 -0
  119. package/dist/pipeline/parallel.d.ts +83 -0
  120. package/dist/pipeline/parallel.d.ts.map +1 -0
  121. package/dist/pipeline/parallel.js +277 -0
  122. package/dist/pipeline/parallel.js.map +1 -0
  123. package/dist/pipeline/state-machine.d.ts +65 -0
  124. package/dist/pipeline/state-machine.d.ts.map +1 -0
  125. package/dist/pipeline/state-machine.js +176 -0
  126. package/dist/pipeline/state-machine.js.map +1 -0
  127. package/dist/query/graph-queries.d.ts +84 -0
  128. package/dist/query/graph-queries.d.ts.map +1 -0
  129. package/dist/query/graph-queries.js +216 -0
  130. package/dist/query/graph-queries.js.map +1 -0
  131. package/dist/query/hybrid-search.d.ts +34 -0
  132. package/dist/query/hybrid-search.d.ts.map +1 -0
  133. package/dist/query/hybrid-search.js +263 -0
  134. package/dist/query/hybrid-search.js.map +1 -0
  135. package/dist/query/intent-detector.d.ts +35 -0
  136. package/dist/query/intent-detector.d.ts.map +1 -0
  137. package/dist/query/intent-detector.js +115 -0
  138. package/dist/query/intent-detector.js.map +1 -0
  139. package/dist/query/ranking.d.ts +57 -0
  140. package/dist/query/ranking.d.ts.map +1 -0
  141. package/dist/query/ranking.js +109 -0
  142. package/dist/query/ranking.js.map +1 -0
  143. package/dist/server.d.ts +3 -0
  144. package/dist/server.d.ts.map +1 -0
  145. package/dist/server.js +291 -0
  146. package/dist/server.js.map +1 -0
  147. package/dist/storage/falkordb-store.d.ts +73 -0
  148. package/dist/storage/falkordb-store.d.ts.map +1 -0
  149. package/dist/storage/falkordb-store.js +346 -0
  150. package/dist/storage/falkordb-store.js.map +1 -0
  151. package/dist/storage/file-cache.d.ts +32 -0
  152. package/dist/storage/file-cache.d.ts.map +1 -0
  153. package/dist/storage/file-cache.js +115 -0
  154. package/dist/storage/file-cache.js.map +1 -0
  155. package/dist/storage/interfaces.d.ts +151 -0
  156. package/dist/storage/interfaces.d.ts.map +1 -0
  157. package/dist/storage/interfaces.js +7 -0
  158. package/dist/storage/interfaces.js.map +1 -0
  159. package/dist/storage/qdrant-store.d.ts +110 -0
  160. package/dist/storage/qdrant-store.d.ts.map +1 -0
  161. package/dist/storage/qdrant-store.js +467 -0
  162. package/dist/storage/qdrant-store.js.map +1 -0
  163. package/dist/storage/schema.d.ts +4 -0
  164. package/dist/storage/schema.d.ts.map +1 -0
  165. package/dist/storage/schema.js +136 -0
  166. package/dist/storage/schema.js.map +1 -0
  167. package/dist/storage/sqlite.d.ts +35 -0
  168. package/dist/storage/sqlite.d.ts.map +1 -0
  169. package/dist/storage/sqlite.js +132 -0
  170. package/dist/storage/sqlite.js.map +1 -0
  171. package/dist/tools/collaboration-tools.d.ts +111 -0
  172. package/dist/tools/collaboration-tools.d.ts.map +1 -0
  173. package/dist/tools/collaboration-tools.js +174 -0
  174. package/dist/tools/collaboration-tools.js.map +1 -0
  175. package/dist/tools/context-tools.d.ts +293 -0
  176. package/dist/tools/context-tools.d.ts.map +1 -0
  177. package/dist/tools/context-tools.js +437 -0
  178. package/dist/tools/context-tools.js.map +1 -0
  179. package/dist/tools/graph-tools.d.ts +129 -0
  180. package/dist/tools/graph-tools.d.ts.map +1 -0
  181. package/dist/tools/graph-tools.js +237 -0
  182. package/dist/tools/graph-tools.js.map +1 -0
  183. package/dist/tools/ingestion-tools.d.ts +96 -0
  184. package/dist/tools/ingestion-tools.d.ts.map +1 -0
  185. package/dist/tools/ingestion-tools.js +90 -0
  186. package/dist/tools/ingestion-tools.js.map +1 -0
  187. package/dist/tools/learning-tools.d.ts +168 -0
  188. package/dist/tools/learning-tools.d.ts.map +1 -0
  189. package/dist/tools/learning-tools.js +158 -0
  190. package/dist/tools/learning-tools.js.map +1 -0
  191. package/dist/tools/memory-tools.d.ts +183 -0
  192. package/dist/tools/memory-tools.d.ts.map +1 -0
  193. package/dist/tools/memory-tools.js +197 -0
  194. package/dist/tools/memory-tools.js.map +1 -0
  195. package/dist/tools/phase-tools.d.ts +954 -0
  196. package/dist/tools/phase-tools.d.ts.map +1 -0
  197. package/dist/tools/phase-tools.js +1215 -0
  198. package/dist/tools/phase-tools.js.map +1 -0
  199. package/dist/tools/pipeline-tools.d.ts +140 -0
  200. package/dist/tools/pipeline-tools.d.ts.map +1 -0
  201. package/dist/tools/pipeline-tools.js +162 -0
  202. package/dist/tools/pipeline-tools.js.map +1 -0
  203. package/dist/tools/registration-tools.d.ts +220 -0
  204. package/dist/tools/registration-tools.d.ts.map +1 -0
  205. package/dist/tools/registration-tools.js +391 -0
  206. package/dist/tools/registration-tools.js.map +1 -0
  207. package/dist/util/circuit-breaker.d.ts +75 -0
  208. package/dist/util/circuit-breaker.d.ts.map +1 -0
  209. package/dist/util/circuit-breaker.js +159 -0
  210. package/dist/util/circuit-breaker.js.map +1 -0
  211. package/dist/util/config.d.ts +23 -0
  212. package/dist/util/config.d.ts.map +1 -0
  213. package/dist/util/config.js +164 -0
  214. package/dist/util/config.js.map +1 -0
  215. package/dist/util/logger.d.ts +13 -0
  216. package/dist/util/logger.d.ts.map +1 -0
  217. package/dist/util/logger.js +45 -0
  218. package/dist/util/logger.js.map +1 -0
  219. package/dist/util/token-counter.d.ts +24 -0
  220. package/dist/util/token-counter.d.ts.map +1 -0
  221. package/dist/util/token-counter.js +48 -0
  222. package/dist/util/token-counter.js.map +1 -0
  223. package/dist/util/types.d.ts +525 -0
  224. package/dist/util/types.d.ts.map +1 -0
  225. package/dist/util/types.js +5 -0
  226. package/dist/util/types.js.map +1 -0
  227. package/docker-compose.yml +20 -0
  228. package/docs/plans/2026-02-27-swarm-coordination/architecture.md +203 -0
  229. package/docs/plans/2026-02-27-swarm-coordination/vision.md +57 -0
  230. package/docs/plans/completed/2026-02-26-forge-plugin-bundling/architecture.md +1 -0
  231. package/docs/plans/completed/2026-02-26-forge-plugin-bundling/vision.md +300 -0
  232. package/docs/plans/completed/2026-02-27-forge-swarm-learning/architecture.md +480 -0
  233. package/docs/plans/completed/2026-02-27-forge-swarm-learning/verification-checklist.md +462 -0
  234. package/docs/plans/completed/2026-02-27-git-history-atlassian/git-jira-plan.md +181 -0
  235. package/package.json +39 -0
  236. package/plugin/.claude-plugin/plugin.json +8 -0
  237. package/plugin/.mcp.json +15 -0
  238. package/plugin/README.md +134 -0
  239. package/plugin/agents/architect.md +367 -0
  240. package/plugin/agents/backend-specialist.md +263 -0
  241. package/plugin/agents/brainstormer.md +122 -0
  242. package/plugin/agents/data-specialist.md +266 -0
  243. package/plugin/agents/designer.md +408 -0
  244. package/plugin/agents/frontend-specialist.md +241 -0
  245. package/plugin/agents/inspector.md +406 -0
  246. package/plugin/agents/knowledge-keeper.md +443 -0
  247. package/plugin/agents/platform-engineer.md +326 -0
  248. package/plugin/agents/product-manager.md +268 -0
  249. package/plugin/agents/product-owner.md +438 -0
  250. package/plugin/agents/pulse-checker.md +73 -0
  251. package/plugin/agents/qa-strategist.md +500 -0
  252. package/plugin/agents/self-improver.md +310 -0
  253. package/plugin/agents/strategist.md +360 -0
  254. package/plugin/agents/supervisor.md +380 -0
  255. package/plugin/commands/brainstorm.md +25 -0
  256. package/plugin/commands/forge.md +88 -0
  257. package/plugin/docs/atlassian-integration.md +110 -0
  258. package/plugin/docs/workflow.md +126 -0
  259. package/plugin/skills/agent-development/.skillfish.json +10 -0
  260. package/plugin/skills/agent-development/SKILL.md +415 -0
  261. package/plugin/skills/agent-development/examples/agent-creation-prompt.md +238 -0
  262. package/plugin/skills/agent-development/examples/complete-agent-examples.md +427 -0
  263. package/plugin/skills/agent-development/references/agent-creation-system-prompt.md +207 -0
  264. package/plugin/skills/agent-development/references/system-prompt-design.md +411 -0
  265. package/plugin/skills/agent-development/references/triggering-examples.md +491 -0
  266. package/plugin/skills/agent-development/scripts/validate-agent.sh +217 -0
  267. package/plugin/skills/agent-handoff/SKILL.md +335 -0
  268. package/plugin/skills/anti-stub/SKILL.md +317 -0
  269. package/plugin/skills/brainstorm/SKILL.md +31 -0
  270. package/plugin/skills/debugging/SKILL.md +276 -0
  271. package/plugin/skills/fix/SKILL.md +62 -0
  272. package/plugin/skills/frontend-design/.skillfish.json +10 -0
  273. package/plugin/skills/frontend-design/SKILL.md +42 -0
  274. package/plugin/skills/gotchas/SKILL.md +61 -0
  275. package/plugin/skills/graph-orchestrator/SKILL.md +38 -0
  276. package/plugin/skills/history/SKILL.md +58 -0
  277. package/plugin/skills/impact/SKILL.md +59 -0
  278. package/plugin/skills/implementation-execution/SKILL.md +291 -0
  279. package/plugin/skills/index-repo/SKILL.md +55 -0
  280. package/plugin/skills/interviewing/SKILL.md +225 -0
  281. package/plugin/skills/knowledge-curation/SKILL.md +393 -0
  282. package/plugin/skills/learn/SKILL.md +69 -0
  283. package/plugin/skills/mcp-integration/.skillfish.json +10 -0
  284. package/plugin/skills/mcp-integration/SKILL.md +554 -0
  285. package/plugin/skills/mcp-integration/examples/http-server.json +20 -0
  286. package/plugin/skills/mcp-integration/examples/sse-server.json +19 -0
  287. package/plugin/skills/mcp-integration/examples/stdio-server.json +26 -0
  288. package/plugin/skills/mcp-integration/references/authentication.md +549 -0
  289. package/plugin/skills/mcp-integration/references/server-types.md +536 -0
  290. package/plugin/skills/mcp-integration/references/tool-usage.md +538 -0
  291. package/plugin/skills/nestjs/.skillfish.json +10 -0
  292. package/plugin/skills/nestjs/SKILL.md +669 -0
  293. package/plugin/skills/nestjs/drizzle-reference.md +1894 -0
  294. package/plugin/skills/nestjs/reference.md +1447 -0
  295. package/plugin/skills/nestjs/workflow-optimization.md +229 -0
  296. package/plugin/skills/parallel-dispatch/SKILL.md +308 -0
  297. package/plugin/skills/project-discovery/SKILL.md +304 -0
  298. package/plugin/skills/search/SKILL.md +56 -0
  299. package/plugin/skills/security-audit/SKILL.md +362 -0
  300. package/plugin/skills/skill-development/.skillfish.json +10 -0
  301. package/plugin/skills/skill-development/SKILL.md +637 -0
  302. package/plugin/skills/skill-development/references/skill-creator-original.md +209 -0
  303. package/plugin/skills/tdd/SKILL.md +273 -0
  304. package/plugin/skills/terminal-presentation/SKILL.md +395 -0
  305. package/plugin/skills/test-strategy/SKILL.md +365 -0
  306. package/plugin/skills/verification-protocol/SKILL.md +256 -0
  307. package/plugin/skills/visual-explainer/CHANGELOG.md +97 -0
  308. package/plugin/skills/visual-explainer/LICENSE +21 -0
  309. package/plugin/skills/visual-explainer/README.md +137 -0
  310. package/plugin/skills/visual-explainer/SKILL.md +352 -0
  311. package/plugin/skills/visual-explainer/banner.png +0 -0
  312. package/plugin/skills/visual-explainer/package.json +11 -0
  313. package/plugin/skills/visual-explainer/prompts/diff-review.md +68 -0
  314. package/plugin/skills/visual-explainer/prompts/fact-check.md +63 -0
  315. package/plugin/skills/visual-explainer/prompts/generate-slides.md +18 -0
  316. package/plugin/skills/visual-explainer/prompts/generate-web-diagram.md +10 -0
  317. package/plugin/skills/visual-explainer/prompts/plan-review.md +86 -0
  318. package/plugin/skills/visual-explainer/prompts/project-recap.md +61 -0
  319. package/plugin/skills/visual-explainer/references/css-patterns.md +1188 -0
  320. package/plugin/skills/visual-explainer/references/libraries.md +470 -0
  321. package/plugin/skills/visual-explainer/references/responsive-nav.md +212 -0
  322. package/plugin/skills/visual-explainer/references/slide-patterns.md +1403 -0
  323. package/plugin/skills/visual-explainer/templates/architecture.html +596 -0
  324. package/plugin/skills/visual-explainer/templates/data-table.html +540 -0
  325. package/plugin/skills/visual-explainer/templates/mermaid-flowchart.html +435 -0
  326. package/plugin/skills/visual-explainer/templates/slide-deck.html +913 -0
  327. package/src/cli.ts +655 -0
  328. package/src/context/.gitkeep +0 -0
  329. package/src/context/codebase.ts +393 -0
  330. package/src/context/injector.ts +797 -0
  331. package/src/context/memory.ts +187 -0
  332. package/src/context/session-index.ts +327 -0
  333. package/src/context/session.ts +152 -0
  334. package/src/index.ts +47 -0
  335. package/src/ingestion/.gitkeep +0 -0
  336. package/src/ingestion/chunker.ts +277 -0
  337. package/src/ingestion/embedder.ts +167 -0
  338. package/src/ingestion/git-analyzer.ts +545 -0
  339. package/src/ingestion/indexer.ts +984 -0
  340. package/src/ingestion/markdown-chunker.ts +337 -0
  341. package/src/ingestion/markdown-knowledge.ts +175 -0
  342. package/src/ingestion/parser.ts +475 -0
  343. package/src/ingestion/watcher.ts +182 -0
  344. package/src/knowledge/.gitkeep +0 -0
  345. package/src/knowledge/hydrator.ts +246 -0
  346. package/src/knowledge/registry.ts +463 -0
  347. package/src/knowledge/search.ts +565 -0
  348. package/src/knowledge/store.ts +262 -0
  349. package/src/learning/.gitkeep +0 -0
  350. package/src/learning/confidence.ts +193 -0
  351. package/src/learning/patterns.ts +360 -0
  352. package/src/learning/trajectory.ts +268 -0
  353. package/src/memory/.gitkeep +0 -0
  354. package/src/memory/memory-compat.ts +233 -0
  355. package/src/memory/observation-store.ts +224 -0
  356. package/src/memory/session-tracker.ts +332 -0
  357. package/src/pipeline/.gitkeep +0 -0
  358. package/src/pipeline/engine.ts +1139 -0
  359. package/src/pipeline/events.ts +253 -0
  360. package/src/pipeline/parallel.ts +394 -0
  361. package/src/pipeline/state-machine.ts +199 -0
  362. package/src/query/.gitkeep +0 -0
  363. package/src/query/graph-queries.ts +262 -0
  364. package/src/query/hybrid-search.ts +337 -0
  365. package/src/query/intent-detector.ts +131 -0
  366. package/src/query/ranking.ts +161 -0
  367. package/src/server.ts +352 -0
  368. package/src/storage/.gitkeep +0 -0
  369. package/src/storage/falkordb-store.ts +388 -0
  370. package/src/storage/file-cache.ts +141 -0
  371. package/src/storage/interfaces.ts +201 -0
  372. package/src/storage/qdrant-store.ts +557 -0
  373. package/src/storage/schema.ts +139 -0
  374. package/src/storage/sqlite.ts +168 -0
  375. package/src/tools/.gitkeep +0 -0
  376. package/src/tools/collaboration-tools.ts +208 -0
  377. package/src/tools/context-tools.ts +493 -0
  378. package/src/tools/graph-tools.ts +295 -0
  379. package/src/tools/ingestion-tools.ts +122 -0
  380. package/src/tools/learning-tools.ts +181 -0
  381. package/src/tools/memory-tools.ts +234 -0
  382. package/src/tools/phase-tools.ts +1452 -0
  383. package/src/tools/pipeline-tools.ts +188 -0
  384. package/src/tools/registration-tools.ts +450 -0
  385. package/src/util/.gitkeep +0 -0
  386. package/src/util/circuit-breaker.ts +193 -0
  387. package/src/util/config.ts +177 -0
  388. package/src/util/logger.ts +53 -0
  389. package/src/util/token-counter.ts +52 -0
  390. package/src/util/types.ts +710 -0
  391. package/tests/context/.gitkeep +0 -0
  392. package/tests/integration/.gitkeep +0 -0
  393. package/tests/knowledge/.gitkeep +0 -0
  394. package/tests/learning/.gitkeep +0 -0
  395. package/tests/pipeline/.gitkeep +0 -0
  396. package/tests/tools/.gitkeep +0 -0
  397. package/tsconfig.json +21 -0
  398. package/vitest.config.ts +10 -0
  399. package/vscode-extension/.vscodeignore +7 -0
  400. package/vscode-extension/README.md +43 -0
  401. package/vscode-extension/out/edge-collector.js +274 -0
  402. package/vscode-extension/out/edge-collector.js.map +1 -0
  403. package/vscode-extension/out/extension.js +264 -0
  404. package/vscode-extension/out/extension.js.map +1 -0
  405. package/vscode-extension/out/forge-client.js +318 -0
  406. package/vscode-extension/out/forge-client.js.map +1 -0
  407. package/vscode-extension/package-lock.json +59 -0
  408. package/vscode-extension/package.json +71 -0
  409. package/vscode-extension/src/edge-collector.ts +320 -0
  410. package/vscode-extension/src/extension.ts +269 -0
  411. package/vscode-extension/src/forge-client.ts +364 -0
  412. package/vscode-extension/tsconfig.json +19 -0
@@ -0,0 +1,391 @@
1
+ // registration-tools.ts — B16
2
+ // MCP tool handlers for repo registration and initialization.
3
+ // Tools: register, init, list_repos, get_manifest, index_repo, get_index_status
4
+ import { z } from 'zod';
5
+ import { fullIndex } from '../ingestion/indexer.js';
6
+ import { getUptimeSeconds } from '../memory/session-tracker.js';
7
+ import { join, resolve } from 'node:path';
8
+ import { logger } from '../util/logger.js';
9
+ // ---------------------------------------------------------------------------
10
+ // Input schemas
11
+ // ---------------------------------------------------------------------------
12
+ export const registerSchema = z.object({
13
+ path: z.string().optional(),
14
+ auto: z.boolean().default(true),
15
+ });
16
+ export const initSchema = z.object({
17
+ path: z.string().optional(),
18
+ name: z.string().min(1, 'name is required'),
19
+ stack: z.array(z.string()).min(1, 'at least one stack tag is required'),
20
+ sharing: z.enum(['team', 'private', 'public']).default('private'),
21
+ org: z.string().optional(),
22
+ });
23
+ export const listReposSchema = z.object({});
24
+ export const getManifestSchema = z.object({
25
+ repo_path: z.string().optional(),
26
+ });
27
+ export const indexRepoSchema = z.object({
28
+ repo_id: z.string().optional(),
29
+ path: z.string().optional(),
30
+ force: z.boolean().default(false),
31
+ });
32
+ export const getIndexStatusSchema = z.object({
33
+ repo_id: z.string().optional(),
34
+ });
35
+ // ---------------------------------------------------------------------------
36
+ // Tool factory
37
+ // ---------------------------------------------------------------------------
38
+ export function createRegistrationTools(registry, stores) {
39
+ return {
40
+ 'register': {
41
+ schema: registerSchema,
42
+ description: 'Register a repo with the forge server, loading its .forge/manifest.yaml and syncing knowledge to Qdrant',
43
+ handler: async (input) => {
44
+ try {
45
+ const repoPath = resolve(input.path ?? process.cwd());
46
+ const result = await registry.register(repoPath, {
47
+ auto: input.auto,
48
+ });
49
+ logger.info('registration-tools.register: completed', {
50
+ repoId: result.repo_id,
51
+ name: result.name,
52
+ newlyRegistered: result.newly_registered,
53
+ });
54
+ const response = {
55
+ repo_id: result.repo_id,
56
+ name: result.name,
57
+ path: result.path,
58
+ remote: result.remote,
59
+ stack: result.stack,
60
+ sharing: result.sharing,
61
+ newly_registered: result.newly_registered,
62
+ knowledge_items_loaded: result.knowledge_items_loaded,
63
+ };
64
+ // Auto-index after registration when auto: true and stores are available
65
+ if (input.auto && stores?.vectorStore) {
66
+ try {
67
+ const indexResult = await runIndex(registry, stores, result.repo_id, result.path, result.stack, false);
68
+ response.index_result = {
69
+ files_processed: indexResult.filesProcessed,
70
+ files_skipped: indexResult.filesSkipped,
71
+ files_errored: indexResult.filesErrored,
72
+ chunks_created: indexResult.chunksCreated,
73
+ knowledge_items_extracted: indexResult.knowledgeItemsExtracted,
74
+ duration_ms: indexResult.durationMs,
75
+ };
76
+ logger.info('registration-tools.register: auto-index completed', {
77
+ repoId: result.repo_id,
78
+ filesProcessed: indexResult.filesProcessed,
79
+ chunksCreated: indexResult.chunksCreated,
80
+ });
81
+ }
82
+ catch (indexErr) {
83
+ // Non-fatal: registration still succeeds
84
+ response.index_warning = `Auto-index failed: ${String(indexErr)}`;
85
+ logger.warn('registration-tools.register: auto-index failed (non-fatal)', {
86
+ repoId: result.repo_id,
87
+ error: String(indexErr),
88
+ });
89
+ }
90
+ }
91
+ return response;
92
+ }
93
+ catch (err) {
94
+ logger.error('registration-tools.register: error', { error: String(err) });
95
+ return {
96
+ error: 'REGISTER_FAILED',
97
+ message: String(err),
98
+ };
99
+ }
100
+ },
101
+ },
102
+ 'init': {
103
+ schema: initSchema,
104
+ description: 'Initialize the .forge/ directory structure in a repo, creating manifest.yaml and empty knowledge YAML files',
105
+ handler: async (input) => {
106
+ try {
107
+ const repoPath = resolve(input.path ?? process.cwd());
108
+ const manifest = {
109
+ name: input.name,
110
+ stack: input.stack,
111
+ sharing: input.sharing,
112
+ org: input.org,
113
+ repo_id_override: null,
114
+ };
115
+ const { forgePath, filesCreated } = registry.initForge(repoPath, manifest);
116
+ logger.info('registration-tools.init: completed', {
117
+ forgePath,
118
+ filesCreated: filesCreated.length,
119
+ });
120
+ return {
121
+ forge_dir: forgePath,
122
+ manifest_path: join(forgePath, 'manifest.yaml'),
123
+ knowledge_dir: join(forgePath, 'knowledge'),
124
+ files_created: filesCreated,
125
+ };
126
+ }
127
+ catch (err) {
128
+ logger.error('registration-tools.init: error', { error: String(err) });
129
+ return {
130
+ error: 'INIT_FAILED',
131
+ message: String(err),
132
+ };
133
+ }
134
+ },
135
+ },
136
+ 'list_repos': {
137
+ schema: listReposSchema,
138
+ description: 'List all registered repos with their stack tags and last-seen timestamps',
139
+ handler: async (_input) => {
140
+ try {
141
+ const repos = registry.listRepos();
142
+ logger.debug('registration-tools.list_repos: completed', {
143
+ count: repos.length,
144
+ });
145
+ return {
146
+ repos: repos.map(r => ({
147
+ id: r.id,
148
+ name: r.name,
149
+ path: r.path,
150
+ stack: r.stack,
151
+ sharing: r.sharing,
152
+ last_seen: new Date(r.lastSeenAt).toISOString(),
153
+ knowledge_count: r.knowledge_count,
154
+ })),
155
+ };
156
+ }
157
+ catch (err) {
158
+ logger.error('registration-tools.list_repos: error', { error: String(err) });
159
+ return {
160
+ error: 'LIST_REPOS_FAILED',
161
+ message: String(err),
162
+ repos: [],
163
+ };
164
+ }
165
+ },
166
+ },
167
+ 'get_manifest': {
168
+ schema: getManifestSchema,
169
+ description: 'Read and return the .forge/manifest.yaml from a repo path',
170
+ handler: async (input) => {
171
+ try {
172
+ const repoPath = resolve(input.repo_path ?? process.cwd());
173
+ const forgePath = join(repoPath, '.forge');
174
+ const manifest = registry.readManifest(forgePath);
175
+ const forgeExists = registry.forgeDirectoryExists(repoPath);
176
+ return {
177
+ manifest: manifest
178
+ ? {
179
+ name: manifest.name,
180
+ stack: manifest.stack,
181
+ sharing: manifest.sharing,
182
+ org: manifest.org,
183
+ }
184
+ : null,
185
+ forge_dir_exists: forgeExists,
186
+ };
187
+ }
188
+ catch (err) {
189
+ logger.error('registration-tools.get_manifest: error', { error: String(err) });
190
+ return {
191
+ error: 'GET_MANIFEST_FAILED',
192
+ message: String(err),
193
+ manifest: null,
194
+ forge_dir_exists: false,
195
+ };
196
+ }
197
+ },
198
+ },
199
+ 'index_repo': {
200
+ schema: indexRepoSchema,
201
+ description: 'Index a registered repo, populating vector and graph stores with code chunks and extracting knowledge from markdown docs',
202
+ handler: async (input) => {
203
+ if (!stores?.vectorStore) {
204
+ return {
205
+ error: 'INDEX_UNAVAILABLE',
206
+ message: 'Vector store (Qdrant) is not available. Cannot index without it.',
207
+ };
208
+ }
209
+ try {
210
+ // Resolve repo by ID or path
211
+ let repo;
212
+ if (input.repo_id) {
213
+ repo = registry.getRepo(input.repo_id);
214
+ if (!repo) {
215
+ return {
216
+ error: 'REPO_NOT_FOUND',
217
+ message: `Repo not found: ${input.repo_id}. Run 'forge register' first.`,
218
+ };
219
+ }
220
+ }
221
+ else {
222
+ const repoPath = resolve(input.path ?? process.cwd());
223
+ repo = registry.findRepoByPath(repoPath);
224
+ if (!repo) {
225
+ return {
226
+ error: 'REPO_NOT_FOUND',
227
+ message: `No registered repo found at ${repoPath}. Run 'forge register' first.`,
228
+ };
229
+ }
230
+ }
231
+ const result = await runIndex(registry, stores, repo.id, repo.path, repo.stack, input.force);
232
+ logger.info('registration-tools.index_repo: completed', {
233
+ repoId: repo.id,
234
+ filesProcessed: result.filesProcessed,
235
+ chunksCreated: result.chunksCreated,
236
+ knowledgeItemsExtracted: result.knowledgeItemsExtracted,
237
+ });
238
+ return {
239
+ repo_id: result.repoId,
240
+ files_processed: result.filesProcessed,
241
+ files_skipped: result.filesSkipped,
242
+ files_errored: result.filesErrored,
243
+ chunks_created: result.chunksCreated,
244
+ knowledge_items_extracted: result.knowledgeItemsExtracted,
245
+ duration_ms: result.durationMs,
246
+ errors: result.errors.length > 0 ? result.errors.slice(0, 10) : undefined,
247
+ };
248
+ }
249
+ catch (err) {
250
+ logger.error('registration-tools.index_repo: error', { error: String(err) });
251
+ return {
252
+ error: 'INDEX_FAILED',
253
+ message: String(err),
254
+ };
255
+ }
256
+ },
257
+ },
258
+ 'get_index_status': {
259
+ schema: getIndexStatusSchema,
260
+ description: 'Get health and index status for all backends (Qdrant, FalkorDB, file cache). Optionally filter by repo_id for per-repo stats.',
261
+ handler: async (input) => {
262
+ try {
263
+ // --- Repo listing ---
264
+ const allRepos = registry.listRepos();
265
+ const filteredRepos = input.repo_id
266
+ ? allRepos.filter(r => r.id === input.repo_id)
267
+ : allRepos;
268
+ const repoStatuses = filteredRepos.map(r => ({
269
+ repoId: r.id,
270
+ path: r.path,
271
+ filesIndexed: 0, // Approximate — exact count requires scanning
272
+ filesTotal: 0,
273
+ lastIndexedAt: r.lastSeenAt,
274
+ staleFiles: 0,
275
+ }));
276
+ // --- Vector store health ---
277
+ let vectorCounts = { codeChunks: 0, observations: 0 };
278
+ let qdrantHealthy = false;
279
+ if (stores?.vectorStore) {
280
+ try {
281
+ qdrantHealthy = await stores.vectorStore.isHealthy();
282
+ if (qdrantHealthy) {
283
+ vectorCounts = await stores.vectorStore.getCounts();
284
+ }
285
+ }
286
+ catch (err) {
287
+ logger.warn('get_index_status: Qdrant getCounts failed', { error: String(err) });
288
+ }
289
+ }
290
+ // --- Graph store health ---
291
+ let graphCounts = { totalNodes: 0, totalEdges: 0, byLabel: {} };
292
+ let graphHealthy = false;
293
+ if (stores?.graphStore) {
294
+ try {
295
+ graphHealthy = await stores.graphStore.isHealthy();
296
+ if (graphHealthy) {
297
+ graphCounts = await stores.graphStore.getCounts();
298
+ }
299
+ }
300
+ catch (err) {
301
+ logger.warn('get_index_status: FalkorDB getCounts failed', { error: String(err) });
302
+ }
303
+ }
304
+ // --- Cache stats ---
305
+ const cacheStats = stores?.fileCache
306
+ ? stores.fileCache.getStats()
307
+ : { entries: 0, memoryUsageMb: 0, hitRate: 0, evictionCount: 0, oldestEntryAgeSeconds: 0 };
308
+ // --- Overall health ---
309
+ let health;
310
+ if (qdrantHealthy && graphHealthy) {
311
+ health = 'healthy';
312
+ }
313
+ else if (qdrantHealthy || graphHealthy) {
314
+ health = 'degraded';
315
+ }
316
+ else {
317
+ health = 'unavailable';
318
+ }
319
+ const result = {
320
+ repos: repoStatuses,
321
+ graph: {
322
+ total_nodes: graphCounts.totalNodes,
323
+ total_edges: graphCounts.totalEdges,
324
+ node_counts: graphCounts.byLabel,
325
+ },
326
+ vector: {
327
+ code_chunks_count: vectorCounts.codeChunks,
328
+ observations_count: vectorCounts.observations,
329
+ },
330
+ cache: cacheStats,
331
+ health,
332
+ embedding_model: 'Xenova/all-MiniLM-L6-v2',
333
+ uptime_seconds: getUptimeSeconds(),
334
+ };
335
+ logger.debug('registration-tools.get_index_status: completed', {
336
+ repoCount: repoStatuses.length,
337
+ health,
338
+ uptimeSeconds: result.uptime_seconds,
339
+ });
340
+ return result;
341
+ }
342
+ catch (err) {
343
+ logger.error('registration-tools.get_index_status: error', { error: String(err) });
344
+ return {
345
+ error: 'GET_INDEX_STATUS_FAILED',
346
+ message: String(err),
347
+ };
348
+ }
349
+ },
350
+ },
351
+ };
352
+ }
353
+ // ---------------------------------------------------------------------------
354
+ // Shared indexing helper
355
+ // ---------------------------------------------------------------------------
356
+ async function runIndex(registry, stores, repoId, repoPath, stackTags, force) {
357
+ // Build a no-op graph store if FalkorDB is unavailable
358
+ const effectiveGraphStore = stores.graphStore ?? createNoOpGraphStore();
359
+ const repoConfig = {
360
+ id: repoId,
361
+ path: repoPath,
362
+ ownership: 'owned',
363
+ watch: false,
364
+ languages: stackTags,
365
+ };
366
+ return fullIndex({
367
+ repoConfig,
368
+ graphStore: effectiveGraphStore,
369
+ vectorStore: stores.vectorStore,
370
+ fileCache: stores.fileCache,
371
+ allRepos: [repoConfig],
372
+ });
373
+ }
374
+ /**
375
+ * Minimal no-op graph store for when FalkorDB is unavailable.
376
+ */
377
+ function createNoOpGraphStore() {
378
+ return {
379
+ connect: async () => { },
380
+ disconnect: async () => { },
381
+ isHealthy: async () => false,
382
+ query: async () => ({ nodes: [], edges: [], raw: [] }),
383
+ upsertNode: async () => { },
384
+ upsertEdge: async () => { },
385
+ deleteFile: async () => { },
386
+ deleteRepo: async () => { },
387
+ getCounts: async () => ({ totalNodes: 0, totalEdges: 0, byLabel: {} }),
388
+ ensureIndexes: async () => { },
389
+ };
390
+ }
391
+ //# sourceMappingURL=registration-tools.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"registration-tools.js","sourceRoot":"","sources":["../../src/tools/registration-tools.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,75 @@
1
+ /**
2
+ * Circuit breaker pattern for external service calls.
3
+ *
4
+ * States:
5
+ * CLOSED — normal operation, failures are counted
6
+ * OPEN — failing, all calls rejected immediately after failureThreshold reached
7
+ * HALF_OPEN — testing recovery, one probe request allowed
8
+ *
9
+ * Transitions:
10
+ * CLOSED → OPEN after failureThreshold consecutive failures
11
+ * OPEN → HALF_OPEN after resetTimeoutMs has elapsed
12
+ * HALF_OPEN → CLOSED if the probe request succeeds
13
+ * HALF_OPEN → OPEN if the probe request fails
14
+ */
15
+ export type CircuitState = 'CLOSED' | 'OPEN' | 'HALF_OPEN';
16
+ export interface CircuitBreakerOptions {
17
+ /** Number of consecutive failures before tripping to OPEN. Default: 5 */
18
+ failureThreshold?: number;
19
+ /** Milliseconds to wait in OPEN before transitioning to HALF_OPEN. Default: 30000 */
20
+ resetTimeoutMs?: number;
21
+ /** Name for log messages. */
22
+ name: string;
23
+ }
24
+ export declare class CircuitBreaker {
25
+ private state;
26
+ private consecutiveFailures;
27
+ private lastFailureTime;
28
+ private halfOpenProbeInFlight;
29
+ private readonly failureThreshold;
30
+ private readonly resetTimeoutMs;
31
+ private readonly name;
32
+ constructor(opts: CircuitBreakerOptions);
33
+ /**
34
+ * Current state of the circuit breaker.
35
+ */
36
+ getState(): CircuitState;
37
+ /**
38
+ * Returns true when the breaker allows the call to proceed.
39
+ * Returns false when OPEN (or HALF_OPEN but a probe is already in flight).
40
+ */
41
+ isAllowed(): boolean;
42
+ /**
43
+ * Call this when a protected operation succeeds.
44
+ */
45
+ recordSuccess(): void;
46
+ /**
47
+ * Call this when a protected operation fails.
48
+ */
49
+ recordFailure(): void;
50
+ /**
51
+ * Execute a function with circuit breaker protection.
52
+ * Throws CircuitOpenError if the circuit is open.
53
+ * Automatically records success/failure.
54
+ */
55
+ execute<T>(fn: () => Promise<T>): Promise<T>;
56
+ /**
57
+ * Convenience: returns true when the circuit is healthy enough to allow requests.
58
+ * Equivalent to `isAllowed()` but does not reserve the HALF_OPEN probe slot.
59
+ */
60
+ isHealthy(): boolean;
61
+ /**
62
+ * Force-reset the breaker to CLOSED state. Useful for testing or manual recovery.
63
+ */
64
+ reset(): void;
65
+ /** Internal: check if we should transition from OPEN to HALF_OPEN due to timeout. */
66
+ private maybeTransitionToHalfOpen;
67
+ private transitionTo;
68
+ }
69
+ /**
70
+ * Thrown when a call is rejected because the circuit is open.
71
+ */
72
+ export declare class CircuitOpenError extends Error {
73
+ constructor(message: string);
74
+ }
75
+ //# sourceMappingURL=circuit-breaker.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"circuit-breaker.d.ts","sourceRoot":"","sources":["../../src/util/circuit-breaker.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;GAaG;AAIH,MAAM,MAAM,YAAY,GAAG,QAAQ,GAAG,MAAM,GAAG,WAAW,CAAC;AAE3D,MAAM,WAAW,qBAAqB;IACpC,yEAAyE;IACzE,gBAAgB,CAAC,EAAE,MAAM,CAAC;IAC1B,qFAAqF;IACrF,cAAc,CAAC,EAAE,MAAM,CAAC;IACxB,6BAA6B;IAC7B,IAAI,EAAE,MAAM,CAAC;CACd;AAED,qBAAa,cAAc;IACzB,OAAO,CAAC,KAAK,CAA0B;IACvC,OAAO,CAAC,mBAAmB,CAAK;IAChC,OAAO,CAAC,eAAe,CAAK;IAC5B,OAAO,CAAC,qBAAqB,CAAS;IAEtC,OAAO,CAAC,QAAQ,CAAC,gBAAgB,CAAS;IAC1C,OAAO,CAAC,QAAQ,CAAC,cAAc,CAAS;IACxC,OAAO,CAAC,QAAQ,CAAC,IAAI,CAAS;gBAElB,IAAI,EAAE,qBAAqB;IAMvC;;OAEG;IACH,QAAQ,IAAI,YAAY;IAKxB;;;OAGG;IACH,SAAS,IAAI,OAAO;IAkBpB;;OAEG;IACH,aAAa,IAAI,IAAI;IAcrB;;OAEG;IACH,aAAa,IAAI,IAAI;IA0BrB;;;;OAIG;IACG,OAAO,CAAC,CAAC,EAAE,EAAE,EAAE,MAAM,OAAO,CAAC,CAAC,CAAC,GAAG,OAAO,CAAC,CAAC,CAAC;IAiBlD;;;OAGG;IACH,SAAS,IAAI,OAAO;IAKpB;;OAEG;IACH,KAAK,IAAI,IAAI;IAOb,qFAAqF;IACrF,OAAO,CAAC,yBAAyB;IAajC,OAAO,CAAC,YAAY;CAOrB;AAED;;GAEG;AACH,qBAAa,gBAAiB,SAAQ,KAAK;gBAC7B,OAAO,EAAE,MAAM;CAI5B"}
@@ -0,0 +1,159 @@
1
+ /**
2
+ * Circuit breaker pattern for external service calls.
3
+ *
4
+ * States:
5
+ * CLOSED — normal operation, failures are counted
6
+ * OPEN — failing, all calls rejected immediately after failureThreshold reached
7
+ * HALF_OPEN — testing recovery, one probe request allowed
8
+ *
9
+ * Transitions:
10
+ * CLOSED → OPEN after failureThreshold consecutive failures
11
+ * OPEN → HALF_OPEN after resetTimeoutMs has elapsed
12
+ * HALF_OPEN → CLOSED if the probe request succeeds
13
+ * HALF_OPEN → OPEN if the probe request fails
14
+ */
15
+ import { logger } from './logger.js';
16
+ export class CircuitBreaker {
17
+ state = 'CLOSED';
18
+ consecutiveFailures = 0;
19
+ lastFailureTime = 0;
20
+ halfOpenProbeInFlight = false;
21
+ failureThreshold;
22
+ resetTimeoutMs;
23
+ name;
24
+ constructor(opts) {
25
+ this.name = opts.name;
26
+ this.failureThreshold = opts.failureThreshold ?? 5;
27
+ this.resetTimeoutMs = opts.resetTimeoutMs ?? 30_000;
28
+ }
29
+ /**
30
+ * Current state of the circuit breaker.
31
+ */
32
+ getState() {
33
+ this.maybeTransitionToHalfOpen();
34
+ return this.state;
35
+ }
36
+ /**
37
+ * Returns true when the breaker allows the call to proceed.
38
+ * Returns false when OPEN (or HALF_OPEN but a probe is already in flight).
39
+ */
40
+ isAllowed() {
41
+ this.maybeTransitionToHalfOpen();
42
+ if (this.state === 'CLOSED')
43
+ return true;
44
+ if (this.state === 'HALF_OPEN') {
45
+ // Allow exactly one probe through
46
+ if (!this.halfOpenProbeInFlight) {
47
+ this.halfOpenProbeInFlight = true;
48
+ return true;
49
+ }
50
+ return false;
51
+ }
52
+ // OPEN — reject
53
+ return false;
54
+ }
55
+ /**
56
+ * Call this when a protected operation succeeds.
57
+ */
58
+ recordSuccess() {
59
+ // Ensure state is up to date before evaluating
60
+ this.maybeTransitionToHalfOpen();
61
+ if (this.state === 'HALF_OPEN') {
62
+ logger.info(`Circuit breaker [${this.name}] recovered — closing`, {
63
+ previousFailures: this.consecutiveFailures,
64
+ });
65
+ this.transitionTo('CLOSED');
66
+ }
67
+ this.consecutiveFailures = 0;
68
+ this.halfOpenProbeInFlight = false;
69
+ }
70
+ /**
71
+ * Call this when a protected operation fails.
72
+ */
73
+ recordFailure() {
74
+ // Ensure state is up to date before evaluating
75
+ this.maybeTransitionToHalfOpen();
76
+ this.consecutiveFailures++;
77
+ this.lastFailureTime = Date.now();
78
+ this.halfOpenProbeInFlight = false;
79
+ if (this.state === 'HALF_OPEN') {
80
+ logger.warn(`Circuit breaker [${this.name}] probe failed — re-opening`, {
81
+ consecutiveFailures: this.consecutiveFailures,
82
+ });
83
+ this.transitionTo('OPEN');
84
+ return;
85
+ }
86
+ if (this.state === 'CLOSED' && this.consecutiveFailures >= this.failureThreshold) {
87
+ logger.warn(`Circuit breaker [${this.name}] tripped OPEN`, {
88
+ consecutiveFailures: this.consecutiveFailures,
89
+ failureThreshold: this.failureThreshold,
90
+ resetTimeoutMs: this.resetTimeoutMs,
91
+ });
92
+ this.transitionTo('OPEN');
93
+ }
94
+ }
95
+ /**
96
+ * Execute a function with circuit breaker protection.
97
+ * Throws CircuitOpenError if the circuit is open.
98
+ * Automatically records success/failure.
99
+ */
100
+ async execute(fn) {
101
+ if (!this.isAllowed()) {
102
+ throw new CircuitOpenError(`Circuit breaker [${this.name}] is OPEN — request rejected`);
103
+ }
104
+ try {
105
+ const result = await fn();
106
+ this.recordSuccess();
107
+ return result;
108
+ }
109
+ catch (err) {
110
+ this.recordFailure();
111
+ throw err;
112
+ }
113
+ }
114
+ /**
115
+ * Convenience: returns true when the circuit is healthy enough to allow requests.
116
+ * Equivalent to `isAllowed()` but does not reserve the HALF_OPEN probe slot.
117
+ */
118
+ isHealthy() {
119
+ this.maybeTransitionToHalfOpen();
120
+ return this.state === 'CLOSED' || this.state === 'HALF_OPEN';
121
+ }
122
+ /**
123
+ * Force-reset the breaker to CLOSED state. Useful for testing or manual recovery.
124
+ */
125
+ reset() {
126
+ logger.info(`Circuit breaker [${this.name}] manually reset`);
127
+ this.transitionTo('CLOSED');
128
+ this.consecutiveFailures = 0;
129
+ this.halfOpenProbeInFlight = false;
130
+ }
131
+ /** Internal: check if we should transition from OPEN to HALF_OPEN due to timeout. */
132
+ maybeTransitionToHalfOpen() {
133
+ if (this.state === 'OPEN' &&
134
+ Date.now() - this.lastFailureTime >= this.resetTimeoutMs) {
135
+ logger.info(`Circuit breaker [${this.name}] entering HALF_OPEN — probing`, {
136
+ elapsed: Date.now() - this.lastFailureTime,
137
+ resetTimeoutMs: this.resetTimeoutMs,
138
+ });
139
+ this.transitionTo('HALF_OPEN');
140
+ }
141
+ }
142
+ transitionTo(next) {
143
+ this.state = next;
144
+ if (next === 'CLOSED') {
145
+ this.lastFailureTime = 0;
146
+ this.halfOpenProbeInFlight = false;
147
+ }
148
+ }
149
+ }
150
+ /**
151
+ * Thrown when a call is rejected because the circuit is open.
152
+ */
153
+ export class CircuitOpenError extends Error {
154
+ constructor(message) {
155
+ super(message);
156
+ this.name = 'CircuitOpenError';
157
+ }
158
+ }
159
+ //# sourceMappingURL=circuit-breaker.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"circuit-breaker.js","sourceRoot":"","sources":["../../src/util/circuit-breaker.ts"],"names":[],"mappings":"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"}