fix: align decision-center desired state

This commit is contained in:
Codex
2026-05-21 10:14:16 +00:00
parent 4fd214bf25
commit 28a0c8b275
3 changed files with 108 additions and 2 deletions
+2 -2
View File
@@ -76,7 +76,7 @@
{
"id": "decision-center",
"repo": "https://github.com/pikasTech/unidesk",
"commitId": "54c1f8e165f90fa8509fda1f0c01f8c3fa82cbee"
"commitId": "b5486a61ab0aa6c227366a95d1afa68281584359"
}
]
},
@@ -120,7 +120,7 @@
{
"id": "decision-center",
"repo": "https://github.com/pikasTech/unidesk",
"commitId": "54c1f8e165f90fa8509fda1f0c01f8c3fa82cbee"
"commitId": "b5486a61ab0aa6c227366a95d1afa68281584359"
},
{
"id": "mdtodo",
@@ -0,0 +1,103 @@
import { readFileSync } from "node:fs";
import { spawnSync } from "node:child_process";
import { rootPath } from "./src/config";
type JsonRecord = Record<string, unknown>;
const verifiedCommit = "b5486a61ab0aa6c227366a95d1afa68281584359";
function assertCondition(condition: unknown, message: string, detail: unknown = {}): void {
if (!condition) throw new Error(`${message}: ${JSON.stringify(detail)}`);
}
function asRecord(value: unknown, label: string): JsonRecord {
assertCondition(typeof value === "object" && value !== null && !Array.isArray(value), `${label} must be an object`, value);
return value as JsonRecord;
}
function asArray(value: unknown, label: string): unknown[] {
assertCondition(Array.isArray(value), `${label} must be an array`, value);
return value as unknown[];
}
function findService(environment: "dev" | "prod", serviceId: string): JsonRecord {
const manifest = asRecord(JSON.parse(readFileSync(rootPath("deploy.json"), "utf8")) as unknown, "deploy.json");
const environments = asRecord(manifest.environments, "deploy.json.environments");
const env = asRecord(environments[environment], `deploy.json.environments.${environment}`);
const services = asArray(env.services, `deploy.json.environments.${environment}.services`);
const service = services.map((item, index) => asRecord(item, `${environment}.services[${index}]`))
.find((item) => item.id === serviceId);
assertCondition(service !== undefined, `${environment}/${serviceId} must exist in deploy.json`);
return service as JsonRecord;
}
function runDeployPlan(environment: "dev" | "prod", serviceId: string): JsonRecord {
const result = spawnSync("bun", [
"scripts/cli.ts",
"artifact-registry",
"deploy-service",
"--env",
environment,
"--service",
serviceId,
"--commit",
verifiedCommit,
"--dry-run",
], {
cwd: rootPath(),
encoding: "utf8",
maxBuffer: 8 * 1024 * 1024,
});
assertCondition(result.status === 0, `artifact consumer dry-run should exit 0 for ${environment}/${serviceId}`, {
status: result.status,
stdout: result.stdout.slice(-2000),
stderr: result.stderr.slice(-2000),
});
const envelope = asRecord(JSON.parse(result.stdout) as unknown, "artifact consumer envelope");
return asRecord(envelope.data, "artifact consumer dry-run data");
}
function assertNoBuildK3sDecisionCenter(environment: "dev" | "prod", expectedDeployment: string, expectedNamespace: string): void {
const service = findService(environment, "decision-center");
assertCondition(service.commitId === verifiedCommit, `${environment}/decision-center desired commit must match verified live commit`, service);
const plan = runDeployPlan(environment, "decision-center");
const registry = asRecord(plan.registry, `${environment}/decision-center registry`);
const build = asRecord(plan.build, `${environment}/decision-center build`);
const target = asRecord(plan.target, `${environment}/decision-center target`);
assertCondition(plan.ok === true && plan.dryRun === true && plan.mutation === false, `${environment}/decision-center dry-run must be non-mutating`, plan);
assertCondition(plan.commit === verifiedCommit, `${environment}/decision-center dry-run commit mismatch`, plan);
assertCondition(plan.serviceId === "decision-center", `${environment}/decision-center service id mismatch`, plan);
assertCondition(plan.sourceImage === `127.0.0.1:5000/unidesk/decision-center:${verifiedCommit}`, `${environment}/decision-center source image mismatch`, plan);
assertCondition(registry.repository === "unidesk/decision-center", `${environment}/decision-center registry repository mismatch`, registry);
assertCondition(registry.tag === verifiedCommit, `${environment}/decision-center registry tag mismatch`, registry);
assertCondition(build.producerBoundary === "ci publish-user-service", `${environment}/decision-center producer boundary mismatch`, build);
assertCondition(build.willCompile === false, `${environment}/decision-center CD must not compile`, build);
assertCondition(build.willRunDockerBuild === false, `${environment}/decision-center CD must not docker build`, build);
assertCondition(build.willRunDockerComposeBuild === false, `${environment}/decision-center CD must not compose build`, build);
assertCondition(target.kind === "d601-k3s", `${environment}/decision-center target must be D601 k3s`, target);
assertCondition(target.namespace === expectedNamespace, `${environment}/decision-center namespace mismatch`, target);
assertCondition(target.deployment === expectedDeployment, `${environment}/decision-center deployment mismatch`, target);
assertCondition(String(target.deployCommandShape ?? "").includes("kubectl set image"), `${environment}/decision-center command shape must be k3s artifact update`, target);
assertCondition(!JSON.stringify(plan).includes("server rebuild"), `${environment}/decision-center plan must not mention server rebuild`, plan);
}
for (const environment of ["dev", "prod"] as const) {
const frontend = findService(environment, "frontend");
assertCondition(frontend.commitId === verifiedCommit, `${environment}/frontend desired commit must stay aligned to verified UI artifact`, frontend);
}
assertNoBuildK3sDecisionCenter("dev", "decision-center-dev", "unidesk-dev");
assertNoBuildK3sDecisionCenter("prod", "decision-center", "unidesk");
process.stdout.write(`${JSON.stringify({
ok: true,
verifiedCommit,
checks: [
"decision-center dev/prod desired commits match the verified live/artifact commit",
"frontend dev/prod desired commits remain aligned to the same verified UI artifact",
"decision-center dev/prod dry-run plans are D601 k3s artifact consumers",
"decision-center dry-run plans declare no compile, docker build, compose build, or server rebuild path",
],
}, null, 2)}\n`);
+3
View File
@@ -289,6 +289,7 @@ export function runChecks(config: UniDeskConfig, options: CheckOptions = default
fileItem("scripts/src/ci.ts"),
fileItem("scripts/src/e2e.ts"),
fileItem("scripts/deploy-artifact-matrix-contract-test.ts"),
fileItem("scripts/decision-center-desired-state-contract-test.ts"),
fileItem("scripts/code-queue-prompt-observation-test.ts"),
fileItem("scripts/gh-cli-issue-guard-contract-test.ts"),
fileItem("scripts/gh-cli-pr-contract-test.ts"),
@@ -310,6 +311,7 @@ export function runChecks(config: UniDeskConfig, options: CheckOptions = default
items.push(commandItem("code-queue:submit-routing-contract", ["bun", "scripts/code-queue-submit-routing-contract-test.ts"], 30_000));
items.push(commandItem("provider:runner-triage-contract", ["bun", "scripts/provider-runner-triage-contract-test.ts"], 30_000));
items.push(commandItem("deploy:artifact-matrix-contract", ["bun", "scripts/deploy-artifact-matrix-contract-test.ts"], 30_000));
items.push(commandItem("decision-center:desired-state-contract", ["bun", "scripts/decision-center-desired-state-contract-test.ts"], 30_000));
items.push(commandItem("code-queue:active-run-heartbeat-visible", ["bun", "scripts/code-queue-liveness-diagnostics-test.ts", "--only", "code-queue:active-run-heartbeat-visible"], 30_000));
items.push(commandItem("code-queue:trace-gap-not-stale", ["bun", "scripts/code-queue-liveness-diagnostics-test.ts", "--only", "code-queue:trace-gap-not-stale"], 30_000));
items.push(commandItem("code-queue:stale-active-owner-expired", ["bun", "scripts/code-queue-liveness-diagnostics-test.ts", "--only", "code-queue:stale-active-owner-expired"], 30_000));
@@ -329,6 +331,7 @@ export function runChecks(config: UniDeskConfig, options: CheckOptions = default
items.push(skippedItem("code-queue:submit-routing-contract", "Code Queue submit routing contract is opt-in with script checks", "--scripts-typecheck or --full"));
items.push(skippedItem("provider:runner-triage-contract", "Provider runner triage contract is opt-in with script checks", "--scripts-typecheck or --full"));
items.push(skippedItem("deploy:artifact-matrix-contract", "deploy artifact matrix contract is opt-in with script checks", "--scripts-typecheck or --full"));
items.push(skippedItem("decision-center:desired-state-contract", "Decision Center desired-state drift contract is opt-in with script checks", "--scripts-typecheck or --full"));
items.push(skippedItem("code-queue:liveness-diagnostics-fixtures", "Code Queue liveness diagnostics fixtures are opt-in with script checks", "--scripts-typecheck or --full"));
items.push(skippedItem("baidu-netdisk:artifact-guard-contract", "Baidu Netdisk artifact guard contract is opt-in with script checks", "--scripts-typecheck or --full"));
items.push(skippedItem("artifact-registry:direct-compose-dry-run-matrix", "main-server direct artifact consumer dry-run matrix is opt-in with script checks", "--scripts-typecheck or --full"));