@urateam/core 0.1.54 → 0.1.56

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 (225) hide show
  1. package/dist/__tests__/auth-monitor.test.js +119 -20
  2. package/dist/__tests__/auth-monitor.test.js.map +1 -1
  3. package/dist/__tests__/bec-238-repro.test.d.ts +2 -0
  4. package/dist/__tests__/bec-238-repro.test.d.ts.map +1 -0
  5. package/dist/__tests__/bec-238-repro.test.js +116 -0
  6. package/dist/__tests__/bec-238-repro.test.js.map +1 -0
  7. package/dist/__tests__/bec146-repro.test.d.ts +2 -0
  8. package/dist/__tests__/bec146-repro.test.d.ts.map +1 -0
  9. package/dist/__tests__/bec146-repro.test.js +153 -0
  10. package/dist/__tests__/bec146-repro.test.js.map +1 -0
  11. package/dist/__tests__/db-migrations.test.js +7 -3
  12. package/dist/__tests__/db-migrations.test.js.map +1 -1
  13. package/dist/__tests__/decision-artifact-schema.test.d.ts +2 -0
  14. package/dist/__tests__/decision-artifact-schema.test.d.ts.map +1 -0
  15. package/dist/__tests__/decision-artifact-schema.test.js +45 -0
  16. package/dist/__tests__/decision-artifact-schema.test.js.map +1 -0
  17. package/dist/__tests__/decisions-parser.test.d.ts +2 -0
  18. package/dist/__tests__/decisions-parser.test.d.ts.map +1 -0
  19. package/dist/__tests__/decisions-parser.test.js +59 -0
  20. package/dist/__tests__/decisions-parser.test.js.map +1 -0
  21. package/dist/__tests__/decisions-store.test.d.ts +2 -0
  22. package/dist/__tests__/decisions-store.test.d.ts.map +1 -0
  23. package/dist/__tests__/decisions-store.test.js +68 -0
  24. package/dist/__tests__/decisions-store.test.js.map +1 -0
  25. package/dist/__tests__/decisions-table-migration.test.d.ts +2 -0
  26. package/dist/__tests__/decisions-table-migration.test.d.ts.map +1 -0
  27. package/dist/__tests__/decisions-table-migration.test.js +56 -0
  28. package/dist/__tests__/decisions-table-migration.test.js.map +1 -0
  29. package/dist/__tests__/execute-stage-session-opts.test.js +1 -0
  30. package/dist/__tests__/execute-stage-session-opts.test.js.map +1 -1
  31. package/dist/__tests__/executor-issue-id.test.js +1 -0
  32. package/dist/__tests__/executor-issue-id.test.js.map +1 -1
  33. package/dist/__tests__/extract-handoff-decisions.test.d.ts +2 -0
  34. package/dist/__tests__/extract-handoff-decisions.test.d.ts.map +1 -0
  35. package/dist/__tests__/extract-handoff-decisions.test.js +26 -0
  36. package/dist/__tests__/extract-handoff-decisions.test.js.map +1 -0
  37. package/dist/__tests__/force-push-agent-branches.test.js +38 -15
  38. package/dist/__tests__/force-push-agent-branches.test.js.map +1 -1
  39. package/dist/__tests__/github-webhook.test.js +117 -0
  40. package/dist/__tests__/github-webhook.test.js.map +1 -1
  41. package/dist/__tests__/implement-template-decisions.test.d.ts +2 -0
  42. package/dist/__tests__/implement-template-decisions.test.d.ts.map +1 -0
  43. package/dist/__tests__/implement-template-decisions.test.js +41 -0
  44. package/dist/__tests__/implement-template-decisions.test.js.map +1 -0
  45. package/dist/__tests__/migrator.test.js +137 -39
  46. package/dist/__tests__/migrator.test.js.map +1 -1
  47. package/dist/__tests__/pipeline-runner.test.js +118 -0
  48. package/dist/__tests__/pipeline-runner.test.js.map +1 -1
  49. package/dist/__tests__/pm-conflict.test.js +145 -5
  50. package/dist/__tests__/pm-conflict.test.js.map +1 -1
  51. package/dist/__tests__/preflight-claude-auth.test.js +3 -2
  52. package/dist/__tests__/preflight-claude-auth.test.js.map +1 -1
  53. package/dist/__tests__/ralph-review-fix-regression.test.js +1 -0
  54. package/dist/__tests__/ralph-review-fix-regression.test.js.map +1 -1
  55. package/dist/__tests__/ralph.test.js +1 -0
  56. package/dist/__tests__/ralph.test.js.map +1 -1
  57. package/dist/__tests__/release-manager-qarun-query.test.d.ts +2 -0
  58. package/dist/__tests__/release-manager-qarun-query.test.d.ts.map +1 -0
  59. package/dist/__tests__/release-manager-qarun-query.test.js +233 -0
  60. package/dist/__tests__/release-manager-qarun-query.test.js.map +1 -0
  61. package/dist/__tests__/release-manager-scheduler.test.js +62 -0
  62. package/dist/__tests__/release-manager-scheduler.test.js.map +1 -1
  63. package/dist/__tests__/session-lazy-creation.test.js +1 -0
  64. package/dist/__tests__/session-lazy-creation.test.js.map +1 -1
  65. package/dist/__tests__/session-resolver.test.d.ts +13 -0
  66. package/dist/__tests__/session-resolver.test.d.ts.map +1 -0
  67. package/dist/__tests__/session-resolver.test.js +152 -0
  68. package/dist/__tests__/session-resolver.test.js.map +1 -0
  69. package/dist/__tests__/session-resume-fallback.test.js +1 -0
  70. package/dist/__tests__/session-resume-fallback.test.js.map +1 -1
  71. package/dist/__tests__/stage-models.test.js +1 -0
  72. package/dist/__tests__/stage-models.test.js.map +1 -1
  73. package/dist/__tests__/surgical-review-fix-prompt.test.d.ts +2 -0
  74. package/dist/__tests__/surgical-review-fix-prompt.test.d.ts.map +1 -0
  75. package/dist/__tests__/surgical-review-fix-prompt.test.js +71 -0
  76. package/dist/__tests__/surgical-review-fix-prompt.test.js.map +1 -0
  77. package/dist/__tests__/surgical-review-fix-runner.test.d.ts +2 -0
  78. package/dist/__tests__/surgical-review-fix-runner.test.d.ts.map +1 -0
  79. package/dist/__tests__/surgical-review-fix-runner.test.js +171 -0
  80. package/dist/__tests__/surgical-review-fix-runner.test.js.map +1 -0
  81. package/dist/__tests__/validate-run-mode.test.js +1 -0
  82. package/dist/__tests__/validate-run-mode.test.js.map +1 -1
  83. package/dist/__tests__/validate.test.js +1 -0
  84. package/dist/__tests__/validate.test.js.map +1 -1
  85. package/dist/audit/auth-error-messages.d.ts +6 -0
  86. package/dist/audit/auth-error-messages.d.ts.map +1 -0
  87. package/dist/audit/auth-error-messages.js +18 -0
  88. package/dist/audit/auth-error-messages.js.map +1 -0
  89. package/dist/audit/events.d.ts +22 -4
  90. package/dist/audit/events.d.ts.map +1 -1
  91. package/dist/audit/events.js +34 -5
  92. package/dist/audit/events.js.map +1 -1
  93. package/dist/audit/index.d.ts +1 -0
  94. package/dist/audit/index.d.ts.map +1 -1
  95. package/dist/audit/index.js +1 -0
  96. package/dist/audit/index.js.map +1 -1
  97. package/dist/db/client.d.ts.map +1 -1
  98. package/dist/db/client.js +10 -0
  99. package/dist/db/client.js.map +1 -1
  100. package/dist/db/decisions-store.d.ts +25 -0
  101. package/dist/db/decisions-store.d.ts.map +1 -0
  102. package/dist/db/decisions-store.js +56 -0
  103. package/dist/db/decisions-store.js.map +1 -0
  104. package/dist/db/index.d.ts +1 -1
  105. package/dist/db/index.d.ts.map +1 -1
  106. package/dist/db/index.js +1 -1
  107. package/dist/db/index.js.map +1 -1
  108. package/dist/db/migrations/postgres/008_sso.sql +9 -20
  109. package/dist/db/migrations/postgres/009_review_model_runs.sql +9 -18
  110. package/dist/db/migrations/postgres/009_sso.sql +20 -0
  111. package/dist/db/migrations/postgres/010_release_manager.sql +9 -38
  112. package/dist/db/migrations/postgres/010_review_model_runs.sql +18 -0
  113. package/dist/db/migrations/postgres/011_qa_run_columns.sql +9 -8
  114. package/dist/db/migrations/postgres/011_release_manager.sql +38 -0
  115. package/dist/db/migrations/postgres/012_qa_gap_issues.sql +9 -18
  116. package/dist/db/migrations/postgres/012_qa_run_columns.sql +8 -0
  117. package/dist/db/migrations/postgres/013_qa_gap_issues.sql +18 -0
  118. package/dist/db/migrations/postgres/013_stage_runs_cache_tokens.sql +11 -6
  119. package/dist/db/migrations/postgres/014_missing_indexes.sql +9 -28
  120. package/dist/db/migrations/postgres/014_stage_runs_cache_tokens.sql +6 -0
  121. package/dist/db/migrations/postgres/015_missing_indexes.sql +28 -0
  122. package/dist/db/migrations/postgres/015_triage_results.sql +9 -11
  123. package/dist/db/migrations/postgres/016_triage_results.sql +11 -0
  124. package/dist/db/migrations/sqlite/007_sso.sql +9 -20
  125. package/dist/db/migrations/sqlite/008_review_model_runs.sql +9 -18
  126. package/dist/db/migrations/sqlite/008_sso.sql +20 -0
  127. package/dist/db/migrations/sqlite/009_release_manager.sql +9 -43
  128. package/dist/db/migrations/sqlite/009_review_model_runs.sql +18 -0
  129. package/dist/db/migrations/sqlite/010_qa_run_columns.sql +9 -9
  130. package/dist/db/migrations/sqlite/010_release_manager.sql +43 -0
  131. package/dist/db/migrations/sqlite/011_qa_gap_issues.sql +9 -22
  132. package/dist/db/migrations/sqlite/011_qa_run_columns.sql +9 -0
  133. package/dist/db/migrations/sqlite/012_qa_gap_issues.sql +22 -0
  134. package/dist/db/migrations/sqlite/012_stage_runs_cache_tokens.sql +11 -6
  135. package/dist/db/migrations/sqlite/013_missing_indexes.sql +10 -28
  136. package/dist/db/migrations/sqlite/013_stage_runs_cache_tokens.sql +6 -0
  137. package/dist/db/migrations/sqlite/013_triage_results.sql +10 -11
  138. package/dist/db/migrations/sqlite/014_missing_indexes.sql +28 -0
  139. package/dist/db/migrations/sqlite/015_triage_results.sql +11 -0
  140. package/dist/db/migrator.d.ts +45 -2
  141. package/dist/db/migrator.d.ts.map +1 -1
  142. package/dist/db/migrator.js +113 -29
  143. package/dist/db/migrator.js.map +1 -1
  144. package/dist/db/schema.d.ts +125 -0
  145. package/dist/db/schema.d.ts.map +1 -1
  146. package/dist/db/schema.js +13 -0
  147. package/dist/db/schema.js.map +1 -1
  148. package/dist/executor/auth-monitor.d.ts.map +1 -1
  149. package/dist/executor/auth-monitor.js +39 -30
  150. package/dist/executor/auth-monitor.js.map +1 -1
  151. package/dist/executor/deep-review.d.ts.map +1 -1
  152. package/dist/executor/deep-review.js +14 -79
  153. package/dist/executor/deep-review.js.map +1 -1
  154. package/dist/executor/executor.d.ts +21 -0
  155. package/dist/executor/executor.d.ts.map +1 -1
  156. package/dist/executor/executor.js +41 -90
  157. package/dist/executor/executor.js.map +1 -1
  158. package/dist/executor/extract-handoff.d.ts +9 -0
  159. package/dist/executor/extract-handoff.d.ts.map +1 -1
  160. package/dist/executor/extract-handoff.js +48 -0
  161. package/dist/executor/extract-handoff.js.map +1 -1
  162. package/dist/executor/handoff.d.ts +10 -1
  163. package/dist/executor/handoff.d.ts.map +1 -1
  164. package/dist/executor/handoff.js +7 -1
  165. package/dist/executor/handoff.js.map +1 -1
  166. package/dist/executor/index.d.ts +1 -0
  167. package/dist/executor/index.d.ts.map +1 -1
  168. package/dist/executor/index.js +1 -0
  169. package/dist/executor/index.js.map +1 -1
  170. package/dist/executor/prompt/templates.d.ts +12 -1
  171. package/dist/executor/prompt/templates.d.ts.map +1 -1
  172. package/dist/executor/prompt/templates.js +64 -0
  173. package/dist/executor/prompt/templates.js.map +1 -1
  174. package/dist/executor/session-resolver.d.ts +51 -0
  175. package/dist/executor/session-resolver.d.ts.map +1 -0
  176. package/dist/executor/session-resolver.js +83 -0
  177. package/dist/executor/session-resolver.js.map +1 -0
  178. package/dist/index.d.ts +1 -1
  179. package/dist/index.d.ts.map +1 -1
  180. package/dist/index.js +1 -1
  181. package/dist/index.js.map +1 -1
  182. package/dist/notifier/composite.js +1 -1
  183. package/dist/notifier/composite.js.map +1 -1
  184. package/dist/notifier/linear.d.ts +1 -1
  185. package/dist/notifier/linear.d.ts.map +1 -1
  186. package/dist/notifier/linear.js +49 -9
  187. package/dist/notifier/linear.js.map +1 -1
  188. package/dist/pipeline/run-surgical-review-fix.d.ts +29 -0
  189. package/dist/pipeline/run-surgical-review-fix.d.ts.map +1 -0
  190. package/dist/pipeline/run-surgical-review-fix.js +48 -0
  191. package/dist/pipeline/run-surgical-review-fix.js.map +1 -0
  192. package/dist/pipeline/runner.d.ts.map +1 -1
  193. package/dist/pipeline/runner.js +132 -36
  194. package/dist/pipeline/runner.js.map +1 -1
  195. package/dist/pm/conflict.d.ts +2 -0
  196. package/dist/pm/conflict.d.ts.map +1 -1
  197. package/dist/pm/conflict.js +59 -9
  198. package/dist/pm/conflict.js.map +1 -1
  199. package/dist/pm/scheduler.d.ts +2 -0
  200. package/dist/pm/scheduler.d.ts.map +1 -1
  201. package/dist/pm/scheduler.js +51 -43
  202. package/dist/pm/scheduler.js.map +1 -1
  203. package/dist/pm/slack.d.ts +11 -0
  204. package/dist/pm/slack.d.ts.map +1 -1
  205. package/dist/pm/slack.js +48 -5
  206. package/dist/pm/slack.js.map +1 -1
  207. package/dist/release-manager/release-helpers.d.ts +19 -2
  208. package/dist/release-manager/release-helpers.d.ts.map +1 -1
  209. package/dist/release-manager/release-helpers.js +22 -3
  210. package/dist/release-manager/release-helpers.js.map +1 -1
  211. package/dist/release-manager/release-tick.d.ts.map +1 -1
  212. package/dist/release-manager/release-tick.js +20 -8
  213. package/dist/release-manager/release-tick.js.map +1 -1
  214. package/dist/release-manager/state.d.ts.map +1 -1
  215. package/dist/release-manager/state.js +25 -18
  216. package/dist/release-manager/state.js.map +1 -1
  217. package/dist/types.d.ts +232 -13
  218. package/dist/types.d.ts.map +1 -1
  219. package/dist/types.js +52 -0
  220. package/dist/types.js.map +1 -1
  221. package/dist/webhook/github-handler.d.ts +2 -2
  222. package/dist/webhook/github-handler.d.ts.map +1 -1
  223. package/dist/webhook/github-handler.js +18 -6
  224. package/dist/webhook/github-handler.js.map +1 -1
  225. package/package.json +1 -1
