Merge pull request #1539 from pikasTech/fix/1534-stage-interval-attribution

fix: preserve branch follower stage intervals
This commit is contained in:
Lyon
2026-07-05 00:18:19 +08:00
committed by GitHub
7 changed files with 224 additions and 128 deletions
+5 -1
View File
@@ -106,6 +106,7 @@ if (existing) {
}
const startedAt = Date.now();
const startedAtIso = new Date(startedAt).toISOString();
const deadline = startedAt + timeoutSeconds * 1000;
let polls = 0;
let latest = await getJob();
@@ -123,6 +124,7 @@ const failed = condition(latest, "Failed");
const logs = await logsTail();
const summary = parseLastJsonSummary(logs);
const timedOut = !complete && !failed;
const finishedAt = Date.now();
const output = {
ok: Boolean(complete) && !timedOut,
completed: Boolean(complete),
@@ -134,7 +136,9 @@ const output = {
jobName,
namespace,
polls,
elapsedMs: Date.now() - startedAt,
elapsedMs: finishedAt - startedAt,
startedAt: startedAtIso,
finishedAt: new Date(finishedAt).toISOString(),
conditionReason: complete?.reason || failed?.reason || null,
conditionMessage: complete?.message || failed?.message || null,
logsTail: logs || null,
+6 -103
View File
@@ -37,6 +37,7 @@ import { buildCicdHelp } from "./cicd-help";
import { attachReconcileTimeline, compactReconcileTimeline, finishReconcileStep, finishReconcileTimeline, startReconcileStep, startReconcileTimeline } from "./cicd-reconcile-timeline";
import { orderFollowersForControllerCloseout, shouldYieldAfterAutomaticTrigger } from "./cicd-reconcile-scheduler";
import { buildFollowerTimings, compactListTimings, compactTimings, storedFollowerTimingsForStatus, timingPerformanceSummary } from "./cicd-timings";
import { timingAttributionSummary } from "./cicd-timing-attribution";
import type { AdapterSummary, BranchFollowerAction, BranchFollowerDebugStep, BranchFollowerGate, BranchFollowerPhase, BranchFollowerRegistry, ControllerSpec, FollowerSpec, FollowerState, K8sFollowerStateRead, K8sStateRead, NativeCloseoutWaitResult, NativeK8sJobResult, NativeStatusSpec, NativeWorkloadSpec, OutputMode, ParsedOptions, TriggerResult } from "./cicd-types";
import {
arrayField,
@@ -1843,6 +1844,8 @@ async function readAdapterStatus(registry: BranchFollowerRegistry, follower: Fol
const sourceSyncDetail = sourceSync === null || sourceSync.result.ok ? null : redactText(tailText(sourceSync.result.conditionMessage ?? sourceSync.result.logsTail ?? "unknown", 800));
const sourceSyncError = sourceSyncDetail === null ? null : `native source sync failed: ${sourceSyncDetail}`;
const bundle = readNativeObjectBundle(registry, follower, options, remainingSeconds(startedAt, timeoutSeconds), runKubeScript);
const bundleFinishedMs = Date.now();
const bundleStartedMs = Math.max(startedAt, bundleFinishedMs - Math.max(0, bundle.elapsedMs));
const observedSha = sourceSyncError === null ? stringOrNull(bundle.source?.commit) : null;
const runtimeTargetSha = runtimeTargetShaFromWorkloads(follower.nativeStatus.runtime, bundle.workloads);
const pipelineRunName = stringOrNull(asOptionalRecord(bundle.pipelineRun?.metadata)?.name) ?? expectedPipelineRunName(follower, observedSha);
@@ -1902,7 +1905,7 @@ async function readAdapterStatus(registry: BranchFollowerRegistry, follower: Fol
planArtifacts: bundle.planArtifacts,
argo: nativeArgoSummary(bundle.argoApplication),
runtime: nativeRuntimeSummary(follower.nativeStatus.runtime, bundle.workloads, observedSha),
timings: { statusRead: { elapsedMs: bundle.elapsedMs, budgetSeconds: timeoutSeconds } },
timings: { statusRead: { elapsedMs: bundle.elapsedMs, budgetSeconds: timeoutSeconds, startedAt: new Date(bundleStartedMs).toISOString(), finishedAt: new Date(bundleFinishedMs).toISOString() } },
errors,
statusAuthority: "k8s-native",
parsedDownstreamCliOutput: false,
@@ -2632,9 +2635,11 @@ function nativeGateTimingSummary(payload: Record<string, unknown> | null, timing
statusReadSeconds: secondsFromMsValue(numberOrNull(statusRead?.elapsedMs)),
gitMirrorSyncSeconds: secondsFromMsValue(numberOrNull(sourceSync?.elapsedMs)),
pipelineRunSeconds: numberOrNull(tekton?.durationSeconds),
pipelineRunName: stringOrNull(tekton?.name),
pipelineRunStartedAt: stringOrNull(tekton?.startTime),
pipelineRunFinishedAt: stringOrNull(tekton?.completionTime),
argoOperationSeconds: numberOrNull(argo?.operationDurationSeconds),
argoApplication: stringOrNull(argo?.name),
argoOperationStartedAt: stringOrNull(argo?.operationStartedAt),
argoOperationFinishedAt: stringOrNull(argo?.operationFinishedAt),
argoIncludedInStoredTotal: argoStage?.seconds !== null && argoStage?.source === "argocd",
@@ -2644,108 +2649,6 @@ function nativeGateTimingSummary(payload: Record<string, unknown> | null, timing
};
}
function timingAttributionSummary(timings: FollowerState["timings"], nativeGate: Record<string, unknown>): Record<string, unknown> {
const totalStartedMs = timestampMs(timings.startedAt);
const totalFinishedMs = timestampMs(timings.finishedAt);
const totalSeconds = timings.totalSeconds;
if (totalSeconds === null || totalStartedMs === null || totalFinishedMs === null || totalFinishedMs < totalStartedMs) {
return {
status: "unknown",
source: "stored-total-vs-native-intervals",
totalSeconds,
knownIntervalCoverageSeconds: null,
unknownWallClockSeconds: null,
reason: "stored total range is missing or invalid; old wall-clock attribution cannot be reconstructed",
};
}
const intervals = [
timingIntervalOverlap("pipeline", "tekton", nativeGate.pipelineRunStartedAt, nativeGate.pipelineRunFinishedAt, totalStartedMs, totalFinishedMs),
timingIntervalOverlap("argo", "argocd", nativeGate.argoOperationStartedAt, nativeGate.argoOperationFinishedAt, totalStartedMs, totalFinishedMs),
].filter((item): item is Record<string, unknown> => item !== null);
const coverageSeconds = mergedIntervalCoverageSeconds(intervals);
const unknownSeconds = roundSeconds(Math.max(0, totalSeconds - coverageSeconds));
const missingHistoricalIntervals = timings.stages.some((stage) => stage.seconds !== null);
return {
status: unknownSeconds > 0 ? "partial" : "covered",
source: "stored-total-vs-native-intervals",
totalSeconds,
totalStartedAt: timings.startedAt,
totalFinishedAt: timings.finishedAt,
knownIntervalCoverageSeconds: coverageSeconds,
unknownWallClockSeconds: unknownSeconds,
intervalCount: intervals.length,
intervals: intervals.map(({ startMs: _startMs, endMs: _endMs, overlapStartMs: _overlapStartMs, overlapEndMs: _overlapEndMs, ...item }) => item),
reason: unknownSeconds > 0
? missingHistoricalIntervals
? "stored state lacks historical per-stage intervals for the remaining wall-clock; do not infer wait/idle from current native objects"
: "no stored historical native intervals overlap the total range; old wall-clock attribution cannot be reconstructed"
: null,
};
}
function timingIntervalOverlap(stage: string, source: string, startedAtValue: unknown, finishedAtValue: unknown, totalStartedMs: number, totalFinishedMs: number): Record<string, unknown> | null {
const startedAt = stringOrNull(startedAtValue);
const finishedAt = stringOrNull(finishedAtValue);
const startMs = timestampMs(startedAt);
const endMs = timestampMs(finishedAt);
if (startMs === null || endMs === null || endMs < startMs) return null;
const overlapStartMs = Math.max(startMs, totalStartedMs);
const overlapEndMs = Math.min(endMs, totalFinishedMs);
const overlapSeconds = overlapEndMs > overlapStartMs ? roundSeconds((overlapEndMs - overlapStartMs) / 1000) : 0;
return {
stage,
source,
startedAt,
finishedAt,
overlapSeconds,
inStoredTotal: overlapSeconds > 0,
startMs,
endMs,
overlapStartMs,
overlapEndMs,
};
}
function mergedIntervalCoverageSeconds(intervals: Record<string, unknown>[]): number {
const ranges = intervals
.map((item) => ({
startMs: numberOrNull(item.overlapStartMs),
endMs: numberOrNull(item.overlapEndMs),
overlapSeconds: numberOrNull(item.overlapSeconds),
}))
.filter((item): item is { startMs: number; endMs: number; overlapSeconds: number } => item.startMs !== null && item.endMs !== null && item.overlapSeconds !== null && item.overlapSeconds > 0)
.sort((a, b) => a.startMs - b.startMs);
let coveredMs = 0;
let currentStart: number | null = null;
let currentEnd: number | null = null;
for (const range of ranges) {
if (currentStart === null || currentEnd === null) {
currentStart = range.startMs;
currentEnd = range.endMs;
continue;
}
if (range.startMs <= currentEnd) {
currentEnd = Math.max(currentEnd, range.endMs);
continue;
}
coveredMs += currentEnd - currentStart;
currentStart = range.startMs;
currentEnd = range.endMs;
}
if (currentStart !== null && currentEnd !== null) coveredMs += currentEnd - currentStart;
return roundSeconds(coveredMs / 1000);
}
function timestampMs(value: string | null): number | null {
if (value === null) return null;
const parsed = Date.parse(value);
return Number.isFinite(parsed) ? parsed : null;
}
function roundSeconds(value: number): number {
return Math.round(value * 10) / 10;
}
function followerNextCommands(follower: FollowerSpec): Record<string, string> {
const next: Record<string, string> = {
status: `bun scripts/cli.ts cicd branch-follower status --follower ${follower.id}`,
+2
View File
@@ -52,6 +52,8 @@ export function runNativeK8sJob(namespace: string, jobName: string, manifest: Re
namespace,
polls: numberOrNull(parsed?.polls) ?? 0,
elapsedMs: numberOrNull(parsed?.elapsedMs) ?? 0,
startedAt: stringOrNull(parsed?.startedAt),
finishedAt: stringOrNull(parsed?.finishedAt),
logsTail: stringOrNull(parsed?.logsTail),
summary: asOptionalRecord(parsed?.summary),
conditionReason: stringOrNull(parsed?.conditionReason),
+6 -2
View File
@@ -127,7 +127,7 @@ function renderStatusHuman(payload: Record<string, unknown>, _options: ParsedOpt
liveRefreshRows.length === 0 ? "" : `\nLIVE REFRESH\n${table(["MODE", "REQUESTED", "EXECUTED", "JOB", "ELAPSED", "IN_TOTAL"], liveRefreshRows)}`,
"",
table(["FOLLOWER", "PHASE", "ADAPTER", "OBSERVED", "TARGET", "TRIGGERED", "SUCCEEDED", "IN_FLIGHT", "BUDGET", "MESSAGE"], rows),
timingRows.length === 0 ? "" : `\nSTAGE TIMINGS\n${table(["FOLLOWER", "STAGE", "STATUS", "SECONDS", "BUDGET", "OBJECT"], timingRows)}`,
timingRows.length === 0 ? "" : `\nSTAGE TIMINGS\n${table(["FOLLOWER", "STAGE", "STATUS", "SECONDS", "BUDGET", "STARTED", "FINISHED", "OBJECT"], timingRows)}`,
timingContextRows.length === 0 ? "" : `\nTIMING CONTEXT\n${table(["FOLLOWER", "CONTEXT", "SOURCE", "SECONDS", "STARTED", "FINISHED", "IN_TOTAL"], timingContextRows)}`,
performanceRows.length === 0 ? "" : `\nSLOW STAGES\n${table(["FOLLOWER", "STAGE", "STATUS", "SECONDS", "SOURCE", "OBJECT"], performanceRows)}`,
evidenceRows.length === 0 ? "" : `\nEVIDENCE\n${table(["FOLLOWER", "TYPE", "STATUS", "DETAIL", "OBJECT"], evidenceRows)}`,
@@ -188,7 +188,7 @@ function renderRunOnceHuman(payload: Record<string, unknown>): string {
`CI/CD BRANCH-FOLLOWER RUN-ONCE (${payload.ok === false ? "blocked" : payload.dryRun === true ? "dry-run" : "ok"})`,
"",
table(["FOLLOWER", "PHASE", "OBSERVED", "TARGET", "TRIGGERED", "IN_FLIGHT", "DECISION"], rows),
timingRows.length === 0 ? "" : `\nSTAGE TIMINGS\n${table(["FOLLOWER", "STAGE", "STATUS", "SECONDS", "BUDGET", "OBJECT"], timingRows)}`,
timingRows.length === 0 ? "" : `\nSTAGE TIMINGS\n${table(["FOLLOWER", "STAGE", "STATUS", "SECONDS", "BUDGET", "STARTED", "FINISHED", "OBJECT"], timingRows)}`,
reconcileRows.length === 0 ? "" : `\nRECONCILE TIMELINE\n${table(["FOLLOWER", "STEP", "STATUS", "SECONDS", "STARTED", "OBJECT"], reconcileRows)}`,
writeRows.length === 0 ? "" : `\nSTATE WRITES\n${table(["FOLLOWER", "STATUS", "BEFORE_RV", "AFTER_RV", "PRESERVED", "EXIT", "MESSAGE"], writeRows)}`,
"",
@@ -237,6 +237,8 @@ function timingRowsForFollower(item: Record<string, unknown>): unknown[][] {
stringOrNull(timings.totalStatus) ?? "unknown",
formatSeconds(numberOrNull(timings.totalSeconds)),
formatSeconds(budget),
stringOrNull(timings.startedAt) ?? "-",
stringOrNull(timings.finishedAt) ?? "-",
[stringOrNull(timings.totalSource), shortSha(stringOrNull(timings.sourceCommit))].filter((value) => value !== null && value !== "-").join(":") || "-",
]];
for (const stage of arrayRecords(timings.stages)) {
@@ -246,6 +248,8 @@ function timingRowsForFollower(item: Record<string, unknown>): unknown[][] {
stage.status,
formatSeconds(numberOrNull(stage.seconds)),
formatSeconds(numberOrNull(stage.budgetSeconds)),
stringOrNull(stage.startedAt) ?? "-",
stringOrNull(stage.finishedAt) ?? "-",
stringOrNull(stage.object) ?? "-",
]);
}
+163
View File
@@ -0,0 +1,163 @@
// SPEC: PJ2026-01060703 CI/CD branch follower timing attribution.
// Responsibility: derive bounded wall-clock attribution from stored totals and stage/native intervals.
import type { FollowerState, StageTiming } from "./cicd-types";
export function timingAttributionSummary(timings: FollowerState["timings"], nativeGate: Record<string, unknown>): Record<string, unknown> {
const totalStartedMs = timestampMs(timings.startedAt);
const totalFinishedMs = timestampMs(timings.finishedAt);
const totalSeconds = timings.totalSeconds;
if (totalSeconds === null || totalStartedMs === null || totalFinishedMs === null || totalFinishedMs < totalStartedMs) {
return {
status: "unknown",
source: "stored-total-vs-stage-intervals",
totalSeconds,
knownIntervalCoverageSeconds: null,
unknownWallClockSeconds: null,
missingIntervalStages: [],
reason: "stored total range is missing or invalid; old wall-clock attribution cannot be reconstructed",
};
}
const stageIntervals = timings.stages
.map((stage) => timingIntervalOverlap(stage.stage, stage.source, stage.object, stage.startedAt, stage.finishedAt, totalStartedMs, totalFinishedMs))
.filter((item): item is TimingInterval => item !== null);
const nativeIntervals = [
timingIntervalOverlap("pipelinerun", "tekton-native-gate", String(nativeGate.pipelineRunObject ?? nativeGate.pipelineRunName ?? ""), nativeGate.pipelineRunStartedAt, nativeGate.pipelineRunFinishedAt, totalStartedMs, totalFinishedMs),
timingIntervalOverlap("argo", "argocd-native-gate", String(nativeGate.argoApplication ?? ""), nativeGate.argoOperationStartedAt, nativeGate.argoOperationFinishedAt, totalStartedMs, totalFinishedMs),
].filter((item): item is TimingInterval => item !== null);
const intervals = dedupeIntervals([...stageIntervals, ...nativeIntervals]);
const coverageSeconds = mergedIntervalCoverageSeconds(intervals);
const unknownSeconds = roundSeconds(Math.max(0, totalSeconds - coverageSeconds));
const missingIntervalCount = countMissingStageIntervals(timings.stages);
const missingIntervalStages = missingStageIntervals(timings.stages);
return {
status: unknownSeconds > 0 ? "partial" : "covered",
source: "stored-total-vs-stage-intervals",
totalSeconds,
totalStartedAt: timings.startedAt,
totalFinishedAt: timings.finishedAt,
knownIntervalCoverageSeconds: coverageSeconds,
unknownWallClockSeconds: unknownSeconds,
intervalCount: intervals.length,
stageIntervalCount: stageIntervals.length,
nativeIntervalCount: nativeIntervals.length,
excludedIntervalCount: intervals.filter((item) => !item.inStoredTotal).length,
missingIntervalCount,
missingIntervalReason: missingIntervalCount > 0 ? "missing-startedAt-or-finishedAt" : null,
missingIntervalStages,
intervals: intervals
.filter((item) => item.inStoredTotal)
.map(({ startMs: _startMs, endMs: _endMs, overlapStartMs: _overlapStartMs, overlapEndMs: _overlapEndMs, ...item }) => item),
reason: unknownSeconds > 0
? missingIntervalCount > 0
? "some timed stages lack startedAt/finishedAt; remaining wall-clock must stay unknown until future state records those intervals"
: "no stored historical stage intervals cover the remaining wall-clock; old data cannot be reconstructed"
: null,
};
}
type TimingInterval = {
stage: string;
source: string;
object: string | null;
startedAt: string;
finishedAt: string;
overlapSeconds: number;
inStoredTotal: boolean;
startMs: number;
endMs: number;
overlapStartMs: number;
overlapEndMs: number;
};
function timingIntervalOverlap(stage: string, source: string, object: unknown, startedAtValue: unknown, finishedAtValue: unknown, totalStartedMs: number, totalFinishedMs: number): TimingInterval | null {
const startedAt = stringOrNull(startedAtValue);
const finishedAt = stringOrNull(finishedAtValue);
const startMs = timestampMs(startedAt);
const endMs = timestampMs(finishedAt);
if (startedAt === null || finishedAt === null || startMs === null || endMs === null || endMs < startMs) return null;
const overlapStartMs = Math.max(startMs, totalStartedMs);
const overlapEndMs = Math.min(endMs, totalFinishedMs);
const overlapSeconds = overlapEndMs > overlapStartMs ? roundSeconds((overlapEndMs - overlapStartMs) / 1000) : 0;
return {
stage,
source,
object: stringOrNull(object),
startedAt,
finishedAt,
overlapSeconds,
inStoredTotal: overlapSeconds > 0,
startMs,
endMs,
overlapStartMs,
overlapEndMs,
};
}
function missingStageIntervals(stages: readonly StageTiming[]): Record<string, unknown>[] {
return stages
.filter((stage) => stage.seconds !== null && (timestampMs(stage.startedAt ?? null) === null || timestampMs(stage.finishedAt ?? null) === null))
.slice(0, 3)
.map((stage) => {
const row: Record<string, unknown> = {
stage: stage.stage,
source: stage.source,
seconds: stage.seconds,
};
if (stage.object !== null) row.object = stage.object;
return row;
});
}
function countMissingStageIntervals(stages: readonly StageTiming[]): number {
return stages.filter((stage) => stage.seconds !== null && (timestampMs(stage.startedAt ?? null) === null || timestampMs(stage.finishedAt ?? null) === null)).length;
}
function dedupeIntervals(intervals: TimingInterval[]): TimingInterval[] {
const byKey = new Map<string, TimingInterval>();
for (const interval of intervals) {
const key = `${interval.stage}\t${interval.source}\t${interval.startedAt}\t${interval.finishedAt}`;
const previous = byKey.get(key);
if (previous === undefined || previous.inStoredTotal === false && interval.inStoredTotal === true) byKey.set(key, interval);
}
return [...byKey.values()];
}
function mergedIntervalCoverageSeconds(intervals: TimingInterval[]): number {
const ranges = intervals
.filter((item) => item.overlapSeconds > 0)
.map((item) => ({ startMs: item.overlapStartMs, endMs: item.overlapEndMs }))
.sort((a, b) => a.startMs - b.startMs);
let coveredMs = 0;
let currentStart: number | null = null;
let currentEnd: number | null = null;
for (const range of ranges) {
if (currentStart === null || currentEnd === null) {
currentStart = range.startMs;
currentEnd = range.endMs;
continue;
}
if (range.startMs <= currentEnd) {
currentEnd = Math.max(currentEnd, range.endMs);
continue;
}
coveredMs += currentEnd - currentStart;
currentStart = range.startMs;
currentEnd = range.endMs;
}
if (currentStart !== null && currentEnd !== null) coveredMs += currentEnd - currentStart;
return roundSeconds(coveredMs / 1000);
}
function timestampMs(value: string | null): number | null {
if (value === null) return null;
const parsed = Date.parse(value);
return Number.isFinite(parsed) ? parsed : null;
}
function roundSeconds(value: number): number {
return Math.round(value * 10) / 10;
}
function stringOrNull(value: unknown): string | null {
return typeof value === "string" && value.length > 0 ? value : null;
}
+38 -22
View File
@@ -73,14 +73,7 @@ export function compactTimings(timings: FollowerState["timings"]): FollowerState
startedAt: timings.startedAt,
finishedAt: timings.finishedAt,
overBudget: timings.overBudget,
stages: timings.stages.slice(0, 24).map((stage) => ({
stage: stage.stage,
status: stage.status,
seconds: stage.seconds,
budgetSeconds: stage.budgetSeconds,
source: stage.source,
object: stage.object,
})),
stages: timings.stages.slice(0, 24).map((stage) => compactStageTiming(stage, true)),
};
}
@@ -94,11 +87,7 @@ export function compactListTimings(timings: FollowerState["timings"]): Record<st
startedAt: timings.startedAt,
finishedAt: timings.finishedAt,
overBudget: timings.overBudget,
stages: timings.stages.slice(0, 4).map((stage) => ({
stage: stage.stage,
status: stage.status,
seconds: stage.seconds,
})),
stages: timings.stages.slice(0, 4).map((stage) => compactListStageTiming(stage)),
};
}
@@ -141,6 +130,31 @@ function noopStoredTotalFinishOverride(
return new Date().toISOString();
}
function compactStageTiming(stage: StageTiming, includeDetail: boolean): StageTiming {
const row: StageTiming = {
stage: stage.stage,
status: stage.status,
seconds: stage.seconds,
budgetSeconds: includeDetail ? stage.budgetSeconds : null,
source: includeDetail ? stage.source : "",
object: includeDetail ? stage.object : null,
};
if (stage.startedAt !== null && stage.startedAt !== undefined) row.startedAt = stage.startedAt;
if (stage.finishedAt !== null && stage.finishedAt !== undefined) row.finishedAt = stage.finishedAt;
return row;
}
function compactListStageTiming(stage: StageTiming): Record<string, unknown> {
const row: Record<string, unknown> = {
stage: stage.stage,
status: stage.status,
seconds: stage.seconds,
};
if (stage.startedAt !== null && stage.startedAt !== undefined) row.startedAt = stage.startedAt;
if (stage.finishedAt !== null && stage.finishedAt !== undefined) row.finishedAt = stage.finishedAt;
return row;
}
function storedStageTimings(storedTimings: Record<string, unknown> | null): StageTiming[] {
if (storedTimings === null) return [];
return arrayRecords(storedTimings.stages)
@@ -151,6 +165,8 @@ function storedStageTimings(storedTimings: Record<string, unknown> | null): Stag
numberOrNull(stage.budgetSeconds),
stringOrNull(stage.source) ?? "stored-state",
stringOrNull(stage.object),
stringOrNull(stage.startedAt),
stringOrNull(stage.finishedAt),
))
.filter((stage) => stage.stage.length > 0)
.slice(0, 24);
@@ -222,7 +238,7 @@ function stageTimingsFromNativePayload(payload: Record<string, unknown> | null,
if (payload === null) return [];
const stages: StageTiming[] = [];
const statusRead = asOptionalRecord(asOptionalRecord(payload.timings)?.statusRead);
stages.push(stageTiming("status-read", "ok", secondsFromMs(numberOrNull(statusRead?.elapsedMs)), numberOrNull(statusRead?.budgetSeconds), "native-status", null));
stages.push(stageTiming("status-read", "ok", secondsFromMs(numberOrNull(statusRead?.elapsedMs)), numberOrNull(statusRead?.budgetSeconds), "native-status", null, stringOrNull(statusRead?.startedAt), stringOrNull(statusRead?.finishedAt)));
const sourceSyncStage = k8sJobTiming("git-mirror-sync", asOptionalRecord(payload.sourceSync));
if (sourceSyncStage !== null) stages.push(sourceSyncStage);
const reuseConfig = asOptionalRecord(payload.reuseConfig);
@@ -237,18 +253,18 @@ function stageTimingsFromNativePayload(payload: Record<string, unknown> | null,
const tekton = asOptionalRecord(payload.tekton);
if (tekton !== null) {
const status = tekton.succeeded === true ? "succeeded" : tekton.succeeded === false ? `failed:${stringOrNull(tekton.reason) ?? "unknown"}` : "running";
stages.push(stageTiming("pipelinerun", status, numberOrNull(tekton.durationSeconds), null, "tekton", stringOrNull(tekton.name)));
stages.push(stageTiming("pipelinerun", status, numberOrNull(tekton.durationSeconds), null, "tekton", stringOrNull(tekton.name), stringOrNull(tekton.startTime), stringOrNull(tekton.completionTime)));
}
const taskRuns = asOptionalRecord(payload.taskRuns);
for (const record of taskRuns === null ? [] : prioritizedTaskRunItems(taskRuns)) {
const name = stringOrNull(record.pipelineTask) ?? stringOrNull(record.name) ?? "unknown";
const status = record.status === "True" ? "succeeded" : record.status === "False" ? `failed:${stringOrNull(record.reason) ?? "unknown"}` : "running";
stages.push(stageTiming(`task:${name}`, status, numberOrNull(record.durationSeconds), null, "tekton-taskrun", stringOrNull(record.name)));
stages.push(stageTiming(`task:${name}`, status, numberOrNull(record.durationSeconds), null, "tekton-taskrun", stringOrNull(record.name), stringOrNull(record.startTime), stringOrNull(record.completionTime)));
}
const argo = asOptionalRecord(payload.argo);
if (argo !== null) {
const sameWindow = total === null || timingOverlapsTotal(total, stringOrNull(argo.operationStartedAt), stringOrNull(argo.operationFinishedAt));
stages.push(stageTiming("argo", `${sameWindow ? "" : "current:"}${stringOrNull(argo.syncStatus) ?? "unknown"}/${stringOrNull(argo.healthStatus) ?? "unknown"}`, sameWindow ? numberOrNull(argo.operationDurationSeconds) : null, null, sameWindow ? "argocd" : "argocd-current", stringOrNull(argo.name)));
stages.push(stageTiming("argo", `${sameWindow ? "" : "current:"}${stringOrNull(argo.syncStatus) ?? "unknown"}/${stringOrNull(argo.healthStatus) ?? "unknown"}`, sameWindow ? numberOrNull(argo.operationDurationSeconds) : null, null, sameWindow ? "argocd" : "argocd-current", stringOrNull(argo.name), stringOrNull(argo.operationStartedAt), stringOrNull(argo.operationFinishedAt)));
}
const runtime = asOptionalRecord(payload.runtime);
if (runtime !== null) {
@@ -296,7 +312,7 @@ function stageTimingsFromCommand(command: Record<string, unknown> | undefined):
const tektonSeconds = secondsFromMs(numberOrNull(payload.elapsedMs));
if (tektonSeconds !== null) {
const status = payload.completed === true ? "completed" : payload.failed === true ? "failed" : payload.stillRunning === true ? "running" : "submitted";
stages.push(stageTiming("pipelinerun-wait", status, tektonSeconds, null, "tekton-submit", stringOrNull(command.pipelineRun)));
stages.push(stageTiming("pipelinerun-wait", status, tektonSeconds, null, "tekton-submit", stringOrNull(command.pipelineRun), stringOrNull(payload.startedAt), stringOrNull(payload.finishedAt)));
}
}
const closeout = asOptionalRecord(command.closeout);
@@ -305,7 +321,7 @@ function stageTimingsFromCommand(command: Record<string, unknown> | undefined):
const gitMirrorFlushStage = k8sJobTiming("git-mirror-flush", asOptionalRecord(gitMirrorFlush?.result), stringOrNull(gitMirrorFlush?.jobName));
if (gitMirrorFlushStage !== null) stages.push(gitMirrorFlushStage);
const status = closeout.completed === true ? "completed" : closeout.timedOut === true ? "over-budget" : "pending";
stages.push(stageTiming("closeout", status, secondsFromMs(numberOrNull(closeout.elapsedMs)), null, "k8s-native-closeout", stringOrNull(command.pipelineRun)));
stages.push(stageTiming("closeout", status, secondsFromMs(numberOrNull(closeout.elapsedMs)), null, "k8s-native-closeout", stringOrNull(command.pipelineRun), stringOrNull(closeout.startedAt), stringOrNull(closeout.finishedAt)));
}
return stages;
}
@@ -313,11 +329,11 @@ function stageTimingsFromCommand(command: Record<string, unknown> | undefined):
function k8sJobTiming(stage: string, job: Record<string, unknown> | null, objectOverride?: string | null): StageTiming | null {
if (job === null) return null;
const status = job.completed === true ? job.reused === true ? "reused" : "completed" : job.failed === true ? "failed" : job.timedOut === true ? "over-budget" : "running";
return stageTiming(stage, status, secondsFromMs(numberOrNull(job.elapsedMs)), null, "kubernetes-job", objectOverride ?? stringOrNull(job.jobName));
return stageTiming(stage, status, secondsFromMs(numberOrNull(job.elapsedMs)), null, "kubernetes-job", objectOverride ?? stringOrNull(job.jobName), stringOrNull(job.startedAt), stringOrNull(job.finishedAt));
}
function stageTiming(stage: string, status: string, seconds: number | null, budgetSeconds: number | null, source: string, object: string | null): StageTiming {
return { stage, status, seconds, budgetSeconds, source, object };
function stageTiming(stage: string, status: string, seconds: number | null, budgetSeconds: number | null, source: string, object: string | null, startedAt?: string | null, finishedAt?: string | null): StageTiming {
return { stage, status, seconds, budgetSeconds, source, object, startedAt: startedAt ?? null, finishedAt: finishedAt ?? null };
}
function dedupeTimingStages(stages: StageTiming[]): StageTiming[] {
+4
View File
@@ -269,6 +269,8 @@ export interface NativeK8sJobResult {
namespace: string;
polls: number;
elapsedMs: number;
startedAt: string | null;
finishedAt: string | null;
logsTail: string | null;
summary: Record<string, unknown> | null;
conditionReason: string | null;
@@ -284,6 +286,8 @@ export interface StageTiming {
budgetSeconds: number | null;
source: string;
object: string | null;
startedAt?: string | null;
finishedAt?: string | null;
}
export interface FollowerState {