2024-04-03 19:56:04 +03:00
|
|
|
import { FrigateConfig } from "@/types/frigateConfig";
|
2024-04-04 06:22:11 +03:00
|
|
|
import {
|
|
|
|
|
CameraDetectThreshold,
|
|
|
|
|
CameraFfmpegThreshold,
|
|
|
|
|
InferenceThreshold,
|
|
|
|
|
} from "@/types/graph";
|
2024-03-30 21:45:13 +03:00
|
|
|
import { FrigateStats, PotentialProblem } from "@/types/stats";
|
|
|
|
|
import { useMemo } from "react";
|
2024-04-03 19:56:04 +03:00
|
|
|
import useSWR from "swr";
|
2024-03-30 21:45:13 +03:00
|
|
|
|
|
|
|
|
export default function useStats(stats: FrigateStats | undefined) {
|
2024-04-03 19:56:04 +03:00
|
|
|
const { data: config } = useSWR<FrigateConfig>("config");
|
|
|
|
|
|
2024-03-30 21:45:13 +03:00
|
|
|
const potentialProblems = useMemo<PotentialProblem[]>(() => {
|
|
|
|
|
const problems: PotentialProblem[] = [];
|
|
|
|
|
|
|
|
|
|
if (!stats) {
|
|
|
|
|
return problems;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// check detectors for high inference speeds
|
|
|
|
|
Object.entries(stats["detectors"]).forEach(([key, det]) => {
|
2024-04-04 06:22:11 +03:00
|
|
|
if (det["inference_speed"] > InferenceThreshold.error) {
|
2024-03-30 21:45:13 +03:00
|
|
|
problems.push({
|
|
|
|
|
text: `${key} is very slow (${det["inference_speed"]} ms)`,
|
|
|
|
|
color: "text-danger",
|
|
|
|
|
});
|
2024-04-04 06:22:11 +03:00
|
|
|
} else if (det["inference_speed"] > InferenceThreshold.warning) {
|
2024-03-30 21:45:13 +03:00
|
|
|
problems.push({
|
|
|
|
|
text: `${key} is slow (${det["inference_speed"]} ms)`,
|
|
|
|
|
color: "text-orange-400",
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
// check for offline cameras
|
|
|
|
|
Object.entries(stats["cameras"]).forEach(([name, cam]) => {
|
2024-04-03 19:56:04 +03:00
|
|
|
if (!config) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (config.cameras[name].enabled && cam["camera_fps"] == 0) {
|
2024-03-30 21:45:13 +03:00
|
|
|
problems.push({
|
|
|
|
|
text: `${name.replaceAll("_", " ")} is offline`,
|
|
|
|
|
color: "text-danger",
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
// check camera cpu usages
|
|
|
|
|
Object.entries(stats["cameras"]).forEach(([name, cam]) => {
|
|
|
|
|
const ffmpegAvg = parseFloat(
|
2024-04-01 17:23:57 +03:00
|
|
|
stats["cpu_usages"][cam["ffmpeg_pid"]]?.cpu_average,
|
|
|
|
|
);
|
|
|
|
|
const detectAvg = parseFloat(
|
|
|
|
|
stats["cpu_usages"][cam["pid"]]?.cpu_average,
|
2024-03-30 21:45:13 +03:00
|
|
|
);
|
|
|
|
|
|
2024-04-04 06:22:11 +03:00
|
|
|
if (!isNaN(ffmpegAvg) && ffmpegAvg >= CameraFfmpegThreshold.error) {
|
2024-03-30 21:45:13 +03:00
|
|
|
problems.push({
|
|
|
|
|
text: `${name.replaceAll("_", " ")} has high FFMPEG CPU usage (${ffmpegAvg}%)`,
|
|
|
|
|
color: "text-danger",
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
2024-04-04 06:22:11 +03:00
|
|
|
if (!isNaN(detectAvg) && detectAvg >= CameraDetectThreshold.error) {
|
2024-03-30 21:45:13 +03:00
|
|
|
problems.push({
|
|
|
|
|
text: `${name.replaceAll("_", " ")} has high detect CPU usage (${detectAvg}%)`,
|
|
|
|
|
color: "text-danger",
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
return problems;
|
2024-04-03 19:56:04 +03:00
|
|
|
}, [config, stats]);
|
2024-03-30 21:45:13 +03:00
|
|
|
|
|
|
|
|
return { potentialProblems };
|
|
|
|
|
}
|