Files
claudetools/.claude/scripts/component.b64
2026-05-23 10:56:40 -07:00

1 line
9.8 KiB
Plaintext


// WatchdogAlertsSection

type WdogStatus = "active" | "acknowledged" | "resolved";

function wdogStatus(alert: WatchdogAlert): WdogStatus {
  if (alert.resolved_at) return "resolved";
  if (alert.acknowledged_at) return "acknowledged";
  return "active";
}

function WatchdogAlertsSection() {
  const queryClient = useQueryClient();
  const { toast } = useToast();
  const [showAll, setShowAll] = useState(false);
  const [expandedLogId, setExpandedLogId] = useState<string | null>(null);

  const { data: allAlerts = [], isLoading } = useQuery({
    queryKey: ["watchdog-alerts"],
    queryFn: () => watchdogAlertsApi.list().then((r) => r.data),
    refetchInterval: 30000,
  });

  const alerts = showAll ? allAlerts : allAlerts.filter((a) => !a.resolved_at);

  const acknowledgeMutation = useMutation({
    mutationFn: (id: string) => watchdogAlertsApi.acknowledge(id),
    onSuccess: () => {
      queryClient.invalidateQueries({ queryKey: ["watchdog-alerts"] });
      toast({ type: "success", title: "Alert acknowledged" });
    },
    onError: (err: Error) =>
      toast({ type: "error", title: "Could not acknowledge", message: err.message }),
  });

  const resolveMutation = useMutation({
    mutationFn: (id: string) => watchdogAlertsApi.resolve(id),
    onSuccess: () => {
      queryClient.invalidateQueries({ queryKey: ["watchdog-alerts"] });
      toast({ type: "success", title: "Alert resolved" });
    },
    onError: (err: Error) =>
      toast({ type: "error", title: "Could not resolve", message: err.message }),
  });

  const deleteMutation = useMutation({
    mutationFn: (id: string) => watchdogAlertsApi.delete(id),
    onSuccess: () => {
      queryClient.invalidateQueries({ queryKey: ["watchdog-alerts"] });
      toast({ type: "success", title: "Alert deleted" });
    },
    onError: (err: Error) =>
      toast({ type: "error", title: "Could not delete", message: err.message }),
  });

  const isMutating =
    acknowledgeMutation.isPending ||
    resolveMutation.isPending ||
    deleteMutation.isPending;

  const activeCount = allAlerts.filter((a) => wdogStatus(a) === "active").length;

  return (
    <Card>
      <CardHeader>
        <div className="flex items-center justify-between gap-3">
          <CardTitle className="flex items-center gap-2">
            <AlertTriangle className="h-4 w-4 text-amber-500" />
            Watchdog Alerts
            {activeCount > 0 && (
              <span className="ml-1 rounded-full bg-amber-500/15 px-2 py-0.5 text-xs font-medium text-amber-600 dark:text-amber-400">
                {activeCount} active
              </span>
            )}
          </CardTitle>
          <button
            className="text-xs text-[hsl(var(--muted-foreground))] hover:text-[hsl(var(--foreground))] underline-offset-2 hover:underline"
            onClick={() => setShowAll((v) => !v)}
          >
            {showAll ? "Active only" : "Show all"}
          </button>
        </div>
      </CardHeader>
      <CardContent>
        {isLoading && (
          <p className="py-6 text-center text-sm text-[hsl(var(--muted-foreground))]">Loading...</p>
        )}
        {!isLoading && alerts.length === 0 && (
          <p className="py-6 text-center text-sm text-[hsl(var(--muted-foreground))]">
            {showAll ? "No watchdog alerts recorded." : "No active watchdog alerts."}
          </p>
        )}
        <div className="space-y-3">
          {alerts.map((alert) => {
            const status = wdogStatus(alert);
            const logExpanded = expandedLogId === alert.id;
            const statusColor =
              status === "active"
                ? "text-red-600 dark:text-red-400"
                : status === "acknowledged"
                ? "text-amber-600 dark:text-amber-400"
                : "text-[hsl(var(--muted-foreground))]";

            return (
              <div
                key={alert.id}
                className="rounded-lg border border-[hsl(var(--border))] p-4 space-y-2"
              >
                <div className="flex items-start justify-between gap-4">
                  <div className="space-y-0.5">
                    <div className="flex items-center gap-2">
                      <span className={`text-xs font-semibold uppercase tracking-wider ${statusColor}`}>
                        {status}
                      </span>
                      <span className="text-xs text-[hsl(var(--muted-foreground))]">
                        · {alert.restart_attempts} restart attempt
                        {alert.restart_attempts !== 1 ? "s" : ""}
                      </span>
                    </div>
                    <p className="text-xs text-[hsl(var(--muted-foreground))]">
                      Agent ID:{" "}
                      <span className="font-mono" title={alert.agent_id}>
                        {alert.agent_id.slice(0, 8)}…
                      </span>
                      {" "}·{" "}
                      Triggered: {formatRelative(alert.triggered_at)}
                      {alert.acknowledged_at && (
                        <>{" "}·{" "}Ack: {formatRelative(alert.acknowledged_at)}</>
                      )}
                      {alert.resolved_at && (
                        <>{" "}·{" "}Resolved: {formatRelative(alert.resolved_at)}</>
                      )}
                    </p>
                    {alert.last_error && (
                      <p className="text-sm text-[hsl(var(--foreground))]">
                        <span className="font-medium">Error:</span> {alert.last_error}
                      </p>
                    )}
                  </div>

                  <div className="flex shrink-0 items-center gap-1.5">
                    {status === "active" && (
                      <Button size="sm" variant="secondary" disabled={isMutating}
                        onClick={() => acknowledgeMutation.mutate(alert.id)}>
                        Acknowledge
                      </Button>
                    )}
                    {status !== "resolved" && (
                      <Button size="sm" variant="secondary" disabled={isMutating}
                        onClick={() => resolveMutation.mutate(alert.id)}>
                        Resolve
                      </Button>
                    )}
                    <Button size="sm" variant="ghost" disabled={isMutating}
                      onClick={() => deleteMutation.mutate(alert.id)}>
                      Delete
                    </Button>
                  </div>
                </div>

                {alert.log_tail && (
                  <div>
                    <button
                      className="flex items-center gap-1 text-xs text-[hsl(var(--muted-foreground))] hover:text-[hsl(var(--foreground))]"
                      onClick={() => setExpandedLogId(logExpanded ? null : alert.id)}
                    >
                      {logExpanded ? <ChevronDown className="h-3 w-3" /> : <ChevronRight className="h-3 w-3" />}
                      Agent log tail
                    </button>
                    {logExpanded && (
                      <pre className="mt-1 max-h-48 overflow-auto rounded bg-[hsl(var(--muted))]/50 p-2 text-xs font-mono text-[hsl(var(--foreground))]">
                        {alert.log_tail}
                      </pre>
                    )}
                  </div>
                )}
              </div>
            );
          })}
        </div>
      </CardContent>
    </Card>
  );
}

