Files
netbird/client/netbird-electron/src/pages/Debug.tsx
2025-11-01 12:17:02 +01:00

222 lines
8.4 KiB
TypeScript

import { useState } from 'react';
import { motion } from 'framer-motion';
import { Bug, Package, AlertCircle, CheckCircle2, Copy, Check, ArrowLeft } from 'lucide-react';
interface DebugPageProps {
onBack?: () => void;
}
export default function DebugPage({ onBack }: DebugPageProps) {
const [creating, setCreating] = useState(false);
const [anonymize, setAnonymize] = useState(true);
const [bundlePath, setBundlePath] = useState('');
const [error, setError] = useState('');
const [copied, setCopied] = useState(false);
const handleCreateBundle = async () => {
try {
setCreating(true);
setError('');
setBundlePath('');
setCopied(false);
// TODO: Implement debug bundle creation via IPC
// const path = await window.electronAPI.daemon.createDebugBundle(anonymize);
// setBundlePath(path);
// Simulated for now
await new Promise((resolve) => setTimeout(resolve, 2000));
setBundlePath('/tmp/netbird-debug-bundle-20241030.zip');
} catch (err) {
setError('Failed to create debug bundle');
console.error('Debug bundle error:', err);
} finally {
setCreating(false);
}
};
const handleCopyPath = async () => {
try {
await navigator.clipboard.writeText(bundlePath);
setCopied(true);
setTimeout(() => setCopied(false), 2000);
} catch (err) {
console.error('Failed to copy path:', err);
}
};
return (
<div className="h-full overflow-auto p-8">
<div className="max-w-4xl mx-auto space-y-6">
{/* Back Button */}
{onBack && (
<motion.button
whileHover={{ scale: 1.05 }}
whileTap={{ scale: 0.95 }}
onClick={onBack}
className="flex items-center gap-2 px-4 py-2 nb-frosted rounded-lg nb-border hover:nb-border-strong transition-all"
>
<ArrowLeft className="w-4 h-4 text-nb-orange" />
<span className="text-text-light font-medium">Back</span>
</motion.button>
)}
{/* Header */}
<motion.div
initial={{ opacity: 0, y: -20 }}
animate={{ opacity: 1, y: 0 }}
className="mb-8"
>
<h1 className="text-3xl font-bold text-text-light text-orange-glow mb-2">Debug Bundle</h1>
<p className="text-text-muted">Create diagnostic bundle for troubleshooting</p>
</motion.div>
{/* Info Card */}
<motion.div
initial={{ opacity: 0, y: 20 }}
animate={{ opacity: 1, y: 0 }}
className="nb-card rounded-nb-card p-6 shadow-nb-card"
>
<div className="flex items-start gap-4 mb-6">
<div className="p-3 bg-nb-orange/20 rounded-lg">
<AlertCircle className="w-6 h-6 text-nb-orange" />
</div>
<div>
<h3 className="text-lg font-bold text-text-light mb-2">What's included?</h3>
<ul className="space-y-2 text-sm text-text-muted">
<li>• System information</li>
<li>• NetBird configuration</li>
<li>• Network interfaces</li>
<li>• Routing tables</li>
<li>• Daemon logs</li>
</ul>
</div>
</div>
{/* Anonymize option */}
<div
className="flex items-start gap-3 p-4 rounded-lg hover:bg-nb-orange/5 transition-all cursor-pointer border border-transparent hover:border-nb-orange/20"
onClick={() => setAnonymize(!anonymize)}
>
<div
className={`relative w-12 h-6 rounded-full p-1 transition-all ${
anonymize ? 'bg-nb-orange shadow-icy-glow' : 'bg-text-muted/30'
}`}
>
<motion.div
animate={{ x: anonymize ? 24 : 0 }}
transition={{ type: "spring", stiffness: 500, damping: 30 }}
className="w-4 h-4 bg-white rounded-full shadow-lg"
/>
</div>
<div className="flex-1">
<h3 className="font-semibold text-text-light">Anonymize sensitive data</h3>
<p className="text-sm text-text-muted mt-1">
Replace IP addresses, emails, and other identifying information
</p>
</div>
</div>
</motion.div>
{/* Create Button */}
<motion.button
whileHover={{ scale: 1.02 }}
whileTap={{ scale: 0.98 }}
onClick={handleCreateBundle}
disabled={creating}
className="w-full py-4 bg-nb-orange/30 text-nb-orange hover:bg-nb-orange/40 rounded-lg font-bold flex items-center justify-center gap-2 nb-border hover:nb-border-strong transition-all disabled:opacity-50 disabled:cursor-not-allowed"
>
<Package className="w-5 h-5" />
{creating ? 'Creating Bundle...' : 'Create Debug Bundle'}
</motion.button>
{/* Success message */}
{bundlePath && (
<motion.div
initial={{ opacity: 0, scale: 0.95 }}
animate={{ opacity: 1, scale: 1 }}
className="nb-card rounded-nb-card p-6 border-2 border-nb-orange/30 shadow-nb-card"
>
<div className="flex items-start gap-4">
<div className="p-3 bg-nb-orange/20 rounded-lg">
<CheckCircle2 className="w-6 h-6 text-nb-orange" />
</div>
<div className="flex-1">
<h3 className="text-lg font-bold text-text-light mb-2">Bundle Created!</h3>
<p className="text-sm text-text-muted mb-3">
Your debug bundle has been created successfully
</p>
<div className="p-3 bg-gray-bg-card rounded-lg border border-nb-orange/10">
<div className="flex items-center justify-between gap-2 mb-1">
<p className="text-xs text-text-muted">File location:</p>
<button
onClick={handleCopyPath}
className="p-1 hover:bg-nb-orange/20 rounded transition-all"
title="Copy path"
>
{copied ? (
<Check className="w-4 h-4 text-green-400" />
) : (
<Copy className="w-4 h-4 text-text-muted hover:text-nb-orange" />
)}
</button>
</div>
<p className="text-sm text-nb-orange font-mono break-all">{bundlePath}</p>
</div>
</div>
</div>
</motion.div>
)}
{/* Error message */}
{error && (
<motion.div
initial={{ opacity: 0, scale: 0.95 }}
animate={{ opacity: 1, scale: 1 }}
className="nb-card rounded-nb-card p-6 border-2 border-red-500/30 shadow-nb-card"
>
<div className="flex items-start gap-4">
<div className="p-3 bg-red-500/20 rounded-lg">
<AlertCircle className="w-6 h-6 text-red-500" />
</div>
<div>
<h3 className="text-lg font-bold text-text-light mb-2">Error</h3>
<p className="text-sm text-text-muted">{error}</p>
</div>
</div>
</motion.div>
)}
{/* Additional Info */}
<motion.div
initial={{ opacity: 0, y: 20 }}
animate={{ opacity: 1, y: 0 }}
transition={{ delay: 0.1 }}
className="nb-card rounded-nb-card p-6 shadow-nb-card"
>
<div className="flex items-start gap-4">
<div className="p-3 bg-text-muted/20 rounded-lg">
<Bug className="w-6 h-6 text-text-muted" />
</div>
<div>
<h3 className="text-lg font-bold text-text-light mb-2">Need Help?</h3>
<p className="text-sm text-text-muted mb-3">
If you're experiencing issues, create a debug bundle and share it with the NetBird
support team.
</p>
<a
href="https://github.com/netbirdio/netbird/issues"
target="_blank"
rel="noopener noreferrer"
className="text-sm text-nb-orange hover:underline"
>
Report an issue on GitHub
</a>
</div>
</div>
</motion.div>
</div>
</div>
);
}