@yasserkhanorg/impact-gate 2.0.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 (587) hide show
  1. package/LICENSE +168 -0
  2. package/README.md +520 -0
  3. package/dist/adapters/cypress.d.ts +10 -0
  4. package/dist/adapters/cypress.d.ts.map +1 -0
  5. package/dist/adapters/cypress.js +86 -0
  6. package/dist/adapters/framework_adapter.d.ts +41 -0
  7. package/dist/adapters/framework_adapter.d.ts.map +1 -0
  8. package/dist/adapters/framework_adapter.js +152 -0
  9. package/dist/adapters/playwright.d.ts +10 -0
  10. package/dist/adapters/playwright.d.ts.map +1 -0
  11. package/dist/adapters/playwright.js +86 -0
  12. package/dist/adapters/pytest.d.ts +10 -0
  13. package/dist/adapters/pytest.d.ts.map +1 -0
  14. package/dist/adapters/pytest.js +96 -0
  15. package/dist/adapters/supertest.d.ts +12 -0
  16. package/dist/adapters/supertest.d.ts.map +1 -0
  17. package/dist/adapters/supertest.js +85 -0
  18. package/dist/agent/api_catalog.d.ts +11 -0
  19. package/dist/agent/api_catalog.d.ts.map +1 -0
  20. package/dist/agent/api_catalog.js +210 -0
  21. package/dist/agent/config.d.ts +193 -0
  22. package/dist/agent/config.d.ts.map +1 -0
  23. package/dist/agent/config.js +875 -0
  24. package/dist/agent/feedback.d.ts +91 -0
  25. package/dist/agent/feedback.d.ts.map +1 -0
  26. package/dist/agent/feedback.js +323 -0
  27. package/dist/agent/git.d.ts +19 -0
  28. package/dist/agent/git.d.ts.map +1 -0
  29. package/dist/agent/git.js +257 -0
  30. package/dist/agent/handoff.d.ts +22 -0
  31. package/dist/agent/handoff.d.ts.map +1 -0
  32. package/dist/agent/handoff.js +180 -0
  33. package/dist/agent/llm_agents_flow.d.ts +15 -0
  34. package/dist/agent/llm_agents_flow.d.ts.map +1 -0
  35. package/dist/agent/llm_agents_flow.js +434 -0
  36. package/dist/agent/native_flow.d.ts +6 -0
  37. package/dist/agent/native_flow.d.ts.map +1 -0
  38. package/dist/agent/native_flow.js +179 -0
  39. package/dist/agent/pipeline.d.ts +7 -0
  40. package/dist/agent/pipeline.d.ts.map +1 -0
  41. package/dist/agent/pipeline.js +260 -0
  42. package/dist/agent/pipeline_types.d.ts +54 -0
  43. package/dist/agent/pipeline_types.d.ts.map +1 -0
  44. package/dist/agent/pipeline_types.js +4 -0
  45. package/dist/agent/pipeline_utils.d.ts +12 -0
  46. package/dist/agent/pipeline_utils.d.ts.map +1 -0
  47. package/dist/agent/pipeline_utils.js +156 -0
  48. package/dist/agent/plan.d.ts +170 -0
  49. package/dist/agent/plan.d.ts.map +1 -0
  50. package/dist/agent/plan.js +86 -0
  51. package/dist/agent/playwright_report.d.ts +8 -0
  52. package/dist/agent/playwright_report.d.ts.map +1 -0
  53. package/dist/agent/playwright_report.js +126 -0
  54. package/dist/agent/process_runner.d.ts +10 -0
  55. package/dist/agent/process_runner.d.ts.map +1 -0
  56. package/dist/agent/process_runner.js +92 -0
  57. package/dist/agent/spec_generator.d.ts +5 -0
  58. package/dist/agent/spec_generator.d.ts.map +1 -0
  59. package/dist/agent/spec_generator.js +253 -0
  60. package/dist/agent/test_path.d.ts +2 -0
  61. package/dist/agent/test_path.d.ts.map +1 -0
  62. package/dist/agent/test_path.js +23 -0
  63. package/dist/agent/traceability_capture.d.ts +18 -0
  64. package/dist/agent/traceability_capture.d.ts.map +1 -0
  65. package/dist/agent/traceability_capture.js +313 -0
  66. package/dist/agent/traceability_ingest.d.ts +21 -0
  67. package/dist/agent/traceability_ingest.d.ts.map +1 -0
  68. package/dist/agent/traceability_ingest.js +237 -0
  69. package/dist/agent/types.d.ts +42 -0
  70. package/dist/agent/types.d.ts.map +1 -0
  71. package/dist/agent/types.js +4 -0
  72. package/dist/agent/utils.d.ts +13 -0
  73. package/dist/agent/utils.d.ts.map +1 -0
  74. package/dist/agent/utils.js +152 -0
  75. package/dist/agent/validation_runner.d.ts +5 -0
  76. package/dist/agent/validation_runner.d.ts.map +1 -0
  77. package/dist/agent/validation_runner.js +77 -0
  78. package/dist/agentic/fix_loop.d.ts +26 -0
  79. package/dist/agentic/fix_loop.d.ts.map +1 -0
  80. package/dist/agentic/fix_loop.js +96 -0
  81. package/dist/agentic/playwright_runner.d.ts +43 -0
  82. package/dist/agentic/playwright_runner.d.ts.map +1 -0
  83. package/dist/agentic/playwright_runner.js +165 -0
  84. package/dist/agentic/runner.d.ts +27 -0
  85. package/dist/agentic/runner.d.ts.map +1 -0
  86. package/dist/agentic/runner.js +210 -0
  87. package/dist/agentic/types.d.ts +62 -0
  88. package/dist/agentic/types.d.ts.map +1 -0
  89. package/dist/agentic/types.js +4 -0
  90. package/dist/agents/coverage-evaluator.d.ts +8 -0
  91. package/dist/agents/coverage-evaluator.d.ts.map +1 -0
  92. package/dist/agents/coverage-evaluator.js +41 -0
  93. package/dist/agents/cross-impact.d.ts +13 -0
  94. package/dist/agents/cross-impact.d.ts.map +1 -0
  95. package/dist/agents/cross-impact.js +140 -0
  96. package/dist/agents/executor.d.ts +8 -0
  97. package/dist/agents/executor.d.ts.map +1 -0
  98. package/dist/agents/executor.js +75 -0
  99. package/dist/agents/explorer.d.ts +12 -0
  100. package/dist/agents/explorer.d.ts.map +1 -0
  101. package/dist/agents/explorer.js +43 -0
  102. package/dist/agents/generator.d.ts +8 -0
  103. package/dist/agents/generator.d.ts.map +1 -0
  104. package/dist/agents/generator.js +77 -0
  105. package/dist/agents/healer.d.ts +8 -0
  106. package/dist/agents/healer.d.ts.map +1 -0
  107. package/dist/agents/healer.js +31 -0
  108. package/dist/agents/impact-analyst.d.ts +8 -0
  109. package/dist/agents/impact-analyst.d.ts.map +1 -0
  110. package/dist/agents/impact-analyst.js +38 -0
  111. package/dist/agents/regression-advisor.d.ts +8 -0
  112. package/dist/agents/regression-advisor.d.ts.map +1 -0
  113. package/dist/agents/regression-advisor.js +116 -0
  114. package/dist/agents/strategist.d.ts +9 -0
  115. package/dist/agents/strategist.d.ts.map +1 -0
  116. package/dist/agents/strategist.js +92 -0
  117. package/dist/agents/test-designer.d.ts +8 -0
  118. package/dist/agents/test-designer.d.ts.map +1 -0
  119. package/dist/agents/test-designer.js +111 -0
  120. package/dist/anthropic_provider.d.ts +65 -0
  121. package/dist/anthropic_provider.d.ts.map +1 -0
  122. package/dist/anthropic_provider.js +334 -0
  123. package/dist/api.d.ts +48 -0
  124. package/dist/api.d.ts.map +1 -0
  125. package/dist/api.js +151 -0
  126. package/dist/base_provider.d.ts +109 -0
  127. package/dist/base_provider.d.ts.map +1 -0
  128. package/dist/base_provider.js +203 -0
  129. package/dist/budget_ledger.d.ts +28 -0
  130. package/dist/budget_ledger.d.ts.map +1 -0
  131. package/dist/budget_ledger.js +62 -0
  132. package/dist/cache/cached_provider.d.ts +49 -0
  133. package/dist/cache/cached_provider.d.ts.map +1 -0
  134. package/dist/cache/cached_provider.js +91 -0
  135. package/dist/cache/response_cache.d.ts +79 -0
  136. package/dist/cache/response_cache.d.ts.map +1 -0
  137. package/dist/cache/response_cache.js +177 -0
  138. package/dist/cli/commands/analyze.d.ts +3 -0
  139. package/dist/cli/commands/analyze.d.ts.map +1 -0
  140. package/dist/cli/commands/analyze.js +77 -0
  141. package/dist/cli/commands/bootstrap.d.ts +3 -0
  142. package/dist/cli/commands/bootstrap.d.ts.map +1 -0
  143. package/dist/cli/commands/bootstrap.js +109 -0
  144. package/dist/cli/commands/cost_report.d.ts +3 -0
  145. package/dist/cli/commands/cost_report.d.ts.map +1 -0
  146. package/dist/cli/commands/cost_report.js +115 -0
  147. package/dist/cli/commands/crew.d.ts +3 -0
  148. package/dist/cli/commands/crew.d.ts.map +1 -0
  149. package/dist/cli/commands/crew.js +255 -0
  150. package/dist/cli/commands/feedback.d.ts +3 -0
  151. package/dist/cli/commands/feedback.d.ts.map +1 -0
  152. package/dist/cli/commands/feedback.js +39 -0
  153. package/dist/cli/commands/finalize.d.ts +3 -0
  154. package/dist/cli/commands/finalize.d.ts.map +1 -0
  155. package/dist/cli/commands/finalize.js +41 -0
  156. package/dist/cli/commands/gate.d.ts +3 -0
  157. package/dist/cli/commands/gate.d.ts.map +1 -0
  158. package/dist/cli/commands/gate.js +89 -0
  159. package/dist/cli/commands/generate.d.ts +4 -0
  160. package/dist/cli/commands/generate.d.ts.map +1 -0
  161. package/dist/cli/commands/generate.js +108 -0
  162. package/dist/cli/commands/heal.d.ts +3 -0
  163. package/dist/cli/commands/heal.d.ts.map +1 -0
  164. package/dist/cli/commands/heal.js +60 -0
  165. package/dist/cli/commands/impact.d.ts +4 -0
  166. package/dist/cli/commands/impact.d.ts.map +1 -0
  167. package/dist/cli/commands/impact.js +33 -0
  168. package/dist/cli/commands/init.d.ts +2 -0
  169. package/dist/cli/commands/init.d.ts.map +1 -0
  170. package/dist/cli/commands/init.js +169 -0
  171. package/dist/cli/commands/llm_health.d.ts +2 -0
  172. package/dist/cli/commands/llm_health.d.ts.map +1 -0
  173. package/dist/cli/commands/llm_health.js +22 -0
  174. package/dist/cli/commands/plan.d.ts +4 -0
  175. package/dist/cli/commands/plan.d.ts.map +1 -0
  176. package/dist/cli/commands/plan.js +120 -0
  177. package/dist/cli/commands/plan_crew.d.ts +17 -0
  178. package/dist/cli/commands/plan_crew.d.ts.map +1 -0
  179. package/dist/cli/commands/plan_crew.js +316 -0
  180. package/dist/cli/commands/traceability.d.ts +4 -0
  181. package/dist/cli/commands/traceability.d.ts.map +1 -0
  182. package/dist/cli/commands/traceability.js +77 -0
  183. package/dist/cli/commands/train.d.ts +3 -0
  184. package/dist/cli/commands/train.d.ts.map +1 -0
  185. package/dist/cli/commands/train.js +391 -0
  186. package/dist/cli/defaults.d.ts +35 -0
  187. package/dist/cli/defaults.d.ts.map +1 -0
  188. package/dist/cli/defaults.js +172 -0
  189. package/dist/cli/errors.d.ts +27 -0
  190. package/dist/cli/errors.d.ts.map +1 -0
  191. package/dist/cli/errors.js +57 -0
  192. package/dist/cli/parse_args.d.ts +6 -0
  193. package/dist/cli/parse_args.d.ts.map +1 -0
  194. package/dist/cli/parse_args.js +257 -0
  195. package/dist/cli/types.d.ts +87 -0
  196. package/dist/cli/types.d.ts.map +1 -0
  197. package/dist/cli/types.js +4 -0
  198. package/dist/cli/usage.d.ts +2 -0
  199. package/dist/cli/usage.d.ts.map +1 -0
  200. package/dist/cli/usage.js +109 -0
  201. package/dist/cli.d.ts +3 -0
  202. package/dist/cli.d.ts.map +1 -0
  203. package/dist/cli.js +194 -0
  204. package/dist/crew/context.d.ts +55 -0
  205. package/dist/crew/context.d.ts.map +1 -0
  206. package/dist/crew/context.js +36 -0
  207. package/dist/crew/orchestrator.d.ts +50 -0
  208. package/dist/crew/orchestrator.d.ts.map +1 -0
  209. package/dist/crew/orchestrator.js +329 -0
  210. package/dist/crew/protocol.d.ts +46 -0
  211. package/dist/crew/protocol.d.ts.map +1 -0
  212. package/dist/crew/protocol.js +4 -0
  213. package/dist/crew/provider.d.ts +17 -0
  214. package/dist/crew/provider.d.ts.map +1 -0
  215. package/dist/crew/provider.js +36 -0
  216. package/dist/crew/sanitize.d.ts +3 -0
  217. package/dist/crew/sanitize.d.ts.map +1 -0
  218. package/dist/crew/sanitize.js +31 -0
  219. package/dist/crew/types.d.ts +52 -0
  220. package/dist/crew/types.d.ts.map +1 -0
  221. package/dist/crew/types.js +4 -0
  222. package/dist/crew/workflows.d.ts +52 -0
  223. package/dist/crew/workflows.d.ts.map +1 -0
  224. package/dist/crew/workflows.js +36 -0
  225. package/dist/custom_provider.d.ts +20 -0
  226. package/dist/custom_provider.d.ts.map +1 -0
  227. package/dist/custom_provider.js +277 -0
  228. package/dist/engine/ai_enrichment.d.ts +44 -0
  229. package/dist/engine/ai_enrichment.d.ts.map +1 -0
  230. package/dist/engine/ai_enrichment.js +267 -0
  231. package/dist/engine/diff_loader.d.ts +11 -0
  232. package/dist/engine/diff_loader.d.ts.map +1 -0
  233. package/dist/engine/diff_loader.js +63 -0
  234. package/dist/engine/impact_engine.d.ts +72 -0
  235. package/dist/engine/impact_engine.d.ts.map +1 -0
  236. package/dist/engine/impact_engine.js +298 -0
  237. package/dist/engine/plan_builder.d.ts +11 -0
  238. package/dist/engine/plan_builder.d.ts.map +1 -0
  239. package/dist/engine/plan_builder.js +599 -0
  240. package/dist/esm/adapters/cypress.js +49 -0
  241. package/dist/esm/adapters/framework_adapter.js +114 -0
  242. package/dist/esm/adapters/playwright.js +49 -0
  243. package/dist/esm/adapters/pytest.js +59 -0
  244. package/dist/esm/adapters/supertest.js +48 -0
  245. package/dist/esm/agent/api_catalog.js +199 -0
  246. package/dist/esm/agent/config.js +872 -0
  247. package/dist/esm/agent/feedback.js +317 -0
  248. package/dist/esm/agent/git.js +252 -0
  249. package/dist/esm/agent/handoff.js +177 -0
  250. package/dist/esm/agent/llm_agents_flow.js +421 -0
  251. package/dist/esm/agent/native_flow.js +175 -0
  252. package/dist/esm/agent/pipeline.js +256 -0
  253. package/dist/esm/agent/pipeline_types.js +3 -0
  254. package/dist/esm/agent/pipeline_utils.js +146 -0
  255. package/dist/esm/agent/plan.js +83 -0
  256. package/dist/esm/agent/playwright_report.js +123 -0
  257. package/dist/esm/agent/process_runner.js +83 -0
  258. package/dist/esm/agent/spec_generator.js +249 -0
  259. package/dist/esm/agent/test_path.js +20 -0
  260. package/dist/esm/agent/traceability_capture.js +310 -0
  261. package/dist/esm/agent/traceability_ingest.js +234 -0
  262. package/dist/esm/agent/types.js +3 -0
  263. package/dist/esm/agent/utils.js +138 -0
  264. package/dist/esm/agent/validation_runner.js +73 -0
  265. package/dist/esm/agentic/fix_loop.js +91 -0
  266. package/dist/esm/agentic/playwright_runner.js +161 -0
  267. package/dist/esm/agentic/runner.js +207 -0
  268. package/dist/esm/agentic/types.js +3 -0
  269. package/dist/esm/agents/coverage-evaluator.js +37 -0
  270. package/dist/esm/agents/cross-impact.js +136 -0
  271. package/dist/esm/agents/executor.js +71 -0
  272. package/dist/esm/agents/explorer.js +39 -0
  273. package/dist/esm/agents/generator.js +73 -0
  274. package/dist/esm/agents/healer.js +27 -0
  275. package/dist/esm/agents/impact-analyst.js +34 -0
  276. package/dist/esm/agents/regression-advisor.js +112 -0
  277. package/dist/esm/agents/strategist.js +88 -0
  278. package/dist/esm/agents/test-designer.js +107 -0
  279. package/dist/esm/anthropic_provider.js +326 -0
  280. package/dist/esm/api.js +143 -0
  281. package/dist/esm/base_provider.js +198 -0
  282. package/dist/esm/budget_ledger.js +58 -0
  283. package/dist/esm/cache/cached_provider.js +85 -0
  284. package/dist/esm/cache/response_cache.js +140 -0
  285. package/dist/esm/cli/commands/analyze.js +74 -0
  286. package/dist/esm/cli/commands/bootstrap.js +106 -0
  287. package/dist/esm/cli/commands/cost_report.js +112 -0
  288. package/dist/esm/cli/commands/crew.js +252 -0
  289. package/dist/esm/cli/commands/feedback.js +36 -0
  290. package/dist/esm/cli/commands/finalize.js +38 -0
  291. package/dist/esm/cli/commands/gate.js +86 -0
  292. package/dist/esm/cli/commands/generate.js +105 -0
  293. package/dist/esm/cli/commands/heal.js +57 -0
  294. package/dist/esm/cli/commands/impact.js +30 -0
  295. package/dist/esm/cli/commands/init.js +133 -0
  296. package/dist/esm/cli/commands/llm_health.js +19 -0
  297. package/dist/esm/cli/commands/plan.js +117 -0
  298. package/dist/esm/cli/commands/plan_crew.js +309 -0
  299. package/dist/esm/cli/commands/traceability.js +73 -0
  300. package/dist/esm/cli/commands/train.js +355 -0
  301. package/dist/esm/cli/defaults.js +165 -0
  302. package/dist/esm/cli/errors.js +52 -0
  303. package/dist/esm/cli/parse_args.js +251 -0
  304. package/dist/esm/cli/types.js +3 -0
  305. package/dist/esm/cli/usage.js +106 -0
  306. package/dist/esm/cli.js +192 -0
  307. package/dist/esm/crew/context.js +32 -0
  308. package/dist/esm/crew/orchestrator.js +325 -0
  309. package/dist/esm/crew/protocol.js +3 -0
  310. package/dist/esm/crew/provider.js +33 -0
  311. package/dist/esm/crew/sanitize.js +27 -0
  312. package/dist/esm/crew/types.js +3 -0
  313. package/dist/esm/crew/workflows.js +33 -0
  314. package/dist/esm/custom_provider.js +273 -0
  315. package/dist/esm/engine/ai_enrichment.js +264 -0
  316. package/dist/esm/engine/diff_loader.js +59 -0
  317. package/dist/esm/engine/impact_engine.js +291 -0
  318. package/dist/esm/engine/plan_builder.js +593 -0
  319. package/dist/esm/index.js +72 -0
  320. package/dist/esm/knowledge/api_surface.js +408 -0
  321. package/dist/esm/knowledge/cluster_utils.js +60 -0
  322. package/dist/esm/knowledge/context_loader.js +85 -0
  323. package/dist/esm/knowledge/failure_history.js +121 -0
  324. package/dist/esm/knowledge/kg_bridge.js +381 -0
  325. package/dist/esm/knowledge/kg_types.js +3 -0
  326. package/dist/esm/knowledge/route_families.js +393 -0
  327. package/dist/esm/knowledge/spec_index.js +122 -0
  328. package/dist/esm/logger.js +115 -0
  329. package/dist/esm/mcp-server.js +621 -0
  330. package/dist/esm/metrics/prometheus.js +149 -0
  331. package/dist/esm/model_router.js +59 -0
  332. package/dist/esm/ollama_provider.js +301 -0
  333. package/dist/esm/openai_provider.js +243 -0
  334. package/dist/esm/package.json +3 -0
  335. package/dist/esm/pipeline/orchestrator.js +228 -0
  336. package/dist/esm/pipeline/spec_verifier.js +75 -0
  337. package/dist/esm/pipeline/stage0_preprocess.js +102 -0
  338. package/dist/esm/pipeline/stage1_impact.js +140 -0
  339. package/dist/esm/pipeline/stage2_coverage.js +153 -0
  340. package/dist/esm/pipeline/stage3_generation.js +284 -0
  341. package/dist/esm/pipeline/stage4_heal.js +288 -0
  342. package/dist/esm/progress.js +112 -0
  343. package/dist/esm/prompts/coverage.js +57 -0
  344. package/dist/esm/prompts/cross-impact.js +53 -0
  345. package/dist/esm/prompts/generation.js +297 -0
  346. package/dist/esm/prompts/generation_profile.js +147 -0
  347. package/dist/esm/prompts/heal.js +91 -0
  348. package/dist/esm/prompts/impact.js +63 -0
  349. package/dist/esm/prompts/json_extract.js +36 -0
  350. package/dist/esm/prompts/strategist.js +61 -0
  351. package/dist/esm/prompts/test-designer.js +92 -0
  352. package/dist/esm/provider_factory.js +366 -0
  353. package/dist/esm/provider_interface.js +23 -0
  354. package/dist/esm/provider_utils.js +96 -0
  355. package/dist/esm/qa-agent/cli.js +205 -0
  356. package/dist/esm/qa-agent/orchestrator.js +120 -0
  357. package/dist/esm/qa-agent/phase1/runner.js +139 -0
  358. package/dist/esm/qa-agent/phase1/scope.js +126 -0
  359. package/dist/esm/qa-agent/phase2/agent_browser.js +95 -0
  360. package/dist/esm/qa-agent/phase2/agent_loop.js +351 -0
  361. package/dist/esm/qa-agent/phase2/exploration_state.js +97 -0
  362. package/dist/esm/qa-agent/phase2/tools.js +386 -0
  363. package/dist/esm/qa-agent/phase2/vision.js +75 -0
  364. package/dist/esm/qa-agent/phase3/feedback.js +34 -0
  365. package/dist/esm/qa-agent/phase3/reporter.js +145 -0
  366. package/dist/esm/qa-agent/phase3/spec_generator.js +62 -0
  367. package/dist/esm/qa-agent/phase3/verdict.js +66 -0
  368. package/dist/esm/qa-agent/safe_env.js +23 -0
  369. package/dist/esm/qa-agent/types.js +3 -0
  370. package/dist/esm/reporters/junit.js +86 -0
  371. package/dist/esm/reporters/reporter.js +3 -0
  372. package/dist/esm/reporters/sarif.js +132 -0
  373. package/dist/esm/resilience/circuit_breaker.js +78 -0
  374. package/dist/esm/resilience/retry.js +56 -0
  375. package/dist/esm/sanitize.js +66 -0
  376. package/dist/esm/training/enricher.js +345 -0
  377. package/dist/esm/training/kg_scanner.js +115 -0
  378. package/dist/esm/training/merger.js +204 -0
  379. package/dist/esm/training/scanner.js +923 -0
  380. package/dist/esm/training/types.js +6 -0
  381. package/dist/esm/training/validator.js +254 -0
  382. package/dist/esm/validation/guardrails.js +101 -0
  383. package/dist/esm/validation/output_schema.js +80 -0
  384. package/dist/esm/version.js +33 -0
  385. package/dist/index.d.ts +99 -0
  386. package/dist/index.d.ts.map +1 -0
  387. package/dist/index.js +169 -0
  388. package/dist/knowledge/api_surface.d.ts +37 -0
  389. package/dist/knowledge/api_surface.d.ts.map +1 -0
  390. package/dist/knowledge/api_surface.js +418 -0
  391. package/dist/knowledge/cluster_utils.d.ts +28 -0
  392. package/dist/knowledge/cluster_utils.d.ts.map +1 -0
  393. package/dist/knowledge/cluster_utils.js +67 -0
  394. package/dist/knowledge/context_loader.d.ts +13 -0
  395. package/dist/knowledge/context_loader.d.ts.map +1 -0
  396. package/dist/knowledge/context_loader.js +90 -0
  397. package/dist/knowledge/failure_history.d.ts +39 -0
  398. package/dist/knowledge/failure_history.d.ts.map +1 -0
  399. package/dist/knowledge/failure_history.js +128 -0
  400. package/dist/knowledge/kg_bridge.d.ts +31 -0
  401. package/dist/knowledge/kg_bridge.d.ts.map +1 -0
  402. package/dist/knowledge/kg_bridge.js +388 -0
  403. package/dist/knowledge/kg_types.d.ts +75 -0
  404. package/dist/knowledge/kg_types.d.ts.map +1 -0
  405. package/dist/knowledge/kg_types.js +4 -0
  406. package/dist/knowledge/route_families.d.ts +98 -0
  407. package/dist/knowledge/route_families.d.ts.map +1 -0
  408. package/dist/knowledge/route_families.js +410 -0
  409. package/dist/knowledge/spec_index.d.ts +18 -0
  410. package/dist/knowledge/spec_index.d.ts.map +1 -0
  411. package/dist/knowledge/spec_index.js +128 -0
  412. package/dist/logger.d.ts +31 -0
  413. package/dist/logger.d.ts.map +1 -0
  414. package/dist/logger.js +119 -0
  415. package/dist/mcp-server.d.ts +68 -0
  416. package/dist/mcp-server.d.ts.map +1 -0
  417. package/dist/mcp-server.js +629 -0
  418. package/dist/metrics/prometheus.d.ts +37 -0
  419. package/dist/metrics/prometheus.d.ts.map +1 -0
  420. package/dist/metrics/prometheus.js +153 -0
  421. package/dist/model_router.d.ts +28 -0
  422. package/dist/model_router.d.ts.map +1 -0
  423. package/dist/model_router.js +63 -0
  424. package/dist/ollama_provider.d.ts +65 -0
  425. package/dist/ollama_provider.d.ts.map +1 -0
  426. package/dist/ollama_provider.js +309 -0
  427. package/dist/openai_provider.d.ts +23 -0
  428. package/dist/openai_provider.d.ts.map +1 -0
  429. package/dist/openai_provider.js +251 -0
  430. package/dist/pipeline/orchestrator.d.ts +33 -0
  431. package/dist/pipeline/orchestrator.d.ts.map +1 -0
  432. package/dist/pipeline/orchestrator.js +231 -0
  433. package/dist/pipeline/spec_verifier.d.ts +20 -0
  434. package/dist/pipeline/spec_verifier.d.ts.map +1 -0
  435. package/dist/pipeline/spec_verifier.js +79 -0
  436. package/dist/pipeline/stage0_preprocess.d.ts +31 -0
  437. package/dist/pipeline/stage0_preprocess.d.ts.map +1 -0
  438. package/dist/pipeline/stage0_preprocess.js +105 -0
  439. package/dist/pipeline/stage1_impact.d.ts +19 -0
  440. package/dist/pipeline/stage1_impact.d.ts.map +1 -0
  441. package/dist/pipeline/stage1_impact.js +143 -0
  442. package/dist/pipeline/stage2_coverage.d.ts +19 -0
  443. package/dist/pipeline/stage2_coverage.d.ts.map +1 -0
  444. package/dist/pipeline/stage2_coverage.js +156 -0
  445. package/dist/pipeline/stage3_generation.d.ts +43 -0
  446. package/dist/pipeline/stage3_generation.d.ts.map +1 -0
  447. package/dist/pipeline/stage3_generation.js +287 -0
  448. package/dist/pipeline/stage4_heal.d.ts +62 -0
  449. package/dist/pipeline/stage4_heal.d.ts.map +1 -0
  450. package/dist/pipeline/stage4_heal.js +294 -0
  451. package/dist/progress.d.ts +22 -0
  452. package/dist/progress.d.ts.map +1 -0
  453. package/dist/progress.js +116 -0
  454. package/dist/prompts/coverage.d.ts +39 -0
  455. package/dist/prompts/coverage.d.ts.map +1 -0
  456. package/dist/prompts/coverage.js +61 -0
  457. package/dist/prompts/cross-impact.d.ts +23 -0
  458. package/dist/prompts/cross-impact.d.ts.map +1 -0
  459. package/dist/prompts/cross-impact.js +57 -0
  460. package/dist/prompts/generation.d.ts +25 -0
  461. package/dist/prompts/generation.d.ts.map +1 -0
  462. package/dist/prompts/generation.js +302 -0
  463. package/dist/prompts/generation_profile.d.ts +29 -0
  464. package/dist/prompts/generation_profile.d.ts.map +1 -0
  465. package/dist/prompts/generation_profile.js +151 -0
  466. package/dist/prompts/heal.d.ts +23 -0
  467. package/dist/prompts/heal.d.ts.map +1 -0
  468. package/dist/prompts/heal.js +95 -0
  469. package/dist/prompts/impact.d.ts +31 -0
  470. package/dist/prompts/impact.d.ts.map +1 -0
  471. package/dist/prompts/impact.js +67 -0
  472. package/dist/prompts/json_extract.d.ts +14 -0
  473. package/dist/prompts/json_extract.d.ts.map +1 -0
  474. package/dist/prompts/json_extract.js +39 -0
  475. package/dist/prompts/strategist.d.ts +25 -0
  476. package/dist/prompts/strategist.d.ts.map +1 -0
  477. package/dist/prompts/strategist.js +65 -0
  478. package/dist/prompts/test-designer.d.ts +35 -0
  479. package/dist/prompts/test-designer.d.ts.map +1 -0
  480. package/dist/prompts/test-designer.js +96 -0
  481. package/dist/provider_factory.d.ts +104 -0
  482. package/dist/provider_factory.d.ts.map +1 -0
  483. package/dist/provider_factory.js +371 -0
  484. package/dist/provider_interface.d.ts +365 -0
  485. package/dist/provider_interface.d.ts.map +1 -0
  486. package/dist/provider_interface.js +28 -0
  487. package/dist/provider_utils.d.ts +39 -0
  488. package/dist/provider_utils.d.ts.map +1 -0
  489. package/dist/provider_utils.js +103 -0
  490. package/dist/qa-agent/cli.d.ts +3 -0
  491. package/dist/qa-agent/cli.d.ts.map +1 -0
  492. package/dist/qa-agent/cli.js +207 -0
  493. package/dist/qa-agent/orchestrator.d.ts +3 -0
  494. package/dist/qa-agent/orchestrator.d.ts.map +1 -0
  495. package/dist/qa-agent/orchestrator.js +123 -0
  496. package/dist/qa-agent/phase1/runner.d.ts +3 -0
  497. package/dist/qa-agent/phase1/runner.d.ts.map +1 -0
  498. package/dist/qa-agent/phase1/runner.js +142 -0
  499. package/dist/qa-agent/phase1/scope.d.ts +6 -0
  500. package/dist/qa-agent/phase1/scope.d.ts.map +1 -0
  501. package/dist/qa-agent/phase1/scope.js +129 -0
  502. package/dist/qa-agent/phase2/agent_browser.d.ts +35 -0
  503. package/dist/qa-agent/phase2/agent_browser.d.ts.map +1 -0
  504. package/dist/qa-agent/phase2/agent_browser.js +99 -0
  505. package/dist/qa-agent/phase2/agent_loop.d.ts +3 -0
  506. package/dist/qa-agent/phase2/agent_loop.d.ts.map +1 -0
  507. package/dist/qa-agent/phase2/agent_loop.js +357 -0
  508. package/dist/qa-agent/phase2/exploration_state.d.ts +12 -0
  509. package/dist/qa-agent/phase2/exploration_state.d.ts.map +1 -0
  510. package/dist/qa-agent/phase2/exploration_state.js +109 -0
  511. package/dist/qa-agent/phase2/tools.d.ts +28 -0
  512. package/dist/qa-agent/phase2/tools.d.ts.map +1 -0
  513. package/dist/qa-agent/phase2/tools.js +390 -0
  514. package/dist/qa-agent/phase2/vision.d.ts +3 -0
  515. package/dist/qa-agent/phase2/vision.d.ts.map +1 -0
  516. package/dist/qa-agent/phase2/vision.js +78 -0
  517. package/dist/qa-agent/phase3/feedback.d.ts +3 -0
  518. package/dist/qa-agent/phase3/feedback.d.ts.map +1 -0
  519. package/dist/qa-agent/phase3/feedback.js +37 -0
  520. package/dist/qa-agent/phase3/reporter.d.ts +3 -0
  521. package/dist/qa-agent/phase3/reporter.d.ts.map +1 -0
  522. package/dist/qa-agent/phase3/reporter.js +148 -0
  523. package/dist/qa-agent/phase3/spec_generator.d.ts +3 -0
  524. package/dist/qa-agent/phase3/spec_generator.d.ts.map +1 -0
  525. package/dist/qa-agent/phase3/spec_generator.js +65 -0
  526. package/dist/qa-agent/phase3/verdict.d.ts +3 -0
  527. package/dist/qa-agent/phase3/verdict.d.ts.map +1 -0
  528. package/dist/qa-agent/phase3/verdict.js +69 -0
  529. package/dist/qa-agent/safe_env.d.ts +3 -0
  530. package/dist/qa-agent/safe_env.d.ts.map +1 -0
  531. package/dist/qa-agent/safe_env.js +26 -0
  532. package/dist/qa-agent/types.d.ts +130 -0
  533. package/dist/qa-agent/types.d.ts.map +1 -0
  534. package/dist/qa-agent/types.js +4 -0
  535. package/dist/reporters/junit.d.ts +6 -0
  536. package/dist/reporters/junit.d.ts.map +1 -0
  537. package/dist/reporters/junit.js +89 -0
  538. package/dist/reporters/reporter.d.ts +42 -0
  539. package/dist/reporters/reporter.d.ts.map +1 -0
  540. package/dist/reporters/reporter.js +4 -0
  541. package/dist/reporters/sarif.d.ts +7 -0
  542. package/dist/reporters/sarif.d.ts.map +1 -0
  543. package/dist/reporters/sarif.js +135 -0
  544. package/dist/resilience/circuit_breaker.d.ts +36 -0
  545. package/dist/resilience/circuit_breaker.d.ts.map +1 -0
  546. package/dist/resilience/circuit_breaker.js +82 -0
  547. package/dist/resilience/retry.d.ts +11 -0
  548. package/dist/resilience/retry.d.ts.map +1 -0
  549. package/dist/resilience/retry.js +59 -0
  550. package/dist/sanitize.d.ts +15 -0
  551. package/dist/sanitize.d.ts.map +1 -0
  552. package/dist/sanitize.js +71 -0
  553. package/dist/training/enricher.d.ts +17 -0
  554. package/dist/training/enricher.d.ts.map +1 -0
  555. package/dist/training/enricher.js +350 -0
  556. package/dist/training/kg_scanner.d.ts +13 -0
  557. package/dist/training/kg_scanner.d.ts.map +1 -0
  558. package/dist/training/kg_scanner.js +118 -0
  559. package/dist/training/merger.d.ts +15 -0
  560. package/dist/training/merger.d.ts.map +1 -0
  561. package/dist/training/merger.js +208 -0
  562. package/dist/training/scanner.d.ts +36 -0
  563. package/dist/training/scanner.d.ts.map +1 -0
  564. package/dist/training/scanner.js +932 -0
  565. package/dist/training/types.d.ts +117 -0
  566. package/dist/training/types.d.ts.map +1 -0
  567. package/dist/training/types.js +9 -0
  568. package/dist/training/validator.d.ts +21 -0
  569. package/dist/training/validator.d.ts.map +1 -0
  570. package/dist/training/validator.js +262 -0
  571. package/dist/validation/guardrails.d.ts +31 -0
  572. package/dist/validation/guardrails.d.ts.map +1 -0
  573. package/dist/validation/guardrails.js +112 -0
  574. package/dist/validation/output_schema.d.ts +67 -0
  575. package/dist/validation/output_schema.d.ts.map +1 -0
  576. package/dist/validation/output_schema.js +84 -0
  577. package/dist/version.d.ts +6 -0
  578. package/dist/version.d.ts.map +1 -0
  579. package/dist/version.js +36 -0
  580. package/package.json +126 -0
  581. package/schemas/flow-decision.schema.json +83 -0
  582. package/schemas/gap.schema.json +18 -0
  583. package/schemas/impact.schema.json +455 -0
  584. package/schemas/plan.schema.json +491 -0
  585. package/schemas/route-families.schema.json +137 -0
  586. package/schemas/subsystem-risk-map.schema.json +62 -0
  587. package/schemas/traceability-input.schema.json +122 -0
