vito/resources/js/pages/servers/create-server.tsx
2025-05-10 11:10:11 +03:00

349 lines
14 KiB
TypeScript

import { ClipboardCheckIcon, ClipboardIcon, LoaderCircle, TriangleAlert } from 'lucide-react';
import { Button } from '@/components/ui/button';
import { Sheet, SheetContent, SheetDescription, SheetFooter, SheetHeader, SheetTitle, SheetTrigger } from '@/components/ui/sheet';
import { useForm, usePage } from '@inertiajs/react';
import React, { FormEventHandler, useState } from 'react';
import { Label } from '@/components/ui/label';
import { Select, SelectContent, SelectGroup, SelectItem, SelectTrigger, SelectValue } from '@/components/ui/select';
import InputError from '@/components/input-error';
import { Input } from '@/components/ui/input';
import { Alert, AlertDescription } from '@/components/ui/alert';
import { ServerProvider } from '@/types/server-provider';
import CreateServerProvider from '@/pages/server-providers/create-server-provider';
import axios from 'axios';
import { Form, FormField, FormFields } from '@/components/ui/form';
import type { SharedData } from '@/types';
type CreateServerForm = {
provider: string;
server_provider: number;
name: string;
os: string;
ip: string;
port: number;
region: string;
plan: string;
webserver: string;
database: string;
php: string;
};
export default function CreateServer({ children }: { children: React.ReactNode }) {
const page = usePage<SharedData>();
const form = useForm<Required<CreateServerForm>>({
provider: 'custom',
server_provider: 0,
name: '',
os: '',
ip: '',
port: 22,
region: '',
plan: '',
webserver: '',
database: '',
php: '',
});
const submit: FormEventHandler = (e) => {
e.preventDefault();
form.post(route('servers'));
};
const [copySuccess, setCopySuccess] = useState(false);
const copyToClipboard = () => {
navigator.clipboard.writeText(page.props.publicKeyText).then(() => {
setCopySuccess(true);
setTimeout(() => {
setCopySuccess(false);
}, 2000);
});
};
const [serverProviders, setServerProviders] = useState<ServerProvider[]>([]);
const fetchServerProviders = async () => {
const serverProviders = await axios.get(route('server-providers.all'));
setServerProviders(serverProviders.data);
};
const selectProvider = (provider: string) => {
form.setData('provider', provider);
form.clearErrors();
if (provider !== 'custom') {
form.setData('server_provider', 0);
form.setData('region', '');
form.setData('plan', '');
fetchServerProviders();
}
};
const selectServerProvider = async (serverProvider: string) => {
form.setData('server_provider', parseInt(serverProvider));
await fetchRegions(parseInt(serverProvider));
};
const [regions, setRegions] = useState<{ [key: string]: string }>({});
const fetchRegions = async (serverProvider: number) => {
const regions = await axios.get(route('server-providers.regions', { serverProvider: serverProvider }));
setRegions(regions.data);
};
const selectRegion = async (region: string) => {
form.setData('region', region);
if (region !== '') {
await fetchPlans(form.data.server_provider, region);
}
};
const [plans, setPlans] = useState<{ [key: string]: string }>({});
const fetchPlans = async (serverProvider: number, region: string) => {
const plans = await axios.get(route('server-providers.plans', { serverProvider: serverProvider, region: region }));
setPlans(plans.data);
};
const selectPlan = (plan: string) => {
form.setData('plan', plan);
};
return (
<Sheet>
<SheetTrigger asChild>{children}</SheetTrigger>
<SheetContent className="w-full lg:max-w-4xl">
<SheetHeader>
<SheetTitle>Create new server</SheetTitle> <SheetDescription>Fill in the details to create a new server.</SheetDescription>
</SheetHeader>
<Form id="create-server-form" className="p-4" onSubmit={submit}>
<FormFields>
<FormField>
<Label htmlFor="provider">Provider</Label>
<Select value={form.data.provider} onValueChange={(value) => selectProvider(value)}>
<SelectTrigger id="provider">
<SelectValue placeholder="Select a provider" />
</SelectTrigger>
<SelectContent>
<SelectGroup>
{page.props.configs.server_providers.map((provider) => (
<SelectItem key={provider} value={provider}>
{provider}
</SelectItem>
))}
</SelectGroup>
</SelectContent>
</Select>
<InputError />
</FormField>
{form.data.provider && form.data.provider !== 'custom' && (
<FormField>
<Label htmlFor="server-provider">Server provider connection</Label>
<div className="flex items-center gap-2">
<Select value={form.data.server_provider.toString()} onValueChange={selectServerProvider}>
<SelectTrigger id="provider">
<SelectValue placeholder="Select a provider" />
</SelectTrigger>
<SelectContent>
<SelectGroup>
{serverProviders
.filter((item: ServerProvider) => item.provider === form.data.provider)
.map((provider) => (
<SelectItem key={`server-provider-${provider.id}`} value={provider.id.toString()}>
{provider.name}
</SelectItem>
))}
</SelectGroup>
</SelectContent>
</Select>
<CreateServerProvider
trigger="icon"
providers={page.props.configs.server_providers.filter((item) => item !== 'custom')}
defaultProvider={form.data.provider}
onProviderAdded={fetchServerProviders}
/>
</div>
<InputError />
</FormField>
)}
{form.data.provider && form.data.provider !== 'custom' && (
<div className="grid grid-cols-2 gap-6">
<FormField>
<Label htmlFor="region">Region</Label>
<Select value={form.data.region} onValueChange={selectRegion} disabled={form.data.server_provider === 0}>
<SelectTrigger id="region">
<SelectValue placeholder="Select a region" />
</SelectTrigger>
<SelectContent>
<SelectGroup>
{Object.entries(regions).map(([key, value]) => (
<SelectItem key={`region-${key}`} value={key}>
{value}
</SelectItem>
))}
</SelectGroup>
</SelectContent>
</Select>
<InputError message={form.errors.region} />
</FormField>
<FormField>
<Label htmlFor="plan">Plan</Label>
<Select value={form.data.plan} onValueChange={selectPlan} disabled={form.data.region === ''}>
<SelectTrigger id="plan">
<SelectValue placeholder="Select a plan" />
</SelectTrigger>
<SelectContent>
<SelectGroup>
{Object.entries(plans).map(([key, value]) => (
<SelectItem key={`plan-${key}`} value={key}>
{value}
</SelectItem>
))}
</SelectGroup>
</SelectContent>
</Select>
<InputError message={form.errors.plan} />
</FormField>
</div>
)}
{form.data.provider === 'custom' && (
<>
<Alert>
<TriangleAlert size={5} />
<AlertDescription>
Your server needs to have a new unused installation of supported operating systems and must have a root user. To get started, add
our public key to /root/.ssh/authorized_keys file by running the bellow command on your server as root.
</AlertDescription>
</Alert>
<FormField>
<Label htmlFor="public_key">Public Key</Label>
<Button
onClick={copyToClipboard}
variant="outline"
id="public_key"
type="button"
value={page.props.publicKeyText}
className="justify-between truncate font-normal"
>
<span className="w-full max-w-2/3 overflow-x-hidden overflow-ellipsis">{page.props.publicKeyText}</span>
{copySuccess ? <ClipboardCheckIcon size={5} className="text-success!" /> : <ClipboardIcon size={5} />}
</Button>
</FormField>
</>
)}
<div className="grid grid-cols-2 items-start gap-6">
<FormField>
<Label htmlFor="name">Server Name</Label>
<Input id="name" type="text" autoComplete="name" value={form.data.name} onChange={(e) => form.setData('name', e.target.value)} />
<InputError message={form.errors.name} />
</FormField>
<FormField>
<Label htmlFor="os">Operating System</Label>
<Select value={form.data.os} onValueChange={(value) => form.setData('os', value)}>
<SelectTrigger id="os">
<SelectValue placeholder="Select an operating system" />
</SelectTrigger>
<SelectContent>
<SelectGroup>
{page.props.configs.operating_systems.map((value) => (
<SelectItem key={`os-${value}`} value={value}>
{value}
</SelectItem>
))}
</SelectGroup>
</SelectContent>
</Select>
<InputError message={form.errors.os} />
</FormField>
</div>
{form.data.provider === 'custom' && (
<div className="grid grid-cols-2 items-start gap-6">
<FormField>
<Label htmlFor="ip">SSH IP</Label>
<Input id="ip" type="text" autoComplete="ip" value={form.data.ip} onChange={(e) => form.setData('ip', e.target.value)} />
<InputError message={form.errors.ip} />
</FormField>
<FormField>
<Label htmlFor="port">SSH Port</Label>
<Input
id="port"
type="text"
autoComplete="port"
value={form.data.port}
onChange={(e) => form.setData('port', parseInt(e.target.value))}
/>
<InputError message={form.errors.port} />
</FormField>
</div>
)}
<div className="grid grid-cols-3 items-start gap-6">
<FormField>
<Label htmlFor="webserver">Webserver</Label>
<Select value={form.data.webserver} onValueChange={(value) => form.setData('webserver', value)}>
<SelectTrigger id="webserver">
<SelectValue placeholder="Select webserver" />
</SelectTrigger>
<SelectContent>
<SelectGroup>
{page.props.configs.webservers.map((value) => (
<SelectItem key={`webserver-${value}`} value={value}>
{value}
</SelectItem>
))}
</SelectGroup>
</SelectContent>
</Select>
<InputError message={form.errors.webserver} />
</FormField>
<FormField>
<Label htmlFor="database">Database</Label>
<Select value={form.data.database} onValueChange={(value) => form.setData('database', value)}>
<SelectTrigger id="database">
<SelectValue placeholder="Select database" />
</SelectTrigger>
<SelectContent>
<SelectGroup>
{page.props.configs.databases.map((value) => (
<SelectItem key={`database-${value}`} value={value}>
{value}
</SelectItem>
))}
</SelectGroup>
</SelectContent>
</Select>
<InputError message={form.errors.database} />
</FormField>
<FormField>
<Label htmlFor="php">PHP</Label>
<Select value={form.data.php} onValueChange={(value) => form.setData('php', value)}>
<SelectTrigger id="php">
<SelectValue placeholder="Select PHP version" />
</SelectTrigger>
<SelectContent>
<SelectGroup>
{page.props.configs.php_versions.map((value) => (
<SelectItem key={`php-${value}`} value={value}>
{value}
</SelectItem>
))}
</SelectGroup>
</SelectContent>
</Select>
<InputError message={form.errors.php} />
</FormField>
</div>
</FormFields>
</Form>
<SheetFooter>
<div className="flex items-center">
<Button type="submit" form="create-server-form" tabIndex={4} disabled={form.processing}>
{form.processing && <LoaderCircle className="animate-spin" />} Create
</Button>
</div>
</SheetFooter>
</SheetContent>
</Sheet>
);
}