mirror of
https://github.com/nicotsx/ironmount.git
synced 2025-12-10 12:10:51 +01:00
411 lines
13 KiB
TypeScript
411 lines
13 KiB
TypeScript
import { arktypeResolver } from "@hookform/resolvers/arktype";
|
|
import { COMPRESSION_MODES, repositoryConfigSchema } from "@ironmount/schemas/restic";
|
|
import { type } from "arktype";
|
|
import { useEffect } from "react";
|
|
import { useForm } from "react-hook-form";
|
|
import { cn, slugify } from "~/lib/utils";
|
|
import { deepClean } from "~/utils/object";
|
|
import { Button } from "./ui/button";
|
|
import { Form, FormControl, FormDescription, FormField, FormItem, FormLabel, FormMessage } from "./ui/form";
|
|
import { Input } from "./ui/input";
|
|
import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from "./ui/select";
|
|
import { listRcloneRemotesOptions } from "~/api-client/@tanstack/react-query.gen";
|
|
import { useQuery } from "@tanstack/react-query";
|
|
import { Alert, AlertDescription } from "./ui/alert";
|
|
import { ExternalLink } from "lucide-react";
|
|
import { Tooltip, TooltipContent, TooltipTrigger } from "./ui/tooltip";
|
|
import { useSystemInfo } from "~/hooks/use-system-info";
|
|
|
|
export const formSchema = type({
|
|
name: "2<=string<=32",
|
|
compressionMode: type.valueOf(COMPRESSION_MODES).optional(),
|
|
}).and(repositoryConfigSchema);
|
|
const cleanSchema = type.pipe((d) => formSchema(deepClean(d)));
|
|
|
|
export type RepositoryFormValues = typeof formSchema.inferIn;
|
|
|
|
type Props = {
|
|
onSubmit: (values: RepositoryFormValues) => void;
|
|
mode?: "create" | "update";
|
|
initialValues?: Partial<RepositoryFormValues>;
|
|
formId?: string;
|
|
loading?: boolean;
|
|
className?: string;
|
|
};
|
|
|
|
const defaultValuesForType = {
|
|
local: { backend: "local" as const, compressionMode: "auto" as const },
|
|
s3: { backend: "s3" as const, compressionMode: "auto" as const },
|
|
gcs: { backend: "gcs" as const, compressionMode: "auto" as const },
|
|
azure: { backend: "azure" as const, compressionMode: "auto" as const },
|
|
rclone: { backend: "rclone" as const, compressionMode: "auto" as const },
|
|
};
|
|
|
|
export const CreateRepositoryForm = ({
|
|
onSubmit,
|
|
mode = "create",
|
|
initialValues,
|
|
formId,
|
|
loading,
|
|
className,
|
|
}: Props) => {
|
|
const form = useForm<RepositoryFormValues>({
|
|
resolver: arktypeResolver(cleanSchema as unknown as typeof formSchema),
|
|
defaultValues: initialValues,
|
|
resetOptions: {
|
|
keepDefaultValues: true,
|
|
keepDirtyValues: false,
|
|
},
|
|
});
|
|
|
|
const { watch } = form;
|
|
|
|
const watchedBackend = watch("backend");
|
|
const watchedName = watch("name");
|
|
|
|
const { data: rcloneRemotes, isLoading: isLoadingRemotes } = useQuery({
|
|
...listRcloneRemotesOptions(),
|
|
});
|
|
|
|
useEffect(() => {
|
|
if (watchedBackend && watchedBackend in defaultValuesForType) {
|
|
form.reset({ name: watchedName, ...defaultValuesForType[watchedBackend as keyof typeof defaultValuesForType] });
|
|
}
|
|
}, [watchedBackend, watchedName, form]);
|
|
|
|
const { capabilities } = useSystemInfo();
|
|
|
|
return (
|
|
<Form {...form}>
|
|
<form id={formId} onSubmit={form.handleSubmit(onSubmit)} className={cn("space-y-4", className)}>
|
|
<FormField
|
|
control={form.control}
|
|
name="name"
|
|
render={({ field }) => (
|
|
<FormItem>
|
|
<FormLabel>Name</FormLabel>
|
|
<FormControl>
|
|
<Input
|
|
{...field}
|
|
placeholder="Repository name"
|
|
onChange={(e) => field.onChange(slugify(e.target.value))}
|
|
max={32}
|
|
min={2}
|
|
disabled={mode === "update"}
|
|
className={mode === "update" ? "bg-gray-50" : ""}
|
|
/>
|
|
</FormControl>
|
|
<FormDescription>Unique identifier for the repository.</FormDescription>
|
|
<FormMessage />
|
|
</FormItem>
|
|
)}
|
|
/>
|
|
<FormField
|
|
control={form.control}
|
|
name="backend"
|
|
render={({ field }) => (
|
|
<FormItem>
|
|
<FormLabel>Backend</FormLabel>
|
|
<Select onValueChange={field.onChange} defaultValue={field.value} value={field.value}>
|
|
<FormControl>
|
|
<SelectTrigger>
|
|
<SelectValue placeholder="Select a backend" />
|
|
</SelectTrigger>
|
|
</FormControl>
|
|
<SelectContent>
|
|
<SelectItem value="local">Local</SelectItem>
|
|
<SelectItem value="s3">S3</SelectItem>
|
|
<SelectItem value="gcs">Google Cloud Storage</SelectItem>
|
|
<SelectItem value="azure">Azure Blob Storage</SelectItem>
|
|
<Tooltip>
|
|
<TooltipTrigger>
|
|
<SelectItem disabled={!capabilities.rclone} value="rclone">
|
|
rclone (40+ cloud providers)
|
|
</SelectItem>
|
|
</TooltipTrigger>
|
|
<TooltipContent className={cn({ hidden: capabilities.rclone })}>
|
|
<p>Setup rclone to use this backend</p>
|
|
</TooltipContent>
|
|
</Tooltip>
|
|
</SelectContent>
|
|
</Select>
|
|
<FormDescription>Choose the storage backend for this repository.</FormDescription>
|
|
<FormMessage />
|
|
</FormItem>
|
|
)}
|
|
/>
|
|
|
|
<FormField
|
|
control={form.control}
|
|
name="compressionMode"
|
|
render={({ field }) => (
|
|
<FormItem>
|
|
<FormLabel>Compression Mode</FormLabel>
|
|
<Select onValueChange={field.onChange} defaultValue={field.value} value={field.value}>
|
|
<FormControl>
|
|
<SelectTrigger>
|
|
<SelectValue placeholder="Select compression mode" />
|
|
</SelectTrigger>
|
|
</FormControl>
|
|
<SelectContent>
|
|
<SelectItem value="off">Off</SelectItem>
|
|
<SelectItem value="auto">Auto</SelectItem>
|
|
<SelectItem value="fastest">Fastest</SelectItem>
|
|
<SelectItem value="better">Better</SelectItem>
|
|
<SelectItem value="max">Max</SelectItem>
|
|
</SelectContent>
|
|
</Select>
|
|
<FormDescription>Compression mode for backups stored in this repository.</FormDescription>
|
|
<FormMessage />
|
|
</FormItem>
|
|
)}
|
|
/>
|
|
|
|
{watchedBackend === "s3" && (
|
|
<>
|
|
<FormField
|
|
control={form.control}
|
|
name="endpoint"
|
|
render={({ field }) => (
|
|
<FormItem>
|
|
<FormLabel>Endpoint</FormLabel>
|
|
<FormControl>
|
|
<Input placeholder="s3.amazonaws.com" {...field} />
|
|
</FormControl>
|
|
<FormDescription>S3-compatible endpoint URL.</FormDescription>
|
|
<FormMessage />
|
|
</FormItem>
|
|
)}
|
|
/>
|
|
<FormField
|
|
control={form.control}
|
|
name="bucket"
|
|
render={({ field }) => (
|
|
<FormItem>
|
|
<FormLabel>Bucket</FormLabel>
|
|
<FormControl>
|
|
<Input placeholder="my-backup-bucket" {...field} />
|
|
</FormControl>
|
|
<FormDescription>S3 bucket name for storing backups.</FormDescription>
|
|
<FormMessage />
|
|
</FormItem>
|
|
)}
|
|
/>
|
|
<FormField
|
|
control={form.control}
|
|
name="accessKeyId"
|
|
render={({ field }) => (
|
|
<FormItem>
|
|
<FormLabel>Access Key ID</FormLabel>
|
|
<FormControl>
|
|
<Input placeholder="AKIAIOSFODNN7EXAMPLE" {...field} />
|
|
</FormControl>
|
|
<FormDescription>S3 access key ID for authentication.</FormDescription>
|
|
<FormMessage />
|
|
</FormItem>
|
|
)}
|
|
/>
|
|
<FormField
|
|
control={form.control}
|
|
name="secretAccessKey"
|
|
render={({ field }) => (
|
|
<FormItem>
|
|
<FormLabel>Secret Access Key</FormLabel>
|
|
<FormControl>
|
|
<Input type="password" placeholder="••••••••" {...field} />
|
|
</FormControl>
|
|
<FormDescription>S3 secret access key for authentication.</FormDescription>
|
|
<FormMessage />
|
|
</FormItem>
|
|
)}
|
|
/>
|
|
</>
|
|
)}
|
|
|
|
{watchedBackend === "gcs" && (
|
|
<>
|
|
<FormField
|
|
control={form.control}
|
|
name="bucket"
|
|
render={({ field }) => (
|
|
<FormItem>
|
|
<FormLabel>Bucket</FormLabel>
|
|
<FormControl>
|
|
<Input placeholder="my-backup-bucket" {...field} />
|
|
</FormControl>
|
|
<FormDescription>GCS bucket name for storing backups.</FormDescription>
|
|
<FormMessage />
|
|
</FormItem>
|
|
)}
|
|
/>
|
|
<FormField
|
|
control={form.control}
|
|
name="projectId"
|
|
render={({ field }) => (
|
|
<FormItem>
|
|
<FormLabel>Project ID</FormLabel>
|
|
<FormControl>
|
|
<Input placeholder="my-gcp-project-123" {...field} />
|
|
</FormControl>
|
|
<FormDescription>Google Cloud project ID.</FormDescription>
|
|
<FormMessage />
|
|
</FormItem>
|
|
)}
|
|
/>
|
|
<FormField
|
|
control={form.control}
|
|
name="credentialsJson"
|
|
render={({ field }) => (
|
|
<FormItem>
|
|
<FormLabel>Service Account JSON</FormLabel>
|
|
<FormControl>
|
|
<Input type="password" placeholder="Paste service account JSON key..." {...field} />
|
|
</FormControl>
|
|
<FormDescription>Service account JSON credentials for authentication.</FormDescription>
|
|
<FormMessage />
|
|
</FormItem>
|
|
)}
|
|
/>
|
|
</>
|
|
)}
|
|
|
|
{watchedBackend === "azure" && (
|
|
<>
|
|
<FormField
|
|
control={form.control}
|
|
name="container"
|
|
render={({ field }) => (
|
|
<FormItem>
|
|
<FormLabel>Container</FormLabel>
|
|
<FormControl>
|
|
<Input placeholder="my-backup-container" {...field} />
|
|
</FormControl>
|
|
<FormDescription>Azure Blob Storage container name for storing backups.</FormDescription>
|
|
<FormMessage />
|
|
</FormItem>
|
|
)}
|
|
/>
|
|
<FormField
|
|
control={form.control}
|
|
name="accountName"
|
|
render={({ field }) => (
|
|
<FormItem>
|
|
<FormLabel>Account Name</FormLabel>
|
|
<FormControl>
|
|
<Input placeholder="mystorageaccount" {...field} />
|
|
</FormControl>
|
|
<FormDescription>Azure Storage account name.</FormDescription>
|
|
<FormMessage />
|
|
</FormItem>
|
|
)}
|
|
/>
|
|
<FormField
|
|
control={form.control}
|
|
name="accountKey"
|
|
render={({ field }) => (
|
|
<FormItem>
|
|
<FormLabel>Account Key</FormLabel>
|
|
<FormControl>
|
|
<Input type="password" placeholder="••••••••" {...field} />
|
|
</FormControl>
|
|
<FormDescription>Azure Storage account key for authentication.</FormDescription>
|
|
<FormMessage />
|
|
</FormItem>
|
|
)}
|
|
/>
|
|
<FormField
|
|
control={form.control}
|
|
name="endpointSuffix"
|
|
render={({ field }) => (
|
|
<FormItem>
|
|
<FormLabel>Endpoint Suffix (Optional)</FormLabel>
|
|
<FormControl>
|
|
<Input placeholder="core.windows.net" {...field} />
|
|
</FormControl>
|
|
<FormDescription>Custom Azure endpoint suffix (defaults to core.windows.net).</FormDescription>
|
|
<FormMessage />
|
|
</FormItem>
|
|
)}
|
|
/>
|
|
</>
|
|
)}
|
|
|
|
{watchedBackend === "rclone" &&
|
|
(!rcloneRemotes || rcloneRemotes.length === 0 ? (
|
|
<Alert>
|
|
<AlertDescription className="space-y-2">
|
|
<p className="font-medium">No rclone remotes configured</p>
|
|
<p className="text-sm text-muted-foreground">
|
|
To use rclone, you need to configure remotes on your host system
|
|
</p>
|
|
<a
|
|
href="https://rclone.org/docs/"
|
|
target="_blank"
|
|
rel="noopener noreferrer"
|
|
className="text-sm text-strong-accent inline-flex items-center gap-1"
|
|
>
|
|
View rclone documentation
|
|
<ExternalLink className="w-3 h-3" />
|
|
</a>
|
|
</AlertDescription>
|
|
</Alert>
|
|
) : (
|
|
<>
|
|
<FormField
|
|
control={form.control}
|
|
name="remote"
|
|
render={({ field }) => (
|
|
<FormItem>
|
|
<FormLabel>Remote</FormLabel>
|
|
<Select onValueChange={(v) => field.onChange(v)} defaultValue={field.value} value={field.value}>
|
|
<FormControl>
|
|
<SelectTrigger>
|
|
<SelectValue placeholder="Select an rclone remote" />
|
|
</SelectTrigger>
|
|
</FormControl>
|
|
<SelectContent>
|
|
{isLoadingRemotes ? (
|
|
<SelectItem value="loading" disabled>
|
|
Loading remotes...
|
|
</SelectItem>
|
|
) : (
|
|
rcloneRemotes.map((remote: { name: string; type: string }) => (
|
|
<SelectItem key={remote.name} value={remote.name}>
|
|
{remote.name} ({remote.type})
|
|
</SelectItem>
|
|
))
|
|
)}
|
|
</SelectContent>
|
|
</Select>
|
|
<FormDescription>Select the rclone remote configured on your host system.</FormDescription>
|
|
<FormMessage />
|
|
</FormItem>
|
|
)}
|
|
/>
|
|
<FormField
|
|
control={form.control}
|
|
name="path"
|
|
render={({ field }) => (
|
|
<FormItem>
|
|
<FormLabel>Path</FormLabel>
|
|
<FormControl>
|
|
<Input placeholder="backups/ironmount" {...field} />
|
|
</FormControl>
|
|
<FormDescription>Path within the remote where backups will be stored.</FormDescription>
|
|
<FormMessage />
|
|
</FormItem>
|
|
)}
|
|
/>
|
|
</>
|
|
))}
|
|
|
|
{mode === "update" && (
|
|
<Button type="submit" className="w-full" loading={loading}>
|
|
Save Changes
|
|
</Button>
|
|
)}
|
|
</form>
|
|
</Form>
|
|
);
|
|
};
|