- Bundle analyzer: @next/bundle-analyzer wired through next.config.ts,
new `bun run analyze` script (ANALYZE=true next build).
- Scheduled tasks gain:
- enabled flag round-tripped via a #DISABLED prefix on the crontab line
(preserves the mc-task marker + payload for re-enable).
- PATCH /api/schedule/tasks to toggle enabled.
- POST /api/schedule/tasks/run to execute a task immediately via
the same buildCommand used for the cron line (60s timeout, kills
child on client abort).
- Weekly preset in the UI (day-of-week selector), broader aria-labels
on all form selects. Human-readable cron renders "Tue at 04:30"
and next-run calc accounts for weekly.
- Hover-reveal Run now / Enable / Disable / Remove actions; disabled
tasks render at 60% opacity with a "disabled" badge and no next-run.
- /api/errors: minimal append-only JSONL reporter (200ms throttle, UA
and IP captured, fields length-bounded). Falls back to a stable
path under /home/minecraft/logs/ and never fails the client on
logging errors.
- ErrorReporter client (production-only) listens to window.error and
unhandledrejection, fingerprint-dedups via a bounded in-memory set,
sends with keepalive so unloads still flush.
- app/opengraph-image.tsx: 1200x630 dynamic PNG with live status dot
(green/red), player count, address. 60s memo on the status probe.
- A11y: aria-labels on log-line count select, scheduled-restart hour
and minute selects, copy-server-address button (plus focus-visible
ring). ModManager selected-mod pill upgraded to role=button with a
real accessible name and a proper × glyph.
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
104 lines
3.5 KiB
TypeScript
104 lines
3.5 KiB
TypeScript
"use client";
|
|
|
|
import { useQuery } from "@tanstack/react-query";
|
|
import { useState } from "react";
|
|
import { Card, CardContent, CardHeader, CardTitle } from "@/components/ui/card";
|
|
import { Skeleton } from "@/components/ui/skeleton";
|
|
import { StatusBadge, statusFromServer } from "@/components/StatusBadge";
|
|
|
|
type ServerStatus = {
|
|
online: boolean;
|
|
starting?: boolean;
|
|
players: { online: number; max: number };
|
|
version?: string;
|
|
motd?: string;
|
|
};
|
|
|
|
export function StatusCard() {
|
|
const [copied, setCopied] = useState(false);
|
|
|
|
const { data, isLoading } = useQuery<ServerStatus>({
|
|
queryKey: ["status"],
|
|
queryFn: () => fetch("/api/status").then((r) => r.json()),
|
|
refetchInterval: 60_000,
|
|
staleTime: 5000,
|
|
});
|
|
|
|
const copyIP = () => {
|
|
navigator.clipboard.writeText("minecraft.hurkicorgi.com");
|
|
setCopied(true);
|
|
setTimeout(() => setCopied(false), 1500);
|
|
};
|
|
|
|
return (
|
|
<Card>
|
|
<CardHeader className="pb-0">
|
|
<CardTitle className="text-base">Server Status</CardTitle>
|
|
</CardHeader>
|
|
<CardContent>
|
|
<div className="grid grid-cols-1 sm:grid-cols-2 gap-3">
|
|
{/* Status */}
|
|
<div className="rounded-lg bg-muted p-3 sm:p-4 text-center">
|
|
<p className="text-xs text-muted-foreground uppercase tracking-wider mb-2">
|
|
Status
|
|
</p>
|
|
{isLoading || !data ? (
|
|
<Skeleton className="h-5 w-20 mx-auto" />
|
|
) : (
|
|
<StatusBadge status={statusFromServer(data)} />
|
|
)}
|
|
</div>
|
|
|
|
{/* Players */}
|
|
<div className="rounded-lg bg-muted p-3 sm:p-4 text-center">
|
|
<p className="text-xs text-muted-foreground uppercase tracking-wider mb-1">
|
|
Players
|
|
</p>
|
|
{isLoading || !data ? (
|
|
<Skeleton className="h-8 w-16 mx-auto" />
|
|
) : (
|
|
<p className="text-2xl font-bold tabular-nums">
|
|
{data.online ? `${data.players.online}/${data.players.max}` : "0"}
|
|
</p>
|
|
)}
|
|
</div>
|
|
|
|
{/* Version */}
|
|
<div className="rounded-lg bg-muted p-3 sm:p-4 text-center">
|
|
<p className="text-xs text-muted-foreground uppercase tracking-wider mb-1">
|
|
Version
|
|
</p>
|
|
{isLoading || !data ? (
|
|
<Skeleton className="h-5 w-20 mx-auto" />
|
|
) : (
|
|
<p className="text-sm font-semibold">{data.version || "-"}</p>
|
|
)}
|
|
</div>
|
|
|
|
{/* Connect */}
|
|
<div className="rounded-lg bg-muted p-3 sm:p-4 text-center">
|
|
<p className="text-xs text-muted-foreground uppercase tracking-wider mb-1">
|
|
Connect
|
|
</p>
|
|
<button
|
|
onClick={copyIP}
|
|
aria-label="Copy server address"
|
|
className="w-full rounded-md border border-dashed border-border px-2 py-2 hover:border-primary/50 transition cursor-pointer min-h-[44px] focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring"
|
|
>
|
|
<p className="font-mono text-xs font-semibold text-foreground">
|
|
minecraft.hurkicorgi.com
|
|
</p>
|
|
<p className="text-xs text-muted-foreground mt-0.5">
|
|
{copied ? (
|
|
<span className="text-emerald-300">Copied!</span>
|
|
) : (
|
|
"tap to copy"
|
|
)}
|
|
</p>
|
|
</button>
|
|
</div>
|
|
</div>
|
|
</CardContent>
|
|
</Card>
|
|
);
|
|
}
|