prism-pr 1.0.0-alpha.47

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 (677) hide show
  1. package/README.md +964 -0
  2. package/bin/run.js +9 -0
  3. package/dist/ai/agents/architecture-reviewer.d.ts +3 -0
  4. package/dist/ai/agents/architecture-reviewer.d.ts.map +1 -0
  5. package/dist/ai/agents/architecture-reviewer.js +67 -0
  6. package/dist/ai/agents/architecture-reviewer.js.map +1 -0
  7. package/dist/ai/agents/blade-reviewer.d.ts +3 -0
  8. package/dist/ai/agents/blade-reviewer.d.ts.map +1 -0
  9. package/dist/ai/agents/blade-reviewer.js +59 -0
  10. package/dist/ai/agents/blade-reviewer.js.map +1 -0
  11. package/dist/ai/agents/config-reviewer.d.ts +3 -0
  12. package/dist/ai/agents/config-reviewer.d.ts.map +1 -0
  13. package/dist/ai/agents/config-reviewer.js +106 -0
  14. package/dist/ai/agents/config-reviewer.js.map +1 -0
  15. package/dist/ai/agents/csharp-reviewer.d.ts +3 -0
  16. package/dist/ai/agents/csharp-reviewer.d.ts.map +1 -0
  17. package/dist/ai/agents/csharp-reviewer.js +59 -0
  18. package/dist/ai/agents/csharp-reviewer.js.map +1 -0
  19. package/dist/ai/agents/css-reviewer.d.ts +3 -0
  20. package/dist/ai/agents/css-reviewer.d.ts.map +1 -0
  21. package/dist/ai/agents/css-reviewer.js +65 -0
  22. package/dist/ai/agents/css-reviewer.js.map +1 -0
  23. package/dist/ai/agents/framework-rules/index.d.ts +33 -0
  24. package/dist/ai/agents/framework-rules/index.d.ts.map +1 -0
  25. package/dist/ai/agents/framework-rules/index.js +67 -0
  26. package/dist/ai/agents/framework-rules/index.js.map +1 -0
  27. package/dist/ai/agents/framework-rules/normalize.d.ts +40 -0
  28. package/dist/ai/agents/framework-rules/normalize.d.ts.map +1 -0
  29. package/dist/ai/agents/framework-rules/normalize.js +102 -0
  30. package/dist/ai/agents/framework-rules/normalize.js.map +1 -0
  31. package/dist/ai/agents/framework-rules/routing-map.d.ts +3 -0
  32. package/dist/ai/agents/framework-rules/routing-map.d.ts.map +1 -0
  33. package/dist/ai/agents/framework-rules/routing-map.js +15 -0
  34. package/dist/ai/agents/framework-rules/routing-map.js.map +1 -0
  35. package/dist/ai/agents/html-reviewer.d.ts +3 -0
  36. package/dist/ai/agents/html-reviewer.d.ts.map +1 -0
  37. package/dist/ai/agents/html-reviewer.js +61 -0
  38. package/dist/ai/agents/html-reviewer.js.map +1 -0
  39. package/dist/ai/agents/performance-reviewer.d.ts +3 -0
  40. package/dist/ai/agents/performance-reviewer.d.ts.map +1 -0
  41. package/dist/ai/agents/performance-reviewer.js +60 -0
  42. package/dist/ai/agents/performance-reviewer.js.map +1 -0
  43. package/dist/ai/agents/php-reviewer.d.ts +3 -0
  44. package/dist/ai/agents/php-reviewer.d.ts.map +1 -0
  45. package/dist/ai/agents/php-reviewer.js +62 -0
  46. package/dist/ai/agents/php-reviewer.js.map +1 -0
  47. package/dist/ai/agents/prompts/architecture-reviewer.txt +39 -0
  48. package/dist/ai/agents/prompts/blade-reviewer.txt +39 -0
  49. package/dist/ai/agents/prompts/config-reviewer.txt +47 -0
  50. package/dist/ai/agents/prompts/csharp-reviewer.txt +39 -0
  51. package/dist/ai/agents/prompts/css-reviewer.txt +41 -0
  52. package/dist/ai/agents/prompts/html-reviewer.txt +43 -0
  53. package/dist/ai/agents/prompts/performance-reviewer.txt +38 -0
  54. package/dist/ai/agents/prompts/php-reviewer.txt +87 -0
  55. package/dist/ai/agents/prompts/python-reviewer.txt +42 -0
  56. package/dist/ai/agents/prompts/security-reviewer.txt +36 -0
  57. package/dist/ai/agents/prompts/sql-reviewer.txt +43 -0
  58. package/dist/ai/agents/prompts/testing-reviewer.txt +38 -0
  59. package/dist/ai/agents/prompts/ts-reviewer.txt +54 -0
  60. package/dist/ai/agents/prompts/ux-text-reviewer.txt +68 -0
  61. package/dist/ai/agents/python-reviewer.d.ts +3 -0
  62. package/dist/ai/agents/python-reviewer.d.ts.map +1 -0
  63. package/dist/ai/agents/python-reviewer.js +59 -0
  64. package/dist/ai/agents/python-reviewer.js.map +1 -0
  65. package/dist/ai/agents/registry.d.ts +20 -0
  66. package/dist/ai/agents/registry.d.ts.map +1 -0
  67. package/dist/ai/agents/registry.js +65 -0
  68. package/dist/ai/agents/registry.js.map +1 -0
  69. package/dist/ai/agents/security-reviewer.d.ts +3 -0
  70. package/dist/ai/agents/security-reviewer.d.ts.map +1 -0
  71. package/dist/ai/agents/security-reviewer.js +69 -0
  72. package/dist/ai/agents/security-reviewer.js.map +1 -0
  73. package/dist/ai/agents/shared/content-builder.d.ts +14 -0
  74. package/dist/ai/agents/shared/content-builder.d.ts.map +1 -0
  75. package/dist/ai/agents/shared/content-builder.js +158 -0
  76. package/dist/ai/agents/shared/content-builder.js.map +1 -0
  77. package/dist/ai/agents/shared/finding-schema.d.ts +54 -0
  78. package/dist/ai/agents/shared/finding-schema.d.ts.map +1 -0
  79. package/dist/ai/agents/shared/finding-schema.js +65 -0
  80. package/dist/ai/agents/shared/finding-schema.js.map +1 -0
  81. package/dist/ai/agents/shared/line-validator.d.ts +8 -0
  82. package/dist/ai/agents/shared/line-validator.d.ts.map +1 -0
  83. package/dist/ai/agents/shared/line-validator.js +12 -0
  84. package/dist/ai/agents/shared/line-validator.js.map +1 -0
  85. package/dist/ai/agents/shared/prompt-builder.d.ts +3 -0
  86. package/dist/ai/agents/shared/prompt-builder.d.ts.map +1 -0
  87. package/dist/ai/agents/shared/prompt-builder.js +68 -0
  88. package/dist/ai/agents/shared/prompt-builder.js.map +1 -0
  89. package/dist/ai/agents/sql-reviewer.d.ts +3 -0
  90. package/dist/ai/agents/sql-reviewer.d.ts.map +1 -0
  91. package/dist/ai/agents/sql-reviewer.js +61 -0
  92. package/dist/ai/agents/sql-reviewer.js.map +1 -0
  93. package/dist/ai/agents/testing-reviewer.d.ts +3 -0
  94. package/dist/ai/agents/testing-reviewer.d.ts.map +1 -0
  95. package/dist/ai/agents/testing-reviewer.js +80 -0
  96. package/dist/ai/agents/testing-reviewer.js.map +1 -0
  97. package/dist/ai/agents/ts-reviewer.d.ts +3 -0
  98. package/dist/ai/agents/ts-reviewer.d.ts.map +1 -0
  99. package/dist/ai/agents/ts-reviewer.js +63 -0
  100. package/dist/ai/agents/ts-reviewer.js.map +1 -0
  101. package/dist/ai/agents/ux-text-reviewer.d.ts +3 -0
  102. package/dist/ai/agents/ux-text-reviewer.d.ts.map +1 -0
  103. package/dist/ai/agents/ux-text-reviewer.js +116 -0
  104. package/dist/ai/agents/ux-text-reviewer.js.map +1 -0
  105. package/dist/ai/orchestrator/agent-classifier.d.ts +28 -0
  106. package/dist/ai/orchestrator/agent-classifier.d.ts.map +1 -0
  107. package/dist/ai/orchestrator/agent-classifier.js +137 -0
  108. package/dist/ai/orchestrator/agent-classifier.js.map +1 -0
  109. package/dist/ai/orchestrator/consolidation.d.ts +12 -0
  110. package/dist/ai/orchestrator/consolidation.d.ts.map +1 -0
  111. package/dist/ai/orchestrator/consolidation.js +34 -0
  112. package/dist/ai/orchestrator/consolidation.js.map +1 -0
  113. package/dist/ai/orchestrator/context-analyzer.d.ts +47 -0
  114. package/dist/ai/orchestrator/context-analyzer.d.ts.map +1 -0
  115. package/dist/ai/orchestrator/context-analyzer.js +368 -0
  116. package/dist/ai/orchestrator/context-analyzer.js.map +1 -0
  117. package/dist/ai/orchestrator/deduplicator.d.ts +43 -0
  118. package/dist/ai/orchestrator/deduplicator.d.ts.map +1 -0
  119. package/dist/ai/orchestrator/deduplicator.js +97 -0
  120. package/dist/ai/orchestrator/deduplicator.js.map +1 -0
  121. package/dist/ai/orchestrator/orchestrator.d.ts +57 -0
  122. package/dist/ai/orchestrator/orchestrator.d.ts.map +1 -0
  123. package/dist/ai/orchestrator/orchestrator.js +181 -0
  124. package/dist/ai/orchestrator/orchestrator.js.map +1 -0
  125. package/dist/ai/orchestrator/rule-based-strategy.d.ts +20 -0
  126. package/dist/ai/orchestrator/rule-based-strategy.d.ts.map +1 -0
  127. package/dist/ai/orchestrator/rule-based-strategy.js +55 -0
  128. package/dist/ai/orchestrator/rule-based-strategy.js.map +1 -0
  129. package/dist/ai/orchestrator/semaphore.d.ts +25 -0
  130. package/dist/ai/orchestrator/semaphore.d.ts.map +1 -0
  131. package/dist/ai/orchestrator/semaphore.js +52 -0
  132. package/dist/ai/orchestrator/semaphore.js.map +1 -0
  133. package/dist/ai/orchestrator/severity-pipeline.d.ts +21 -0
  134. package/dist/ai/orchestrator/severity-pipeline.d.ts.map +1 -0
  135. package/dist/ai/orchestrator/severity-pipeline.js +69 -0
  136. package/dist/ai/orchestrator/severity-pipeline.js.map +1 -0
  137. package/dist/ai/orchestrator/strategy.d.ts +20 -0
  138. package/dist/ai/orchestrator/strategy.d.ts.map +1 -0
  139. package/dist/ai/orchestrator/strategy.js +2 -0
  140. package/dist/ai/orchestrator/strategy.js.map +1 -0
  141. package/dist/ai/providers/anthropic-provider.d.ts +14 -0
  142. package/dist/ai/providers/anthropic-provider.d.ts.map +1 -0
  143. package/dist/ai/providers/anthropic-provider.js +108 -0
  144. package/dist/ai/providers/anthropic-provider.js.map +1 -0
  145. package/dist/ai/providers/claude-code-provider.d.ts +12 -0
  146. package/dist/ai/providers/claude-code-provider.d.ts.map +1 -0
  147. package/dist/ai/providers/claude-code-provider.js +103 -0
  148. package/dist/ai/providers/claude-code-provider.js.map +1 -0
  149. package/dist/ai/providers/errors.d.ts +22 -0
  150. package/dist/ai/providers/errors.d.ts.map +1 -0
  151. package/dist/ai/providers/errors.js +44 -0
  152. package/dist/ai/providers/errors.js.map +1 -0
  153. package/dist/ai/skills/skill-registry.d.ts +21 -0
  154. package/dist/ai/skills/skill-registry.d.ts.map +1 -0
  155. package/dist/ai/skills/skill-registry.js +27 -0
  156. package/dist/ai/skills/skill-registry.js.map +1 -0
  157. package/dist/ai/text-extractor/extractor.d.ts +15 -0
  158. package/dist/ai/text-extractor/extractor.d.ts.map +1 -0
  159. package/dist/ai/text-extractor/extractor.js +63 -0
  160. package/dist/ai/text-extractor/extractor.js.map +1 -0
  161. package/dist/ai/text-extractor/index.d.ts +4 -0
  162. package/dist/ai/text-extractor/index.d.ts.map +1 -0
  163. package/dist/ai/text-extractor/index.js +3 -0
  164. package/dist/ai/text-extractor/index.js.map +1 -0
  165. package/dist/ai/text-extractor/patterns.d.ts +21 -0
  166. package/dist/ai/text-extractor/patterns.d.ts.map +1 -0
  167. package/dist/ai/text-extractor/patterns.js +169 -0
  168. package/dist/ai/text-extractor/patterns.js.map +1 -0
  169. package/dist/ai/text-extractor/types.d.ts +47 -0
  170. package/dist/ai/text-extractor/types.d.ts.map +1 -0
  171. package/dist/ai/text-extractor/types.js +2 -0
  172. package/dist/ai/text-extractor/types.js.map +1 -0
  173. package/dist/bitbucket/client.d.ts +38 -0
  174. package/dist/bitbucket/client.d.ts.map +1 -0
  175. package/dist/bitbucket/client.js +448 -0
  176. package/dist/bitbucket/client.js.map +1 -0
  177. package/dist/bitbucket/errors.d.ts +75 -0
  178. package/dist/bitbucket/errors.d.ts.map +1 -0
  179. package/dist/bitbucket/errors.js +127 -0
  180. package/dist/bitbucket/errors.js.map +1 -0
  181. package/dist/bitbucket/mappers.d.ts +3 -0
  182. package/dist/bitbucket/mappers.d.ts.map +1 -0
  183. package/dist/bitbucket/mappers.js +14 -0
  184. package/dist/bitbucket/mappers.js.map +1 -0
  185. package/dist/commands/guard/check.d.ts +19 -0
  186. package/dist/commands/guard/check.d.ts.map +1 -0
  187. package/dist/commands/guard/check.js +224 -0
  188. package/dist/commands/guard/check.js.map +1 -0
  189. package/dist/commands/guard/index.d.ts +7 -0
  190. package/dist/commands/guard/index.d.ts.map +1 -0
  191. package/dist/commands/guard/index.js +65 -0
  192. package/dist/commands/guard/index.js.map +1 -0
  193. package/dist/commands/interactive.d.ts +10 -0
  194. package/dist/commands/interactive.d.ts.map +1 -0
  195. package/dist/commands/interactive.js +88 -0
  196. package/dist/commands/interactive.js.map +1 -0
  197. package/dist/commands/login.d.ts +35 -0
  198. package/dist/commands/login.d.ts.map +1 -0
  199. package/dist/commands/login.js +164 -0
  200. package/dist/commands/login.js.map +1 -0
  201. package/dist/commands/logout.d.ts +12 -0
  202. package/dist/commands/logout.d.ts.map +1 -0
  203. package/dist/commands/logout.js +23 -0
  204. package/dist/commands/logout.js.map +1 -0
  205. package/dist/commands/review/start.d.ts +22 -0
  206. package/dist/commands/review/start.d.ts.map +1 -0
  207. package/dist/commands/review/start.js +120 -0
  208. package/dist/commands/review/start.js.map +1 -0
  209. package/dist/commands/rules/push.d.ts +13 -0
  210. package/dist/commands/rules/push.d.ts.map +1 -0
  211. package/dist/commands/rules/push.js +75 -0
  212. package/dist/commands/rules/push.js.map +1 -0
  213. package/dist/commands/rules/stats.d.ts +14 -0
  214. package/dist/commands/rules/stats.d.ts.map +1 -0
  215. package/dist/commands/rules/stats.js +101 -0
  216. package/dist/commands/rules/stats.js.map +1 -0
  217. package/dist/commands/rules/sync.d.ts +15 -0
  218. package/dist/commands/rules/sync.d.ts.map +1 -0
  219. package/dist/commands/rules/sync.js +144 -0
  220. package/dist/commands/rules/sync.js.map +1 -0
  221. package/dist/config/config-manager.d.ts +14 -0
  222. package/dist/config/config-manager.d.ts.map +1 -0
  223. package/dist/config/config-manager.js +85 -0
  224. package/dist/config/config-manager.js.map +1 -0
  225. package/dist/config/convention-builder.d.ts +13 -0
  226. package/dist/config/convention-builder.d.ts.map +1 -0
  227. package/dist/config/convention-builder.js +31 -0
  228. package/dist/config/convention-builder.js.map +1 -0
  229. package/dist/config/credentials.d.ts +14 -0
  230. package/dist/config/credentials.d.ts.map +1 -0
  231. package/dist/config/credentials.js +85 -0
  232. package/dist/config/credentials.js.map +1 -0
  233. package/dist/config/encryption.d.ts +20 -0
  234. package/dist/config/encryption.d.ts.map +1 -0
  235. package/dist/config/encryption.js +94 -0
  236. package/dist/config/encryption.js.map +1 -0
  237. package/dist/config/repo-config-loader.d.ts +34 -0
  238. package/dist/config/repo-config-loader.d.ts.map +1 -0
  239. package/dist/config/repo-config-loader.js +103 -0
  240. package/dist/config/repo-config-loader.js.map +1 -0
  241. package/dist/core/cli-workflow-callbacks.d.ts +35 -0
  242. package/dist/core/cli-workflow-callbacks.d.ts.map +1 -0
  243. package/dist/core/cli-workflow-callbacks.js +141 -0
  244. package/dist/core/cli-workflow-callbacks.js.map +1 -0
  245. package/dist/core/comment-publisher.d.ts +61 -0
  246. package/dist/core/comment-publisher.d.ts.map +1 -0
  247. package/dist/core/comment-publisher.js +172 -0
  248. package/dist/core/comment-publisher.js.map +1 -0
  249. package/dist/core/integrity-checker.d.ts +22 -0
  250. package/dist/core/integrity-checker.d.ts.map +1 -0
  251. package/dist/core/integrity-checker.js +83 -0
  252. package/dist/core/integrity-checker.js.map +1 -0
  253. package/dist/core/review-workflow.d.ts +34 -0
  254. package/dist/core/review-workflow.d.ts.map +1 -0
  255. package/dist/core/review-workflow.js +21 -0
  256. package/dist/core/review-workflow.js.map +1 -0
  257. package/dist/core/session-manager.d.ts +107 -0
  258. package/dist/core/session-manager.d.ts.map +1 -0
  259. package/dist/core/session-manager.js +164 -0
  260. package/dist/core/session-manager.js.map +1 -0
  261. package/dist/core/workflow-callbacks.d.ts +74 -0
  262. package/dist/core/workflow-callbacks.d.ts.map +1 -0
  263. package/dist/core/workflow-callbacks.js +2 -0
  264. package/dist/core/workflow-callbacks.js.map +1 -0
  265. package/dist/core/workflow-engine.d.ts +48 -0
  266. package/dist/core/workflow-engine.d.ts.map +1 -0
  267. package/dist/core/workflow-engine.js +442 -0
  268. package/dist/core/workflow-engine.js.map +1 -0
  269. package/dist/errors/base.d.ts +6 -0
  270. package/dist/errors/base.d.ts.map +1 -0
  271. package/dist/errors/base.js +13 -0
  272. package/dist/errors/base.js.map +1 -0
  273. package/dist/errors/config-error.d.ts +5 -0
  274. package/dist/errors/config-error.d.ts.map +1 -0
  275. package/dist/errors/config-error.js +9 -0
  276. package/dist/errors/config-error.js.map +1 -0
  277. package/dist/errors/index.d.ts +5 -0
  278. package/dist/errors/index.d.ts.map +1 -0
  279. package/dist/errors/index.js +5 -0
  280. package/dist/errors/index.js.map +1 -0
  281. package/dist/errors/validation-error.d.ts +5 -0
  282. package/dist/errors/validation-error.d.ts.map +1 -0
  283. package/dist/errors/validation-error.js +9 -0
  284. package/dist/errors/validation-error.js.map +1 -0
  285. package/dist/errors/workflow-error.d.ts +6 -0
  286. package/dist/errors/workflow-error.d.ts.map +1 -0
  287. package/dist/errors/workflow-error.js +11 -0
  288. package/dist/errors/workflow-error.js.map +1 -0
  289. package/dist/guard/ast-grep-strategy.d.ts +25 -0
  290. package/dist/guard/ast-grep-strategy.d.ts.map +1 -0
  291. package/dist/guard/ast-grep-strategy.js +112 -0
  292. package/dist/guard/ast-grep-strategy.js.map +1 -0
  293. package/dist/guard/diff-pattern-matcher.d.ts +52 -0
  294. package/dist/guard/diff-pattern-matcher.d.ts.map +1 -0
  295. package/dist/guard/diff-pattern-matcher.js +325 -0
  296. package/dist/guard/diff-pattern-matcher.js.map +1 -0
  297. package/dist/guard/file-fetcher.d.ts +3 -0
  298. package/dist/guard/file-fetcher.d.ts.map +1 -0
  299. package/dist/guard/file-fetcher.js +21 -0
  300. package/dist/guard/file-fetcher.js.map +1 -0
  301. package/dist/guard/guard-engine.d.ts +8 -0
  302. package/dist/guard/guard-engine.d.ts.map +1 -0
  303. package/dist/guard/guard-engine.js +78 -0
  304. package/dist/guard/guard-engine.js.map +1 -0
  305. package/dist/guard/match-strategy.d.ts +14 -0
  306. package/dist/guard/match-strategy.d.ts.map +1 -0
  307. package/dist/guard/match-strategy.js +7 -0
  308. package/dist/guard/match-strategy.js.map +1 -0
  309. package/dist/guard/patterns-loader.d.ts +16 -0
  310. package/dist/guard/patterns-loader.d.ts.map +1 -0
  311. package/dist/guard/patterns-loader.js +61 -0
  312. package/dist/guard/patterns-loader.js.map +1 -0
  313. package/dist/guard/types.d.ts +61 -0
  314. package/dist/guard/types.d.ts.map +1 -0
  315. package/dist/guard/types.js +2 -0
  316. package/dist/guard/types.js.map +1 -0
  317. package/dist/memory/engram-client.d.ts +15 -0
  318. package/dist/memory/engram-client.d.ts.map +1 -0
  319. package/dist/memory/engram-client.js +167 -0
  320. package/dist/memory/engram-client.js.map +1 -0
  321. package/dist/memory/review-memory-service.d.ts +13 -0
  322. package/dist/memory/review-memory-service.d.ts.map +1 -0
  323. package/dist/memory/review-memory-service.js +43 -0
  324. package/dist/memory/review-memory-service.js.map +1 -0
  325. package/dist/persistence/database.d.ts +8 -0
  326. package/dist/persistence/database.d.ts.map +1 -0
  327. package/dist/persistence/database.js +58 -0
  328. package/dist/persistence/database.js.map +1 -0
  329. package/dist/persistence/false-positive-repository.d.ts +12 -0
  330. package/dist/persistence/false-positive-repository.d.ts.map +1 -0
  331. package/dist/persistence/false-positive-repository.js +58 -0
  332. package/dist/persistence/false-positive-repository.js.map +1 -0
  333. package/dist/persistence/migrations/0000_peaceful_blur.sql +66 -0
  334. package/dist/persistence/migrations/0001_phase3a_schema.sql +15 -0
  335. package/dist/persistence/migrations/0002_stack_profile.sql +1 -0
  336. package/dist/persistence/migrations/0003_known_workspaces.sql +5 -0
  337. package/dist/persistence/migrations/0004_review_memory.sql +24 -0
  338. package/dist/persistence/migrations/meta/0000_snapshot.json +468 -0
  339. package/dist/persistence/migrations/meta/_journal.json +41 -0
  340. package/dist/persistence/profile-repository.d.ts +33 -0
  341. package/dist/persistence/profile-repository.d.ts.map +1 -0
  342. package/dist/persistence/profile-repository.js +94 -0
  343. package/dist/persistence/profile-repository.js.map +1 -0
  344. package/dist/persistence/repository.d.ts +70 -0
  345. package/dist/persistence/repository.d.ts.map +1 -0
  346. package/dist/persistence/repository.js +225 -0
  347. package/dist/persistence/repository.js.map +1 -0
  348. package/dist/persistence/schema.d.ts +1333 -0
  349. package/dist/persistence/schema.d.ts.map +1 -0
  350. package/dist/persistence/schema.js +92 -0
  351. package/dist/persistence/schema.js.map +1 -0
  352. package/dist/persistence/workspace-repository.d.ts +14 -0
  353. package/dist/persistence/workspace-repository.d.ts.map +1 -0
  354. package/dist/persistence/workspace-repository.js +30 -0
  355. package/dist/persistence/workspace-repository.js.map +1 -0
  356. package/dist/rules-engine/finding-aggregator.d.ts +13 -0
  357. package/dist/rules-engine/finding-aggregator.d.ts.map +1 -0
  358. package/dist/rules-engine/finding-aggregator.js +135 -0
  359. package/dist/rules-engine/finding-aggregator.js.map +1 -0
  360. package/dist/rules-engine/pattern-generator.d.ts +26 -0
  361. package/dist/rules-engine/pattern-generator.d.ts.map +1 -0
  362. package/dist/rules-engine/pattern-generator.js +116 -0
  363. package/dist/rules-engine/pattern-generator.js.map +1 -0
  364. package/dist/rules-engine/rule-prompt-builder.d.ts +32 -0
  365. package/dist/rules-engine/rule-prompt-builder.d.ts.map +1 -0
  366. package/dist/rules-engine/rule-prompt-builder.js +124 -0
  367. package/dist/rules-engine/rule-prompt-builder.js.map +1 -0
  368. package/dist/rules-engine/types.d.ts +80 -0
  369. package/dist/rules-engine/types.d.ts.map +1 -0
  370. package/dist/rules-engine/types.js +18 -0
  371. package/dist/rules-engine/types.js.map +1 -0
  372. package/dist/rules-repo/rules-cache.d.ts +14 -0
  373. package/dist/rules-repo/rules-cache.d.ts.map +1 -0
  374. package/dist/rules-repo/rules-cache.js +43 -0
  375. package/dist/rules-repo/rules-cache.js.map +1 -0
  376. package/dist/rules-repo/rules-repo-client.d.ts +27 -0
  377. package/dist/rules-repo/rules-repo-client.d.ts.map +1 -0
  378. package/dist/rules-repo/rules-repo-client.js +85 -0
  379. package/dist/rules-repo/rules-repo-client.js.map +1 -0
  380. package/dist/rules-repo/types.d.ts +16 -0
  381. package/dist/rules-repo/types.d.ts.map +1 -0
  382. package/dist/rules-repo/types.js +10 -0
  383. package/dist/rules-repo/types.js.map +1 -0
  384. package/dist/stack-detector/ai-stack-fallback.d.ts +12 -0
  385. package/dist/stack-detector/ai-stack-fallback.d.ts.map +1 -0
  386. package/dist/stack-detector/ai-stack-fallback.js +67 -0
  387. package/dist/stack-detector/ai-stack-fallback.js.map +1 -0
  388. package/dist/stack-detector/ai-stack-schema.d.ts +28 -0
  389. package/dist/stack-detector/ai-stack-schema.d.ts.map +1 -0
  390. package/dist/stack-detector/ai-stack-schema.js +13 -0
  391. package/dist/stack-detector/ai-stack-schema.js.map +1 -0
  392. package/dist/stack-detector/formatter.d.ts +7 -0
  393. package/dist/stack-detector/formatter.d.ts.map +1 -0
  394. package/dist/stack-detector/formatter.js +37 -0
  395. package/dist/stack-detector/formatter.js.map +1 -0
  396. package/dist/stack-detector/index.d.ts +7 -0
  397. package/dist/stack-detector/index.d.ts.map +1 -0
  398. package/dist/stack-detector/index.js +5 -0
  399. package/dist/stack-detector/index.js.map +1 -0
  400. package/dist/stack-detector/merge-profiles.d.ts +5 -0
  401. package/dist/stack-detector/merge-profiles.d.ts.map +1 -0
  402. package/dist/stack-detector/merge-profiles.js +36 -0
  403. package/dist/stack-detector/merge-profiles.js.map +1 -0
  404. package/dist/stack-detector/parsers.d.ts +12 -0
  405. package/dist/stack-detector/parsers.d.ts.map +1 -0
  406. package/dist/stack-detector/parsers.js +259 -0
  407. package/dist/stack-detector/parsers.js.map +1 -0
  408. package/dist/stack-detector/stack-detector.d.ts +11 -0
  409. package/dist/stack-detector/stack-detector.d.ts.map +1 -0
  410. package/dist/stack-detector/stack-detector.js +83 -0
  411. package/dist/stack-detector/stack-detector.js.map +1 -0
  412. package/dist/stack-detector/types.d.ts +13 -0
  413. package/dist/stack-detector/types.d.ts.map +1 -0
  414. package/dist/stack-detector/types.js +2 -0
  415. package/dist/stack-detector/types.js.map +1 -0
  416. package/dist/tui/app.d.ts +9 -0
  417. package/dist/tui/app.d.ts.map +1 -0
  418. package/dist/tui/app.js +45 -0
  419. package/dist/tui/app.js.map +1 -0
  420. package/dist/tui/callbacks/tui-workflow-callbacks.d.ts +32 -0
  421. package/dist/tui/callbacks/tui-workflow-callbacks.d.ts.map +1 -0
  422. package/dist/tui/callbacks/tui-workflow-callbacks.js +88 -0
  423. package/dist/tui/callbacks/tui-workflow-callbacks.js.map +1 -0
  424. package/dist/tui/components/key-hint.d.ts +7 -0
  425. package/dist/tui/components/key-hint.d.ts.map +1 -0
  426. package/dist/tui/components/key-hint.js +6 -0
  427. package/dist/tui/components/key-hint.js.map +1 -0
  428. package/dist/tui/components/scroll-list.d.ts +20 -0
  429. package/dist/tui/components/scroll-list.d.ts.map +1 -0
  430. package/dist/tui/components/scroll-list.js +85 -0
  431. package/dist/tui/components/scroll-list.js.map +1 -0
  432. package/dist/tui/components/searchable-list.d.ts +15 -0
  433. package/dist/tui/components/searchable-list.d.ts.map +1 -0
  434. package/dist/tui/components/searchable-list.js +97 -0
  435. package/dist/tui/components/searchable-list.js.map +1 -0
  436. package/dist/tui/components/severity-badge.d.ts +6 -0
  437. package/dist/tui/components/severity-badge.d.ts.map +1 -0
  438. package/dist/tui/components/severity-badge.js +17 -0
  439. package/dist/tui/components/severity-badge.js.map +1 -0
  440. package/dist/tui/components/severity-selector.d.ts +9 -0
  441. package/dist/tui/components/severity-selector.d.ts.map +1 -0
  442. package/dist/tui/components/severity-selector.js +29 -0
  443. package/dist/tui/components/severity-selector.js.map +1 -0
  444. package/dist/tui/components/spinner.d.ts +6 -0
  445. package/dist/tui/components/spinner.d.ts.map +1 -0
  446. package/dist/tui/components/spinner.js +19 -0
  447. package/dist/tui/components/spinner.js.map +1 -0
  448. package/dist/tui/components/status-bar.d.ts +10 -0
  449. package/dist/tui/components/status-bar.d.ts.map +1 -0
  450. package/dist/tui/components/status-bar.js +7 -0
  451. package/dist/tui/components/status-bar.js.map +1 -0
  452. package/dist/tui/deps-context.d.ts +27 -0
  453. package/dist/tui/deps-context.d.ts.map +1 -0
  454. package/dist/tui/deps-context.js +9 -0
  455. package/dist/tui/deps-context.js.map +1 -0
  456. package/dist/tui/error-boundary.d.ts +16 -0
  457. package/dist/tui/error-boundary.d.ts.map +1 -0
  458. package/dist/tui/error-boundary.js +30 -0
  459. package/dist/tui/error-boundary.js.map +1 -0
  460. package/dist/tui/hooks/use-agent-recommendations.d.ts +10 -0
  461. package/dist/tui/hooks/use-agent-recommendations.d.ts.map +1 -0
  462. package/dist/tui/hooks/use-agent-recommendations.js +66 -0
  463. package/dist/tui/hooks/use-agent-recommendations.js.map +1 -0
  464. package/dist/tui/hooks/use-auth.d.ts +14 -0
  465. package/dist/tui/hooks/use-auth.d.ts.map +1 -0
  466. package/dist/tui/hooks/use-auth.js +32 -0
  467. package/dist/tui/hooks/use-auth.js.map +1 -0
  468. package/dist/tui/hooks/use-bitbucket.d.ts +18 -0
  469. package/dist/tui/hooks/use-bitbucket.d.ts.map +1 -0
  470. package/dist/tui/hooks/use-bitbucket.js +31 -0
  471. package/dist/tui/hooks/use-bitbucket.js.map +1 -0
  472. package/dist/tui/hooks/use-review.d.ts +6 -0
  473. package/dist/tui/hooks/use-review.d.ts.map +1 -0
  474. package/dist/tui/hooks/use-review.js +24 -0
  475. package/dist/tui/hooks/use-review.js.map +1 -0
  476. package/dist/tui/hooks/use-session.d.ts +47 -0
  477. package/dist/tui/hooks/use-session.d.ts.map +1 -0
  478. package/dist/tui/hooks/use-session.js +17 -0
  479. package/dist/tui/hooks/use-session.js.map +1 -0
  480. package/dist/tui/screen-router.d.ts +3 -0
  481. package/dist/tui/screen-router.d.ts.map +1 -0
  482. package/dist/tui/screen-router.js +73 -0
  483. package/dist/tui/screen-router.js.map +1 -0
  484. package/dist/tui/screens/agent-select.d.ts +3 -0
  485. package/dist/tui/screens/agent-select.d.ts.map +1 -0
  486. package/dist/tui/screens/agent-select.js +173 -0
  487. package/dist/tui/screens/agent-select.js.map +1 -0
  488. package/dist/tui/screens/finding-detail.d.ts +3 -0
  489. package/dist/tui/screens/finding-detail.d.ts.map +1 -0
  490. package/dist/tui/screens/finding-detail.js +48 -0
  491. package/dist/tui/screens/finding-detail.js.map +1 -0
  492. package/dist/tui/screens/findings-list.d.ts +3 -0
  493. package/dist/tui/screens/findings-list.d.ts.map +1 -0
  494. package/dist/tui/screens/findings-list.js +108 -0
  495. package/dist/tui/screens/findings-list.js.map +1 -0
  496. package/dist/tui/screens/guard-detail.d.ts +3 -0
  497. package/dist/tui/screens/guard-detail.d.ts.map +1 -0
  498. package/dist/tui/screens/guard-detail.js +41 -0
  499. package/dist/tui/screens/guard-detail.js.map +1 -0
  500. package/dist/tui/screens/guard-home.d.ts +3 -0
  501. package/dist/tui/screens/guard-home.d.ts.map +1 -0
  502. package/dist/tui/screens/guard-home.js +81 -0
  503. package/dist/tui/screens/guard-home.js.map +1 -0
  504. package/dist/tui/screens/guard-results.d.ts +3 -0
  505. package/dist/tui/screens/guard-results.d.ts.map +1 -0
  506. package/dist/tui/screens/guard-results.js +103 -0
  507. package/dist/tui/screens/guard-results.js.map +1 -0
  508. package/dist/tui/screens/guard-running.d.ts +3 -0
  509. package/dist/tui/screens/guard-running.d.ts.map +1 -0
  510. package/dist/tui/screens/guard-running.js +92 -0
  511. package/dist/tui/screens/guard-running.js.map +1 -0
  512. package/dist/tui/screens/home.d.ts +3 -0
  513. package/dist/tui/screens/home.d.ts.map +1 -0
  514. package/dist/tui/screens/home.js +44 -0
  515. package/dist/tui/screens/home.js.map +1 -0
  516. package/dist/tui/screens/integrity-report.d.ts +3 -0
  517. package/dist/tui/screens/integrity-report.d.ts.map +1 -0
  518. package/dist/tui/screens/integrity-report.js +29 -0
  519. package/dist/tui/screens/integrity-report.js.map +1 -0
  520. package/dist/tui/screens/login.d.ts +3 -0
  521. package/dist/tui/screens/login.d.ts.map +1 -0
  522. package/dist/tui/screens/login.js +81 -0
  523. package/dist/tui/screens/login.js.map +1 -0
  524. package/dist/tui/screens/pr-list.d.ts +3 -0
  525. package/dist/tui/screens/pr-list.d.ts.map +1 -0
  526. package/dist/tui/screens/pr-list.js +76 -0
  527. package/dist/tui/screens/pr-list.js.map +1 -0
  528. package/dist/tui/screens/publish-confirm.d.ts +3 -0
  529. package/dist/tui/screens/publish-confirm.d.ts.map +1 -0
  530. package/dist/tui/screens/publish-confirm.js +42 -0
  531. package/dist/tui/screens/publish-confirm.js.map +1 -0
  532. package/dist/tui/screens/publishing.d.ts +3 -0
  533. package/dist/tui/screens/publishing.d.ts.map +1 -0
  534. package/dist/tui/screens/publishing.js +50 -0
  535. package/dist/tui/screens/publishing.js.map +1 -0
  536. package/dist/tui/screens/repo-select.d.ts +3 -0
  537. package/dist/tui/screens/repo-select.d.ts.map +1 -0
  538. package/dist/tui/screens/repo-select.js +76 -0
  539. package/dist/tui/screens/repo-select.js.map +1 -0
  540. package/dist/tui/screens/review-running.d.ts +3 -0
  541. package/dist/tui/screens/review-running.d.ts.map +1 -0
  542. package/dist/tui/screens/review-running.js +187 -0
  543. package/dist/tui/screens/review-running.js.map +1 -0
  544. package/dist/tui/screens/session-detail.d.ts +7 -0
  545. package/dist/tui/screens/session-detail.d.ts.map +1 -0
  546. package/dist/tui/screens/session-detail.js +28 -0
  547. package/dist/tui/screens/session-detail.js.map +1 -0
  548. package/dist/tui/screens/session-list.d.ts +7 -0
  549. package/dist/tui/screens/session-list.d.ts.map +1 -0
  550. package/dist/tui/screens/session-list.js +38 -0
  551. package/dist/tui/screens/session-list.js.map +1 -0
  552. package/dist/tui/screens/url-input.d.ts +3 -0
  553. package/dist/tui/screens/url-input.d.ts.map +1 -0
  554. package/dist/tui/screens/url-input.js +111 -0
  555. package/dist/tui/screens/url-input.js.map +1 -0
  556. package/dist/tui/screens/welcome.d.ts +3 -0
  557. package/dist/tui/screens/welcome.d.ts.map +1 -0
  558. package/dist/tui/screens/welcome.js +48 -0
  559. package/dist/tui/screens/welcome.js.map +1 -0
  560. package/dist/tui/screens/workspace-select.d.ts +3 -0
  561. package/dist/tui/screens/workspace-select.d.ts.map +1 -0
  562. package/dist/tui/screens/workspace-select.js +128 -0
  563. package/dist/tui/screens/workspace-select.js.map +1 -0
  564. package/dist/tui/state/router-context.d.ts +14 -0
  565. package/dist/tui/state/router-context.d.ts.map +1 -0
  566. package/dist/tui/state/router-context.js +9 -0
  567. package/dist/tui/state/router-context.js.map +1 -0
  568. package/dist/tui/state/router-reducer.d.ts +23 -0
  569. package/dist/tui/state/router-reducer.d.ts.map +1 -0
  570. package/dist/tui/state/router-reducer.js +21 -0
  571. package/dist/tui/state/router-reducer.js.map +1 -0
  572. package/dist/tui/state/tui-context.d.ts +11 -0
  573. package/dist/tui/state/tui-context.d.ts.map +1 -0
  574. package/dist/tui/state/tui-context.js +9 -0
  575. package/dist/tui/state/tui-context.js.map +1 -0
  576. package/dist/tui/state/tui-reducer.d.ts +212 -0
  577. package/dist/tui/state/tui-reducer.d.ts.map +1 -0
  578. package/dist/tui/state/tui-reducer.js +321 -0
  579. package/dist/tui/state/tui-reducer.js.map +1 -0
  580. package/dist/tui/utils/group-by-file.d.ts +7 -0
  581. package/dist/tui/utils/group-by-file.d.ts.map +1 -0
  582. package/dist/tui/utils/group-by-file.js +18 -0
  583. package/dist/tui/utils/group-by-file.js.map +1 -0
  584. package/dist/tui/utils/group-matches-by-file.d.ts +7 -0
  585. package/dist/tui/utils/group-matches-by-file.d.ts.map +1 -0
  586. package/dist/tui/utils/group-matches-by-file.js +18 -0
  587. package/dist/tui/utils/group-matches-by-file.js.map +1 -0
  588. package/dist/tui/utils/group-repos-by-project.d.ts +9 -0
  589. package/dist/tui/utils/group-repos-by-project.d.ts.map +1 -0
  590. package/dist/tui/utils/group-repos-by-project.js +48 -0
  591. package/dist/tui/utils/group-repos-by-project.js.map +1 -0
  592. package/dist/types/agent.d.ts +46 -0
  593. package/dist/types/agent.d.ts.map +1 -0
  594. package/dist/types/agent.js +2 -0
  595. package/dist/types/agent.js.map +1 -0
  596. package/dist/types/bitbucket.d.ts +90 -0
  597. package/dist/types/bitbucket.d.ts.map +1 -0
  598. package/dist/types/bitbucket.js +2 -0
  599. package/dist/types/bitbucket.js.map +1 -0
  600. package/dist/types/config.d.ts +17 -0
  601. package/dist/types/config.d.ts.map +1 -0
  602. package/dist/types/config.js +2 -0
  603. package/dist/types/config.js.map +1 -0
  604. package/dist/types/diff.d.ts +24 -0
  605. package/dist/types/diff.d.ts.map +1 -0
  606. package/dist/types/diff.js +2 -0
  607. package/dist/types/diff.js.map +1 -0
  608. package/dist/types/engram.d.ts +29 -0
  609. package/dist/types/engram.d.ts.map +1 -0
  610. package/dist/types/engram.js +2 -0
  611. package/dist/types/engram.js.map +1 -0
  612. package/dist/types/false-positive.d.ts +12 -0
  613. package/dist/types/false-positive.d.ts.map +1 -0
  614. package/dist/types/false-positive.js +2 -0
  615. package/dist/types/false-positive.js.map +1 -0
  616. package/dist/types/index.d.ts +14 -0
  617. package/dist/types/index.d.ts.map +1 -0
  618. package/dist/types/index.js +3 -0
  619. package/dist/types/index.js.map +1 -0
  620. package/dist/types/provider.d.ts +31 -0
  621. package/dist/types/provider.d.ts.map +1 -0
  622. package/dist/types/provider.js +2 -0
  623. package/dist/types/provider.js.map +1 -0
  624. package/dist/types/repo-config.d.ts +336 -0
  625. package/dist/types/repo-config.d.ts.map +1 -0
  626. package/dist/types/repo-config.js +62 -0
  627. package/dist/types/repo-config.js.map +1 -0
  628. package/dist/types/review-finding.d.ts +34 -0
  629. package/dist/types/review-finding.d.ts.map +1 -0
  630. package/dist/types/review-finding.js +11 -0
  631. package/dist/types/review-finding.js.map +1 -0
  632. package/dist/types/review-session.d.ts +21 -0
  633. package/dist/types/review-session.d.ts.map +1 -0
  634. package/dist/types/review-session.js +2 -0
  635. package/dist/types/review-session.js.map +1 -0
  636. package/dist/types/skill.d.ts +7 -0
  637. package/dist/types/skill.d.ts.map +1 -0
  638. package/dist/types/skill.js +2 -0
  639. package/dist/types/skill.js.map +1 -0
  640. package/dist/utils/comment-formatter.d.ts +19 -0
  641. package/dist/utils/comment-formatter.d.ts.map +1 -0
  642. package/dist/utils/comment-formatter.js +53 -0
  643. package/dist/utils/comment-formatter.js.map +1 -0
  644. package/dist/utils/cost-estimator.d.ts +19 -0
  645. package/dist/utils/cost-estimator.d.ts.map +1 -0
  646. package/dist/utils/cost-estimator.js +32 -0
  647. package/dist/utils/cost-estimator.js.map +1 -0
  648. package/dist/utils/diff-parser.d.ts +7 -0
  649. package/dist/utils/diff-parser.d.ts.map +1 -0
  650. package/dist/utils/diff-parser.js +175 -0
  651. package/dist/utils/diff-parser.js.map +1 -0
  652. package/dist/utils/file-filter.d.ts +16 -0
  653. package/dist/utils/file-filter.d.ts.map +1 -0
  654. package/dist/utils/file-filter.js +64 -0
  655. package/dist/utils/file-filter.js.map +1 -0
  656. package/dist/utils/hash.d.ts +25 -0
  657. package/dist/utils/hash.d.ts.map +1 -0
  658. package/dist/utils/hash.js +45 -0
  659. package/dist/utils/hash.js.map +1 -0
  660. package/dist/utils/logger.d.ts +17 -0
  661. package/dist/utils/logger.d.ts.map +1 -0
  662. package/dist/utils/logger.js +40 -0
  663. package/dist/utils/logger.js.map +1 -0
  664. package/dist/utils/parse-bitbucket-url.d.ts +12 -0
  665. package/dist/utils/parse-bitbucket-url.d.ts.map +1 -0
  666. package/dist/utils/parse-bitbucket-url.js +41 -0
  667. package/dist/utils/parse-bitbucket-url.js.map +1 -0
  668. package/dist/utils/paths.d.ts +6 -0
  669. package/dist/utils/paths.d.ts.map +1 -0
  670. package/dist/utils/paths.js +27 -0
  671. package/dist/utils/paths.js.map +1 -0
  672. package/dist/utils/summary-generator.d.ts +17 -0
  673. package/dist/utils/summary-generator.d.ts.map +1 -0
  674. package/dist/utils/summary-generator.js +76 -0
  675. package/dist/utils/summary-generator.js.map +1 -0
  676. package/oclif.manifest.json +467 -0
  677. package/package.json +100 -0
