meatloaf-config/src/app/components/ToolsPage.tsx

363 lines
14 KiB
TypeScript

import { useState, useEffect } from 'react';
import { Wrench, Download, Upload, RotateCcw, Database, FileText, HardDrive, X, Loader2, Cpu, ChevronRight, AlertCircle } from 'lucide-react';
import { toast } from 'sonner';
import { getFileContents, putFileContents, humanFileSize } from '../webdav';
import { ConfirmDialog, type ConfirmOptions } from './ui/confirm-dialog';
interface ToolsPageProps {
config: any;
setConfig: (config: any) => void;
}
interface FirmwareEntry {
version: string;
date: string;
description: string;
url: string;
size?: number;
}
function FirmwareOverlay({ onClose }: { onClose: () => void }) {
const [entries, setEntries] = useState<FirmwareEntry[] | null>(null);
const [loadError, setLoadError] = useState<string | null>(null);
const [installing, setInstalling] = useState<string | null>(null);
useEffect(() => {
getFileContents('/.sys/firmware.json')
.then(async (blob) => {
const parsed = JSON.parse(await blob.text());
if (Array.isArray(parsed)) {
setEntries(parsed);
} else {
setLoadError('Unexpected firmware manifest format.');
}
})
.catch((e: any) => setLoadError(e?.message || 'Failed to load firmware list.'));
}, []);
const install = async (entry: FirmwareEntry) => {
setInstalling(entry.url);
try {
let data: ArrayBuffer;
if (/^https?:\/\//.test(entry.url)) {
const r = await fetch(entry.url);
if (!r.ok) throw new Error(`Download failed: ${r.status} ${r.statusText}`);
data = await r.arrayBuffer();
} else {
const blob = await getFileContents(entry.url);
data = await blob.arrayBuffer();
}
await putFileContents('/sd/.bin', data);
toast.success(`Firmware ${entry.version} saved to /sd/.bin`);
onClose();
} catch (e: any) {
toast.error(`Install failed: ${e?.message || String(e)}`);
} finally {
setInstalling(null);
}
};
return (
<div className="fixed inset-0 bg-white z-50 flex flex-col">
<div className="flex items-center justify-between p-4 border-b border-neutral-200">
<div>
<h2 className="font-semibold text-lg">Change Firmware</h2>
<p className="text-xs text-neutral-500 mt-0.5">Selected firmware will be saved to /sd/.bin</p>
</div>
<button onClick={onClose} className="p-2 -m-2 hover:bg-neutral-100 rounded-lg">
<X className="w-6 h-6" />
</button>
</div>
<div className="flex-1 overflow-y-auto p-4 space-y-3">
{entries === null && !loadError && (
<div className="flex flex-col items-center justify-center gap-3 py-16 text-neutral-500">
<Loader2 className="w-6 h-6 animate-spin" />
<span className="text-sm">Loading firmware list</span>
</div>
)}
{loadError && (
<div className="flex items-start gap-3 p-4 bg-red-50 border border-red-200 rounded-lg text-sm text-red-700">
<AlertCircle className="w-5 h-5 flex-shrink-0 mt-0.5" />
<span>{loadError}</span>
</div>
)}
{entries && entries.length === 0 && (
<div className="text-center py-16 text-sm text-neutral-500">
No firmware updates available.
</div>
)}
{entries && entries.map((entry) => {
const isInstalling = installing === entry.url;
const busy = installing !== null;
return (
<div key={entry.url} className="border border-neutral-200 rounded-lg p-4">
<div className="flex items-start gap-3">
<div className="flex-1 min-w-0">
<div className="flex flex-wrap items-center gap-2 mb-1">
<span className="font-semibold text-neutral-900">{entry.version}</span>
<span className="text-xs text-neutral-400">{entry.date}</span>
{entry.size != null && (
<span className="text-xs text-neutral-400">· {humanFileSize(entry.size)}</span>
)}
</div>
<p className="text-sm text-neutral-600 leading-snug">{entry.description}</p>
</div>
<button
onClick={() => void install(entry)}
disabled={busy}
className={`flex-shrink-0 flex items-center gap-1.5 px-3 py-1.5 rounded-lg text-sm font-medium transition ${
isInstalling
? 'bg-blue-100 text-blue-600'
: busy
? 'bg-neutral-100 text-neutral-400 cursor-not-allowed'
: 'bg-blue-600 text-white hover:bg-blue-700'
}`}
>
{isInstalling ? (
<>
<Loader2 className="w-3.5 h-3.5 animate-spin" />
Installing
</>
) : (
<>
<Download className="w-3.5 h-3.5" />
Install
</>
)}
</button>
</div>
</div>
);
})}
</div>
</div>
);
}
export default function ToolsPage({ config }: ToolsPageProps) {
const [showFirmware, setShowFirmware] = useState(false);
const [confirm, setConfirm] = useState<ConfirmOptions | null>(null);
const handleBackup = () => {
const dataStr = JSON.stringify(config, null, 2);
const dataUri = 'data:application/json;charset=utf-8,' + encodeURIComponent(dataStr);
const exportFileDefaultName = `meatloaf-config-${new Date().toISOString().split('T')[0]}.json`;
const linkElement = document.createElement('a');
linkElement.setAttribute('href', dataUri);
linkElement.setAttribute('download', exportFileDefaultName);
linkElement.click();
toast.success('Configuration backed up');
};
const handleRestore = () => {
const input = document.createElement('input');
input.type = 'file';
input.accept = '.json';
input.onchange = (e: any) => {
const file = e.target.files[0];
const reader = new FileReader();
reader.onload = (event: any) => {
try {
const config = JSON.parse(event.target.result);
toast.success('Configuration restored');
console.log('Restored config:', config);
} catch (error) {
toast.error('Invalid configuration file');
}
};
reader.readAsText(file);
};
input.click();
};
const handleFactoryReset = () => {
setConfirm({
title: 'Factory Reset',
description: 'Reset to default settings? This cannot be undone.',
confirmLabel: 'Reset',
destructive: true,
onConfirm: () => toast.success('Configuration reset to factory defaults'),
});
};
const handleFormatSD = () => {
setConfirm({
title: 'Format SD Card',
description: 'All data on the SD card will be lost.',
confirmLabel: 'Format',
destructive: true,
onConfirm: () => toast.success('SD card formatting started'),
});
};
const handleSystemUpdate = () => {
toast.loading('Checking for updates...');
setTimeout(() => {
toast.dismiss();
toast.success('System is up to date');
}, 2000);
};
const handleExportLogs = () => {
toast.success('System logs exported');
};
return (
<div className="p-4 space-y-4">
<div className="bg-white border border-neutral-200 rounded-lg divide-y divide-neutral-200">
<div className="p-4">
<div className="flex items-center gap-3 mb-3">
<div className="w-10 h-10 bg-neutral-100 rounded-lg flex items-center justify-center">
<Wrench className="w-5 h-5 text-neutral-600" />
</div>
<div className="flex-1">
<div className="font-medium">System Information</div>
</div>
</div>
<div className="space-y-2 text-sm">
<div className="flex justify-between">
<span className="text-neutral-500">Firmware Version</span>
<span className="text-neutral-900">v2.5.1</span>
</div>
<div className="flex justify-between">
<span className="text-neutral-500">Hardware Revision</span>
<span className="text-neutral-900">Rev C</span>
</div>
<div className="flex justify-between">
<span className="text-neutral-500">Serial Number</span>
<span className="text-neutral-900">ML-2024-0420</span>
</div>
</div>
</div>
<button
onClick={handleSystemUpdate}
className="w-full p-4 flex items-center gap-3 hover:bg-neutral-50 text-left"
>
<div className="w-10 h-10 bg-purple-100 rounded-lg flex items-center justify-center">
<Download className="w-5 h-5 text-purple-600" />
</div>
<div className="flex-1">
<div className="font-medium">System Update</div>
<div className="text-sm text-neutral-500">Check for firmware updates</div>
</div>
</button>
<button
onClick={() => setShowFirmware(true)}
className="w-full p-4 flex items-center gap-3 hover:bg-neutral-50 text-left"
>
<div className="w-10 h-10 bg-purple-100 rounded-lg flex items-center justify-center">
<Cpu className="w-5 h-5 text-purple-600" />
</div>
<div className="flex-1">
<div className="font-medium">Change Firmware</div>
<div className="text-sm text-neutral-500">Select a firmware version to install</div>
</div>
<ChevronRight className="w-4 h-4 text-neutral-400" />
</button>
<button
onClick={handleExportLogs}
className="w-full p-4 flex items-center gap-3 hover:bg-neutral-50 text-left"
>
<div className="w-10 h-10 bg-neutral-100 rounded-lg flex items-center justify-center">
<FileText className="w-5 h-5 text-neutral-600" />
</div>
<div className="flex-1">
<div className="font-medium">Export System Logs</div>
<div className="text-sm text-neutral-500">Download diagnostic logs</div>
</div>
</button>
</div>
<h2 className="text-sm text-neutral-500 flex items-center gap-2"><Wrench className="w-4 h-4" /> System Tools</h2>
<div className="bg-white border border-neutral-200 rounded-lg divide-y divide-neutral-200">
<button
onClick={handleBackup}
className="w-full p-4 flex items-center gap-3 hover:bg-neutral-50 text-left"
>
<div className="w-10 h-10 bg-blue-100 rounded-lg flex items-center justify-center">
<Download className="w-5 h-5 text-blue-600" />
</div>
<div className="flex-1">
<div className="font-medium">Backup Configuration</div>
<div className="text-sm text-neutral-500">Download current config as JSON</div>
</div>
</button>
<button
onClick={handleRestore}
className="w-full p-4 flex items-center gap-3 hover:bg-neutral-50 text-left"
>
<div className="w-10 h-10 bg-green-100 rounded-lg flex items-center justify-center">
<Upload className="w-5 h-5 text-green-600" />
</div>
<div className="flex-1">
<div className="font-medium">Restore Configuration</div>
<div className="text-sm text-neutral-500">Upload a config backup file</div>
</div>
</button>
<button
onClick={handleFactoryReset}
className="w-full p-4 flex items-center gap-3 hover:bg-neutral-50 text-left"
>
<div className="w-10 h-10 bg-red-100 rounded-lg flex items-center justify-center">
<RotateCcw className="w-5 h-5 text-red-600" />
</div>
<div className="flex-1">
<div className="font-medium">Factory Reset</div>
<div className="text-sm text-neutral-500">Reset to default settings</div>
</div>
</button>
</div>
<h2 className="text-sm text-neutral-500 pt-4 flex items-center gap-2"><HardDrive className="w-4 h-4" /> Storage Tools</h2>
<div className="bg-white border border-neutral-200 rounded-lg divide-y divide-neutral-200">
<button
onClick={handleFormatSD}
className="w-full p-4 flex items-center gap-3 hover:bg-neutral-50 text-left"
>
<div className="w-10 h-10 bg-orange-100 rounded-lg flex items-center justify-center">
<HardDrive className="w-5 h-5 text-orange-600" />
</div>
<div className="flex-1">
<div className="font-medium">Format SD Card</div>
<div className="text-sm text-neutral-500">Erase and reformat storage</div>
</div>
</button>
<div className="p-4 flex items-center gap-3">
<div className="w-10 h-10 bg-neutral-100 rounded-lg flex items-center justify-center">
<Database className="w-5 h-5 text-neutral-600" />
</div>
<div className="flex-1">
<div className="font-medium">Storage Info</div>
<div className="text-sm text-neutral-500">4.2 GB used of 8 GB</div>
</div>
</div>
</div>
{showFirmware && <FirmwareOverlay onClose={() => setShowFirmware(false)} />}
<ConfirmDialog
open={confirm !== null}
{...(confirm ?? { title: '', onConfirm: () => {} })}
onCancel={() => setConfirm(null)}
/>
</div>
);
}