2023-12-31 16:31:33 +03:00
|
|
|
import { Button } from "@/components/ui/button";
|
2024-03-24 20:23:39 +03:00
|
|
|
import { ToggleGroup, ToggleGroupItem } from "@/components/ui/toggle-group";
|
2025-01-14 04:37:29 +03:00
|
|
|
import { LogLine, LogSeverity, LogType, logTypes } from "@/types/log";
|
2023-12-31 16:31:33 +03:00
|
|
|
import copy from "copy-to-clipboard";
|
2025-01-14 04:37:29 +03:00
|
|
|
import { useCallback, useEffect, useRef, useState } from "react";
|
2024-04-03 19:55:13 +03:00
|
|
|
import axios from "axios";
|
2024-04-07 23:36:08 +03:00
|
|
|
import LogInfoDialog from "@/components/overlay/LogInfoDialog";
|
|
|
|
|
import { LogChip } from "@/components/indicators/Chip";
|
|
|
|
|
import { LogLevelFilterButton } from "@/components/filter/LogLevelFilter";
|
2025-01-14 04:37:29 +03:00
|
|
|
import { FaCopy, FaDownload } from "react-icons/fa";
|
2024-04-07 23:36:08 +03:00
|
|
|
import { Toaster } from "@/components/ui/sonner";
|
|
|
|
|
import { toast } from "sonner";
|
2024-04-14 19:14:10 +03:00
|
|
|
import ActivityIndicator from "@/components/indicators/activity-indicator";
|
2024-05-07 17:00:25 +03:00
|
|
|
import { cn } from "@/lib/utils";
|
2024-05-29 21:05:39 +03:00
|
|
|
import { parseLogLines } from "@/utils/logUtil";
|
2024-08-19 17:53:33 +03:00
|
|
|
import { ScrollArea, ScrollBar } from "@/components/ui/scroll-area";
|
|
|
|
|
import scrollIntoView from "scroll-into-view-if-needed";
|
2025-01-14 04:37:29 +03:00
|
|
|
import { LazyLog } from "@melloware/react-logviewer";
|
|
|
|
|
import useKeyboardListener from "@/hooks/use-keyboard-listener";
|
|
|
|
|
import EnhancedScrollFollow from "@/components/dynamic/EnhancedScrollFollow";
|
|
|
|
|
import { MdCircle } from "react-icons/md";
|
2024-04-03 19:55:13 +03:00
|
|
|
|
2023-12-08 16:33:22 +03:00
|
|
|
function Logs() {
|
2023-12-31 16:31:33 +03:00
|
|
|
const [logService, setLogService] = useState<LogType>("frigate");
|
2024-08-19 17:53:33 +03:00
|
|
|
const tabsRef = useRef<HTMLDivElement | null>(null);
|
2025-01-14 04:37:29 +03:00
|
|
|
const lazyLogWrapperRef = useRef<HTMLDivElement>(null);
|
|
|
|
|
const [logs, setLogs] = useState<string[]>([]);
|
|
|
|
|
const [filterSeverity, setFilterSeverity] = useState<LogSeverity[]>();
|
|
|
|
|
const [selectedLog, setSelectedLog] = useState<LogLine>();
|
|
|
|
|
const lazyLogRef = useRef<LazyLog>(null);
|
2023-12-31 16:31:33 +03:00
|
|
|
|
2024-04-12 15:31:30 +03:00
|
|
|
useEffect(() => {
|
2024-04-16 23:55:24 +03:00
|
|
|
document.title = `${logService[0].toUpperCase()}${logService.substring(1)} Logs - Frigate`;
|
2024-04-12 15:31:30 +03:00
|
|
|
}, [logService]);
|
|
|
|
|
|
2024-08-19 17:53:33 +03:00
|
|
|
useEffect(() => {
|
|
|
|
|
if (tabsRef.current) {
|
|
|
|
|
const element = tabsRef.current.querySelector(
|
|
|
|
|
`[data-nav-item="${logService}"]`,
|
|
|
|
|
);
|
|
|
|
|
if (element instanceof HTMLElement) {
|
|
|
|
|
scrollIntoView(element, {
|
|
|
|
|
behavior: "smooth",
|
|
|
|
|
inline: "start",
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}, [tabsRef, logService]);
|
|
|
|
|
|
2025-01-14 04:37:29 +03:00
|
|
|
// handlers
|
2024-04-03 19:55:13 +03:00
|
|
|
|
2023-12-31 16:31:33 +03:00
|
|
|
const handleCopyLogs = useCallback(() => {
|
2025-01-14 04:37:29 +03:00
|
|
|
if (logs.length) {
|
2024-04-03 19:55:13 +03:00
|
|
|
copy(logs.join("\n"));
|
2025-01-14 04:37:29 +03:00
|
|
|
toast.success("Copied logs to clipboard");
|
2024-04-07 23:36:08 +03:00
|
|
|
} else {
|
|
|
|
|
toast.error("Could not copy logs to clipboard");
|
2024-04-03 19:55:13 +03:00
|
|
|
}
|
2025-01-14 04:37:29 +03:00
|
|
|
}, [logs]);
|
2023-12-31 16:31:33 +03:00
|
|
|
|
2024-08-19 17:53:33 +03:00
|
|
|
const handleDownloadLogs = useCallback(() => {
|
|
|
|
|
axios
|
2025-01-14 04:37:29 +03:00
|
|
|
.get(`api/logs/${logService}?download=true`)
|
2024-08-19 17:53:33 +03:00
|
|
|
.then((resp) => {
|
|
|
|
|
const element = document.createElement("a");
|
|
|
|
|
element.setAttribute(
|
|
|
|
|
"href",
|
|
|
|
|
"data:text/plain;charset=utf-8," + encodeURIComponent(resp.data),
|
|
|
|
|
);
|
|
|
|
|
element.setAttribute("download", `${logService}-logs.txt`);
|
|
|
|
|
|
|
|
|
|
element.style.display = "none";
|
|
|
|
|
document.body.appendChild(element);
|
|
|
|
|
|
|
|
|
|
element.click();
|
|
|
|
|
|
|
|
|
|
document.body.removeChild(element);
|
|
|
|
|
})
|
|
|
|
|
.catch(() => {});
|
|
|
|
|
}, [logService]);
|
|
|
|
|
|
2025-01-14 04:37:29 +03:00
|
|
|
const handleRowClick = useCallback(
|
|
|
|
|
(rowInfo: { lineNumber: number; rowIndex: number }) => {
|
|
|
|
|
const clickedLine = parseLogLines(logService, [
|
|
|
|
|
logs[rowInfo.rowIndex],
|
|
|
|
|
])[0];
|
|
|
|
|
setSelectedLog(clickedLine);
|
2024-02-24 03:25:00 +03:00
|
|
|
},
|
2025-01-14 04:37:29 +03:00
|
|
|
[logs, logService],
|
2024-02-24 03:25:00 +03:00
|
|
|
);
|
2024-04-03 19:55:13 +03:00
|
|
|
|
2025-01-14 04:37:29 +03:00
|
|
|
// filter
|
2024-04-03 19:55:13 +03:00
|
|
|
|
2025-01-14 04:37:29 +03:00
|
|
|
const filterLines = useCallback(
|
|
|
|
|
(lines: string[]) => {
|
|
|
|
|
// console.log(lines);
|
|
|
|
|
if (!filterSeverity?.length) return lines;
|
|
|
|
|
|
|
|
|
|
return lines.filter((line) => {
|
|
|
|
|
// console.log(line);
|
|
|
|
|
const parsedLine = parseLogLines(logService, [line])[0];
|
|
|
|
|
return filterSeverity.includes(parsedLine.severity);
|
|
|
|
|
});
|
2024-04-03 19:55:13 +03:00
|
|
|
},
|
2025-01-14 04:37:29 +03:00
|
|
|
[filterSeverity, logService],
|
2024-04-03 19:55:13 +03:00
|
|
|
);
|
|
|
|
|
|
2025-01-14 04:37:29 +03:00
|
|
|
// fetchers
|
|
|
|
|
|
|
|
|
|
const fetchInitialLogs = useCallback(async () => {
|
|
|
|
|
try {
|
|
|
|
|
const response = await axios.get(`logs/${logService}`);
|
|
|
|
|
if (
|
|
|
|
|
response.status === 200 &&
|
|
|
|
|
response.data &&
|
|
|
|
|
Array.isArray(response.data.lines)
|
|
|
|
|
) {
|
|
|
|
|
const filteredLines = filterLines(response.data.lines);
|
|
|
|
|
setLogs(filteredLines);
|
|
|
|
|
}
|
|
|
|
|
} catch (error) {
|
|
|
|
|
const errorMessage =
|
|
|
|
|
error instanceof Error ? error.message : "An unknown error occurred";
|
|
|
|
|
toast.error(`Error fetching logs: ${errorMessage}`, {
|
|
|
|
|
position: "top-center",
|
|
|
|
|
});
|
2024-04-03 19:55:13 +03:00
|
|
|
}
|
2025-01-14 04:37:29 +03:00
|
|
|
}, [logService, filterLines]);
|
|
|
|
|
|
|
|
|
|
const [isStreaming, setIsStreaming] = useState(false);
|
|
|
|
|
const abortControllerRef = useRef<AbortController | null>(null);
|
|
|
|
|
|
|
|
|
|
const fetchLogsStream = useCallback(() => {
|
|
|
|
|
// Cancel any existing stream
|
|
|
|
|
abortControllerRef.current?.abort();
|
|
|
|
|
const abortController = new AbortController();
|
|
|
|
|
abortControllerRef.current = abortController;
|
|
|
|
|
let buffer = "";
|
|
|
|
|
const decoder = new TextDecoder();
|
|
|
|
|
|
|
|
|
|
const processStreamChunk = (
|
|
|
|
|
reader: ReadableStreamDefaultReader<Uint8Array>,
|
|
|
|
|
): Promise<void> => {
|
|
|
|
|
return reader.read().then(({ done, value }) => {
|
|
|
|
|
if (done) return;
|
|
|
|
|
|
|
|
|
|
// Decode the chunk and add it to our buffer
|
|
|
|
|
buffer += decoder.decode(value, { stream: true });
|
|
|
|
|
|
|
|
|
|
// Split on newlines, keeping any partial line in the buffer
|
|
|
|
|
const lines = buffer.split("\n");
|
|
|
|
|
|
|
|
|
|
// Keep the last partial line
|
|
|
|
|
buffer = lines.pop() || "";
|
|
|
|
|
|
|
|
|
|
// Filter and append complete lines
|
|
|
|
|
if (lines.length > 0) {
|
|
|
|
|
const filteredLines = filterSeverity?.length
|
|
|
|
|
? lines.filter((line) => {
|
|
|
|
|
const parsedLine = parseLogLines(logService, [line])[0];
|
|
|
|
|
return filterSeverity.includes(parsedLine.severity);
|
|
|
|
|
})
|
|
|
|
|
: lines;
|
|
|
|
|
if (filteredLines.length > 0) {
|
|
|
|
|
lazyLogRef.current?.appendLines(filteredLines);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
// Process next chunk
|
|
|
|
|
return processStreamChunk(reader);
|
|
|
|
|
});
|
|
|
|
|
};
|
2024-04-03 19:55:13 +03:00
|
|
|
|
2025-01-14 04:37:29 +03:00
|
|
|
fetch(`api/logs/${logService}?stream=true`, {
|
|
|
|
|
signal: abortController.signal,
|
|
|
|
|
})
|
|
|
|
|
.then((response): Promise<void> => {
|
|
|
|
|
if (!response.ok) {
|
|
|
|
|
throw new Error(
|
|
|
|
|
`Error while fetching log stream, status: ${response.status}`,
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
const reader = response.body?.getReader();
|
|
|
|
|
if (!reader) {
|
|
|
|
|
throw new Error("No reader available");
|
|
|
|
|
}
|
|
|
|
|
return processStreamChunk(reader);
|
|
|
|
|
})
|
|
|
|
|
.catch((error) => {
|
|
|
|
|
if (error.name !== "AbortError") {
|
|
|
|
|
const errorMessage =
|
|
|
|
|
error instanceof Error
|
|
|
|
|
? error.message
|
|
|
|
|
: "An unknown error occurred";
|
|
|
|
|
toast.error(`Error while streaming logs: ${errorMessage}`);
|
|
|
|
|
setIsStreaming(false);
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
}, [logService, filterSeverity]);
|
2024-04-03 19:55:13 +03:00
|
|
|
|
2025-01-14 04:37:29 +03:00
|
|
|
useEffect(() => {
|
|
|
|
|
fetchInitialLogs().then(() => {
|
|
|
|
|
// Start streaming after initial load
|
|
|
|
|
setIsStreaming(true);
|
|
|
|
|
fetchLogsStream();
|
2024-04-03 19:55:13 +03:00
|
|
|
});
|
2024-04-07 23:36:08 +03:00
|
|
|
|
2025-01-14 04:37:29 +03:00
|
|
|
return () => {
|
|
|
|
|
abortControllerRef.current?.abort();
|
|
|
|
|
setIsStreaming(false);
|
|
|
|
|
};
|
|
|
|
|
}, [fetchInitialLogs, fetchLogsStream]);
|
2024-04-07 23:36:08 +03:00
|
|
|
|
2025-01-14 04:37:29 +03:00
|
|
|
// keyboard listener
|
2024-05-29 21:05:39 +03:00
|
|
|
|
|
|
|
|
useKeyboardListener(
|
|
|
|
|
["PageDown", "PageUp", "ArrowDown", "ArrowUp"],
|
2024-06-18 17:32:17 +03:00
|
|
|
(key, modifiers) => {
|
2025-01-14 04:37:29 +03:00
|
|
|
if (!key || !modifiers.down || !lazyLogWrapperRef.current) {
|
2024-05-29 21:05:39 +03:00
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2025-01-14 04:37:29 +03:00
|
|
|
const container =
|
|
|
|
|
lazyLogWrapperRef.current.querySelector(".react-lazylog");
|
|
|
|
|
|
|
|
|
|
const logLineHeight = container?.querySelector(".log-line")?.clientHeight;
|
|
|
|
|
|
|
|
|
|
if (!logLineHeight) {
|
|
|
|
|
return;
|
2024-05-29 21:05:39 +03:00
|
|
|
}
|
2025-01-14 04:37:29 +03:00
|
|
|
|
|
|
|
|
const scrollAmount = key.includes("Page")
|
|
|
|
|
? logLineHeight * 10
|
|
|
|
|
: logLineHeight;
|
|
|
|
|
const direction = key.includes("Down") ? 1 : -1;
|
|
|
|
|
container?.scrollBy({ top: scrollAmount * direction });
|
2024-05-29 21:05:39 +03:00
|
|
|
},
|
|
|
|
|
);
|
|
|
|
|
|
2025-01-14 04:37:29 +03:00
|
|
|
// format lines
|
2024-10-31 22:48:26 +03:00
|
|
|
|
2025-01-14 04:37:29 +03:00
|
|
|
const lineBufferRef = useRef<string>("");
|
2024-10-31 22:48:26 +03:00
|
|
|
|
2025-01-14 04:37:29 +03:00
|
|
|
const formatPart = useCallback(
|
|
|
|
|
(text: string) => {
|
|
|
|
|
lineBufferRef.current += text;
|
2024-10-31 22:48:26 +03:00
|
|
|
|
2025-01-14 04:37:29 +03:00
|
|
|
if (text.endsWith("\n")) {
|
|
|
|
|
const completeLine = lineBufferRef.current.trim();
|
|
|
|
|
lineBufferRef.current = "";
|
2024-10-31 22:48:26 +03:00
|
|
|
|
2025-01-14 04:37:29 +03:00
|
|
|
if (completeLine) {
|
|
|
|
|
const parsedLine = parseLogLines(logService, [completeLine])[0];
|
|
|
|
|
return (
|
|
|
|
|
<LogLineData
|
|
|
|
|
line={parsedLine}
|
|
|
|
|
logService={logService}
|
|
|
|
|
onClickSeverity={() => setFilterSeverity([parsedLine.severity])}
|
|
|
|
|
onSelect={() => setSelectedLog(parsedLine)}
|
|
|
|
|
/>
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
}
|
2024-10-31 22:48:26 +03:00
|
|
|
|
2025-01-14 04:37:29 +03:00
|
|
|
return null;
|
|
|
|
|
},
|
|
|
|
|
[logService, setFilterSeverity, setSelectedLog],
|
|
|
|
|
);
|
2024-10-31 22:48:26 +03:00
|
|
|
|
2023-12-08 16:33:22 +03:00
|
|
|
return (
|
2024-05-14 18:06:44 +03:00
|
|
|
<div className="flex size-full flex-col p-2">
|
2024-05-04 22:54:50 +03:00
|
|
|
<Toaster position="top-center" closeButton={true} />
|
2024-04-07 23:36:08 +03:00
|
|
|
<LogInfoDialog logLine={selectedLog} setLogLine={setSelectedLog} />
|
|
|
|
|
|
2024-08-19 17:53:33 +03:00
|
|
|
<div className="relative flex h-11 w-full items-center justify-between">
|
|
|
|
|
<ScrollArea className="w-full whitespace-nowrap">
|
|
|
|
|
<div ref={tabsRef} className="flex flex-row">
|
|
|
|
|
<ToggleGroup
|
|
|
|
|
type="single"
|
|
|
|
|
size="sm"
|
|
|
|
|
value={logService}
|
|
|
|
|
onValueChange={(value: LogType) => {
|
|
|
|
|
if (value) {
|
|
|
|
|
setLogs([]);
|
|
|
|
|
setFilterSeverity(undefined);
|
|
|
|
|
setLogService(value);
|
|
|
|
|
}
|
2025-01-14 04:37:29 +03:00
|
|
|
}}
|
2024-03-24 20:23:39 +03:00
|
|
|
>
|
2024-08-19 17:53:33 +03:00
|
|
|
{Object.values(logTypes).map((item) => (
|
|
|
|
|
<ToggleGroupItem
|
|
|
|
|
key={item}
|
|
|
|
|
className={`flex items-center justify-between gap-2 ${logService == item ? "" : "text-muted-foreground"}`}
|
|
|
|
|
value={item}
|
|
|
|
|
data-nav-item={item}
|
|
|
|
|
aria-label={`Select ${item}`}
|
|
|
|
|
>
|
|
|
|
|
<div className="capitalize">{item}</div>
|
|
|
|
|
</ToggleGroupItem>
|
|
|
|
|
))}
|
|
|
|
|
</ToggleGroup>
|
|
|
|
|
<ScrollBar orientation="horizontal" className="h-0" />
|
|
|
|
|
</div>
|
|
|
|
|
</ScrollArea>
|
2024-04-07 23:36:08 +03:00
|
|
|
<div className="flex items-center gap-2">
|
2024-03-24 20:23:39 +03:00
|
|
|
<Button
|
2024-05-14 18:06:44 +03:00
|
|
|
className="flex items-center justify-between gap-2"
|
2024-10-23 01:07:42 +03:00
|
|
|
aria-label="Copy logs to clipboard"
|
2024-03-24 20:23:39 +03:00
|
|
|
size="sm"
|
|
|
|
|
onClick={handleCopyLogs}
|
|
|
|
|
>
|
2024-04-16 23:55:24 +03:00
|
|
|
<FaCopy className="text-secondary-foreground" />
|
2024-05-14 18:06:44 +03:00
|
|
|
<div className="hidden text-primary md:block">
|
2024-04-07 23:36:08 +03:00
|
|
|
Copy to Clipboard
|
|
|
|
|
</div>
|
2024-03-24 20:23:39 +03:00
|
|
|
</Button>
|
2024-08-19 17:53:33 +03:00
|
|
|
<Button
|
|
|
|
|
className="flex items-center justify-between gap-2"
|
2024-10-23 01:07:42 +03:00
|
|
|
aria-label="Download logs"
|
2024-08-19 17:53:33 +03:00
|
|
|
size="sm"
|
|
|
|
|
onClick={handleDownloadLogs}
|
|
|
|
|
>
|
|
|
|
|
<FaDownload className="text-secondary-foreground" />
|
|
|
|
|
<div className="hidden text-primary md:block">Download</div>
|
|
|
|
|
</Button>
|
2024-04-07 23:36:08 +03:00
|
|
|
<LogLevelFilterButton
|
|
|
|
|
selectedLabels={filterSeverity}
|
|
|
|
|
updateLabelFilter={setFilterSeverity}
|
2025-01-14 04:37:29 +03:00
|
|
|
// setStreaming={setStreaming}
|
2024-04-07 23:36:08 +03:00
|
|
|
/>
|
2023-12-31 16:31:33 +03:00
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
|
2025-01-14 04:37:29 +03:00
|
|
|
<div className="font-mono relative my-2 flex size-full flex-col overflow-hidden whitespace-pre-wrap rounded-md border border-secondary bg-background_alt text-xs sm:p-1">
|
2025-01-15 22:33:25 +03:00
|
|
|
<div className="grid grid-cols-5 *:px-0 *:py-3 *:text-sm *:text-primary/40 md:grid-cols-12">
|
2025-01-14 04:37:29 +03:00
|
|
|
<div className="ml-1 flex items-center p-1 capitalize">Type</div>
|
2025-01-15 22:33:25 +03:00
|
|
|
<div className="col-span-2 flex items-center lg:col-span-1">
|
2024-04-03 19:55:13 +03:00
|
|
|
Timestamp
|
|
|
|
|
</div>
|
2025-01-14 04:37:29 +03:00
|
|
|
<div
|
|
|
|
|
className={cn(
|
|
|
|
|
"flex items-center",
|
|
|
|
|
logService == "frigate" ? "col-span-2" : "col-span-1",
|
|
|
|
|
)}
|
|
|
|
|
>
|
|
|
|
|
Tag
|
|
|
|
|
</div>
|
|
|
|
|
<div
|
|
|
|
|
className={cn(
|
2025-01-15 22:33:25 +03:00
|
|
|
"col-span-5 flex items-center",
|
|
|
|
|
logService == "frigate" ? "md:col-span-7" : "md:col-span-8",
|
2025-01-14 04:37:29 +03:00
|
|
|
)}
|
|
|
|
|
>
|
|
|
|
|
<div className="flex flex-1">Message</div>
|
|
|
|
|
{isStreaming && (
|
|
|
|
|
<div className="flex flex-row justify-end">
|
|
|
|
|
<MdCircle className="mr-2 size-2 animate-pulse text-selected shadow-selected drop-shadow-md" />
|
|
|
|
|
</div>
|
|
|
|
|
)}
|
2024-04-03 19:55:13 +03:00
|
|
|
</div>
|
|
|
|
|
</div>
|
2024-04-07 23:36:08 +03:00
|
|
|
|
2025-01-14 04:37:29 +03:00
|
|
|
<div ref={lazyLogWrapperRef} className="size-full">
|
|
|
|
|
<EnhancedScrollFollow
|
|
|
|
|
startFollowing={true}
|
|
|
|
|
render={({ follow, onScroll }) => (
|
|
|
|
|
<LazyLog
|
|
|
|
|
ref={lazyLogRef}
|
|
|
|
|
enableLineNumbers={false}
|
|
|
|
|
selectableLines
|
|
|
|
|
lineClassName="text-primary bg-background"
|
|
|
|
|
highlightLineClassName="bg-primary/20"
|
|
|
|
|
onRowClick={handleRowClick}
|
|
|
|
|
formatPart={formatPart}
|
|
|
|
|
text={logs.join("\n")}
|
|
|
|
|
follow={follow}
|
|
|
|
|
onScroll={onScroll}
|
|
|
|
|
loadingComponent={<ActivityIndicator />}
|
|
|
|
|
/>
|
|
|
|
|
)}
|
|
|
|
|
/>
|
2024-04-07 23:36:08 +03:00
|
|
|
</div>
|
2024-04-03 19:55:13 +03:00
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
type LogLineDataProps = {
|
2025-01-14 04:37:29 +03:00
|
|
|
className?: string;
|
2024-04-03 19:55:13 +03:00
|
|
|
line: LogLine;
|
2025-01-14 04:37:29 +03:00
|
|
|
logService: string;
|
2024-04-07 23:36:08 +03:00
|
|
|
onClickSeverity: () => void;
|
|
|
|
|
onSelect: () => void;
|
2024-04-03 19:55:13 +03:00
|
|
|
};
|
2025-01-14 04:37:29 +03:00
|
|
|
|
2024-04-07 23:36:08 +03:00
|
|
|
function LogLineData({
|
|
|
|
|
className,
|
|
|
|
|
line,
|
2025-01-14 04:37:29 +03:00
|
|
|
logService,
|
2024-04-07 23:36:08 +03:00
|
|
|
onClickSeverity,
|
|
|
|
|
onSelect,
|
|
|
|
|
}: LogLineDataProps) {
|
2024-04-03 19:55:13 +03:00
|
|
|
return (
|
|
|
|
|
<div
|
2024-05-07 17:00:25 +03:00
|
|
|
className={cn(
|
2025-01-15 22:33:25 +03:00
|
|
|
"grid w-full cursor-pointer grid-cols-5 gap-2 border-t border-secondary py-0 hover:bg-muted md:grid-cols-12",
|
2024-05-07 17:00:25 +03:00
|
|
|
className,
|
2025-01-14 04:37:29 +03:00
|
|
|
"*:text-xs",
|
2024-05-07 17:00:25 +03:00
|
|
|
)}
|
2024-04-07 23:36:08 +03:00
|
|
|
onClick={onSelect}
|
2024-04-03 19:55:13 +03:00
|
|
|
>
|
2024-10-31 22:48:26 +03:00
|
|
|
<div className="log-severity flex h-full items-center gap-2 p-1">
|
2024-04-07 23:36:08 +03:00
|
|
|
<LogChip severity={line.severity} onClickSeverity={onClickSeverity} />
|
2024-04-03 19:55:13 +03:00
|
|
|
</div>
|
2025-01-15 22:33:25 +03:00
|
|
|
<div className="log-timestamp col-span-2 flex h-full items-center lg:col-span-1">
|
2024-04-03 19:55:13 +03:00
|
|
|
{line.dateStamp}
|
|
|
|
|
</div>
|
2025-01-14 04:37:29 +03:00
|
|
|
<div
|
|
|
|
|
className={cn(
|
|
|
|
|
"log-section flex size-full items-center pr-2",
|
|
|
|
|
logService == "frigate" ? "col-span-2" : "col-span-1",
|
|
|
|
|
)}
|
|
|
|
|
>
|
2024-05-14 18:06:44 +03:00
|
|
|
<div className="w-full overflow-hidden text-ellipsis whitespace-nowrap">
|
2024-04-07 23:36:08 +03:00
|
|
|
{line.section}
|
|
|
|
|
</div>
|
2024-04-03 19:55:13 +03:00
|
|
|
</div>
|
2025-01-14 04:37:29 +03:00
|
|
|
<div
|
|
|
|
|
className={cn(
|
2025-01-15 22:33:25 +03:00
|
|
|
"log-content col-span-5 flex size-full items-center justify-between pr-2",
|
|
|
|
|
logService == "frigate" ? "md:col-span-7" : "md:col-span-8",
|
2025-01-14 04:37:29 +03:00
|
|
|
)}
|
|
|
|
|
>
|
2024-05-14 18:06:44 +03:00
|
|
|
<div className="w-full overflow-hidden text-ellipsis whitespace-nowrap">
|
2024-04-03 19:55:13 +03:00
|
|
|
{line.content}
|
|
|
|
|
</div>
|
2023-12-31 16:31:33 +03:00
|
|
|
</div>
|
2024-02-21 23:07:32 +03:00
|
|
|
</div>
|
2023-12-08 16:33:22 +03:00
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export default Logs;
|