mirror of
https://github.com/Buriburizaem0n/admin-frontend-domain.git
synced 2026-02-04 12:40:08 +00:00
implement install-commands button (#10)
* fix: type conversion * implement install-commands button
This commit is contained in:
@@ -6,19 +6,40 @@ import {
|
||||
} from "@/components/ui/dropdown-menu"
|
||||
import { Button, ButtonProps } from "@/components/ui/button"
|
||||
import { forwardRef, useState } from "react"
|
||||
import { useConfig } from "@/hooks/useConfig"
|
||||
import { ConfigEssential } from "@/types"
|
||||
import { Check, Clipboard } from "lucide-react"
|
||||
import { t } from "i18next"
|
||||
import { toast } from "sonner"
|
||||
|
||||
import { useTranslation } from "react-i18next"
|
||||
|
||||
enum OSTypes {
|
||||
Linux = 1,
|
||||
macOS,
|
||||
Windows
|
||||
}
|
||||
|
||||
export const InstallCommandsMenu = forwardRef<HTMLButtonElement, ButtonProps>((props, ref) => {
|
||||
const [copy, setCopy] = useState(false);
|
||||
const { config } = useConfig();
|
||||
const { t } = useTranslation();
|
||||
|
||||
const switchState = async () => {
|
||||
const switchState = async (type: number) => {
|
||||
if (!copy) {
|
||||
try {
|
||||
setCopy(true);
|
||||
await navigator.clipboard.writeText("stub");
|
||||
if (config)
|
||||
await navigator.clipboard.writeText(generateCommand(type, config));
|
||||
} catch (e) {
|
||||
console.error(e);
|
||||
toast(t("Error"), {
|
||||
description: t("Results.UnExpectedError"),
|
||||
})
|
||||
} finally {
|
||||
setTimeout(() => {
|
||||
setCopy(false);
|
||||
}, 1000);
|
||||
}, 2 * 1000);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -31,10 +52,30 @@ export const InstallCommandsMenu = forwardRef<HTMLButtonElement, ButtonProps>((p
|
||||
</Button>
|
||||
</DropdownMenuTrigger>
|
||||
<DropdownMenuContent>
|
||||
<DropdownMenuItem onClick={switchState}>Linux</DropdownMenuItem>
|
||||
<DropdownMenuItem onClick={switchState}>macOS</DropdownMenuItem>
|
||||
<DropdownMenuItem onClick={switchState}>Windows</DropdownMenuItem>
|
||||
<DropdownMenuItem onClick={async () => { switchState(OSTypes.Linux) }}>Linux</DropdownMenuItem>
|
||||
<DropdownMenuItem onClick={async () => { switchState(OSTypes.macOS) }}>macOS</DropdownMenuItem>
|
||||
<DropdownMenuItem onClick={async () => { switchState(OSTypes.Windows) }}>Windows</DropdownMenuItem>
|
||||
</DropdownMenuContent>
|
||||
</DropdownMenu>
|
||||
);
|
||||
})
|
||||
|
||||
const generateCommand = (type: number, { agent_secret_key, install_host, listen_port, tls }: ConfigEssential) => {
|
||||
if (!install_host)
|
||||
throw new Error("You have not specify the installed host.");
|
||||
|
||||
const env = `NZ_SERVER=${install_host}:${listen_port} NZ_TLS=${tls || false} NZ_CLIENT_SECRET=${agent_secret_key}`;
|
||||
|
||||
switch (type) {
|
||||
case OSTypes.Linux:
|
||||
case OSTypes.macOS: {
|
||||
return `curl -L https://raw.githubusercontent.com/nezhahq/scripts/main/agent/install.sh -o nezha.sh && chmod +x nezha.sh && env ${env} ./nezha.sh`
|
||||
}
|
||||
case OSTypes.Windows: {
|
||||
return `${env} [Net.ServicePointManager]::SecurityProtocol = [Net.SecurityProtocolType]::Ssl3 -bor [Net.SecurityProtocolType]::Tls -bor [Net.SecurityProtocolType]::Tls11 -bor [Net.SecurityProtocolType]::Tls12;set-ExecutionPolicy RemoteSigned;Invoke-WebRequest https://raw.githubusercontent.com/nezhahq/scripts/main/agent/install.ps1 -OutFile C:\install.ps1;powershell.exe C:\install.ps1`
|
||||
}
|
||||
default: {
|
||||
throw new Error(`Unknown OS: ${type}`);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
48
src/hooks/useConfig.tsx
Normal file
48
src/hooks/useConfig.tsx
Normal file
@@ -0,0 +1,48 @@
|
||||
import { createContext, useContext, useEffect, useMemo } from "react"
|
||||
import { useConfigStore } from "./useConfigStore"
|
||||
import { getSettings } from "@/api/settings"
|
||||
import { ConfigContextProps } from "@/types"
|
||||
import { useLocation } from "react-router-dom"
|
||||
|
||||
const ConfigContext = createContext<ConfigContextProps>({});
|
||||
|
||||
interface ConfigProviderProps {
|
||||
children: React.ReactNode;
|
||||
}
|
||||
|
||||
export const ConfigProvider: React.FC<ConfigProviderProps> = ({ children }) => {
|
||||
const config = useConfigStore(store => store.config);
|
||||
const setConfig = useConfigStore(store => store.setConfig);
|
||||
|
||||
const location = useLocation();
|
||||
|
||||
useEffect(() => {
|
||||
(async () => {
|
||||
if (location.pathname !== "/dashboard/settings")
|
||||
try {
|
||||
const c = await getSettings();
|
||||
const { agent_secret_key, language, listen_port, install_host, site_name, tls } = c;
|
||||
const data = {
|
||||
agent_secret_key,
|
||||
language,
|
||||
listen_port,
|
||||
install_host,
|
||||
site_name,
|
||||
tls,
|
||||
};
|
||||
setConfig(data);
|
||||
} catch (error) {
|
||||
setConfig(undefined);
|
||||
}
|
||||
})();
|
||||
}, [location.pathname])
|
||||
|
||||
const value: ConfigContextProps = useMemo(() => ({
|
||||
config: config,
|
||||
}), [config]);
|
||||
return <ConfigContext.Provider value={value}>{children}</ConfigContext.Provider>;
|
||||
}
|
||||
|
||||
export const useConfig = () => {
|
||||
return useContext(ConfigContext);
|
||||
};
|
||||
16
src/hooks/useConfigStore.ts
Normal file
16
src/hooks/useConfigStore.ts
Normal file
@@ -0,0 +1,16 @@
|
||||
import { ConfigStore } from '@/types'
|
||||
import { create } from 'zustand'
|
||||
import { persist, createJSONStorage } from 'zustand/middleware'
|
||||
|
||||
export const useConfigStore = create<ConfigStore, [['zustand/persist', ConfigStore]]>(
|
||||
persist(
|
||||
(set, get) => ({
|
||||
config: get()?.config,
|
||||
setConfig: config => set({ config }),
|
||||
}),
|
||||
{
|
||||
name: 'configStore',
|
||||
storage: createJSONStorage(() => localStorage),
|
||||
},
|
||||
),
|
||||
)
|
||||
11
src/main.tsx
11
src/main.tsx
@@ -22,6 +22,7 @@ import ServerGroupPage from './routes/server-group';
|
||||
import NotificationGroupPage from './routes/notification-group';
|
||||
import { ServerProvider } from './hooks/useServer';
|
||||
import { NotificationProvider } from './hooks/useNotfication';
|
||||
import { ConfigProvider } from './hooks/useConfig';
|
||||
import CronPage from './routes/cron';
|
||||
import NotificationPage from './routes/notification';
|
||||
import AlertRulePage from './routes/alert-rule';
|
||||
@@ -33,7 +34,15 @@ import ProfilePage from './routes/profile';
|
||||
const router = createBrowserRouter([
|
||||
{
|
||||
path: "/dashboard",
|
||||
element: <AuthProvider><ProtectedRoute><Root /></ProtectedRoute></AuthProvider>,
|
||||
element: (
|
||||
<AuthProvider>
|
||||
<ProtectedRoute>
|
||||
<ConfigProvider>
|
||||
<Root />
|
||||
</ConfigProvider>
|
||||
</ProtectedRoute>
|
||||
</AuthProvider>
|
||||
),
|
||||
errorElement: <ErrorPage />,
|
||||
children: [
|
||||
{
|
||||
|
||||
@@ -74,7 +74,10 @@ export default function SettingsPage() {
|
||||
const form = useForm<z.infer<typeof settingFormSchema>>({
|
||||
resolver: zodResolver(settingFormSchema),
|
||||
defaultValues: config
|
||||
? config
|
||||
? {
|
||||
...config,
|
||||
site_name: config.site_name || "",
|
||||
}
|
||||
: {
|
||||
ip_change_notification_group_id: 0,
|
||||
cover: 1,
|
||||
|
||||
5
src/types/configContext.ts
Normal file
5
src/types/configContext.ts
Normal file
@@ -0,0 +1,5 @@
|
||||
import { ConfigEssential } from "@/types";
|
||||
|
||||
export interface ConfigContextProps {
|
||||
config?: ConfigEssential;
|
||||
}
|
||||
13
src/types/configStore.ts
Normal file
13
src/types/configStore.ts
Normal file
@@ -0,0 +1,13 @@
|
||||
export interface ConfigEssential {
|
||||
language: string;
|
||||
agent_secret_key: string;
|
||||
install_host: string;
|
||||
listen_port: number;
|
||||
site_name: string;
|
||||
tls: boolean;
|
||||
}
|
||||
|
||||
export interface ConfigStore {
|
||||
config?: ConfigEssential;
|
||||
setConfig: (config?: ConfigEssential) => void;
|
||||
}
|
||||
@@ -12,3 +12,5 @@ export * from './notificationStore';
|
||||
export * from './notificationContext';
|
||||
export * from './fm';
|
||||
export * from './settings';
|
||||
export * from './configStore';
|
||||
export * from './configContext';
|
||||
|
||||
Reference in New Issue
Block a user