tftsr-devops_investigation/src/pages/Proxmox/BackupPage.tsx

202 lines
7.0 KiB
TypeScript
Raw Normal View History

import React, { useState, useEffect, useCallback } from 'react';
import { Button } from '@/components/ui/index';
import { RefreshCw, Plus } from 'lucide-react';
import { BackupJobList } from '@/components/Proxmox';
import { listProxmoxClusters, listProxmoxBackupJobs } from '@/lib/proxmoxClient';
import type { ClusterInfo } from '@/lib/domain';
import { toast } from 'sonner';
import { Dialog, DialogContent, DialogHeader, DialogTitle, DialogFooter } from '@/components/ui/index';
import { Input } from '@/components/ui/index';
import { Label } from '@/components/ui/index';
export function ProxmoxBackupPage() {
const [clusters, setClusters] = useState<ClusterInfo[]>([]);
const [selectedClusterId, setSelectedClusterId] = useState<string>('');
// eslint-disable-next-line @typescript-eslint/no-explicit-any
const [jobs, setJobs] = useState<any[]>([]);
const [isLoading, setIsLoading] = useState(false);
const [showNewJobDialog, setShowNewJobDialog] = useState(false);
// New job form state
const [jobName, setJobName] = useState('');
const [jobNode, setJobNode] = useState('');
const [jobSchedule, setJobSchedule] = useState('');
const [jobVms, setJobVms] = useState('');
useEffect(() => {
listProxmoxClusters()
.then((cls) => {
setClusters(cls);
if (cls.length > 0) setSelectedClusterId(cls[0].id);
})
.catch((err) => {
console.error('Failed to load clusters:', err);
toast.error('Failed to load clusters');
});
}, []);
const loadJobs = useCallback(async (clusterId: string) => {
if (!clusterId) return;
setIsLoading(true);
try {
fix(proxmox): resolve 7 dashboard and AI chat issues 1. VM Actions: pass clusterId/clusters props from VMsPage to VMList; rename node→node_id in 14 Rust Tauri command handlers to match Tauri 2.x camelCase→snake_case mapping; wire action menu items through handleAction so menu closes on click. 2. Migration: add Target Remote dropdown in MigrationDialog showing available clusters for cross-datacenter migration; targetCluster passed through to migrate_vm invoke. 3. Storage: switch list_proxmox_datastores to cluster/resources?type=storage (single API call, cluster-wide); normalize plugintype→type, disk/maxdisk→used/size, compute available via saturating_sub. 4. Network: replace free-text Interface Type Input with a Select dropdown listing all PVE network interface types. 5. Firewall New Rule: add onNewRule prop to FirewallRuleList, wire button; add full dialog in FirewallPage with action/protocol/ source/dest/port fields that calls add_firewall_rule; rewrite Rust command to accept rule as serde_json::Value instead of flat params (matches frontend invoke signature). 6. Backup: normalize raw PVE cluster/backup fields (id, storage, node, schedule, enabled, next-run timestamp) to BackupJobInfo shape; update BackupJobList columns to show storage, vmid, mode. 7. AI chat: merge all system prompt sections into a single system message (fixes Qwen 3.5 / LiteLLM rejection of multiple system messages); push assistant message with tool_calls before tool result messages to satisfy OpenAI API contract.
2026-06-21 20:08:56 +00:00
const raw = await listProxmoxBackupJobs(clusterId, '');
// eslint-disable-next-line @typescript-eslint/no-explicit-any
const normalized = (raw as any[]).map((job) => {
const enabledRaw = job.enabled ?? job.enable ?? 1;
const isEnabled = enabledRaw === 1 || enabledRaw === true || enabledRaw === '1';
const nextRunRaw = job['next-run'] ?? job.next_run ?? job.nextRun;
const nextRunStr = nextRunRaw
? new Date(Number(nextRunRaw) * 1000).toLocaleString()
: undefined;
return {
id: job.id || String(job.jobid || ''),
name: job.id || job.comment || `job-${job.jobid || '?'}`,
node: job.node || 'all',
schedule: job.schedule || '-',
status: isEnabled ? ('idle' as const) : ('idle' as const),
lastRun: undefined,
nextRun: nextRunStr,
size: undefined,
count: undefined,
enabled: isEnabled,
vmid: job.vmid,
storage: job.storage,
mode: job.mode,
compress: job.compress,
comment: job.comment,
};
});
setJobs(normalized);
} catch (err) {
console.error('Failed to load backup jobs:', err);
toast.error('Failed to load backup jobs');
} finally {
setIsLoading(false);
}
}, []);
useEffect(() => {
if (selectedClusterId) loadJobs(selectedClusterId);
}, [selectedClusterId, loadJobs]);
const handleNewJob = () => {
toast.warning(
'Backup job creation requires additional backend implementation (POST cluster/backup) and is not yet available.',
);
};
const handleSubmitNewJob = async () => {
toast.warning('Backup job creation is not yet available.');
setShowNewJobDialog(false);
};
if (clusters.length === 0 && !isLoading) {
return (
<div className="space-y-4">
<div>
<h1 className="text-2xl font-bold">Backup Jobs</h1>
<p className="text-muted-foreground">Manage Proxmox backup schedules</p>
</div>
<div className="text-center py-12 text-muted-foreground">
<p>No Proxmox clusters configured.</p>
<p className="text-sm mt-1">Add a remote connection first.</p>
</div>
</div>
);
}
return (
<div className="space-y-4">
<div className="flex items-center justify-between">
<div>
<h1 className="text-2xl font-bold">Backup Jobs</h1>
<p className="text-muted-foreground">Manage Proxmox backup schedules</p>
</div>
<div className="flex items-center space-x-2">
{clusters.length > 1 && (
<select
className="rounded-md border px-3 py-1.5 text-sm bg-background"
value={selectedClusterId}
onChange={(e) => setSelectedClusterId(e.target.value)}
>
{clusters.map((c) => (
<option key={c.id} value={c.id}>{c.name}</option>
))}
</select>
)}
<Button variant="outline" size="sm" onClick={() => loadJobs(selectedClusterId)}>
<RefreshCw className="mr-2 h-4 w-4" />
Refresh
</Button>
<Button size="sm" onClick={handleNewJob}>
<Plus className="mr-2 h-4 w-4" />
New Job
</Button>
</div>
</div>
<BackupJobList
jobs={jobs}
onRefresh={() => loadJobs(selectedClusterId)}
/>
<Dialog open={showNewJobDialog} onOpenChange={setShowNewJobDialog}>
<DialogContent>
<DialogHeader>
<DialogTitle>Create New Backup Job</DialogTitle>
</DialogHeader>
<div className="space-y-4 py-4">
<div className="space-y-2">
<Label htmlFor="jobName">Job Name</Label>
<Input
id="jobName"
value={jobName}
onChange={(e) => setJobName(e.target.value)}
placeholder="daily-backup"
/>
</div>
<div className="space-y-2">
<Label htmlFor="jobNode">Node</Label>
<Input
id="jobNode"
value={jobNode}
onChange={(e) => setJobNode(e.target.value)}
placeholder="pve"
/>
</div>
<div className="space-y-2">
<Label htmlFor="jobSchedule">Schedule (cron format)</Label>
<Input
id="jobSchedule"
value={jobSchedule}
onChange={(e) => setJobSchedule(e.target.value)}
placeholder="0 2 * * *"
/>
<p className="text-xs text-muted-foreground">
Example: "0 2 * * *" for daily at 2:00 AM
</p>
</div>
<div className="space-y-2">
<Label htmlFor="jobVms">VMs to Backup (comma-separated IDs)</Label>
<Input
id="jobVms"
value={jobVms}
onChange={(e) => setJobVms(e.target.value)}
placeholder="100, 101, 102"
/>
</div>
</div>
<DialogFooter>
<Button variant="outline" onClick={() => setShowNewJobDialog(false)}>
Cancel
</Button>
<Button onClick={handleSubmitNewJob}>
Create Job
</Button>
</DialogFooter>
</DialogContent>
</Dialog>
</div>
);
}