@@ -0,0 +1,872 @@
1
+ // Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
2
+ // See LICENSE.txt for license information.
3
+ import { existsSync, readFileSync } from 'fs';
4
+ import { dirname, resolve } from 'path';
5
+ const DEFAULT_CONFIG = {
6
+ path: '.',
7
+ profile: 'default',
8
+ testsRoot: undefined,
9
+ flowCatalogPath: undefined,
10
+ mode: 'impact',
11
+ framework: 'auto',
12
+ timeLimitMinutes: 10,
13
+ budget: {
14
+ maxUSD: 2,
15
+ maxTokens: 20000,
16
+ },
17
+ artifacts: {
18
+ mode: 'commit',
19
+ specsDir: '.e2e-ai-agents/reports',
20
+ },
21
+ selectors: {
22
+ patchOnApply: true,
23
+ },
24
+ testDiscovery: {
25
+ patterns: [],
26
+ },
27
+ flowDiscovery: {
28
+ patterns: [],
29
+ exclude: [],
30
+ },
31
+ catalogScoring: {
32
+ priorityScores: {
33
+ P0: 10,
34
+ P1: 6,
35
+ P2: 3,
36
+ },
37
+ fileMatchWeight: 1,
38
+ },
39
+ impact: {
40
+ allowFallback: false,
41
+ dependencyGraph: {
42
+ enabled: true,
43
+ maxDepth: 3,
44
+ maxExpandedFiles: 1000,
45
+ filePatterns: ['**/*.{ts,tsx,js,jsx}'],
46
+ excludePatterns: [
47
+ '**/node_modules/**',
48
+ '**/.git/**',
49
+ '**/dist/**',
50
+ '**/build/**',
51
+ '**/coverage/**',
52
+ '**/__tests__/**',
53
+ '**/tests/**',
54
+ '**/*.spec.*',
55
+ '**/*.test.*',
56
+ ],
57
+ aliasRoots: ['src'],
58
+ pathAliases: {},
59
+ },
60
+ traceability: {
61
+ enabled: true,
62
+ manifestPath: '.e2e-ai-agents/traceability.json',
63
+ minSignalsPerTest: 1,
64
+ },
65
+ subsystemRisk: {
66
+ enabled: false,
67
+ mapPath: '.e2e-ai-agents/subsystem-risk-map.json',
68
+ maxRulesPerFile: 4,
69
+ },
70
+ aiFlow: {
71
+ enabled: false,
72
+ strict: false,
73
+ provider: 'auto',
74
+ contextFiles: [
75
+ 'CLAUDE.OPTIONAL.md',
76
+ '.claude/CLAUDE.OPTIONAL.md',
77
+ ],
78
+ maxFilesPerRequest: 220,
79
+ maxFlowsPerRequest: 80,
80
+ maxTokens: 4000,
81
+ temperature: 0,
82
+ },
83
+ aiMapping: {
84
+ enabled: false,
85
+ provider: 'auto',
86
+ contextFiles: [
87
+ 'CLAUDE.OPTIONAL.md',
88
+ '.claude/CLAUDE.OPTIONAL.md',
89
+ ],
90
+ maxFlowsPerRequest: 30,
91
+ maxCandidateTests: 400,
92
+ maxTokens: 4000,
93
+ temperature: 0,
94
+ },
95
+ },
96
+ pipeline: {
97
+ enabled: false,
98
+ scenarios: 3,
99
+ outputDir: 'specs/functional/ai-assisted',
100
+ heal: true,
101
+ project: 'chrome',
102
+ mcp: false,
103
+ mcpAllowFallback: false,
104
+ mcpOnly: false,
105
+ mcpCommandTimeoutMs: 180000,
106
+ mcpRetries: 1,
107
+ },
108
+ llm: {
109
+ provider: 'auto',
110
+ fallback: 'ollama',
111
+ },
112
+ risk: {
113
+ p0Threshold: 7,
114
+ p1Threshold: 4,
115
+ criticalKeywords: [
116
+ 'auth',
117
+ 'login',
118
+ 'logout',
119
+ 'signup',
120
+ 'register',
121
+ 'onboarding',
122
+ 'checkout',
123
+ 'payment',
124
+ 'billing',
125
+ 'subscription',
126
+ 'admin',
127
+ 'permissions',
128
+ 'settings',
129
+ 'profile',
130
+ 'search',
131
+ 'dashboard',
132
+ 'message',
133
+ 'notifications',
134
+ ],
135
+ },
136
+ policy: {
137
+ minConfidenceForTargeted: 60,
138
+ safeMergeMinConfidence: 85,
139
+ forceFullOnWarningsAtOrAbove: 2,
140
+ forceFullOnP0WithGaps: true,
141
+ forceFullOnRiskyFiles: true,
142
+ riskyFilePatterns: [
143
+ '**/auth/**',
144
+ '**/login/**',
145
+ '**/permissions/**',
146
+ '**/*permission*',
147
+ '**/admin/**',
148
+ '**/security/**',
149
+ '**/migrations/**',
150
+ '**/*migration*',
151
+ '**/schema/**',
152
+ '**/*.sql',
153
+ '**/webhook/**',
154
+ ],
155
+ enforcementMode: 'advisory',
156
+ blockOnActions: ['must-add-tests'],
157
+ },
158
+ flags: {
159
+ defaultState: 'on',
160
+ },
161
+ audience: {
162
+ defaultRoles: ['member'],
163
+ },
164
+ blastRadius: {
165
+ memberBonus: 1,
166
+ guestBonus: 1,
167
+ adminOnlyPenalty: -1,
168
+ flagOffPenalty: -2,
169
+ },
170
+ git: {
171
+ since: 'HEAD~1',
172
+ includeUncommitted: true,
173
+ },
174
+ routeFamilies: {
175
+ strict: false,
176
+ },
177
+ apiSurface: {
178
+ enabled: false,
179
+ },
180
+ };
181
+ function normalizeMattermostProvider(rawProvider, fallback, strict = false) {
182
+ if (typeof rawProvider !== 'string') {
183
+ return fallback;
184
+ }
185
+ const normalized = rawProvider.trim().toLowerCase();
186
+ if (normalized === 'anthropic'
187
+ || normalized === 'openai'
188
+ || normalized === 'ollama'
189
+ || normalized === 'auto') {
190
+ return normalized;
191
+ }
192
+ if (strict) {
193
+ throw new Error(`Invalid provider "${rawProvider}". Allowed values for Mattermost flows: ollama, anthropic, openai, auto.`);
194
+ }
195
+ return fallback;
196
+ }
197
+ function resolveMattermostProvider(config) {
198
+ const envProvider = process.env.LLM_PROVIDER?.trim().toLowerCase();
199
+ const envResolved = envProvider ? normalizeMattermostProvider(envProvider, 'auto', true) : 'auto';
200
+ const llmProvider = normalizeMattermostProvider(config.llm.provider, 'auto', true);
201
+ return envResolved !== 'auto' ? envResolved : llmProvider;
202
+ }
203
+ function safeReadJson(path) {
204
+ try {
205
+ if (!existsSync(path)) {
206
+ return undefined;
207
+ }
208
+ const raw = readFileSync(path, 'utf-8');
209
+ return JSON.parse(raw);
210
+ }
211
+ catch {
212
+ return undefined;
213
+ }
214
+ }
215
+ function mergeConfig(base, patch) {
216
+ return {
217
+ ...base,
218
+ ...patch,
219
+ budget: {
220
+ ...base.budget,
221
+ ...(patch.budget || {}),
222
+ },
223
+ artifacts: {
224
+ ...base.artifacts,
225
+ ...(patch.artifacts || {}),
226
+ },
227
+ selectors: {
228
+ ...base.selectors,
229
+ ...(patch.selectors || {}),
230
+ },
231
+ testDiscovery: {
232
+ ...base.testDiscovery,
233
+ ...(patch.testDiscovery || {}),
234
+ },
235
+ flowDiscovery: {
236
+ ...base.flowDiscovery,
237
+ ...(patch.flowDiscovery || {}),
238
+ },
239
+ catalogScoring: {
240
+ ...base.catalogScoring,
241
+ ...(patch.catalogScoring || {}),
242
+ },
243
+ impact: {
244
+ ...base.impact,
245
+ ...(patch.impact || {}),
246
+ dependencyGraph: {
247
+ ...base.impact.dependencyGraph,
248
+ ...(patch.impact?.dependencyGraph || {}),
249
+ pathAliases: {
250
+ ...base.impact.dependencyGraph.pathAliases,
251
+ ...(patch.impact?.dependencyGraph?.pathAliases || {}),
252
+ },
253
+ },
254
+ traceability: {
255
+ ...base.impact.traceability,
256
+ ...(patch.impact?.traceability || {}),
257
+ },
258
+ subsystemRisk: {
259
+ ...base.impact.subsystemRisk,
260
+ ...(patch.impact?.subsystemRisk || {}),
261
+ },
262
+ aiFlow: {
263
+ ...base.impact.aiFlow,
264
+ ...(patch.impact?.aiFlow || {}),
265
+ },
266
+ aiMapping: {
267
+ ...base.impact.aiMapping,
268
+ ...(patch.impact?.aiMapping || {}),
269
+ },
270
+ },
271
+ pipeline: {
272
+ ...base.pipeline,
273
+ ...(patch.pipeline || {}),
274
+ },
275
+ llm: {
276
+ ...base.llm,
277
+ ...(patch.llm || {}),
278
+ },
279
+ risk: {
280
+ ...base.risk,
281
+ ...(patch.risk || {}),
282
+ },
283
+ policy: {
284
+ ...base.policy,
285
+ ...(patch.policy || {}),
286
+ },
287
+ flags: {
288
+ ...base.flags,
289
+ ...(patch.flags || {}),
290
+ },
291
+ audience: {
292
+ ...base.audience,
293
+ ...(patch.audience || {}),
294
+ },
295
+ blastRadius: {
296
+ ...base.blastRadius,
297
+ ...(patch.blastRadius || {}),
298
+ },
299
+ git: {
300
+ ...base.git,
301
+ ...(patch.git || {}),
302
+ },
303
+ routeFamilies: {
304
+ ...base.routeFamilies,
305
+ ...(patch.routeFamilies || {}),
306
+ },
307
+ apiSurface: {
308
+ ...base.apiSurface,
309
+ ...(patch.apiSurface || {}),
310
+ },
311
+ };
312
+ }
313
+ function coerceNumber(value) {
314
+ if (typeof value === 'number' && Number.isFinite(value)) {
315
+ return value;
316
+ }
317
+ if (typeof value === 'string') {
318
+ const parsed = Number(value);
319
+ if (Number.isFinite(parsed)) {
320
+ return parsed;
321
+ }
322
+ }
323
+ return undefined;
324
+ }
325
+ function normalizeMode(value) {
326
+ if (value === 'impact' || value === 'gap') {
327
+ return value;
328
+ }
329
+ return undefined;
330
+ }
331
+ function normalizeProfile(value) {
332
+ if (value === 'default' || value === 'mattermost') {
333
+ return value;
334
+ }
335
+ return undefined;
336
+ }
337
+ function normalizeFramework(value) {
338
+ if (value === 'auto' ||
339
+ value === 'playwright' ||
340
+ value === 'cypress' ||
341
+ value === 'pytest' ||
342
+ value === 'supertest' ||
343
+ value === 'selenium') {
344
+ return value;
345
+ }
346
+ return undefined;
347
+ }
348
+ function normalizeFlagState(value) {
349
+ if (typeof value !== 'string') {
350
+ return undefined;
351
+ }
352
+ const normalized = value.trim().toLowerCase();
353
+ if (normalized === 'on' || normalized === 'off' || normalized === 'unknown') {
354
+ return normalized;
355
+ }
356
+ return undefined;
357
+ }
358
+ function normalizePolicyEnforcementMode(value) {
359
+ if (value === 'advisory' || value === 'warn' || value === 'block') {
360
+ return value;
361
+ }
362
+ return undefined;
363
+ }
364
+ function normalizePolicyBlockAction(value) {
365
+ if (value === 'run-now' || value === 'must-add-tests' || value === 'safe-to-merge') {
366
+ return value;
367
+ }
368
+ return undefined;
369
+ }
370
+ function extractConfigPatch(raw) {
371
+ const patch = {};
372
+ if (typeof raw.path === 'string') {
373
+ patch.path = raw.path;
374
+ }
375
+ const profile = normalizeProfile(raw.profile);
376
+ if (profile) {
377
+ patch.profile = profile;
378
+ }
379
+ if (typeof raw.testsRoot === 'string') {
380
+ patch.testsRoot = raw.testsRoot;
381
+ }
382
+ if (typeof raw.flowCatalogPath === 'string') {
383
+ patch.flowCatalogPath = raw.flowCatalogPath;
384
+ }
385
+ const mode = normalizeMode(raw.mode);
386
+ if (mode) {
387
+ patch.mode = mode;
388
+ }
389
+ const framework = normalizeFramework(raw.framework);
390
+ if (framework) {
391
+ patch.framework = framework;
392
+ }
393
+ const timeLimitMinutes = coerceNumber(raw.timeLimitMinutes);
394
+ if (timeLimitMinutes !== undefined) {
395
+ patch.timeLimitMinutes = timeLimitMinutes;
396
+ }
397
+ if (raw.budget && typeof raw.budget === 'object') {
398
+ const budget = raw.budget;
399
+ patch.budget = {
400
+ maxUSD: coerceNumber(budget.maxUSD),
401
+ maxTokens: coerceNumber(budget.maxTokens),
402
+ };
403
+ }
404
+ if (raw.artifacts && typeof raw.artifacts === 'object') {
405
+ const artifacts = raw.artifacts;
406
+ const mode = artifacts.mode === 'commit' || artifacts.mode === 'keep-local' || artifacts.mode === 'none'
407
+ ? artifacts.mode
408
+ : undefined;
409
+ const specsDir = typeof artifacts.specsDir === 'string' ? artifacts.specsDir : undefined;
410
+ if (mode || specsDir) {
411
+ patch.artifacts = {
412
+ mode: mode ?? DEFAULT_CONFIG.artifacts.mode,
413
+ specsDir: specsDir ?? DEFAULT_CONFIG.artifacts.specsDir,
414
+ };
415
+ }
416
+ }
417
+ if (raw.selectors && typeof raw.selectors === 'object') {
418
+ const selectors = raw.selectors;
419
+ patch.selectors = {
420
+ patchOnApply: selectors.patchOnApply !== undefined ? Boolean(selectors.patchOnApply) : DEFAULT_CONFIG.selectors.patchOnApply,
421
+ };
422
+ }
423
+ if (raw.testDiscovery && typeof raw.testDiscovery === 'object') {
424
+ const testDiscovery = raw.testDiscovery;
425
+ patch.testDiscovery = {
426
+ patterns: Array.isArray(testDiscovery.patterns)
427
+ ? testDiscovery.patterns.filter((pattern) => typeof pattern === 'string')
428
+ : undefined,
429
+ };
430
+ }
431
+ if (raw.flowDiscovery && typeof raw.flowDiscovery === 'object') {
432
+ const flowDiscovery = raw.flowDiscovery;
433
+ patch.flowDiscovery = {
434
+ patterns: Array.isArray(flowDiscovery.patterns)
435
+ ? flowDiscovery.patterns.filter((pattern) => typeof pattern === 'string')
436
+ : undefined,
437
+ exclude: Array.isArray(flowDiscovery.exclude)
438
+ ? flowDiscovery.exclude.filter((pattern) => typeof pattern === 'string')
439
+ : undefined,
440
+ };
441
+ }
442
+ if (raw.catalogScoring && typeof raw.catalogScoring === 'object') {
443
+ const catalogScoring = raw.catalogScoring;
444
+ const priorityScores = catalogScoring.priorityScores;
445
+ patch.catalogScoring = {
446
+ priorityScores: {
447
+ P0: coerceNumber(priorityScores?.P0) ?? DEFAULT_CONFIG.catalogScoring.priorityScores.P0,
448
+ P1: coerceNumber(priorityScores?.P1) ?? DEFAULT_CONFIG.catalogScoring.priorityScores.P1,
449
+ P2: coerceNumber(priorityScores?.P2) ?? DEFAULT_CONFIG.catalogScoring.priorityScores.P2,
450
+ },
451
+ fileMatchWeight: coerceNumber(catalogScoring.fileMatchWeight) ?? DEFAULT_CONFIG.catalogScoring.fileMatchWeight,
452
+ };
453
+ }
454
+ if (raw.impact && typeof raw.impact === 'object') {
455
+ const impact = raw.impact;
456
+ const dependencyGraphRaw = impact.dependencyGraph && typeof impact.dependencyGraph === 'object'
457
+ ? impact.dependencyGraph
458
+ : undefined;
459
+ const traceabilityRaw = impact.traceability && typeof impact.traceability === 'object'
460
+ ? impact.traceability
461
+ : undefined;
462
+ const subsystemRiskRaw = impact.subsystemRisk && typeof impact.subsystemRisk === 'object'
463
+ ? impact.subsystemRisk
464
+ : undefined;
465
+ const aiMappingRaw = impact.aiMapping && typeof impact.aiMapping === 'object'
466
+ ? impact.aiMapping
467
+ : undefined;
468
+ const aiFlowRaw = impact.aiFlow && typeof impact.aiFlow === 'object'
469
+ ? impact.aiFlow
470
+ : undefined;
471
+ const pathAliasesRaw = dependencyGraphRaw?.pathAliases && typeof dependencyGraphRaw.pathAliases === 'object'
472
+ ? dependencyGraphRaw.pathAliases
473
+ : undefined;
474
+ const parsedPathAliases = {};
475
+ if (pathAliasesRaw) {
476
+ for (const [key, value] of Object.entries(pathAliasesRaw)) {
477
+ if (typeof value === 'string') {
478
+ parsedPathAliases[key] = [value];
479
+ continue;
480
+ }
481
+ if (Array.isArray(value)) {
482
+ const targets = value.filter((item) => typeof item === 'string');
483
+ if (targets.length > 0) {
484
+ parsedPathAliases[key] = targets;
485
+ }
486
+ }
487
+ }
488
+ }
489
+ patch.impact = {
490
+ allowFallback: impact.allowFallback !== undefined ? Boolean(impact.allowFallback) : DEFAULT_CONFIG.impact.allowFallback,
491
+ dependencyGraph: {
492
+ enabled: dependencyGraphRaw?.enabled !== undefined
493
+ ? Boolean(dependencyGraphRaw.enabled)
494
+ : DEFAULT_CONFIG.impact.dependencyGraph.enabled,
495
+ maxDepth: coerceNumber(dependencyGraphRaw?.maxDepth) ?? DEFAULT_CONFIG.impact.dependencyGraph.maxDepth,
496
+ maxExpandedFiles: coerceNumber(dependencyGraphRaw?.maxExpandedFiles) ?? DEFAULT_CONFIG.impact.dependencyGraph.maxExpandedFiles,
497
+ filePatterns: Array.isArray(dependencyGraphRaw?.filePatterns)
498
+ ? dependencyGraphRaw?.filePatterns.filter((pattern) => typeof pattern === 'string')
499
+ : DEFAULT_CONFIG.impact.dependencyGraph.filePatterns,
500
+ excludePatterns: Array.isArray(dependencyGraphRaw?.excludePatterns)
501
+ ? dependencyGraphRaw?.excludePatterns.filter((pattern) => typeof pattern === 'string')
502
+ : DEFAULT_CONFIG.impact.dependencyGraph.excludePatterns,
503
+ aliasRoots: Array.isArray(dependencyGraphRaw?.aliasRoots)
504
+ ? dependencyGraphRaw?.aliasRoots.filter((pattern) => typeof pattern === 'string')
505
+ : DEFAULT_CONFIG.impact.dependencyGraph.aliasRoots,
506
+ pathAliases: pathAliasesRaw
507
+ ? parsedPathAliases
508
+ : DEFAULT_CONFIG.impact.dependencyGraph.pathAliases,
509
+ },
510
+ traceability: {
511
+ enabled: traceabilityRaw?.enabled !== undefined
512
+ ? Boolean(traceabilityRaw.enabled)
513
+ : DEFAULT_CONFIG.impact.traceability.enabled,
514
+ manifestPath: typeof traceabilityRaw?.manifestPath === 'string'
515
+ ? traceabilityRaw.manifestPath
516
+ : DEFAULT_CONFIG.impact.traceability.manifestPath,
517
+ minSignalsPerTest: coerceNumber(traceabilityRaw?.minSignalsPerTest) ?? DEFAULT_CONFIG.impact.traceability.minSignalsPerTest,
518
+ },
519
+ subsystemRisk: {
520
+ enabled: subsystemRiskRaw?.enabled !== undefined
521
+ ? Boolean(subsystemRiskRaw.enabled)
522
+ : DEFAULT_CONFIG.impact.subsystemRisk.enabled,
523
+ mapPath: typeof subsystemRiskRaw?.mapPath === 'string'
524
+ ? subsystemRiskRaw.mapPath
525
+ : DEFAULT_CONFIG.impact.subsystemRisk.mapPath,
526
+ maxRulesPerFile: coerceNumber(subsystemRiskRaw?.maxRulesPerFile) ?? DEFAULT_CONFIG.impact.subsystemRisk.maxRulesPerFile,
527
+ },
528
+ aiFlow: {
529
+ enabled: aiFlowRaw?.enabled !== undefined
530
+ ? Boolean(aiFlowRaw.enabled)
531
+ : DEFAULT_CONFIG.impact.aiFlow.enabled,
532
+ strict: aiFlowRaw?.strict !== undefined
533
+ ? Boolean(aiFlowRaw.strict)
534
+ : DEFAULT_CONFIG.impact.aiFlow.strict,
535
+ provider: aiFlowRaw?.provider === 'anthropic' ||
536
+ aiFlowRaw?.provider === 'openai' ||
537
+ aiFlowRaw?.provider === 'ollama' ||
538
+ aiFlowRaw?.provider === 'auto'
539
+ ? aiFlowRaw.provider
540
+ : DEFAULT_CONFIG.impact.aiFlow.provider,
541
+ contextFiles: Array.isArray(aiFlowRaw?.contextFiles)
542
+ ? aiFlowRaw.contextFiles.filter((entry) => typeof entry === 'string')
543
+ : DEFAULT_CONFIG.impact.aiFlow.contextFiles,
544
+ maxFilesPerRequest: coerceNumber(aiFlowRaw?.maxFilesPerRequest) ?? DEFAULT_CONFIG.impact.aiFlow.maxFilesPerRequest,
545
+ maxFlowsPerRequest: coerceNumber(aiFlowRaw?.maxFlowsPerRequest) ?? DEFAULT_CONFIG.impact.aiFlow.maxFlowsPerRequest,
546
+ maxTokens: coerceNumber(aiFlowRaw?.maxTokens) ?? DEFAULT_CONFIG.impact.aiFlow.maxTokens,
547
+ temperature: coerceNumber(aiFlowRaw?.temperature) ?? DEFAULT_CONFIG.impact.aiFlow.temperature,
548
+ },
549
+ aiMapping: {
550
+ enabled: aiMappingRaw?.enabled !== undefined
551
+ ? Boolean(aiMappingRaw.enabled)
552
+ : DEFAULT_CONFIG.impact.aiMapping.enabled,
553
+ provider: aiMappingRaw?.provider === 'anthropic' ||
554
+ aiMappingRaw?.provider === 'openai' ||
555
+ aiMappingRaw?.provider === 'ollama' ||
556
+ aiMappingRaw?.provider === 'auto'
557
+ ? aiMappingRaw.provider
558
+ : DEFAULT_CONFIG.impact.aiMapping.provider,
559
+ contextFiles: Array.isArray(aiMappingRaw?.contextFiles)
560
+ ? aiMappingRaw.contextFiles.filter((path) => typeof path === 'string')
561
+ : DEFAULT_CONFIG.impact.aiMapping.contextFiles,
562
+ maxFlowsPerRequest: coerceNumber(aiMappingRaw?.maxFlowsPerRequest) ?? DEFAULT_CONFIG.impact.aiMapping.maxFlowsPerRequest,
563
+ maxCandidateTests: coerceNumber(aiMappingRaw?.maxCandidateTests) ?? DEFAULT_CONFIG.impact.aiMapping.maxCandidateTests,
564
+ maxTokens: coerceNumber(aiMappingRaw?.maxTokens) ?? DEFAULT_CONFIG.impact.aiMapping.maxTokens,
565
+ temperature: coerceNumber(aiMappingRaw?.temperature) ?? DEFAULT_CONFIG.impact.aiMapping.temperature,
566
+ },
567
+ };
568
+ }
569
+ if (raw.pipeline && typeof raw.pipeline === 'object') {
570
+ const pipeline = raw.pipeline;
571
+ patch.pipeline = {
572
+ enabled: pipeline.enabled !== undefined ? Boolean(pipeline.enabled) : DEFAULT_CONFIG.pipeline.enabled,
573
+ scenarios: coerceNumber(pipeline.scenarios) ?? DEFAULT_CONFIG.pipeline.scenarios,
574
+ outputDir: typeof pipeline.outputDir === 'string' ? pipeline.outputDir : DEFAULT_CONFIG.pipeline.outputDir,
575
+ heal: pipeline.heal !== undefined ? Boolean(pipeline.heal) : DEFAULT_CONFIG.pipeline.heal,
576
+ baseUrl: typeof pipeline.baseUrl === 'string' ? pipeline.baseUrl : undefined,
577
+ browser: pipeline.browser === 'chrome' ||
578
+ pipeline.browser === 'chromium' ||
579
+ pipeline.browser === 'firefox' ||
580
+ pipeline.browser === 'webkit'
581
+ ? pipeline.browser
582
+ : undefined,
583
+ headless: pipeline.headless !== undefined ? Boolean(pipeline.headless) : undefined,
584
+ project: typeof pipeline.project === 'string' ? pipeline.project : undefined,
585
+ parallel: pipeline.parallel !== undefined ? Boolean(pipeline.parallel) : undefined,
586
+ dryRun: pipeline.dryRun !== undefined ? Boolean(pipeline.dryRun) : undefined,
587
+ mcp: pipeline.mcp !== undefined ? Boolean(pipeline.mcp) : DEFAULT_CONFIG.pipeline.mcp,
588
+ mcpAllowFallback: pipeline.mcpAllowFallback !== undefined
589
+ ? Boolean(pipeline.mcpAllowFallback)
590
+ : DEFAULT_CONFIG.pipeline.mcpAllowFallback,
591
+ mcpOnly: pipeline.mcpOnly !== undefined
592
+ ? Boolean(pipeline.mcpOnly)
593
+ : DEFAULT_CONFIG.pipeline.mcpOnly,
594
+ mcpCommandTimeoutMs: coerceNumber(pipeline.mcpCommandTimeoutMs) ?? DEFAULT_CONFIG.pipeline.mcpCommandTimeoutMs,
595
+ mcpRetries: coerceNumber(pipeline.mcpRetries) ?? DEFAULT_CONFIG.pipeline.mcpRetries,
596
+ };
597
+ }
598
+ if (raw.llm && typeof raw.llm === 'object') {
599
+ const llm = raw.llm;
600
+ patch.llm = {
601
+ provider: typeof llm.provider === 'string' ? llm.provider : undefined,
602
+ fallback: typeof llm.fallback === 'string' ? llm.fallback : undefined,
603
+ };
604
+ }
605
+ if (typeof raw.specPDF === 'string') {
606
+ patch.specPDF = raw.specPDF;
607
+ }
608
+ if (raw.risk && typeof raw.risk === 'object') {
609
+ const risk = raw.risk;
610
+ patch.risk = {
611
+ p0Threshold: coerceNumber(risk.p0Threshold) ?? DEFAULT_CONFIG.risk.p0Threshold,
612
+ p1Threshold: coerceNumber(risk.p1Threshold) ?? DEFAULT_CONFIG.risk.p1Threshold,
613
+ criticalKeywords: Array.isArray(risk.criticalKeywords)
614
+ ? risk.criticalKeywords.filter((keyword) => typeof keyword === 'string')
615
+ : DEFAULT_CONFIG.risk.criticalKeywords,
616
+ };
617
+ }
618
+ if (raw.policy && typeof raw.policy === 'object') {
619
+ const policy = raw.policy;
620
+ const blockOnActions = Array.isArray(policy.blockOnActions)
621
+ ? policy.blockOnActions
622
+ .map((value) => normalizePolicyBlockAction(value))
623
+ .filter((value) => Boolean(value))
624
+ : DEFAULT_CONFIG.policy.blockOnActions;
625
+ patch.policy = {
626
+ minConfidenceForTargeted: coerceNumber(policy.minConfidenceForTargeted) ?? DEFAULT_CONFIG.policy.minConfidenceForTargeted,
627
+ safeMergeMinConfidence: coerceNumber(policy.safeMergeMinConfidence) ?? DEFAULT_CONFIG.policy.safeMergeMinConfidence,
628
+ forceFullOnWarningsAtOrAbove: coerceNumber(policy.forceFullOnWarningsAtOrAbove) ?? DEFAULT_CONFIG.policy.forceFullOnWarningsAtOrAbove,
629
+ forceFullOnP0WithGaps: policy.forceFullOnP0WithGaps !== undefined
630
+ ? Boolean(policy.forceFullOnP0WithGaps)
631
+ : DEFAULT_CONFIG.policy.forceFullOnP0WithGaps,
632
+ forceFullOnRiskyFiles: policy.forceFullOnRiskyFiles !== undefined
633
+ ? Boolean(policy.forceFullOnRiskyFiles)
634
+ : DEFAULT_CONFIG.policy.forceFullOnRiskyFiles,
635
+ riskyFilePatterns: Array.isArray(policy.riskyFilePatterns)
636
+ ? policy.riskyFilePatterns.filter((pattern) => typeof pattern === 'string')
637
+ : DEFAULT_CONFIG.policy.riskyFilePatterns,
638
+ enforcementMode: normalizePolicyEnforcementMode(policy.enforcementMode) ?? DEFAULT_CONFIG.policy.enforcementMode,
639
+ blockOnActions: blockOnActions.length > 0 ? blockOnActions : DEFAULT_CONFIG.policy.blockOnActions,
640
+ };
641
+ }
642
+ if (raw.flags && typeof raw.flags === 'object') {
643
+ const flags = raw.flags;
644
+ patch.flags = {
645
+ defaultState: normalizeFlagState(flags.defaultState) ?? DEFAULT_CONFIG.flags.defaultState,
646
+ };
647
+ }
648
+ if (raw.audience && typeof raw.audience === 'object') {
649
+ const audience = raw.audience;
650
+ const roles = Array.isArray(audience.defaultRoles)
651
+ ? audience.defaultRoles.filter((role) => typeof role === 'string')
652
+ : [];
653
+ patch.audience = {
654
+ defaultRoles: (roles.length > 0 ? roles : DEFAULT_CONFIG.audience.defaultRoles),
655
+ };
656
+ }
657
+ if (raw.blastRadius && typeof raw.blastRadius === 'object') {
658
+ const blastRadius = raw.blastRadius;
659
+ patch.blastRadius = {
660
+ memberBonus: coerceNumber(blastRadius.memberBonus) ?? DEFAULT_CONFIG.blastRadius.memberBonus,
661
+ guestBonus: coerceNumber(blastRadius.guestBonus) ?? DEFAULT_CONFIG.blastRadius.guestBonus,
662
+ adminOnlyPenalty: coerceNumber(blastRadius.adminOnlyPenalty) ?? DEFAULT_CONFIG.blastRadius.adminOnlyPenalty,
663
+ flagOffPenalty: coerceNumber(blastRadius.flagOffPenalty) ?? DEFAULT_CONFIG.blastRadius.flagOffPenalty,
664
+ };
665
+ }
666
+ if (raw.git && typeof raw.git === 'object') {
667
+ const git = raw.git;
668
+ patch.git = {
669
+ since: typeof git.since === 'string' ? git.since : DEFAULT_CONFIG.git.since,
670
+ includeUncommitted: git.includeUncommitted !== undefined ? Boolean(git.includeUncommitted) : DEFAULT_CONFIG.git.includeUncommitted,
671
+ };
672
+ }
673
+ if (raw.routeFamilies && typeof raw.routeFamilies === 'object') {
674
+ const rf = raw.routeFamilies;
675
+ patch.routeFamilies = {
676
+ manifestPath: typeof rf.manifestPath === 'string' ? rf.manifestPath : undefined,
677
+ strict: rf.strict !== undefined ? Boolean(rf.strict) : DEFAULT_CONFIG.routeFamilies.strict,
678
+ };
679
+ }
680
+ if (raw.apiSurface && typeof raw.apiSurface === 'object') {
681
+ const api = raw.apiSurface;
682
+ patch.apiSurface = {
683
+ enabled: api.enabled !== undefined ? Boolean(api.enabled) : DEFAULT_CONFIG.apiSurface.enabled,
684
+ pageObjectsDir: typeof api.pageObjectsDir === 'string' ? api.pageObjectsDir : undefined,
685
+ componentsDir: typeof api.componentsDir === 'string' ? api.componentsDir : undefined,
686
+ cachePath: typeof api.cachePath === 'string' ? api.cachePath : undefined,
687
+ };
688
+ }
689
+ return patch;
690
+ }
691
+ export function resolveConfig(cwd, configPath, overrides) {
692
+ const resolvedConfigPath = configPath ? resolve(cwd, configPath) : undefined;
693
+ const configDir = resolvedConfigPath ? dirname(resolvedConfigPath) : cwd;
694
+ const rawConfig = resolvedConfigPath ? safeReadJson(resolvedConfigPath) : undefined;
695
+ const configPatch = rawConfig ? extractConfigPatch(rawConfig) : {};
696
+ let config = mergeConfig(DEFAULT_CONFIG, configPatch);
697
+ if (overrides?.profile) {
698
+ config.profile = overrides.profile;
699
+ }
700
+ if (overrides?.mode) {
701
+ config.mode = overrides.mode;
702
+ }
703
+ if (overrides?.framework) {
704
+ config.framework = overrides.framework;
705
+ }
706
+ if (overrides?.timeLimitMinutes !== undefined) {
707
+ config.timeLimitMinutes = overrides.timeLimitMinutes;
708
+ }
709
+ if (overrides?.budget) {
710
+ const budgetPatch = {};
711
+ if (overrides.budget.maxUSD !== undefined) {
712
+ budgetPatch.maxUSD = overrides.budget.maxUSD;
713
+ }
714
+ if (overrides.budget.maxTokens !== undefined) {
715
+ budgetPatch.maxTokens = overrides.budget.maxTokens;
716
+ }
717
+ config.budget = { ...config.budget, ...budgetPatch };
718
+ }
719
+ if (overrides?.testPatterns && overrides.testPatterns.length > 0) {
720
+ config.testDiscovery = { patterns: overrides.testPatterns };
721
+ }
722
+ if (overrides?.flowPatterns && overrides.flowPatterns.length > 0) {
723
+ config.flowDiscovery = {
724
+ patterns: overrides.flowPatterns,
725
+ exclude: overrides.flowExclude,
726
+ };
727
+ }
728
+ else if (overrides?.flowExclude && overrides.flowExclude.length > 0) {
729
+ config.flowDiscovery = {
730
+ ...config.flowDiscovery,
731
+ exclude: overrides.flowExclude,
732
+ };
733
+ }
734
+ if (overrides?.pipeline) {
735
+ const pipelinePatch = {};
736
+ if (overrides.pipeline.enabled !== undefined) {
737
+ pipelinePatch.enabled = overrides.pipeline.enabled;
738
+ }
739
+ if (overrides.pipeline.scenarios !== undefined) {
740
+ pipelinePatch.scenarios = overrides.pipeline.scenarios;
741
+ }
742
+ if (overrides.pipeline.outputDir) {
743
+ pipelinePatch.outputDir = overrides.pipeline.outputDir;
744
+ }
745
+ if (overrides.pipeline.heal !== undefined) {
746
+ pipelinePatch.heal = overrides.pipeline.heal;
747
+ }
748
+ if (overrides.pipeline.baseUrl) {
749
+ pipelinePatch.baseUrl = overrides.pipeline.baseUrl;
750
+ }
751
+ if (overrides.pipeline.browser) {
752
+ pipelinePatch.browser = overrides.pipeline.browser;
753
+ }
754
+ if (overrides.pipeline.headless !== undefined) {
755
+ pipelinePatch.headless = overrides.pipeline.headless;
756
+ }
757
+ if (overrides.pipeline.project) {
758
+ pipelinePatch.project = overrides.pipeline.project;
759
+ }
760
+ if (overrides.pipeline.parallel !== undefined) {
761
+ pipelinePatch.parallel = overrides.pipeline.parallel;
762
+ }
763
+ if (overrides.pipeline.dryRun !== undefined) {
764
+ pipelinePatch.dryRun = overrides.pipeline.dryRun;
765
+ }
766
+ if (overrides.pipeline.mcp !== undefined) {
767
+ pipelinePatch.mcp = overrides.pipeline.mcp;
768
+ }
769
+ if (overrides.pipeline.mcpAllowFallback !== undefined) {
770
+ pipelinePatch.mcpAllowFallback = overrides.pipeline.mcpAllowFallback;
771
+ }
772
+ if (overrides.pipeline.mcpOnly !== undefined) {
773
+ pipelinePatch.mcpOnly = overrides.pipeline.mcpOnly;
774
+ }
775
+ if (overrides.pipeline.mcpCommandTimeoutMs !== undefined) {
776
+ pipelinePatch.mcpCommandTimeoutMs = overrides.pipeline.mcpCommandTimeoutMs;
777
+ }
778
+ if (overrides.pipeline.mcpRetries !== undefined) {
779
+ pipelinePatch.mcpRetries = overrides.pipeline.mcpRetries;
780
+ }
781
+ config.pipeline = { ...config.pipeline, ...pipelinePatch };
782
+ }
783
+ if (overrides?.policy) {
784
+ const policyPatch = {};
785
+ if (overrides.policy.minConfidenceForTargeted !== undefined) {
786
+ policyPatch.minConfidenceForTargeted = overrides.policy.minConfidenceForTargeted;
787
+ }
788
+ if (overrides.policy.safeMergeMinConfidence !== undefined) {
789
+ policyPatch.safeMergeMinConfidence = overrides.policy.safeMergeMinConfidence;
790
+ }
791
+ if (overrides.policy.forceFullOnWarningsAtOrAbove !== undefined) {
792
+ policyPatch.forceFullOnWarningsAtOrAbove = overrides.policy.forceFullOnWarningsAtOrAbove;
793
+ }
794
+ if (overrides.policy.forceFullOnP0WithGaps !== undefined) {
795
+ policyPatch.forceFullOnP0WithGaps = overrides.policy.forceFullOnP0WithGaps;
796
+ }
797
+ if (overrides.policy.forceFullOnRiskyFiles !== undefined) {
798
+ policyPatch.forceFullOnRiskyFiles = overrides.policy.forceFullOnRiskyFiles;
799
+ }
800
+ if (overrides.policy.riskyFilePatterns && overrides.policy.riskyFilePatterns.length > 0) {
801
+ policyPatch.riskyFilePatterns = overrides.policy.riskyFilePatterns;
802
+ }
803
+ if (overrides.policy.enforcementMode !== undefined) {
804
+ policyPatch.enforcementMode = overrides.policy.enforcementMode;
805
+ }
806
+ if (overrides.policy.blockOnActions && overrides.policy.blockOnActions.length > 0) {
807
+ policyPatch.blockOnActions = overrides.policy.blockOnActions;
808
+ }
809
+ config.policy = { ...config.policy, ...policyPatch };
810
+ }
811
+ if (overrides?.specPDF) {
812
+ config.specPDF = overrides.specPDF;
813
+ }
814
+ if (overrides?.gitSince) {
815
+ config.git.since = overrides.gitSince;
816
+ }
817
+ if (overrides?.path) {
818
+ config.path = overrides.path;
819
+ }
820
+ if (overrides?.testsRoot) {
821
+ config.testsRoot = overrides.testsRoot;
822
+ }
823
+ if (overrides?.flowCatalogPath) {
824
+ config.flowCatalogPath = overrides.flowCatalogPath;
825
+ }
826
+ if (overrides?.llmProvider !== undefined) {
827
+ config.llm.provider = normalizeMattermostProvider(overrides.llmProvider, normalizeMattermostProvider(config.llm.provider, 'auto'), true);
828
+ }
829
+ if (overrides?.llmFallback !== undefined) {
830
+ config.llm.fallback = overrides.llmFallback;
831
+ }
832
+ if (config.profile === 'mattermost') {
833
+ config.impact.allowFallback = false;
834
+ config.impact.traceability.enabled = true;
835
+ config.impact.traceability.minSignalsPerTest = Math.max(2, config.impact.traceability.minSignalsPerTest);
836
+ config.impact.aiFlow.enabled = true;
837
+ config.impact.aiFlow.strict = true;
838
+ const provider = resolveMattermostProvider(config);
839
+ config.impact.aiFlow.provider = provider;
840
+ config.impact.aiMapping.enabled = true;
841
+ config.impact.aiMapping.provider = provider;
842
+ config.pipeline.mcp = true;
843
+ config.pipeline.mcpOnly = true;
844
+ config.pipeline.mcpAllowFallback = false;
845
+ config.pipeline.mcpCommandTimeoutMs = Math.min(240000, Math.max(60000, config.pipeline.mcpCommandTimeoutMs || 180000));
846
+ config.pipeline.mcpRetries = Math.max(1, Math.min(3, config.pipeline.mcpRetries || 1));
847
+ config.policy.minConfidenceForTargeted = Math.max(75, config.policy.minConfidenceForTargeted);
848
+ config.policy.safeMergeMinConfidence = Math.max(90, config.policy.safeMergeMinConfidence);
849
+ config.policy.forceFullOnWarningsAtOrAbove = 1;
850
+ config.policy.forceFullOnP0WithGaps = true;
851
+ config.policy.forceFullOnRiskyFiles = true;
852
+ config.routeFamilies.strict = true;
853
+ config.apiSurface.enabled = true;
854
+ }
855
+ const resolvedRoot = resolve(configDir, config.path);
856
+ config.path = resolvedRoot;
857
+ if (config.testsRoot) {
858
+ config.testsRoot = resolve(configDir, config.testsRoot);
859
+ }
860
+ else {
861
+ config.testsRoot = resolvedRoot;
862
+ }
863
+ if (config.flowCatalogPath) {
864
+ config.flowCatalogPath = resolve(configDir, config.flowCatalogPath);
865
+ }
866
+ config.impact.subsystemRisk.mapPath = resolve(configDir, config.impact.subsystemRisk.mapPath);
867
+ return {
868
+ config,
869
+ configPath: resolvedConfigPath,
870
+ rootDir: resolvedRoot,
871
+ };
872
+ }