mirror of
https://github.com/henrygd/beszel.git
synced 2026-03-23 05:56:17 +01:00
Compare commits
1 Commits
split-inte
...
battery-fi
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
3e73399b87 |
1
.gitignore
vendored
1
.gitignore
vendored
@@ -20,4 +20,3 @@ __debug_*
|
|||||||
agent/lhm/obj
|
agent/lhm/obj
|
||||||
agent/lhm/bin
|
agent/lhm/bin
|
||||||
dockerfile_agent_dev
|
dockerfile_agent_dev
|
||||||
.vite
|
|
||||||
2
Makefile
2
Makefile
@@ -77,7 +77,7 @@ dev-hub: export ENV=dev
|
|||||||
dev-hub:
|
dev-hub:
|
||||||
mkdir -p ./internal/site/dist && touch ./internal/site/dist/index.html
|
mkdir -p ./internal/site/dist && touch ./internal/site/dist/index.html
|
||||||
@if command -v entr >/dev/null 2>&1; then \
|
@if command -v entr >/dev/null 2>&1; then \
|
||||||
find ./internal -type f -name '*.go' | entr -r -s "cd ./internal/cmd/hub && go run -tags development . serve --http 0.0.0.0:8090"; \
|
find ./internal/cmd/hub/*.go ./internal/{alerts,hub,records,users}/*.go | entr -r -s "cd ./internal/cmd/hub && go run -tags development . serve --http 0.0.0.0:8090"; \
|
||||||
else \
|
else \
|
||||||
cd ./internal/cmd/hub && go run -tags development . serve --http 0.0.0.0:8090; \
|
cd ./internal/cmd/hub && go run -tags development . serve --http 0.0.0.0:8090; \
|
||||||
fi
|
fi
|
||||||
|
|||||||
@@ -22,23 +22,23 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
type Agent struct {
|
type Agent struct {
|
||||||
sync.Mutex // Used to lock agent while collecting data
|
sync.Mutex // Used to lock agent while collecting data
|
||||||
debug bool // true if LOG_LEVEL is set to debug
|
debug bool // true if LOG_LEVEL is set to debug
|
||||||
zfs bool // true if system has arcstats
|
zfs bool // true if system has arcstats
|
||||||
memCalc string // Memory calculation formula
|
memCalc string // Memory calculation formula
|
||||||
fsNames []string // List of filesystem device names being monitored
|
fsNames []string // List of filesystem device names being monitored
|
||||||
fsStats map[string]*system.FsStats // Keeps track of disk stats for each filesystem
|
fsStats map[string]*system.FsStats // Keeps track of disk stats for each filesystem
|
||||||
netInterfaces map[string]struct{} // Stores all valid network interfaces
|
netInterfaces map[string]struct{} // Stores all valid network interfaces
|
||||||
netIoStats map[string]system.NetIoStats // Keeps track of per-interface bandwidth usage
|
netIoStats system.NetIoStats // Keeps track of bandwidth usage
|
||||||
dockerManager *dockerManager // Manages Docker API requests
|
dockerManager *dockerManager // Manages Docker API requests
|
||||||
sensorConfig *SensorConfig // Sensors config
|
sensorConfig *SensorConfig // Sensors config
|
||||||
systemInfo system.Info // Host system info
|
systemInfo system.Info // Host system info
|
||||||
gpuManager *GPUManager // Manages GPU data
|
gpuManager *GPUManager // Manages GPU data
|
||||||
cache *SessionCache // Cache for system stats based on primary session ID
|
cache *SessionCache // Cache for system stats based on primary session ID
|
||||||
connectionManager *ConnectionManager // Channel to signal connection events
|
connectionManager *ConnectionManager // Channel to signal connection events
|
||||||
server *ssh.Server // SSH server
|
server *ssh.Server // SSH server
|
||||||
dataDir string // Directory for persisting data
|
dataDir string // Directory for persisting data
|
||||||
keys []gossh.PublicKey // SSH public keys
|
keys []gossh.PublicKey // SSH public keys
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewAgent creates a new agent with the given data directory for persisting data.
|
// NewAgent creates a new agent with the given data directory for persisting data.
|
||||||
|
|||||||
@@ -20,9 +20,8 @@ func HasReadableBattery() bool {
|
|||||||
}
|
}
|
||||||
haveCheckedBattery = true
|
haveCheckedBattery = true
|
||||||
bat, err := battery.Get(0)
|
bat, err := battery.Get(0)
|
||||||
if err == nil && bat != nil {
|
systemHasBattery = err == nil && bat != nil && bat.Design != 0 && bat.Full != 0
|
||||||
systemHasBattery = true
|
if !systemHasBattery {
|
||||||
} else {
|
|
||||||
slog.Debug("No battery found", "err", err)
|
slog.Debug("No battery found", "err", err)
|
||||||
}
|
}
|
||||||
return systemHasBattery
|
return systemHasBattery
|
||||||
|
|||||||
@@ -5,16 +5,12 @@ import (
|
|||||||
"strings"
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/henrygd/beszel/internal/entities/system"
|
|
||||||
|
|
||||||
psutilNet "github.com/shirou/gopsutil/v4/net"
|
psutilNet "github.com/shirou/gopsutil/v4/net"
|
||||||
)
|
)
|
||||||
|
|
||||||
func (a *Agent) initializeNetIoStats() {
|
func (a *Agent) initializeNetIoStats() {
|
||||||
// reset valid network interfaces
|
// reset valid network interfaces
|
||||||
a.netInterfaces = make(map[string]struct{}, 0)
|
a.netInterfaces = make(map[string]struct{}, 0)
|
||||||
// reset network I/O stats per interface
|
|
||||||
a.netIoStats = make(map[string]system.NetIoStats, 0)
|
|
||||||
|
|
||||||
// map of network interface names passed in via NICS env var
|
// map of network interface names passed in via NICS env var
|
||||||
var nicsMap map[string]struct{}
|
var nicsMap map[string]struct{}
|
||||||
@@ -26,10 +22,13 @@ func (a *Agent) initializeNetIoStats() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// reset network I/O stats
|
||||||
|
a.netIoStats.BytesSent = 0
|
||||||
|
a.netIoStats.BytesRecv = 0
|
||||||
|
|
||||||
// get intial network I/O stats
|
// get intial network I/O stats
|
||||||
if netIO, err := psutilNet.IOCounters(true); err == nil {
|
if netIO, err := psutilNet.IOCounters(true); err == nil {
|
||||||
now := time.Now()
|
a.netIoStats.Time = time.Now()
|
||||||
|
|
||||||
for _, v := range netIO {
|
for _, v := range netIO {
|
||||||
switch {
|
switch {
|
||||||
// skip if nics exists and the interface is not in the list
|
// skip if nics exists and the interface is not in the list
|
||||||
@@ -44,15 +43,10 @@ func (a *Agent) initializeNetIoStats() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
slog.Info("Detected network interface", "name", v.Name, "sent", v.BytesSent, "recv", v.BytesRecv)
|
slog.Info("Detected network interface", "name", v.Name, "sent", v.BytesSent, "recv", v.BytesRecv)
|
||||||
|
a.netIoStats.BytesSent += v.BytesSent
|
||||||
|
a.netIoStats.BytesRecv += v.BytesRecv
|
||||||
// store as a valid network interface
|
// store as a valid network interface
|
||||||
a.netInterfaces[v.Name] = struct{}{}
|
a.netInterfaces[v.Name] = struct{}{}
|
||||||
// initialize per-interface stats
|
|
||||||
a.netIoStats[v.Name] = system.NetIoStats{
|
|
||||||
BytesRecv: v.BytesRecv,
|
|
||||||
BytesSent: v.BytesSent,
|
|
||||||
Time: now,
|
|
||||||
Name: v.Name,
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
195
agent/system.go
195
agent/system.go
@@ -176,85 +176,53 @@ func (a *Agent) getSystemStats() system.Stats {
|
|||||||
if len(a.netInterfaces) == 0 {
|
if len(a.netInterfaces) == 0 {
|
||||||
// if no network interfaces, initialize again
|
// if no network interfaces, initialize again
|
||||||
// this is a fix if agent started before network is online (#466)
|
// this is a fix if agent started before network is online (#466)
|
||||||
|
// maybe refactor this in the future to not cache interface names at all so we
|
||||||
|
// don't miss an interface that's been added after agent started in any circumstance
|
||||||
a.initializeNetIoStats()
|
a.initializeNetIoStats()
|
||||||
}
|
}
|
||||||
if netIO, err := psutilNet.IOCounters(true); err == nil {
|
if netIO, err := psutilNet.IOCounters(true); err == nil {
|
||||||
now := time.Now()
|
msElapsed := uint64(time.Since(a.netIoStats.Time).Milliseconds())
|
||||||
|
a.netIoStats.Time = time.Now()
|
||||||
// pre-allocate maps with known capacity
|
totalBytesSent := uint64(0)
|
||||||
interfaceCount := len(a.netInterfaces)
|
totalBytesRecv := uint64(0)
|
||||||
if systemStats.NetworkInterfaces == nil || len(systemStats.NetworkInterfaces) != interfaceCount {
|
// sum all bytes sent and received
|
||||||
systemStats.NetworkInterfaces = make(map[string]system.NetworkInterfaceStats, interfaceCount)
|
|
||||||
}
|
|
||||||
|
|
||||||
var totalSent, totalRecv float64
|
|
||||||
|
|
||||||
// single pass through interfaces
|
|
||||||
for _, v := range netIO {
|
for _, v := range netIO {
|
||||||
// skip if not in valid network interfaces list
|
// skip if not in valid network interfaces list
|
||||||
if _, exists := a.netInterfaces[v.Name]; !exists {
|
if _, exists := a.netInterfaces[v.Name]; !exists {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
totalBytesSent += v.BytesSent
|
||||||
// get previous stats for this interface
|
totalBytesRecv += v.BytesRecv
|
||||||
prevStats, exists := a.netIoStats[v.Name]
|
|
||||||
var networkSentPs, networkRecvPs float64
|
|
||||||
|
|
||||||
if exists {
|
|
||||||
secondsElapsed := time.Since(prevStats.Time).Seconds()
|
|
||||||
if secondsElapsed > 0 {
|
|
||||||
// direct calculation to MB/s, avoiding intermediate bytes/sec
|
|
||||||
networkSentPs = bytesToMegabytes(float64(v.BytesSent-prevStats.BytesSent) / secondsElapsed)
|
|
||||||
networkRecvPs = bytesToMegabytes(float64(v.BytesRecv-prevStats.BytesRecv) / secondsElapsed)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// accumulate totals
|
|
||||||
totalSent += networkSentPs
|
|
||||||
totalRecv += networkRecvPs
|
|
||||||
|
|
||||||
// store per-interface stats
|
|
||||||
systemStats.NetworkInterfaces[v.Name] = system.NetworkInterfaceStats{
|
|
||||||
NetworkSent: networkSentPs,
|
|
||||||
NetworkRecv: networkRecvPs,
|
|
||||||
TotalBytesSent: v.BytesSent,
|
|
||||||
TotalBytesRecv: v.BytesRecv,
|
|
||||||
}
|
|
||||||
|
|
||||||
// update previous stats (reuse existing struct if possible)
|
|
||||||
if prevStats.Name == v.Name {
|
|
||||||
prevStats.BytesRecv = v.BytesRecv
|
|
||||||
prevStats.BytesSent = v.BytesSent
|
|
||||||
prevStats.PacketsSent = v.PacketsSent
|
|
||||||
prevStats.PacketsRecv = v.PacketsRecv
|
|
||||||
prevStats.Time = now
|
|
||||||
a.netIoStats[v.Name] = prevStats
|
|
||||||
} else {
|
|
||||||
a.netIoStats[v.Name] = system.NetIoStats{
|
|
||||||
BytesRecv: v.BytesRecv,
|
|
||||||
BytesSent: v.BytesSent,
|
|
||||||
PacketsSent: v.PacketsSent,
|
|
||||||
PacketsRecv: v.PacketsRecv,
|
|
||||||
Time: now,
|
|
||||||
Name: v.Name,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
// add to systemStats
|
||||||
|
var bytesSentPerSecond, bytesRecvPerSecond uint64
|
||||||
|
if msElapsed > 0 {
|
||||||
|
bytesSentPerSecond = (totalBytesSent - a.netIoStats.BytesSent) * 1000 / msElapsed
|
||||||
|
bytesRecvPerSecond = (totalBytesRecv - a.netIoStats.BytesRecv) * 1000 / msElapsed
|
||||||
|
}
|
||||||
|
networkSentPs := bytesToMegabytes(float64(bytesSentPerSecond))
|
||||||
|
networkRecvPs := bytesToMegabytes(float64(bytesRecvPerSecond))
|
||||||
// add check for issue (#150) where sent is a massive number
|
// add check for issue (#150) where sent is a massive number
|
||||||
if totalSent > 10_000 || totalRecv > 10_000 {
|
if networkSentPs > 10_000 || networkRecvPs > 10_000 {
|
||||||
slog.Warn("Invalid net stats. Resetting.", "sent", totalSent, "recv", totalRecv)
|
slog.Warn("Invalid net stats. Resetting.", "sent", networkSentPs, "recv", networkRecvPs)
|
||||||
|
for _, v := range netIO {
|
||||||
|
if _, exists := a.netInterfaces[v.Name]; !exists {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
slog.Info(v.Name, "recv", v.BytesRecv, "sent", v.BytesSent)
|
||||||
|
}
|
||||||
// reset network I/O stats
|
// reset network I/O stats
|
||||||
a.initializeNetIoStats()
|
a.initializeNetIoStats()
|
||||||
} else {
|
} else {
|
||||||
systemStats.NetworkSent = totalSent
|
systemStats.NetworkSent = networkSentPs
|
||||||
systemStats.NetworkRecv = totalRecv
|
systemStats.NetworkRecv = networkRecvPs
|
||||||
|
systemStats.Bandwidth[0], systemStats.Bandwidth[1] = bytesSentPerSecond, bytesRecvPerSecond
|
||||||
|
// update netIoStats
|
||||||
|
a.netIoStats.BytesSent = totalBytesSent
|
||||||
|
a.netIoStats.BytesRecv = totalBytesRecv
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// connection counts
|
|
||||||
a.updateConnectionCounts(&systemStats)
|
|
||||||
|
|
||||||
// temperatures
|
// temperatures
|
||||||
// TODO: maybe refactor to methods on systemStats
|
// TODO: maybe refactor to methods on systemStats
|
||||||
a.updateTemperatures(&systemStats)
|
a.updateTemperatures(&systemStats)
|
||||||
@@ -302,109 +270,14 @@ func (a *Agent) getSystemStats() system.Stats {
|
|||||||
a.systemInfo.MemPct = systemStats.MemPct
|
a.systemInfo.MemPct = systemStats.MemPct
|
||||||
a.systemInfo.DiskPct = systemStats.DiskPct
|
a.systemInfo.DiskPct = systemStats.DiskPct
|
||||||
a.systemInfo.Uptime, _ = host.Uptime()
|
a.systemInfo.Uptime, _ = host.Uptime()
|
||||||
|
// TODO: in future release, remove MB bandwidth values in favor of bytes
|
||||||
// Sum all per-interface network sent/recv and assign to systemInfo
|
a.systemInfo.Bandwidth = twoDecimals(systemStats.NetworkSent + systemStats.NetworkRecv)
|
||||||
var totalSent, totalRecv float64
|
a.systemInfo.BandwidthBytes = systemStats.Bandwidth[0] + systemStats.Bandwidth[1]
|
||||||
for _, iface := range systemStats.NetworkInterfaces {
|
|
||||||
totalSent += iface.NetworkSent
|
|
||||||
totalRecv += iface.NetworkRecv
|
|
||||||
}
|
|
||||||
a.systemInfo.NetworkSent = twoDecimals(totalSent)
|
|
||||||
a.systemInfo.NetworkRecv = twoDecimals(totalRecv)
|
|
||||||
slog.Debug("sysinfo", "data", a.systemInfo)
|
slog.Debug("sysinfo", "data", a.systemInfo)
|
||||||
|
|
||||||
return systemStats
|
return systemStats
|
||||||
}
|
}
|
||||||
|
|
||||||
func (a *Agent) updateConnectionCounts(systemStats *system.Stats) {
|
|
||||||
// Get IPv4 connections
|
|
||||||
connectionsIPv4, err := psutilNet.Connections("inet")
|
|
||||||
if err != nil {
|
|
||||||
slog.Debug("Failed to get IPv4 connection stats", "err", err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// Get IPv6 connections
|
|
||||||
connectionsIPv6, err := psutilNet.Connections("inet6")
|
|
||||||
if err != nil {
|
|
||||||
slog.Debug("Failed to get IPv6 connection stats", "err", err)
|
|
||||||
// Continue with IPv4 only if IPv6 fails
|
|
||||||
}
|
|
||||||
|
|
||||||
// Initialize Nets map if needed
|
|
||||||
if systemStats.Nets == nil {
|
|
||||||
systemStats.Nets = make(map[string]float64)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Count IPv4 connection states
|
|
||||||
connStatsIPv4 := map[string]int{
|
|
||||||
"established": 0,
|
|
||||||
"listen": 0,
|
|
||||||
"time_wait": 0,
|
|
||||||
"close_wait": 0,
|
|
||||||
"syn_recv": 0,
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, conn := range connectionsIPv4 {
|
|
||||||
// Only count TCP connections (Type 1 = SOCK_STREAM)
|
|
||||||
if conn.Type == 1 {
|
|
||||||
switch strings.ToUpper(conn.Status) {
|
|
||||||
case "ESTABLISHED":
|
|
||||||
connStatsIPv4["established"]++
|
|
||||||
case "LISTEN":
|
|
||||||
connStatsIPv4["listen"]++
|
|
||||||
case "TIME_WAIT":
|
|
||||||
connStatsIPv4["time_wait"]++
|
|
||||||
case "CLOSE_WAIT":
|
|
||||||
connStatsIPv4["close_wait"]++
|
|
||||||
case "SYN_RECV":
|
|
||||||
connStatsIPv4["syn_recv"]++
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Count IPv6 connection states
|
|
||||||
connStatsIPv6 := map[string]int{
|
|
||||||
"established": 0,
|
|
||||||
"listen": 0,
|
|
||||||
"time_wait": 0,
|
|
||||||
"close_wait": 0,
|
|
||||||
"syn_recv": 0,
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, conn := range connectionsIPv6 {
|
|
||||||
// Only count TCP connections (Type 1 = SOCK_STREAM)
|
|
||||||
if conn.Type == 1 {
|
|
||||||
switch strings.ToUpper(conn.Status) {
|
|
||||||
case "ESTABLISHED":
|
|
||||||
connStatsIPv6["established"]++
|
|
||||||
case "LISTEN":
|
|
||||||
connStatsIPv6["listen"]++
|
|
||||||
case "TIME_WAIT":
|
|
||||||
connStatsIPv6["time_wait"]++
|
|
||||||
case "CLOSE_WAIT":
|
|
||||||
connStatsIPv6["close_wait"]++
|
|
||||||
case "SYN_RECV":
|
|
||||||
connStatsIPv6["syn_recv"]++
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Add IPv4 connection counts to Nets
|
|
||||||
systemStats.Nets["conn_established"] = float64(connStatsIPv4["established"])
|
|
||||||
systemStats.Nets["conn_listen"] = float64(connStatsIPv4["listen"])
|
|
||||||
systemStats.Nets["conn_timewait"] = float64(connStatsIPv4["time_wait"])
|
|
||||||
systemStats.Nets["conn_closewait"] = float64(connStatsIPv4["close_wait"])
|
|
||||||
systemStats.Nets["conn_synrecv"] = float64(connStatsIPv4["syn_recv"])
|
|
||||||
|
|
||||||
// Add IPv6 connection counts to Nets
|
|
||||||
systemStats.Nets["conn6_established"] = float64(connStatsIPv6["established"])
|
|
||||||
systemStats.Nets["conn6_listen"] = float64(connStatsIPv6["listen"])
|
|
||||||
systemStats.Nets["conn6_timewait"] = float64(connStatsIPv6["time_wait"])
|
|
||||||
systemStats.Nets["conn6_closewait"] = float64(connStatsIPv6["close_wait"])
|
|
||||||
systemStats.Nets["conn6_synrecv"] = float64(connStatsIPv6["syn_recv"])
|
|
||||||
}
|
|
||||||
|
|
||||||
// Returns the size of the ZFS ARC memory cache in bytes
|
// Returns the size of the ZFS ARC memory cache in bytes
|
||||||
func getARCSize() (uint64, error) {
|
func getARCSize() (uint64, error) {
|
||||||
file, err := os.Open("/proc/spl/kstat/zfs/arcstats")
|
file, err := os.Open("/proc/spl/kstat/zfs/arcstats")
|
||||||
|
|||||||
@@ -38,7 +38,7 @@ func (am *AlertManager) HandleSystemAlerts(systemRecord *core.Record, data *syst
|
|||||||
case "Memory":
|
case "Memory":
|
||||||
val = data.Info.MemPct
|
val = data.Info.MemPct
|
||||||
case "Bandwidth":
|
case "Bandwidth":
|
||||||
val = data.Info.NetworkSent + data.Info.NetworkRecv
|
val = data.Info.Bandwidth
|
||||||
unit = " MB/s"
|
unit = " MB/s"
|
||||||
case "Disk":
|
case "Disk":
|
||||||
maxUsedPct := data.Info.DiskPct
|
maxUsedPct := data.Info.DiskPct
|
||||||
|
|||||||
@@ -8,47 +8,39 @@ import (
|
|||||||
"github.com/henrygd/beszel/internal/entities/container"
|
"github.com/henrygd/beszel/internal/entities/container"
|
||||||
)
|
)
|
||||||
|
|
||||||
type NetworkInterfaceStats struct {
|
|
||||||
NetworkSent float64 `json:"ns"`
|
|
||||||
NetworkRecv float64 `json:"nr"`
|
|
||||||
MaxNetworkSent float64 `json:"nsm,omitempty"`
|
|
||||||
MaxNetworkRecv float64 `json:"nrm,omitempty"`
|
|
||||||
TotalBytesSent uint64 `json:"tbs,omitempty"` // Total bytes sent since boot
|
|
||||||
TotalBytesRecv uint64 `json:"tbr,omitempty"` // Total bytes received since boot
|
|
||||||
}
|
|
||||||
|
|
||||||
type Stats struct {
|
type Stats struct {
|
||||||
Cpu float64 `json:"cpu" cbor:"0,keyasint"`
|
Cpu float64 `json:"cpu" cbor:"0,keyasint"`
|
||||||
MaxCpu float64 `json:"cpum,omitempty" cbor:"1,keyasint,omitempty"`
|
MaxCpu float64 `json:"cpum,omitempty" cbor:"1,keyasint,omitempty"`
|
||||||
Mem float64 `json:"m" cbor:"2,keyasint"`
|
Mem float64 `json:"m" cbor:"2,keyasint"`
|
||||||
MemUsed float64 `json:"mu" cbor:"3,keyasint"`
|
MemUsed float64 `json:"mu" cbor:"3,keyasint"`
|
||||||
MemPct float64 `json:"mp" cbor:"4,keyasint"`
|
MemPct float64 `json:"mp" cbor:"4,keyasint"`
|
||||||
MemBuffCache float64 `json:"mb" cbor:"5,keyasint"`
|
MemBuffCache float64 `json:"mb" cbor:"5,keyasint"`
|
||||||
MemZfsArc float64 `json:"mz,omitempty" cbor:"6,keyasint,omitempty"` // ZFS ARC memory
|
MemZfsArc float64 `json:"mz,omitempty" cbor:"6,keyasint,omitempty"` // ZFS ARC memory
|
||||||
Swap float64 `json:"s,omitempty" cbor:"7,keyasint,omitempty"`
|
Swap float64 `json:"s,omitempty" cbor:"7,keyasint,omitempty"`
|
||||||
SwapUsed float64 `json:"su,omitempty" cbor:"8,keyasint,omitempty"`
|
SwapUsed float64 `json:"su,omitempty" cbor:"8,keyasint,omitempty"`
|
||||||
DiskTotal float64 `json:"d" cbor:"9,keyasint"`
|
DiskTotal float64 `json:"d" cbor:"9,keyasint"`
|
||||||
DiskUsed float64 `json:"du" cbor:"10,keyasint"`
|
DiskUsed float64 `json:"du" cbor:"10,keyasint"`
|
||||||
DiskPct float64 `json:"dp" cbor:"11,keyasint"`
|
DiskPct float64 `json:"dp" cbor:"11,keyasint"`
|
||||||
DiskReadPs float64 `json:"dr" cbor:"12,keyasint"`
|
DiskReadPs float64 `json:"dr" cbor:"12,keyasint"`
|
||||||
DiskWritePs float64 `json:"dw" cbor:"13,keyasint"`
|
DiskWritePs float64 `json:"dw" cbor:"13,keyasint"`
|
||||||
MaxDiskReadPs float64 `json:"drm,omitempty" cbor:"14,keyasint,omitempty"`
|
MaxDiskReadPs float64 `json:"drm,omitempty" cbor:"14,keyasint,omitempty"`
|
||||||
MaxDiskWritePs float64 `json:"dwm,omitempty" cbor:"15,keyasint,omitempty"`
|
MaxDiskWritePs float64 `json:"dwm,omitempty" cbor:"15,keyasint,omitempty"`
|
||||||
NetworkInterfaces map[string]NetworkInterfaceStats `json:"ni" cbor:"16,omitempty"` // Per-interface network stats
|
NetworkSent float64 `json:"ns" cbor:"16,keyasint"`
|
||||||
NetworkSent float64 `json:"ns" cbor:"17,keyasint"` // Total network sent (MB/s)
|
NetworkRecv float64 `json:"nr" cbor:"17,keyasint"`
|
||||||
NetworkRecv float64 `json:"nr" cbor:"18,keyasint"` // Total network recv (MB/s)
|
MaxNetworkSent float64 `json:"nsm,omitempty" cbor:"18,keyasint,omitempty"`
|
||||||
MaxNetworkSent float64 `json:"nsm,omitempty" cbor:"19,keyasint,omitempty"`
|
MaxNetworkRecv float64 `json:"nrm,omitempty" cbor:"19,keyasint,omitempty"`
|
||||||
MaxNetworkRecv float64 `json:"nrm,omitempty" cbor:"20,keyasint,omitempty"`
|
Temperatures map[string]float64 `json:"t,omitempty" cbor:"20,keyasint,omitempty"`
|
||||||
Temperatures map[string]float64 `json:"t,omitempty" cbor:"21,keyasint,omitempty"`
|
ExtraFs map[string]*FsStats `json:"efs,omitempty" cbor:"21,keyasint,omitempty"`
|
||||||
ExtraFs map[string]*FsStats `json:"efs,omitempty" cbor:"22,keyasint,omitempty"`
|
GPUData map[string]GPUData `json:"g,omitempty" cbor:"22,keyasint,omitempty"`
|
||||||
GPUData map[string]GPUData `json:"g,omitempty" cbor:"23,keyasint,omitempty"`
|
LoadAvg1 float64 `json:"l1,omitempty" cbor:"23,keyasint,omitempty"`
|
||||||
LoadAvg1 float64 `json:"l1,omitempty" cbor:"24,keyasint,omitempty"`
|
LoadAvg5 float64 `json:"l5,omitempty" cbor:"24,keyasint,omitempty"`
|
||||||
LoadAvg5 float64 `json:"l5,omitempty" cbor:"25,keyasint,omitempty"`
|
LoadAvg15 float64 `json:"l15,omitempty" cbor:"25,keyasint,omitempty"`
|
||||||
LoadAvg15 float64 `json:"l15,omitempty" cbor:"26,keyasint,omitempty"`
|
Bandwidth [2]uint64 `json:"b,omitzero" cbor:"26,keyasint,omitzero"` // [sent bytes, recv bytes]
|
||||||
LoadAvg [3]float64 `json:"la,omitempty" cbor:"27,keyasint"` // [1min, 5min, 15min]
|
MaxBandwidth [2]uint64 `json:"bm,omitzero" cbor:"27,keyasint,omitzero"` // [sent bytes, recv bytes]
|
||||||
Battery [2]uint8 `json:"bat,omitzero" cbor:"28,keyasint,omitzero"` // [percent, charge state]
|
// TODO: remove other load fields in future release in favor of load avg array
|
||||||
MaxMem float64 `json:"mm,omitempty" cbor:"29,keyasint,omitempty"`
|
LoadAvg [3]float64 `json:"la,omitempty" cbor:"28,keyasint"`
|
||||||
Nets map[string]float64 `json:"nets,omitempty" cbor:"30,keyasint,omitempty"` // Network connection statistics
|
Battery [2]uint8 `json:"bat,omitzero" cbor:"29,keyasint,omitzero"` // [percent, charge state, current]
|
||||||
|
MaxMem float64 `json:"mm,omitempty" cbor:"30,keyasint,omitempty"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type GPUData struct {
|
type GPUData struct {
|
||||||
@@ -76,12 +68,10 @@ type FsStats struct {
|
|||||||
}
|
}
|
||||||
|
|
||||||
type NetIoStats struct {
|
type NetIoStats struct {
|
||||||
BytesRecv uint64
|
BytesRecv uint64
|
||||||
BytesSent uint64
|
BytesSent uint64
|
||||||
PacketsSent uint64
|
Time time.Time
|
||||||
PacketsRecv uint64
|
Name string
|
||||||
Time time.Time
|
|
||||||
Name string
|
|
||||||
}
|
}
|
||||||
|
|
||||||
type Os = uint8
|
type Os = uint8
|
||||||
@@ -94,26 +84,27 @@ const (
|
|||||||
)
|
)
|
||||||
|
|
||||||
type Info struct {
|
type Info struct {
|
||||||
Hostname string `json:"h" cbor:"0,keyasint"`
|
Hostname string `json:"h" cbor:"0,keyasint"`
|
||||||
KernelVersion string `json:"k,omitempty" cbor:"1,keyasint,omitempty"`
|
KernelVersion string `json:"k,omitempty" cbor:"1,keyasint,omitempty"`
|
||||||
Cores int `json:"c" cbor:"2,keyasint"`
|
Cores int `json:"c" cbor:"2,keyasint"`
|
||||||
Threads int `json:"t,omitempty" cbor:"3,keyasint,omitempty"`
|
Threads int `json:"t,omitempty" cbor:"3,keyasint,omitempty"`
|
||||||
CpuModel string `json:"m" cbor:"4,keyasint"`
|
CpuModel string `json:"m" cbor:"4,keyasint"`
|
||||||
Uptime uint64 `json:"u" cbor:"5,keyasint"`
|
Uptime uint64 `json:"u" cbor:"5,keyasint"`
|
||||||
Cpu float64 `json:"cpu" cbor:"6,keyasint"`
|
Cpu float64 `json:"cpu" cbor:"6,keyasint"`
|
||||||
MemPct float64 `json:"mp" cbor:"7,keyasint"`
|
MemPct float64 `json:"mp" cbor:"7,keyasint"`
|
||||||
DiskPct float64 `json:"dp" cbor:"8,keyasint"`
|
DiskPct float64 `json:"dp" cbor:"8,keyasint"`
|
||||||
NetworkSent float64 `json:"ns" cbor:"9,keyasint"` // Per-interface total (MB/s)
|
Bandwidth float64 `json:"b" cbor:"9,keyasint"`
|
||||||
NetworkRecv float64 `json:"nr" cbor:"10,keyasint"` // Per-interface total (MB/s)
|
AgentVersion string `json:"v" cbor:"10,keyasint"`
|
||||||
AgentVersion string `json:"v" cbor:"11,keyasint"`
|
Podman bool `json:"p,omitempty" cbor:"11,keyasint,omitempty"`
|
||||||
Podman bool `json:"p,omitempty" cbor:"12,keyasint,omitempty"`
|
GpuPct float64 `json:"g,omitempty" cbor:"12,keyasint,omitempty"`
|
||||||
GpuPct float64 `json:"g,omitempty" cbor:"13,keyasint,omitempty"`
|
DashboardTemp float64 `json:"dt,omitempty" cbor:"13,keyasint,omitempty"`
|
||||||
DashboardTemp float64 `json:"dt,omitempty" cbor:"14,keyasint,omitempty"`
|
Os Os `json:"os" cbor:"14,keyasint"`
|
||||||
Os Os `json:"os" cbor:"15,keyasint"`
|
LoadAvg1 float64 `json:"l1,omitempty" cbor:"15,keyasint,omitempty"`
|
||||||
LoadAvg1 float64 `json:"l1,omitempty" cbor:"16,keyasint,omitempty"`
|
LoadAvg5 float64 `json:"l5,omitempty" cbor:"16,keyasint,omitempty"`
|
||||||
LoadAvg5 float64 `json:"l5,omitempty" cbor:"17,keyasint,omitempty"`
|
LoadAvg15 float64 `json:"l15,omitempty" cbor:"17,keyasint,omitempty"`
|
||||||
LoadAvg15 float64 `json:"l15,omitempty" cbor:"18,keyasint,omitempty"`
|
BandwidthBytes uint64 `json:"bb" cbor:"18,keyasint"`
|
||||||
LoadAvg [3]float64 `json:"la,omitempty" cbor:"19,keyasint"` // [1min, 5min, 15min]
|
// TODO: remove load fields in future release in favor of load avg array
|
||||||
|
LoadAvg [3]float64 `json:"la,omitempty" cbor:"19,keyasint"`
|
||||||
}
|
}
|
||||||
|
|
||||||
// Final data structure to return to the hub
|
// Final data structure to return to the hub
|
||||||
|
|||||||
@@ -206,51 +206,15 @@ func (rm *RecordManager) AverageSystemStats(db dbx.Builder, records RecordIds) *
|
|||||||
sum.DiskPct += stats.DiskPct
|
sum.DiskPct += stats.DiskPct
|
||||||
sum.DiskReadPs += stats.DiskReadPs
|
sum.DiskReadPs += stats.DiskReadPs
|
||||||
sum.DiskWritePs += stats.DiskWritePs
|
sum.DiskWritePs += stats.DiskWritePs
|
||||||
sum.LoadAvg1 += stats.LoadAvg1
|
|
||||||
sum.LoadAvg5 += stats.LoadAvg5
|
|
||||||
sum.LoadAvg15 += stats.LoadAvg15
|
|
||||||
sum.NetworkSent += stats.NetworkSent
|
sum.NetworkSent += stats.NetworkSent
|
||||||
sum.NetworkRecv += stats.NetworkRecv
|
sum.NetworkRecv += stats.NetworkRecv
|
||||||
sum.LoadAvg[0] += stats.LoadAvg[0]
|
sum.LoadAvg[0] += stats.LoadAvg[0]
|
||||||
sum.LoadAvg[1] += stats.LoadAvg[1]
|
sum.LoadAvg[1] += stats.LoadAvg[1]
|
||||||
sum.LoadAvg[2] += stats.LoadAvg[2]
|
sum.LoadAvg[2] += stats.LoadAvg[2]
|
||||||
|
sum.Bandwidth[0] += stats.Bandwidth[0]
|
||||||
|
sum.Bandwidth[1] += stats.Bandwidth[1]
|
||||||
batterySum += int(stats.Battery[0])
|
batterySum += int(stats.Battery[0])
|
||||||
sum.Battery[1] = stats.Battery[1]
|
sum.Battery[1] = stats.Battery[1]
|
||||||
|
|
||||||
if stats.NetworkInterfaces != nil {
|
|
||||||
if sum.NetworkInterfaces == nil {
|
|
||||||
sum.NetworkInterfaces = make(map[string]system.NetworkInterfaceStats, len(stats.NetworkInterfaces))
|
|
||||||
}
|
|
||||||
for key, value := range stats.NetworkInterfaces {
|
|
||||||
if _, ok := sum.NetworkInterfaces[key]; !ok {
|
|
||||||
sum.NetworkInterfaces[key] = system.NetworkInterfaceStats{}
|
|
||||||
}
|
|
||||||
ni := sum.NetworkInterfaces[key]
|
|
||||||
ni.NetworkSent += value.NetworkSent
|
|
||||||
ni.NetworkRecv += value.NetworkRecv
|
|
||||||
ni.MaxNetworkSent += value.MaxNetworkSent
|
|
||||||
ni.MaxNetworkRecv += value.MaxNetworkRecv
|
|
||||||
// For cumulative totals, use the maximum value (most recent)
|
|
||||||
if value.TotalBytesSent > ni.TotalBytesSent {
|
|
||||||
ni.TotalBytesSent = value.TotalBytesSent
|
|
||||||
}
|
|
||||||
if value.TotalBytesRecv > ni.TotalBytesRecv {
|
|
||||||
ni.TotalBytesRecv = value.TotalBytesRecv
|
|
||||||
}
|
|
||||||
sum.NetworkInterfaces[key] = ni
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Handle network connection stats - use the latest values (most recent sample)
|
|
||||||
if stats.Nets != nil {
|
|
||||||
if sum.Nets == nil {
|
|
||||||
sum.Nets = make(map[string]float64)
|
|
||||||
}
|
|
||||||
for key, value := range stats.Nets {
|
|
||||||
sum.Nets[key] = value
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Set peak values
|
// Set peak values
|
||||||
sum.MaxCpu = max(sum.MaxCpu, stats.MaxCpu, stats.Cpu)
|
sum.MaxCpu = max(sum.MaxCpu, stats.MaxCpu, stats.Cpu)
|
||||||
sum.MaxMem = max(sum.MaxMem, stats.MaxMem, stats.MemUsed)
|
sum.MaxMem = max(sum.MaxMem, stats.MaxMem, stats.MemUsed)
|
||||||
@@ -258,6 +222,8 @@ func (rm *RecordManager) AverageSystemStats(db dbx.Builder, records RecordIds) *
|
|||||||
sum.MaxNetworkRecv = max(sum.MaxNetworkRecv, stats.MaxNetworkRecv, stats.NetworkRecv)
|
sum.MaxNetworkRecv = max(sum.MaxNetworkRecv, stats.MaxNetworkRecv, stats.NetworkRecv)
|
||||||
sum.MaxDiskReadPs = max(sum.MaxDiskReadPs, stats.MaxDiskReadPs, stats.DiskReadPs)
|
sum.MaxDiskReadPs = max(sum.MaxDiskReadPs, stats.MaxDiskReadPs, stats.DiskReadPs)
|
||||||
sum.MaxDiskWritePs = max(sum.MaxDiskWritePs, stats.MaxDiskWritePs, stats.DiskWritePs)
|
sum.MaxDiskWritePs = max(sum.MaxDiskWritePs, stats.MaxDiskWritePs, stats.DiskWritePs)
|
||||||
|
sum.MaxBandwidth[0] = max(sum.MaxBandwidth[0], stats.MaxBandwidth[0], stats.Bandwidth[0])
|
||||||
|
sum.MaxBandwidth[1] = max(sum.MaxBandwidth[1], stats.MaxBandwidth[1], stats.Bandwidth[1])
|
||||||
|
|
||||||
// Accumulate temperatures
|
// Accumulate temperatures
|
||||||
if stats.Temperatures != nil {
|
if stats.Temperatures != nil {
|
||||||
@@ -325,26 +291,14 @@ func (rm *RecordManager) AverageSystemStats(db dbx.Builder, records RecordIds) *
|
|||||||
sum.DiskPct = twoDecimals(sum.DiskPct / count)
|
sum.DiskPct = twoDecimals(sum.DiskPct / count)
|
||||||
sum.DiskReadPs = twoDecimals(sum.DiskReadPs / count)
|
sum.DiskReadPs = twoDecimals(sum.DiskReadPs / count)
|
||||||
sum.DiskWritePs = twoDecimals(sum.DiskWritePs / count)
|
sum.DiskWritePs = twoDecimals(sum.DiskWritePs / count)
|
||||||
sum.LoadAvg1 = twoDecimals(sum.LoadAvg1 / count)
|
|
||||||
sum.LoadAvg5 = twoDecimals(sum.LoadAvg5 / count)
|
|
||||||
sum.LoadAvg15 = twoDecimals(sum.LoadAvg15 / count)
|
|
||||||
sum.NetworkSent = twoDecimals(sum.NetworkSent / count)
|
sum.NetworkSent = twoDecimals(sum.NetworkSent / count)
|
||||||
sum.NetworkRecv = twoDecimals(sum.NetworkRecv / count)
|
sum.NetworkRecv = twoDecimals(sum.NetworkRecv / count)
|
||||||
sum.LoadAvg[0] = twoDecimals(sum.LoadAvg[0] / count)
|
sum.LoadAvg[0] = twoDecimals(sum.LoadAvg[0] / count)
|
||||||
sum.LoadAvg[1] = twoDecimals(sum.LoadAvg[1] / count)
|
sum.LoadAvg[1] = twoDecimals(sum.LoadAvg[1] / count)
|
||||||
sum.LoadAvg[2] = twoDecimals(sum.LoadAvg[2] / count)
|
sum.LoadAvg[2] = twoDecimals(sum.LoadAvg[2] / count)
|
||||||
|
sum.Bandwidth[0] = sum.Bandwidth[0] / uint64(count)
|
||||||
|
sum.Bandwidth[1] = sum.Bandwidth[1] / uint64(count)
|
||||||
sum.Battery[0] = uint8(batterySum / int(count))
|
sum.Battery[0] = uint8(batterySum / int(count))
|
||||||
|
|
||||||
if sum.NetworkInterfaces != nil {
|
|
||||||
for key := range sum.NetworkInterfaces {
|
|
||||||
ni := sum.NetworkInterfaces[key]
|
|
||||||
ni.NetworkSent = twoDecimals(ni.NetworkSent / count)
|
|
||||||
ni.NetworkRecv = twoDecimals(ni.NetworkRecv / count)
|
|
||||||
ni.MaxNetworkSent = twoDecimals(max(ni.MaxNetworkSent, ni.NetworkSent))
|
|
||||||
ni.MaxNetworkRecv = twoDecimals(max(ni.MaxNetworkRecv, ni.NetworkRecv))
|
|
||||||
sum.NetworkInterfaces[key] = ni
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// Average temperatures
|
// Average temperatures
|
||||||
if sum.Temperatures != nil && tempCount > 0 {
|
if sum.Temperatures != nil && tempCount > 0 {
|
||||||
for key := range sum.Temperatures {
|
for key := range sum.Temperatures {
|
||||||
@@ -409,15 +363,19 @@ func (rm *RecordManager) AverageContainerStats(db dbx.Builder, records RecordIds
|
|||||||
}
|
}
|
||||||
sums[stat.Name].Cpu += stat.Cpu
|
sums[stat.Name].Cpu += stat.Cpu
|
||||||
sums[stat.Name].Mem += stat.Mem
|
sums[stat.Name].Mem += stat.Mem
|
||||||
|
sums[stat.Name].NetworkSent += stat.NetworkSent
|
||||||
|
sums[stat.Name].NetworkRecv += stat.NetworkRecv
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
result := make([]container.Stats, 0, len(sums))
|
result := make([]container.Stats, 0, len(sums))
|
||||||
for _, value := range sums {
|
for _, value := range sums {
|
||||||
result = append(result, container.Stats{
|
result = append(result, container.Stats{
|
||||||
Name: value.Name,
|
Name: value.Name,
|
||||||
Cpu: twoDecimals(value.Cpu / count),
|
Cpu: twoDecimals(value.Cpu / count),
|
||||||
Mem: twoDecimals(value.Mem / count),
|
Mem: twoDecimals(value.Mem / count),
|
||||||
|
NetworkSent: twoDecimals(value.NetworkSent / count),
|
||||||
|
NetworkRecv: twoDecimals(value.NetworkRecv / count),
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
return result
|
return result
|
||||||
|
|||||||
843
internal/site/package-lock.json
generated
843
internal/site/package-lock.json
generated
File diff suppressed because it is too large
Load Diff
@@ -1,124 +0,0 @@
|
|||||||
import { memo } from "react"
|
|
||||||
import { useLingui } from "@lingui/react/macro"
|
|
||||||
import { Area, AreaChart, CartesianGrid, YAxis } from "recharts"
|
|
||||||
import {
|
|
||||||
ChartContainer,
|
|
||||||
ChartTooltip,
|
|
||||||
ChartTooltipContent,
|
|
||||||
ChartLegend,
|
|
||||||
ChartLegendContent,
|
|
||||||
xAxis,
|
|
||||||
} from "@/components/ui/chart"
|
|
||||||
import { cn, formatShortDate, chartMargin } from "@/lib/utils"
|
|
||||||
import { ChartData } from "@/types"
|
|
||||||
import { useYAxisWidth } from "./hooks"
|
|
||||||
|
|
||||||
export default memo(function ConnectionChart({ chartData }: { chartData: ChartData }) {
|
|
||||||
const { yAxisWidth, updateYAxisWidth } = useYAxisWidth()
|
|
||||||
const { t } = useLingui()
|
|
||||||
|
|
||||||
if (chartData.systemStats.length === 0) {
|
|
||||||
return null
|
|
||||||
}
|
|
||||||
|
|
||||||
const dataKeys = [
|
|
||||||
{
|
|
||||||
name: t`IPv4 Established`,
|
|
||||||
dataKey: "stats.nets.conn_established",
|
|
||||||
color: "hsl(220, 70%, 50%)", // Blue
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: t`IPv4 Listen`,
|
|
||||||
dataKey: "stats.nets.conn_listen",
|
|
||||||
color: "hsl(142, 70%, 45%)", // Green
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: t`IPv4 Time Wait`,
|
|
||||||
dataKey: "stats.nets.conn_timewait",
|
|
||||||
color: "hsl(48, 96%, 53%)", // Yellow
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: t`IPv4 Close Wait`,
|
|
||||||
dataKey: "stats.nets.conn_closewait",
|
|
||||||
color: "hsl(271, 81%, 56%)", // Purple
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: t`IPv4 Syn Recv`,
|
|
||||||
dataKey: "stats.nets.conn_synrecv",
|
|
||||||
color: "hsl(9, 78%, 56%)", // Red
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: t`IPv6 Established`,
|
|
||||||
dataKey: "stats.nets.conn6_established",
|
|
||||||
color: "hsl(220, 70%, 65%)", // Light Blue
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: t`IPv6 Listen`,
|
|
||||||
dataKey: "stats.nets.conn6_listen",
|
|
||||||
color: "hsl(142, 70%, 60%)", // Light Green
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: t`IPv6 Time Wait`,
|
|
||||||
dataKey: "stats.nets.conn6_timewait",
|
|
||||||
color: "hsl(48, 96%, 68%)", // Light Yellow
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: t`IPv6 Close Wait`,
|
|
||||||
dataKey: "stats.nets.conn6_closewait",
|
|
||||||
color: "hsl(271, 81%, 71%)", // Light Purple
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: t`IPv6 Syn Recv`,
|
|
||||||
dataKey: "stats.nets.conn6_synrecv",
|
|
||||||
color: "hsl(9, 78%, 71%)", // Light Red
|
|
||||||
},
|
|
||||||
]
|
|
||||||
|
|
||||||
return (
|
|
||||||
<div>
|
|
||||||
<ChartContainer
|
|
||||||
className={cn("h-full w-full absolute aspect-auto bg-card opacity-0 transition-opacity", {
|
|
||||||
"opacity-100": yAxisWidth,
|
|
||||||
})}
|
|
||||||
>
|
|
||||||
<AreaChart accessibilityLayer data={chartData.systemStats} margin={chartMargin}>
|
|
||||||
<CartesianGrid vertical={false} />
|
|
||||||
<YAxis
|
|
||||||
direction="ltr"
|
|
||||||
orientation={chartData.orientation}
|
|
||||||
className="tracking-tighter"
|
|
||||||
width={yAxisWidth}
|
|
||||||
tickFormatter={(value) => updateYAxisWidth(value.toString())}
|
|
||||||
tickLine={false}
|
|
||||||
axisLine={false}
|
|
||||||
/>
|
|
||||||
{xAxis(chartData)}
|
|
||||||
<ChartTooltip
|
|
||||||
animationEasing="ease-out"
|
|
||||||
animationDuration={150}
|
|
||||||
content={
|
|
||||||
<ChartTooltipContent
|
|
||||||
labelFormatter={(_, data) => formatShortDate(data[0].payload.created)}
|
|
||||||
contentFormatter={({ value }) => value.toString()}
|
|
||||||
/>
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
<ChartLegend content={<ChartLegendContent />} />
|
|
||||||
{dataKeys.map((key, i) => (
|
|
||||||
<Area
|
|
||||||
key={i}
|
|
||||||
dataKey={key.dataKey}
|
|
||||||
name={key.name}
|
|
||||||
type="monotoneX"
|
|
||||||
fill={key.color}
|
|
||||||
fillOpacity={0.3}
|
|
||||||
stroke={key.color}
|
|
||||||
strokeOpacity={1}
|
|
||||||
isAnimationActive={false}
|
|
||||||
/>
|
|
||||||
))}
|
|
||||||
</AreaChart>
|
|
||||||
</ChartContainer>
|
|
||||||
</div>
|
|
||||||
)
|
|
||||||
})
|
|
||||||
@@ -1,164 +0,0 @@
|
|||||||
import { memo, useMemo } from "react"
|
|
||||||
import { useLingui } from "@lingui/react/macro"
|
|
||||||
import { Area, AreaChart, CartesianGrid, YAxis } from "recharts"
|
|
||||||
import {
|
|
||||||
ChartContainer,
|
|
||||||
ChartTooltip,
|
|
||||||
ChartTooltipContent,
|
|
||||||
xAxis,
|
|
||||||
ChartLegend,
|
|
||||||
ChartLegendContent,
|
|
||||||
} from "@/components/ui/chart"
|
|
||||||
import { cn, formatShortDate, chartMargin, formatBytes, toFixedFloat, decimalString } from "@/lib/utils"
|
|
||||||
import { ChartData } from "@/types"
|
|
||||||
import { useStore } from "@nanostores/react"
|
|
||||||
import { $networkInterfaceFilter, $userSettings } from "@/lib/stores"
|
|
||||||
import { Unit } from "@/lib/enums"
|
|
||||||
import { useYAxisWidth } from "./hooks"
|
|
||||||
|
|
||||||
const getNestedValue = (path: string, max = false, data: any): number | null => {
|
|
||||||
// path format is like "eth0.ns" or "eth0.nr"
|
|
||||||
// need to access data.stats.ni[interface][property]
|
|
||||||
const parts = path.split(".")
|
|
||||||
if (parts.length !== 2) return null
|
|
||||||
|
|
||||||
const [interfaceName, property] = parts
|
|
||||||
const propertyKey = property + (max ? "m" : "")
|
|
||||||
|
|
||||||
return data?.stats?.ni?.[interfaceName]?.[propertyKey] ?? null
|
|
||||||
}
|
|
||||||
|
|
||||||
export default memo(function NetworkInterfaceChart({
|
|
||||||
chartData,
|
|
||||||
maxToggled = false,
|
|
||||||
max,
|
|
||||||
}: {
|
|
||||||
chartData: ChartData
|
|
||||||
maxToggled?: boolean
|
|
||||||
max?: number
|
|
||||||
}) {
|
|
||||||
const { yAxisWidth, updateYAxisWidth } = useYAxisWidth()
|
|
||||||
const { i18n } = useLingui()
|
|
||||||
const networkInterfaceFilter = useStore($networkInterfaceFilter)
|
|
||||||
const userSettings = useStore($userSettings)
|
|
||||||
|
|
||||||
const { chartTime } = chartData
|
|
||||||
const showMax = chartTime !== "1h" && maxToggled
|
|
||||||
|
|
||||||
// Get network interface names from the latest stats
|
|
||||||
const networkInterfaces = useMemo(() => {
|
|
||||||
if (chartData.systemStats.length === 0) return []
|
|
||||||
const latestStats = chartData.systemStats[chartData.systemStats.length - 1]
|
|
||||||
const allInterfaces = Object.keys(latestStats.stats.ni || {})
|
|
||||||
|
|
||||||
// Filter interfaces based on filter value
|
|
||||||
if (networkInterfaceFilter) {
|
|
||||||
return allInterfaces.filter((iface) => iface.toLowerCase().includes(networkInterfaceFilter.toLowerCase()))
|
|
||||||
}
|
|
||||||
|
|
||||||
return allInterfaces
|
|
||||||
}, [chartData.systemStats, networkInterfaceFilter])
|
|
||||||
|
|
||||||
const dataKeys = useMemo(() => {
|
|
||||||
// Generate colors for each interface - each interface gets a unique hue
|
|
||||||
// and sent/received use different shades of that hue
|
|
||||||
const interfaceColors = networkInterfaces.map((iface, index) => {
|
|
||||||
const hue = ((index * 360) / Math.max(networkInterfaces.length, 1)) % 360
|
|
||||||
return {
|
|
||||||
interface: iface,
|
|
||||||
sentColor: `hsl(${hue}, 70%, 45%)`, // Darker shade for sent
|
|
||||||
receivedColor: `hsl(${hue}, 70%, 65%)`, // Lighter shade for received
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
return interfaceColors.flatMap(({ interface: iface, sentColor, receivedColor }) => [
|
|
||||||
{
|
|
||||||
name: `${iface} Sent`,
|
|
||||||
dataKey: `${iface}.ns`,
|
|
||||||
color: sentColor,
|
|
||||||
type: "sent" as const,
|
|
||||||
interface: iface,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: `${iface} Received`,
|
|
||||||
dataKey: `${iface}.nr`,
|
|
||||||
color: receivedColor,
|
|
||||||
type: "received" as const,
|
|
||||||
interface: iface,
|
|
||||||
},
|
|
||||||
])
|
|
||||||
}, [networkInterfaces, i18n.locale])
|
|
||||||
|
|
||||||
const colors = dataKeys.map((key) => key.name)
|
|
||||||
|
|
||||||
return (
|
|
||||||
<div>
|
|
||||||
<ChartContainer
|
|
||||||
className={cn("h-full w-full absolute aspect-auto bg-card opacity-0 transition-opacity", {
|
|
||||||
"opacity-100": yAxisWidth,
|
|
||||||
})}
|
|
||||||
>
|
|
||||||
<AreaChart accessibilityLayer data={chartData.systemStats} margin={chartMargin}>
|
|
||||||
<CartesianGrid vertical={false} />
|
|
||||||
<YAxis
|
|
||||||
direction="ltr"
|
|
||||||
orientation={chartData.orientation}
|
|
||||||
className="tracking-tighter"
|
|
||||||
width={yAxisWidth}
|
|
||||||
tickFormatter={(value) => {
|
|
||||||
const { value: formattedValue, unit } = formatBytes(value, true, userSettings.unitNet ?? Unit.Bits, true)
|
|
||||||
const rounded = toFixedFloat(formattedValue, formattedValue >= 10 ? 1 : 2)
|
|
||||||
return updateYAxisWidth(`${rounded} ${unit}`)
|
|
||||||
}}
|
|
||||||
tickLine={false}
|
|
||||||
axisLine={false}
|
|
||||||
/>
|
|
||||||
{xAxis(chartData)}
|
|
||||||
<ChartTooltip
|
|
||||||
animationEasing="ease-out"
|
|
||||||
animationDuration={150}
|
|
||||||
content={
|
|
||||||
<ChartTooltipContent
|
|
||||||
labelFormatter={(_: any, data: any) => formatShortDate(data[0].payload.created)}
|
|
||||||
contentFormatter={({ value }: any) => {
|
|
||||||
const { value: formattedValue, unit } = formatBytes(
|
|
||||||
value,
|
|
||||||
true,
|
|
||||||
userSettings.unitNet ?? Unit.Bits,
|
|
||||||
true
|
|
||||||
)
|
|
||||||
return (
|
|
||||||
<span className="flex">
|
|
||||||
{decimalString(formattedValue, formattedValue >= 10 ? 1 : 2)} {unit}
|
|
||||||
</span>
|
|
||||||
)
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
{dataKeys.map((key, i) => {
|
|
||||||
const filtered =
|
|
||||||
networkInterfaceFilter && !key.interface.toLowerCase().includes(networkInterfaceFilter.toLowerCase())
|
|
||||||
let fillOpacity = filtered ? 0.05 : 0.4
|
|
||||||
let strokeOpacity = filtered ? 0.1 : 1
|
|
||||||
return (
|
|
||||||
<Area
|
|
||||||
key={i}
|
|
||||||
dataKey={getNestedValue.bind(null, key.dataKey, showMax)}
|
|
||||||
name={key.name}
|
|
||||||
type="monotoneX"
|
|
||||||
fill={key.color}
|
|
||||||
fillOpacity={fillOpacity}
|
|
||||||
stroke={key.color}
|
|
||||||
strokeOpacity={strokeOpacity}
|
|
||||||
activeDot={{ opacity: filtered ? 0 : 1 }}
|
|
||||||
isAnimationActive={false}
|
|
||||||
/>
|
|
||||||
)
|
|
||||||
})}
|
|
||||||
{colors.length < 12 && <ChartLegend content={<ChartLegendContent />} />}
|
|
||||||
</AreaChart>
|
|
||||||
</ChartContainer>
|
|
||||||
</div>
|
|
||||||
)
|
|
||||||
})
|
|
||||||
@@ -1,159 +0,0 @@
|
|||||||
import { memo, useMemo } from "react"
|
|
||||||
import { useLingui } from "@lingui/react/macro"
|
|
||||||
import { Area, AreaChart, CartesianGrid, YAxis } from "recharts"
|
|
||||||
import {
|
|
||||||
ChartContainer,
|
|
||||||
ChartTooltip,
|
|
||||||
ChartTooltipContent,
|
|
||||||
ChartLegend,
|
|
||||||
ChartLegendContent,
|
|
||||||
xAxis,
|
|
||||||
} from "@/components/ui/chart"
|
|
||||||
import { cn, formatShortDate, chartMargin, formatBytes, toFixedFloat, decimalString } from "@/lib/utils"
|
|
||||||
import { ChartData } from "@/types"
|
|
||||||
import { useStore } from "@nanostores/react"
|
|
||||||
import { $userSettings } from "@/lib/stores"
|
|
||||||
import { Unit } from "@/lib/enums"
|
|
||||||
import { useYAxisWidth } from "./hooks"
|
|
||||||
|
|
||||||
const getPerInterfaceBandwidth = (data: any): Record<string, { sent: number; recv: number }> | null => {
|
|
||||||
const networkInterfaces = data?.stats?.ni
|
|
||||||
if (!networkInterfaces) {
|
|
||||||
return null
|
|
||||||
}
|
|
||||||
|
|
||||||
const interfaceData: Record<string, { sent: number; recv: number }> = {}
|
|
||||||
let hasData = false
|
|
||||||
|
|
||||||
Object.entries(networkInterfaces).forEach(([name, iface]: [string, any]) => {
|
|
||||||
if (iface.tbs || iface.tbr) {
|
|
||||||
interfaceData[name] = {
|
|
||||||
sent: iface.tbs || 0,
|
|
||||||
recv: iface.tbr || 0,
|
|
||||||
}
|
|
||||||
hasData = true
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
return hasData ? interfaceData : null
|
|
||||||
}
|
|
||||||
|
|
||||||
export default memo(function TotalBandwidthChart({ chartData }: { chartData: ChartData }) {
|
|
||||||
const { yAxisWidth, updateYAxisWidth } = useYAxisWidth()
|
|
||||||
const { i18n } = useLingui()
|
|
||||||
const userSettings = useStore($userSettings)
|
|
||||||
|
|
||||||
// Transform data to include per-interface bandwidth
|
|
||||||
const { transformedData, interfaceNames } = useMemo(() => {
|
|
||||||
const allInterfaces = new Set<string>()
|
|
||||||
|
|
||||||
// First pass: collect all interface names
|
|
||||||
chartData.systemStats.forEach((dataPoint) => {
|
|
||||||
const interfaceData = getPerInterfaceBandwidth(dataPoint)
|
|
||||||
if (interfaceData) {
|
|
||||||
Object.keys(interfaceData).forEach((name) => allInterfaces.add(name))
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
const interfaceNames = Array.from(allInterfaces).sort()
|
|
||||||
|
|
||||||
// Second pass: transform data with per-interface values
|
|
||||||
const transformedData = chartData.systemStats.map((dataPoint) => {
|
|
||||||
const interfaceData = getPerInterfaceBandwidth(dataPoint)
|
|
||||||
const result: any = { ...dataPoint }
|
|
||||||
|
|
||||||
interfaceNames.forEach((interfaceName) => {
|
|
||||||
const data = interfaceData?.[interfaceName]
|
|
||||||
result[`${interfaceName}_sent`] = data?.sent || 0
|
|
||||||
result[`${interfaceName}_recv`] = data?.recv || 0
|
|
||||||
})
|
|
||||||
|
|
||||||
return result
|
|
||||||
})
|
|
||||||
|
|
||||||
return { transformedData, interfaceNames }
|
|
||||||
}, [chartData.systemStats])
|
|
||||||
|
|
||||||
// Generate dynamic data keys for each interface using same color scheme as NetworkInterfaceChart
|
|
||||||
const dataKeys = useMemo(() => {
|
|
||||||
const keys: Array<{ name: string; dataKey: string; color: string }> = []
|
|
||||||
|
|
||||||
interfaceNames.forEach((interfaceName, index) => {
|
|
||||||
// Use the same color calculation as NetworkInterfaceChart
|
|
||||||
const hue = ((index * 360) / Math.max(interfaceNames.length, 1)) % 360
|
|
||||||
|
|
||||||
keys.push({
|
|
||||||
name: `${interfaceName} Sent`,
|
|
||||||
dataKey: `${interfaceName}_sent`,
|
|
||||||
color: `hsl(${hue}, 70%, 45%)`, // Darker shade for sent (same as NetworkInterfaceChart)
|
|
||||||
})
|
|
||||||
|
|
||||||
keys.push({
|
|
||||||
name: `${interfaceName} Received`,
|
|
||||||
dataKey: `${interfaceName}_recv`,
|
|
||||||
color: `hsl(${hue}, 70%, 65%)`, // Lighter shade for received (same as NetworkInterfaceChart)
|
|
||||||
})
|
|
||||||
})
|
|
||||||
|
|
||||||
return keys
|
|
||||||
}, [interfaceNames])
|
|
||||||
|
|
||||||
return (
|
|
||||||
<div>
|
|
||||||
<ChartContainer
|
|
||||||
className={cn("h-full w-full absolute aspect-auto bg-card opacity-0 transition-opacity", {
|
|
||||||
"opacity-100": yAxisWidth,
|
|
||||||
})}
|
|
||||||
>
|
|
||||||
<AreaChart accessibilityLayer data={transformedData} margin={chartMargin}>
|
|
||||||
<CartesianGrid vertical={false} />
|
|
||||||
<YAxis
|
|
||||||
direction="ltr"
|
|
||||||
orientation={chartData.orientation}
|
|
||||||
className="tracking-tighter"
|
|
||||||
width={yAxisWidth}
|
|
||||||
tickFormatter={(value) => {
|
|
||||||
const { value: formattedValue, unit } = formatBytes(value, false, userSettings.unitNet ?? Unit.Bytes)
|
|
||||||
const rounded = toFixedFloat(formattedValue, formattedValue >= 10 ? 1 : 2)
|
|
||||||
return updateYAxisWidth(`${rounded} ${unit}`)
|
|
||||||
}}
|
|
||||||
tickLine={false}
|
|
||||||
axisLine={false}
|
|
||||||
/>
|
|
||||||
{xAxis(chartData)}
|
|
||||||
<ChartTooltip
|
|
||||||
animationEasing="ease-out"
|
|
||||||
animationDuration={150}
|
|
||||||
content={
|
|
||||||
<ChartTooltipContent
|
|
||||||
labelFormatter={(_: any, data: any) => formatShortDate(data[0].payload.created)}
|
|
||||||
contentFormatter={({ value }: any) => {
|
|
||||||
const { value: formattedValue, unit } = formatBytes(value, false, userSettings.unitNet ?? Unit.Bytes)
|
|
||||||
return (
|
|
||||||
<span className="flex">
|
|
||||||
{decimalString(formattedValue, formattedValue >= 10 ? 1 : 2)} {unit}
|
|
||||||
</span>
|
|
||||||
)
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
<ChartLegend content={<ChartLegendContent />} />
|
|
||||||
{dataKeys.map((key, i) => (
|
|
||||||
<Area
|
|
||||||
key={i}
|
|
||||||
dataKey={key.dataKey}
|
|
||||||
name={key.name}
|
|
||||||
type="monotoneX"
|
|
||||||
fill={key.color}
|
|
||||||
fillOpacity={0.3}
|
|
||||||
stroke={key.color}
|
|
||||||
strokeOpacity={1}
|
|
||||||
isAnimationActive={false}
|
|
||||||
/>
|
|
||||||
))}
|
|
||||||
</AreaChart>
|
|
||||||
</ChartContainer>
|
|
||||||
</div>
|
|
||||||
)
|
|
||||||
})
|
|
||||||
@@ -24,7 +24,6 @@ import {
|
|||||||
$containerFilter,
|
$containerFilter,
|
||||||
$direction,
|
$direction,
|
||||||
$maxValues,
|
$maxValues,
|
||||||
$networkInterfaceFilter,
|
|
||||||
$systems,
|
$systems,
|
||||||
$temperatureFilter,
|
$temperatureFilter,
|
||||||
$userSettings,
|
$userSettings,
|
||||||
@@ -53,9 +52,6 @@ import { Input } from "../ui/input"
|
|||||||
import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from "../ui/select"
|
import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from "../ui/select"
|
||||||
import { Separator } from "../ui/separator"
|
import { Separator } from "../ui/separator"
|
||||||
import { Tooltip, TooltipContent, TooltipProvider, TooltipTrigger } from "../ui/tooltip"
|
import { Tooltip, TooltipContent, TooltipProvider, TooltipTrigger } from "../ui/tooltip"
|
||||||
import ConnectionChart from "../charts/connection-chart"
|
|
||||||
import NetworkInterfaceChart from "../charts/network-interface-chart"
|
|
||||||
import TotalBandwidthChart from "../charts/total-bandwidth-chart"
|
|
||||||
|
|
||||||
type ChartTimeData = {
|
type ChartTimeData = {
|
||||||
time: number
|
time: number
|
||||||
@@ -151,7 +147,6 @@ export default memo(function SystemDetail({ name }: { name: string }) {
|
|||||||
const netCardRef = useRef<HTMLDivElement>(null)
|
const netCardRef = useRef<HTMLDivElement>(null)
|
||||||
const persistChartTime = useRef(false)
|
const persistChartTime = useRef(false)
|
||||||
const [containerFilterBar, setContainerFilterBar] = useState(null as null | JSX.Element)
|
const [containerFilterBar, setContainerFilterBar] = useState(null as null | JSX.Element)
|
||||||
const [networkInterfaceFilterBar, setNetworkInterfaceFilterBar] = useState(null as null | JSX.Element)
|
|
||||||
const [bottomSpacing, setBottomSpacing] = useState(0)
|
const [bottomSpacing, setBottomSpacing] = useState(0)
|
||||||
const [chartLoading, setChartLoading] = useState(true)
|
const [chartLoading, setChartLoading] = useState(true)
|
||||||
const isLongerChart = chartTime !== "1h"
|
const isLongerChart = chartTime !== "1h"
|
||||||
@@ -168,9 +163,7 @@ export default memo(function SystemDetail({ name }: { name: string }) {
|
|||||||
setSystemStats([])
|
setSystemStats([])
|
||||||
setContainerData([])
|
setContainerData([])
|
||||||
setContainerFilterBar(null)
|
setContainerFilterBar(null)
|
||||||
setNetworkInterfaceFilterBar(null)
|
|
||||||
$containerFilter.set("")
|
$containerFilter.set("")
|
||||||
$networkInterfaceFilter.set("")
|
|
||||||
}
|
}
|
||||||
}, [name])
|
}, [name])
|
||||||
|
|
||||||
@@ -267,19 +260,6 @@ export default memo(function SystemDetail({ name }: { name: string }) {
|
|||||||
})
|
})
|
||||||
}, [system, chartTime])
|
}, [system, chartTime])
|
||||||
|
|
||||||
// Set up network interface filter bar
|
|
||||||
useEffect(() => {
|
|
||||||
if (systemStats.length > 0) {
|
|
||||||
const latestStats = systemStats[systemStats.length - 1]
|
|
||||||
const networkInterfaces = Object.keys(latestStats.stats.ns || {})
|
|
||||||
if (networkInterfaces.length > 0) {
|
|
||||||
!networkInterfaceFilterBar && setNetworkInterfaceFilterBar(<FilterBar store={$networkInterfaceFilter} />)
|
|
||||||
} else if (networkInterfaceFilterBar) {
|
|
||||||
setNetworkInterfaceFilterBar(null)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}, [systemStats, networkInterfaceFilterBar])
|
|
||||||
|
|
||||||
// values for system info bar
|
// values for system info bar
|
||||||
const systemInfo = useMemo(() => {
|
const systemInfo = useMemo(() => {
|
||||||
if (!system.info) {
|
if (!system.info) {
|
||||||
@@ -409,7 +389,6 @@ export default memo(function SystemDetail({ name }: { name: string }) {
|
|||||||
const lastGpuVals = Object.values(systemStats.at(-1)?.stats.g ?? {})
|
const lastGpuVals = Object.values(systemStats.at(-1)?.stats.g ?? {})
|
||||||
const hasGpuData = lastGpuVals.length > 0
|
const hasGpuData = lastGpuVals.length > 0
|
||||||
const hasGpuPowerData = lastGpuVals.some((gpu) => gpu.p !== undefined)
|
const hasGpuPowerData = lastGpuVals.some((gpu) => gpu.p !== undefined)
|
||||||
const latestNetworkStats = systemStats.at(-1)?.stats.ni
|
|
||||||
|
|
||||||
let translatedStatus: string = system.status
|
let translatedStatus: string = system.status
|
||||||
if (system.status === SystemStatus.Up) {
|
if (system.status === SystemStatus.Up) {
|
||||||
@@ -576,7 +555,7 @@ export default memo(function SystemDetail({ name }: { name: string }) {
|
|||||||
empty={dataEmpty}
|
empty={dataEmpty}
|
||||||
grid={grid}
|
grid={grid}
|
||||||
title={t`Disk I/O`}
|
title={t`Disk I/O`}
|
||||||
description={t`Disk read and write throughput`}
|
description={t`Throughput of root filesystem`}
|
||||||
cornerEl={maxValSelect}
|
cornerEl={maxValSelect}
|
||||||
>
|
>
|
||||||
<AreaChartDefault
|
<AreaChartDefault
|
||||||
@@ -607,44 +586,51 @@ export default memo(function SystemDetail({ name }: { name: string }) {
|
|||||||
/>
|
/>
|
||||||
</ChartCard>
|
</ChartCard>
|
||||||
|
|
||||||
{/* Network interface charts */}
|
<ChartCard
|
||||||
{Object.keys(latestNetworkStats ?? {}).length > 0 && (
|
empty={dataEmpty}
|
||||||
<ChartCard
|
grid={grid}
|
||||||
empty={dataEmpty}
|
title={t`Bandwidth`}
|
||||||
grid={grid}
|
cornerEl={maxValSelect}
|
||||||
title={t`Network Interfaces`}
|
description={t`Network traffic of public interfaces`}
|
||||||
description={t`Network traffic per interface`}
|
>
|
||||||
cornerEl={networkInterfaceFilterBar}
|
<AreaChartDefault
|
||||||
>
|
chartData={chartData}
|
||||||
{/* @ts-ignore */}
|
maxToggled={maxValues}
|
||||||
<NetworkInterfaceChart chartData={chartData} />
|
dataPoints={[
|
||||||
</ChartCard>
|
{
|
||||||
)}
|
label: t`Sent`,
|
||||||
|
// use bytes if available, otherwise multiply old MB (can remove in future)
|
||||||
{/* Per-Interface Cumulative Bandwidth chart */}
|
dataKey(data) {
|
||||||
{Object.keys(latestNetworkStats ?? {}).length > 0 && (
|
if (showMax) {
|
||||||
<ChartCard
|
return data?.stats?.bm?.[0] ?? (data?.stats?.nsm ?? 0) * 1024 * 1024
|
||||||
empty={dataEmpty}
|
}
|
||||||
grid={grid}
|
return data?.stats?.b?.[0] ?? data?.stats?.ns * 1024 * 1024
|
||||||
title={t`Cumulative Bandwidth`}
|
},
|
||||||
description={t`Total bytes sent and received per network interface since boot`}
|
color: 5,
|
||||||
>
|
opacity: 0.2,
|
||||||
{/* @ts-ignore */}
|
},
|
||||||
<TotalBandwidthChart chartData={chartData} />
|
{
|
||||||
</ChartCard>
|
label: t`Received`,
|
||||||
)}
|
dataKey(data) {
|
||||||
|
if (showMax) {
|
||||||
{/* TCP Connection States chart */}
|
return data?.stats?.bm?.[1] ?? (data?.stats?.nrm ?? 0) * 1024 * 1024
|
||||||
{systemStats.at(-1)?.stats.nets && Object.keys(systemStats.at(-1)?.stats.nets ?? {}).length > 0 && (
|
}
|
||||||
<ChartCard
|
return data?.stats?.b?.[1] ?? data?.stats?.nr * 1024 * 1024
|
||||||
empty={dataEmpty}
|
},
|
||||||
grid={grid}
|
color: 2,
|
||||||
title={t`TCP Connection States`}
|
opacity: 0.2,
|
||||||
description={t`TCP connection states for IPv4 and IPv6`}
|
},
|
||||||
>
|
]}
|
||||||
<ConnectionChart chartData={chartData} />
|
tickFormatter={(val) => {
|
||||||
</ChartCard>
|
const { value, unit } = formatBytes(val, true, userSettings.unitNet, false)
|
||||||
)}
|
return `${toFixedFloat(value, value >= 10 ? 0 : 1)} ${unit}`
|
||||||
|
}}
|
||||||
|
contentFormatter={(data) => {
|
||||||
|
const { value, unit } = formatBytes(data.value, true, userSettings.unitNet, false)
|
||||||
|
return `${decimalString(value, value >= 100 ? 1 : 2)} ${unit}`
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</ChartCard>
|
||||||
|
|
||||||
{containerFilterBar && containerData.length > 0 && (
|
{containerFilterBar && containerData.length > 0 && (
|
||||||
<div
|
<div
|
||||||
|
|||||||
@@ -216,32 +216,22 @@ export default function SystemsTableColumns(viewMode: "table" | "grid"): ColumnD
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
accessorFn: (row) => (row.info.ns || 0) + (row.info.nr || 0),
|
accessorFn: ({ info }) => info.bb || (info.b || 0) * 1024 * 1024,
|
||||||
id: "net",
|
id: "net",
|
||||||
name: () => t`Net`,
|
name: () => t`Net`,
|
||||||
size: 0,
|
size: 0,
|
||||||
Icon: EthernetIcon,
|
Icon: EthernetIcon,
|
||||||
header: sortableHeader,
|
header: sortableHeader,
|
||||||
sortDescFirst: true,
|
|
||||||
sortingFn: (rowA, rowB) => {
|
|
||||||
const a = (rowA.original.info.ns || 0) + (rowA.original.info.nr || 0)
|
|
||||||
const b = (rowB.original.info.ns || 0) + (rowB.original.info.nr || 0)
|
|
||||||
return a - b
|
|
||||||
},
|
|
||||||
cell(info) {
|
cell(info) {
|
||||||
const system = info.row.original
|
const sys = info.row.original
|
||||||
const sent = system.info.ns || 0
|
|
||||||
const received = system.info.nr || 0
|
|
||||||
const userSettings = useStore($userSettings, { keys: ["unitNet"] })
|
const userSettings = useStore($userSettings, { keys: ["unitNet"] })
|
||||||
if (system.status === SystemStatus.Paused) {
|
if (sys.status === SystemStatus.Paused) {
|
||||||
return null
|
return null
|
||||||
}
|
}
|
||||||
const sentFmt = formatBytes(sent, true, userSettings.unitNet, true)
|
const { value, unit } = formatBytes(info.getValue() as number, true, userSettings.unitNet, false)
|
||||||
const receivedFmt = formatBytes(received, true, userSettings.unitNet, true)
|
|
||||||
return (
|
return (
|
||||||
<span className={cn("tabular-nums whitespace-nowrap", { "ps-1": viewMode === "table" })}>
|
<span className="tabular-nums whitespace-nowrap">
|
||||||
<span className="text-green-600">↑</span> {Math.round(sentFmt.value)} {sentFmt.unit}{" "}
|
{decimalString(value, value >= 100 ? 1 : 2)} {unit}
|
||||||
<span className="text-blue-600">↓</span> {Math.round(receivedFmt.value)} {receivedFmt.unit}
|
|
||||||
</span>
|
</span>
|
||||||
)
|
)
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -55,9 +55,6 @@ listenKeys($userSettings, ["chartTime"], ({ chartTime }) => $chartTime.set(chart
|
|||||||
/** Container chart filter */
|
/** Container chart filter */
|
||||||
export const $containerFilter = atom("")
|
export const $containerFilter = atom("")
|
||||||
|
|
||||||
/** Network interface chart filter */
|
|
||||||
export const $networkInterfaceFilter = atom("")
|
|
||||||
|
|
||||||
/** Temperature chart filter */
|
/** Temperature chart filter */
|
||||||
export const $temperatureFilter = atom("")
|
export const $temperatureFilter = atom("")
|
||||||
|
|
||||||
|
|||||||
@@ -283,22 +283,6 @@ export const getHubURL = () => BESZEL?.HUB_URL || window.location.origin
|
|||||||
/** Map of system IDs to their corresponding tokens (used to avoid fetching in add-system dialog) */
|
/** Map of system IDs to their corresponding tokens (used to avoid fetching in add-system dialog) */
|
||||||
export const tokenMap = new Map<SystemRecord["id"], FingerprintRecord["token"]>()
|
export const tokenMap = new Map<SystemRecord["id"], FingerprintRecord["token"]>()
|
||||||
|
|
||||||
/**
|
|
||||||
* Formats a network speed value (in MB/s) to the most readable unit (B/s, KB/s, MB/s, GB/s, TB/s).
|
|
||||||
* @param valueMBps The value in MB/s
|
|
||||||
* @returns A string with the value and the appropriate unit
|
|
||||||
*/
|
|
||||||
export function formatSpeed(valueMBps: number): string {
|
|
||||||
const bitsPerSec = valueMBps * 8_000_000
|
|
||||||
if (bitsPerSec >= 1_000_000_000) {
|
|
||||||
return (bitsPerSec / 1_000_000_000).toFixed(2) + ' Gbit/s'
|
|
||||||
} else if (bitsPerSec >= 1_000_000) {
|
|
||||||
return (bitsPerSec / 1_000_000).toFixed(2) + ' Mbit/s'
|
|
||||||
} else {
|
|
||||||
return (bitsPerSec / 1_000).toFixed(2) + ' kbit/s'
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Calculate duration between two dates and format as human-readable string */
|
/** Calculate duration between two dates and format as human-readable string */
|
||||||
export function formatDuration(
|
export function formatDuration(
|
||||||
createdDate: string | null | undefined,
|
createdDate: string | null | undefined,
|
||||||
|
|||||||
25
internal/site/src/types.d.ts
vendored
25
internal/site/src/types.d.ts
vendored
@@ -75,25 +75,6 @@ export interface SystemInfo {
|
|||||||
dt?: number
|
dt?: number
|
||||||
/** operating system */
|
/** operating system */
|
||||||
os?: Os
|
os?: Os
|
||||||
/** network sent (mb) */
|
|
||||||
ns?: number
|
|
||||||
/** network received (mb) */
|
|
||||||
nr?: number
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface NetworkInterfaceStats {
|
|
||||||
/** network sent (mb) */
|
|
||||||
ns: number
|
|
||||||
/** network received (mb) */
|
|
||||||
nr: number
|
|
||||||
/** max network sent (mb) */
|
|
||||||
nsm?: number
|
|
||||||
/** max network received (mb) */
|
|
||||||
nrm?: number
|
|
||||||
/** total bytes sent since boot */
|
|
||||||
tbs?: number
|
|
||||||
/** total bytes received since boot */
|
|
||||||
tbr?: number
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface SystemStats {
|
export interface SystemStats {
|
||||||
@@ -150,9 +131,7 @@ export interface SystemStats {
|
|||||||
nsm?: number
|
nsm?: number
|
||||||
/** max network received (mb) */
|
/** max network received (mb) */
|
||||||
nrm?: number
|
nrm?: number
|
||||||
/** per-interface network stats */
|
/** max network sent (bytes) */
|
||||||
ni?: Record<string, NetworkInterfaceStats>
|
|
||||||
/** max bandwidth (bytes) [sent, received] */
|
|
||||||
bm?: [number, number]
|
bm?: [number, number]
|
||||||
/** temperatures */
|
/** temperatures */
|
||||||
t?: Record<string, number>
|
t?: Record<string, number>
|
||||||
@@ -162,8 +141,6 @@ export interface SystemStats {
|
|||||||
g?: Record<string, GPUData>
|
g?: Record<string, GPUData>
|
||||||
/** battery percent and state */
|
/** battery percent and state */
|
||||||
bat?: [number, BatteryState]
|
bat?: [number, BatteryState]
|
||||||
/** network connection statistics */
|
|
||||||
nets?: Record<string, number>
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface GPUData {
|
export interface GPUData {
|
||||||
|
|||||||
Reference in New Issue
Block a user