"use client" import { useState, useEffect } from "react" import { Card, CardContent, CardHeader, CardTitle } from "./ui/card" import { Badge } from "./ui/badge" import { Dialog, DialogContent, DialogHeader, DialogTitle } from "./ui/dialog" import { Wifi, Globe, Shield, Activity, Network, Router, AlertCircle, Zap } from "lucide-react" interface NetworkData { interfaces: NetworkInterface[] traffic: { bytes_sent: number bytes_recv: number packets_sent?: number packets_recv?: number } } interface NetworkInterface { name: string type: string status: string speed: number duplex: string mtu: number mac_address: string | null addresses: Array<{ ip: string netmask: string }> bytes_sent?: number bytes_recv?: number packets_sent?: number packets_recv?: number errors_in?: number errors_out?: number drops_in?: number drops_out?: number bond_mode?: string bond_slaves?: string[] bond_active_slave?: string | null bridge_members?: string[] } const getInterfaceTypeBadge = (type: string) => { switch (type) { case "physical": return { color: "bg-blue-500/10 text-blue-500 border-blue-500/20", label: "Physical" } case "bridge": return { color: "bg-green-500/10 text-green-500 border-green-500/20", label: "Bridge" } case "bond": return { color: "bg-purple-500/10 text-purple-500 border-purple-500/20", label: "Bond" } case "vlan": return { color: "bg-cyan-500/10 text-cyan-500 border-cyan-500/20", label: "VLAN" } case "virtual": return { color: "bg-orange-500/10 text-orange-500 border-orange-500/20", label: "Virtual" } default: return { color: "bg-gray-500/10 text-gray-500 border-gray-500/20", label: "Unknown" } } } const formatBytes = (bytes: number | undefined): string => { if (!bytes || bytes === 0) return "0 B" const k = 1024 const sizes = ["B", "KB", "MB", "GB", "TB"] const i = Math.floor(Math.log(bytes) / Math.log(k)) return `${(bytes / Math.pow(k, i)).toFixed(2)} ${sizes[i]}` } const formatSpeed = (speed: number): string => { if (speed === 0) return "N/A" if (speed >= 1000) return `${(speed / 1000).toFixed(1)} Gbps` return `${speed} Mbps` } const fetchNetworkData = async (): Promise => { try { console.log("[v0] Fetching network data from Flask server...") const response = await fetch("/api/network", { method: "GET", headers: { "Content-Type": "application/json", }, signal: AbortSignal.timeout(5000), }) if (!response.ok) { throw new Error(`Flask server responded with status: ${response.status}`) } const data = await response.json() console.log("[v0] Successfully fetched network data from Flask:", data) return data } catch (error) { console.error("[v0] Failed to fetch network data from Flask server:", error) return null } } export function NetworkMetrics() { const [networkData, setNetworkData] = useState(null) const [loading, setLoading] = useState(true) const [error, setError] = useState(null) const [selectedInterface, setSelectedInterface] = useState(null) useEffect(() => { const fetchData = async () => { setLoading(true) setError(null) const result = await fetchNetworkData() if (!result) { setError("Flask server not available. Please ensure the server is running.") } else { setNetworkData(result) } setLoading(false) } fetchData() const interval = setInterval(fetchData, 30000) return () => clearInterval(interval) }, []) if (loading) { return (
Loading network data...
) } if (error || !networkData) { return (
Flask Server Not Available
{error || "Unable to connect to the Flask server. Please ensure the server is running and try again."}
) } const trafficInMB = (networkData.traffic.bytes_recv / (1024 * 1024)).toFixed(1) const trafficOutMB = (networkData.traffic.bytes_sent / (1024 * 1024)).toFixed(1) return (
{/* Network Overview Cards */}
Network Traffic
{trafficInMB} MB
↓ {trafficInMB} MB ↑ {trafficOutMB} MB

Total data transferred

Active Interfaces
{networkData.interfaces.filter((i) => i.status === "up").length}
Online

{networkData.interfaces.length} total interfaces

Firewall Status
Active
Protected

System protected

Packets
{networkData.traffic.packets_recv ? (networkData.traffic.packets_recv / 1000).toFixed(0) : "N/A"}K
Received

Total packets received

{/* Network Interfaces */} Network Interfaces
{networkData.interfaces.map((interface_, index) => { const typeBadge = getInterfaceTypeBadge(interface_.type) return (
setSelectedInterface(interface_)} > {/* First row: Icon, Name, Type Badge */}
{interface_.name}
{typeBadge.label}
{interface_.status.toUpperCase()}
{/* Second row: Details */}
IP Address
{interface_.addresses.length > 0 ? interface_.addresses[0].ip : "N/A"}
Speed
{formatSpeed(interface_.speed)}
Traffic
↓ {formatBytes(interface_.bytes_recv)} {" / "} ↑ {formatBytes(interface_.bytes_sent)}
{interface_.mac_address && (
MAC
{interface_.mac_address}
)}
) })}
{/* Interface Details Modal */} setSelectedInterface(null)}> {selectedInterface?.name} - Interface Details {selectedInterface && (
{/* Basic Information */}

Basic Information

Interface Name
{selectedInterface.name}
Type
{getInterfaceTypeBadge(selectedInterface.type).label}
Status
{selectedInterface.status.toUpperCase()}
Speed
{formatSpeed(selectedInterface.speed)}
Duplex
{selectedInterface.duplex}
MTU
{selectedInterface.mtu}
{selectedInterface.mac_address && (
MAC Address
{selectedInterface.mac_address}
)}
{/* IP Addresses */} {selectedInterface.addresses.length > 0 && (

IP Addresses

{selectedInterface.addresses.map((addr, idx) => (
{addr.ip}
Netmask: {addr.netmask}
))}
)} {/* Traffic Statistics */}

Traffic Statistics

Bytes Received
{formatBytes(selectedInterface.bytes_recv)}
Bytes Sent
{formatBytes(selectedInterface.bytes_sent)}
Packets Received
{selectedInterface.packets_recv?.toLocaleString() || "N/A"}
Packets Sent
{selectedInterface.packets_sent?.toLocaleString() || "N/A"}
Errors In
{selectedInterface.errors_in || 0}
Errors Out
{selectedInterface.errors_out || 0}
Drops In
{selectedInterface.drops_in || 0}
Drops Out
{selectedInterface.drops_out || 0}
{/* Bond Information */} {selectedInterface.type === "bond" && selectedInterface.bond_slaves && (

Bond Configuration

Bonding Mode
{selectedInterface.bond_mode || "Unknown"}
{selectedInterface.bond_active_slave && (
Active Slave
{selectedInterface.bond_active_slave}
)}
Slave Interfaces
{selectedInterface.bond_slaves.map((slave, idx) => ( {slave} ))}
)} {/* Bridge Information */} {selectedInterface.type === "bridge" && selectedInterface.bridge_members && (

Bridge Configuration

Member Interfaces
{selectedInterface.bridge_members.length > 0 ? ( selectedInterface.bridge_members.map((member, idx) => ( {member} )) ) : (
No members
)}
)}
)}
) }