mirror of
https://github.com/nicotsx/zerobyte.git
synced 2026-04-19 06:18:12 -04:00
Separate include patters and included path cleanly to avoid path with special characters to be expanded. Closes https://github.com/nicotsx/zerobyte/discussions/680 <!-- This is an auto-generated comment: release notes by coderabbit.ai --> ## Summary by CodeRabbit * **New Features** * Added ability to select specific directories and paths for inclusion in backup schedules, separate from pattern-based rules. * **Bug Fixes & Improvements** * Automatically migrates existing backup configurations to work with the new path selection system. * Enhanced backup restoration to properly handle both selected paths and pattern-based inclusions. * **Chores** * Updated database schema to support path selections in backup schedules. <!-- end of auto-generated comment: release notes by coderabbit.ai -->
101 lines
3.3 KiB
TypeScript
101 lines
3.3 KiB
TypeScript
import { useQuery } from "@tanstack/react-query";
|
|
import { ByteSize } from "~/client/components/bytes-size";
|
|
import { Card } from "~/client/components/ui/card";
|
|
import { Progress } from "~/client/components/ui/progress";
|
|
import { getBackupProgressOptions } from "~/client/api-client/@tanstack/react-query.gen";
|
|
import type { GetBackupProgressResponse } from "~/client/api-client/types.gen";
|
|
import { formatDuration } from "~/utils/utils";
|
|
import { formatBytes } from "~/utils/format-bytes";
|
|
|
|
type Props = {
|
|
scheduleShortId: string;
|
|
initialProgress: GetBackupProgressResponse;
|
|
};
|
|
|
|
export const BackupProgressCard = ({ scheduleShortId, initialProgress }: Props) => {
|
|
const { data: progress } = useQuery({
|
|
...getBackupProgressOptions({ path: { shortId: scheduleShortId } }),
|
|
initialData: initialProgress,
|
|
refetchInterval: 1000,
|
|
});
|
|
|
|
const {
|
|
percent_done = 0,
|
|
bytes_done = 0,
|
|
total_bytes = 0,
|
|
seconds_elapsed = 0,
|
|
files_done = 0,
|
|
total_files = 0,
|
|
} = progress ?? {};
|
|
|
|
const percentDone = progress ? Math.round(percent_done * 100) : 0;
|
|
const currentFile = progress?.current_files?.[0] || "";
|
|
const fileName = currentFile.split("/").pop() || currentFile;
|
|
const speed = progress ? formatBytes(bytes_done / seconds_elapsed) : null;
|
|
const eta = progress?.seconds_remaining ? formatDuration(progress.seconds_remaining) : null;
|
|
|
|
return (
|
|
<Card className="p-4">
|
|
<div className="flex items-center justify-between">
|
|
<div className="flex items-center gap-2">
|
|
<div className="h-4 w-4 animate-spin rounded-full border-2 border-primary border-t-transparent" />
|
|
<span className="font-medium">Backup in progress</span>
|
|
</div>
|
|
<span className="text-sm font-medium text-primary">{progress ? `${percentDone}%` : "—"}</span>
|
|
</div>
|
|
|
|
<Progress value={percentDone} className="h-2" />
|
|
|
|
<div className="grid grid-cols-2 gap-4 text-sm">
|
|
<div>
|
|
<p className="text-xs uppercase text-muted-foreground">Files</p>
|
|
<p className="font-medium">
|
|
{progress ? (
|
|
<>
|
|
{files_done.toLocaleString()} / {total_files.toLocaleString()}
|
|
</>
|
|
) : (
|
|
"—"
|
|
)}
|
|
</p>
|
|
</div>
|
|
<div>
|
|
<p className="text-xs uppercase text-muted-foreground">Data</p>
|
|
<p className="font-medium">
|
|
{progress ? (
|
|
<>
|
|
<ByteSize bytes={bytes_done} base={1024} />
|
|
/
|
|
<ByteSize bytes={total_bytes} base={1024} />
|
|
</>
|
|
) : (
|
|
"—"
|
|
)}
|
|
</p>
|
|
</div>
|
|
<div>
|
|
<p className="text-xs uppercase text-muted-foreground">Elapsed</p>
|
|
<p className="font-medium">{progress ? formatDuration(seconds_elapsed) : "—"}</p>
|
|
</div>
|
|
<div>
|
|
<p className="text-xs uppercase text-muted-foreground">Speed</p>
|
|
<p className="font-medium">
|
|
{progress ? (seconds_elapsed > 0 ? `${speed?.text} ${speed?.unit}/s` : "Calculating...") : "—"}
|
|
</p>
|
|
</div>
|
|
<div>
|
|
<p className="text-xs uppercase text-muted-foreground">ETA</p>
|
|
<p className="font-medium">{progress ? (eta ?? "Calculating...") : "—"}</p>
|
|
</div>
|
|
</div>
|
|
|
|
<div className="pt-2 border-t border-border">
|
|
<p className="text-xs uppercase text-muted-foreground mb-1">Current file</p>
|
|
<p className="text-xs font-mono text-muted-foreground truncate" title={currentFile || undefined}>
|
|
{fileName || "—"}
|
|
</p>
|
|
</div>
|
|
</Card>
|
|
);
|
|
};
|