@urateam/core 0.1.16 → 0.1.19

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 (293) hide show
  1. package/dist/__tests__/agentic-deep-review-provider.test.d.ts +2 -0
  2. package/dist/__tests__/agentic-deep-review-provider.test.d.ts.map +1 -0
  3. package/dist/__tests__/agentic-deep-review-provider.test.js +70 -0
  4. package/dist/__tests__/agentic-deep-review-provider.test.js.map +1 -0
  5. package/dist/__tests__/audit-immutability.test.js +16 -1
  6. package/dist/__tests__/audit-immutability.test.js.map +1 -1
  7. package/dist/__tests__/cost/per-run-multi-model.test.d.ts +2 -0
  8. package/dist/__tests__/cost/per-run-multi-model.test.d.ts.map +1 -0
  9. package/dist/__tests__/cost/per-run-multi-model.test.js +102 -0
  10. package/dist/__tests__/cost/per-run-multi-model.test.js.map +1 -0
  11. package/dist/__tests__/db-qa-gap-issues.test.d.ts +2 -0
  12. package/dist/__tests__/db-qa-gap-issues.test.d.ts.map +1 -0
  13. package/dist/__tests__/db-qa-gap-issues.test.js +113 -0
  14. package/dist/__tests__/db-qa-gap-issues.test.js.map +1 -0
  15. package/dist/__tests__/db-release-decisions.test.d.ts +2 -0
  16. package/dist/__tests__/db-release-decisions.test.d.ts.map +1 -0
  17. package/dist/__tests__/db-release-decisions.test.js +98 -0
  18. package/dist/__tests__/db-release-decisions.test.js.map +1 -0
  19. package/dist/__tests__/db-review-model-runs.test.d.ts +2 -0
  20. package/dist/__tests__/db-review-model-runs.test.d.ts.map +1 -0
  21. package/dist/__tests__/db-review-model-runs.test.js +90 -0
  22. package/dist/__tests__/db-review-model-runs.test.js.map +1 -0
  23. package/dist/__tests__/e2e-fanout.test.d.ts +17 -0
  24. package/dist/__tests__/e2e-fanout.test.d.ts.map +1 -0
  25. package/dist/__tests__/e2e-fanout.test.js +184 -0
  26. package/dist/__tests__/e2e-fanout.test.js.map +1 -0
  27. package/dist/__tests__/e2e-pipeline.test.js +1 -0
  28. package/dist/__tests__/e2e-pipeline.test.js.map +1 -1
  29. package/dist/__tests__/notifier-discord.test.js +3 -3
  30. package/dist/__tests__/notifier-discord.test.js.map +1 -1
  31. package/dist/__tests__/notifier-slack.test.js +2 -2
  32. package/dist/__tests__/notifier-slack.test.js.map +1 -1
  33. package/dist/__tests__/notifier.test.js +1 -0
  34. package/dist/__tests__/notifier.test.js.map +1 -1
  35. package/dist/__tests__/openrouter-client.test.d.ts +2 -0
  36. package/dist/__tests__/openrouter-client.test.d.ts.map +1 -0
  37. package/dist/__tests__/openrouter-client.test.js +56 -0
  38. package/dist/__tests__/openrouter-client.test.js.map +1 -0
  39. package/dist/__tests__/openrouter-fanout.test.d.ts +2 -0
  40. package/dist/__tests__/openrouter-fanout.test.d.ts.map +1 -0
  41. package/dist/__tests__/openrouter-fanout.test.js +93 -0
  42. package/dist/__tests__/openrouter-fanout.test.js.map +1 -0
  43. package/dist/__tests__/pm-budget.test.js +1 -0
  44. package/dist/__tests__/pm-budget.test.js.map +1 -1
  45. package/dist/__tests__/pm-promote.test.js +189 -0
  46. package/dist/__tests__/pm-promote.test.js.map +1 -1
  47. package/dist/__tests__/pm-scheduler.test.js +1 -0
  48. package/dist/__tests__/pm-scheduler.test.js.map +1 -1
  49. package/dist/__tests__/pm-types.test.js +2 -0
  50. package/dist/__tests__/pm-types.test.js.map +1 -1
  51. package/dist/__tests__/post-fanout-comments.test.d.ts +2 -0
  52. package/dist/__tests__/post-fanout-comments.test.d.ts.map +1 -0
  53. package/dist/__tests__/post-fanout-comments.test.js +84 -0
  54. package/dist/__tests__/post-fanout-comments.test.js.map +1 -0
  55. package/dist/__tests__/qa-audit-events.test.d.ts +2 -0
  56. package/dist/__tests__/qa-audit-events.test.d.ts.map +1 -0
  57. package/dist/__tests__/qa-audit-events.test.js +57 -0
  58. package/dist/__tests__/qa-audit-events.test.js.map +1 -0
  59. package/dist/__tests__/qa-config.test.d.ts +2 -0
  60. package/dist/__tests__/qa-config.test.d.ts.map +1 -0
  61. package/dist/__tests__/qa-config.test.js +80 -0
  62. package/dist/__tests__/qa-config.test.js.map +1 -0
  63. package/dist/__tests__/qa-eval.test.d.ts +2 -0
  64. package/dist/__tests__/qa-eval.test.d.ts.map +1 -0
  65. package/dist/__tests__/qa-eval.test.js +146 -0
  66. package/dist/__tests__/qa-eval.test.js.map +1 -0
  67. package/dist/__tests__/qa-gap.test.d.ts +2 -0
  68. package/dist/__tests__/qa-gap.test.d.ts.map +1 -0
  69. package/dist/__tests__/qa-gap.test.js +100 -0
  70. package/dist/__tests__/qa-gap.test.js.map +1 -0
  71. package/dist/__tests__/qa-github.test.d.ts +2 -0
  72. package/dist/__tests__/qa-github.test.d.ts.map +1 -0
  73. package/dist/__tests__/qa-github.test.js +162 -0
  74. package/dist/__tests__/qa-github.test.js.map +1 -0
  75. package/dist/__tests__/ralph-review-fix-regression.test.js +1 -0
  76. package/dist/__tests__/ralph-review-fix-regression.test.js.map +1 -1
  77. package/dist/__tests__/release-manager-audit-events.test.d.ts +2 -0
  78. package/dist/__tests__/release-manager-audit-events.test.d.ts.map +1 -0
  79. package/dist/__tests__/release-manager-audit-events.test.js +64 -0
  80. package/dist/__tests__/release-manager-audit-events.test.js.map +1 -0
  81. package/dist/__tests__/release-manager-config.test.d.ts +2 -0
  82. package/dist/__tests__/release-manager-config.test.d.ts.map +1 -0
  83. package/dist/__tests__/release-manager-config.test.js +56 -0
  84. package/dist/__tests__/release-manager-config.test.js.map +1 -0
  85. package/dist/__tests__/release-manager-decide.test.d.ts +2 -0
  86. package/dist/__tests__/release-manager-decide.test.d.ts.map +1 -0
  87. package/dist/__tests__/release-manager-decide.test.js +102 -0
  88. package/dist/__tests__/release-manager-decide.test.js.map +1 -0
  89. package/dist/__tests__/release-manager-github.test.d.ts +2 -0
  90. package/dist/__tests__/release-manager-github.test.d.ts.map +1 -0
  91. package/dist/__tests__/release-manager-github.test.js +83 -0
  92. package/dist/__tests__/release-manager-github.test.js.map +1 -0
  93. package/dist/__tests__/release-manager-license-gate.test.d.ts +2 -0
  94. package/dist/__tests__/release-manager-license-gate.test.d.ts.map +1 -0
  95. package/dist/__tests__/release-manager-license-gate.test.js +25 -0
  96. package/dist/__tests__/release-manager-license-gate.test.js.map +1 -0
  97. package/dist/__tests__/release-manager-scheduler.test.d.ts +2 -0
  98. package/dist/__tests__/release-manager-scheduler.test.d.ts.map +1 -0
  99. package/dist/__tests__/release-manager-scheduler.test.js +395 -0
  100. package/dist/__tests__/release-manager-scheduler.test.js.map +1 -0
  101. package/dist/__tests__/release-manager-slack-handler.test.d.ts +2 -0
  102. package/dist/__tests__/release-manager-slack-handler.test.d.ts.map +1 -0
  103. package/dist/__tests__/release-manager-slack-handler.test.js +175 -0
  104. package/dist/__tests__/release-manager-slack-handler.test.js.map +1 -0
  105. package/dist/__tests__/release-manager-triggers.test.d.ts +2 -0
  106. package/dist/__tests__/release-manager-triggers.test.d.ts.map +1 -0
  107. package/dist/__tests__/release-manager-triggers.test.js +67 -0
  108. package/dist/__tests__/release-manager-triggers.test.js.map +1 -0
  109. package/dist/__tests__/release-manager-versioning.test.d.ts +2 -0
  110. package/dist/__tests__/release-manager-versioning.test.d.ts.map +1 -0
  111. package/dist/__tests__/release-manager-versioning.test.js +51 -0
  112. package/dist/__tests__/release-manager-versioning.test.js.map +1 -0
  113. package/dist/__tests__/reproduce-bec113-pagination-warning.test.js +1 -0
  114. package/dist/__tests__/reproduce-bec113-pagination-warning.test.js.map +1 -1
  115. package/dist/__tests__/reproduce-bec62.test.js +1 -0
  116. package/dist/__tests__/reproduce-bec62.test.js.map +1 -1
  117. package/dist/__tests__/reproduce-bec91-stuck-in-progress.test.js +2 -0
  118. package/dist/__tests__/reproduce-bec91-stuck-in-progress.test.js.map +1 -1
  119. package/dist/__tests__/review-prompt.test.d.ts +2 -0
  120. package/dist/__tests__/review-prompt.test.d.ts.map +1 -0
  121. package/dist/__tests__/review-prompt.test.js +88 -0
  122. package/dist/__tests__/review-prompt.test.js.map +1 -0
  123. package/dist/__tests__/review-provider-registry.test.d.ts +2 -0
  124. package/dist/__tests__/review-provider-registry.test.d.ts.map +1 -0
  125. package/dist/__tests__/review-provider-registry.test.js +61 -0
  126. package/dist/__tests__/review-provider-registry.test.js.map +1 -0
  127. package/dist/__tests__/runner-fanout-integration.test.d.ts +2 -0
  128. package/dist/__tests__/runner-fanout-integration.test.d.ts.map +1 -0
  129. package/dist/__tests__/runner-fanout-integration.test.js +107 -0
  130. package/dist/__tests__/runner-fanout-integration.test.js.map +1 -0
  131. package/dist/__tests__/webhook-handler.test.js +1 -0
  132. package/dist/__tests__/webhook-handler.test.js.map +1 -1
  133. package/dist/audit/events.d.ts +54 -0
  134. package/dist/audit/events.d.ts.map +1 -1
  135. package/dist/audit/events.js +100 -0
  136. package/dist/audit/events.js.map +1 -1
  137. package/dist/audit/writer.d.ts +22 -8
  138. package/dist/audit/writer.d.ts.map +1 -1
  139. package/dist/audit/writer.js +22 -8
  140. package/dist/audit/writer.js.map +1 -1
  141. package/dist/cost/aggregate.d.ts.map +1 -1
  142. package/dist/cost/aggregate.js +14 -2
  143. package/dist/cost/aggregate.js.map +1 -1
  144. package/dist/cost/csv.d.ts.map +1 -1
  145. package/dist/cost/csv.js +14 -2
  146. package/dist/cost/csv.js.map +1 -1
  147. package/dist/cost/per-run.d.ts +11 -0
  148. package/dist/cost/per-run.d.ts.map +1 -1
  149. package/dist/cost/per-run.js +21 -8
  150. package/dist/cost/per-run.js.map +1 -1
  151. package/dist/cost/rollup.d.ts.map +1 -1
  152. package/dist/cost/rollup.js +14 -2
  153. package/dist/cost/rollup.js.map +1 -1
  154. package/dist/db/client.d.ts.map +1 -1
  155. package/dist/db/client.js +20 -1
  156. package/dist/db/client.js.map +1 -1
  157. package/dist/db/migrations/postgres/009_review_model_runs.sql +18 -0
  158. package/dist/db/migrations/postgres/010_release_manager.sql +38 -0
  159. package/dist/db/migrations/postgres/011_qa_run_columns.sql +8 -0
  160. package/dist/db/migrations/postgres/012_qa_gap_issues.sql +18 -0
  161. package/dist/db/migrations/sqlite/008_review_model_runs.sql +18 -0
  162. package/dist/db/migrations/sqlite/009_release_manager.sql +43 -0
  163. package/dist/db/migrations/sqlite/010_qa_run_columns.sql +9 -0
  164. package/dist/db/migrations/sqlite/011_qa_gap_issues.sql +22 -0
  165. package/dist/db/review-model-runs.d.ts +4 -0
  166. package/dist/db/review-model-runs.d.ts.map +1 -0
  167. package/dist/db/review-model-runs.js +25 -0
  168. package/dist/db/review-model-runs.js.map +1 -0
  169. package/dist/db/schema.d.ts +761 -0
  170. package/dist/db/schema.d.ts.map +1 -1
  171. package/dist/db/schema.js +67 -0
  172. package/dist/db/schema.js.map +1 -1
  173. package/dist/executor/executor.d.ts.map +1 -1
  174. package/dist/executor/executor.js +2 -0
  175. package/dist/executor/executor.js.map +1 -1
  176. package/dist/executor/review/agentic-deep-review.d.ts +6 -0
  177. package/dist/executor/review/agentic-deep-review.d.ts.map +1 -0
  178. package/dist/executor/review/agentic-deep-review.js +37 -0
  179. package/dist/executor/review/agentic-deep-review.js.map +1 -0
  180. package/dist/executor/review/openrouter-client.d.ts +23 -0
  181. package/dist/executor/review/openrouter-client.d.ts.map +1 -0
  182. package/dist/executor/review/openrouter-client.js +34 -0
  183. package/dist/executor/review/openrouter-client.js.map +1 -0
  184. package/dist/executor/review/openrouter-fanout.d.ts +17 -0
  185. package/dist/executor/review/openrouter-fanout.d.ts.map +1 -0
  186. package/dist/executor/review/openrouter-fanout.js +84 -0
  187. package/dist/executor/review/openrouter-fanout.js.map +1 -0
  188. package/dist/executor/review/post-fanout-comments.d.ts +4 -0
  189. package/dist/executor/review/post-fanout-comments.d.ts.map +1 -0
  190. package/dist/executor/review/post-fanout-comments.js +45 -0
  191. package/dist/executor/review/post-fanout-comments.js.map +1 -0
  192. package/dist/executor/review/review-prompt.d.ts +21 -0
  193. package/dist/executor/review/review-prompt.d.ts.map +1 -0
  194. package/dist/executor/review/review-prompt.js +130 -0
  195. package/dist/executor/review/review-prompt.js.map +1 -0
  196. package/dist/executor/review/review-provider.d.ts +27 -0
  197. package/dist/executor/review/review-provider.d.ts.map +1 -0
  198. package/dist/executor/review/review-provider.js +36 -0
  199. package/dist/executor/review/review-provider.js.map +1 -0
  200. package/dist/executor/review/workdir-snapshot.d.ts +13 -0
  201. package/dist/executor/review/workdir-snapshot.d.ts.map +1 -0
  202. package/dist/executor/review/workdir-snapshot.js +24 -0
  203. package/dist/executor/review/workdir-snapshot.js.map +1 -0
  204. package/dist/index.d.ts +4 -0
  205. package/dist/index.d.ts.map +1 -1
  206. package/dist/index.js +4 -0
  207. package/dist/index.js.map +1 -1
  208. package/dist/license.d.ts.map +1 -1
  209. package/dist/license.js +1 -0
  210. package/dist/license.js.map +1 -1
  211. package/dist/pipeline/review-providers-runner.d.ts +31 -0
  212. package/dist/pipeline/review-providers-runner.d.ts.map +1 -0
  213. package/dist/pipeline/review-providers-runner.js +64 -0
  214. package/dist/pipeline/review-providers-runner.js.map +1 -0
  215. package/dist/pipeline/runner.d.ts.map +1 -1
  216. package/dist/pipeline/runner.js +99 -7
  217. package/dist/pipeline/runner.js.map +1 -1
  218. package/dist/pm/actions/promote.d.ts +12 -0
  219. package/dist/pm/actions/promote.d.ts.map +1 -1
  220. package/dist/pm/actions/promote.js +27 -2
  221. package/dist/pm/actions/promote.js.map +1 -1
  222. package/dist/pm/actions/resolve-approvals.js +3 -3
  223. package/dist/pm/actions/resolve-approvals.js.map +1 -1
  224. package/dist/pm/actions/triage.js +2 -2
  225. package/dist/pm/actions/triage.js.map +1 -1
  226. package/dist/pm/scheduler.d.ts.map +1 -1
  227. package/dist/pm/scheduler.js +5 -3
  228. package/dist/pm/scheduler.js.map +1 -1
  229. package/dist/pm/slack-interface.d.ts +8 -0
  230. package/dist/pm/slack-interface.d.ts.map +1 -1
  231. package/dist/pm/slack-interface.js +20 -3
  232. package/dist/pm/slack-interface.js.map +1 -1
  233. package/dist/pm/types.d.ts +1 -0
  234. package/dist/pm/types.d.ts.map +1 -1
  235. package/dist/pm/types.js +6 -0
  236. package/dist/pm/types.js.map +1 -1
  237. package/dist/qa/gap.d.ts +29 -0
  238. package/dist/qa/gap.d.ts.map +1 -0
  239. package/dist/qa/gap.js +91 -0
  240. package/dist/qa/gap.js.map +1 -0
  241. package/dist/qa/github.d.ts +77 -0
  242. package/dist/qa/github.d.ts.map +1 -0
  243. package/dist/qa/github.js +106 -0
  244. package/dist/qa/github.js.map +1 -0
  245. package/dist/qa/index.d.ts +4 -0
  246. package/dist/qa/index.d.ts.map +1 -0
  247. package/dist/qa/index.js +4 -0
  248. package/dist/qa/index.js.map +1 -0
  249. package/dist/qa/types.d.ts +44 -0
  250. package/dist/qa/types.d.ts.map +1 -0
  251. package/dist/qa/types.js +12 -0
  252. package/dist/qa/types.js.map +1 -0
  253. package/dist/release-manager/decide.d.ts +26 -0
  254. package/dist/release-manager/decide.d.ts.map +1 -0
  255. package/dist/release-manager/decide.js +58 -0
  256. package/dist/release-manager/decide.js.map +1 -0
  257. package/dist/release-manager/github.d.ts +47 -0
  258. package/dist/release-manager/github.d.ts.map +1 -0
  259. package/dist/release-manager/github.js +66 -0
  260. package/dist/release-manager/github.js.map +1 -0
  261. package/dist/release-manager/index.d.ts +9 -0
  262. package/dist/release-manager/index.d.ts.map +1 -0
  263. package/dist/release-manager/index.js +9 -0
  264. package/dist/release-manager/index.js.map +1 -0
  265. package/dist/release-manager/scheduler.d.ts +30 -0
  266. package/dist/release-manager/scheduler.d.ts.map +1 -0
  267. package/dist/release-manager/scheduler.js +427 -0
  268. package/dist/release-manager/scheduler.js.map +1 -0
  269. package/dist/release-manager/slack-handler.d.ts +39 -0
  270. package/dist/release-manager/slack-handler.d.ts.map +1 -0
  271. package/dist/release-manager/slack-handler.js +124 -0
  272. package/dist/release-manager/slack-handler.js.map +1 -0
  273. package/dist/release-manager/state.d.ts +18 -0
  274. package/dist/release-manager/state.d.ts.map +1 -0
  275. package/dist/release-manager/state.js +154 -0
  276. package/dist/release-manager/state.js.map +1 -0
  277. package/dist/release-manager/triggers.d.ts +32 -0
  278. package/dist/release-manager/triggers.d.ts.map +1 -0
  279. package/dist/release-manager/triggers.js +82 -0
  280. package/dist/release-manager/triggers.js.map +1 -0
  281. package/dist/release-manager/types.d.ts +79 -0
  282. package/dist/release-manager/types.d.ts.map +1 -0
  283. package/dist/release-manager/types.js +48 -0
  284. package/dist/release-manager/types.js.map +1 -0
  285. package/dist/release-manager/versioning.d.ts +20 -0
  286. package/dist/release-manager/versioning.d.ts.map +1 -0
  287. package/dist/release-manager/versioning.js +60 -0
  288. package/dist/release-manager/versioning.js.map +1 -0
  289. package/dist/types.d.ts +48 -0
  290. package/dist/types.d.ts.map +1 -1
  291. package/dist/types.js +8 -0
  292. package/dist/types.js.map +1 -1
  293. package/package.json +3 -1