@@ -1,10 +1,12 @@
1
1
  /**
2
- * Tests for AuthMonitor (BEC-207) — periodic Claude session health-check.
2
+ * Tests for AuthMonitor (BEC-207 / BEC-237) — periodic Claude session health-check.
3
3
  *
4
4
  * Covers:
5
- * - Returns early when CLAUDE_CODE_OAUTH_TOKEN or ANTHROPIC_API_KEY is set.
6
- * - Runs `claude auth status` only when neither env var is set.
7
- * - On expiry: sends Slack alert and writes claude.auth_expired audit event.
5
+ * - Returns early ONLY when ANTHROPIC_API_KEY is set and CLAUDE_CODE_OAUTH_TOKEN is NOT set.
6
+ * - When CLAUDE_CODE_OAUTH_TOKEN is set, runs the auth probe regardless of ANTHROPIC_API_KEY.
7
+ * - Runs `claude auth status` when neither env var is set (mounted-session path).
8
+ * - On expiry: sends Slack alert (text branches on authMethod) and writes claude.auth_expired
9
+ * audit event (payload includes authMethod).
8
10
  * - 6-hour throttle: skips check if interval has not elapsed.
9
11
  * - createAuthMonitor stateful wrapper manages lastCheckTime across calls.
10
12
  */
