2023-12-13 00:22:02 +01:00
|
|
|
import useSWR from "swr";
|
|
|
|
import * as monaco from "monaco-editor";
|
|
|
|
import { configureMonacoYaml } from "monaco-yaml";
|
2023-12-16 00:24:50 +01:00
|
|
|
import { useCallback, useEffect, useRef, useState } from "react";
|
2023-12-13 00:22:02 +01:00
|
|
|
import { useApiHost } from "@/api";
|
2023-12-08 14:33:22 +01:00
|
|
|
import Heading from "@/components/ui/heading";
|
2024-03-03 17:32:47 +01:00
|
|
|
import ActivityIndicator from "@/components/indicators/activity-indicator";
|
2023-12-13 00:22:02 +01:00
|
|
|
import { Button } from "@/components/ui/button";
|
|
|
|
import axios from "axios";
|
|
|
|
import copy from "copy-to-clipboard";
|
|
|
|
import { useTheme } from "@/context/theme-provider";
|
2024-02-06 00:54:08 +01:00
|
|
|
import { Toaster } from "@/components/ui/sonner";
|
|
|
|
import { toast } from "sonner";
|
2024-05-21 20:06:17 +02:00
|
|
|
import { LuCopy, LuSave } from "react-icons/lu";
|
|
|
|
import { MdOutlineRestartAlt } from "react-icons/md";
|
2024-11-05 16:33:41 +01:00
|
|
|
import RestartDialog from "@/components/overlay/dialog/RestartDialog";
|
2023-12-13 00:22:02 +01:00
|
|
|
|
|
|
|
type SaveOptions = "saveonly" | "restart";
|
2023-12-08 14:33:22 +01:00
|
|
|
|
|
|
|
function ConfigEditor() {
|
2023-12-13 00:22:02 +01:00
|
|
|
const apiHost = useApiHost();
|
|
|
|
|
2024-04-12 14:31:30 +02:00
|
|
|
useEffect(() => {
|
|
|
|
document.title = "Config Editor - Frigate";
|
|
|
|
}, []);
|
|
|
|
|
2023-12-13 00:22:02 +01:00
|
|
|
const { data: config } = useSWR<string>("config/raw");
|
|
|
|
|
2024-03-21 02:46:45 +01:00
|
|
|
const { theme, systemTheme } = useTheme();
|
2023-12-13 00:22:02 +01:00
|
|
|
const [error, setError] = useState<string | undefined>();
|
|
|
|
|
2023-12-20 15:33:57 +01:00
|
|
|
const editorRef = useRef<monaco.editor.IStandaloneCodeEditor | null>(null);
|
2024-05-27 17:31:58 +02:00
|
|
|
const modelRef = useRef<monaco.editor.ITextModel | null>(null);
|
2023-12-20 15:33:57 +01:00
|
|
|
const configRef = useRef<HTMLDivElement | null>(null);
|
2024-09-20 01:36:07 +02:00
|
|
|
const schemaConfiguredRef = useRef(false);
|
2023-12-13 00:22:02 +01:00
|
|
|
|
2024-11-05 16:33:41 +01:00
|
|
|
const [restartDialogOpen, setRestartDialogOpen] = useState(false);
|
|
|
|
|
2023-12-13 00:22:02 +01:00
|
|
|
const onHandleSaveConfig = useCallback(
|
|
|
|
async (save_option: SaveOptions) => {
|
|
|
|
if (!editorRef.current) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
axios
|
|
|
|
.post(
|
|
|
|
`config/save?save_option=${save_option}`,
|
|
|
|
editorRef.current.getValue(),
|
|
|
|
{
|
|
|
|
headers: { "Content-Type": "text/plain" },
|
2024-02-28 23:23:56 +01:00
|
|
|
},
|
2023-12-13 00:22:02 +01:00
|
|
|
)
|
|
|
|
.then((response) => {
|
|
|
|
if (response.status === 200) {
|
|
|
|
setError("");
|
2024-02-06 00:54:08 +01:00
|
|
|
toast.success(response.data.message, { position: "top-center" });
|
2023-12-13 00:22:02 +01:00
|
|
|
}
|
|
|
|
})
|
|
|
|
.catch((error) => {
|
2024-02-06 00:54:08 +01:00
|
|
|
toast.error("Error saving config", { position: "top-center" });
|
2023-12-13 00:22:02 +01:00
|
|
|
|
|
|
|
if (error.response) {
|
|
|
|
setError(error.response.data.message);
|
|
|
|
} else {
|
|
|
|
setError(error.message);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
},
|
2024-02-28 23:23:56 +01:00
|
|
|
[editorRef],
|
2023-12-13 00:22:02 +01:00
|
|
|
);
|
|
|
|
|
|
|
|
const handleCopyConfig = useCallback(async () => {
|
|
|
|
if (!editorRef.current) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
copy(editorRef.current.getValue());
|
2024-05-21 20:06:17 +02:00
|
|
|
toast.success("Config copied to clipboard.", { position: "top-center" });
|
2023-12-13 00:22:02 +01:00
|
|
|
}, [editorRef]);
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
if (!config) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2024-09-20 01:36:07 +02:00
|
|
|
const modelUri = monaco.Uri.parse(
|
|
|
|
`a://b/api/config/schema_${Date.now()}.json`,
|
|
|
|
);
|
|
|
|
|
|
|
|
// Configure Monaco YAML schema only once
|
|
|
|
if (!schemaConfiguredRef.current) {
|
|
|
|
configureMonacoYaml(monaco, {
|
|
|
|
enableSchemaRequest: true,
|
|
|
|
hover: true,
|
|
|
|
completion: true,
|
|
|
|
validate: true,
|
|
|
|
format: true,
|
|
|
|
schemas: [
|
|
|
|
{
|
|
|
|
uri: `${apiHost}api/config/schema.json`,
|
|
|
|
fileMatch: [String(modelUri)],
|
|
|
|
},
|
|
|
|
],
|
|
|
|
});
|
|
|
|
schemaConfiguredRef.current = true;
|
2023-12-13 00:22:02 +01:00
|
|
|
}
|
|
|
|
|
2024-09-20 01:36:07 +02:00
|
|
|
if (!modelRef.current) {
|
2023-12-13 00:22:02 +01:00
|
|
|
modelRef.current = monaco.editor.createModel(config, "yaml", modelUri);
|
2024-09-20 01:36:07 +02:00
|
|
|
} else {
|
|
|
|
modelRef.current.setValue(config);
|
2023-12-13 00:22:02 +01:00
|
|
|
}
|
|
|
|
|
2023-12-20 15:33:57 +01:00
|
|
|
const container = configRef.current;
|
2023-12-13 00:22:02 +01:00
|
|
|
|
2024-09-20 01:36:07 +02:00
|
|
|
if (container && !editorRef.current) {
|
2023-12-13 00:22:02 +01:00
|
|
|
editorRef.current = monaco.editor.create(container, {
|
|
|
|
language: "yaml",
|
|
|
|
model: modelRef.current,
|
|
|
|
scrollBeyondLastLine: false,
|
2024-03-21 02:46:45 +01:00
|
|
|
theme: (systemTheme || theme) == "dark" ? "vs-dark" : "vs-light",
|
2023-12-13 00:22:02 +01:00
|
|
|
});
|
2024-09-20 01:36:07 +02:00
|
|
|
} else if (editorRef.current) {
|
|
|
|
editorRef.current.setModel(modelRef.current);
|
2023-12-13 00:22:02 +01:00
|
|
|
}
|
2023-12-20 15:33:57 +01:00
|
|
|
|
|
|
|
return () => {
|
2024-09-20 01:36:07 +02:00
|
|
|
if (editorRef.current) {
|
|
|
|
editorRef.current.dispose();
|
|
|
|
editorRef.current = null;
|
|
|
|
}
|
|
|
|
if (modelRef.current) {
|
|
|
|
modelRef.current.dispose();
|
|
|
|
modelRef.current = null;
|
|
|
|
}
|
|
|
|
schemaConfiguredRef.current = false;
|
2023-12-20 15:33:57 +01:00
|
|
|
};
|
2024-09-20 01:36:07 +02:00
|
|
|
}, [config, apiHost, systemTheme, theme]);
|
2023-12-13 00:22:02 +01:00
|
|
|
|
2024-08-21 16:19:07 +02:00
|
|
|
// monitoring state
|
|
|
|
|
|
|
|
const [hasChanges, setHasChanges] = useState(false);
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
if (!config || !modelRef.current) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
modelRef.current.onDidChangeContent(() => {
|
|
|
|
if (modelRef.current?.getValue() != config) {
|
|
|
|
setHasChanges(true);
|
|
|
|
} else {
|
|
|
|
setHasChanges(false);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}, [config]);
|
|
|
|
|
2024-05-27 17:31:58 +02:00
|
|
|
useEffect(() => {
|
|
|
|
if (config && modelRef.current) {
|
|
|
|
modelRef.current.setValue(config);
|
2024-08-21 16:19:07 +02:00
|
|
|
setHasChanges(false);
|
2024-05-27 17:31:58 +02:00
|
|
|
}
|
|
|
|
}, [config]);
|
|
|
|
|
2024-08-21 16:19:07 +02:00
|
|
|
useEffect(() => {
|
|
|
|
let listener: ((e: BeforeUnloadEvent) => void) | undefined;
|
|
|
|
if (hasChanges) {
|
|
|
|
listener = (e) => {
|
|
|
|
e.preventDefault();
|
|
|
|
e.returnValue = true;
|
|
|
|
return "Exit without saving?";
|
|
|
|
};
|
|
|
|
window.addEventListener("beforeunload", listener);
|
|
|
|
}
|
|
|
|
|
|
|
|
return () => {
|
|
|
|
if (listener) {
|
|
|
|
window.removeEventListener("beforeunload", listener);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}, [hasChanges]);
|
|
|
|
|
2023-12-13 00:22:02 +01:00
|
|
|
if (!config) {
|
|
|
|
return <ActivityIndicator />;
|
|
|
|
}
|
|
|
|
|
2023-12-08 14:33:22 +01:00
|
|
|
return (
|
2024-05-21 20:06:17 +02:00
|
|
|
<div className="absolute bottom-2 left-0 right-0 top-2 md:left-2">
|
|
|
|
<div className="relative h-full overflow-hidden">
|
|
|
|
<div className="mr-1 flex items-center justify-between">
|
|
|
|
<Heading as="h2" className="mb-0 ml-1 md:ml-0">
|
|
|
|
Config Editor
|
|
|
|
</Heading>
|
|
|
|
<div className="flex flex-row gap-1">
|
|
|
|
<Button
|
|
|
|
size="sm"
|
|
|
|
className="flex items-center gap-2"
|
2024-10-23 00:07:42 +02:00
|
|
|
aria-label="Copy config"
|
2024-05-21 20:06:17 +02:00
|
|
|
onClick={() => handleCopyConfig()}
|
|
|
|
>
|
|
|
|
<LuCopy className="text-secondary-foreground" />
|
|
|
|
<span className="hidden md:block">Copy Config</span>
|
|
|
|
</Button>
|
|
|
|
<Button
|
|
|
|
size="sm"
|
|
|
|
className="flex items-center gap-2"
|
2024-10-23 00:07:42 +02:00
|
|
|
aria-label="Save and restart"
|
2024-11-05 16:33:41 +01:00
|
|
|
onClick={() => setRestartDialogOpen(true)}
|
2024-05-21 20:06:17 +02:00
|
|
|
>
|
|
|
|
<div className="relative size-5">
|
|
|
|
<LuSave className="absolute left-0 top-0 size-3 text-secondary-foreground" />
|
|
|
|
<MdOutlineRestartAlt className="absolute size-4 translate-x-1 translate-y-1/2 text-secondary-foreground" />
|
|
|
|
</div>
|
|
|
|
<span className="hidden md:block">Save & Restart</span>
|
|
|
|
</Button>
|
|
|
|
<Button
|
|
|
|
size="sm"
|
|
|
|
className="flex items-center gap-2"
|
2024-10-23 00:07:42 +02:00
|
|
|
aria-label="Save only without restarting"
|
2024-05-21 20:06:17 +02:00
|
|
|
onClick={() => onHandleSaveConfig("saveonly")}
|
|
|
|
>
|
|
|
|
<LuSave className="text-secondary-foreground" />
|
|
|
|
<span className="hidden md:block">Save Only</span>
|
|
|
|
</Button>
|
|
|
|
</div>
|
2023-12-13 00:22:02 +01:00
|
|
|
</div>
|
|
|
|
|
2024-05-21 20:06:17 +02:00
|
|
|
{error && (
|
2024-10-10 15:09:03 +02:00
|
|
|
<div className="mt-2 max-h-[30%] overflow-auto whitespace-pre-wrap border-2 border-muted bg-background_alt p-4 text-sm text-danger md:max-h-[40%]">
|
2024-05-21 20:06:17 +02:00
|
|
|
{error}
|
|
|
|
</div>
|
|
|
|
)}
|
2023-12-13 00:22:02 +01:00
|
|
|
|
2024-05-21 20:06:17 +02:00
|
|
|
<div ref={configRef} className="mt-2 h-[calc(100%-2.75rem)]" />
|
|
|
|
</div>
|
2024-05-04 21:54:50 +02:00
|
|
|
<Toaster closeButton={true} />
|
2024-11-05 16:33:41 +01:00
|
|
|
<RestartDialog
|
|
|
|
isOpen={restartDialogOpen}
|
|
|
|
onClose={() => setRestartDialogOpen(false)}
|
|
|
|
onRestart={() => onHandleSaveConfig("restart")}
|
|
|
|
/>
|
2023-12-13 00:22:02 +01:00
|
|
|
</div>
|
2023-12-08 14:33:22 +01:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
export default ConfigEditor;
|