@@ -0,0 +1,80 @@
1
+ import { readFileSync } from 'node:fs';
2
+ import { fileURLToPath } from 'node:url';
3
+ import path from 'node:path';
4
+ import { Logger, LogLevel } from '../../utils/logger.js';
5
+ import { FINDING_TOOL_SCHEMA, normalizeFinding } from './shared/finding-schema.js';
6
+ import { buildSystemPrompt } from './shared/prompt-builder.js';
7
+ import { buildUserContent } from './shared/content-builder.js';
8
+ import { isValidLineNumber } from './shared/line-validator.js';
9
+ // Load the system prompt at module load time (relative to this source file)
10
+ const __filename = fileURLToPath(import.meta.url);
11
+ const __dir = path.dirname(__filename);
12
+ const BASE_SYSTEM_PROMPT = readFileSync(path.resolve(__dir, 'prompts', 'testing-reviewer.txt'), 'utf-8');
13
+ /**
14
+ * Check if a file path is a test file (spec/test extensions or in test directories).
15
+ */
16
+ function isTestFile(filePath) {
17
+ const lower = filePath.toLowerCase();
18
+ const basename = path.basename(lower);
19
+ // Match test file extensions
20
+ const testExtensions = [
21
+ '.test.ts', '.test.tsx', '.test.js',
22
+ '.spec.ts', '.spec.tsx', '.spec.js',
23
+ ];
24
+ for (const ext of testExtensions) {
25
+ if (basename.endsWith(ext))
26
+ return true;
27
+ }
28
+ // Match files in __tests__ or test directories (anywhere in the path, or at root)
29
+ const normalized = lower.replace(/\\/g, '/');
30
+ return (normalized.includes('/__tests__/') ||
31
+ normalized.includes('/test/') ||
32
+ normalized.startsWith('test/'));
33
+ }
34
+ export const testingReviewerAgent = {
35
+ id: 'testing-reviewer',
36
+ displayName: 'Testing',
37
+ description: 'Test coverage, patterns, and quality',
38
+ domain: 'quality',
39
+ contextTags: ['conventions', 'frameworkRules'],
40
+ fileFilter(filePath) {
41
+ return isTestFile(filePath);
42
+ },
43
+ async analyze(input, provider) {
44
+ const logger = new Logger(LogLevel.warn);
45
+ const relevantFiles = input.files.filter((f) => this.fileFilter(f.newPath));
46
+ if (relevantFiles.length === 0) {
47
+ return {
48
+ agentId: this.id,
49
+ findings: [],
50
+ usage: { inputTokens: 0, outputTokens: 0 },
51
+ };
52
+ }
53
+ const legacyProjectContext = input.projectContext;
54
+ const systemPrompt = buildSystemPrompt(BASE_SYSTEM_PROMPT, input.context, legacyProjectContext);
55
+ const userContent = buildUserContent(relevantFiles, input.prMetadata, input.context?.contentMode ?? 'full');
56
+ const result = await provider.analyze({
57
+ systemPrompt,
58
+ userContent,
59
+ toolName: 'report_findings',
60
+ toolSchema: FINDING_TOOL_SCHEMA,
61
+ });
62
+ const findings = [];
63
+ for (const raw of result.findings) {
64
+ const finding = normalizeFinding(raw, this.id);
65
+ if (finding === null) {
66
+ logger.warn(`[testing-reviewer] Dropping invalid finding — failed schema validation`);
67
+ continue;
68
+ }
69
+ if (isValidLineNumber(finding.lineNumber, finding.filePath, relevantFiles)) {
70
+ findings.push(finding);
71
+ }
72
+ else {
73
+ logger.warn(`[testing-reviewer] Dropping finding at ${finding.filePath}:${finding.lineNumber} — not an addition line`);
74
+ }
75
+ }
76
+ const usage = result.usage;
77
+ return { agentId: this.id, findings, usage };
78
+ },
79
+ };
80
+ //# sourceMappingURL=testing-reviewer.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"testing-reviewer.js","sourceRoot":"","sources":["../../../src/ai/agents/testing-reviewer.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ import type { Agent } from '../../types/index.js';
2
+ export declare const tsReviewerAgent: Agent;
3
+ //# sourceMappingURL=ts-reviewer.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ts-reviewer.d.ts","sourceRoot":"","sources":["../../../src/ai/agents/ts-reviewer.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,KAAK,EAAgE,MAAM,sBAAsB,CAAC;AAchH,eAAO,MAAM,eAAe,EAAE,KAuD7B,CAAC"}
@@ -0,0 +1,63 @@
1
+ import { readFileSync } from 'node:fs';
2
+ import { fileURLToPath } from 'node:url';
3
+ import path from 'node:path';
4
+ import { Logger, LogLevel } from '../../utils/logger.js';
5
+ import { FINDING_TOOL_SCHEMA, normalizeFinding } from './shared/finding-schema.js';
6
+ import { buildSystemPrompt } from './shared/prompt-builder.js';
7
+ import { buildUserContent } from './shared/content-builder.js';
8
+ import { isValidLineNumber } from './shared/line-validator.js';
9
+ const TS_EXTENSIONS = new Set(['.ts', '.tsx', '.js', '.jsx', '.mts', '.cts', '.mjs', '.cjs']);
10
+ // Load the system prompt at module load time (relative to this source file)
11
+ const __filename = fileURLToPath(import.meta.url);
12
+ const __dir = path.dirname(__filename);
13
+ const BASE_SYSTEM_PROMPT = readFileSync(path.resolve(__dir, 'prompts', 'ts-reviewer.txt'), 'utf-8');
14
+ export const tsReviewerAgent = {
15
+ id: 'ts-reviewer',
16
+ displayName: 'TypeScript',
17
+ description: 'TS/JS type safety, patterns, and best practices',
18
+ domain: 'language',
19
+ contextTags: ['conventions', 'frameworkRules'],
20
+ fileFilter(filePath) {
21
+ const ext = path.extname(filePath).toLowerCase();
22
+ return TS_EXTENSIONS.has(ext);
23
+ },
24
+ async analyze(input, provider) {
25
+ const logger = new Logger(LogLevel.warn);
26
+ const relevantFiles = input.files.filter((f) => this.fileFilter(f.newPath));
27
+ if (relevantFiles.length === 0) {
28
+ return {
29
+ agentId: this.id,
30
+ findings: [],
31
+ usage: { inputTokens: 0, outputTokens: 0 },
32
+ };
33
+ }
34
+ // Support both legacy projectContext and new context field
35
+ const legacyProjectContext = input.projectContext;
36
+ const systemPrompt = buildSystemPrompt(BASE_SYSTEM_PROMPT, input.context, legacyProjectContext);
37
+ const userContent = buildUserContent(relevantFiles, input.prMetadata, input.context?.contentMode ?? 'full');
38
+ const result = await provider.analyze({
39
+ systemPrompt,
40
+ userContent,
41
+ toolName: 'report_findings',
42
+ toolSchema: FINDING_TOOL_SCHEMA,
43
+ });
44
+ // Normalize and validate findings using shared schema
45
+ const findings = [];
46
+ for (const raw of result.findings) {
47
+ const finding = normalizeFinding(raw, this.id);
48
+ if (finding === null) {
49
+ logger.warn(`[ts-reviewer] Dropping invalid finding — failed schema validation`);
50
+ continue;
51
+ }
52
+ if (isValidLineNumber(finding.lineNumber, finding.filePath, relevantFiles)) {
53
+ findings.push(finding);
54
+ }
55
+ else {
56
+ logger.warn(`[ts-reviewer] Dropping finding at ${finding.filePath}:${finding.lineNumber} — not an addition line`);
57
+ }
58
+ }
59
+ const usage = result.usage;
60
+ return { agentId: this.id, findings, usage };
61
+ },
62
+ };
63
+ //# sourceMappingURL=ts-reviewer.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ts-reviewer.js","sourceRoot":"","sources":["../../../src/ai/agents/ts-reviewer.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ import type { Agent } from '../../types/index.js';
2
+ export declare const uxTextReviewerAgent: Agent;
3
+ //# sourceMappingURL=ux-text-reviewer.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ux-text-reviewer.d.ts","sourceRoot":"","sources":["../../../src/ai/agents/ux-text-reviewer.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,KAAK,EAAsD,MAAM,sBAAsB,CAAC;AAqEtG,eAAO,MAAM,mBAAmB,EAAE,KAqEjC,CAAC"}
@@ -0,0 +1,116 @@
1
+ import { readFileSync } from 'node:fs';
2
+ import { fileURLToPath } from 'node:url';
3
+ import path from 'node:path';
4
+ import { Logger, LogLevel } from '../../utils/logger.js';
5
+ import { FINDING_TOOL_SCHEMA, normalizeFinding } from './shared/finding-schema.js';
6
+ import { buildSystemPrompt } from './shared/prompt-builder.js';
7
+ import { isValidLineNumber } from './shared/line-validator.js';
8
+ import { extractUserFacingText } from '../text-extractor/index.js';
9
+ const UX_TEXT_EXTENSIONS = new Set([
10
+ '.tsx', '.jsx', '.ts', '.js', '.vue', '.html', '.svelte', '.json',
11
+ ]);
12
+ // Load the system prompt at module load time (relative to this source file)
13
+ const __filename = fileURLToPath(import.meta.url);
14
+ const __dir = path.dirname(__filename);
15
+ const BASE_SYSTEM_PROMPT = readFileSync(path.resolve(__dir, 'prompts', 'ux-text-reviewer.txt'), 'utf-8');
16
+ /**
17
+ * Build compact user content from extracted texts instead of full diff.
18
+ * This is the KEY DIFFERENCE from other agents — achieves ~85-90% token savings.
19
+ *
20
+ * Format:
21
+ * ## Pull Request #42: Fix login screen
22
+ * Branch: feature/login → main
23
+ *
24
+ * ## Extracted User-Facing Text
25
+ *
26
+ * ### src/tui/screens/login.tsx
27
+ * - [L91] (jsx-literal) "Login to Bitbucket"
28
+ */
29
+ function buildCompactContent(texts, prMetadata) {
30
+ const lines = [];
31
+ lines.push(`## Pull Request #${prMetadata.id}: ${prMetadata.title}`);
32
+ lines.push(`Branch: ${prMetadata.sourceBranch} → ${prMetadata.targetBranch}`);
33
+ lines.push('');
34
+ lines.push('## Extracted User-Facing Text');
35
+ lines.push('');
36
+ // Group texts by file for readability
37
+ const byFile = new Map();
38
+ for (const t of texts) {
39
+ const existing = byFile.get(t.filePath);
40
+ if (existing !== undefined) {
41
+ existing.push(t);
42
+ }
43
+ else {
44
+ byFile.set(t.filePath, [t]);
45
+ }
46
+ }
47
+ for (const [filePath, fileTexts] of byFile) {
48
+ lines.push(`### ${filePath}`);
49
+ // Sort by lineNumber ascending within each file
50
+ const sorted = [...fileTexts].sort((a, b) => a.lineNumber - b.lineNumber);
51
+ for (const t of sorted) {
52
+ lines.push(`- [L${t.lineNumber}] (${t.patternType}) "${t.text}"`);
53
+ }
54
+ lines.push('');
55
+ }
56
+ return lines.join('\n');
57
+ }
58
+ export const uxTextReviewerAgent = {
59
+ id: 'ux-text-reviewer',
60
+ displayName: 'UX Text',
61
+ description: 'Grammar, tone, and clarity in UI strings',
62
+ domain: 'quality',
63
+ contextTags: ['conventions'],
64
+ contentMode: 'additions-only',
65
+ fileFilter(filePath) {
66
+ const ext = path.extname(filePath).toLowerCase();
67
+ return UX_TEXT_EXTENSIONS.has(ext);
68
+ },
69
+ async analyze(input, provider) {
70
+ const logger = new Logger(LogLevel.warn);
71
+ const relevantFiles = input.files.filter((f) => this.fileFilter(f.newPath));
72
+ if (relevantFiles.length === 0) {
73
+ return {
74
+ agentId: this.id,
75
+ findings: [],
76
+ usage: { inputTokens: 0, outputTokens: 0 },
77
+ };
78
+ }
79
+ // KEY: Pre-process with text extractor before sending to AI
80
+ const extraction = extractUserFacingText(relevantFiles);
81
+ if (extraction.texts.length === 0) {
82
+ logger.warn(`[ux-text-reviewer] No user-facing text found in ${extraction.fileCount} file(s) — skipping AI call`);
83
+ return {
84
+ agentId: this.id,
85
+ findings: [],
86
+ usage: { inputTokens: 0, outputTokens: 0 },
87
+ };
88
+ }
89
+ const legacyProjectContext = input.projectContext;
90
+ const systemPrompt = buildSystemPrompt(BASE_SYSTEM_PROMPT, input.context, legacyProjectContext);
91
+ const userContent = buildCompactContent(extraction.texts, input.prMetadata);
92
+ const result = await provider.analyze({
93
+ systemPrompt,
94
+ userContent,
95
+ toolName: 'report_findings',
96
+ toolSchema: FINDING_TOOL_SCHEMA,
97
+ });
98
+ const findings = [];
99
+ for (const raw of result.findings) {
100
+ const finding = normalizeFinding(raw, this.id);
101
+ if (finding === null) {
102
+ logger.warn(`[ux-text-reviewer] Dropping invalid finding — failed schema validation`);
103
+ continue;
104
+ }
105
+ if (isValidLineNumber(finding.lineNumber, finding.filePath, relevantFiles)) {
106
+ findings.push(finding);
107
+ }
108
+ else {
109
+ logger.warn(`[ux-text-reviewer] Dropping finding at ${finding.filePath}:${finding.lineNumber} — not an addition line`);
110
+ }
111
+ }
112
+ const usage = result.usage;
113
+ return { agentId: this.id, findings, usage };
114
+ },
115
+ };
116
+ //# sourceMappingURL=ux-text-reviewer.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ux-text-reviewer.js","sourceRoot":"","sources":["../../../src/ai/agents/ux-text-reviewer.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,28 @@
1
+ import type { Agent } from '../../types/agent.js';
2
+ export type AgentTier = 'recommended' | 'available' | 'not-applicable';
3
+ export interface ClassifiedAgent {
4
+ agent: Agent;
5
+ tier: AgentTier;
6
+ reason: string;
7
+ }
8
+ /**
9
+ * Classify each agent as recommended, available, or not-applicable
10
+ * based on the provided file paths.
11
+ *
12
+ * Rules:
13
+ * 1. Threshold agents (architecture-reviewer, performance-reviewer) have special
14
+ * count-based logic that takes precedence over fileFilter:
15
+ * - architecture-reviewer: recommended at >= 5 files, available otherwise
16
+ * - performance-reviewer: recommended at >= 8 files, available otherwise
17
+ * 2. For all other agents, check fileFilter match:
18
+ * - File match (any path passes fileFilter) → recommended, reason lists unique extensions.
19
+ * - No match:
20
+ * - security domain → recommended ("Security review is always recommended")
21
+ * - quality domain → available ("Cross-cutting review — opt in if needed")
22
+ * - infra (non-performance) → not-applicable
23
+ * - language domain → not-applicable
24
+ * 3. Empty filePaths:
25
+ * - No recommended; security → available; quality → available; language → not-applicable
26
+ */
27
+ export declare function classifyAgents(allAgents: readonly Agent[], filePaths: readonly string[]): ClassifiedAgent[];
28
+ //# sourceMappingURL=agent-classifier.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"agent-classifier.d.ts","sourceRoot":"","sources":["../../../src/ai/orchestrator/agent-classifier.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,KAAK,EAAe,MAAM,sBAAsB,CAAC;AAE/D,MAAM,MAAM,SAAS,GAAG,aAAa,GAAG,WAAW,GAAG,gBAAgB,CAAC;AAEvE,MAAM,WAAW,eAAe;IAC9B,KAAK,EAAE,KAAK,CAAC;IACb,IAAI,EAAE,SAAS,CAAC;IAChB,MAAM,EAAE,MAAM,CAAC;CAChB;AAgCD;;;;;;;;;;;;;;;;;;GAkBG;AACH,wBAAgB,cAAc,CAC5B,SAAS,EAAE,SAAS,KAAK,EAAE,EAC3B,SAAS,EAAE,SAAS,MAAM,EAAE,GAC3B,eAAe,EAAE,CAyBnB"}
@@ -0,0 +1,137 @@
1
+ import path from 'node:path';
2
+ /**
3
+ * Safely calls agent.fileFilter for a single path, returning false on any exception.
4
+ */
5
+ function safeFileFilter(agent, filePath) {
6
+ try {
7
+ return agent.fileFilter(filePath);
8
+ }
9
+ catch {
10
+ return false;
11
+ }
12
+ }
13
+ /**
14
+ * Returns the unique file extensions from the matched paths (e.g. ".ts", ".tsx").
15
+ * Only returns extensions found in matchedPaths, sorted alphabetically.
16
+ */
17
+ function matchedExtensions(matchedPaths) {
18
+ const exts = new Set();
19
+ for (const p of matchedPaths) {
20
+ const ext = path.extname(p).toLowerCase();
21
+ if (ext)
22
+ exts.add(ext);
23
+ }
24
+ return [...exts].sort();
25
+ }
26
+ /**
27
+ * IDs of agents that use threshold-based recommendation logic instead of
28
+ * (or in addition to) fileFilter matching.
29
+ */
30
+ const THRESHOLD_AGENTS = new Set(['architecture-reviewer', 'performance-reviewer']);
31
+ /**
32
+ * Classify each agent as recommended, available, or not-applicable
33
+ * based on the provided file paths.
34
+ *
35
+ * Rules:
36
+ * 1. Threshold agents (architecture-reviewer, performance-reviewer) have special
37
+ * count-based logic that takes precedence over fileFilter:
38
+ * - architecture-reviewer: recommended at >= 5 files, available otherwise
39
+ * - performance-reviewer: recommended at >= 8 files, available otherwise
40
+ * 2. For all other agents, check fileFilter match:
41
+ * - File match (any path passes fileFilter) → recommended, reason lists unique extensions.
42
+ * - No match:
43
+ * - security domain → recommended ("Security review is always recommended")
44
+ * - quality domain → available ("Cross-cutting review — opt in if needed")
45
+ * - infra (non-performance) → not-applicable
46
+ * - language domain → not-applicable
47
+ * 3. Empty filePaths:
48
+ * - No recommended; security → available; quality → available; language → not-applicable
49
+ */
50
+ export function classifyAgents(allAgents, filePaths) {
51
+ const paths = Array.from(filePaths);
52
+ const isEmpty = paths.length === 0;
53
+ return allAgents.map((agent) => {
54
+ // Threshold agents bypass fileFilter — count-based logic takes full control
55
+ if (THRESHOLD_AGENTS.has(agent.id)) {
56
+ return applyThresholdRule(agent, paths.length, isEmpty);
57
+ }
58
+ // Check if agent matches any file
59
+ const matchedPaths = isEmpty ? [] : paths.filter((p) => safeFileFilter(agent, p));
60
+ const hasMatch = matchedPaths.length > 0;
61
+ if (hasMatch) {
62
+ const exts = matchedExtensions(matchedPaths);
63
+ const reason = exts.length > 0
64
+ ? `PR contains ${exts.join(', ')} files`
65
+ : 'PR contains matching files';
66
+ return { agent, tier: 'recommended', reason };
67
+ }
68
+ // No file match — apply domain fallback rules
69
+ return applyFallbackRule(agent, paths.length, isEmpty);
70
+ });
71
+ }
72
+ function applyThresholdRule(agent, fileCount, isEmpty) {
73
+ if (agent.id === 'architecture-reviewer') {
74
+ if (!isEmpty && fileCount >= 5) {
75
+ return {
76
+ agent,
77
+ tier: 'recommended',
78
+ reason: `PR touches ${fileCount} files across the codebase`,
79
+ };
80
+ }
81
+ return { agent, tier: 'available', reason: 'Cross-cutting review — opt in if needed' };
82
+ }
83
+ if (agent.id === 'performance-reviewer') {
84
+ if (!isEmpty && fileCount >= 8) {
85
+ return {
86
+ agent,
87
+ tier: 'recommended',
88
+ reason: `Large PR with ${fileCount} files`,
89
+ };
90
+ }
91
+ if (isEmpty) {
92
+ return { agent, tier: 'not-applicable', reason: `No ${agent.displayName} files in this PR` };
93
+ }
94
+ return { agent, tier: 'available', reason: 'Available for additional review' };
95
+ }
96
+ // Fallback for unknown threshold agents — treat as available
97
+ return { agent, tier: 'available', reason: 'Available for additional review' };
98
+ }
99
+ function applyFallbackRule(agent, fileCount, isEmpty) {
100
+ const domain = agent.domain;
101
+ if (domain === 'security') {
102
+ if (isEmpty) {
103
+ return { agent, tier: 'available', reason: 'Available for additional review' };
104
+ }
105
+ return { agent, tier: 'recommended', reason: 'Security review is always recommended' };
106
+ }
107
+ if (domain === 'quality') {
108
+ if (agent.id === 'architecture-reviewer' && !isEmpty && fileCount >= 5) {
109
+ return {
110
+ agent,
111
+ tier: 'recommended',
112
+ reason: `PR touches ${fileCount} files across the codebase`,
113
+ };
114
+ }
115
+ return { agent, tier: 'available', reason: 'Cross-cutting review — opt in if needed' };
116
+ }
117
+ if (domain === 'infra') {
118
+ if (agent.id === 'performance-reviewer') {
119
+ if (!isEmpty && fileCount >= 8) {
120
+ return {
121
+ agent,
122
+ tier: 'recommended',
123
+ reason: `Large PR with ${fileCount} files`,
124
+ };
125
+ }
126
+ if (isEmpty) {
127
+ return { agent, tier: 'not-applicable', reason: `No ${agent.displayName} files in this PR` };
128
+ }
129
+ return { agent, tier: 'available', reason: 'Available for additional review' };
130
+ }
131
+ // config-reviewer and other infra agents without a file match → not-applicable
132
+ return { agent, tier: 'not-applicable', reason: `No ${agent.displayName} files in this PR` };
133
+ }
134
+ // language domain (and any unknown domain)
135
+ return { agent, tier: 'not-applicable', reason: `No ${agent.displayName} files in this PR` };
136
+ }
137
+ //# sourceMappingURL=agent-classifier.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"agent-classifier.js","sourceRoot":"","sources":["../../../src/ai/orchestrator/agent-classifier.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,12 @@
1
+ import type { ReviewFinding } from '../../types/index.js';
2
+ /**
3
+ * Consolidate findings from multiple agents.
4
+ *
5
+ * Groups findings by `filePath + lineNumber + category`. When multiple agents
6
+ * flag the same location and category, keeps the one with the highest severity
7
+ * (lowest rank). All other findings are passed through unchanged.
8
+ *
9
+ * Pure function — does not mutate input.
10
+ */
11
+ export declare function consolidateFindings(findings: ReviewFinding[]): ReviewFinding[];
12
+ //# sourceMappingURL=consolidation.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"consolidation.d.ts","sourceRoot":"","sources":["../../../src/ai/orchestrator/consolidation.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,aAAa,EAAmB,MAAM,sBAAsB,CAAC;AAU3E;;;;;;;;GAQG;AACH,wBAAgB,mBAAmB,CAAC,QAAQ,EAAE,aAAa,EAAE,GAAG,aAAa,EAAE,CAmB9E"}
@@ -0,0 +1,34 @@
1
+ // Severity priority: lower index = higher priority (more severe)
2
+ const SEVERITY_PRIORITY = ['critical', 'high', 'medium', 'low', 'info'];
3
+ function severityRank(severity) {
4
+ const rank = SEVERITY_PRIORITY.indexOf(severity);
5
+ return rank === -1 ? SEVERITY_PRIORITY.length : rank;
6
+ }
7
+ /**
8
+ * Consolidate findings from multiple agents.
9
+ *
10
+ * Groups findings by `filePath + lineNumber + category`. When multiple agents
11
+ * flag the same location and category, keeps the one with the highest severity
12
+ * (lowest rank). All other findings are passed through unchanged.
13
+ *
14
+ * Pure function — does not mutate input.
15
+ */
16
+ export function consolidateFindings(findings) {
17
+ // Key: filePath::lineNumber::category
18
+ const grouped = new Map();
19
+ for (const finding of findings) {
20
+ const key = `${finding.filePath}::${finding.lineNumber}::${finding.category}`;
21
+ const existing = grouped.get(key);
22
+ if (existing === undefined) {
23
+ grouped.set(key, finding);
24
+ }
25
+ else {
26
+ // Keep the finding with the highest severity (lowest rank)
27
+ if (severityRank(finding.severity) < severityRank(existing.severity)) {
28
+ grouped.set(key, finding);
29
+ }
30
+ }
31
+ }
32
+ return Array.from(grouped.values());
33
+ }
34
+ //# sourceMappingURL=consolidation.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"consolidation.js","sourceRoot":"","sources":["../../../src/ai/orchestrator/consolidation.ts"],"names":[],"mappings":"AAEA,iEAAiE;AACjE,MAAM,iBAAiB,GAAsB,CAAC,UAAU,EAAE,MAAM,EAAE,QAAQ,EAAE,KAAK,EAAE,MAAM,CAAC,CAAC;AAE3F,SAAS,YAAY,CAAC,QAAyB;IAC7C,MAAM,IAAI,GAAG,iBAAiB,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC;IACjD,OAAO,IAAI,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,iBAAiB,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC;AACvD,CAAC;AAED;;;;;;;;GAQG;AACH,MAAM,UAAU,mBAAmB,CAAC,QAAyB;IAC3D,sCAAsC;IACtC,MAAM,OAAO,GAAG,IAAI,GAAG,EAAyB,CAAC;IAEjD,KAAK,MAAM,OAAO,IAAI,QAAQ,EAAE,CAAC;QAC/B,MAAM,GAAG,GAAG,GAAG,OAAO,CAAC,QAAQ,KAAK,OAAO,CAAC,UAAU,KAAK,OAAO,CAAC,QAAQ,EAAE,CAAC;QAC9E,MAAM,QAAQ,GAAG,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;QAElC,IAAI,QAAQ,KAAK,SAAS,EAAE,CAAC;YAC3B,OAAO,CAAC,GAAG,CAAC,GAAG,EAAE,OAAO,CAAC,CAAC;QAC5B,CAAC;aAAM,CAAC;YACN,2DAA2D;YAC3D,IAAI,YAAY,CAAC,OAAO,CAAC,QAAQ,CAAC,GAAG,YAAY,CAAC,QAAQ,CAAC,QAAQ,CAAC,EAAE,CAAC;gBACrE,OAAO,CAAC,GAAG,CAAC,GAAG,EAAE,OAAO,CAAC,CAAC;YAC5B,CAAC;QACH,CAAC;IACH,CAAC;IAED,OAAO,KAAK,CAAC,IAAI,CAAC,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC;AACtC,CAAC"}
@@ -0,0 +1,47 @@
1
+ import type { DiffFile } from '../../types/diff.js';
2
+ import type { Agent } from '../../types/agent.js';
3
+ import type { StackProfile } from '../../stack-detector/types.js';
4
+ import type { RepoConfig } from '../../types/repo-config.js';
5
+ export interface DetectedContext {
6
+ languages: Set<string>;
7
+ fileTypes: Map<string, number>;
8
+ hasTests: boolean;
9
+ hasStyles: boolean;
10
+ hasCSharp: boolean;
11
+ hasPhp: boolean;
12
+ hasBlade: boolean;
13
+ hasPython: boolean;
14
+ hasHtml: boolean;
15
+ hasSql: boolean;
16
+ hasConfig: boolean;
17
+ hasUiText: boolean;
18
+ totalFiles: number;
19
+ totalLines: number;
20
+ }
21
+ /**
22
+ * Analyze the diff files to detect what languages, file types, and patterns are present.
23
+ */
24
+ export declare function analyzeContext(files: DiffFile[]): DetectedContext;
25
+ /**
26
+ * Recommend which agents to run based on detected context and available agents.
27
+ * Only returns IDs that exist in allAgents.
28
+ * @public
29
+ */
30
+ export declare function getRecommendedAgents(context: DetectedContext, allAgents: readonly Agent[]): string[];
31
+ /**
32
+ * Map from StackProfile (framework + keyLibraries) to additional agent IDs.
33
+ * Rules are ADDITIVE — they never remove agents, only suggest more.
34
+ * All matching is case-insensitive.
35
+ * Only returns IDs that exist in allAgents.
36
+ */
37
+ export declare function getStackAwareAgents(stackProfile: StackProfile, allAgents: readonly Agent[]): string[];
38
+ /**
39
+ * Resolve the final set of agent IDs using the priority chain:
40
+ * 1. config.agents (explicit override — highest priority)
41
+ * 2. Set union of context-recommended + stack-aware agents
42
+ * 3. Context-recommended agents only (when stackProfile is absent)
43
+ *
44
+ * Used by both WorkflowEngine (cost estimation) and RuleBasedStrategy.
45
+ */
46
+ export declare function resolveAgentIds(context: DetectedContext, allAgents: readonly Agent[], config?: RepoConfig, stackProfile?: StackProfile): string[];
47
+ //# sourceMappingURL=context-analyzer.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"context-analyzer.d.ts","sourceRoot":"","sources":["../../../src/ai/orchestrator/context-analyzer.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,qBAAqB,CAAC;AACpD,OAAO,KAAK,EAAE,KAAK,EAAE,MAAM,sBAAsB,CAAC;AAClD,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,+BAA+B,CAAC;AAClE,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,4BAA4B,CAAC;AAE7D,MAAM,WAAW,eAAe;IAC9B,SAAS,EAAE,GAAG,CAAC,MAAM,CAAC,CAAC;IACvB,SAAS,EAAE,GAAG,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;IAC/B,QAAQ,EAAE,OAAO,CAAC;IAClB,SAAS,EAAE,OAAO,CAAC;IACnB,SAAS,EAAE,OAAO,CAAC;IACnB,MAAM,EAAE,OAAO,CAAC;IAChB,QAAQ,EAAE,OAAO,CAAC;IAClB,SAAS,EAAE,OAAO,CAAC;IACnB,OAAO,EAAE,OAAO,CAAC;IACjB,MAAM,EAAE,OAAO,CAAC;IAChB,SAAS,EAAE,OAAO,CAAC;IACnB,SAAS,EAAE,OAAO,CAAC;IACnB,UAAU,EAAE,MAAM,CAAC;IACnB,UAAU,EAAE,MAAM,CAAC;CACpB;AAqJD;;GAEG;AACH,wBAAgB,cAAc,CAAC,KAAK,EAAE,QAAQ,EAAE,GAAG,eAAe,CAmEjE;AAED;;;;GAIG;AACH,wBAAgB,oBAAoB,CAAC,OAAO,EAAE,eAAe,EAAE,SAAS,EAAE,SAAS,KAAK,EAAE,GAAG,MAAM,EAAE,CA6EpG;AA0CD;;;;;GAKG;AACH,wBAAgB,mBAAmB,CACjC,YAAY,EAAE,YAAY,EAC1B,SAAS,EAAE,SAAS,KAAK,EAAE,GAC1B,MAAM,EAAE,CA2BV;AAED;;;;;;;GAOG;AACH,wBAAgB,eAAe,CAC7B,OAAO,EAAE,eAAe,EACxB,SAAS,EAAE,SAAS,KAAK,EAAE,EAC3B,MAAM,CAAC,EAAE,UAAU,EACnB,YAAY,CAAC,EAAE,YAAY,GAC1B,MAAM,EAAE,CAaV"}