forked from CopilotKit/CopilotKit
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathredeploy-env.summary.test.ts
More file actions
135 lines (127 loc) · 4.66 KB
/
Copy pathredeploy-env.summary.test.ts
File metadata and controls
135 lines (127 loc) · 4.66 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
import { mkdtempSync, readFileSync, readdirSync, rmSync } from "node:fs";
import { tmpdir } from "node:os";
import { join } from "node:path";
import { afterEach, beforeEach, describe, expect, it } from "vitest";
import { runRedeploy } from "../redeploy-env";
import { SERVICES } from "../railway-envs";
// A.4: per-service JSON summary contract.
//
// Shape (cross-workstream contract, consumed by showcase_deploy.yml's
// `enforce-redeploy-gate` job in A.7):
// Array<{ service: string; status: "ok" | "error"; error?: string }>
//
// When REDEPLOY_SUMMARY_JSON env var is set, runRedeploy MUST write the
// records array to that path atomically (stage to .tmp, rename). When
// unset, no JSON is written.
//
// PR #5093's exit-code contract MUST be preserved verbatim:
// - staging: exitCode === 0 even when all per-service redeploys fail
// - prod: exitCode === 1 on any per-service failure
// The JSON write happens BEFORE the exit-code computation; it never
// changes exit semantics on disk hiccups (write failures are warn-only).
describe("redeploy-env per-service JSON summary", () => {
let dir: string;
let summaryPath: string;
beforeEach(() => {
dir = mkdtempSync(join(tmpdir(), "redeploy-summary-"));
summaryPath = join(dir, "summary.json");
process.env.REDEPLOY_SUMMARY_JSON = summaryPath;
});
afterEach(() => {
delete process.env.REDEPLOY_SUMMARY_JSON;
rmSync(dir, { recursive: true, force: true });
});
it("writes one JSON record per attempted service with status", async () => {
const ag2ServiceId = SERVICES["showcase-ag2"].serviceId;
const summary = await runRedeploy({
env: "staging",
services: ["showcase-mastra", "showcase-ag2"],
appendSummary: () => {},
redeploy: async (serviceId) => {
if (serviceId === ag2ServiceId) {
return { ok: false, error: "boom" };
}
return { ok: true };
},
});
expect(summary.exitCode).toBe(0); // staging contract intact
const records = JSON.parse(readFileSync(summaryPath, "utf8")) as Array<{
service: string;
status: "ok" | "error";
error?: string;
}>;
expect(records).toEqual(
expect.arrayContaining([
{ service: "showcase-ag2", status: "error", error: "boom" },
{ service: "showcase-mastra", status: "ok" },
]),
);
expect(records).toHaveLength(2);
});
it("does NOT write JSON when REDEPLOY_SUMMARY_JSON is unset", async () => {
delete process.env.REDEPLOY_SUMMARY_JSON;
await runRedeploy({
env: "staging",
services: ["showcase-mastra"],
appendSummary: () => {},
redeploy: async () => ({ ok: true }),
});
// The temp dir we created is empty — assert nothing was written.
// (No path was given; the function had nowhere to write.)
expect(readdirSync(dir)).toEqual([]);
});
it("preserves PR #5093 contract: staging exits 0 even when all fail", async () => {
const summary = await runRedeploy({
env: "staging",
services: ["showcase-mastra", "showcase-ag2"],
appendSummary: () => {},
redeploy: async () => ({ ok: false, error: "boom" }),
});
expect(summary.exitCode).toBe(0);
expect(summary.failed).toBe(2);
const records = JSON.parse(readFileSync(summaryPath, "utf8")) as Array<{
service: string;
status: "ok" | "error";
error?: string;
}>;
expect(records.every((r) => r.status === "error")).toBe(true);
expect(records).toHaveLength(2);
});
it("preserves PR #5093 contract: prod exits 1 on any per-service failure", async () => {
const summary = await runRedeploy({
env: "prod",
services: ["showcase-mastra"],
appendSummary: () => {},
redeploy: async () => ({ ok: false, error: "boom" }),
});
expect(summary.exitCode).toBe(1);
const records = JSON.parse(readFileSync(summaryPath, "utf8")) as Array<{
service: string;
status: "ok" | "error";
error?: string;
}>;
expect(records).toEqual([
{ service: "showcase-mastra", status: "error", error: "boom" },
]);
});
it("records caught throws as status:error with the thrown message", async () => {
const summary = await runRedeploy({
env: "staging",
services: ["showcase-mastra"],
appendSummary: () => {},
redeploy: async () => {
throw new Error("kaboom");
},
});
expect(summary.exitCode).toBe(0);
expect(summary.failed).toBe(1);
const records = JSON.parse(readFileSync(summaryPath, "utf8")) as Array<{
service: string;
status: "ok" | "error";
error?: string;
}>;
expect(records).toEqual([
{ service: "showcase-mastra", status: "error", error: "kaboom" },
]);
});
});