@@ -0,0 +1,57 @@
1
+ import { describe, it, expect } from "vitest";
2
+ import { qaRunTriggeredEvent, qaRunCompletedEvent, qaGapIssueFiledEvent, } from "../audit/events.js";
3
+ import { AuditEventSchema } from "../types.js";
4
+ describe("qa audit events", () => {
5
+ it("qaRunTriggeredEvent passes schema validation", () => {
6
+ const evt = qaRunTriggeredEvent({
7
+ repoUrl: "https://github.com/org/repo",
8
+ branch: "main",
9
+ workflow: ".github/workflows/smoke.yml",
10
+ runId: 12345,
11
+ sha: "abcdef0",
12
+ });
13
+ expect(evt.eventType).toBe("qa.run_triggered");
14
+ expect(evt.actor).toBe("release-manager");
15
+ expect(evt.actorType).toBe("release-manager");
16
+ expect(evt.scope).toBe("repo:https://github.com/org/repo");
17
+ expect(() => AuditEventSchema.parse(evt)).not.toThrow();
18
+ expect(evt.payload.runId).toBe(12345);
19
+ });
20
+ it("qaRunCompletedEvent passes schema validation with conclusion", () => {
21
+ const evt = qaRunCompletedEvent({
22
+ repoUrl: "https://github.com/org/repo",
23
+ branch: "main",
24
+ runId: 12345,
25
+ conclusion: "success",
26
+ durationMs: 600_000,
27
+ });
28
+ expect(evt.eventType).toBe("qa.run_completed");
29
+ expect(() => AuditEventSchema.parse(evt)).not.toThrow();
30
+ expect(evt.payload.conclusion).toBe("success");
31
+ expect(evt.payload.durationMs).toBe(600_000);
32
+ });
33
+ it("qaRunCompletedEvent supports synthetic timeout flag", () => {
34
+ const evt = qaRunCompletedEvent({
35
+ repoUrl: "https://github.com/org/repo",
36
+ branch: "main",
37
+ runId: 12345,
38
+ conclusion: "timed_out",
39
+ durationMs: 1_800_000,
40
+ synthetic: true,
41
+ });
42
+ expect(() => AuditEventSchema.parse(evt)).not.toThrow();
43
+ expect(evt.payload.synthetic).toBe(true);
44
+ });
45
+ it("qaGapIssueFiledEvent passes schema validation", () => {
46
+ const evt = qaGapIssueFiledEvent({
47
+ repoUrl: "https://github.com/org/repo",
48
+ branch: "main",
49
+ workflowPath: ".github/workflows/smoke.yml",
50
+ linearIssueId: "BEC-150",
51
+ });
52
+ expect(evt.eventType).toBe("qa.gap_issue_filed");
53
+ expect(() => AuditEventSchema.parse(evt)).not.toThrow();
54
+ expect(evt.payload.linearIssueId).toBe("BEC-150");
55
+ });
56
+ });
57
+ //# sourceMappingURL=qa-audit-events.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"qa-audit-events.test.js","sourceRoot":"","sources":["../../src/__tests__/qa-audit-events.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=qa-config.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"qa-config.test.d.ts","sourceRoot":"","sources":["../../src/__tests__/qa-config.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,80 @@
1
+ import { describe, it, expect } from "vitest";
2
+ import { QaCheckConfigSchema } from "../qa/types.js";
3
+ import { ReleaseManagerConfigSchema } from "../release-manager/types.js";
4
+ describe("QaCheckConfigSchema", () => {
5
+ it("parses minimal valid config", () => {
6
+ const cfg = QaCheckConfigSchema.parse({
7
+ workflow: ".github/workflows/smoke.yml",
8
+ linearTeamId: "team-uuid-123",
9
+ });
10
+ expect(cfg.workflow).toBe(".github/workflows/smoke.yml");
11
+ expect(cfg.timeoutMinutes).toBe(30); // default
12
+ expect(cfg.linearTeamId).toBe("team-uuid-123");
13
+ expect(cfg.workflowInputs).toBeUndefined();
14
+ });
15
+ it("respects custom timeoutMinutes", () => {
16
+ const cfg = QaCheckConfigSchema.parse({
17
+ workflow: ".github/workflows/smoke.yml",
18
+ linearTeamId: "team-uuid-123",
19
+ timeoutMinutes: 60,
20
+ });
21
+ expect(cfg.timeoutMinutes).toBe(60);
22
+ });
23
+ it("accepts workflowInputs object", () => {
24
+ const cfg = QaCheckConfigSchema.parse({
25
+ workflow: ".github/workflows/smoke.yml",
26
+ linearTeamId: "team-uuid-123",
27
+ workflowInputs: { environment: "preview" },
28
+ });
29
+ expect(cfg.workflowInputs).toEqual({ environment: "preview" });
30
+ });
31
+ it("rejects empty workflow string", () => {
32
+ expect(() => QaCheckConfigSchema.parse({
33
+ workflow: "",
34
+ linearTeamId: "team-uuid-123",
35
+ })).toThrow();
36
+ });
37
+ it("rejects empty linearTeamId", () => {
38
+ expect(() => QaCheckConfigSchema.parse({
39
+ workflow: ".github/workflows/smoke.yml",
40
+ linearTeamId: "",
41
+ })).toThrow();
42
+ });
43
+ it("rejects non-positive timeoutMinutes", () => {
44
+ expect(() => QaCheckConfigSchema.parse({
45
+ workflow: ".github/workflows/smoke.yml",
46
+ linearTeamId: "team-uuid-123",
47
+ timeoutMinutes: 0,
48
+ })).toThrow();
49
+ });
50
+ });
51
+ describe("ReleaseManagerConfigSchema with qaCheck", () => {
52
+ it("accepts a qaCheck trigger alongside existing triggers", () => {
53
+ const cfg = ReleaseManagerConfigSchema.parse({
54
+ enabled: true,
55
+ triggers: {
56
+ mergedPRsSince: 5,
57
+ qaCheck: {
58
+ workflow: ".github/workflows/smoke.yml",
59
+ linearTeamId: "team-uuid-123",
60
+ },
61
+ },
62
+ });
63
+ expect(cfg.triggers.qaCheck?.workflow).toBe(".github/workflows/smoke.yml");
64
+ expect(cfg.triggers.qaCheck?.timeoutMinutes).toBe(30);
65
+ });
66
+ it("treats qaCheck as a valid trigger to satisfy 'at least one trigger' guard", () => {
67
+ // qaCheck alone (no mergedPRsSince/timeSinceLastHours/ciGreenForMinutes/requireSlackApproval) should pass
68
+ const cfg = ReleaseManagerConfigSchema.parse({
69
+ enabled: true,
70
+ triggers: {
71
+ qaCheck: {
72
+ workflow: ".github/workflows/smoke.yml",
73
+ linearTeamId: "team-uuid-123",
74
+ },
75
+ },
76
+ });
77
+ expect(cfg.triggers.qaCheck).toBeDefined();
78
+ });
79
+ });
80
+ //# sourceMappingURL=qa-config.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"qa-config.test.js","sourceRoot":"","sources":["../../src/__tests__/qa-config.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=qa-eval.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"qa-eval.test.d.ts","sourceRoot":"","sources":["../../src/__tests__/qa-eval.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,146 @@
1
+ import { describe, it, expect } from "vitest";
2
+ import { evalQaCheck } from "../release-manager/triggers.js";
3
+ const NOW = new Date("2026-05-04T12:00:00Z");
4
+ const cfg = {
5
+ workflow: ".github/workflows/smoke.yml",
6
+ timeoutMinutes: 30,
7
+ linearTeamId: "team-uuid-123",
8
+ };
9
+ describe("evalQaCheck", () => {
10
+ it("returns qa_no_workflow when workflowFileExists=false", () => {
11
+ const result = evalQaCheck({
12
+ qaConfig: cfg,
13
+ headSha: "head_sha_1",
14
+ workflowFileExists: false,
15
+ qaRun: null,
16
+ runConclusion: null,
17
+ now: NOW,
18
+ });
19
+ expect(result.pass).toBe(false);
20
+ expect(result.reason).toBe("qa_no_workflow");
21
+ });
22
+ it("returns qa_needs_trigger when workflow exists but no run for current SHA", () => {
23
+ const result = evalQaCheck({
24
+ qaConfig: cfg,
25
+ headSha: "head_sha_1",
26
+ workflowFileExists: true,
27
+ qaRun: null,
28
+ runConclusion: null,
29
+ now: NOW,
30
+ });
31
+ expect(result.pass).toBe(false);
32
+ expect(result.reason).toBe("qa_needs_trigger");
33
+ });
34
+ it("returns qa_needs_trigger when in-flight run is for stale SHA", () => {
35
+ const qaRun = {
36
+ runId: 99999,
37
+ runSha: "old_sha",
38
+ triggeredAt: new Date(NOW.getTime() - 5 * 60 * 1000),
39
+ };
40
+ const result = evalQaCheck({
41
+ qaConfig: cfg,
42
+ headSha: "head_sha_1",
43
+ workflowFileExists: true,
44
+ qaRun,
45
+ runConclusion: null,
46
+ now: NOW,
47
+ });
48
+ expect(result.pass).toBe(false);
49
+ expect(result.reason).toBe("qa_needs_trigger");
50
+ });
51
+ it("returns qa_running when run is in flight for current SHA, not yet timed out", () => {
52
+ const qaRun = {
53
+ runId: 99999,
54
+ runSha: "head_sha_1",
55
+ triggeredAt: new Date(NOW.getTime() - 5 * 60 * 1000), // 5 min ago, < 30 min timeout
56
+ };
57
+ const result = evalQaCheck({
58
+ qaConfig: cfg,
59
+ headSha: "head_sha_1",
60
+ workflowFileExists: true,
61
+ qaRun,
62
+ runConclusion: null,
63
+ now: NOW,
64
+ });
65
+ expect(result.pass).toBe(false);
66
+ expect(result.reason).toBe("qa_running");
67
+ if (result.pass === false && result.reason === "qa_running") {
68
+ expect(result.runId).toBe(99999);
69
+ }
70
+ });
71
+ it("returns qa_timed_out when run has been running > timeoutMinutes", () => {
72
+ const qaRun = {
73
+ runId: 99999,
74
+ runSha: "head_sha_1",
75
+ triggeredAt: new Date(NOW.getTime() - 35 * 60 * 1000), // 35 min ago, > 30 min timeout
76
+ };
77
+ const result = evalQaCheck({
78
+ qaConfig: cfg,
79
+ headSha: "head_sha_1",
80
+ workflowFileExists: true,
81
+ qaRun,
82
+ runConclusion: null,
83
+ now: NOW,
84
+ });
85
+ expect(result.pass).toBe(false);
86
+ expect(result.reason).toBe("qa_timed_out");
87
+ });
88
+ it("returns pass:true when run completed with conclusion=success", () => {
89
+ const qaRun = {
90
+ runId: 99999,
91
+ runSha: "head_sha_1",
92
+ triggeredAt: new Date(NOW.getTime() - 10 * 60 * 1000),
93
+ };
94
+ const result = evalQaCheck({
95
+ qaConfig: cfg,
96
+ headSha: "head_sha_1",
97
+ workflowFileExists: true,
98
+ qaRun,
99
+ runConclusion: "success",
100
+ now: NOW,
101
+ });
102
+ expect(result.pass).toBe(true);
103
+ expect(result.reason).toMatch(/qa passed/i);
104
+ });
105
+ it("returns qa_failed for conclusion=failure", () => {
106
+ const qaRun = {
107
+ runId: 99999,
108
+ runSha: "head_sha_1",
109
+ triggeredAt: new Date(NOW.getTime() - 10 * 60 * 1000),
110
+ };
111
+ const result = evalQaCheck({
112
+ qaConfig: cfg,
113
+ headSha: "head_sha_1",
114
+ workflowFileExists: true,
115
+ qaRun,
116
+ runConclusion: "failure",
117
+ now: NOW,
118
+ });
119
+ expect(result.pass).toBe(false);
120
+ expect(result.reason).toBe("qa_failed");
121
+ if (result.pass === false && result.reason === "qa_failed") {
122
+ expect(result.conclusion).toBe("failure");
123
+ }
124
+ });
125
+ it("returns qa_failed for conclusion=cancelled", () => {
126
+ const qaRun = {
127
+ runId: 99999,
128
+ runSha: "head_sha_1",
129
+ triggeredAt: new Date(NOW.getTime() - 10 * 60 * 1000),
130
+ };
131
+ const result = evalQaCheck({
132
+ qaConfig: cfg,
133
+ headSha: "head_sha_1",
134
+ workflowFileExists: true,
135
+ qaRun,
136
+ runConclusion: "cancelled",
137
+ now: NOW,
138
+ });
139
+ expect(result.pass).toBe(false);
140
+ expect(result.reason).toBe("qa_failed");
141
+ if (result.pass === false && result.reason === "qa_failed") {
142
+ expect(result.conclusion).toBe("cancelled");
143
+ }
144
+ });
145
+ });
146
+ //# sourceMappingURL=qa-eval.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"qa-eval.test.js","sourceRoot":"","sources":["../../src/__tests__/qa-eval.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=qa-gap.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"qa-gap.test.d.ts","sourceRoot":"","sources":["../../src/__tests__/qa-gap.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,100 @@
1
+ import { describe, it, expect, vi, beforeEach, afterEach } from "vitest";
2
+ import { randomBytes } from "node:crypto";
3
+ import { unlinkSync } from "node:fs";
4
+ import { eq, and, isNull } from "drizzle-orm";
5
+ import { createDb } from "../db/index.js";
6
+ import { qaGapIssues } from "../db/schema.js";
7
+ import { fileGapIssue } from "../qa/gap.js";
8
+ function tmpDbPath() {
9
+ const id = randomBytes(8).toString("hex");
10
+ return `/tmp/laf-qa-gap-fn-${id}.sqlite`;
11
+ }
12
+ describe("fileGapIssue", () => {
13
+ const paths = [];
14
+ let db;
15
+ const repoUrl = "https://github.com/org/repo";
16
+ const branch = "main";
17
+ const workflowPath = ".github/workflows/smoke.yml";
18
+ beforeEach(async () => {
19
+ const path = tmpDbPath();
20
+ paths.push(path);
21
+ const created = await createDb({ driver: "sqlite", connectionString: path });
22
+ db = created;
23
+ });
24
+ afterEach(() => {
25
+ for (const p of paths) {
26
+ try {
27
+ unlinkSync(p);
28
+ }
29
+ catch { }
30
+ try {
31
+ unlinkSync(p + "-wal");
32
+ }
33
+ catch { }
34
+ try {
35
+ unlinkSync(p + "-shm");
36
+ }
37
+ catch { }
38
+ }
39
+ paths.length = 0;
40
+ });
41
+ function makeMockLinear(over = {}) {
42
+ return {
43
+ createIssue: vi.fn(async () => ({
44
+ issue: Promise.resolve({ identifier: "BEC-150", url: "https://linear.app/beckerspace/issue/BEC-150" }),
45
+ })),
46
+ ...over,
47
+ };
48
+ }
49
+ it("files a new issue when none exists, persists qa_gap_issues row", async () => {
50
+ const linear = makeMockLinear();
51
+ const result = await fileGapIssue({
52
+ db,
53
+ linear: linear,
54
+ repoUrl,
55
+ branch,
56
+ workflowPath,
57
+ linearTeamId: "team-uuid-123",
58
+ });
59
+ expect(result.kind).toBe("filed");
60
+ expect(result.kind === "filed" && result.linearIssueId).toBe("BEC-150");
61
+ expect(linear.createIssue).toHaveBeenCalledTimes(1);
62
+ const rows = await db.select().from(qaGapIssues).where(and(eq(qaGapIssues.repoUrl, repoUrl), isNull(qaGapIssues.resolvedAt)));
63
+ expect(rows).toHaveLength(1);
64
+ expect(rows[0].linearIssueId).toBe("BEC-150");
65
+ });
66
+ it("is idempotent — second call with existing open row is a no-op", async () => {
67
+ const linear = makeMockLinear();
68
+ await fileGapIssue({ db, linear: linear, repoUrl, branch, workflowPath, linearTeamId: "team-uuid-123" });
69
+ linear.createIssue.mockClear();
70
+ const result = await fileGapIssue({ db, linear: linear, repoUrl, branch, workflowPath, linearTeamId: "team-uuid-123" });
71
+ expect(result.kind).toBe("already_filed");
72
+ expect(result.kind === "already_filed" && result.linearIssueId).toBe("BEC-150");
73
+ expect(linear.createIssue).not.toHaveBeenCalled();
74
+ });
75
+ it("re-files when the previous row was resolved", async () => {
76
+ const linear = makeMockLinear();
77
+ await fileGapIssue({ db, linear: linear, repoUrl, branch, workflowPath, linearTeamId: "team-uuid-123" });
78
+ // Resolve the existing row
79
+ await db.update(qaGapIssues)
80
+ .set({ resolvedAt: new Date() })
81
+ .where(eq(qaGapIssues.linearIssueId, "BEC-150"));
82
+ // Update the mock to return a new ID for the second filing
83
+ linear.createIssue.mockResolvedValueOnce({
84
+ issue: Promise.resolve({ identifier: "BEC-160", url: "https://linear.app/beckerspace/issue/BEC-160" }),
85
+ });
86
+ const result = await fileGapIssue({ db, linear: linear, repoUrl, branch, workflowPath, linearTeamId: "team-uuid-123" });
87
+ expect(result.kind).toBe("filed");
88
+ expect(result.kind === "filed" && result.linearIssueId).toBe("BEC-160");
89
+ expect(linear.createIssue).toHaveBeenCalledTimes(2);
90
+ });
91
+ it("returns linear_error on Linear API failure", async () => {
92
+ const linear = makeMockLinear({
93
+ createIssue: vi.fn(async () => { throw new Error("Linear unauthorized"); }),
94
+ });
95
+ const result = await fileGapIssue({ db, linear: linear, repoUrl, branch, workflowPath, linearTeamId: "team-uuid-123" });
96
+ expect(result.kind).toBe("linear_error");
97
+ expect(result.kind === "linear_error" && result.message).toMatch(/unauthorized/);
98
+ });
99
+ });
100
+ //# sourceMappingURL=qa-gap.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"qa-gap.test.js","sourceRoot":"","sources":["../../src/__tests__/qa-gap.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=qa-github.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"qa-github.test.d.ts","sourceRoot":"","sources":["../../src/__tests__/qa-github.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,162 @@
1
+ import { describe, it, expect, vi } from "vitest";
2
+ import { triggerWorkflow, pollWorkflowRun, workflowFileExists, } from "../qa/github.js";
3
+ const fakeDb = {};
4
+ const owner = "org";
5
+ const repo = "repo";
6
+ function makeMockOctokit(over = {}) {
7
+ return {
8
+ actions: {
9
+ createWorkflowDispatch: vi.fn(async () => ({ status: 204 })),
10
+ listWorkflowRuns: vi.fn(async () => ({
11
+ data: { workflow_runs: [{ id: 99999, head_sha: "abc123", status: "in_progress", conclusion: null, created_at: "2026-05-04T12:00:00Z", run_started_at: "2026-05-04T12:00:00Z" }] },
12
+ })),
13
+ getWorkflowRun: vi.fn(async () => ({
14
+ data: { id: 99999, head_sha: "abc123", status: "in_progress", conclusion: null, run_started_at: "2026-05-04T12:00:00Z", updated_at: "2026-05-04T12:10:00Z" },
15
+ })),
16
+ },
17
+ repos: {
18
+ getContent: vi.fn(async () => ({ data: { type: "file", path: ".github/workflows/smoke.yml" } })),
19
+ },
20
+ ...over,
21
+ };
22
+ }
23
+ describe("workflowFileExists", () => {
24
+ it("returns true when file content is fetched successfully", async () => {
25
+ const octokit = makeMockOctokit();
26
+ const exists = await workflowFileExists({ octokit, owner, repo, path: ".github/workflows/smoke.yml", ref: "main" });
27
+ expect(exists).toBe(true);
28
+ expect(octokit.repos.getContent).toHaveBeenCalledWith({ owner, repo, path: ".github/workflows/smoke.yml", ref: "main" });
29
+ });
30
+ it("returns false on 404", async () => {
31
+ const octokit = makeMockOctokit({
32
+ repos: {
33
+ getContent: vi.fn(async () => {
34
+ const err = new Error("Not Found");
35
+ err.status = 404;
36
+ throw err;
37
+ }),
38
+ },
39
+ });
40
+ const exists = await workflowFileExists({ octokit, owner, repo, path: ".github/workflows/smoke.yml", ref: "main" });
41
+ expect(exists).toBe(false);
42
+ });
43
+ it("rethrows on 5xx", async () => {
44
+ const octokit = makeMockOctokit({
45
+ repos: {
46
+ getContent: vi.fn(async () => {
47
+ const err = new Error("Server Error");
48
+ err.status = 500;
49
+ throw err;
50
+ }),
51
+ },
52
+ });
53
+ await expect(workflowFileExists({ octokit, owner, repo, path: ".github/workflows/smoke.yml", ref: "main" })).rejects.toThrow("Server Error");
54
+ });
55
+ });
56
+ describe("triggerWorkflow", () => {
57
+ it("calls createWorkflowDispatch then finds the new run by SHA", async () => {
58
+ const octokit = makeMockOctokit();
59
+ const result = await triggerWorkflow({
60
+ octokit,
61
+ db: fakeDb,
62
+ owner,
63
+ repo,
64
+ repoUrl: `https://github.com/${owner}/${repo}`,
65
+ branch: "main",
66
+ workflow: ".github/workflows/smoke.yml",
67
+ ref: "abc123",
68
+ inputs: { environment: "preview" },
69
+ });
70
+ expect(result.kind).toBe("ok");
71
+ expect(result.kind === "ok" && result.runId).toBe(99999);
72
+ expect(octokit.actions.createWorkflowDispatch).toHaveBeenCalledWith({
73
+ owner,
74
+ repo,
75
+ workflow_id: ".github/workflows/smoke.yml",
76
+ ref: "abc123",
77
+ inputs: { environment: "preview" },
78
+ });
79
+ });
80
+ it("returns dispatch_404 on 404 (workflow file missing)", async () => {
81
+ const octokit = makeMockOctokit({
82
+ actions: {
83
+ createWorkflowDispatch: vi.fn(async () => {
84
+ const err = new Error("Not Found");
85
+ err.status = 404;
86
+ throw err;
87
+ }),
88
+ },
89
+ });
90
+ const result = await triggerWorkflow({
91
+ octokit, db: fakeDb, owner, repo, repoUrl: `https://github.com/${owner}/${repo}`, branch: "main",
92
+ workflow: ".github/workflows/smoke.yml", ref: "abc123",
93
+ });
94
+ expect(result.kind).toBe("dispatch_404");
95
+ });
96
+ it("returns dispatch_422 on 422 (workflow not workflow_dispatch-triggered)", async () => {
97
+ const octokit = makeMockOctokit({
98
+ actions: {
99
+ createWorkflowDispatch: vi.fn(async () => {
100
+ const err = new Error("Unprocessable Entity");
101
+ err.status = 422;
102
+ throw err;
103
+ }),
104
+ },
105
+ });
106
+ const result = await triggerWorkflow({
107
+ octokit, db: fakeDb, owner, repo, repoUrl: `https://github.com/${owner}/${repo}`, branch: "main",
108
+ workflow: ".github/workflows/smoke.yml", ref: "abc123",
109
+ });
110
+ expect(result.kind).toBe("dispatch_422");
111
+ });
112
+ it("returns dispatch_error on 5xx", async () => {
113
+ const octokit = makeMockOctokit({
114
+ actions: {
115
+ createWorkflowDispatch: vi.fn(async () => {
116
+ const err = new Error("Server Error");
117
+ err.status = 502;
118
+ throw err;
119
+ }),
120
+ },
121
+ });
122
+ const result = await triggerWorkflow({
123
+ octokit, db: fakeDb, owner, repo, repoUrl: `https://github.com/${owner}/${repo}`, branch: "main",
124
+ workflow: ".github/workflows/smoke.yml", ref: "abc123",
125
+ });
126
+ expect(result.kind).toBe("dispatch_error");
127
+ expect(result.kind === "dispatch_error" && result.message).toMatch(/Server Error/);
128
+ });
129
+ });
130
+ describe("pollWorkflowRun", () => {
131
+ it("returns running for in_progress runs", async () => {
132
+ const octokit = makeMockOctokit();
133
+ const result = await pollWorkflowRun({ octokit, owner, repo, runId: 99999 });
134
+ expect(result.kind).toBe("running");
135
+ });
136
+ it("returns completed with conclusion=success", async () => {
137
+ const octokit = makeMockOctokit({
138
+ actions: {
139
+ getWorkflowRun: vi.fn(async () => ({
140
+ data: { id: 99999, status: "completed", conclusion: "success", run_started_at: "2026-05-04T12:00:00Z", updated_at: "2026-05-04T12:10:00Z" },
141
+ })),
142
+ },
143
+ });
144
+ const result = await pollWorkflowRun({ octokit, owner, repo, runId: 99999 });
145
+ expect(result.kind).toBe("completed");
146
+ expect(result.kind === "completed" && result.conclusion).toBe("success");
147
+ expect(result.kind === "completed" && result.durationMs).toBe(10 * 60 * 1000);
148
+ });
149
+ it("returns completed with conclusion=failure", async () => {
150
+ const octokit = makeMockOctokit({
151
+ actions: {
152
+ getWorkflowRun: vi.fn(async () => ({
153
+ data: { id: 99999, status: "completed", conclusion: "failure", run_started_at: "2026-05-04T12:00:00Z", updated_at: "2026-05-04T12:05:00Z" },
154
+ })),
155
+ },
156
+ });
157
+ const result = await pollWorkflowRun({ octokit, owner, repo, runId: 99999 });
158
+ expect(result.kind).toBe("completed");
159
+ expect(result.kind === "completed" && result.conclusion).toBe("failure");
160
+ });
161
+ });
162
+ //# sourceMappingURL=qa-github.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"qa-github.test.js","sourceRoot":"","sources":["../../src/__tests__/qa-github.test.ts"],"names":[],"mappings":"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"}
@@ -86,6 +86,7 @@ vi.mock("../executor/executor.js", () => ({
86
86
  turns: 1,
87
87
  handoffArtifact: makeHandoffArtifact(runId, issueId, stage, reviewFindings),
88
88
  handoffIsStructured: true,
89
+ stageRunId: `mock-${runId}-${stage}`,
89
90
  };
90
91
  }),
91
92
  }));