2024-05-29 20:05:39 +02:00
|
|
|
import { StorageGraph } from "@/components/graph/StorageGraph";
|
2024-04-04 18:24:23 +02:00
|
|
|
import { FrigateStats } from "@/types/stats";
|
2024-08-30 15:11:54 +02:00
|
|
|
import { getUnitSize } from "@/utils/storageUtil";
|
2024-04-04 18:24:23 +02:00
|
|
|
import { useMemo } from "react";
|
|
|
|
import useSWR from "swr";
|
|
|
|
|
|
|
|
type CameraStorage = {
|
|
|
|
[key: string]: {
|
|
|
|
bandwidth: number;
|
|
|
|
usage: number;
|
|
|
|
usage_percent: number;
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
type StorageMetricsProps = {
|
|
|
|
setLastUpdated: (last: number) => void;
|
|
|
|
};
|
|
|
|
export default function StorageMetrics({
|
|
|
|
setLastUpdated,
|
|
|
|
}: StorageMetricsProps) {
|
|
|
|
const { data: cameraStorage } = useSWR<CameraStorage>("recordings/storage");
|
|
|
|
const { data: stats } = useSWR<FrigateStats>("stats");
|
|
|
|
|
|
|
|
const totalStorage = useMemo(() => {
|
|
|
|
if (!cameraStorage || !stats) {
|
|
|
|
return undefined;
|
|
|
|
}
|
|
|
|
|
|
|
|
const totalStorage = {
|
|
|
|
used: 0,
|
|
|
|
total: stats.service.storage["/media/frigate/recordings"]["total"],
|
|
|
|
};
|
|
|
|
|
|
|
|
Object.values(cameraStorage).forEach(
|
|
|
|
(cam) => (totalStorage.used += cam.usage),
|
|
|
|
);
|
|
|
|
setLastUpdated(Date.now() / 1000);
|
|
|
|
return totalStorage;
|
|
|
|
}, [cameraStorage, stats, setLastUpdated]);
|
|
|
|
|
|
|
|
if (!cameraStorage || !stats || !totalStorage) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
return (
|
2024-06-03 20:43:30 +02:00
|
|
|
<div className="scrollbar-container mt-4 flex size-full flex-col overflow-y-auto">
|
2024-05-14 17:06:44 +02:00
|
|
|
<div className="text-sm font-medium text-muted-foreground">Overview</div>
|
|
|
|
<div className="mt-4 grid grid-cols-1 gap-2 sm:grid-cols-3">
|
|
|
|
<div className="flex-col rounded-lg bg-background_alt p-2.5 md:rounded-2xl">
|
2024-04-04 18:24:23 +02:00
|
|
|
<div className="mb-5">Recordings</div>
|
|
|
|
<StorageGraph
|
|
|
|
graphId="general-recordings"
|
|
|
|
used={totalStorage.used}
|
|
|
|
total={totalStorage.total}
|
|
|
|
/>
|
|
|
|
</div>
|
2024-05-14 17:06:44 +02:00
|
|
|
<div className="flex-col rounded-lg bg-background_alt p-2.5 md:rounded-2xl">
|
2024-04-04 18:24:23 +02:00
|
|
|
<div className="mb-5">/tmp/cache</div>
|
|
|
|
<StorageGraph
|
|
|
|
graphId="general-cache"
|
|
|
|
used={stats.service.storage["/tmp/cache"]["used"]}
|
|
|
|
total={stats.service.storage["/tmp/cache"]["total"]}
|
|
|
|
/>
|
|
|
|
</div>
|
2024-05-14 17:06:44 +02:00
|
|
|
<div className="flex-col rounded-lg bg-background_alt p-2.5 md:rounded-2xl">
|
2024-04-04 18:24:23 +02:00
|
|
|
<div className="mb-5">/dev/shm</div>
|
|
|
|
<StorageGraph
|
|
|
|
graphId="general-shared-memory"
|
|
|
|
used={stats.service.storage["/dev/shm"]["used"]}
|
|
|
|
total={stats.service.storage["/dev/shm"]["total"]}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
</div>
|
2024-05-14 17:06:44 +02:00
|
|
|
<div className="mt-4 text-sm font-medium text-muted-foreground">
|
2024-04-04 18:24:23 +02:00
|
|
|
Camera Storage
|
|
|
|
</div>
|
2024-05-14 17:06:44 +02:00
|
|
|
<div className="mt-4 grid grid-cols-1 gap-2 sm:grid-cols-3">
|
2024-04-04 18:24:23 +02:00
|
|
|
{Object.keys(cameraStorage).map((camera) => (
|
2024-05-14 17:06:44 +02:00
|
|
|
<div className="flex-col rounded-lg bg-background_alt p-2.5 md:rounded-2xl">
|
2024-08-30 15:11:54 +02:00
|
|
|
<div className="mb-5 flex flex-row items-center justify-between">
|
|
|
|
<div className="capitalize">{camera.replaceAll("_", " ")}</div>
|
|
|
|
<div className="text-xs text-muted-foreground">
|
|
|
|
{getUnitSize(cameraStorage[camera].bandwidth)} / hour
|
|
|
|
</div>
|
|
|
|
</div>
|
2024-04-04 18:24:23 +02:00
|
|
|
<StorageGraph
|
|
|
|
graphId={`${camera}-storage`}
|
|
|
|
used={cameraStorage[camera].usage}
|
|
|
|
total={totalStorage.used}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
))}
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|