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,188 @@
1
+ // pipeline-tools.ts — B13
2
+ // MCP tool handlers for project lifecycle management.
3
+ // Tools: create_project, get_state
4
+
5
+ import { z } from 'zod';
6
+ import type { PipelineEngine } from '../pipeline/engine.js';
7
+ import type { RepoRegistry } from '../knowledge/registry.js';
8
+ import type { Phase, PipelineTier } from '../util/types.js';
9
+ import { getAvailableTransitions } from '../pipeline/state-machine.js';
10
+ import { logger } from '../util/logger.js';
11
+ import { resolve } from 'node:path';
12
+
13
+ // ---------------------------------------------------------------------------
14
+ // Input schemas
15
+ // ---------------------------------------------------------------------------
16
+
17
+ export const createProjectSchema = z.object({
18
+ name: z.string().min(1, 'Project name is required'),
19
+ description: z.string().optional(),
20
+ repo_path: z.string().optional(),
21
+ tier: z.enum(['full', 'abbreviated', 'trivial']).default('full'),
22
+ jira_ticket: z.string().optional(),
23
+ jira_epic: z.string().optional(),
24
+ });
25
+
26
+ export const getStateSchema = z.object({
27
+ project_id: z.string().min(1, 'project_id is required'),
28
+ });
29
+
30
+ // ---------------------------------------------------------------------------
31
+ // Tool factory
32
+ // ---------------------------------------------------------------------------
33
+
34
+ export function createPipelineTools(
35
+ engine: PipelineEngine,
36
+ registry: RepoRegistry,
37
+ ) {
38
+ return {
39
+ 'create_project': {
40
+ schema: createProjectSchema,
41
+ description: 'Create a new pipeline project for a task/feature',
42
+ handler: async (input: z.infer<typeof createProjectSchema>) => {
43
+ try {
44
+ // Determine repo path
45
+ const repoPath = input.repo_path
46
+ ? resolve(input.repo_path)
47
+ : resolve(process.cwd());
48
+
49
+ // Find or auto-register the repo
50
+ let repo = registry.findRepoByPath(repoPath);
51
+
52
+ if (!repo) {
53
+ // Auto-register using current directory
54
+ try {
55
+ const result = await registry.register(repoPath);
56
+ repo = registry.getRepo(result.repo_id);
57
+ logger.info('pipeline-tools.create_project: auto-registered repo', {
58
+ repoId: result.repo_id,
59
+ path: repoPath,
60
+ });
61
+ } catch (regErr) {
62
+ // If registration fails (no manifest), create a minimal fallback
63
+ // repo entry so the project can still be created.
64
+ logger.warn('pipeline-tools.create_project: repo auto-register failed', {
65
+ path: repoPath,
66
+ error: String(regErr),
67
+ });
68
+ // Try to find any registered repo as fallback
69
+ const allRepos = registry.listRepos();
70
+ repo = allRepos[0] ?? null;
71
+
72
+ if (!repo) {
73
+ return {
74
+ error: 'NO_REPO_REGISTERED',
75
+ message:
76
+ 'No registered repo found. Run forge.init + forge.register first, ' +
77
+ 'or provide a valid repo_path that has a .forge/manifest.yaml.',
78
+ };
79
+ }
80
+ }
81
+ }
82
+
83
+ if (!repo) {
84
+ return {
85
+ error: 'REPO_NOT_FOUND',
86
+ message: `No repo found at ${repoPath}. Run forge.init to initialize.`,
87
+ };
88
+ }
89
+
90
+ // Build metadata with optional Jira context
91
+ const metadata: Record<string, unknown> = {};
92
+ if (input.jira_ticket) metadata.jira_ticket = input.jira_ticket;
93
+ if (input.jira_epic) metadata.jira_epic = input.jira_epic;
94
+
95
+ // Create the project
96
+ const project = engine.createProject({
97
+ name: input.name,
98
+ description: input.description,
99
+ repoId: repo.id,
100
+ tier: input.tier as PipelineTier,
101
+ metadata: Object.keys(metadata).length > 0 ? metadata : undefined,
102
+ });
103
+
104
+ const availableTransitions = getAvailableTransitions(
105
+ project.currentPhase as Phase,
106
+ project.tier,
107
+ );
108
+
109
+ logger.info('pipeline-tools.create_project: project created', {
110
+ projectId: project.id,
111
+ name: project.name,
112
+ repoId: repo.id,
113
+ });
114
+
115
+ return {
116
+ project_id: project.id,
117
+ repo_id: repo.id,
118
+ repo_name: repo.name,
119
+ current_phase: project.currentPhase,
120
+ tier: project.tier,
121
+ available_transitions: availableTransitions,
122
+ };
123
+ } catch (err) {
124
+ logger.error('pipeline-tools.create_project: error', { error: String(err) });
125
+ return {
126
+ error: 'CREATE_PROJECT_FAILED',
127
+ message: String(err),
128
+ };
129
+ }
130
+ },
131
+ },
132
+
133
+ 'get_state': {
134
+ schema: getStateSchema,
135
+ description: 'Returns the current pipeline state and history for a project',
136
+ handler: async (input: z.infer<typeof getStateSchema>) => {
137
+ try {
138
+ const project = engine.getProject(input.project_id);
139
+ if (!project) {
140
+ return {
141
+ error: 'PROJECT_NOT_FOUND',
142
+ project_id: input.project_id,
143
+ };
144
+ }
145
+
146
+ const cycleCounts = engine.getCycleCounts(input.project_id);
147
+ const activeClaims = engine.getActiveClaims(input.project_id);
148
+ const { history } = engine.getHistory(input.project_id);
149
+
150
+ const availableTransitions = getAvailableTransitions(
151
+ project.currentPhase as Phase,
152
+ project.tier,
153
+ );
154
+
155
+ return {
156
+ project_id: project.id,
157
+ name: project.name,
158
+ current_phase: project.currentPhase,
159
+ status: project.status,
160
+ tier: project.tier,
161
+ metadata: project.metadata,
162
+ history: history.map(h => ({
163
+ phase: h.phase,
164
+ agent: h.agent,
165
+ started_at: h.startedAt,
166
+ completed_at: h.completedAt,
167
+ success: h.success,
168
+ })),
169
+ active_claims: activeClaims.map(c => ({
170
+ module: c.moduleName,
171
+ agent: c.agent,
172
+ status: c.status,
173
+ })),
174
+ cycle_counts: cycleCounts,
175
+ available_transitions: availableTransitions,
176
+ };
177
+ } catch (err) {
178
+ logger.error('pipeline-tools.get_state: error', { error: String(err) });
179
+ return {
180
+ error: 'GET_STATE_FAILED',
181
+ project_id: input.project_id,
182
+ message: String(err),
183
+ };
184
+ }
185
+ },
186
+ },
187
+ };
188
+ }
@@ -0,0 +1,450 @@
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
+
5
+ import { z } from 'zod';
6
+ import type { RepoRegistry } from '../knowledge/registry.js';
7
+ import type { VectorStore, GraphStore, FileContentCache } from '../storage/interfaces.js';
8
+ import { fullIndex, type IndexResult } from '../ingestion/indexer.js';
9
+ import type { RepoConfig, IndexStatusOutput } from '../util/types.js';
10
+ import { getUptimeSeconds } from '../memory/session-tracker.js';
11
+ import { join, resolve } from 'node:path';
12
+ import { logger } from '../util/logger.js';
13
+
14
+ // ---------------------------------------------------------------------------
15
+ // Input schemas
16
+ // ---------------------------------------------------------------------------
17
+
18
+ export const registerSchema = z.object({
19
+ path: z.string().optional(),
20
+ auto: z.boolean().default(true),
21
+ });
22
+
23
+ export const initSchema = z.object({
24
+ path: z.string().optional(),
25
+ name: z.string().min(1, 'name is required'),
26
+ stack: z.array(z.string()).min(1, 'at least one stack tag is required'),
27
+ sharing: z.enum(['team', 'private', 'public']).default('private'),
28
+ org: z.string().optional(),
29
+ });
30
+
31
+ export const listReposSchema = z.object({});
32
+
33
+ export const getManifestSchema = z.object({
34
+ repo_path: z.string().optional(),
35
+ });
36
+
37
+ export const indexRepoSchema = z.object({
38
+ repo_id: z.string().optional(),
39
+ path: z.string().optional(),
40
+ force: z.boolean().default(false),
41
+ });
42
+
43
+ export const getIndexStatusSchema = z.object({
44
+ repo_id: z.string().optional(),
45
+ });
46
+
47
+ // ---------------------------------------------------------------------------
48
+ // Store dependencies for indexing
49
+ // ---------------------------------------------------------------------------
50
+
51
+ export interface RegistrationToolStores {
52
+ vectorStore: VectorStore | null;
53
+ graphStore: GraphStore | null;
54
+ fileCache: FileContentCache;
55
+ }
56
+
57
+ // ---------------------------------------------------------------------------
58
+ // Tool factory
59
+ // ---------------------------------------------------------------------------
60
+
61
+ export function createRegistrationTools(
62
+ registry: RepoRegistry,
63
+ stores?: RegistrationToolStores,
64
+ ) {
65
+ return {
66
+ 'register': {
67
+ schema: registerSchema,
68
+ description: 'Register a repo with the forge server, loading its .forge/manifest.yaml and syncing knowledge to Qdrant',
69
+ handler: async (input: z.infer<typeof registerSchema>) => {
70
+ try {
71
+ const repoPath = resolve(input.path ?? process.cwd());
72
+
73
+ const result = await registry.register(repoPath, {
74
+ auto: input.auto,
75
+ });
76
+
77
+ logger.info('registration-tools.register: completed', {
78
+ repoId: result.repo_id,
79
+ name: result.name,
80
+ newlyRegistered: result.newly_registered,
81
+ });
82
+
83
+ const response: Record<string, unknown> = {
84
+ repo_id: result.repo_id,
85
+ name: result.name,
86
+ path: result.path,
87
+ remote: result.remote,
88
+ stack: result.stack,
89
+ sharing: result.sharing,
90
+ newly_registered: result.newly_registered,
91
+ knowledge_items_loaded: result.knowledge_items_loaded,
92
+ };
93
+
94
+ // Auto-index after registration when auto: true and stores are available
95
+ if (input.auto && stores?.vectorStore) {
96
+ try {
97
+ const indexResult = await runIndex(
98
+ registry, stores, result.repo_id, result.path, result.stack, false,
99
+ );
100
+ response.index_result = {
101
+ files_processed: indexResult.filesProcessed,
102
+ files_skipped: indexResult.filesSkipped,
103
+ files_errored: indexResult.filesErrored,
104
+ chunks_created: indexResult.chunksCreated,
105
+ knowledge_items_extracted: indexResult.knowledgeItemsExtracted,
106
+ duration_ms: indexResult.durationMs,
107
+ };
108
+ logger.info('registration-tools.register: auto-index completed', {
109
+ repoId: result.repo_id,
110
+ filesProcessed: indexResult.filesProcessed,
111
+ chunksCreated: indexResult.chunksCreated,
112
+ });
113
+ } catch (indexErr) {
114
+ // Non-fatal: registration still succeeds
115
+ response.index_warning = `Auto-index failed: ${String(indexErr)}`;
116
+ logger.warn('registration-tools.register: auto-index failed (non-fatal)', {
117
+ repoId: result.repo_id,
118
+ error: String(indexErr),
119
+ });
120
+ }
121
+ }
122
+
123
+ return response;
124
+ } catch (err) {
125
+ logger.error('registration-tools.register: error', { error: String(err) });
126
+ return {
127
+ error: 'REGISTER_FAILED',
128
+ message: String(err),
129
+ };
130
+ }
131
+ },
132
+ },
133
+
134
+ 'init': {
135
+ schema: initSchema,
136
+ description: 'Initialize the .forge/ directory structure in a repo, creating manifest.yaml and empty knowledge YAML files',
137
+ handler: async (input: z.infer<typeof initSchema>) => {
138
+ try {
139
+ const repoPath = resolve(input.path ?? process.cwd());
140
+
141
+ const manifest = {
142
+ name: input.name,
143
+ stack: input.stack,
144
+ sharing: input.sharing as 'team' | 'private' | 'public',
145
+ org: input.org,
146
+ repo_id_override: null,
147
+ };
148
+
149
+ const { forgePath, filesCreated } = registry.initForge(repoPath, manifest);
150
+
151
+ logger.info('registration-tools.init: completed', {
152
+ forgePath,
153
+ filesCreated: filesCreated.length,
154
+ });
155
+
156
+ return {
157
+ forge_dir: forgePath,
158
+ manifest_path: join(forgePath, 'manifest.yaml'),
159
+ knowledge_dir: join(forgePath, 'knowledge'),
160
+ files_created: filesCreated,
161
+ };
162
+ } catch (err) {
163
+ logger.error('registration-tools.init: error', { error: String(err) });
164
+ return {
165
+ error: 'INIT_FAILED',
166
+ message: String(err),
167
+ };
168
+ }
169
+ },
170
+ },
171
+
172
+ 'list_repos': {
173
+ schema: listReposSchema,
174
+ description: 'List all registered repos with their stack tags and last-seen timestamps',
175
+ handler: async (_input: z.infer<typeof listReposSchema>) => {
176
+ try {
177
+ const repos = registry.listRepos();
178
+
179
+ logger.debug('registration-tools.list_repos: completed', {
180
+ count: repos.length,
181
+ });
182
+
183
+ return {
184
+ repos: repos.map(r => ({
185
+ id: r.id,
186
+ name: r.name,
187
+ path: r.path,
188
+ stack: r.stack,
189
+ sharing: r.sharing,
190
+ last_seen: new Date(r.lastSeenAt).toISOString(),
191
+ knowledge_count: r.knowledge_count,
192
+ })),
193
+ };
194
+ } catch (err) {
195
+ logger.error('registration-tools.list_repos: error', { error: String(err) });
196
+ return {
197
+ error: 'LIST_REPOS_FAILED',
198
+ message: String(err),
199
+ repos: [],
200
+ };
201
+ }
202
+ },
203
+ },
204
+
205
+ 'get_manifest': {
206
+ schema: getManifestSchema,
207
+ description: 'Read and return the .forge/manifest.yaml from a repo path',
208
+ handler: async (input: z.infer<typeof getManifestSchema>) => {
209
+ try {
210
+ const repoPath = resolve(input.repo_path ?? process.cwd());
211
+ const forgePath = join(repoPath, '.forge');
212
+
213
+ const manifest = registry.readManifest(forgePath);
214
+ const forgeExists = registry.forgeDirectoryExists(repoPath);
215
+
216
+ return {
217
+ manifest: manifest
218
+ ? {
219
+ name: manifest.name,
220
+ stack: manifest.stack,
221
+ sharing: manifest.sharing,
222
+ org: manifest.org,
223
+ }
224
+ : null,
225
+ forge_dir_exists: forgeExists,
226
+ };
227
+ } catch (err) {
228
+ logger.error('registration-tools.get_manifest: error', { error: String(err) });
229
+ return {
230
+ error: 'GET_MANIFEST_FAILED',
231
+ message: String(err),
232
+ manifest: null,
233
+ forge_dir_exists: false,
234
+ };
235
+ }
236
+ },
237
+ },
238
+
239
+ 'index_repo': {
240
+ schema: indexRepoSchema,
241
+ description: 'Index a registered repo, populating vector and graph stores with code chunks and extracting knowledge from markdown docs',
242
+ handler: async (input: z.infer<typeof indexRepoSchema>) => {
243
+ if (!stores?.vectorStore) {
244
+ return {
245
+ error: 'INDEX_UNAVAILABLE',
246
+ message: 'Vector store (Qdrant) is not available. Cannot index without it.',
247
+ };
248
+ }
249
+
250
+ try {
251
+ // Resolve repo by ID or path
252
+ let repo;
253
+ if (input.repo_id) {
254
+ repo = registry.getRepo(input.repo_id);
255
+ if (!repo) {
256
+ return {
257
+ error: 'REPO_NOT_FOUND',
258
+ message: `Repo not found: ${input.repo_id}. Run 'forge register' first.`,
259
+ };
260
+ }
261
+ } else {
262
+ const repoPath = resolve(input.path ?? process.cwd());
263
+ repo = registry.findRepoByPath(repoPath);
264
+ if (!repo) {
265
+ return {
266
+ error: 'REPO_NOT_FOUND',
267
+ message: `No registered repo found at ${repoPath}. Run 'forge register' first.`,
268
+ };
269
+ }
270
+ }
271
+
272
+ const result = await runIndex(
273
+ registry, stores, repo.id, repo.path, repo.stack, input.force,
274
+ );
275
+
276
+ logger.info('registration-tools.index_repo: completed', {
277
+ repoId: repo.id,
278
+ filesProcessed: result.filesProcessed,
279
+ chunksCreated: result.chunksCreated,
280
+ knowledgeItemsExtracted: result.knowledgeItemsExtracted,
281
+ });
282
+
283
+ return {
284
+ repo_id: result.repoId,
285
+ files_processed: result.filesProcessed,
286
+ files_skipped: result.filesSkipped,
287
+ files_errored: result.filesErrored,
288
+ chunks_created: result.chunksCreated,
289
+ knowledge_items_extracted: result.knowledgeItemsExtracted,
290
+ duration_ms: result.durationMs,
291
+ errors: result.errors.length > 0 ? result.errors.slice(0, 10) : undefined,
292
+ };
293
+ } catch (err) {
294
+ logger.error('registration-tools.index_repo: error', { error: String(err) });
295
+ return {
296
+ error: 'INDEX_FAILED',
297
+ message: String(err),
298
+ };
299
+ }
300
+ },
301
+ },
302
+
303
+ 'get_index_status': {
304
+ schema: getIndexStatusSchema,
305
+ description: 'Get health and index status for all backends (Qdrant, FalkorDB, file cache). Optionally filter by repo_id for per-repo stats.',
306
+ handler: async (input: z.infer<typeof getIndexStatusSchema>) => {
307
+ try {
308
+ // --- Repo listing ---
309
+ const allRepos = registry.listRepos();
310
+ const filteredRepos = input.repo_id
311
+ ? allRepos.filter(r => r.id === input.repo_id)
312
+ : allRepos;
313
+
314
+ const repoStatuses = filteredRepos.map(r => ({
315
+ repoId: r.id,
316
+ path: r.path,
317
+ filesIndexed: 0, // Approximate — exact count requires scanning
318
+ filesTotal: 0,
319
+ lastIndexedAt: r.lastSeenAt,
320
+ staleFiles: 0,
321
+ }));
322
+
323
+ // --- Vector store health ---
324
+ let vectorCounts = { codeChunks: 0, observations: 0 };
325
+ let qdrantHealthy = false;
326
+ if (stores?.vectorStore) {
327
+ try {
328
+ qdrantHealthy = await stores.vectorStore.isHealthy();
329
+ if (qdrantHealthy) {
330
+ vectorCounts = await stores.vectorStore.getCounts();
331
+ }
332
+ } catch (err) {
333
+ logger.warn('get_index_status: Qdrant getCounts failed', { error: String(err) });
334
+ }
335
+ }
336
+
337
+ // --- Graph store health ---
338
+ let graphCounts = { totalNodes: 0, totalEdges: 0, byLabel: {} as Record<string, number> };
339
+ let graphHealthy = false;
340
+ if (stores?.graphStore) {
341
+ try {
342
+ graphHealthy = await stores.graphStore.isHealthy();
343
+ if (graphHealthy) {
344
+ graphCounts = await stores.graphStore.getCounts();
345
+ }
346
+ } catch (err) {
347
+ logger.warn('get_index_status: FalkorDB getCounts failed', { error: String(err) });
348
+ }
349
+ }
350
+
351
+ // --- Cache stats ---
352
+ const cacheStats = stores?.fileCache
353
+ ? stores.fileCache.getStats()
354
+ : { entries: 0, memoryUsageMb: 0, hitRate: 0, evictionCount: 0, oldestEntryAgeSeconds: 0 };
355
+
356
+ // --- Overall health ---
357
+ let health: 'healthy' | 'degraded' | 'unavailable';
358
+ if (qdrantHealthy && graphHealthy) {
359
+ health = 'healthy';
360
+ } else if (qdrantHealthy || graphHealthy) {
361
+ health = 'degraded';
362
+ } else {
363
+ health = 'unavailable';
364
+ }
365
+
366
+ const result: IndexStatusOutput = {
367
+ repos: repoStatuses,
368
+ graph: {
369
+ total_nodes: graphCounts.totalNodes,
370
+ total_edges: graphCounts.totalEdges,
371
+ node_counts: graphCounts.byLabel,
372
+ },
373
+ vector: {
374
+ code_chunks_count: vectorCounts.codeChunks,
375
+ observations_count: vectorCounts.observations,
376
+ },
377
+ cache: cacheStats,
378
+ health,
379
+ embedding_model: 'Xenova/all-MiniLM-L6-v2',
380
+ uptime_seconds: getUptimeSeconds(),
381
+ };
382
+
383
+ logger.debug('registration-tools.get_index_status: completed', {
384
+ repoCount: repoStatuses.length,
385
+ health,
386
+ uptimeSeconds: result.uptime_seconds,
387
+ });
388
+
389
+ return result;
390
+ } catch (err) {
391
+ logger.error('registration-tools.get_index_status: error', { error: String(err) });
392
+ return {
393
+ error: 'GET_INDEX_STATUS_FAILED',
394
+ message: String(err),
395
+ };
396
+ }
397
+ },
398
+ },
399
+ };
400
+ }
401
+
402
+ // ---------------------------------------------------------------------------
403
+ // Shared indexing helper
404
+ // ---------------------------------------------------------------------------
405
+
406
+ async function runIndex(
407
+ registry: RepoRegistry,
408
+ stores: RegistrationToolStores,
409
+ repoId: string,
410
+ repoPath: string,
411
+ stackTags: string[],
412
+ force: boolean,
413
+ ): Promise<IndexResult> {
414
+ // Build a no-op graph store if FalkorDB is unavailable
415
+ const effectiveGraphStore = stores.graphStore ?? createNoOpGraphStore();
416
+
417
+ const repoConfig: RepoConfig = {
418
+ id: repoId,
419
+ path: repoPath,
420
+ ownership: 'owned',
421
+ watch: false,
422
+ languages: stackTags,
423
+ };
424
+
425
+ return fullIndex({
426
+ repoConfig,
427
+ graphStore: effectiveGraphStore,
428
+ vectorStore: stores.vectorStore!,
429
+ fileCache: stores.fileCache,
430
+ allRepos: [repoConfig],
431
+ });
432
+ }
433
+
434
+ /**
435
+ * Minimal no-op graph store for when FalkorDB is unavailable.
436
+ */
437
+ function createNoOpGraphStore(): GraphStore {
438
+ return {
439
+ connect: async () => {},
440
+ disconnect: async () => {},
441
+ isHealthy: async () => false,
442
+ query: async () => ({ nodes: [], edges: [], raw: [] }),
443
+ upsertNode: async () => {},
444
+ upsertEdge: async () => {},
445
+ deleteFile: async () => {},
446
+ deleteRepo: async () => {},
447
+ getCounts: async () => ({ totalNodes: 0, totalEdges: 0, byLabel: {} }),
448
+ ensureIndexes: async () => {},
449
+ };
450
+ }
File without changes