@@ -99,29 +101,92 @@ describe("runAuthMonitorCheck", () => {
99
101
  await runAuthMonitorCheck(0, {}, SMALL_INTERVAL);
100
102
  expect(mockExecFile).toHaveBeenCalledTimes(1);
101
103
  });
102
- // --- Env-var short-circuit (AC #8) ---
103
- it("returns early without subprocess when CLAUDE_CODE_OAUTH_TOKEN is set", async () => {
104
- process.env.CLAUDE_CODE_OAUTH_TOKEN = "sk-ant-oat-test";
104
+ // --- Env-var skip logic (BEC-237) ---
105
+ it("skips check without subprocess when ANTHROPIC_API_KEY is set and CLAUDE_CODE_OAUTH_TOKEN is NOT set", async () => {
106
+ process.env.ANTHROPIC_API_KEY = "sk-ant-api03-test";
105
107
  await runAuthMonitorCheck(0, {}, SMALL_INTERVAL);
106
108
  expect(mockExecFile).not.toHaveBeenCalled();
107
109
  expect(mockPostSlackMessage).not.toHaveBeenCalled();
108
110
  expect(mockLogAuditEvent).not.toHaveBeenCalled();
109
111
  });
110
- it("returns early without subprocess when ANTHROPIC_API_KEY is set", async () => {
112
+ it("runs the auth probe when CLAUDE_CODE_OAUTH_TOKEN is set (OAuth tokens can expire)", async () => {
113
+ process.env.CLAUDE_CODE_OAUTH_TOKEN = "sk-ant-oat-test";
114
+ simulateCliSuccess();
115
+ await runAuthMonitorCheck(0, {}, SMALL_INTERVAL);
116
+ expect(mockExecFile).toHaveBeenCalledTimes(1);
117
+ });
118
+ it("runs the auth probe when both CLAUDE_CODE_OAUTH_TOKEN and ANTHROPIC_API_KEY are set", async () => {
119
+ // CLAUDE_CODE_OAUTH_TOKEN takes precedence — it can expire even if ANTHROPIC_API_KEY is also set
120
+ process.env.CLAUDE_CODE_OAUTH_TOKEN = "sk-ant-oat-test";
111
121
  process.env.ANTHROPIC_API_KEY = "sk-ant-api03-test";
122
+ simulateCliSuccess();
112
123
  await runAuthMonitorCheck(0, {}, SMALL_INTERVAL);
113
- expect(mockExecFile).not.toHaveBeenCalled();
124
+ expect(mockExecFile).toHaveBeenCalledTimes(1);
125
+ });
126
+ it("runs the auth probe when neither env var is set (mounted-session path)", async () => {
127
+ simulateCliSuccess();
128
+ await runAuthMonitorCheck(0, {}, SMALL_INTERVAL);
129
+ expect(mockExecFile).toHaveBeenCalledTimes(1);
130
+ });
131
+ // --- OAuth token probe — success ---
132
+ it("does not alert when CLAUDE_CODE_OAUTH_TOKEN probe succeeds", async () => {
133
+ process.env.CLAUDE_CODE_OAUTH_TOKEN = "sk-ant-oat-test";
134
+ simulateCliSuccess();
135
+ await runAuthMonitorCheck(0, {
136
+ slackBotToken: "xoxb-test",
137
+ slackErrorChannel: "CTEST",
138
+ }, SMALL_INTERVAL);
114
139
  expect(mockPostSlackMessage).not.toHaveBeenCalled();
115
140
  expect(mockLogAuditEvent).not.toHaveBeenCalled();
116
141
  });
117
- it("returns early without subprocess when both env vars are set", async () => {
142
+ // --- OAuth token probe failure (BEC-237 core scenario) ---
143
+ it("sends Slack alert with oauth-token instructions when CLAUDE_CODE_OAUTH_TOKEN probe fails", async () => {
144
+ process.env.CLAUDE_CODE_OAUTH_TOKEN = "sk-ant-oat-test";
145
+ simulateCliFailure();
146
+ await runAuthMonitorCheck(0, {
147
+ slackBotToken: "xoxb-bot-token",
148
+ slackErrorChannel: "CERROR",
149
+ }, SMALL_INTERVAL);
150
+ expect(mockPostSlackMessage).toHaveBeenCalledTimes(1);
151
+ const [calledToken, calledPayload] = mockPostSlackMessage.mock.calls[0];
152
+ expect(calledToken).toBe("xoxb-bot-token");
153
+ expect(calledPayload.channel).toBe("CERROR");
154
+ // Alert text should mention setup-token, not claude login
155
+ expect(calledPayload.text).toContain("setup-token");
156
+ expect(calledPayload.text).not.toContain("claude login");
157
+ });
158
+ it("logs claude.auth_expired with authMethod=oauth-token when CLAUDE_CODE_OAUTH_TOKEN probe fails", async () => {
159
+ process.env.CLAUDE_CODE_OAUTH_TOKEN = "sk-ant-oat-test";
160
+ simulateCliFailure();
161
+ const fakeDb = {};
162
+ await runAuthMonitorCheck(0, { db: fakeDb }, SMALL_INTERVAL);
163
+ expect(mockLogAuditEvent).toHaveBeenCalledTimes(1);
164
+ const [calledDb, calledEvent] = mockLogAuditEvent.mock.calls[0];
165
+ expect(calledDb).toBe(fakeDb);
166
+ expect(calledEvent.eventType).toBe("claude.auth_expired");
167
+ expect(calledEvent.actor).toBe("system");
168
+ expect(calledEvent.payload).toMatchObject({
169
+ detectedAt: expect.any(String),
170
+ authMethod: "oauth-token",
171
+ });
172
+ });
173
+ it("sends oauth-token Slack alert when both env vars set and probe fails", async () => {
118
174
  process.env.CLAUDE_CODE_OAUTH_TOKEN = "sk-ant-oat-test";
119
175
  process.env.ANTHROPIC_API_KEY = "sk-ant-api03-test";
120
- await runAuthMonitorCheck(0, {}, SMALL_INTERVAL);
121
- expect(mockExecFile).not.toHaveBeenCalled();
176
+ simulateCliFailure();
177
+ const fakeDb = {};
178
+ await runAuthMonitorCheck(0, {
179
+ slackBotToken: "xoxb-test",
180
+ slackErrorChannel: "CTEST",
181
+ db: fakeDb,
182
+ }, SMALL_INTERVAL);
183
+ expect(mockPostSlackMessage).toHaveBeenCalledTimes(1);
184
+ expect(mockPostSlackMessage.mock.calls[0][1].text).toContain("setup-token");
185
+ const [, calledEvent] = mockLogAuditEvent.mock.calls[0];
186
+ expect(calledEvent.payload.authMethod).toBe("oauth-token");
122
187
  });
123
188
  // --- Session valid ---
124
- it("runs subprocess and skips alerts when session is valid", async () => {
189
+ it("runs subprocess and skips alerts when mounted session is valid", async () => {
125
190
  simulateCliSuccess();
126
191
  await runAuthMonitorCheck(0, {
127
192
  slackBotToken: "xoxb-test",
@@ -131,8 +196,8 @@ describe("runAuthMonitorCheck", () => {
131
196
  expect(mockPostSlackMessage).not.toHaveBeenCalled();
132
197
  expect(mockLogAuditEvent).not.toHaveBeenCalled();
133
198
  });
134
- // --- Session expired (AC #7, #15) ---
135
- it("sends Slack alert to configured channel when session is expired", async () => {
199
+ // --- Mounted session expired (existing behavior, preserved) ---
200
+ it("sends Slack alert to configured channel when mounted session is expired", async () => {
136
201
  simulateCliFailure();
137
202
  await runAuthMonitorCheck(0, {
138
203
  slackBotToken: "xoxb-bot-token",
@@ -141,7 +206,16 @@ describe("runAuthMonitorCheck", () => {
141
206
  expect(mockPostSlackMessage).toHaveBeenCalledTimes(1);
142
207
  expect(mockPostSlackMessage).toHaveBeenCalledWith("xoxb-bot-token", expect.objectContaining({ channel: "CERROR" }));
143
208
  });
144
- it("logs claude.auth_expired audit event when session is expired and db provided (AC #7)", async () => {
209
+ it("Slack alert for mounted-session expiry mentions claude login, not setup-token", async () => {
210
+ simulateCliFailure();
211
+ await runAuthMonitorCheck(0, {
212
+ slackBotToken: "xoxb-test",
213
+ slackErrorChannel: "CTEST",
214
+ }, SMALL_INTERVAL);
215
+ const text = mockPostSlackMessage.mock.calls[0][1].text;
216
+ expect(text).toContain("claude login");
217
+ });
218
+ it("logs claude.auth_expired audit event with authMethod=mounted-session when session expired", async () => {
145
219
  simulateCliFailure();
146
220
  const fakeDb = {};
147
221
  await runAuthMonitorCheck(0, { db: fakeDb }, SMALL_INTERVAL);
@@ -152,6 +226,7 @@ describe("runAuthMonitorCheck", () => {
152
226
  expect(calledEvent.actor).toBe("system");
153
227
  expect(calledEvent.payload).toMatchObject({
154
228
  detectedAt: expect.any(String),
229
+ authMethod: "mounted-session",
155
230
  });
156
231
  });
157
232
  it("does not throw when Slack post fails", async () => {
@@ -188,7 +263,7 @@ describe("runAuthMonitorCheck", () => {
188
263
  });
189
264
  });
190
265
  // ---------------------------------------------------------------------------
191
- // Tests: createAuthMonitor stateful wrapper (AC #6)
266
+ // Tests: createAuthMonitor stateful wrapper
192
267
  // ---------------------------------------------------------------------------
193
268
  describe("createAuthMonitor", () => {
194
269
  let savedOauthToken;
@@ -229,13 +304,36 @@ describe("createAuthMonitor", () => {
229
304
  await monitor.tick(); // second tick immediately — throttled
230
305
  expect(mockExecFile).toHaveBeenCalledTimes(1); // only one subprocess
231
306
  });
232
- it("tick() skips session check when CLAUDE_CODE_OAUTH_TOKEN is set (AC #8)", async () => {
233
- process.env.CLAUDE_CODE_OAUTH_TOKEN = "sk-ant-oat-test";
307
+ it("tick() skips session check when only ANTHROPIC_API_KEY is set (static key, never expires)", async () => {
308
+ process.env.ANTHROPIC_API_KEY = "sk-ant-api03-test";
234
309
  const monitor = createAuthMonitor({}, SMALL_INTERVAL);
235
310
  await monitor.tick();
236
311
  expect(mockExecFile).not.toHaveBeenCalled();
237
312
  });
238
- it("tick() sends alert on expired session and logs audit event (AC #6, #7)", async () => {
313
+ it("tick() runs probe when CLAUDE_CODE_OAUTH_TOKEN is set (OAuth tokens can expire)", async () => {
314
+ process.env.CLAUDE_CODE_OAUTH_TOKEN = "sk-ant-oat-test";
315
+ simulateCliSuccess();
316
+ const monitor = createAuthMonitor({}, SMALL_INTERVAL);
317
+ await monitor.tick();
318
+ expect(mockExecFile).toHaveBeenCalledTimes(1);
319
+ });
320
+ it("tick() sends alert on expired OAuth token and logs audit event with authMethod=oauth-token", async () => {
321
+ process.env.CLAUDE_CODE_OAUTH_TOKEN = "sk-ant-oat-test";
322
+ simulateCliFailure();
323
+ const fakeDb = {};
324
+ const monitor = createAuthMonitor({
325
+ slackBotToken: "xoxb-test",
326
+ slackErrorChannel: "CALERTS",
327
+ db: fakeDb,
328
+ }, SMALL_INTERVAL);
329
+ await monitor.tick();
330
+ expect(mockPostSlackMessage).toHaveBeenCalledWith("xoxb-test", expect.objectContaining({ channel: "CALERTS" }));
331
+ expect(mockLogAuditEvent).toHaveBeenCalledTimes(1);
332
+ const [, calledEvent] = mockLogAuditEvent.mock.calls[0];
333
+ expect(calledEvent.eventType).toBe("claude.auth_expired");
334
+ expect(calledEvent.payload.authMethod).toBe("oauth-token");
335
+ });
336
+ it("tick() sends alert on expired mounted session and logs audit event with authMethod=mounted-session", async () => {
239
337
  simulateCliFailure();
240
338
  const fakeDb = {};
241
339
  const monitor = createAuthMonitor({
@@ -248,6 +346,7 @@ describe("createAuthMonitor", () => {
248
346
  expect(mockLogAuditEvent).toHaveBeenCalledTimes(1);
249
347
  const [, calledEvent] = mockLogAuditEvent.mock.calls[0];
250
348
  expect(calledEvent.eventType).toBe("claude.auth_expired");
349
+ expect(calledEvent.payload.authMethod).toBe("mounted-session");
251
350
  });
252
351
  });
253
352
  //# sourceMappingURL=auth-monitor.test.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"auth-monitor.test.js","sourceRoot":"","sources":["../../src/__tests__/auth-monitor.test.ts"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"auth-monitor.test.js","sourceRoot":"","sources":["../../src/__tests__/auth-monitor.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=bec-238-repro.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"bec-238-repro.test.d.ts","sourceRoot":"","sources":["../../src/__tests__/bec-238-repro.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,116 @@
1
+ /**
2
+ * BEC-238: Slack bot token missing reactions:read scope.
3
+ *
4
+ * Verifies the fix:
5
+ * 1. probeReactionsScope() logs error exactly once on missing_scope.
6
+ * 2. probeReactionsScope() stays silent when scope is present (any other error).
7
+ * 3. checkApprovalReactions() deduplicates missing_scope warns to one per
8
+ * notifier instance — no 7×/tick flood.
9
+ * 4. checkApprovalReactions() still logs every occurrence of non-missing_scope
10
+ * errors (other error types are not deduped).
11
+ */
12
+ import { describe, it, expect, vi, beforeEach } from "vitest";
13
+ import { PmSlackNotifier } from "../pm/slack.js";
14
+ vi.mock("../logger.js", () => {
15
+ const warnSpy = vi.fn();
16
+ const errorSpy = vi.fn();
17
+ return {
18
+ createLogger: () => ({
19
+ warn: warnSpy,
20
+ error: errorSpy,
21
+ info: vi.fn(),
22
+ debug: vi.fn(),
23
+ }),
24
+ __warnSpy: warnSpy,
25
+ __errorSpy: errorSpy,
26
+ };
27
+ });
28
+ import * as loggerModule from "../logger.js";
29
+ const warnSpy = loggerModule.__warnSpy;
30
+ const errorSpy = loggerModule.__errorSpy;
31
+ const mockFetch = vi.fn();
32
+ beforeEach(() => {
33
+ vi.stubGlobal("fetch", mockFetch);
34
+ mockFetch.mockReset();
35
+ warnSpy.mockReset();
36
+ errorSpy.mockReset();
37
+ });
38
+ function fetchReturning(body) {
39
+ return { ok: true, json: () => Promise.resolve(body) };
40
+ }
41
+ // ── probeReactionsScope ───────────────────────────────────────────────────────
42
+ describe("PmSlackNotifier.probeReactionsScope (BEC-238 AC2)", () => {
43
+ it("logs error exactly once when reactions:read scope is missing", async () => {
44
+ mockFetch.mockResolvedValueOnce(fetchReturning({ ok: false, error: "missing_scope" }));
45
+ const notifier = new PmSlackNotifier({ botToken: "xoxb-test", channelId: "C0000" });
46
+ await notifier.probeReactionsScope();
47
+ expect(errorSpy).toHaveBeenCalledTimes(1);
48
+ const [ctx, msg] = errorSpy.mock.calls[0];
49
+ expect(ctx?.error).toBe("missing_scope");
50
+ expect(msg).toMatch(/reactions:read/);
51
+ expect(msg).toMatch(/Fix:/);
52
+ expect(warnSpy).not.toHaveBeenCalled();
53
+ });
54
+ it("stays silent when scope is present (message_not_found means token can call reactions.get)", async () => {
55
+ mockFetch.mockResolvedValueOnce(fetchReturning({ ok: false, error: "message_not_found" }));
56
+ const notifier = new PmSlackNotifier({ botToken: "xoxb-test", channelId: "C0000" });
57
+ await notifier.probeReactionsScope();
58
+ expect(errorSpy).not.toHaveBeenCalled();
59
+ expect(warnSpy).not.toHaveBeenCalled();
60
+ });
61
+ it("stays silent when reactions.get succeeds (scope definitely present)", async () => {
62
+ mockFetch.mockResolvedValueOnce(fetchReturning({ ok: true, message: { reactions: [] } }));
63
+ const notifier = new PmSlackNotifier({ botToken: "xoxb-test", channelId: "C0000" });
64
+ await notifier.probeReactionsScope();
65
+ expect(errorSpy).not.toHaveBeenCalled();
66
+ expect(warnSpy).not.toHaveBeenCalled();
67
+ });
68
+ it("logs warn (not error) on fetch exception — probe is best-effort", async () => {
69
+ mockFetch.mockRejectedValueOnce(new Error("network timeout"));
70
+ const notifier = new PmSlackNotifier({ botToken: "xoxb-test", channelId: "C0000" });
71
+ await notifier.probeReactionsScope();
72
+ expect(errorSpy).not.toHaveBeenCalled();
73
+ expect(warnSpy).toHaveBeenCalledTimes(1);
74
+ expect(warnSpy.mock.calls[0][1]).toMatch(/scope probe failed/);
75
+ });
76
+ });
77
+ // ── checkApprovalReactions dedup (AC3) ───────────────────────────────────────
78
+ describe("PmSlackNotifier.checkApprovalReactions missing_scope dedup (BEC-238 AC3)", () => {
79
+ it("logs warn only once across 7 consecutive missing_scope calls", async () => {
80
+ const PENDING_COUNT = 7;
81
+ for (let i = 0; i < PENDING_COUNT; i++) {
82
+ mockFetch.mockResolvedValueOnce(fetchReturning({ ok: false, error: "missing_scope" }));
83
+ }
84
+ const notifier = new PmSlackNotifier({ botToken: "xoxb-test", channelId: "C0000" });
85
+ for (let i = 0; i < PENDING_COUNT; i++) {
86
+ const result = await notifier.checkApprovalReactions(`ts-${i}`);
87
+ expect(result).toBe("pending");
88
+ }
89
+ // First call logs; subsequent 6 are silently deduped.
90
+ expect(warnSpy).toHaveBeenCalledTimes(1);
91
+ const [ctx, msg] = warnSpy.mock.calls[0];
92
+ expect(ctx?.error).toBe("missing_scope");
93
+ expect(msg).toMatch(/reactions\.get.*ok:false/i);
94
+ });
95
+ it("does NOT dedup non-missing_scope errors — each occurrence is logged", async () => {
96
+ const ERROR_COUNT = 3;
97
+ for (let i = 0; i < ERROR_COUNT; i++) {
98
+ mockFetch.mockResolvedValueOnce(fetchReturning({ ok: false, error: "channel_not_found" }));
99
+ }
100
+ const notifier = new PmSlackNotifier({ botToken: "xoxb-test", channelId: "C0000" });
101
+ for (let i = 0; i < ERROR_COUNT; i++) {
102
+ await notifier.checkApprovalReactions(`ts-${i}`);
103
+ }
104
+ expect(warnSpy).toHaveBeenCalledTimes(ERROR_COUNT);
105
+ });
106
+ it("returns approved / rejected normally when scope is present", async () => {
107
+ mockFetch.mockResolvedValueOnce(fetchReturning({ ok: true, message: { reactions: [{ name: "white_check_mark", count: 1 }] } }));
108
+ mockFetch.mockResolvedValueOnce(fetchReturning({ ok: true, message: { reactions: [{ name: "x", count: 1 }] } }));
109
+ const notifier = new PmSlackNotifier({ botToken: "xoxb-test", channelId: "C0000" });
110
+ expect(await notifier.checkApprovalReactions("ts-approved")).toBe("approved");
111
+ expect(await notifier.checkApprovalReactions("ts-rejected")).toBe("rejected");
112
+ expect(warnSpy).not.toHaveBeenCalled();
113
+ expect(errorSpy).not.toHaveBeenCalled();
114
+ });
115
+ });
116
+ //# sourceMappingURL=bec-238-repro.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"bec-238-repro.test.js","sourceRoot":"","sources":["../../src/__tests__/bec-238-repro.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=bec146-repro.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"bec146-repro.test.d.ts","sourceRoot":"","sources":["../../src/__tests__/bec146-repro.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,153 @@
1
+ /**
2
+ * BEC-146 Regression Test
3
+ *
4
+ * Verifies that the QA retry counter resets correctly after a successful dispatch.
5
+ *
6
+ * Original bug: MAX(attemptCount) returned the old high-water mark even after a
7
+ * successful dispatch wrote attemptCount=0, causing the next failure to falsely
8
+ * escalate to a permanent skip.
9
+ *
10
+ * Fix mechanism (this version): `clearFailureRowsForSha` is called from the
11
+ * `release-tick` dispatch_pending / ok branches BEFORE persisting the reset row.
12
+ * That removes prior failure rows so MAX(attemptCount) naturally returns 0.
13
+ *
14
+ * (An earlier attempted fix used ORDER BY decidedAt DESC LIMIT 1 in the helper,
15
+ * but `crossTimestamp` stores SQLite epoch at SECOND resolution — rapid consecutive
16
+ * ticks tied on decidedAt and the LIMIT 1 query returned a non-deterministic row.)
17
+ */
18
+ import { describe, it, expect, beforeEach, afterEach } from "vitest";
19
+ import { randomBytes } from "node:crypto";
20
+ import { unlinkSync } from "node:fs";
21
+ import { createDb } from "../db/index.js";
22
+ import { releaseDecisions } from "../db/schema.js";
23
+ import { getMaxAttemptCountForReason, clearFailureRowsForSha, } from "../release-manager/release-helpers.js";
24
+ function tmpDbPath() {
25
+ const id = randomBytes(8).toString("hex");
26
+ return `/tmp/bec146-repro-${id}.sqlite`;
27
+ }
28
+ describe("BEC-146 — retry counter resets to 0 on successful dispatch (regression)", () => {
29
+ const paths = [];
30
+ let db;
31
+ const repoUrl = "https://github.com/org/repo";
32
+ const branch = "main";
33
+ const SHA1 = "head_sha_x";
34
+ beforeEach(async () => {
35
+ const path = tmpDbPath();
36
+ paths.push(path);
37
+ const created = await createDb({ driver: "sqlite", connectionString: path });
38
+ db = created;
39
+ });
40
+ afterEach(() => {
41
+ for (const p of paths) {
42
+ try {
43
+ unlinkSync(p);
44
+ }
45
+ catch { }
46
+ try {
47
+ unlinkSync(p + "-wal");
48
+ }
49
+ catch { }
50
+ try {
51
+ unlinkSync(p + "-shm");
52
+ }
53
+ catch { }
54
+ }
55
+ paths.length = 0;
56
+ });
57
+ it("clearFailureRowsForSha removes prior failure rows so MAX(attemptCount) returns 0 after reset", async () => {
58
+ // Seed the DB to represent the scenario:
59
+ // Tick 1: dispatch_error → qa_needs_trigger, attemptCount=1, qaRunId=null
60
+ // Tick 2: dispatch_error → qa_needs_trigger, attemptCount=2, qaRunId=null
61
+ // At tick 3 (dispatch_pending or ok), release-tick.ts calls
62
+ // clearFailureRowsForSha BEFORE persisting the reset row.
63
+ await db.insert(releaseDecisions).values([
64
+ {
65
+ id: "rd_fail1",
66
+ repoUrl,
67
+ branch,
68
+ decidedAt: new Date(Date.now() - 3000),
69
+ decision: "skip",
70
+ reason: "qa_needs_trigger",
71
+ triggerStateJson: "{}",
72
+ qaRunSha: SHA1,
73
+ qaRunId: null,
74
+ attemptCount: 1,
75
+ },
76
+ {
77
+ id: "rd_fail2",
78
+ repoUrl,
79
+ branch,
80
+ decidedAt: new Date(Date.now() - 2000),
81
+ decision: "skip",
82
+ reason: "qa_needs_trigger",
83
+ triggerStateJson: "{}",
84
+ qaRunSha: SHA1,
85
+ qaRunId: null,
86
+ attemptCount: 2,
87
+ },
88
+ ]);
89
+ // Before clear: MAX returns the high-water mark.
90
+ expect(await getMaxAttemptCountForReason(db, repoUrl, branch, "qa_needs_trigger", SHA1)).toBe(2);
91
+ // Simulate tick 3 (dispatch_pending): clear prior failure rows for this SHA.
92
+ await clearFailureRowsForSha(db, repoUrl, branch, "qa_needs_trigger", SHA1);
93
+ // After clear + reset row write, MAX returns 0 (only the reset row remains).
94
+ await db.insert(releaseDecisions).values({
95
+ id: "rd_reset",
96
+ repoUrl,
97
+ branch,
98
+ decidedAt: new Date(Date.now() - 1000),
99
+ decision: "skip",
100
+ reason: "qa_needs_trigger",
101
+ triggerStateJson: "{}",
102
+ qaRunSha: SHA1,
103
+ qaRunId: null, // dispatch_pending — HTTP succeeded but runId not yet visible
104
+ attemptCount: 0,
105
+ });
106
+ const maxCount = await getMaxAttemptCountForReason(db, repoUrl, branch, "qa_needs_trigger", SHA1);
107
+ expect(maxCount).toBe(0);
108
+ });
109
+ it("clearFailureRowsForSha does NOT delete rows with qaRunId set (success rows are preserved)", async () => {
110
+ // Seed a row that represents a SUCCESSFUL dispatch (qaRunId IS NOT NULL).
111
+ // state.ts uses these rows for the latest-QA-run snapshot; they must survive.
112
+ await db.insert(releaseDecisions).values({
113
+ id: "rd_success",
114
+ repoUrl,
115
+ branch,
116
+ decidedAt: new Date(),
117
+ decision: "skip",
118
+ reason: "qa_needs_trigger",
119
+ triggerStateJson: "{}",
120
+ qaRunSha: SHA1,
121
+ qaRunId: 12345,
122
+ attemptCount: 0,
123
+ });
124
+ await clearFailureRowsForSha(db, repoUrl, branch, "qa_needs_trigger", SHA1);
125
+ const rows = await db.select().from(releaseDecisions);
126
+ expect(rows).toHaveLength(1);
127
+ expect(rows[0].id).toBe("rd_success");
128
+ });
129
+ it("subsequent dispatch failure after reset starts attemptCount=1, not 3, no false permanent skip", async () => {
130
+ // Seed: 2 failures, then clear (simulating tick 3 dispatch_pending), then write the reset row.
131
+ await db.insert(releaseDecisions).values([
132
+ { id: "rd_fail1", repoUrl, branch, decidedAt: new Date(Date.now() - 3000), decision: "skip",
133
+ reason: "qa_needs_trigger", triggerStateJson: "{}", qaRunSha: SHA1, qaRunId: null, attemptCount: 1 },
134
+ { id: "rd_fail2", repoUrl, branch, decidedAt: new Date(Date.now() - 2000), decision: "skip",
135
+ reason: "qa_needs_trigger", triggerStateJson: "{}", qaRunSha: SHA1, qaRunId: null, attemptCount: 2 },
136
+ ]);
137
+ await clearFailureRowsForSha(db, repoUrl, branch, "qa_needs_trigger", SHA1);
138
+ await db.insert(releaseDecisions).values({
139
+ id: "rd_reset", repoUrl, branch, decidedAt: new Date(Date.now() - 1000), decision: "skip",
140
+ reason: "qa_needs_trigger", triggerStateJson: "{}", qaRunSha: SHA1, qaRunId: null, attemptCount: 0,
141
+ });
142
+ // Simulate tick 4: dispatch fails again. release-tick computes
143
+ // newAttemptCount = getMaxAttemptCountForReason(...) + 1.
144
+ const prevMax = await getMaxAttemptCountForReason(db, repoUrl, branch, "qa_needs_trigger", SHA1);
145
+ const newAttemptCount = prevMax + 1;
146
+ const MAX_QA_RETRY_ATTEMPTS = 3;
147
+ const wouldBePermanentSkip = newAttemptCount >= MAX_QA_RETRY_ATTEMPTS;
148
+ expect(prevMax).toBe(0);
149
+ expect(newAttemptCount).toBe(1);
150
+ expect(wouldBePermanentSkip).toBe(false);
151
+ });
152
+ });
153
+ //# sourceMappingURL=bec146-repro.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"bec146-repro.test.js","sourceRoot":"","sources":["../../src/__tests__/bec146-repro.test.ts"],"names":[],"mappings":"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"}
@@ -39,11 +39,13 @@ describe("database migrations", () => {
39
39
  it("includes the new missing_indexes migration", () => {
40
40
  const migrations = loadMigrationFiles("sqlite");
41
41
  const migrationNames = migrations.map(m => m.name);
42
- expect(migrationNames).toContain("013_missing_indexes");
42
+ // BEC-149: missing_indexes was renumbered from 013 to 014 to fix a
43
+ // prefix collision with 013_triage_results.
44
+ expect(migrationNames).toContain("014_missing_indexes");
43
45
  });
44
46
  it("migration file contains CREATE INDEX IF NOT EXISTS statements", () => {
45
47
  const migrations = loadMigrationFiles("sqlite");
46
- const missingIndexesMigration = migrations.find(m => m.name === "013_missing_indexes");
48
+ const missingIndexesMigration = migrations.find(m => m.name === "014_missing_indexes");
47
49
  expect(missingIndexesMigration).toBeDefined();
48
50
  expect(missingIndexesMigration.sql).toContain("CREATE INDEX IF NOT EXISTS");
49
51
  expect(missingIndexesMigration.sql).toContain("idx_pipeline_runs_pr_url");
@@ -224,7 +226,9 @@ describe("database migrations", () => {
224
226
  });
225
227
  it("postgres migration file also includes all 5 indexes", async () => {
226
228
  const pgMigrations = loadMigrationFiles("postgres");
227
- const missingIndexesMigration = pgMigrations.find(m => m.name === "014_missing_indexes");
229
+ // BEC-149: missing_indexes was renumbered from 014 to 015 in Postgres to
230
+ // fix a prefix collision with 014_stage_runs_cache_tokens.
231
+ const missingIndexesMigration = pgMigrations.find(m => m.name === "015_missing_indexes");
228
232
  expect(missingIndexesMigration).toBeDefined();
229
233
  expect(missingIndexesMigration.sql).toContain("CREATE INDEX IF NOT EXISTS");
230
234
  expect(missingIndexesMigration.sql).toContain("idx_pipeline_runs_pr_url");