@urateam/core 0.1.31 → 0.1.33
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.
- package/dist/__tests__/agent-stream.test.js +35 -1
- package/dist/__tests__/agent-stream.test.js.map +1 -1
- package/dist/__tests__/audit-immutability.test.js +7 -1
- package/dist/__tests__/audit-immutability.test.js.map +1 -1
- package/dist/__tests__/auth-monitor.test.d.ts +2 -0
- package/dist/__tests__/auth-monitor.test.d.ts.map +1 -0
- package/dist/__tests__/auth-monitor.test.js +253 -0
- package/dist/__tests__/auth-monitor.test.js.map +1 -0
- package/dist/__tests__/bec-186-repro.test.d.ts +16 -0
- package/dist/__tests__/bec-186-repro.test.d.ts.map +1 -0
- package/dist/__tests__/bec-186-repro.test.js +223 -0
- package/dist/__tests__/bec-186-repro.test.js.map +1 -0
- package/dist/__tests__/control-signals.test.d.ts +2 -0
- package/dist/__tests__/control-signals.test.d.ts.map +1 -0
- package/dist/__tests__/control-signals.test.js +77 -0
- package/dist/__tests__/control-signals.test.js.map +1 -0
- package/dist/__tests__/db-migrations.test.d.ts +2 -0
- package/dist/__tests__/db-migrations.test.d.ts.map +1 -0
- package/dist/__tests__/db-migrations.test.js +237 -0
- package/dist/__tests__/db-migrations.test.js.map +1 -0
- package/dist/__tests__/executor-issue-id.test.js +2 -0
- package/dist/__tests__/executor-issue-id.test.js.map +1 -1
- package/dist/__tests__/pm-slack-interface.test.js +45 -0
- package/dist/__tests__/pm-slack-interface.test.js.map +1 -1
- package/dist/__tests__/pm-triage.test.js +101 -0
- package/dist/__tests__/pm-triage.test.js.map +1 -1
- package/dist/__tests__/post-fanout-comments.test.js +36 -0
- package/dist/__tests__/post-fanout-comments.test.js.map +1 -1
- package/dist/__tests__/preflight-claude-auth.test.d.ts +2 -0
- package/dist/__tests__/preflight-claude-auth.test.d.ts.map +1 -0
- package/dist/__tests__/preflight-claude-auth.test.js +36 -0
- package/dist/__tests__/preflight-claude-auth.test.js.map +1 -0
- package/dist/__tests__/resolve-claude-auth.test.d.ts +2 -0
- package/dist/__tests__/resolve-claude-auth.test.d.ts.map +1 -0
- package/dist/__tests__/resolve-claude-auth.test.js +129 -0
- package/dist/__tests__/resolve-claude-auth.test.js.map +1 -0
- package/dist/__tests__/stage-models.test.js +4 -0
- package/dist/__tests__/stage-models.test.js.map +1 -1
- package/dist/__tests__/util-linear.test.d.ts +10 -0
- package/dist/__tests__/util-linear.test.d.ts.map +1 -0
- package/dist/__tests__/util-linear.test.js +244 -0
- package/dist/__tests__/util-linear.test.js.map +1 -0
- package/dist/audit/events.d.ts +37 -0
- package/dist/audit/events.d.ts.map +1 -1
- package/dist/audit/events.js +53 -0
- package/dist/audit/events.js.map +1 -1
- package/dist/db/client.d.ts.map +1 -1
- package/dist/db/client.js +8 -0
- package/dist/db/client.js.map +1 -1
- package/dist/db/migrations/postgres/014_missing_indexes.sql +28 -0
- package/dist/db/migrations/sqlite/013_missing_indexes.sql +28 -0
- package/dist/executor/agent-stream.d.ts +16 -0
- package/dist/executor/agent-stream.d.ts.map +1 -1
- package/dist/executor/agent-stream.js +43 -1
- package/dist/executor/agent-stream.js.map +1 -1
- package/dist/executor/auth-check.d.ts +39 -0
- package/dist/executor/auth-check.d.ts.map +1 -1
- package/dist/executor/auth-check.js +31 -0
- package/dist/executor/auth-check.js.map +1 -1
- package/dist/executor/auth-monitor.d.ts +40 -0
- package/dist/executor/auth-monitor.d.ts.map +1 -0
- package/dist/executor/auth-monitor.js +114 -0
- package/dist/executor/auth-monitor.js.map +1 -0
- package/dist/executor/executor.d.ts.map +1 -1
- package/dist/executor/executor.js +7 -2
- package/dist/executor/executor.js.map +1 -1
- package/dist/executor/index.d.ts +2 -0
- package/dist/executor/index.d.ts.map +1 -1
- package/dist/executor/index.js +2 -0
- package/dist/executor/index.js.map +1 -1
- package/dist/executor/review/post-fanout-comments.d.ts +8 -0
- package/dist/executor/review/post-fanout-comments.d.ts.map +1 -1
- package/dist/executor/review/post-fanout-comments.js +23 -3
- package/dist/executor/review/post-fanout-comments.js.map +1 -1
- package/dist/index.d.ts +3 -1
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +3 -1
- package/dist/index.js.map +1 -1
- package/dist/notifier/composite.d.ts +1 -0
- package/dist/notifier/composite.d.ts.map +1 -1
- package/dist/notifier/composite.js +3 -0
- package/dist/notifier/composite.js.map +1 -1
- package/dist/notifier/linear.d.ts +1 -0
- package/dist/notifier/linear.d.ts.map +1 -1
- package/dist/notifier/linear.js +7 -0
- package/dist/notifier/linear.js.map +1 -1
- package/dist/pipeline/control-signals.d.ts +49 -0
- package/dist/pipeline/control-signals.d.ts.map +1 -0
- package/dist/pipeline/control-signals.js +93 -0
- package/dist/pipeline/control-signals.js.map +1 -0
- package/dist/pipeline/feedback-pipeline.d.ts +140 -0
- package/dist/pipeline/feedback-pipeline.d.ts.map +1 -0
- package/dist/pipeline/feedback-pipeline.js +427 -0
- package/dist/pipeline/feedback-pipeline.js.map +1 -0
- package/dist/pipeline/index.d.ts +1 -0
- package/dist/pipeline/index.d.ts.map +1 -1
- package/dist/pipeline/index.js +1 -0
- package/dist/pipeline/index.js.map +1 -1
- package/dist/pipeline/runner.d.ts +49 -33
- package/dist/pipeline/runner.d.ts.map +1 -1
- package/dist/pipeline/runner.js +143 -350
- package/dist/pipeline/runner.js.map +1 -1
- package/dist/pm/actions/promote.d.ts.map +1 -1
- package/dist/pm/actions/promote.js +15 -11
- package/dist/pm/actions/promote.js.map +1 -1
- package/dist/pm/actions/recover-stuck.d.ts.map +1 -1
- package/dist/pm/actions/recover-stuck.js +9 -5
- package/dist/pm/actions/recover-stuck.js.map +1 -1
- package/dist/pm/actions/triage.d.ts.map +1 -1
- package/dist/pm/actions/triage.js +67 -1
- package/dist/pm/actions/triage.js.map +1 -1
- package/dist/pm/linear-helpers.d.ts +10 -0
- package/dist/pm/linear-helpers.d.ts.map +1 -1
- package/dist/pm/linear-helpers.js +13 -21
- package/dist/pm/linear-helpers.js.map +1 -1
- package/dist/pm/pause-state.d.ts +29 -0
- package/dist/pm/pause-state.d.ts.map +1 -0
- package/dist/pm/pause-state.js +34 -0
- package/dist/pm/pause-state.js.map +1 -0
- package/dist/pm/scheduler.d.ts.map +1 -1
- package/dist/pm/scheduler.js +19 -0
- package/dist/pm/scheduler.js.map +1 -1
- package/dist/pm/slack-bulk.d.ts +34 -0
- package/dist/pm/slack-bulk.d.ts.map +1 -0
- package/dist/pm/slack-bulk.js +110 -0
- package/dist/pm/slack-bulk.js.map +1 -0
- package/dist/pm/slack-commands.d.ts +101 -0
- package/dist/pm/slack-commands.d.ts.map +1 -0
- package/dist/pm/slack-commands.js +309 -0
- package/dist/pm/slack-commands.js.map +1 -0
- package/dist/pm/slack-helpers.d.ts +10 -0
- package/dist/pm/slack-helpers.d.ts.map +1 -1
- package/dist/pm/slack-helpers.js +32 -0
- package/dist/pm/slack-helpers.js.map +1 -1
- package/dist/pm/slack-interface.d.ts +32 -58
- package/dist/pm/slack-interface.d.ts.map +1 -1
- package/dist/pm/slack-interface.js +150 -320
- package/dist/pm/slack-interface.js.map +1 -1
- package/dist/rbac/matrix.d.ts +2 -0
- package/dist/rbac/matrix.d.ts.map +1 -1
- package/dist/rbac/matrix.js +2 -0
- package/dist/rbac/matrix.js.map +1 -1
- package/dist/release-manager/index.d.ts +2 -0
- package/dist/release-manager/index.d.ts.map +1 -1
- package/dist/release-manager/index.js +2 -0
- package/dist/release-manager/index.js.map +1 -1
- package/dist/release-manager/release-helpers.d.ts +112 -0
- package/dist/release-manager/release-helpers.d.ts.map +1 -0
- package/dist/release-manager/release-helpers.js +164 -0
- package/dist/release-manager/release-helpers.js.map +1 -0
- package/dist/release-manager/release-tick.d.ts +101 -0
- package/dist/release-manager/release-tick.d.ts.map +1 -0
- package/dist/release-manager/release-tick.js +374 -0
- package/dist/release-manager/release-tick.js.map +1 -0
- package/dist/release-manager/scheduler.d.ts +28 -3
- package/dist/release-manager/scheduler.d.ts.map +1 -1
- package/dist/release-manager/scheduler.js +41 -417
- package/dist/release-manager/scheduler.js.map +1 -1
- package/dist/server.d.ts.map +1 -1
- package/dist/server.js +10 -0
- package/dist/server.js.map +1 -1
- package/dist/sync/gh-linear-sync.d.ts.map +1 -1
- package/dist/sync/gh-linear-sync.js +2 -2
- package/dist/sync/gh-linear-sync.js.map +1 -1
- package/dist/types.d.ts +22 -4
- package/dist/types.d.ts.map +1 -1
- package/dist/types.js +10 -0
- package/dist/types.js.map +1 -1
- package/dist/util/linear.d.ts +70 -0
- package/dist/util/linear.d.ts.map +1 -0
- package/dist/util/linear.js +108 -0
- package/dist/util/linear.js.map +1 -0
- package/dist/webhook/github-handler.d.ts +7 -1
- package/dist/webhook/github-handler.d.ts.map +1 -1
- package/dist/webhook/github-handler.js +82 -38
- package/dist/webhook/github-handler.js.map +1 -1
- package/package.json +1 -1
|
@@ -0,0 +1,374 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* release-tick.ts
|
|
3
|
+
*
|
|
4
|
+
* Responsibility: the release-manager decision cycle (QA trigger logic, approval
|
|
5
|
+
* gate, version bump, Git tag creation, and cron rescheduling state).
|
|
6
|
+
*
|
|
7
|
+
* The single exported function `tick(ctx)` receives all dependencies and mutable
|
|
8
|
+
* inter-tick state through an explicit `TickContext` struct instead of closing
|
|
9
|
+
* over variables in the scheduler factory. This makes the function independently
|
|
10
|
+
* testable and removes the 8-variable closure from scheduler.ts.
|
|
11
|
+
*
|
|
12
|
+
* Exports:
|
|
13
|
+
* - TickMutableState — per-instance state that persists between ticks
|
|
14
|
+
* - TickContext — all deps + mutable state for one tick invocation
|
|
15
|
+
* - tick — execute a single release-manager decision cycle
|
|
16
|
+
*/
|
|
17
|
+
import { randomUUID } from "node:crypto";
|
|
18
|
+
import { createLogger } from "../logger.js";
|
|
19
|
+
import { logAuditEventUnchecked } from "../audit/writer.js";
|
|
20
|
+
import { releaseFiredEvent, releaseSkippedEvent, releaseTagConflictEvent, releasePartialEvent, qaRunCompletedEvent, } from "../audit/events.js";
|
|
21
|
+
import { collectState } from "./state.js";
|
|
22
|
+
import { decide } from "./decide.js";
|
|
23
|
+
import { bumpFromConfigAndCommits } from "./versioning.js";
|
|
24
|
+
import { createTagAndRelease, parseRepoFromUrl } from "./github.js";
|
|
25
|
+
import { triggerWorkflow, pollWorkflowRun, workflowFileExists } from "../qa/github.js";
|
|
26
|
+
import { markGapResolved } from "../qa/gap.js";
|
|
27
|
+
import { maybePostSlack, persistDecision, consumeApprovalRow, getMaxAttemptCountForReason, tryFileQaGapIssue, MAX_QA_RETRY_ATTEMPTS, } from "./release-helpers.js";
|
|
28
|
+
const log = createLogger({ component: "ReleaseManager:scheduler" });
|
|
29
|
+
/**
|
|
30
|
+
* Maximum number of completed QA run IDs to track in the dedup set before
|
|
31
|
+
* evicting stale entries. Once this threshold is reached the set is cleared;
|
|
32
|
+
* a cleared entry may produce a single duplicate `qa.run_completed` audit
|
|
33
|
+
* event, which is acceptable (far better than unbounded memory growth in
|
|
34
|
+
* long-running deployments).
|
|
35
|
+
*/
|
|
36
|
+
const MAX_AUDITED_RUN_IDS = 10_000;
|
|
37
|
+
/** Compute the approval TTL in milliseconds from config, defaulting to 24 hours. */
|
|
38
|
+
function approvalTtlMs(config) {
|
|
39
|
+
const hours = config.triggers.timeSinceLastHours;
|
|
40
|
+
if (hours && hours > 0)
|
|
41
|
+
return hours * 3600 * 1000;
|
|
42
|
+
return 24 * 3600 * 1000;
|
|
43
|
+
}
|
|
44
|
+
/**
|
|
45
|
+
* Execute a single release-manager decision cycle.
|
|
46
|
+
*
|
|
47
|
+
* Orchestrates the full state machine:
|
|
48
|
+
* 1. License check and pause gate.
|
|
49
|
+
* 2. Collect world state (branch HEAD, tags, CI, approvals, QA run).
|
|
50
|
+
* 3. Manual-tag detection (re-baseline).
|
|
51
|
+
* 4. QA workflow check — trigger, poll, or file gap issue as needed.
|
|
52
|
+
* 5. Decide (skip / awaiting-approval / fire).
|
|
53
|
+
* 6. On skip: persist decision, emit audit event, post Slack with dedup.
|
|
54
|
+
* 7. On awaiting-approval: persist decision, post Slack prompt.
|
|
55
|
+
* 8. On fire: create Git tag + GitHub release, persist decision, post Slack.
|
|
56
|
+
*
|
|
57
|
+
* All dependencies arrive through `ctx`; no closure variables are read or written.
|
|
58
|
+
* Mutable inter-tick state is stored in `ctx.mutableState` and updated in-place.
|
|
59
|
+
*
|
|
60
|
+
* @param ctx - All dependencies and mutable state for this tick invocation.
|
|
61
|
+
*/
|
|
62
|
+
export async function tick(ctx) {
|
|
63
|
+
const { config, db, octokit, linear, repoUrl, branch, isLicensed, slack, mutableState } = ctx;
|
|
64
|
+
const slackChannel = config.slackChannel;
|
|
65
|
+
if (!isLicensed()) {
|
|
66
|
+
if (!mutableState.licenseWarnLogged) {
|
|
67
|
+
log.warn({ repoUrl, branch }, "release-manager unlicensed — skipping ticks");
|
|
68
|
+
mutableState.licenseWarnLogged = true;
|
|
69
|
+
}
|
|
70
|
+
return;
|
|
71
|
+
}
|
|
72
|
+
if (Date.now() < mutableState.pausedUntilTs) {
|
|
73
|
+
log.info({ pausedUntilTs: mutableState.pausedUntilTs }, "scheduler paused (via /release skip) — skipping tick");
|
|
74
|
+
return;
|
|
75
|
+
}
|
|
76
|
+
let state;
|
|
77
|
+
try {
|
|
78
|
+
state = await collectState({
|
|
79
|
+
octokit, db, repoUrl, branch, approvalTtlMs: approvalTtlMs(config),
|
|
80
|
+
});
|
|
81
|
+
}
|
|
82
|
+
catch (err) {
|
|
83
|
+
log.error({ err, repoUrl, branch }, "collectState failed — skipping tick");
|
|
84
|
+
return;
|
|
85
|
+
}
|
|
86
|
+
const triggerStateJson = JSON.stringify({
|
|
87
|
+
mergedCommitsSinceLastTag: state.mergedCommitsSinceLastTag,
|
|
88
|
+
lastTag: state.lastTag,
|
|
89
|
+
lastTagAt: state.lastTagAt?.toISOString() ?? null,
|
|
90
|
+
ciStatus: state.ciStatus,
|
|
91
|
+
hasFreshApproval: state.hasFreshApproval,
|
|
92
|
+
});
|
|
93
|
+
// 1. Manual-tag detection — re-baseline counters.
|
|
94
|
+
if (state.manualTagDetected) {
|
|
95
|
+
const id = `rd_${randomUUID()}`;
|
|
96
|
+
await persistDecision(db, repoUrl, branch, {
|
|
97
|
+
id,
|
|
98
|
+
decision: "skip",
|
|
99
|
+
reason: "manual_tag_detected",
|
|
100
|
+
triggerStateJson,
|
|
101
|
+
});
|
|
102
|
+
void logAuditEventUnchecked(db, releaseSkippedEvent({ repoUrl, branch, reason: "manual_tag_detected" }));
|
|
103
|
+
log.info({ repoUrl, branch }, "manual tag detected — re-baselining");
|
|
104
|
+
return;
|
|
105
|
+
}
|
|
106
|
+
// 2. BEC-136: Compute QA state when qaCheck is configured.
|
|
107
|
+
let qaState;
|
|
108
|
+
if (config.triggers.qaCheck) {
|
|
109
|
+
const { owner, repo } = parseRepoFromUrl(repoUrl);
|
|
110
|
+
let wfExists = false;
|
|
111
|
+
try {
|
|
112
|
+
wfExists = await workflowFileExists({
|
|
113
|
+
octokit, owner, repo,
|
|
114
|
+
path: config.triggers.qaCheck.workflow,
|
|
115
|
+
ref: state.headSha,
|
|
116
|
+
});
|
|
117
|
+
}
|
|
118
|
+
catch (err) {
|
|
119
|
+
log.warn({ err }, "qa workflowFileExists check failed — treating as exists");
|
|
120
|
+
wfExists = true; // fail-open so retries hit the dispatch path
|
|
121
|
+
}
|
|
122
|
+
if (wfExists) {
|
|
123
|
+
// BEC-136: workflow file is present; if there was an open gap issue for this
|
|
124
|
+
// (repo, branch, workflow), mark it resolved so a future gap can be re-filed.
|
|
125
|
+
await markGapResolved({
|
|
126
|
+
db,
|
|
127
|
+
repoUrl,
|
|
128
|
+
branch,
|
|
129
|
+
workflowPath: config.triggers.qaCheck.workflow,
|
|
130
|
+
});
|
|
131
|
+
}
|
|
132
|
+
let runConclusion = null;
|
|
133
|
+
if (state.qaRun && state.qaRun.runSha === state.headSha) {
|
|
134
|
+
try {
|
|
135
|
+
const polled = await pollWorkflowRun({ octokit, owner, repo, runId: state.qaRun.runId });
|
|
136
|
+
if (polled.kind === "completed") {
|
|
137
|
+
runConclusion = polled.conclusion;
|
|
138
|
+
if (!mutableState.auditedCompletedRunIds.has(state.qaRun.runId)) {
|
|
139
|
+
// BEC-196: bounded-memory eviction. The pre-split scheduler held
|
|
140
|
+
// this set unbounded; the split is the right moment to fix it
|
|
141
|
+
// because state is now passed explicitly via TickMutableState.
|
|
142
|
+
// After eviction a single QA run id can be re-audited once
|
|
143
|
+
// (duplicate `qa.run_completed` event) — acceptable in exchange
|
|
144
|
+
// for bounded memory. The 10k cap is high enough that hitting it
|
|
145
|
+
// requires months of continuous QA runs.
|
|
146
|
+
if (mutableState.auditedCompletedRunIds.size >= MAX_AUDITED_RUN_IDS) {
|
|
147
|
+
mutableState.auditedCompletedRunIds.clear();
|
|
148
|
+
}
|
|
149
|
+
mutableState.auditedCompletedRunIds.add(state.qaRun.runId);
|
|
150
|
+
// Emit qa.run_completed audit on first observation of completion.
|
|
151
|
+
void logAuditEventUnchecked(db, qaRunCompletedEvent({
|
|
152
|
+
repoUrl, branch,
|
|
153
|
+
runId: state.qaRun.runId,
|
|
154
|
+
conclusion: polled.conclusion,
|
|
155
|
+
durationMs: polled.durationMs,
|
|
156
|
+
}));
|
|
157
|
+
}
|
|
158
|
+
// When already in the set: runConclusion is still set above; audit skipped.
|
|
159
|
+
}
|
|
160
|
+
}
|
|
161
|
+
catch (err) {
|
|
162
|
+
log.warn({ err }, "qa pollWorkflowRun failed — treating as still running");
|
|
163
|
+
}
|
|
164
|
+
}
|
|
165
|
+
qaState = { workflowFileExists: wfExists, runConclusion };
|
|
166
|
+
}
|
|
167
|
+
// 3. Decision.
|
|
168
|
+
const result = decide(state, config.triggers, undefined, qaState);
|
|
169
|
+
const proposedVersion = bumpFromConfigAndCommits(state.lastTag, state.commitsSinceLastTag, config.versionBump);
|
|
170
|
+
if (result.kind === "skip") {
|
|
171
|
+
const id = `rd_${randomUUID()}`;
|
|
172
|
+
// Compute attempt count for retry handling on qa_dispatch_error path.
|
|
173
|
+
let attemptCount = 0;
|
|
174
|
+
let qaRunId;
|
|
175
|
+
let qaRunSha;
|
|
176
|
+
let finalReason = result.reason;
|
|
177
|
+
if (result.qaActionNeeded?.reason === "qa_needs_trigger") {
|
|
178
|
+
// Look up the highest attempt count across all qa_needs_trigger rows for this
|
|
179
|
+
// (branch, sha) pair. Using MAX instead of ORDER BY + LIMIT 1 to be stable
|
|
180
|
+
// when multiple rows share the same decidedAt timestamp.
|
|
181
|
+
attemptCount = await getMaxAttemptCountForReason(db, repoUrl, branch, "qa_needs_trigger", state.headSha);
|
|
182
|
+
const { owner, repo } = parseRepoFromUrl(repoUrl);
|
|
183
|
+
const dispatch = await triggerWorkflow({
|
|
184
|
+
octokit, db, owner, repo, repoUrl, branch,
|
|
185
|
+
workflow: config.triggers.qaCheck.workflow,
|
|
186
|
+
ref: state.headSha,
|
|
187
|
+
inputs: config.triggers.qaCheck.workflowInputs,
|
|
188
|
+
});
|
|
189
|
+
if (dispatch.kind === "ok") {
|
|
190
|
+
attemptCount = 0; // reset on successful dispatch
|
|
191
|
+
qaRunId = dispatch.runId;
|
|
192
|
+
qaRunSha = state.headSha;
|
|
193
|
+
}
|
|
194
|
+
else if (dispatch.kind === "dispatch_404") {
|
|
195
|
+
// Workflow disappeared between state cache and dispatch — drop into gap-issue path.
|
|
196
|
+
finalReason = "qa_no_workflow";
|
|
197
|
+
if (linear) {
|
|
198
|
+
const gapResult = await tryFileQaGapIssue({
|
|
199
|
+
db, linear, repoUrl, branch,
|
|
200
|
+
workflowPath: config.triggers.qaCheck.workflow,
|
|
201
|
+
linearTeamId: config.triggers.qaCheck.linearTeamId,
|
|
202
|
+
});
|
|
203
|
+
finalReason = gapResult.finalReason;
|
|
204
|
+
attemptCount = gapResult.attemptCount;
|
|
205
|
+
}
|
|
206
|
+
else {
|
|
207
|
+
log.error({ repoUrl, branch }, "qaCheck requires Linear client but none configured — skipping gap-issue file");
|
|
208
|
+
}
|
|
209
|
+
}
|
|
210
|
+
else if (dispatch.kind === "dispatch_422") {
|
|
211
|
+
finalReason = "qa_dispatch_error";
|
|
212
|
+
attemptCount = 99; // permanent skip — workflow misconfigured, retrying won't help
|
|
213
|
+
}
|
|
214
|
+
else if (dispatch.kind === "dispatch_pending") {
|
|
215
|
+
// GitHub eventual-consistency window. Don't count against retry budget; next tick
|
|
216
|
+
// will re-evaluate and the run should be findable by then.
|
|
217
|
+
// Tag with qaRunSha so the per-SHA retry counter query can find these rows.
|
|
218
|
+
qaRunSha = state.headSha;
|
|
219
|
+
// attemptCount stays as-is; finalReason stays as "qa_needs_trigger" for next tick to retry.
|
|
220
|
+
}
|
|
221
|
+
else {
|
|
222
|
+
// dispatch_error — increment attempt counter.
|
|
223
|
+
// Tag with qaRunSha so the per-SHA retry counter query can find these rows.
|
|
224
|
+
qaRunSha = state.headSha;
|
|
225
|
+
attemptCount += 1;
|
|
226
|
+
if (attemptCount >= MAX_QA_RETRY_ATTEMPTS) {
|
|
227
|
+
finalReason = "qa_dispatch_error";
|
|
228
|
+
}
|
|
229
|
+
}
|
|
230
|
+
}
|
|
231
|
+
else if (result.qaActionNeeded?.reason === "qa_no_workflow") {
|
|
232
|
+
if (linear) {
|
|
233
|
+
const gapResult = await tryFileQaGapIssue({
|
|
234
|
+
db, linear, repoUrl, branch,
|
|
235
|
+
workflowPath: config.triggers.qaCheck.workflow,
|
|
236
|
+
linearTeamId: config.triggers.qaCheck.linearTeamId,
|
|
237
|
+
});
|
|
238
|
+
finalReason = gapResult.finalReason;
|
|
239
|
+
attemptCount = gapResult.attemptCount;
|
|
240
|
+
}
|
|
241
|
+
else {
|
|
242
|
+
log.error({ repoUrl, branch }, "qaCheck requires Linear client but none configured — skipping gap-issue file");
|
|
243
|
+
}
|
|
244
|
+
}
|
|
245
|
+
if (result.qaActionNeeded?.reason === "qa_timed_out" && !result.qaActionNeeded.pass && state.qaRun) {
|
|
246
|
+
const elapsedMs = Date.now() - state.qaRun.triggeredAt.getTime();
|
|
247
|
+
void logAuditEventUnchecked(db, qaRunCompletedEvent({
|
|
248
|
+
repoUrl,
|
|
249
|
+
branch,
|
|
250
|
+
runId: result.qaActionNeeded.runId,
|
|
251
|
+
conclusion: "timed_out",
|
|
252
|
+
durationMs: elapsedMs,
|
|
253
|
+
synthetic: true,
|
|
254
|
+
}));
|
|
255
|
+
}
|
|
256
|
+
await persistDecision(db, repoUrl, branch, {
|
|
257
|
+
id,
|
|
258
|
+
decision: "skip",
|
|
259
|
+
reason: finalReason,
|
|
260
|
+
triggerStateJson,
|
|
261
|
+
proposedVersion,
|
|
262
|
+
qaRunId,
|
|
263
|
+
qaRunSha,
|
|
264
|
+
attemptCount,
|
|
265
|
+
});
|
|
266
|
+
void logAuditEventUnchecked(db, releaseSkippedEvent({ repoUrl, branch, reason: finalReason }));
|
|
267
|
+
// BEC-160: emit a stdout line for every skip so operators tailing
|
|
268
|
+
// docker logs can see the scheduler is alive and why it's skipping.
|
|
269
|
+
log.info({ repoUrl, branch, reason: finalReason, mergedCommitsSinceLastTag: state.mergedCommitsSinceLastTag, lastTag: state.lastTag, proposedVersion }, "tick skip");
|
|
270
|
+
// Slack notification with dedup
|
|
271
|
+
await maybePostSlack(slack, slackChannel, db, mutableState.slackDedup, `:double_vertical_bar: Release skipped for *${repoUrl}* (${branch}): ${finalReason}`, finalReason);
|
|
272
|
+
return;
|
|
273
|
+
}
|
|
274
|
+
if (result.kind === "awaiting-approval") {
|
|
275
|
+
const id = `rd_${randomUUID()}`;
|
|
276
|
+
await persistDecision(db, repoUrl, branch, {
|
|
277
|
+
id,
|
|
278
|
+
decision: "awaiting-approval",
|
|
279
|
+
reason: result.reason,
|
|
280
|
+
triggerStateJson,
|
|
281
|
+
proposedVersion,
|
|
282
|
+
});
|
|
283
|
+
void logAuditEventUnchecked(db, releaseSkippedEvent({ repoUrl, branch, reason: "awaiting-approval" }));
|
|
284
|
+
// BEC-160: stdout visibility for the awaiting-approval skip path.
|
|
285
|
+
log.info({ repoUrl, branch, reason: "awaiting-approval", proposedVersion }, "tick skip");
|
|
286
|
+
// Always post on first transition to awaiting-approval (bypass dedup).
|
|
287
|
+
// Reset lastSkipReason so a subsequent regular-skip will re-post.
|
|
288
|
+
mutableState.slackDedup.lastSkipReason = null;
|
|
289
|
+
await maybePostSlack(slack, slackChannel, db, mutableState.slackDedup, `:hourglass_flowing_sand: Release ready for *${repoUrl}* (${branch}): bumping ${proposedVersion} (${state.mergedCommitsSinceLastTag} commits since last tag). Run \`/release approve\` to fire.`, null);
|
|
290
|
+
return;
|
|
291
|
+
}
|
|
292
|
+
// Fire — create tag + release.
|
|
293
|
+
const id = `rd_${randomUUID()}`;
|
|
294
|
+
const githubResult = await createTagAndRelease({
|
|
295
|
+
octokit,
|
|
296
|
+
...parseRepoFromUrl(repoUrl),
|
|
297
|
+
tag: proposedVersion,
|
|
298
|
+
sha: state.headSha,
|
|
299
|
+
});
|
|
300
|
+
if (githubResult.kind === "tag_exists") {
|
|
301
|
+
await persistDecision(db, repoUrl, branch, {
|
|
302
|
+
id,
|
|
303
|
+
decision: "skip",
|
|
304
|
+
reason: "tag_exists",
|
|
305
|
+
triggerStateJson,
|
|
306
|
+
proposedVersion,
|
|
307
|
+
});
|
|
308
|
+
void logAuditEventUnchecked(db, releaseTagConflictEvent({ repoUrl, branch, tag: proposedVersion }));
|
|
309
|
+
// BEC-160: stdout visibility for the tag-exists skip path.
|
|
310
|
+
log.info({ repoUrl, branch, reason: "tag_exists", proposedVersion }, "tick skip");
|
|
311
|
+
return;
|
|
312
|
+
}
|
|
313
|
+
if (githubResult.kind === "release_create_failed") {
|
|
314
|
+
// Tag was created; release-creation failed. Write a single skip row with the
|
|
315
|
+
// partial-fire details so an operator can see what happened and clean up the
|
|
316
|
+
// orphaned tag manually. v1 does NOT retry release-creation across ticks
|
|
317
|
+
// (the tag is now committed, so the next tick would hit `tag_exists` and
|
|
318
|
+
// skip again — see plan §"Known v1 simplifications"). Proper retry is a v2
|
|
319
|
+
// feature requiring a tick-start sweep that calls only `createRelease` for
|
|
320
|
+
// matching fire-pending rows.
|
|
321
|
+
await persistDecision(db, repoUrl, branch, {
|
|
322
|
+
id,
|
|
323
|
+
decision: "skip",
|
|
324
|
+
reason: "release_create_failed",
|
|
325
|
+
triggerStateJson,
|
|
326
|
+
proposedVersion,
|
|
327
|
+
firedTag: proposedVersion,
|
|
328
|
+
firedSha: state.headSha,
|
|
329
|
+
});
|
|
330
|
+
void logAuditEventUnchecked(db, releasePartialEvent({ repoUrl, branch, tag: proposedVersion, attemptCount: 1 }));
|
|
331
|
+
log.error({ repoUrl, branch, tag: proposedVersion, msg: githubResult.message }, "release create failed — tag exists in GitHub but release page not created; manual cleanup required");
|
|
332
|
+
return;
|
|
333
|
+
}
|
|
334
|
+
if (githubResult.kind === "other_error") {
|
|
335
|
+
await persistDecision(db, repoUrl, branch, {
|
|
336
|
+
id,
|
|
337
|
+
decision: "skip",
|
|
338
|
+
reason: "tag_create_error",
|
|
339
|
+
triggerStateJson,
|
|
340
|
+
proposedVersion,
|
|
341
|
+
});
|
|
342
|
+
void logAuditEventUnchecked(db, releaseSkippedEvent({ repoUrl, branch, reason: "tag_create_error" }));
|
|
343
|
+
log.error({ err: githubResult.message, repoUrl, branch }, "createTagAndRelease unknown error — wrote skip row");
|
|
344
|
+
return;
|
|
345
|
+
}
|
|
346
|
+
// ok — fire succeeded.
|
|
347
|
+
await persistDecision(db, repoUrl, branch, {
|
|
348
|
+
id,
|
|
349
|
+
decision: "fire",
|
|
350
|
+
reason: "all triggers passed",
|
|
351
|
+
triggerStateJson,
|
|
352
|
+
proposedVersion,
|
|
353
|
+
firedTag: proposedVersion,
|
|
354
|
+
firedSha: state.headSha,
|
|
355
|
+
});
|
|
356
|
+
if (state.hasFreshApproval) {
|
|
357
|
+
await consumeApprovalRow(db, repoUrl, branch, id);
|
|
358
|
+
}
|
|
359
|
+
void logAuditEventUnchecked(db, releaseFiredEvent({
|
|
360
|
+
repoUrl,
|
|
361
|
+
branch,
|
|
362
|
+
tag: proposedVersion,
|
|
363
|
+
sha: state.headSha,
|
|
364
|
+
mergedPrCount: state.mergedCommitsSinceLastTag,
|
|
365
|
+
}));
|
|
366
|
+
// BEC-160: stdout visibility for the fire path. A successful release is a
|
|
367
|
+
// production event — operators must see it in `docker logs`, not just the
|
|
368
|
+
// audit table. Slack-suppressed deployments would otherwise be silent.
|
|
369
|
+
log.info({ repoUrl, branch, tag: proposedVersion, sha: state.headSha, mergedPrCount: state.mergedCommitsSinceLastTag, releaseUrl: githubResult.releaseUrl }, "tick fire");
|
|
370
|
+
await maybePostSlack(slack, slackChannel, db, mutableState.slackDedup, `:rocket: Released *${proposedVersion}* for ${repoUrl} (${branch}). ${githubResult.releaseUrl}`, null);
|
|
371
|
+
// Reset Slack dedup so the next skip re-posts.
|
|
372
|
+
mutableState.slackDedup.lastSkipReason = null;
|
|
373
|
+
}
|
|
374
|
+
//# sourceMappingURL=release-tick.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"release-tick.js","sourceRoot":"","sources":["../../src/release-manager/release-tick.ts"],"names":[],"mappings":"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"}
|
|
@@ -1,10 +1,23 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* scheduler.ts
|
|
3
|
+
*
|
|
4
|
+
* Responsibility: cron factory — wires dependencies into a TickContext, constructs
|
|
5
|
+
* the mutable per-instance state, and exposes the public
|
|
6
|
+
* `createReleaseManagerScheduler` entry point.
|
|
7
|
+
*
|
|
8
|
+
* File responsibilities in this module:
|
|
9
|
+
* - scheduler.ts : cron factory + public surface (this file)
|
|
10
|
+
* - release-tick.ts : tick() orchestration — QA trigger logic, approval gate,
|
|
11
|
+
* version bump, Git tag, cron rescheduling state
|
|
12
|
+
* - release-helpers.ts: persistence + QA helpers — Slack dedup, persistDecision,
|
|
13
|
+
* consumeApprovalRow, getMaxAttemptCountForReason,
|
|
14
|
+
* tryFileQaGapIssue, MAX_QA_RETRY_ATTEMPTS
|
|
15
|
+
*/
|
|
1
16
|
import type { Octokit } from "@octokit/rest";
|
|
2
17
|
import type { LinearClient } from "@linear/sdk";
|
|
3
18
|
import type { AnyDb } from "../db/client.js";
|
|
4
19
|
import type { ReleaseManagerConfig } from "./types.js";
|
|
5
|
-
|
|
6
|
-
postMessage: (channel: string, text: string) => Promise<boolean>;
|
|
7
|
-
}
|
|
20
|
+
import type { SlackPoster } from "./release-helpers.js";
|
|
8
21
|
export interface ReleaseManagerSchedulerInput {
|
|
9
22
|
config: ReleaseManagerConfig;
|
|
10
23
|
db: AnyDb;
|
|
@@ -26,5 +39,17 @@ export interface ReleaseManagerScheduler {
|
|
|
26
39
|
/** /release skip → pause future ticks until this timestamp. */
|
|
27
40
|
pauseUntil(ts: Date): void;
|
|
28
41
|
}
|
|
42
|
+
/**
|
|
43
|
+
* Create a release-manager scheduler for a single (repo, branch) pair.
|
|
44
|
+
*
|
|
45
|
+
* The returned object is the sole public API for the release-manager module.
|
|
46
|
+
* Callers should call `start()` to enable the cron driver, or call `tick()`
|
|
47
|
+
* directly in tests.
|
|
48
|
+
*
|
|
49
|
+
* Internally this function constructs a `TickContext` (defined in release-tick.ts)
|
|
50
|
+
* and delegates every decision cycle to `tick(ctx)`. Mutable inter-tick state
|
|
51
|
+
* (Slack dedup counters, license-warn flag, pause timestamp, etc.) lives in
|
|
52
|
+
* `ctx.mutableState` and is updated in-place by each tick invocation.
|
|
53
|
+
*/
|
|
29
54
|
export declare function createReleaseManagerScheduler(input: ReleaseManagerSchedulerInput): ReleaseManagerScheduler;
|
|
30
55
|
//# sourceMappingURL=scheduler.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"scheduler.d.ts","sourceRoot":"","sources":["../../src/release-manager/scheduler.ts"],"names":[],"mappings":"
|
|
1
|
+
{"version":3,"file":"scheduler.d.ts","sourceRoot":"","sources":["../../src/release-manager/scheduler.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;GAcG;AACH,OAAO,KAAK,EAAE,OAAO,EAAE,MAAM,eAAe,CAAC;AAC7C,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,aAAa,CAAC;AAEhD,OAAO,KAAK,EAAE,KAAK,EAAE,MAAM,iBAAiB,CAAC;AAE7C,OAAO,KAAK,EAAE,oBAAoB,EAAE,MAAM,YAAY,CAAC;AACvD,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,sBAAsB,CAAC;AAMxD,MAAM,WAAW,4BAA4B;IAC3C,MAAM,EAAE,oBAAoB,CAAC;IAC7B,EAAE,EAAE,KAAK,CAAC;IACV,OAAO,EAAE,OAAO,CAAC;IACjB,4FAA4F;IAC5F,MAAM,CAAC,EAAE,YAAY,CAAC;IACtB,OAAO,EAAE,MAAM,CAAC;IAChB,iGAAiG;IACjG,UAAU,EAAE,MAAM,OAAO,CAAC;IAC1B,KAAK,CAAC,EAAE,WAAW,CAAC;CACrB;AAED,MAAM,WAAW,uBAAuB;IACtC,kFAAkF;IAClF,IAAI,IAAI,OAAO,CAAC,IAAI,CAAC,CAAC;IACtB,kDAAkD;IAClD,KAAK,IAAI,IAAI,CAAC;IACd,yCAAyC;IACzC,IAAI,IAAI,IAAI,CAAC;IACb,+DAA+D;IAC/D,UAAU,CAAC,EAAE,EAAE,IAAI,GAAG,IAAI,CAAC;CAC5B;AAED;;;;;;;;;;;GAWG;AACH,wBAAgB,6BAA6B,CAC3C,KAAK,EAAE,4BAA4B,GAClC,uBAAuB,CAkDzB"}
|