mirror of
https://github.com/nicotsx/ironmount.git
synced 2025-12-10 12:10:51 +01:00
feat(repositories): allow importing existing repos
This commit is contained in:
@@ -1,6 +1,6 @@
|
|||||||
import { arktypeResolver } from "@hookform/resolvers/arktype";
|
import { arktypeResolver } from "@hookform/resolvers/arktype";
|
||||||
import { type } from "arktype";
|
import { type } from "arktype";
|
||||||
import { useEffect } from "react";
|
import { useEffect, useState } from "react";
|
||||||
import { useForm } from "react-hook-form";
|
import { useForm } from "react-hook-form";
|
||||||
import { cn, slugify } from "~/client/lib/utils";
|
import { cn, slugify } from "~/client/lib/utils";
|
||||||
import { deepClean } from "~/utils/object";
|
import { deepClean } from "~/utils/object";
|
||||||
@@ -15,6 +15,7 @@ import { Tooltip, TooltipContent, TooltipTrigger } from "./ui/tooltip";
|
|||||||
import { useSystemInfo } from "~/client/hooks/use-system-info";
|
import { useSystemInfo } from "~/client/hooks/use-system-info";
|
||||||
import { COMPRESSION_MODES, repositoryConfigSchema } from "~/schemas/restic";
|
import { COMPRESSION_MODES, repositoryConfigSchema } from "~/schemas/restic";
|
||||||
import { listRcloneRemotesOptions } from "../api-client/@tanstack/react-query.gen";
|
import { listRcloneRemotesOptions } from "../api-client/@tanstack/react-query.gen";
|
||||||
|
import { Checkbox } from "./ui/checkbox";
|
||||||
|
|
||||||
export const formSchema = type({
|
export const formSchema = type({
|
||||||
name: "2<=string<=32",
|
name: "2<=string<=32",
|
||||||
@@ -59,9 +60,12 @@ export const CreateRepositoryForm = ({
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
const { watch } = form;
|
const { watch, setValue } = form;
|
||||||
|
|
||||||
const watchedBackend = watch("backend");
|
const watchedBackend = watch("backend");
|
||||||
|
const watchedIsExistingRepository = watch("isExistingRepository");
|
||||||
|
|
||||||
|
const [passwordMode, setPasswordMode] = useState<"default" | "custom">("default");
|
||||||
|
|
||||||
const { data: rcloneRemotes, isLoading: isLoadingRemotes } = useQuery({
|
const { data: rcloneRemotes, isLoading: isLoadingRemotes } = useQuery({
|
||||||
...listRcloneRemotesOptions(),
|
...listRcloneRemotesOptions(),
|
||||||
@@ -70,6 +74,8 @@ export const CreateRepositoryForm = ({
|
|||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
form.reset({
|
form.reset({
|
||||||
name: form.getValues().name,
|
name: form.getValues().name,
|
||||||
|
isExistingRepository: form.getValues().isExistingRepository,
|
||||||
|
customPassword: form.getValues().customPassword,
|
||||||
...defaultValuesForType[watchedBackend as keyof typeof defaultValuesForType],
|
...defaultValuesForType[watchedBackend as keyof typeof defaultValuesForType],
|
||||||
});
|
});
|
||||||
}, [watchedBackend, form]);
|
}, [watchedBackend, form]);
|
||||||
@@ -163,6 +169,81 @@ export const CreateRepositoryForm = ({
|
|||||||
)}
|
)}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
|
<FormField
|
||||||
|
control={form.control}
|
||||||
|
name="isExistingRepository"
|
||||||
|
render={({ field }) => (
|
||||||
|
<FormItem className="flex flex-row items-center space-x-3">
|
||||||
|
<FormControl>
|
||||||
|
<Checkbox
|
||||||
|
checked={field.value}
|
||||||
|
onCheckedChange={(checked) => {
|
||||||
|
field.onChange(checked);
|
||||||
|
if (!checked) {
|
||||||
|
setPasswordMode("default");
|
||||||
|
setValue("customPassword", undefined);
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</FormControl>
|
||||||
|
<div className="space-y-1">
|
||||||
|
<FormLabel>Import existing repository</FormLabel>
|
||||||
|
<FormDescription>Check this if the repository already exists at the specified location</FormDescription>
|
||||||
|
</div>
|
||||||
|
</FormItem>
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
{watchedIsExistingRepository && (
|
||||||
|
<>
|
||||||
|
<FormItem>
|
||||||
|
<FormLabel>Repository Password</FormLabel>
|
||||||
|
<Select
|
||||||
|
onValueChange={(value) => {
|
||||||
|
setPasswordMode(value as "default" | "custom");
|
||||||
|
if (value === "default") {
|
||||||
|
setValue("customPassword", undefined);
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
defaultValue={passwordMode}
|
||||||
|
value={passwordMode}
|
||||||
|
>
|
||||||
|
<FormControl>
|
||||||
|
<SelectTrigger>
|
||||||
|
<SelectValue placeholder="Select password option" />
|
||||||
|
</SelectTrigger>
|
||||||
|
</FormControl>
|
||||||
|
<SelectContent>
|
||||||
|
<SelectItem value="default">Use Ironmount's password</SelectItem>
|
||||||
|
<SelectItem value="custom">Enter password manually</SelectItem>
|
||||||
|
</SelectContent>
|
||||||
|
</Select>
|
||||||
|
<FormDescription>
|
||||||
|
Choose whether to use Ironmount's master password or enter a custom password for the existing
|
||||||
|
repository.
|
||||||
|
</FormDescription>
|
||||||
|
</FormItem>
|
||||||
|
|
||||||
|
{passwordMode === "custom" && (
|
||||||
|
<FormField
|
||||||
|
control={form.control}
|
||||||
|
name="customPassword"
|
||||||
|
render={({ field }) => (
|
||||||
|
<FormItem>
|
||||||
|
<FormLabel>Repository Password</FormLabel>
|
||||||
|
<FormControl>
|
||||||
|
<Input type="password" placeholder="Enter repository password" {...field} />
|
||||||
|
</FormControl>
|
||||||
|
<FormDescription>
|
||||||
|
The password used to encrypt this repository. It will be stored securely.
|
||||||
|
</FormDescription>
|
||||||
|
<FormMessage />
|
||||||
|
</FormItem>
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
|
||||||
{watchedBackend === "s3" && (
|
{watchedBackend === "s3" && (
|
||||||
<>
|
<>
|
||||||
<FormField
|
<FormField
|
||||||
@@ -235,7 +316,9 @@ export const CreateRepositoryForm = ({
|
|||||||
<FormControl>
|
<FormControl>
|
||||||
<Input placeholder="<account-id>.r2.cloudflarestorage.com" {...field} />
|
<Input placeholder="<account-id>.r2.cloudflarestorage.com" {...field} />
|
||||||
</FormControl>
|
</FormControl>
|
||||||
<FormDescription>R2 endpoint (without https://). Find in R2 dashboard under bucket settings.</FormDescription>
|
<FormDescription>
|
||||||
|
R2 endpoint (without https://). Find in R2 dashboard under bucket settings.
|
||||||
|
</FormDescription>
|
||||||
<FormMessage />
|
<FormMessage />
|
||||||
</FormItem>
|
</FormItem>
|
||||||
)}
|
)}
|
||||||
|
|||||||
@@ -11,13 +11,19 @@ export const REPOSITORY_BACKENDS = {
|
|||||||
|
|
||||||
export type RepositoryBackend = keyof typeof REPOSITORY_BACKENDS;
|
export type RepositoryBackend = keyof typeof REPOSITORY_BACKENDS;
|
||||||
|
|
||||||
|
// Common fields for all repository configs
|
||||||
|
const baseRepositoryConfigSchema = type({
|
||||||
|
isExistingRepository: "boolean?",
|
||||||
|
customPassword: "string?",
|
||||||
|
});
|
||||||
|
|
||||||
export const s3RepositoryConfigSchema = type({
|
export const s3RepositoryConfigSchema = type({
|
||||||
backend: "'s3'",
|
backend: "'s3'",
|
||||||
endpoint: "string",
|
endpoint: "string",
|
||||||
bucket: "string",
|
bucket: "string",
|
||||||
accessKeyId: "string",
|
accessKeyId: "string",
|
||||||
secretAccessKey: "string",
|
secretAccessKey: "string",
|
||||||
});
|
}).and(baseRepositoryConfigSchema);
|
||||||
|
|
||||||
export const r2RepositoryConfigSchema = type({
|
export const r2RepositoryConfigSchema = type({
|
||||||
backend: "'r2'",
|
backend: "'r2'",
|
||||||
@@ -25,19 +31,19 @@ export const r2RepositoryConfigSchema = type({
|
|||||||
bucket: "string",
|
bucket: "string",
|
||||||
accessKeyId: "string",
|
accessKeyId: "string",
|
||||||
secretAccessKey: "string",
|
secretAccessKey: "string",
|
||||||
});
|
}).and(baseRepositoryConfigSchema);
|
||||||
|
|
||||||
export const localRepositoryConfigSchema = type({
|
export const localRepositoryConfigSchema = type({
|
||||||
backend: "'local'",
|
backend: "'local'",
|
||||||
name: "string",
|
name: "string",
|
||||||
});
|
}).and(baseRepositoryConfigSchema);
|
||||||
|
|
||||||
export const gcsRepositoryConfigSchema = type({
|
export const gcsRepositoryConfigSchema = type({
|
||||||
backend: "'gcs'",
|
backend: "'gcs'",
|
||||||
bucket: "string",
|
bucket: "string",
|
||||||
projectId: "string",
|
projectId: "string",
|
||||||
credentialsJson: "string",
|
credentialsJson: "string",
|
||||||
});
|
}).and(baseRepositoryConfigSchema);
|
||||||
|
|
||||||
export const azureRepositoryConfigSchema = type({
|
export const azureRepositoryConfigSchema = type({
|
||||||
backend: "'azure'",
|
backend: "'azure'",
|
||||||
@@ -45,13 +51,13 @@ export const azureRepositoryConfigSchema = type({
|
|||||||
accountName: "string",
|
accountName: "string",
|
||||||
accountKey: "string",
|
accountKey: "string",
|
||||||
endpointSuffix: "string?",
|
endpointSuffix: "string?",
|
||||||
});
|
}).and(baseRepositoryConfigSchema);
|
||||||
|
|
||||||
export const rcloneRepositoryConfigSchema = type({
|
export const rcloneRepositoryConfigSchema = type({
|
||||||
backend: "'rclone'",
|
backend: "'rclone'",
|
||||||
remote: "string",
|
remote: "string",
|
||||||
path: "string",
|
path: "string",
|
||||||
});
|
}).and(baseRepositoryConfigSchema);
|
||||||
|
|
||||||
export const repositoryConfigSchema = s3RepositoryConfigSchema
|
export const repositoryConfigSchema = s3RepositoryConfigSchema
|
||||||
.or(r2RepositoryConfigSchema)
|
.or(r2RepositoryConfigSchema)
|
||||||
|
|||||||
@@ -15,7 +15,11 @@ const listRepositories = async () => {
|
|||||||
};
|
};
|
||||||
|
|
||||||
const encryptConfig = async (config: RepositoryConfig): Promise<RepositoryConfig> => {
|
const encryptConfig = async (config: RepositoryConfig): Promise<RepositoryConfig> => {
|
||||||
const encryptedConfig: Record<string, string> = { ...config };
|
const encryptedConfig: Record<string, string | boolean> = { ...config };
|
||||||
|
|
||||||
|
if (config.customPassword) {
|
||||||
|
encryptedConfig.customPassword = await cryptoUtils.encrypt(config.customPassword);
|
||||||
|
}
|
||||||
|
|
||||||
switch (config.backend) {
|
switch (config.backend) {
|
||||||
case "s3":
|
case "s3":
|
||||||
@@ -65,23 +69,30 @@ const createRepository = async (name: string, config: RepositoryConfig, compress
|
|||||||
throw new InternalServerError("Failed to create repository");
|
throw new InternalServerError("Failed to create repository");
|
||||||
}
|
}
|
||||||
|
|
||||||
const { success, error } = await restic.init(encryptedConfig);
|
let error: string | null = null;
|
||||||
|
|
||||||
if (success) {
|
if (config.isExistingRepository) {
|
||||||
|
const result = await restic
|
||||||
|
.snapshots(encryptedConfig)
|
||||||
|
.then(() => ({ error: null }))
|
||||||
|
.catch((error) => ({ error }));
|
||||||
|
|
||||||
|
error = result.error;
|
||||||
|
} else {
|
||||||
|
const initResult = await restic.init(encryptedConfig);
|
||||||
|
error = initResult.error;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!error) {
|
||||||
await db
|
await db
|
||||||
.update(repositoriesTable)
|
.update(repositoriesTable)
|
||||||
.set({
|
.set({ status: "healthy", lastChecked: Date.now(), lastError: null })
|
||||||
status: "healthy",
|
|
||||||
lastChecked: Date.now(),
|
|
||||||
lastError: null,
|
|
||||||
})
|
|
||||||
.where(eq(repositoriesTable.id, id));
|
.where(eq(repositoriesTable.id, id));
|
||||||
|
|
||||||
return { repository: created, status: 201 };
|
return { repository: created, status: 201 };
|
||||||
}
|
}
|
||||||
|
|
||||||
const errorMessage = toMessage(error);
|
const errorMessage = toMessage(error);
|
||||||
|
|
||||||
await db.delete(repositoriesTable).where(eq(repositoriesTable.id, id));
|
await db.delete(repositoriesTable).where(eq(repositoriesTable.id, id));
|
||||||
|
|
||||||
throw new InternalServerError(`Failed to initialize repository: ${errorMessage}`);
|
throw new InternalServerError(`Failed to initialize repository: ${errorMessage}`);
|
||||||
|
|||||||
@@ -75,7 +75,7 @@ const buildRepoUrl = (config: RepositoryConfig): string => {
|
|||||||
case "s3":
|
case "s3":
|
||||||
return `s3:${config.endpoint}/${config.bucket}`;
|
return `s3:${config.endpoint}/${config.bucket}`;
|
||||||
case "r2": {
|
case "r2": {
|
||||||
const endpoint = config.endpoint.replace(/^https?:\/\//, '');
|
const endpoint = config.endpoint.replace(/^https?:\/\//, "");
|
||||||
return `s3:${endpoint}/${config.bucket}`;
|
return `s3:${endpoint}/${config.bucket}`;
|
||||||
}
|
}
|
||||||
case "gcs":
|
case "gcs":
|
||||||
@@ -93,10 +93,19 @@ const buildRepoUrl = (config: RepositoryConfig): string => {
|
|||||||
const buildEnv = async (config: RepositoryConfig) => {
|
const buildEnv = async (config: RepositoryConfig) => {
|
||||||
const env: Record<string, string> = {
|
const env: Record<string, string> = {
|
||||||
RESTIC_CACHE_DIR: "/var/lib/ironmount/restic/cache",
|
RESTIC_CACHE_DIR: "/var/lib/ironmount/restic/cache",
|
||||||
RESTIC_PASSWORD_FILE: RESTIC_PASS_FILE,
|
|
||||||
PATH: process.env.PATH || "/usr/local/bin:/usr/bin:/bin",
|
PATH: process.env.PATH || "/usr/local/bin:/usr/bin:/bin",
|
||||||
};
|
};
|
||||||
|
|
||||||
|
if (config.isExistingRepository && config.customPassword) {
|
||||||
|
const decryptedPassword = await cryptoUtils.decrypt(config.customPassword);
|
||||||
|
const passwordFilePath = path.join("/tmp", `ironmount-pass-${crypto.randomBytes(8).toString("hex")}.txt`);
|
||||||
|
|
||||||
|
await fs.writeFile(passwordFilePath, decryptedPassword, { mode: 0o600 });
|
||||||
|
env.RESTIC_PASSWORD_FILE = passwordFilePath;
|
||||||
|
} else {
|
||||||
|
env.RESTIC_PASSWORD_FILE = RESTIC_PASS_FILE;
|
||||||
|
}
|
||||||
|
|
||||||
switch (config.backend) {
|
switch (config.backend) {
|
||||||
case "s3":
|
case "s3":
|
||||||
env.AWS_ACCESS_KEY_ID = await cryptoUtils.decrypt(config.accessKeyId);
|
env.AWS_ACCESS_KEY_ID = await cryptoUtils.decrypt(config.accessKeyId);
|
||||||
@@ -110,7 +119,7 @@ const buildEnv = async (config: RepositoryConfig) => {
|
|||||||
break;
|
break;
|
||||||
case "gcs": {
|
case "gcs": {
|
||||||
const decryptedCredentials = await cryptoUtils.decrypt(config.credentialsJson);
|
const decryptedCredentials = await cryptoUtils.decrypt(config.credentialsJson);
|
||||||
const credentialsPath = path.join("/tmp", `gcs-credentials-${crypto.randomBytes(8).toString("hex")}.json`);
|
const credentialsPath = path.join("/tmp", `ironmount-gcs-${crypto.randomBytes(8).toString("hex")}.json`);
|
||||||
await fs.writeFile(credentialsPath, decryptedCredentials, { mode: 0o600 });
|
await fs.writeFile(credentialsPath, decryptedCredentials, { mode: 0o600 });
|
||||||
env.GOOGLE_PROJECT_ID = config.projectId;
|
env.GOOGLE_PROJECT_ID = config.projectId;
|
||||||
env.GOOGLE_APPLICATION_CREDENTIALS = credentialsPath;
|
env.GOOGLE_APPLICATION_CREDENTIALS = credentialsPath;
|
||||||
@@ -139,7 +148,7 @@ const init = async (config: RepositoryConfig) => {
|
|||||||
|
|
||||||
if (res.exitCode !== 0) {
|
if (res.exitCode !== 0) {
|
||||||
logger.error(`Restic init failed: ${res.stderr}`);
|
logger.error(`Restic init failed: ${res.stderr}`);
|
||||||
return { success: false, error: res.stderr };
|
return { success: false, error: res.stderr.toString() };
|
||||||
}
|
}
|
||||||
|
|
||||||
logger.info(`Restic repository initialized: ${repoUrl}`);
|
logger.info(`Restic repository initialized: ${repoUrl}`);
|
||||||
|
|||||||
Reference in New Issue
Block a user