package templates import ( "bytes" "context" "fmt" "time" "vetting/internal/model" ) // subStepDuration formats a sub-step's elapsed time the same way // stageDuration does for pipeline nodes. Empty string when not started. func subStepDuration(ss model.SubStep) string { if ss.StartedAt == nil { return "" } end := time.Now() if ss.CompletedAt != nil { end = *ss.CompletedAt } d := end.Sub(*ss.StartedAt) if d < 0 { d = 0 } switch { case d < time.Second: return fmt.Sprintf("%dms", int(d/time.Millisecond)) case d < 10*time.Second: return fmt.Sprintf("%.1fs", d.Seconds()) case d < time.Minute: return fmt.Sprintf("%ds", int(d/time.Second)) case d < time.Hour: return fmt.Sprintf("%dm", int(d/time.Minute)) default: return fmt.Sprintf("%dh", int(d/time.Hour)) } } // subStepMarker mirrors stageMarker — a single-char glyph used inside the // state badge. StageState values reused verbatim for sub-steps. func subStepMarker(s model.StageState) string { switch s { case model.StagePassed: return "✓" case model.StageFailed: return "!" case model.StageRunning: return "●" case model.StageSkipped: return "–" } return "" } // SubStepRow renders one sub-step entry for the expanded-step pane. The // outer