88 lines
3.1 KiB
TypeScript
88 lines
3.1 KiB
TypeScript
import React from "react";
|
|
import { List, Activity } from "lucide-react";
|
|
import { useTranslation } from "react-i18next";
|
|
import type { ServerMetrics } from "@/ui/main-axios.ts";
|
|
|
|
interface ProcessesWidgetProps {
|
|
metrics: ServerMetrics | null;
|
|
metricsHistory: ServerMetrics[];
|
|
}
|
|
|
|
export function ProcessesWidget({ metrics }: ProcessesWidgetProps) {
|
|
const { t } = useTranslation();
|
|
|
|
const metricsWithProcesses = metrics as ServerMetrics & {
|
|
processes?: {
|
|
total?: number;
|
|
running?: number;
|
|
top?: Array<{
|
|
pid: number;
|
|
cpu: number;
|
|
mem: number;
|
|
command: string;
|
|
user: string;
|
|
}>;
|
|
};
|
|
};
|
|
const processes = metricsWithProcesses?.processes;
|
|
const topProcesses = processes?.top || [];
|
|
|
|
return (
|
|
<div className="h-full w-full p-4 rounded-lg bg-dark-bg-darker border border-dark-border/50 hover:bg-dark-bg/70 transition-colors duration-200 flex flex-col overflow-hidden">
|
|
<div className="flex items-center gap-2 flex-shrink-0 mb-3">
|
|
<List className="h-5 w-5 text-yellow-400" />
|
|
<h3 className="font-semibold text-lg text-white">
|
|
{t("serverStats.processes")}
|
|
</h3>
|
|
</div>
|
|
|
|
<div className="flex items-center justify-between mb-3 pb-2 border-b border-dark-border/30">
|
|
<div className="text-sm text-gray-400">
|
|
{t("serverStats.totalProcesses")}:{" "}
|
|
<span className="text-white font-semibold">
|
|
{processes?.total ?? "N/A"}
|
|
</span>
|
|
</div>
|
|
<div className="text-sm text-gray-400">
|
|
{t("serverStats.running")}:{" "}
|
|
<span className="text-green-400 font-semibold">
|
|
{processes?.running ?? "N/A"}
|
|
</span>
|
|
</div>
|
|
</div>
|
|
|
|
<div className="overflow-auto flex-1">
|
|
{topProcesses.length === 0 ? (
|
|
<div className="flex flex-col items-center justify-center py-12 text-gray-400">
|
|
<Activity className="h-10 w-10 mb-3 opacity-50" />
|
|
<p className="text-sm">{t("serverStats.noProcessesFound")}</p>
|
|
</div>
|
|
) : (
|
|
<div className="space-y-2">
|
|
{topProcesses.map((proc) => (
|
|
<div
|
|
key={proc.pid}
|
|
className="p-2.5 rounded-lg bg-dark-bg/30 hover:bg-dark-bg/50 transition-colors border border-dark-border/20"
|
|
>
|
|
<div className="flex items-center justify-between mb-1.5">
|
|
<span className="text-xs font-mono text-gray-400 font-medium">
|
|
PID: {proc.pid}
|
|
</span>
|
|
<div className="flex gap-3 text-xs font-medium">
|
|
<span className="text-blue-400">CPU: {proc.cpu}%</span>
|
|
<span className="text-green-400">MEM: {proc.mem}%</span>
|
|
</div>
|
|
</div>
|
|
<div className="text-xs text-white font-mono truncate mb-1">
|
|
{proc.command}
|
|
</div>
|
|
<div className="text-xs text-gray-500">User: {proc.user}</div>
|
|
</div>
|
|
))}
|
|
</div>
|
|
)}
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|