commit 4fa42f7115b24098233cee13e8fb0c4974d2b778 Author: 树萌芽 <3205788256@qq.com> Date: Sun Dec 14 15:25:31 2025 +0800 初始化提交 diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..425ab26 --- /dev/null +++ b/.gitignore @@ -0,0 +1,47 @@ +# General +.DS_Store +.AppleDouble +.LSOverride +Icon +._* +Thumbs.db +ehthumbs.db +Desktop.ini +*.log +*.tmp +*.bak +*.swp +*.swo +*~ +.idea/ +.vscode/ +*.sublime-project +*.sublime-workspace + +# Backend (Go) +/mengyamonitor-backend/mengyamonitor-backend +/mengyamonitor-backend/mengyamonitor-backend.exe +/mengyamonitor-backend/*.exe +/mengyamonitor-backend/*.out +/mengyamonitor-backend/bin/ +/mengyamonitor-backend/dist/ +/mengyamonitor-backend/vendor/ +/mengyamonitor-backend/.env +/mengyamonitor-backend/debug +/mengyamonitor-backend/__debug_bin + +# Frontend (Node/React/Vite) +/mengyamonitor-frontend/node_modules/ +/mengyamonitor-frontend/dist/ +/mengyamonitor-frontend/build/ +/mengyamonitor-frontend/coverage/ +/mengyamonitor-frontend/.env +/mengyamonitor-frontend/.env.local +/mengyamonitor-frontend/.env.development.local +/mengyamonitor-frontend/.env.test.local +/mengyamonitor-frontend/.env.production.local +/mengyamonitor-frontend/npm-debug.log* +/mengyamonitor-frontend/yarn-debug.log* +/mengyamonitor-frontend/yarn-error.log* +/mengyamonitor-frontend/pnpm-debug.log* +/mengyamonitor-frontend/.eslintcache diff --git a/LICENSE b/LICENSE new file mode 100644 index 0000000..6a8d6e7 --- /dev/null +++ b/LICENSE @@ -0,0 +1,21 @@ +MIT License + +Copyright (c) 2024 萌芽监控面板 + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. diff --git a/README.md b/README.md new file mode 100644 index 0000000..991379d --- /dev/null +++ b/README.md @@ -0,0 +1,156 @@ +# 萌芽监控面板 + +一个简洁高效的 Linux 服务器监控面板,采用前后端分离架构。 + +## 项目概述 + +### 功能 +- 实时监控 Linux 服务器的 CPU、内存、存储、GPU 使用情况 +- 支持同时监控多个服务器 +- 卡片式展示,直观清晰 +- 详情弹窗查看完整信息 + +### 技术栈 +- **前端**: React 19 + TypeScript + Vite +- **后端**: Go (原生 net/http 库) +- **风格**: 白色柔和风格界面 + +## 快速开始 + +### 后端部署 + +1. 进入后端目录 +```bash +cd mengyamonitor-backend +``` + +2. 编译(在 Linux 服务器上) +```bash +go build -o mengyamonitor-backend +``` + +3. 运行 +```bash +./mengyamonitor-backend +``` +默认监听端口: 9292 + +4. 可选:配置环境变量 +```bash +PORT=8080 ./mengyamonitor-backend +``` + +### 前端开发 + +1. 进入前端目录 +```bash +cd mengyamonitor-frontend +``` + +2. 安装依赖 +```bash +npm install +``` + +3. 启动开发服务器 +```bash +npm run dev +``` +默认访问端口: 2929 + +4. 构建生产版本 +```bash +npm run build +``` + +## 使用说明 + +### 1. 部署后端到服务器 +将编译好的 `mengyamonitor-backend` 二进制文件上传到需要监控的 Linux 服务器并运行。 + +### 2. 配置前端 +在前端界面点击"添加服务器",输入: +- 服务器名称:例如"生产服务器1" +- 服务器地址:例如"http://192.168.1.100:9292" + +### 3. 查看监控数据 +- 主界面显示所有服务器的基本信息(CPU、内存使用率) +- 点击"查看详情"可查看完整的系统信息 +- 数据每2秒自动刷新 + +## 项目结构 + +``` +萌芽监控面板/ +├── mengyamonitor-backend/ # 后端服务 +│ ├── main.go # 主程序和HTTP服务器 +│ ├── systeminfo.go # 系统信息采集 +│ ├── go.mod # Go模块配置 +│ └── README.md # 后端文档 +│ +├── mengyamonitor-frontend/ # 前端应用 +│ ├── src/ +│ │ ├── api/ # API调用层 +│ │ ├── components/ # React组件 +│ │ ├── hooks/ # 自定义Hooks +│ │ ├── types/ # TypeScript类型 +│ │ ├── utils/ # 工具函数 +│ │ ├── App.tsx # 主应用 +│ │ └── main.tsx # 入口文件 +│ ├── package.json +│ ├── vite.config.ts +│ └── README.md # 前端文档 +│ +└── 需求.txt # 需求文档 +``` + +## API 接口 + +### GET /api/health +健康检查 + +### GET /api/metrics +获取系统监控指标 +```json +{ + "data": { + "hostname": "server1", + "timestamp": "2025-12-10T10:00:00Z", + "cpu": { "usagePercent": 23.45, ... }, + "memory": { "usedPercent": 50.0, ... }, + "storage": [...], + "gpu": [...], + "os": { ... }, + "uptimeSeconds": 864000.5 + } +} +``` + +## 注意事项 + +1. **系统支持**: 后端仅支持 Linux 系统 +2. **权限要求**: 需要读取 `/proc` 文件系统的权限 +3. **GPU 监控**: 需要安装 `nvidia-smi` 工具(可选) +4. **网络访问**: 确保前端可以访问后端的 9292 端口 +5. **CORS**: 后端已配置允许跨域访问 + +## 常见问题 + +**Q: 前端无法连接后端?** +A: 检查服务器防火墙是否开放 9292 端口,确保后端服务正在运行。 + +**Q: GPU 信息显示不可用?** +A: 如果服务器没有 NVIDIA GPU 或未安装 nvidia-smi,GPU 信息会显示为"不可用",这是正常的。 + +**Q: 如何将前端打包成桌面应用?** +A: 可以使用 Electron 或 Tauri 框架将前端打包成桌面应用,详见前端 README。 + +## 开发者 + +- 前后端分离架构,代码结构清晰 +- 符合企业级开发规范 +- 易于扩展和维护 + +## License + +MIT diff --git a/mengyamonitor-backend/BUILD.md b/mengyamonitor-backend/BUILD.md new file mode 100644 index 0000000..7550137 --- /dev/null +++ b/mengyamonitor-backend/BUILD.md @@ -0,0 +1,88 @@ +# 编译说明 - 兼容旧版本系统 + +## 问题说明 + +如果在 Debian 11 或其他旧版本系统上运行时出现 GLIBC 版本错误: +``` +./mengyamonitor-backend: /lib/x86_64-linux-gnu/libc.so.6: version `GLIBC_2.34' not found +``` + +这是因为编译时使用了较新版本的 GLIBC,而目标系统的 GLIBC 版本较旧。 + +## 解决方案 + +### 方案 1:在目标系统上编译(推荐) + +在 Debian 11 服务器上直接编译: + +```bash +cd mengyamonitor-backend + +# 禁用 CGO,静态链接 +export CGO_ENABLED=0 +go build -ldflags="-s -w" -o mengyamonitor-backend . + +# 或者使用提供的脚本 +chmod +x build.sh +./build.sh +``` + +### 方案 2:使用静态链接编译 + +在任何系统上编译,但使用静态链接: + +```bash +cd mengyamonitor-backend + +# 禁用 CGO +export CGO_ENABLED=0 +export GOOS=linux +export GOARCH=amd64 + +# 编译(静态链接,不依赖系统库) +go build -ldflags="-s -w" -o mengyamonitor-backend . +``` + +### 方案 3:使用 Docker 编译 + +使用 Docker 在 Debian 11 环境中编译: + +```bash +# 使用 Debian 11 镜像编译 +docker run --rm -v $(pwd):/app -w /app golang:1.21-bullseye sh -c " + export CGO_ENABLED=0 + go build -ldflags='-s -w' -o mengyamonitor-backend . +" + +# 或者使用多阶段构建 +docker build -t mengyamonitor-backend -f Dockerfile.build . +``` + +## 验证编译结果 + +编译完成后,检查二进制文件: + +```bash +# 检查文件类型 +file mengyamonitor-backend + +# 检查依赖(如果是静态链接,应该显示 "not a dynamic executable") +ldd mengyamonitor-backend + +# 检查 GLIBC 依赖(应该没有或很少) +objdump -T mengyamonitor-backend | grep GLIBC +``` + +## 编译参数说明 + +- `CGO_ENABLED=0`: 禁用 CGO,使用纯 Go 实现,不依赖系统 C 库 +- `-ldflags="-s -w"`: 减小二进制文件大小 + - `-s`: 省略符号表和调试信息 + - `-w`: 省略 DWARF 符号表 + +## 注意事项 + +1. 禁用 CGO 后,某些需要 C 库的功能可能不可用,但本项目不依赖 CGO +2. 静态链接的二进制文件会稍大一些,但兼容性更好 +3. 如果必须使用 CGO,需要在目标系统上编译,或使用相同 GLIBC 版本的系统编译 + diff --git a/mengyamonitor-backend/README.md b/mengyamonitor-backend/README.md new file mode 100644 index 0000000..160f6fe --- /dev/null +++ b/mengyamonitor-backend/README.md @@ -0,0 +1,121 @@ +# 萌芽监控面板 - 后端服务 + +## 概述 +Linux 服务器监控后端服务,使用 Go 原生 net/http 库实现。 + +## 功能 +- CPU 使用率和负载监控 +- 内存使用情况 +- 磁盘存储监控 +- GPU 监控(支持 NVIDIA) +- 操作系统信息 +- 系统运行时间 + +## API 端点 + +### `GET /api/health` +健康检查端点 +```json +{ + "status": "ok", + "timestamp": "2025-12-10T10:00:00Z" +} +``` + +### `GET /api/metrics` +获取系统监控指标 +```json +{ + "data": { + "hostname": "server1", + "timestamp": "2025-12-10T10:00:00Z", + "cpu": { + "model": "Intel(R) Xeon(R) CPU E5-2680 v4 @ 2.40GHz", + "cores": 8, + "usagePercent": 23.45, + "loadAverages": [1.2, 1.5, 1.8] + }, + "memory": { + "totalBytes": 16777216000, + "usedBytes": 8388608000, + "freeBytes": 8388608000, + "usedPercent": 50.0 + }, + "storage": [{ + "mount": "/", + "totalBytes": 107374182400, + "usedBytes": 53687091200, + "freeBytes": 53687091200, + "usedPercent": 50.0 + }], + "gpu": [{ + "name": "Tesla T4", + "memoryTotalMB": 15360, + "memoryUsedMB": 512, + "utilizationPercent": 15.0, + "status": "ok" + }], + "os": { + "kernel": "Linux version 5.15.0", + "distro": "Ubuntu 22.04 LTS", + "architecture": "amd64" + }, + "uptimeSeconds": 864000.5 + } +} +``` + +## 运行方式 + +### 开发环境 +```bash +go run . +``` + +### 生产环境 + +#### 标准编译 +```bash +# 编译 +go build -o mengyamonitor-backend + +# 运行 +./mengyamonitor-backend +``` + +#### 兼容旧版本系统编译(推荐) + +如果需要在 Debian 11 或其他旧版本系统上运行,使用静态链接编译: + +```bash +# 禁用 CGO,静态链接(不依赖系统 GLIBC 版本) +export CGO_ENABLED=0 +go build -ldflags="-s -w" -o mengyamonitor-backend . + +# 或使用提供的脚本 +chmod +x build.sh +./build.sh +``` + +这样可以避免 GLIBC 版本兼容性问题。详细说明请参考 [BUILD.md](./BUILD.md)。 + +### 环境变量 +- `HOST`: 监听地址,默认 `0.0.0.0` +- `PORT`: 监听端口,默认 `9292` + +示例: +```bash +PORT=8080 ./mengyamonitor-backend +``` + +## 部署到服务器 + +1. 将编译好的二进制文件上传到目标服务器 +2. 赋予执行权限:`chmod +x mengyamonitor-backend` +3. 运行服务:`./mengyamonitor-backend` +4. 可选:使用 systemd 或 supervisor 管理服务进程 + +## 注意事项 +- 仅支持 Linux 系统 +- GPU 监控需要安装 nvidia-smi 工具 +- 需要读取 /proc 文件系统的权限 diff --git a/mengyamonitor-backend/build.sh b/mengyamonitor-backend/build.sh new file mode 100644 index 0000000..1e6e86e --- /dev/null +++ b/mengyamonitor-backend/build.sh @@ -0,0 +1,29 @@ +#!/bin/bash +# 编译脚本 - 兼容旧版本 GLIBC + +echo "开始编译 mengyamonitor-backend..." + +# 禁用 CGO,使用纯 Go 编译(不依赖系统 C 库) +export CGO_ENABLED=0 + +# 设置目标平台 +export GOOS=linux +export GOARCH=amd64 + +# 编译(静态链接) +go build -ldflags="-s -w" -o mengyamonitor-backend . + +if [ $? -eq 0 ]; then + echo "编译成功!" + echo "二进制文件: mengyamonitor-backend" + echo "" + echo "检查文件信息:" + file mengyamonitor-backend + echo "" + echo "检查依赖库:" + ldd mengyamonitor-backend 2>/dev/null || echo "静态链接,无外部依赖" +else + echo "编译失败!" + exit 1 +fi + diff --git a/mengyamonitor-backend/collector.go b/mengyamonitor-backend/collector.go new file mode 100644 index 0000000..0e7078f --- /dev/null +++ b/mengyamonitor-backend/collector.go @@ -0,0 +1,103 @@ +package main + +import ( + "bufio" + "os" + "runtime" + "strconv" + "strings" + "time" +) + +// CollectMetrics 收集所有系统监控指标 +func CollectMetrics() (*Metrics, error) { + hostname, _ := os.Hostname() + + cpuModel := firstMatchInFile("/proc/cpuinfo", "model name") + cpuUsage, err := readCPUUsage() + if err != nil { + return nil, err + } + cpuTemp := readCPUTemperature() + perCoreUsage := readPerCoreUsage() + + mem, err := readMemory() + if err != nil { + return nil, err + } + + storage, err := readAllStorage() + if err != nil { + return nil, err + } + + gpu := readGPU() + network := readNetworkInterfaces() + systemStats := readSystemStats() + systemStats.DockerStats = readDockerStats() + + osInfo := readOSInfo() + uptime := readUptime() + + loads := readLoadAverages() + + return &Metrics{ + Hostname: hostname, + Timestamp: time.Now().UTC(), + CPU: CPUMetrics{ + Model: cpuModel, + Cores: runtime.NumCPU(), + UsagePercent: round(cpuUsage, 2), + LoadAverages: loads, + Temperature: cpuTemp, + PerCoreUsage: perCoreUsage, + }, + Memory: mem, + Storage: storage, + GPU: gpu, + Network: network, + System: systemStats, + OS: osInfo, + UptimeSeconds: uptime, + }, nil +} + +func readOSInfo() OSInfo { + distro := readOSRelease() + kernel := strings.TrimSpace(readFirstLine("/proc/version")) + arch := runtime.GOARCH + return OSInfo{ + Kernel: kernel, + Distro: distro, + Architecture: arch, + } +} + +func readOSRelease() string { + f, err := os.Open("/etc/os-release") + if err != nil { + return runtime.GOOS + } + defer f.Close() + scanner := bufio.NewScanner(f) + for scanner.Scan() { + line := scanner.Text() + if strings.HasPrefix(line, "PRETTY_NAME=") { + return strings.Trim(line[len("PRETTY_NAME="):], "\"") + } + } + return runtime.GOOS +} + +func readUptime() float64 { + line := readFirstLine("/proc/uptime") + fields := strings.Fields(line) + if len(fields) == 0 { + return 0 + } + v, err := strconv.ParseFloat(fields[0], 64) + if err != nil { + return 0 + } + return v +} diff --git a/mengyamonitor-backend/cpu.go b/mengyamonitor-backend/cpu.go new file mode 100644 index 0000000..d6994b1 --- /dev/null +++ b/mengyamonitor-backend/cpu.go @@ -0,0 +1,182 @@ +package main + +import ( + "bufio" + "errors" + "os" + "strconv" + "strings" + "time" +) + +// readCPUUsage 读取CPU整体使用率 +func readCPUUsage() (float64, error) { + idle1, total1, err := readCPUTicks() + if err != nil { + return 0, err + } + time.Sleep(250 * time.Millisecond) + idle2, total2, err := readCPUTicks() + if err != nil { + return 0, err + } + if total2 == total1 { + return 0, errors.New("cpu totals unchanged") + } + idleDelta := float64(idle2 - idle1) + totalDelta := float64(total2 - total1) + usage := (1.0 - idleDelta/totalDelta) * 100 + if usage < 0 { + usage = 0 + } + return usage, nil +} + +// readCPUTicks 读取CPU的idle和total ticks +func readCPUTicks() (idle, total uint64, err error) { + f, err := os.Open("/proc/stat") + if err != nil { + return 0, 0, err + } + defer f.Close() + scanner := bufio.NewScanner(f) + if !scanner.Scan() { + return 0, 0, errors.New("failed to scan /proc/stat") + } + fields := strings.Fields(scanner.Text()) + if len(fields) < 5 { + return 0, 0, errors.New("unexpected /proc/stat format") + } + // fields[0] is "cpu" + var vals []uint64 + for _, f := range fields[1:] { + v, err := strconv.ParseUint(f, 10, 64) + if err != nil { + return 0, 0, err + } + vals = append(vals, v) + } + for _, v := range vals { + total += v + } + if len(vals) > 3 { + idle = vals[3] // idle time + // 注意:不包含 iowait,因为 iowait 不算真正的空闲时间 + // 如果系统有 iowait,它会在 total 中,但不应该算作 idle + } + return idle, total, nil +} + +// readPerCoreUsage 读取每个CPU核心的使用率 +func readPerCoreUsage() []CoreUsage { + coreUsages := []CoreUsage{} + + // 第一次读取 + cores1 := readPerCoreTicks() + time.Sleep(100 * time.Millisecond) // 减少到100ms + // 第二次读取 + cores2 := readPerCoreTicks() + + for i := 0; i < len(cores1) && i < len(cores2); i++ { + idle1, total1 := cores1[i][0], cores1[i][1] + idle2, total2 := cores2[i][0], cores2[i][1] + + if total2 == total1 { + continue + } + + idleDelta := float64(idle2 - idle1) + totalDelta := float64(total2 - total1) + usage := (1.0 - idleDelta/totalDelta) * 100 + if usage < 0 { + usage = 0 + } + + coreUsages = append(coreUsages, CoreUsage{ + Core: i, + Percent: round(usage, 1), + }) + } + + return coreUsages +} + +// readPerCoreTicks 读取每个CPU核心的ticks +func readPerCoreTicks() [][2]uint64 { + var result [][2]uint64 + + f, err := os.Open("/proc/stat") + if err != nil { + return result + } + defer f.Close() + + scanner := bufio.NewScanner(f) + for scanner.Scan() { + line := scanner.Text() + if !strings.HasPrefix(line, "cpu") { + continue + } + if strings.HasPrefix(line, "cpu ") { + continue // 跳过总的cpu行 + } + + fields := strings.Fields(line) + if len(fields) < 5 { + continue + } + + var vals []uint64 + for _, f := range fields[1:] { + v, err := strconv.ParseUint(f, 10, 64) + if err != nil { + break + } + vals = append(vals, v) + } + + if len(vals) < 5 { + continue + } + + var total, idle uint64 + for _, v := range vals { + total += v + } + idle = vals[3] // idle time only, not including iowait + + result = append(result, [2]uint64{idle, total}) + } + + return result +} + +// readCPUTemperature 读取CPU温度 +func readCPUTemperature() float64 { + // 尝试从常见位置读取温度 + paths := []string{ + "/sys/class/thermal/thermal_zone0/temp", + "/sys/class/hwmon/hwmon0/temp1_input", + "/sys/class/hwmon/hwmon1/temp1_input", + } + for _, path := range paths { + if temp := readTempFromFile(path); temp > 0 { + return temp + } + } + return 0 +} + +// readLoadAverages 读取系统负载平均值 +func readLoadAverages() []float64 { + line := readFirstLine("/proc/loadavg") + fields := strings.Fields(line) + res := make([]float64, 0, 3) + for i := 0; i < len(fields) && i < 3; i++ { + v, err := strconv.ParseFloat(fields[i], 64) + if err == nil { + res = append(res, v) + } + } + return res +} diff --git a/mengyamonitor-backend/docker.go b/mengyamonitor-backend/docker.go new file mode 100644 index 0000000..eeb3fab --- /dev/null +++ b/mengyamonitor-backend/docker.go @@ -0,0 +1,78 @@ +package main + +import ( + "context" + "os/exec" + "strings" + "time" +) + +// readDockerStats 读取 Docker 统计信息(简化版) +func readDockerStats() DockerStats { + stats := DockerStats{ + Available: false, + RunningNames: []string{}, + StoppedNames: []string{}, + ImageNames: []string{}, + } + + // 检查 docker 是否可用 + if _, err := exec.LookPath("docker"); err != nil { + return stats + } + + stats.Available = true + + // 获取 Docker 版本 + cmd := exec.Command("docker", "--version") + if out, err := cmd.Output(); err == nil { + stats.Version = strings.TrimSpace(string(out)) + } + + // 获取运行中的容器名 + ctx, cancel := context.WithTimeout(context.Background(), 3*time.Second) + defer cancel() + cmd = exec.CommandContext(ctx, "docker", "ps", "--format", "{{.Names}}") + if out, err := cmd.Output(); err == nil { + lines := strings.Split(strings.TrimSpace(string(out)), "\n") + for _, line := range lines { + line = strings.TrimSpace(line) + if line != "" { + stats.RunningNames = append(stats.RunningNames, line) + stats.Running++ + } + } + } + + // 获取停止的容器名 + ctx, cancel = context.WithTimeout(context.Background(), 3*time.Second) + defer cancel() + cmd = exec.CommandContext(ctx, "docker", "ps", "-a", "--filter", "status=exited", "--format", "{{.Names}}") + if out, err := cmd.Output(); err == nil { + lines := strings.Split(strings.TrimSpace(string(out)), "\n") + for _, line := range lines { + line = strings.TrimSpace(line) + if line != "" { + stats.StoppedNames = append(stats.StoppedNames, line) + stats.Stopped++ + } + } + } + + // 获取镜像名(只获取前20个,避免数据过多) + ctx, cancel = context.WithTimeout(context.Background(), 3*time.Second) + defer cancel() + cmd = exec.CommandContext(ctx, "docker", "images", "--format", "{{.Repository}}:{{.Tag}}") + if out, err := cmd.Output(); err == nil { + lines := strings.Split(strings.TrimSpace(string(out)), "\n") + for _, line := range lines { + line = strings.TrimSpace(line) + if line != "" && line != ":" { + stats.ImageNames = append(stats.ImageNames, line) + stats.ImageCount++ + } + } + } + + return stats +} diff --git a/mengyamonitor-backend/go.mod b/mengyamonitor-backend/go.mod new file mode 100644 index 0000000..9920de7 --- /dev/null +++ b/mengyamonitor-backend/go.mod @@ -0,0 +1,3 @@ +module mengyamonitor-backend + +go 1.22 diff --git a/mengyamonitor-backend/gpu.go b/mengyamonitor-backend/gpu.go new file mode 100644 index 0000000..3bc6900 --- /dev/null +++ b/mengyamonitor-backend/gpu.go @@ -0,0 +1,48 @@ +package main + +import ( + "os/exec" + "strconv" + "strings" +) + +// readGPU 读取GPU信息 +func readGPU() []GPUMetrics { + _, err := exec.LookPath("nvidia-smi") + if err != nil { + return []GPUMetrics{{Status: "not_available"}} + } + // Query GPU info including temperature + cmd := exec.Command("nvidia-smi", "--query-gpu=name,memory.total,memory.used,utilization.gpu,temperature.gpu", "--format=csv,noheader,nounits") + out, err := cmd.Output() + if err != nil { + return []GPUMetrics{{Status: "error", Name: "nvidia-smi", UtilizationPercent: 0}} + } + lines := strings.Split(strings.TrimSpace(string(out)), "\n") + gpus := make([]GPUMetrics, 0, len(lines)) + for _, line := range lines { + parts := strings.Split(line, ",") + for i := range parts { + parts[i] = strings.TrimSpace(parts[i]) + } + if len(parts) < 5 { + continue + } + total, _ := strconv.ParseInt(parts[1], 10, 64) + used, _ := strconv.ParseInt(parts[2], 10, 64) + util, _ := strconv.ParseFloat(parts[3], 64) + temp, _ := strconv.ParseFloat(parts[4], 64) + gpus = append(gpus, GPUMetrics{ + Name: parts[0], + MemoryTotalMB: total, + MemoryUsedMB: used, + UtilizationPercent: round(util, 2), + Temperature: temp, + Status: "ok", + }) + } + if len(gpus) == 0 { + return []GPUMetrics{{Status: "not_available"}} + } + return gpus +} diff --git a/mengyamonitor-backend/latency.go b/mengyamonitor-backend/latency.go new file mode 100644 index 0000000..015835c --- /dev/null +++ b/mengyamonitor-backend/latency.go @@ -0,0 +1,56 @@ +package main + +import ( + "fmt" + "net" + "time" +) + +// LatencyInfo 延迟信息 +type LatencyInfo struct { + ClientToServer float64 `json:"clientToServer"` // 客户端到服务器延迟(ms),由前端计算 + External map[string]string `json:"external"` // 外部网站延迟 +} + +// checkExternalLatency 检测外部网站延迟 +func checkExternalLatency(host string, timeout time.Duration) string { + start := time.Now() + + // 尝试 TCP 连接 80 端口 + conn, err := net.DialTimeout("tcp", host+":80", timeout) + if err != nil { + // 如果 80 端口失败,尝试 443 (HTTPS) + conn, err = net.DialTimeout("tcp", host+":443", timeout) + if err != nil { + return "超时" + } + } + defer conn.Close() + + latency := time.Since(start).Milliseconds() + + // 检查是否超时(超过超时时间的一半就认为可能有问题) + if latency > timeout.Milliseconds()/2 { + return "超时" + } + + return fmt.Sprintf("%d ms", latency) +} + +// readExternalLatencies 读取外部网站延迟 +func readExternalLatencies() map[string]string { + latencies := make(map[string]string) + timeout := 3 * time.Second + + // 检测百度 + latencies["baidu.com"] = checkExternalLatency("baidu.com", timeout) + + // 检测谷歌 + latencies["google.com"] = checkExternalLatency("google.com", timeout) + + // 检测 GitHub + latencies["github.com"] = checkExternalLatency("github.com", timeout) + + return latencies +} + diff --git a/mengyamonitor-backend/main.go b/mengyamonitor-backend/main.go new file mode 100644 index 0000000..12475a8 --- /dev/null +++ b/mengyamonitor-backend/main.go @@ -0,0 +1,265 @@ +package main + +import ( + "encoding/json" + "fmt" + "log" + "net/http" + "os" + "runtime" + "time" +) + +// envelope keeps JSON responses consistent. +type envelope map[string]any + +func main() { + host := getenv("HOST", "0.0.0.0") + port := getenv("PORT", "9292") + + mux := http.NewServeMux() + mux.HandleFunc("/", rootHandler) + mux.HandleFunc("/api/health", healthHandler) + // 拆分的细粒度API端点 + mux.HandleFunc("/api/metrics/cpu", cpuMetricsHandler) + mux.HandleFunc("/api/metrics/memory", memoryMetricsHandler) + mux.HandleFunc("/api/metrics/storage", storageMetricsHandler) + mux.HandleFunc("/api/metrics/gpu", gpuMetricsHandler) + mux.HandleFunc("/api/metrics/network", networkMetricsHandler) + mux.HandleFunc("/api/metrics/system", systemMetricsHandler) + mux.HandleFunc("/api/metrics/docker", dockerMetricsHandler) + mux.HandleFunc("/api/metrics/latency", latencyMetricsHandler) + + srv := &http.Server{ + Addr: fmt.Sprintf("%s:%s", host, port), + Handler: loggingMiddleware(corsMiddleware(mux)), + ReadHeaderTimeout: 5 * time.Second, + } + + log.Printf("server starting on http://%s:%s", host, port) + if err := srv.ListenAndServe(); err != nil && err != http.ErrServerClosed { + log.Fatalf("server error: %v", err) + } +} + +func rootHandler(w http.ResponseWriter, r *http.Request) { + respondJSON(w, http.StatusOK, envelope{ + "service": "萌芽监控面板 API", + "version": "1.0.0", + "endpoints": []map[string]string{ + { + "path": "/", + "description": "API 信息和可用端点列表", + }, + { + "path": "/api/health", + "description": "健康检查", + }, + { + "path": "/api/metrics/cpu", + "description": "获取 CPU 监控数据", + }, + { + "path": "/api/metrics/memory", + "description": "获取内存监控数据", + }, + { + "path": "/api/metrics/storage", + "description": "获取存储监控数据", + }, + { + "path": "/api/metrics/gpu", + "description": "获取 GPU 监控数据", + }, + { + "path": "/api/metrics/network", + "description": "获取网络接口监控数据", + }, + { + "path": "/api/metrics/system", + "description": "获取系统统计信息(进程、包、磁盘速度等)", + }, + { + "path": "/api/metrics/docker", + "description": "获取 Docker 容器监控数据", + }, + }, + }) +} + +func healthHandler(w http.ResponseWriter, r *http.Request) { + respondJSON(w, http.StatusOK, envelope{ + "status": "ok", + "timestamp": time.Now().UTC(), + }) +} + +// CPU监控数据 +func cpuMetricsHandler(w http.ResponseWriter, r *http.Request) { + cpuModel := firstMatchInFile("/proc/cpuinfo", "model name") + cpuUsage, err := readCPUUsage() + if err != nil { + cpuUsage = 0 + } + cpuTemp := readCPUTemperature() + perCoreUsage := readPerCoreUsage() + loads := readLoadAverages() + + cores := runtime.NumCPU() + respondJSON(w, http.StatusOK, envelope{ + "data": CPUMetrics{ + Model: cpuModel, + Cores: cores, + UsagePercent: round(cpuUsage, 2), + LoadAverages: loads, + Temperature: cpuTemp, + PerCoreUsage: perCoreUsage, + }, + }) +} + +// 内存监控数据 +func memoryMetricsHandler(w http.ResponseWriter, r *http.Request) { + mem, err := readMemory() + if err != nil { + respondJSON(w, http.StatusInternalServerError, envelope{ + "error": "failed to read memory", + }) + return + } + respondJSON(w, http.StatusOK, envelope{ + "data": mem, + }) +} + +// 存储监控数据 +func storageMetricsHandler(w http.ResponseWriter, r *http.Request) { + storage, err := readAllStorage() + if err != nil { + respondJSON(w, http.StatusInternalServerError, envelope{ + "error": "failed to read storage", + }) + return + } + respondJSON(w, http.StatusOK, envelope{ + "data": storage, + }) +} + +// GPU监控数据 +func gpuMetricsHandler(w http.ResponseWriter, r *http.Request) { + gpu := readGPU() + respondJSON(w, http.StatusOK, envelope{ + "data": gpu, + }) +} + +// 网络监控数据 +func networkMetricsHandler(w http.ResponseWriter, r *http.Request) { + network := readNetworkInterfaces() + respondJSON(w, http.StatusOK, envelope{ + "data": network, + }) +} + +// 系统统计信息 +func systemMetricsHandler(w http.ResponseWriter, r *http.Request) { + stats := readSystemStats() + + // 读取系统基本信息 + hostname, _ := os.Hostname() + osInfo := readOSInfo() + uptime := readUptime() + + // 计算总网络速度(汇总所有接口) + networkInterfaces := readNetworkInterfaces() + var totalRxSpeed, totalTxSpeed float64 + for _, iface := range networkInterfaces { + totalRxSpeed += iface.RxSpeed // bytes/s + totalTxSpeed += iface.TxSpeed // bytes/s + } + // 转换为 MB/s + stats.NetworkRxSpeed = round(totalRxSpeed/1024/1024, 2) + stats.NetworkTxSpeed = round(totalTxSpeed/1024/1024, 2) + + respondJSON(w, http.StatusOK, envelope{ + "data": map[string]interface{}{ + "hostname": hostname, + "os": osInfo, + "uptimeSeconds": uptime, + "processCount": stats.ProcessCount, + "packageCount": stats.PackageCount, + "packageManager": stats.PackageManager, + "temperature": stats.Temperature, + "diskReadSpeed": stats.DiskReadSpeed, + "diskWriteSpeed": stats.DiskWriteSpeed, + "networkRxSpeed": stats.NetworkRxSpeed, + "networkTxSpeed": stats.NetworkTxSpeed, + "topProcesses": stats.TopProcesses, + "systemLogs": stats.SystemLogs, + }, + }) +} + +// Docker监控数据 +func dockerMetricsHandler(w http.ResponseWriter, r *http.Request) { + docker := readDockerStats() + respondJSON(w, http.StatusOK, envelope{ + "data": docker, + }) +} + +// 延迟监控数据 +func latencyMetricsHandler(w http.ResponseWriter, r *http.Request) { + // 记录请求开始时间,用于计算客户端到服务器的延迟 + startTime := time.Now() + + // 读取外部网站延迟 + externalLatencies := readExternalLatencies() + + // 计算服务器处理时间(这个可以作为参考,实际客户端延迟由前端计算) + serverProcessTime := time.Since(startTime).Milliseconds() + + respondJSON(w, http.StatusOK, envelope{ + "data": map[string]interface{}{ + "serverProcessTime": serverProcessTime, // 服务器处理时间(参考) + "external": externalLatencies, + }, + }) +} + +func respondJSON(w http.ResponseWriter, status int, body envelope) { + w.Header().Set("Content-Type", "application/json") + w.WriteHeader(status) + if err := json.NewEncoder(w).Encode(body); err != nil { + log.Printf("write json error: %v", err) + } +} + +func getenv(key, fallback string) string { + if v, ok := os.LookupEnv(key); ok && v != "" { + return v + } + return fallback +} + +func loggingMiddleware(next http.Handler) http.Handler { + return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + start := time.Now() + next.ServeHTTP(w, r) + log.Printf("%s %s %s", r.Method, r.URL.Path, time.Since(start)) + }) +} + +func corsMiddleware(next http.Handler) http.Handler { + return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + w.Header().Set("Access-Control-Allow-Origin", "*") + w.Header().Set("Access-Control-Allow-Methods", "GET, OPTIONS") + w.Header().Set("Access-Control-Allow-Headers", "Content-Type") + if r.Method == http.MethodOptions { + w.WriteHeader(http.StatusNoContent) + return + } + next.ServeHTTP(w, r) + }) +} diff --git a/mengyamonitor-backend/memory.go b/mengyamonitor-backend/memory.go new file mode 100644 index 0000000..5fd2731 --- /dev/null +++ b/mengyamonitor-backend/memory.go @@ -0,0 +1,64 @@ +package main + +import ( + "bufio" + "os" + "strconv" + "strings" +) + +// readMemory 读取内存信息 +func readMemory() (MemoryMetrics, error) { + totals := map[string]uint64{} + f, err := os.Open("/proc/meminfo") + if err != nil { + return MemoryMetrics{}, err + } + defer f.Close() + scanner := bufio.NewScanner(f) + for scanner.Scan() { + line := scanner.Text() + parts := strings.Split(line, ":") + if len(parts) < 2 { + continue + } + key := strings.TrimSpace(parts[0]) + fields := strings.Fields(strings.TrimSpace(parts[1])) + if len(fields) == 0 { + continue + } + value, err := strconv.ParseUint(fields[0], 10, 64) + if err != nil { + continue + } + totals[key] = value * 1024 // kB to bytes + } + total := totals["MemTotal"] + + // 优先使用 MemAvailable(Linux 3.14+),如果没有则计算 + var free uint64 + if available, ok := totals["MemAvailable"]; ok && available > 0 { + free = available + } else { + // 回退到 MemFree + Buffers + Cached(适用于较老的系统) + free = totals["MemFree"] + if buffers, ok := totals["Buffers"]; ok { + free += buffers + } + if cached, ok := totals["Cached"]; ok { + free += cached + } + } + + used := total - free + usedPercent := 0.0 + if total > 0 { + usedPercent = (float64(used) / float64(total)) * 100 + } + return MemoryMetrics{ + TotalBytes: total, + UsedBytes: used, + FreeBytes: free, + UsedPercent: round(usedPercent, 2), + }, nil +} diff --git a/mengyamonitor-backend/network.go b/mengyamonitor-backend/network.go new file mode 100644 index 0000000..5a61e02 --- /dev/null +++ b/mengyamonitor-backend/network.go @@ -0,0 +1,110 @@ +package main + +import ( + "os" + "os/exec" + "strconv" + "strings" + "time" +) + +// readNetworkInterfaces 读取网络接口信息(包含瞬时流量速度) - 优化版:所有接口并行采样 +func readNetworkInterfaces() []NetworkInterface { + interfaces := []NetworkInterface{} + + // 读取网络接口列表 + entries, err := os.ReadDir("/sys/class/net") + if err != nil { + return interfaces + } + + // 收集要监控的接口名称 + var validIfaces []string + for _, entry := range entries { + ifName := entry.Name() + if ifName == "lo" { // 跳过回环接口 + continue + } + + // 跳过 Docker 相关网络接口 + if strings.HasPrefix(ifName, "docker") || + strings.HasPrefix(ifName, "br-") || + strings.HasPrefix(ifName, "veth") { + continue + } + + validIfaces = append(validIfaces, ifName) + } + + // 第一次批量读取所有接口的流量 + firstReadings := make(map[string][2]uint64) // [rx, tx] + for _, ifName := range validIfaces { + rxPath := "/sys/class/net/" + ifName + "/statistics/rx_bytes" + txPath := "/sys/class/net/" + ifName + "/statistics/tx_bytes" + + var rxBytes, txBytes uint64 + if rxStr := readFirstLine(rxPath); rxStr != "" { + rxBytes, _ = strconv.ParseUint(strings.TrimSpace(rxStr), 10, 64) + } + if txStr := readFirstLine(txPath); txStr != "" { + txBytes, _ = strconv.ParseUint(strings.TrimSpace(txStr), 10, 64) + } + firstReadings[ifName] = [2]uint64{rxBytes, txBytes} + } + + // 等待500ms(所有接口一起等待,而不是每个接口单独等待) + time.Sleep(500 * time.Millisecond) + + // 第二次批量读取并构建结果 + for _, ifName := range validIfaces { + iface := NetworkInterface{ + Name: ifName, + } + + // 读取 MAC 地址 + macPath := "/sys/class/net/" + ifName + "/address" + iface.MACAddress = strings.TrimSpace(readFirstLine(macPath)) + + // 读取 IP 地址 (使用 ip addr show) + cmd := exec.Command("ip", "addr", "show", ifName) + if out, err := cmd.Output(); err == nil { + lines := strings.Split(string(out), "\n") + for _, line := range lines { + line = strings.TrimSpace(line) + if strings.HasPrefix(line, "inet ") { + fields := strings.Fields(line) + if len(fields) >= 2 { + iface.IPAddress = strings.Split(fields[1], "/")[0] + break + } + } + } + } + + // 读取第二次流量统计 + rxPath := "/sys/class/net/" + ifName + "/statistics/rx_bytes" + txPath := "/sys/class/net/" + ifName + "/statistics/tx_bytes" + + var rxBytes2, txBytes2 uint64 + if rxStr := readFirstLine(rxPath); rxStr != "" { + rxBytes2, _ = strconv.ParseUint(strings.TrimSpace(rxStr), 10, 64) + } + if txStr := readFirstLine(txPath); txStr != "" { + txBytes2, _ = strconv.ParseUint(strings.TrimSpace(txStr), 10, 64) + } + + // 设置累计流量 + iface.RxBytes = rxBytes2 + iface.TxBytes = txBytes2 + + // 计算瞬时速度 (bytes/s,乘以2因为采样间隔是0.5秒) + if first, ok := firstReadings[ifName]; ok { + iface.RxSpeed = float64(rxBytes2-first[0]) * 2 + iface.TxSpeed = float64(txBytes2-first[1]) * 2 + } + + interfaces = append(interfaces, iface) + } + + return interfaces +} diff --git a/mengyamonitor-backend/system.go b/mengyamonitor-backend/system.go new file mode 100644 index 0000000..e1d7bc4 --- /dev/null +++ b/mengyamonitor-backend/system.go @@ -0,0 +1,390 @@ +package main + +import ( + "bufio" + "context" + "os" + "os/exec" + "strconv" + "strings" + "time" +) + +// readSystemStats 读取系统统计信息 +func readSystemStats() SystemStats { + stats := SystemStats{} + + // 读取进程数量 + stats.ProcessCount = countProcesses() + + // 读取已安装包数量和包管理器类型 + stats.PackageCount, stats.PackageManager = countPackages() + + // 读取系统温度 + stats.Temperature = readSystemTemperature() + + // 读取磁盘读写速度 + stats.DiskReadSpeed, stats.DiskWriteSpeed = readDiskSpeed() + + // 读取 Top 5 进程 + stats.TopProcesses = readTopProcesses() + + // 读取系统日志 + stats.SystemLogs = readSystemLogs(10) + + return stats +} + +func countProcesses() int { + entries, err := os.ReadDir("/proc") + if err != nil { + return 0 + } + count := 0 + for _, entry := range entries { + if entry.IsDir() { + // 进程目录是数字命名的 + if _, err := strconv.Atoi(entry.Name()); err == nil { + count++ + } + } + } + return count +} + +func countPackages() (int, string) { + // 尝试不同的包管理器 + // dpkg (Debian/Ubuntu) + if _, err := exec.LookPath("dpkg"); err == nil { + cmd := exec.Command("dpkg", "-l") + out, err := cmd.Output() + if err == nil { + lines := strings.Split(string(out), "\n") + count := 0 + for _, line := range lines { + if strings.HasPrefix(line, "ii ") { + count++ + } + } + return count, "dpkg (apt)" + } + } + + // rpm (RedHat/CentOS/Fedora) + if _, err := exec.LookPath("rpm"); err == nil { + cmd := exec.Command("rpm", "-qa") + out, err := cmd.Output() + if err == nil { + lines := strings.Split(strings.TrimSpace(string(out)), "\n") + return len(lines), "rpm (yum/dnf)" + } + } + + // pacman (Arch Linux) + if _, err := exec.LookPath("pacman"); err == nil { + cmd := exec.Command("pacman", "-Q") + out, err := cmd.Output() + if err == nil { + lines := strings.Split(strings.TrimSpace(string(out)), "\n") + return len(lines), "pacman" + } + } + + return 0, "unknown" +} + +func readSystemTemperature() float64 { + var cpuTemp float64 = 0 + var fallbackTemp float64 = 0 + + // 1. 优先读取 thermal_zone (通常是 CPU 温度) + thermalDir := "/sys/class/thermal" + entries, err := os.ReadDir(thermalDir) + if err == nil { + for _, entry := range entries { + if !strings.HasPrefix(entry.Name(), "thermal_zone") { + continue + } + tempPath := thermalDir + "/" + entry.Name() + "/temp" + if temp := readTempFromFile(tempPath); temp > 0 && temp > 20 && temp < 120 { + // thermal_zone0 通常是 CPU + if entry.Name() == "thermal_zone0" { + cpuTemp = temp + break + } else if fallbackTemp == 0 { + fallbackTemp = temp + } + } + } + } + + // 2. 扫描所有 hwmon 设备,查找 CPU 温度 + hwmonDir := "/sys/class/hwmon" + entries, err = os.ReadDir(hwmonDir) + if err == nil { + for _, entry := range entries { + hwmonPath := hwmonDir + "/" + entry.Name() + + // 读取 name 文件,检查是否是 CPU 相关 + namePath := hwmonPath + "/name" + name := strings.ToLower(strings.TrimSpace(readFirstLine(namePath))) + + // 检查是否是 CPU 温度传感器 + isCPU := strings.Contains(name, "cpu") || + strings.Contains(name, "core") || + strings.Contains(name, "k10temp") || + strings.Contains(name, "coretemp") || + strings.Contains(name, "zenpower") + + // 尝试读取 temp1_input (通常是 CPU) + temp1Path := hwmonPath + "/temp1_input" + if temp := readTempFromFile(temp1Path); temp > 0 && temp > 20 && temp < 120 { + if isCPU { + cpuTemp = temp + break + } else if fallbackTemp == 0 { + fallbackTemp = temp + } + } + + // 也尝试 temp2_input + temp2Path := hwmonPath + "/temp2_input" + if temp := readTempFromFile(temp2Path); temp > 0 && temp > 20 && temp < 120 { + if isCPU && cpuTemp == 0 { + cpuTemp = temp + } else if fallbackTemp == 0 { + fallbackTemp = temp + } + } + } + } + + // 优先返回 CPU 温度,如果没有则返回其他温度 + if cpuTemp > 0 { + return cpuTemp + } + return fallbackTemp +} + +// readDiskSpeed 读取磁盘瞬时读写速度 (MB/s) +func readDiskSpeed() (float64, float64) { + // 第一次读取 + readSectors1, writeSectors1 := getDiskSectors() + if readSectors1 == 0 && writeSectors1 == 0 { + return 0, 0 + } + + // 等待1秒 + time.Sleep(1 * time.Second) + + // 第二次读取 + readSectors2, writeSectors2 := getDiskSectors() + + // 计算差值(扇区数) + readDiff := readSectors2 - readSectors1 + writeDiff := writeSectors2 - writeSectors1 + + // 扇区大小通常是 512 字节,转换为 MB/s + readSpeed := float64(readDiff) * 512 / 1024 / 1024 + writeSpeed := float64(writeDiff) * 512 / 1024 / 1024 + + return round(readSpeed, 2), round(writeSpeed, 2) +} + +func getDiskSectors() (uint64, uint64) { + f, err := os.Open("/proc/diskstats") + if err != nil { + return 0, 0 + } + defer f.Close() + + scanner := bufio.NewScanner(f) + + var maxRead uint64 = 0 + var mainDevice string + + // 第一次遍历:找到读写量最大的主磁盘(通常是系统盘) + for scanner.Scan() { + line := scanner.Text() + fields := strings.Fields(line) + if len(fields) < 14 { + continue + } + + deviceName := fields[2] + // 跳过分区(分区名通常包含数字,如 sda1, vda1, nvme0n1p1) + if strings.ContainsAny(deviceName, "0123456789") && + !strings.HasPrefix(deviceName, "nvme") && + !strings.HasPrefix(deviceName, "loop") { + continue + } + + // 跳过虚拟设备 + if strings.HasPrefix(deviceName, "loop") || + strings.HasPrefix(deviceName, "ram") || + strings.HasPrefix(deviceName, "zram") { + continue + } + + readSectors, _ := strconv.ParseUint(fields[5], 10, 64) + // 选择读写量最大的作为主磁盘 + if readSectors > maxRead { + maxRead = readSectors + mainDevice = deviceName + } + } + + // 第二次遍历:读取主磁盘的数据 + f.Close() + f, err = os.Open("/proc/diskstats") + if err != nil { + return 0, 0 + } + scanner = bufio.NewScanner(f) + for scanner.Scan() { + line := scanner.Text() + fields := strings.Fields(line) + if len(fields) < 14 { + continue + } + + if fields[2] == mainDevice { + readSectors, _ := strconv.ParseUint(fields[5], 10, 64) + writeSectors, _ := strconv.ParseUint(fields[9], 10, 64) + return readSectors, writeSectors + } + } + + // 如果没找到,尝试常见的设备名(向后兼容) + f.Close() + f, err = os.Open("/proc/diskstats") + if err != nil { + return 0, 0 + } + scanner = bufio.NewScanner(f) + for scanner.Scan() { + line := scanner.Text() + fields := strings.Fields(line) + if len(fields) < 14 { + continue + } + + deviceName := fields[2] + if deviceName == "sda" || deviceName == "vda" || deviceName == "nvme0n1" { + readSectors, _ := strconv.ParseUint(fields[5], 10, 64) + writeSectors, _ := strconv.ParseUint(fields[9], 10, 64) + return readSectors, writeSectors + } + } + + return 0, 0 +} + +// readTopProcesses 读取 Top 5 进程 (按 CPU 使用率) +func readTopProcesses() []ProcessInfo { + processes := []ProcessInfo{} + + // 读取系统总内存 + memInfo, _ := readMemory() + totalMemGB := float64(memInfo.TotalBytes) / 1024 / 1024 / 1024 + + // 使用 ps 命令获取进程信息,添加超时控制 + ctx, cancel := context.WithTimeout(context.Background(), 2*time.Second) + defer cancel() + cmd := exec.CommandContext(ctx, "ps", "aux", "--sort=-%cpu", "--no-headers") + out, err := cmd.Output() + if err != nil { + return processes + } + + lines := strings.Split(string(out), "\n") + count := 0 + + for _, line := range lines { + if count >= 5 { // 只取前5个 + break + } + + line = strings.TrimSpace(line) + if line == "" { + continue + } + + fields := strings.Fields(line) + if len(fields) < 11 { + continue + } + + pid, _ := strconv.Atoi(fields[1]) + cpu, _ := strconv.ParseFloat(fields[2], 64) + mem, _ := strconv.ParseFloat(fields[3], 64) + + // 计算内存MB数 + memoryMB := (mem / 100) * totalMemGB * 1024 + + // 命令可能包含空格,从第11个字段开始拼接 + command := strings.Join(fields[10:], " ") + if len(command) > 50 { + command = command[:50] + "..." + } + + processes = append(processes, ProcessInfo{ + PID: pid, + Name: fields[10], + CPU: round(cpu, 1), + Memory: round(mem, 1), + MemoryMB: round(memoryMB, 1), + Command: command, + }) + count++ + } + + return processes +} + +// readSystemLogs 读取系统最新日志 +func readSystemLogs(count int) []string { + logs := []string{} + + // 尝试使用 journalctl 读取系统日志 + if _, err := exec.LookPath("journalctl"); err == nil { + cmd := exec.Command("journalctl", "-n", strconv.Itoa(count), "--no-pager", "-o", "short") + out, err := cmd.Output() + if err == nil { + lines := strings.Split(strings.TrimSpace(string(out)), "\n") + for _, line := range lines { + if line != "" { + logs = append(logs, line) + } + } + return logs + } + } + + // 如果 journalctl 不可用,尝试读取 /var/log/syslog 或 /var/log/messages + logFiles := []string{"/var/log/syslog", "/var/log/messages"} + for _, logFile := range logFiles { + f, err := os.Open(logFile) + if err != nil { + continue + } + defer f.Close() + + // 读取最后几行 + var lines []string + scanner := bufio.NewScanner(f) + for scanner.Scan() { + lines = append(lines, scanner.Text()) + } + + // 取最后count行 + start := len(lines) - count + if start < 0 { + start = 0 + } + logs = lines[start:] + break + } + + return logs +} diff --git a/mengyamonitor-backend/systeminfo_linux.go b/mengyamonitor-backend/systeminfo_linux.go new file mode 100644 index 0000000..8b531ae --- /dev/null +++ b/mengyamonitor-backend/systeminfo_linux.go @@ -0,0 +1,113 @@ +//go:build linux +// +build linux + +package main + +import ( + "bufio" + "os" + "strings" + "syscall" +) + +func readStorage() ([]StorageMetrics, error) { + // For simplicity, report root mount. Can be extended to iterate mounts. + var stat syscall.Statfs_t + if err := syscall.Statfs("/", &stat); err != nil { + return nil, err + } + total := stat.Blocks * uint64(stat.Bsize) + free := stat.Bfree * uint64(stat.Bsize) + used := total - free + usedPercent := 0.0 + if total > 0 { + usedPercent = (float64(used) / float64(total)) * 100 + } + return []StorageMetrics{{ + Mount: "/", + TotalBytes: total, + UsedBytes: used, + FreeBytes: free, + UsedPercent: round(usedPercent, 2), + }}, nil +} + +// readAllStorage 读取所有挂载的存储设备 +func readAllStorage() ([]StorageMetrics, error) { + storages := []StorageMetrics{} + + // 读取 /proc/mounts 获取所有挂载点 + f, err := os.Open("/proc/mounts") + if err != nil { + return readStorage() // 降级到只读根目录 + } + defer f.Close() + + scanner := bufio.NewScanner(f) + seen := make(map[string]bool) + + for scanner.Scan() { + line := scanner.Text() + fields := strings.Fields(line) + if len(fields) < 3 { + continue + } + + device := fields[0] + mountPoint := fields[1] + fsType := fields[2] + + // 跳过虚拟文件系统 + if strings.HasPrefix(device, "/dev/") == false { + continue + } + + // 跳过特殊文件系统类型 + skipTypes := map[string]bool{ + "tmpfs": true, "devtmpfs": true, "squashfs": true, + "overlay": true, "aufs": true, "proc": true, + "sysfs": true, "devpts": true, "cgroup": true, + } + if skipTypes[fsType] { + continue + } + + // 避免重复挂载点 + if seen[mountPoint] { + continue + } + seen[mountPoint] = true + + // 获取该挂载点的统计信息 + var stat syscall.Statfs_t + if err := syscall.Statfs(mountPoint, &stat); err != nil { + continue + } + + total := stat.Blocks * uint64(stat.Bsize) + free := stat.Bfree * uint64(stat.Bsize) + used := total - free + usedPercent := 0.0 + if total > 0 { + usedPercent = (float64(used) / float64(total)) * 100 + } + + // 只添加有容量的存储 + if total > 0 { + storages = append(storages, StorageMetrics{ + Mount: mountPoint, + TotalBytes: total, + UsedBytes: used, + FreeBytes: free, + UsedPercent: round(usedPercent, 2), + }) + } + } + + // 如果没有找到任何存储,至少返回根目录 + if len(storages) == 0 { + return readStorage() + } + + return storages, nil +} diff --git a/mengyamonitor-backend/systeminfo_other.go b/mengyamonitor-backend/systeminfo_other.go new file mode 100644 index 0000000..f4c0924 --- /dev/null +++ b/mengyamonitor-backend/systeminfo_other.go @@ -0,0 +1,14 @@ +//go:build !linux +// +build !linux + +package main + +import "errors" + +func readStorage() ([]StorageMetrics, error) { + return nil, errors.New("storage monitoring is only supported on Linux") +} + +func readAllStorage() ([]StorageMetrics, error) { + return nil, errors.New("storage monitoring is only supported on Linux") +} diff --git a/mengyamonitor-backend/types.go b/mengyamonitor-backend/types.go new file mode 100644 index 0000000..c7a1885 --- /dev/null +++ b/mengyamonitor-backend/types.go @@ -0,0 +1,114 @@ +package main + +import "time" + +// CPU相关信息 +type CPUMetrics struct { + Model string `json:"model"` + Cores int `json:"cores"` + UsagePercent float64 `json:"usagePercent"` + LoadAverages []float64 `json:"loadAverages,omitempty"` + Temperature float64 `json:"temperature,omitempty"` // CPU温度(摄氏度) + PerCoreUsage []CoreUsage `json:"perCoreUsage,omitempty"` // 每个核心使用率 +} + +type CoreUsage struct { + Core int `json:"core"` + Percent float64 `json:"percent"` +} + +// 内存相关信息 +type MemoryMetrics struct { + TotalBytes uint64 `json:"totalBytes"` + UsedBytes uint64 `json:"usedBytes"` + FreeBytes uint64 `json:"freeBytes"` + UsedPercent float64 `json:"usedPercent"` +} + +// 储存相关信息 +type StorageMetrics struct { + Mount string `json:"mount"` + TotalBytes uint64 `json:"totalBytes"` + UsedBytes uint64 `json:"usedBytes"` + FreeBytes uint64 `json:"freeBytes"` + UsedPercent float64 `json:"usedPercent"` +} + +// GPU相关信息 +type GPUMetrics struct { + Name string `json:"name"` + MemoryTotalMB int64 `json:"memoryTotalMB"` + MemoryUsedMB int64 `json:"memoryUsedMB"` + UtilizationPercent float64 `json:"utilizationPercent"` + Temperature float64 `json:"temperature,omitempty"` // GPU温度(摄氏度) + Status string `json:"status"` +} + +// 网络相关信息 +type NetworkInterface struct { + Name string `json:"name"` + IPAddress string `json:"ipAddress"` + MACAddress string `json:"macAddress"` + RxBytes uint64 `json:"rxBytes"` // 接收字节数 + TxBytes uint64 `json:"txBytes"` // 发送字节数 + RxSpeed float64 `json:"rxSpeed"` // 接收速度 bytes/s + TxSpeed float64 `json:"txSpeed"` // 发送速度 bytes/s +} + +// 系统状态相关信息 +type SystemStats struct { + ProcessCount int `json:"processCount"` // 进程数量 + PackageCount int `json:"packageCount"` // 已安装软件包数量 + PackageManager string `json:"packageManager"` // 包管理器类型 + Temperature float64 `json:"temperature,omitempty"` // 系统温度(摄氏度) + DiskReadSpeed float64 `json:"diskReadSpeed"` // 磁盘读取速度 MB/s + DiskWriteSpeed float64 `json:"diskWriteSpeed"` // 磁盘写入速度 MB/s + NetworkRxSpeed float64 `json:"networkRxSpeed"` // 网络下载速度 MB/s + NetworkTxSpeed float64 `json:"networkTxSpeed"` // 网络上传速度 MB/s + TopProcesses []ProcessInfo `json:"topProcesses"` // Top 5 进程 + DockerStats DockerStats `json:"dockerStats"` // Docker 统计信息 + SystemLogs []string `json:"systemLogs"` // 系统最新日志 +} + +// 服务器进程相关信息 +type ProcessInfo struct { + PID int `json:"pid"` + Name string `json:"name"` + CPU float64 `json:"cpu"` + Memory float64 `json:"memory"` + MemoryMB float64 `json:"memoryMB"` // 内存占用MB + Command string `json:"command"` +} + +// Docker相关信息(简化版) +type DockerStats struct { + Available bool `json:"available"` // Docker是否可用 + Version string `json:"version"` // Docker版本 + Running int `json:"running"` // 运行中的容器数 + Stopped int `json:"stopped"` // 停止的容器数 + ImageCount int `json:"imageCount"` // 镜像数量 + RunningNames []string `json:"runningNames"` // 运行中的容器名列表 + StoppedNames []string `json:"stoppedNames"` // 停止的容器名列表 + ImageNames []string `json:"imageNames"` // 镜像名列表 +} + +// 操作系统相关信息 +type OSInfo struct { + Kernel string `json:"kernel"` + Distro string `json:"distro"` + Architecture string `json:"architecture"` +} + +// 所有注册指标 +type Metrics struct { + Hostname string `json:"hostname"` + Timestamp time.Time `json:"timestamp"` + CPU CPUMetrics `json:"cpu"` + Memory MemoryMetrics `json:"memory"` + Storage []StorageMetrics `json:"storage"` + GPU []GPUMetrics `json:"gpu"` + Network []NetworkInterface `json:"network"` + System SystemStats `json:"system"` + OS OSInfo `json:"os"` + UptimeSeconds float64 `json:"uptimeSeconds"` +} diff --git a/mengyamonitor-backend/utils.go b/mengyamonitor-backend/utils.go new file mode 100644 index 0000000..7a028e8 --- /dev/null +++ b/mengyamonitor-backend/utils.go @@ -0,0 +1,66 @@ +package main + +import ( + "bufio" + "math" + "os" + "strconv" + "strings" +) + +// round 将浮点数四舍五入到指定小数位 +func round(v float64, places int) float64 { + factor := math.Pow(10, float64(places)) + return math.Round(v*factor) / factor +} + +// readFirstLine 读取文件的第一行 +func readFirstLine(path string) string { + f, err := os.Open(path) + if err != nil { + return "" + } + defer f.Close() + scanner := bufio.NewScanner(f) + if scanner.Scan() { + return scanner.Text() + } + return "" +} + +// firstMatchInFile 在文件中查找第一个匹配指定前缀的行,并返回冒号后的内容 +func firstMatchInFile(path, prefix string) string { + f, err := os.Open(path) + if err != nil { + return "" + } + defer f.Close() + scanner := bufio.NewScanner(f) + for scanner.Scan() { + line := scanner.Text() + if strings.HasPrefix(strings.TrimSpace(line), prefix) { + parts := strings.SplitN(line, ":", 2) + if len(parts) == 2 { + return strings.TrimSpace(parts[1]) + } + } + } + return "" +} + +// readTempFromFile 从文件读取温度值 +func readTempFromFile(path string) float64 { + content := readFirstLine(path) + if content == "" { + return 0 + } + val, err := strconv.ParseFloat(strings.TrimSpace(content), 64) + if err != nil { + return 0 + } + // 温度通常以毫度为单位 + if val > 1000 { + return round(val/1000, 1) + } + return round(val, 1) +} diff --git a/mengyamonitor-frontend/README.md b/mengyamonitor-frontend/README.md new file mode 100644 index 0000000..d17fa10 --- /dev/null +++ b/mengyamonitor-frontend/README.md @@ -0,0 +1,110 @@ +# 萌芽监控面板 - 前端 + +## 概述 +服务器监控面板前端应用,使用 React + TypeScript + Vite 构建。 + +## 功能特性 +- 🖥️ 多服务器监控支持 +- 📊 实时数据展示(CPU、内存、存储、GPU) +- 🔄 自动刷新(2秒轮询) +- 📱 响应式设计 +- 🎨 白色柔和风格界面 +- 💾 本地存储配置 + +## 开发 + +### 安装依赖 +```bash +npm install +``` + +### 启动开发服务器 +```bash +npm run dev +``` +访问 http://localhost:2929 + +### 构建生产版本 +```bash +npm run build +``` +生产文件将输出到 `dist` 目录 + +### 预览生产版本 +```bash +npm run preview +``` + +## 使用说明 + +### 添加服务器 +1. 点击右上角"添加服务器"按钮 +2. 输入服务器名称(如:生产服务器1) +3. 输入服务器地址(如:http://192.168.1.100:9292) +4. 点击"添加"按钮 + +### 查看详情 +点击服务器卡片上的"查看详情"按钮,可以查看完整的系统信息: +- 系统信息(主机名、操作系统、内核、架构、运行时间) +- CPU 详细信息(型号、核心数、使用率、负载) +- 内存详细信息(总容量、已使用、可用、使用率) +- 存储详细信息(挂载点、容量、使用情况) +- GPU 信息(名称、显存、利用率) + +### 移除服务器 +点击服务器卡片右上角的"×"按钮,确认后即可移除。 + +## 项目结构 +``` +src/ +├── api/ # API 调用 +│ └── monitor.ts # 监控 API +├── components/ # React 组件 +│ ├── ServerCard/ # 服务器卡片组件 +│ └── ServerDetail/ # 服务器详情弹窗 +├── hooks/ # 自定义 Hooks +│ └── useServerMonitor.ts # 服务器监控 Hook +├── types/ # TypeScript 类型定义 +│ └── index.ts +├── utils/ # 工具函数 +│ ├── format.ts # 格式化函数 +│ └── storage.ts # 本地存储 +├── App.tsx # 主应用组件 +├── App.css # 主应用样式 +├── main.tsx # 应用入口 +└── index.css # 全局样式 +``` + +## 配置说明 + +### 修改端口 +编辑 `vite.config.ts` 文件中的 `server.port` 配置: +```typescript +server: { + port: 2929, // 修改为你想要的端口 +} +``` + +### 修改轮询间隔 +编辑 `src/App.tsx` 文件中的 `useServerMonitor` 第二个参数: +```typescript +const statuses = useServerMonitor(servers, 2000); // 2000ms = 2秒 +``` + +## 部署 + +### 静态文件部署 +1. 构建项目:`npm run build` +2. 将 `dist` 目录部署到 Web 服务器(如 Nginx、Apache) +3. 或使用静态托管服务(如 Vercel、Netlify) + +### 桌面应用 +可以使用 Electron 或 Tauri 将前端打包成桌面应用: +- Electron: https://www.electronjs.org/ +- Tauri: https://tauri.app/ + +## 注意事项 +- 确保后端服务器已启动并可访问 +- 后端服务器需要配置 CORS 允许跨域访问 +- 服务器地址需要包含协议(http:// 或 https://) +- 本地存储的服务器配置保存在浏览器 localStorage 中 diff --git a/mengyamonitor-frontend/eslint.config.js b/mengyamonitor-frontend/eslint.config.js new file mode 100644 index 0000000..5e6b472 --- /dev/null +++ b/mengyamonitor-frontend/eslint.config.js @@ -0,0 +1,23 @@ +import js from '@eslint/js' +import globals from 'globals' +import reactHooks from 'eslint-plugin-react-hooks' +import reactRefresh from 'eslint-plugin-react-refresh' +import tseslint from 'typescript-eslint' +import { defineConfig, globalIgnores } from 'eslint/config' + +export default defineConfig([ + globalIgnores(['dist']), + { + files: ['**/*.{ts,tsx}'], + extends: [ + js.configs.recommended, + tseslint.configs.recommended, + reactHooks.configs.flat.recommended, + reactRefresh.configs.vite, + ], + languageOptions: { + ecmaVersion: 2020, + globals: globals.browser, + }, + }, +]) diff --git a/mengyamonitor-frontend/index.html b/mengyamonitor-frontend/index.html new file mode 100644 index 0000000..26c2ebe --- /dev/null +++ b/mengyamonitor-frontend/index.html @@ -0,0 +1,13 @@ + + + + + + + 萌芽监控面板 + + +
+ + + diff --git a/mengyamonitor-frontend/package-lock.json b/mengyamonitor-frontend/package-lock.json new file mode 100644 index 0000000..30bd1ff --- /dev/null +++ b/mengyamonitor-frontend/package-lock.json @@ -0,0 +1,3224 @@ +{ + "name": "mengyamonitor-frontend", + "version": "0.0.0", + "lockfileVersion": 3, + "requires": true, + "packages": { + "": { + "name": "mengyamonitor-frontend", + "version": "0.0.0", + "dependencies": { + "react": "^19.2.0", + "react-dom": "^19.2.0" + }, + "devDependencies": { + "@eslint/js": "^9.39.1", + "@types/node": "^24.10.1", + "@types/react": "^19.2.5", + "@types/react-dom": "^19.2.3", + "@vitejs/plugin-react": "^5.1.1", + "eslint": "^9.39.1", + "eslint-plugin-react-hooks": "^7.0.1", + "eslint-plugin-react-refresh": "^0.4.24", + "globals": "^16.5.0", + "typescript": "~5.9.3", + "typescript-eslint": "^8.46.4", + "vite": "^7.2.4" + } + }, + "node_modules/@babel/code-frame": { + "version": "7.27.1", + "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.27.1.tgz", + "integrity": "sha512-cjQ7ZlQ0Mv3b47hABuTevyTuYN4i+loJKGeV9flcCgIK37cCXRh+L1bd3iBHlynerhQ7BhCkn2BPbQUL+rGqFg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-validator-identifier": "^7.27.1", + "js-tokens": "^4.0.0", + "picocolors": "^1.1.1" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/compat-data": { + "version": "7.28.5", + "resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.28.5.tgz", + "integrity": "sha512-6uFXyCayocRbqhZOB+6XcuZbkMNimwfVGFji8CTZnCzOHVGvDqzvitu1re2AU5LROliz7eQPhB8CpAMvnx9EjA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/core": { + "version": "7.28.5", + "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.28.5.tgz", + "integrity": "sha512-e7jT4DxYvIDLk1ZHmU/m/mB19rex9sv0c2ftBtjSBv+kVM/902eh0fINUzD7UwLLNR+jU585GxUJ8/EBfAM5fw==", + "dev": true, + "license": "MIT", + "peer": true, + "dependencies": { + "@babel/code-frame": "^7.27.1", + "@babel/generator": "^7.28.5", + "@babel/helper-compilation-targets": "^7.27.2", + "@babel/helper-module-transforms": "^7.28.3", + "@babel/helpers": "^7.28.4", + "@babel/parser": "^7.28.5", + "@babel/template": "^7.27.2", + "@babel/traverse": "^7.28.5", + "@babel/types": "^7.28.5", + "@jridgewell/remapping": "^2.3.5", + "convert-source-map": "^2.0.0", + "debug": "^4.1.0", + "gensync": "^1.0.0-beta.2", + "json5": "^2.2.3", + "semver": "^6.3.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/babel" + } + }, + "node_modules/@babel/generator": { + "version": "7.28.5", + "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.28.5.tgz", + "integrity": "sha512-3EwLFhZ38J4VyIP6WNtt2kUdW9dokXA9Cr4IVIFHuCpZ3H8/YFOl5JjZHisrn1fATPBmKKqXzDFvh9fUwHz6CQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/parser": "^7.28.5", + "@babel/types": "^7.28.5", + "@jridgewell/gen-mapping": "^0.3.12", + "@jridgewell/trace-mapping": "^0.3.28", + "jsesc": "^3.0.2" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-compilation-targets": { + "version": "7.27.2", + "resolved": "https://registry.npmjs.org/@babel/helper-compilation-targets/-/helper-compilation-targets-7.27.2.tgz", + "integrity": "sha512-2+1thGUUWWjLTYTHZWK1n8Yga0ijBz1XAhUXcKy81rd5g6yh7hGqMp45v7cadSbEHc9G3OTv45SyneRN3ps4DQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/compat-data": "^7.27.2", + "@babel/helper-validator-option": "^7.27.1", + "browserslist": "^4.24.0", + "lru-cache": "^5.1.1", + "semver": "^6.3.1" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-globals": { + "version": "7.28.0", + "resolved": "https://registry.npmjs.org/@babel/helper-globals/-/helper-globals-7.28.0.tgz", + "integrity": "sha512-+W6cISkXFa1jXsDEdYA8HeevQT/FULhxzR99pxphltZcVaugps53THCeiWA8SguxxpSp3gKPiuYfSWopkLQ4hw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-module-imports": { + "version": "7.27.1", + "resolved": "https://registry.npmjs.org/@babel/helper-module-imports/-/helper-module-imports-7.27.1.tgz", + "integrity": "sha512-0gSFWUPNXNopqtIPQvlD5WgXYI5GY2kP2cCvoT8kczjbfcfuIljTbcWrulD1CIPIX2gt1wghbDy08yE1p+/r3w==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/traverse": "^7.27.1", + "@babel/types": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-module-transforms": { + "version": "7.28.3", + "resolved": "https://registry.npmjs.org/@babel/helper-module-transforms/-/helper-module-transforms-7.28.3.tgz", + "integrity": "sha512-gytXUbs8k2sXS9PnQptz5o0QnpLL51SwASIORY6XaBKF88nsOT0Zw9szLqlSGQDP/4TljBAD5y98p2U1fqkdsw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-module-imports": "^7.27.1", + "@babel/helper-validator-identifier": "^7.27.1", + "@babel/traverse": "^7.28.3" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/@babel/helper-plugin-utils": { + "version": "7.27.1", + "resolved": "https://registry.npmjs.org/@babel/helper-plugin-utils/-/helper-plugin-utils-7.27.1.tgz", + "integrity": "sha512-1gn1Up5YXka3YYAHGKpbideQ5Yjf1tDa9qYcgysz+cNCXukyLl6DjPXhD3VRwSb8c0J9tA4b2+rHEZtc6R0tlw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-string-parser": { + "version": "7.27.1", + "resolved": "https://registry.npmjs.org/@babel/helper-string-parser/-/helper-string-parser-7.27.1.tgz", + "integrity": "sha512-qMlSxKbpRlAridDExk92nSobyDdpPijUq2DW6oDnUqd0iOGxmQjyqhMIihI9+zv4LPyZdRje2cavWPbCbWm3eA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-validator-identifier": { + "version": "7.28.5", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.28.5.tgz", + "integrity": "sha512-qSs4ifwzKJSV39ucNjsvc6WVHs6b7S03sOh2OcHF9UHfVPqWWALUsNUVzhSBiItjRZoLHx7nIarVjqKVusUZ1Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-validator-option": { + "version": "7.27.1", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-option/-/helper-validator-option-7.27.1.tgz", + "integrity": "sha512-YvjJow9FxbhFFKDSuFnVCe2WxXk1zWc22fFePVNEaWJEu8IrZVlda6N0uHwzZrUM1il7NC9Mlp4MaJYbYd9JSg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helpers": { + "version": "7.28.4", + "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.28.4.tgz", + "integrity": "sha512-HFN59MmQXGHVyYadKLVumYsA9dBFun/ldYxipEjzA4196jpLZd8UjEEBLkbEkvfYreDqJhZxYAWFPtrfhNpj4w==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/template": "^7.27.2", + "@babel/types": "^7.28.4" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/parser": { + "version": "7.28.5", + "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.28.5.tgz", + "integrity": "sha512-KKBU1VGYR7ORr3At5HAtUQ+TV3SzRCXmA/8OdDZiLDBIZxVyzXuztPjfLd3BV1PRAQGCMWWSHYhL0F8d5uHBDQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/types": "^7.28.5" + }, + "bin": { + "parser": "bin/babel-parser.js" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@babel/plugin-transform-react-jsx-self": { + "version": "7.27.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-react-jsx-self/-/plugin-transform-react-jsx-self-7.27.1.tgz", + "integrity": "sha512-6UzkCs+ejGdZ5mFFC/OCUrv028ab2fp1znZmCZjAOBKiBK2jXD1O+BPSfX8X2qjJ75fZBMSnQn3Rq2mrBJK2mw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-react-jsx-source": { + "version": "7.27.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-react-jsx-source/-/plugin-transform-react-jsx-source-7.27.1.tgz", + "integrity": "sha512-zbwoTsBruTeKB9hSq73ha66iFeJHuaFkUbwvqElnygoNbj/jHRsSeokowZFN3CZ64IvEqcmmkVe89OPXc7ldAw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/template": { + "version": "7.27.2", + "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.27.2.tgz", + "integrity": "sha512-LPDZ85aEJyYSd18/DkjNh4/y1ntkE5KwUHWTiqgRxruuZL2F1yuHligVHLvcHY2vMHXttKFpJn6LwfI7cw7ODw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/code-frame": "^7.27.1", + "@babel/parser": "^7.27.2", + "@babel/types": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/traverse": { + "version": "7.28.5", + "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.28.5.tgz", + "integrity": "sha512-TCCj4t55U90khlYkVV/0TfkJkAkUg3jZFA3Neb7unZT8CPok7iiRfaX0F+WnqWqt7OxhOn0uBKXCw4lbL8W0aQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/code-frame": "^7.27.1", + "@babel/generator": "^7.28.5", + "@babel/helper-globals": "^7.28.0", + "@babel/parser": "^7.28.5", + "@babel/template": "^7.27.2", + "@babel/types": "^7.28.5", + "debug": "^4.3.1" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/types": { + "version": "7.28.5", + "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.28.5.tgz", + "integrity": "sha512-qQ5m48eI/MFLQ5PxQj4PFaprjyCTLI37ElWMmNs0K8Lk3dVeOdNpB3ks8jc7yM5CDmVC73eMVk/trk3fgmrUpA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-string-parser": "^7.27.1", + "@babel/helper-validator-identifier": "^7.28.5" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@esbuild/aix-ppc64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/aix-ppc64/-/aix-ppc64-0.25.12.tgz", + "integrity": "sha512-Hhmwd6CInZ3dwpuGTF8fJG6yoWmsToE+vYgD4nytZVxcu1ulHpUQRAB1UJ8+N1Am3Mz4+xOByoQoSZf4D+CpkA==", + "cpu": [ + "ppc64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "aix" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/android-arm": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/android-arm/-/android-arm-0.25.12.tgz", + "integrity": "sha512-VJ+sKvNA/GE7Ccacc9Cha7bpS8nyzVv0jdVgwNDaR4gDMC/2TTRc33Ip8qrNYUcpkOHUT5OZ0bUcNNVZQ9RLlg==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/android-arm64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/android-arm64/-/android-arm64-0.25.12.tgz", + "integrity": "sha512-6AAmLG7zwD1Z159jCKPvAxZd4y/VTO0VkprYy+3N2FtJ8+BQWFXU+OxARIwA46c5tdD9SsKGZ/1ocqBS/gAKHg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/android-x64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/android-x64/-/android-x64-0.25.12.tgz", + "integrity": "sha512-5jbb+2hhDHx5phYR2By8GTWEzn6I9UqR11Kwf22iKbNpYrsmRB18aX/9ivc5cabcUiAT/wM+YIZ6SG9QO6a8kg==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/darwin-arm64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/darwin-arm64/-/darwin-arm64-0.25.12.tgz", + "integrity": "sha512-N3zl+lxHCifgIlcMUP5016ESkeQjLj/959RxxNYIthIg+CQHInujFuXeWbWMgnTo4cp5XVHqFPmpyu9J65C1Yg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/darwin-x64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/darwin-x64/-/darwin-x64-0.25.12.tgz", + "integrity": "sha512-HQ9ka4Kx21qHXwtlTUVbKJOAnmG1ipXhdWTmNXiPzPfWKpXqASVcWdnf2bnL73wgjNrFXAa3yYvBSd9pzfEIpA==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/freebsd-arm64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/freebsd-arm64/-/freebsd-arm64-0.25.12.tgz", + "integrity": "sha512-gA0Bx759+7Jve03K1S0vkOu5Lg/85dou3EseOGUes8flVOGxbhDDh/iZaoek11Y8mtyKPGF3vP8XhnkDEAmzeg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "freebsd" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/freebsd-x64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/freebsd-x64/-/freebsd-x64-0.25.12.tgz", + "integrity": "sha512-TGbO26Yw2xsHzxtbVFGEXBFH0FRAP7gtcPE7P5yP7wGy7cXK2oO7RyOhL5NLiqTlBh47XhmIUXuGciXEqYFfBQ==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "freebsd" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-arm": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/linux-arm/-/linux-arm-0.25.12.tgz", + "integrity": "sha512-lPDGyC1JPDou8kGcywY0YILzWlhhnRjdof3UlcoqYmS9El818LLfJJc3PXXgZHrHCAKs/Z2SeZtDJr5MrkxtOw==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-arm64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/linux-arm64/-/linux-arm64-0.25.12.tgz", + "integrity": "sha512-8bwX7a8FghIgrupcxb4aUmYDLp8pX06rGh5HqDT7bB+8Rdells6mHvrFHHW2JAOPZUbnjUpKTLg6ECyzvas2AQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-ia32": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/linux-ia32/-/linux-ia32-0.25.12.tgz", + "integrity": "sha512-0y9KrdVnbMM2/vG8KfU0byhUN+EFCny9+8g202gYqSSVMonbsCfLjUO+rCci7pM0WBEtz+oK/PIwHkzxkyharA==", + "cpu": [ + "ia32" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-loong64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/linux-loong64/-/linux-loong64-0.25.12.tgz", + "integrity": "sha512-h///Lr5a9rib/v1GGqXVGzjL4TMvVTv+s1DPoxQdz7l/AYv6LDSxdIwzxkrPW438oUXiDtwM10o9PmwS/6Z0Ng==", + "cpu": [ + "loong64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-mips64el": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/linux-mips64el/-/linux-mips64el-0.25.12.tgz", + "integrity": "sha512-iyRrM1Pzy9GFMDLsXn1iHUm18nhKnNMWscjmp4+hpafcZjrr2WbT//d20xaGljXDBYHqRcl8HnxbX6uaA/eGVw==", + "cpu": [ + "mips64el" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-ppc64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/linux-ppc64/-/linux-ppc64-0.25.12.tgz", + "integrity": "sha512-9meM/lRXxMi5PSUqEXRCtVjEZBGwB7P/D4yT8UG/mwIdze2aV4Vo6U5gD3+RsoHXKkHCfSxZKzmDssVlRj1QQA==", + "cpu": [ + "ppc64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-riscv64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/linux-riscv64/-/linux-riscv64-0.25.12.tgz", + "integrity": "sha512-Zr7KR4hgKUpWAwb1f3o5ygT04MzqVrGEGXGLnj15YQDJErYu/BGg+wmFlIDOdJp0PmB0lLvxFIOXZgFRrdjR0w==", + "cpu": [ + "riscv64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-s390x": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/linux-s390x/-/linux-s390x-0.25.12.tgz", + "integrity": "sha512-MsKncOcgTNvdtiISc/jZs/Zf8d0cl/t3gYWX8J9ubBnVOwlk65UIEEvgBORTiljloIWnBzLs4qhzPkJcitIzIg==", + "cpu": [ + "s390x" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-x64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/linux-x64/-/linux-x64-0.25.12.tgz", + "integrity": "sha512-uqZMTLr/zR/ed4jIGnwSLkaHmPjOjJvnm6TVVitAa08SLS9Z0VM8wIRx7gWbJB5/J54YuIMInDquWyYvQLZkgw==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/netbsd-arm64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/netbsd-arm64/-/netbsd-arm64-0.25.12.tgz", + "integrity": "sha512-xXwcTq4GhRM7J9A8Gv5boanHhRa/Q9KLVmcyXHCTaM4wKfIpWkdXiMog/KsnxzJ0A1+nD+zoecuzqPmCRyBGjg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "netbsd" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/netbsd-x64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/netbsd-x64/-/netbsd-x64-0.25.12.tgz", + "integrity": "sha512-Ld5pTlzPy3YwGec4OuHh1aCVCRvOXdH8DgRjfDy/oumVovmuSzWfnSJg+VtakB9Cm0gxNO9BzWkj6mtO1FMXkQ==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "netbsd" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/openbsd-arm64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/openbsd-arm64/-/openbsd-arm64-0.25.12.tgz", + "integrity": "sha512-fF96T6KsBo/pkQI950FARU9apGNTSlZGsv1jZBAlcLL1MLjLNIWPBkj5NlSz8aAzYKg+eNqknrUJ24QBybeR5A==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "openbsd" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/openbsd-x64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/openbsd-x64/-/openbsd-x64-0.25.12.tgz", + "integrity": "sha512-MZyXUkZHjQxUvzK7rN8DJ3SRmrVrke8ZyRusHlP+kuwqTcfWLyqMOE3sScPPyeIXN/mDJIfGXvcMqCgYKekoQw==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "openbsd" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/openharmony-arm64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/openharmony-arm64/-/openharmony-arm64-0.25.12.tgz", + "integrity": "sha512-rm0YWsqUSRrjncSXGA7Zv78Nbnw4XL6/dzr20cyrQf7ZmRcsovpcRBdhD43Nuk3y7XIoW2OxMVvwuRvk9XdASg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "openharmony" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/sunos-x64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/sunos-x64/-/sunos-x64-0.25.12.tgz", + "integrity": "sha512-3wGSCDyuTHQUzt0nV7bocDy72r2lI33QL3gkDNGkod22EsYl04sMf0qLb8luNKTOmgF/eDEDP5BFNwoBKH441w==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "sunos" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/win32-arm64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/win32-arm64/-/win32-arm64-0.25.12.tgz", + "integrity": "sha512-rMmLrur64A7+DKlnSuwqUdRKyd3UE7oPJZmnljqEptesKM8wx9J8gx5u0+9Pq0fQQW8vqeKebwNXdfOyP+8Bsg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/win32-ia32": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/win32-ia32/-/win32-ia32-0.25.12.tgz", + "integrity": "sha512-HkqnmmBoCbCwxUKKNPBixiWDGCpQGVsrQfJoVGYLPT41XWF8lHuE5N6WhVia2n4o5QK5M4tYr21827fNhi4byQ==", + "cpu": [ + "ia32" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/win32-x64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/win32-x64/-/win32-x64-0.25.12.tgz", + "integrity": "sha512-alJC0uCZpTFrSL0CCDjcgleBXPnCrEAhTBILpeAp7M/OFgoqtAetfBzX0xM00MUsVVPpVjlPuMbREqnZCXaTnA==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@eslint-community/eslint-utils": { + "version": "4.9.0", + "resolved": "https://registry.npmjs.org/@eslint-community/eslint-utils/-/eslint-utils-4.9.0.tgz", + "integrity": "sha512-ayVFHdtZ+hsq1t2Dy24wCmGXGe4q9Gu3smhLYALJrr473ZH27MsnSL+LKUlimp4BWJqMDMLmPpx/Q9R3OAlL4g==", + "dev": true, + "license": "MIT", + "dependencies": { + "eslint-visitor-keys": "^3.4.3" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + }, + "peerDependencies": { + "eslint": "^6.0.0 || ^7.0.0 || >=8.0.0" + } + }, + "node_modules/@eslint-community/eslint-utils/node_modules/eslint-visitor-keys": { + "version": "3.4.3", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-3.4.3.tgz", + "integrity": "sha512-wpc+LXeiyiisxPlEkUzU6svyS1frIO3Mgxj1fdy7Pm8Ygzguax2N3Fa/D/ag1WqbOprdI+uY6wMUl8/a2G+iag==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/@eslint-community/regexpp": { + "version": "4.12.2", + "resolved": "https://registry.npmjs.org/@eslint-community/regexpp/-/regexpp-4.12.2.tgz", + "integrity": "sha512-EriSTlt5OC9/7SXkRSCAhfSxxoSUgBm33OH+IkwbdpgoqsSsUg7y3uh+IICI/Qg4BBWr3U2i39RpmycbxMq4ew==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^12.0.0 || ^14.0.0 || >=16.0.0" + } + }, + "node_modules/@eslint/config-array": { + "version": "0.21.1", + "resolved": "https://registry.npmjs.org/@eslint/config-array/-/config-array-0.21.1.tgz", + "integrity": "sha512-aw1gNayWpdI/jSYVgzN5pL0cfzU02GT3NBpeT/DXbx1/1x7ZKxFPd9bwrzygx/qiwIQiJ1sw/zD8qY/kRvlGHA==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "@eslint/object-schema": "^2.1.7", + "debug": "^4.3.1", + "minimatch": "^3.1.2" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + } + }, + "node_modules/@eslint/config-helpers": { + "version": "0.4.2", + "resolved": "https://registry.npmjs.org/@eslint/config-helpers/-/config-helpers-0.4.2.tgz", + "integrity": "sha512-gBrxN88gOIf3R7ja5K9slwNayVcZgK6SOUORm2uBzTeIEfeVaIhOpCtTox3P6R7o2jLFwLFTLnC7kU/RGcYEgw==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "@eslint/core": "^0.17.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + } + }, + "node_modules/@eslint/core": { + "version": "0.17.0", + "resolved": "https://registry.npmjs.org/@eslint/core/-/core-0.17.0.tgz", + "integrity": "sha512-yL/sLrpmtDaFEiUj1osRP4TI2MDz1AddJL+jZ7KSqvBuliN4xqYY54IfdN8qD8Toa6g1iloph1fxQNkjOxrrpQ==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "@types/json-schema": "^7.0.15" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + } + }, + "node_modules/@eslint/eslintrc": { + "version": "3.3.3", + "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-3.3.3.tgz", + "integrity": "sha512-Kr+LPIUVKz2qkx1HAMH8q1q6azbqBAsXJUxBl/ODDuVPX45Z9DfwB8tPjTi6nNZ8BuM3nbJxC5zCAg5elnBUTQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "ajv": "^6.12.4", + "debug": "^4.3.2", + "espree": "^10.0.1", + "globals": "^14.0.0", + "ignore": "^5.2.0", + "import-fresh": "^3.2.1", + "js-yaml": "^4.1.1", + "minimatch": "^3.1.2", + "strip-json-comments": "^3.1.1" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/@eslint/eslintrc/node_modules/globals": { + "version": "14.0.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-14.0.0.tgz", + "integrity": "sha512-oahGvuMGQlPw/ivIYBjVSrWAfWLBeku5tpPE2fOPLi+WHffIWbuh2tCjhyQhTBPMf5E9jDEH4FOmTYgYwbKwtQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=18" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/@eslint/js": { + "version": "9.39.1", + "resolved": "https://registry.npmjs.org/@eslint/js/-/js-9.39.1.tgz", + "integrity": "sha512-S26Stp4zCy88tH94QbBv3XCuzRQiZ9yXofEILmglYTh/Ug/a9/umqvgFtYBAo3Lp0nsI/5/qH1CCrbdK3AP1Tw==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "url": "https://eslint.org/donate" + } + }, + "node_modules/@eslint/object-schema": { + "version": "2.1.7", + "resolved": "https://registry.npmjs.org/@eslint/object-schema/-/object-schema-2.1.7.tgz", + "integrity": "sha512-VtAOaymWVfZcmZbp6E2mympDIHvyjXs/12LqWYjVw6qjrfF+VK+fyG33kChz3nnK+SU5/NeHOqrTEHS8sXO3OA==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + } + }, + "node_modules/@eslint/plugin-kit": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/@eslint/plugin-kit/-/plugin-kit-0.4.1.tgz", + "integrity": "sha512-43/qtrDUokr7LJqoF2c3+RInu/t4zfrpYdoSDfYyhg52rwLV6TnOvdG4fXm7IkSB3wErkcmJS9iEhjVtOSEjjA==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "@eslint/core": "^0.17.0", + "levn": "^0.4.1" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + } + }, + "node_modules/@humanfs/core": { + "version": "0.19.1", + "resolved": "https://registry.npmjs.org/@humanfs/core/-/core-0.19.1.tgz", + "integrity": "sha512-5DyQ4+1JEUzejeK1JGICcideyfUbGixgS9jNgex5nqkW+cY7WZhxBigmieN5Qnw9ZosSNVC9KQKyb+GUaGyKUA==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": ">=18.18.0" + } + }, + "node_modules/@humanfs/node": { + "version": "0.16.7", + "resolved": "https://registry.npmjs.org/@humanfs/node/-/node-0.16.7.tgz", + "integrity": "sha512-/zUx+yOsIrG4Y43Eh2peDeKCxlRt/gET6aHfaKpuq267qXdYDFViVHfMaLyygZOnl0kGWxFIgsBy8QFuTLUXEQ==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "@humanfs/core": "^0.19.1", + "@humanwhocodes/retry": "^0.4.0" + }, + "engines": { + "node": ">=18.18.0" + } + }, + "node_modules/@humanwhocodes/module-importer": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/@humanwhocodes/module-importer/-/module-importer-1.0.1.tgz", + "integrity": "sha512-bxveV4V8v5Yb4ncFTT3rPSgZBOpCkjfK0y4oVVVJwIuDVBRMDXrPyXRL988i5ap9m9bnyEEjWfm5WkBmtffLfA==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": ">=12.22" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/nzakas" + } + }, + "node_modules/@humanwhocodes/retry": { + "version": "0.4.3", + "resolved": "https://registry.npmjs.org/@humanwhocodes/retry/-/retry-0.4.3.tgz", + "integrity": "sha512-bV0Tgo9K4hfPCek+aMAn81RppFKv2ySDQeMoSZuvTASywNTnVJCArCZE2FWqpvIatKu7VMRLWlR1EazvVhDyhQ==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": ">=18.18" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/nzakas" + } + }, + "node_modules/@jridgewell/gen-mapping": { + "version": "0.3.13", + "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.13.tgz", + "integrity": "sha512-2kkt/7niJ6MgEPxF0bYdQ6etZaA+fQvDcLKckhy1yIQOzaoKjBBjSj63/aLVjYE3qhRt5dvM+uUyfCg6UKCBbA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jridgewell/sourcemap-codec": "^1.5.0", + "@jridgewell/trace-mapping": "^0.3.24" + } + }, + "node_modules/@jridgewell/remapping": { + "version": "2.3.5", + "resolved": "https://registry.npmjs.org/@jridgewell/remapping/-/remapping-2.3.5.tgz", + "integrity": "sha512-LI9u/+laYG4Ds1TDKSJW2YPrIlcVYOwi2fUC6xB43lueCjgxV4lffOCZCtYFiH6TNOX+tQKXx97T4IKHbhyHEQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jridgewell/gen-mapping": "^0.3.5", + "@jridgewell/trace-mapping": "^0.3.24" + } + }, + "node_modules/@jridgewell/resolve-uri": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/@jridgewell/resolve-uri/-/resolve-uri-3.1.2.tgz", + "integrity": "sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@jridgewell/sourcemap-codec": { + "version": "1.5.5", + "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.5.5.tgz", + "integrity": "sha512-cYQ9310grqxueWbl+WuIUIaiUaDcj7WOq5fVhEljNVgRfOUhY9fy2zTvfoqWsnebh8Sl70VScFbICvJnLKB0Og==", + "dev": true, + "license": "MIT" + }, + "node_modules/@jridgewell/trace-mapping": { + "version": "0.3.31", + "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.31.tgz", + "integrity": "sha512-zzNR+SdQSDJzc8joaeP8QQoCQr8NuYx2dIIytl1QeBEZHJ9uW6hebsrYgbz8hJwUQao3TWCMtmfV8Nu1twOLAw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jridgewell/resolve-uri": "^3.1.0", + "@jridgewell/sourcemap-codec": "^1.4.14" + } + }, + "node_modules/@rolldown/pluginutils": { + "version": "1.0.0-beta.53", + "resolved": "https://registry.npmjs.org/@rolldown/pluginutils/-/pluginutils-1.0.0-beta.53.tgz", + "integrity": "sha512-vENRlFU4YbrwVqNDZ7fLvy+JR1CRkyr01jhSiDpE1u6py3OMzQfztQU2jxykW3ALNxO4kSlqIDeYyD0Y9RcQeQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/@rollup/rollup-android-arm-eabi": { + "version": "4.53.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.53.3.tgz", + "integrity": "sha512-mRSi+4cBjrRLoaal2PnqH82Wqyb+d3HsPUN/W+WslCXsZsyHa9ZeQQX/pQsZaVIWDkPcpV6jJ+3KLbTbgnwv8w==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ] + }, + "node_modules/@rollup/rollup-android-arm64": { + "version": "4.53.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.53.3.tgz", + "integrity": "sha512-CbDGaMpdE9sh7sCmTrTUyllhrg65t6SwhjlMJsLr+J8YjFuPmCEjbBSx4Z/e4SmDyH3aB5hGaJUP2ltV/vcs4w==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ] + }, + "node_modules/@rollup/rollup-darwin-arm64": { + "version": "4.53.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.53.3.tgz", + "integrity": "sha512-Nr7SlQeqIBpOV6BHHGZgYBuSdanCXuw09hon14MGOLGmXAFYjx1wNvquVPmpZnl0tLjg25dEdr4IQ6GgyToCUA==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ] + }, + "node_modules/@rollup/rollup-darwin-x64": { + "version": "4.53.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.53.3.tgz", + "integrity": "sha512-DZ8N4CSNfl965CmPktJ8oBnfYr3F8dTTNBQkRlffnUarJ2ohudQD17sZBa097J8xhQ26AwhHJ5mvUyQW8ddTsQ==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ] + }, + "node_modules/@rollup/rollup-freebsd-arm64": { + "version": "4.53.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-arm64/-/rollup-freebsd-arm64-4.53.3.tgz", + "integrity": "sha512-yMTrCrK92aGyi7GuDNtGn2sNW+Gdb4vErx4t3Gv/Tr+1zRb8ax4z8GWVRfr3Jw8zJWvpGHNpss3vVlbF58DZ4w==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "freebsd" + ] + }, + "node_modules/@rollup/rollup-freebsd-x64": { + "version": "4.53.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-x64/-/rollup-freebsd-x64-4.53.3.tgz", + "integrity": "sha512-lMfF8X7QhdQzseM6XaX0vbno2m3hlyZFhwcndRMw8fbAGUGL3WFMBdK0hbUBIUYcEcMhVLr1SIamDeuLBnXS+Q==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "freebsd" + ] + }, + "node_modules/@rollup/rollup-linux-arm-gnueabihf": { + "version": "4.53.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.53.3.tgz", + "integrity": "sha512-k9oD15soC/Ln6d2Wv/JOFPzZXIAIFLp6B+i14KhxAfnq76ajt0EhYc5YPeX6W1xJkAdItcVT+JhKl1QZh44/qw==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-arm-musleabihf": { + "version": "4.53.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-musleabihf/-/rollup-linux-arm-musleabihf-4.53.3.tgz", + "integrity": "sha512-vTNlKq+N6CK/8UktsrFuc+/7NlEYVxgaEgRXVUVK258Z5ymho29skzW1sutgYjqNnquGwVUObAaxae8rZ6YMhg==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-arm64-gnu": { + "version": "4.53.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.53.3.tgz", + "integrity": "sha512-RGrFLWgMhSxRs/EWJMIFM1O5Mzuz3Xy3/mnxJp/5cVhZ2XoCAxJnmNsEyeMJtpK+wu0FJFWz+QF4mjCA7AUQ3w==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-arm64-musl": { + "version": "4.53.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.53.3.tgz", + "integrity": "sha512-kASyvfBEWYPEwe0Qv4nfu6pNkITLTb32p4yTgzFCocHnJLAHs+9LjUu9ONIhvfT/5lv4YS5muBHyuV84epBo/A==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-loong64-gnu": { + "version": "4.53.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-loong64-gnu/-/rollup-linux-loong64-gnu-4.53.3.tgz", + "integrity": "sha512-JiuKcp2teLJwQ7vkJ95EwESWkNRFJD7TQgYmCnrPtlu50b4XvT5MOmurWNrCj3IFdyjBQ5p9vnrX4JM6I8OE7g==", + "cpu": [ + "loong64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-ppc64-gnu": { + "version": "4.53.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-ppc64-gnu/-/rollup-linux-ppc64-gnu-4.53.3.tgz", + "integrity": "sha512-EoGSa8nd6d3T7zLuqdojxC20oBfNT8nexBbB/rkxgKj5T5vhpAQKKnD+h3UkoMuTyXkP5jTjK/ccNRmQrPNDuw==", + "cpu": [ + "ppc64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-riscv64-gnu": { + "version": "4.53.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.53.3.tgz", + "integrity": "sha512-4s+Wped2IHXHPnAEbIB0YWBv7SDohqxobiiPA1FIWZpX+w9o2i4LezzH/NkFUl8LRci/8udci6cLq+jJQlh+0g==", + "cpu": [ + "riscv64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-riscv64-musl": { + "version": "4.53.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-musl/-/rollup-linux-riscv64-musl-4.53.3.tgz", + "integrity": "sha512-68k2g7+0vs2u9CxDt5ktXTngsxOQkSEV/xBbwlqYcUrAVh6P9EgMZvFsnHy4SEiUl46Xf0IObWVbMvPrr2gw8A==", + "cpu": [ + "riscv64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-s390x-gnu": { + "version": "4.53.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-s390x-gnu/-/rollup-linux-s390x-gnu-4.53.3.tgz", + "integrity": "sha512-VYsFMpULAz87ZW6BVYw3I6sWesGpsP9OPcyKe8ofdg9LHxSbRMd7zrVrr5xi/3kMZtpWL/wC+UIJWJYVX5uTKg==", + "cpu": [ + "s390x" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-x64-gnu": { + "version": "4.53.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.53.3.tgz", + "integrity": "sha512-3EhFi1FU6YL8HTUJZ51imGJWEX//ajQPfqWLI3BQq4TlvHy4X0MOr5q3D2Zof/ka0d5FNdPwZXm3Yyib/UEd+w==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-x64-musl": { + "version": "4.53.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.53.3.tgz", + "integrity": "sha512-eoROhjcc6HbZCJr+tvVT8X4fW3/5g/WkGvvmwz/88sDtSJzO7r/blvoBDgISDiCjDRZmHpwud7h+6Q9JxFwq1Q==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-openharmony-arm64": { + "version": "4.53.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-openharmony-arm64/-/rollup-openharmony-arm64-4.53.3.tgz", + "integrity": "sha512-OueLAWgrNSPGAdUdIjSWXw+u/02BRTcnfw9PN41D2vq/JSEPnJnVuBgw18VkN8wcd4fjUs+jFHVM4t9+kBSNLw==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "openharmony" + ] + }, + "node_modules/@rollup/rollup-win32-arm64-msvc": { + "version": "4.53.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.53.3.tgz", + "integrity": "sha512-GOFuKpsxR/whszbF/bzydebLiXIHSgsEUp6M0JI8dWvi+fFa1TD6YQa4aSZHtpmh2/uAlj/Dy+nmby3TJ3pkTw==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ] + }, + "node_modules/@rollup/rollup-win32-ia32-msvc": { + "version": "4.53.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.53.3.tgz", + "integrity": "sha512-iah+THLcBJdpfZ1TstDFbKNznlzoxa8fmnFYK4V67HvmuNYkVdAywJSoteUszvBQ9/HqN2+9AZghbajMsFT+oA==", + "cpu": [ + "ia32" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ] + }, + "node_modules/@rollup/rollup-win32-x64-gnu": { + "version": "4.53.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-gnu/-/rollup-win32-x64-gnu-4.53.3.tgz", + "integrity": "sha512-J9QDiOIZlZLdcot5NXEepDkstocktoVjkaKUtqzgzpt2yWjGlbYiKyp05rWwk4nypbYUNoFAztEgixoLaSETkg==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ] + }, + "node_modules/@rollup/rollup-win32-x64-msvc": { + "version": "4.53.3", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.53.3.tgz", + "integrity": "sha512-UhTd8u31dXadv0MopwGgNOBpUVROFKWVQgAg5N1ESyCz8AuBcMqm4AuTjrwgQKGDfoFuz02EuMRHQIw/frmYKQ==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ] + }, + "node_modules/@types/babel__core": { + "version": "7.20.5", + "resolved": "https://registry.npmjs.org/@types/babel__core/-/babel__core-7.20.5.tgz", + "integrity": "sha512-qoQprZvz5wQFJwMDqeseRXWv3rqMvhgpbXFfVyWhbx9X47POIA6i/+dXefEmZKoAgOaTdaIgNSMqMIU61yRyzA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/parser": "^7.20.7", + "@babel/types": "^7.20.7", + "@types/babel__generator": "*", + "@types/babel__template": "*", + "@types/babel__traverse": "*" + } + }, + "node_modules/@types/babel__generator": { + "version": "7.27.0", + "resolved": "https://registry.npmjs.org/@types/babel__generator/-/babel__generator-7.27.0.tgz", + "integrity": "sha512-ufFd2Xi92OAVPYsy+P4n7/U7e68fex0+Ee8gSG9KX7eo084CWiQ4sdxktvdl0bOPupXtVJPY19zk6EwWqUQ8lg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/types": "^7.0.0" + } + }, + "node_modules/@types/babel__template": { + "version": "7.4.4", + "resolved": "https://registry.npmjs.org/@types/babel__template/-/babel__template-7.4.4.tgz", + "integrity": "sha512-h/NUaSyG5EyxBIp8YRxo4RMe2/qQgvyowRwVMzhYhBCONbW8PUsg4lkFMrhgZhUe5z3L3MiLDuvyJ/CaPa2A8A==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/parser": "^7.1.0", + "@babel/types": "^7.0.0" + } + }, + "node_modules/@types/babel__traverse": { + "version": "7.28.0", + "resolved": "https://registry.npmjs.org/@types/babel__traverse/-/babel__traverse-7.28.0.tgz", + "integrity": "sha512-8PvcXf70gTDZBgt9ptxJ8elBeBjcLOAcOtoO/mPJjtji1+CdGbHgm77om1GrsPxsiE+uXIpNSK64UYaIwQXd4Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/types": "^7.28.2" + } + }, + "node_modules/@types/estree": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.8.tgz", + "integrity": "sha512-dWHzHa2WqEXI/O1E9OjrocMTKJl2mSrEolh1Iomrv6U+JuNwaHXsXx9bLu5gG7BUWFIN0skIQJQ/L1rIex4X6w==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/json-schema": { + "version": "7.0.15", + "resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.15.tgz", + "integrity": "sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/node": { + "version": "24.10.2", + "resolved": "https://registry.npmjs.org/@types/node/-/node-24.10.2.tgz", + "integrity": "sha512-WOhQTZ4G8xZ1tjJTvKOpyEVSGgOTvJAfDK3FNFgELyaTpzhdgHVHeqW8V+UJvzF5BT+/B54T/1S2K6gd9c7bbA==", + "dev": true, + "license": "MIT", + "peer": true, + "dependencies": { + "undici-types": "~7.16.0" + } + }, + "node_modules/@types/react": { + "version": "19.2.7", + "resolved": "https://registry.npmjs.org/@types/react/-/react-19.2.7.tgz", + "integrity": "sha512-MWtvHrGZLFttgeEj28VXHxpmwYbor/ATPYbBfSFZEIRK0ecCFLl2Qo55z52Hss+UV9CRN7trSeq1zbgx7YDWWg==", + "dev": true, + "license": "MIT", + "peer": true, + "dependencies": { + "csstype": "^3.2.2" + } + }, + "node_modules/@types/react-dom": { + "version": "19.2.3", + "resolved": "https://registry.npmjs.org/@types/react-dom/-/react-dom-19.2.3.tgz", + "integrity": "sha512-jp2L/eY6fn+KgVVQAOqYItbF0VY/YApe5Mz2F0aykSO8gx31bYCZyvSeYxCHKvzHG5eZjc+zyaS5BrBWya2+kQ==", + "dev": true, + "license": "MIT", + "peerDependencies": { + "@types/react": "^19.2.0" + } + }, + "node_modules/@typescript-eslint/eslint-plugin": { + "version": "8.49.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-8.49.0.tgz", + "integrity": "sha512-JXij0vzIaTtCwu6SxTh8qBc66kmf1xs7pI4UOiMDFVct6q86G0Zs7KRcEoJgY3Cav3x5Tq0MF5jwgpgLqgKG3A==", + "dev": true, + "license": "MIT", + "dependencies": { + "@eslint-community/regexpp": "^4.10.0", + "@typescript-eslint/scope-manager": "8.49.0", + "@typescript-eslint/type-utils": "8.49.0", + "@typescript-eslint/utils": "8.49.0", + "@typescript-eslint/visitor-keys": "8.49.0", + "ignore": "^7.0.0", + "natural-compare": "^1.4.0", + "ts-api-utils": "^2.1.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "@typescript-eslint/parser": "^8.49.0", + "eslint": "^8.57.0 || ^9.0.0", + "typescript": ">=4.8.4 <6.0.0" + } + }, + "node_modules/@typescript-eslint/eslint-plugin/node_modules/ignore": { + "version": "7.0.5", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-7.0.5.tgz", + "integrity": "sha512-Hs59xBNfUIunMFgWAbGX5cq6893IbWg4KnrjbYwX3tx0ztorVgTDA6B2sxf8ejHJ4wz8BqGUMYlnzNBer5NvGg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 4" + } + }, + "node_modules/@typescript-eslint/parser": { + "version": "8.49.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-8.49.0.tgz", + "integrity": "sha512-N9lBGA9o9aqb1hVMc9hzySbhKibHmB+N3IpoShyV6HyQYRGIhlrO5rQgttypi+yEeKsKI4idxC8Jw6gXKD4THA==", + "dev": true, + "license": "MIT", + "peer": true, + "dependencies": { + "@typescript-eslint/scope-manager": "8.49.0", + "@typescript-eslint/types": "8.49.0", + "@typescript-eslint/typescript-estree": "8.49.0", + "@typescript-eslint/visitor-keys": "8.49.0", + "debug": "^4.3.4" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "eslint": "^8.57.0 || ^9.0.0", + "typescript": ">=4.8.4 <6.0.0" + } + }, + "node_modules/@typescript-eslint/project-service": { + "version": "8.49.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/project-service/-/project-service-8.49.0.tgz", + "integrity": "sha512-/wJN0/DKkmRUMXjZUXYZpD1NEQzQAAn9QWfGwo+Ai8gnzqH7tvqS7oNVdTjKqOcPyVIdZdyCMoqN66Ia789e7g==", + "dev": true, + "license": "MIT", + "dependencies": { + "@typescript-eslint/tsconfig-utils": "^8.49.0", + "@typescript-eslint/types": "^8.49.0", + "debug": "^4.3.4" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "typescript": ">=4.8.4 <6.0.0" + } + }, + "node_modules/@typescript-eslint/scope-manager": { + "version": "8.49.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-8.49.0.tgz", + "integrity": "sha512-npgS3zi+/30KSOkXNs0LQXtsg9ekZ8OISAOLGWA/ZOEn0ZH74Ginfl7foziV8DT+D98WfQ5Kopwqb/PZOaIJGg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@typescript-eslint/types": "8.49.0", + "@typescript-eslint/visitor-keys": "8.49.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + } + }, + "node_modules/@typescript-eslint/tsconfig-utils": { + "version": "8.49.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/tsconfig-utils/-/tsconfig-utils-8.49.0.tgz", + "integrity": "sha512-8prixNi1/6nawsRYxet4YOhnbW+W9FK/bQPxsGB1D3ZrDzbJ5FXw5XmzxZv82X3B+ZccuSxo/X8q9nQ+mFecWA==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "typescript": ">=4.8.4 <6.0.0" + } + }, + "node_modules/@typescript-eslint/type-utils": { + "version": "8.49.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/type-utils/-/type-utils-8.49.0.tgz", + "integrity": "sha512-KTExJfQ+svY8I10P4HdxKzWsvtVnsuCifU5MvXrRwoP2KOlNZ9ADNEWWsQTJgMxLzS5VLQKDjkCT/YzgsnqmZg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@typescript-eslint/types": "8.49.0", + "@typescript-eslint/typescript-estree": "8.49.0", + "@typescript-eslint/utils": "8.49.0", + "debug": "^4.3.4", + "ts-api-utils": "^2.1.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "eslint": "^8.57.0 || ^9.0.0", + "typescript": ">=4.8.4 <6.0.0" + } + }, + "node_modules/@typescript-eslint/types": { + "version": "8.49.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-8.49.0.tgz", + "integrity": "sha512-e9k/fneezorUo6WShlQpMxXh8/8wfyc+biu6tnAqA81oWrEic0k21RHzP9uqqpyBBeBKu4T+Bsjy9/b8u7obXQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + } + }, + "node_modules/@typescript-eslint/typescript-estree": { + "version": "8.49.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-8.49.0.tgz", + "integrity": "sha512-jrLdRuAbPfPIdYNppHJ/D0wN+wwNfJ32YTAm10eJVsFmrVpXQnDWBn8niCSMlWjvml8jsce5E/O+86IQtTbJWA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@typescript-eslint/project-service": "8.49.0", + "@typescript-eslint/tsconfig-utils": "8.49.0", + "@typescript-eslint/types": "8.49.0", + "@typescript-eslint/visitor-keys": "8.49.0", + "debug": "^4.3.4", + "minimatch": "^9.0.4", + "semver": "^7.6.0", + "tinyglobby": "^0.2.15", + "ts-api-utils": "^2.1.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "typescript": ">=4.8.4 <6.0.0" + } + }, + "node_modules/@typescript-eslint/typescript-estree/node_modules/brace-expansion": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.0.2.tgz", + "integrity": "sha512-Jt0vHyM+jmUBqojB7E1NIYadt0vI0Qxjxd2TErW94wDz+E2LAm5vKMXXwg6ZZBTHPuUlDgQHKXvjGBdfcF1ZDQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "balanced-match": "^1.0.0" + } + }, + "node_modules/@typescript-eslint/typescript-estree/node_modules/minimatch": { + "version": "9.0.5", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-9.0.5.tgz", + "integrity": "sha512-G6T0ZX48xgozx7587koeX9Ys2NYy6Gmv//P89sEte9V9whIapMNF4idKxnW2QtCcLiTWlb/wfCabAtAFWhhBow==", + "dev": true, + "license": "ISC", + "dependencies": { + "brace-expansion": "^2.0.1" + }, + "engines": { + "node": ">=16 || 14 >=14.17" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/@typescript-eslint/typescript-estree/node_modules/semver": { + "version": "7.7.3", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.7.3.tgz", + "integrity": "sha512-SdsKMrI9TdgjdweUSR9MweHA4EJ8YxHn8DFaDisvhVlUOe4BF1tLD7GAj0lIqWVl+dPb/rExr0Btby5loQm20Q==", + "dev": true, + "license": "ISC", + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@typescript-eslint/utils": { + "version": "8.49.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-8.49.0.tgz", + "integrity": "sha512-N3W7rJw7Rw+z1tRsHZbK395TWSYvufBXumYtEGzypgMUthlg0/hmCImeA8hgO2d2G4pd7ftpxxul2J8OdtdaFA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@eslint-community/eslint-utils": "^4.7.0", + "@typescript-eslint/scope-manager": "8.49.0", + "@typescript-eslint/types": "8.49.0", + "@typescript-eslint/typescript-estree": "8.49.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "eslint": "^8.57.0 || ^9.0.0", + "typescript": ">=4.8.4 <6.0.0" + } + }, + "node_modules/@typescript-eslint/visitor-keys": { + "version": "8.49.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-8.49.0.tgz", + "integrity": "sha512-LlKaciDe3GmZFphXIc79THF/YYBugZ7FS1pO581E/edlVVNbZKDy93evqmrfQ9/Y4uN0vVhX4iuchq26mK/iiA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@typescript-eslint/types": "8.49.0", + "eslint-visitor-keys": "^4.2.1" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + } + }, + "node_modules/@vitejs/plugin-react": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/@vitejs/plugin-react/-/plugin-react-5.1.2.tgz", + "integrity": "sha512-EcA07pHJouywpzsoTUqNh5NwGayl2PPVEJKUSinGGSxFGYn+shYbqMGBg6FXDqgXum9Ou/ecb+411ssw8HImJQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/core": "^7.28.5", + "@babel/plugin-transform-react-jsx-self": "^7.27.1", + "@babel/plugin-transform-react-jsx-source": "^7.27.1", + "@rolldown/pluginutils": "1.0.0-beta.53", + "@types/babel__core": "^7.20.5", + "react-refresh": "^0.18.0" + }, + "engines": { + "node": "^20.19.0 || >=22.12.0" + }, + "peerDependencies": { + "vite": "^4.2.0 || ^5.0.0 || ^6.0.0 || ^7.0.0" + } + }, + "node_modules/acorn": { + "version": "8.15.0", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.15.0.tgz", + "integrity": "sha512-NZyJarBfL7nWwIq+FDL6Zp/yHEhePMNnnJ0y3qfieCrmNvYct8uvtiV41UvlSe6apAfk0fY1FbWx+NwfmpvtTg==", + "dev": true, + "license": "MIT", + "peer": true, + "bin": { + "acorn": "bin/acorn" + }, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/acorn-jsx": { + "version": "5.3.2", + "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.3.2.tgz", + "integrity": "sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==", + "dev": true, + "license": "MIT", + "peerDependencies": { + "acorn": "^6.0.0 || ^7.0.0 || ^8.0.0" + } + }, + "node_modules/ajv": { + "version": "6.12.6", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz", + "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==", + "dev": true, + "license": "MIT", + "dependencies": { + "fast-deep-equal": "^3.1.1", + "fast-json-stable-stringify": "^2.0.0", + "json-schema-traverse": "^0.4.1", + "uri-js": "^4.2.2" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/epoberezkin" + } + }, + "node_modules/ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "license": "MIT", + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/argparse": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/argparse/-/argparse-2.0.1.tgz", + "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==", + "dev": true, + "license": "Python-2.0" + }, + "node_modules/balanced-match": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", + "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", + "dev": true, + "license": "MIT" + }, + "node_modules/baseline-browser-mapping": { + "version": "2.9.5", + "resolved": "https://registry.npmjs.org/baseline-browser-mapping/-/baseline-browser-mapping-2.9.5.tgz", + "integrity": "sha512-D5vIoztZOq1XM54LUdttJVc96ggEsIfju2JBvht06pSzpckp3C7HReun67Bghzrtdsq9XdMGbSSB3v3GhMNmAA==", + "dev": true, + "license": "Apache-2.0", + "bin": { + "baseline-browser-mapping": "dist/cli.js" + } + }, + "node_modules/brace-expansion": { + "version": "1.1.12", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.12.tgz", + "integrity": "sha512-9T9UjW3r0UW5c1Q7GTwllptXwhvYmEzFhzMfZ9H7FQWt+uZePjZPjBP/W1ZEyZ1twGWom5/56TF4lPcqjnDHcg==", + "dev": true, + "license": "MIT", + "dependencies": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "node_modules/browserslist": { + "version": "4.28.1", + "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.28.1.tgz", + "integrity": "sha512-ZC5Bd0LgJXgwGqUknZY/vkUQ04r8NXnJZ3yYi4vDmSiZmC/pdSN0NbNRPxZpbtO4uAfDUAFffO8IZoM3Gj8IkA==", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/browserslist" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/browserslist" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "license": "MIT", + "peer": true, + "dependencies": { + "baseline-browser-mapping": "^2.9.0", + "caniuse-lite": "^1.0.30001759", + "electron-to-chromium": "^1.5.263", + "node-releases": "^2.0.27", + "update-browserslist-db": "^1.2.0" + }, + "bin": { + "browserslist": "cli.js" + }, + "engines": { + "node": "^6 || ^7 || ^8 || ^9 || ^10 || ^11 || ^12 || >=13.7" + } + }, + "node_modules/callsites": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz", + "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/caniuse-lite": { + "version": "1.0.30001760", + "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001760.tgz", + "integrity": "sha512-7AAMPcueWELt1p3mi13HR/LHH0TJLT11cnwDJEs3xA4+CK/PLKeO9Kl1oru24htkyUKtkGCvAx4ohB0Ttry8Dw==", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/browserslist" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/caniuse-lite" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "license": "CC-BY-4.0" + }, + "node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" + } + }, + "node_modules/color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true, + "license": "MIT" + }, + "node_modules/concat-map": { + "version": "0.0.1", + "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", + "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==", + "dev": true, + "license": "MIT" + }, + "node_modules/convert-source-map": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/convert-source-map/-/convert-source-map-2.0.0.tgz", + "integrity": "sha512-Kvp459HrV2FEJ1CAsi1Ku+MY3kasH19TFykTz2xWmMeq6bk2NU3XXvfJ+Q61m0xktWwt+1HSYf3JZsTms3aRJg==", + "dev": true, + "license": "MIT" + }, + "node_modules/cross-spawn": { + "version": "7.0.6", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.6.tgz", + "integrity": "sha512-uV2QOWP2nWzsy2aMp8aRibhi9dlzF5Hgh5SHaB9OiTGEyDTiJJyx0uy51QXdyWbtAHNua4XJzUKca3OzKUd3vA==", + "dev": true, + "license": "MIT", + "dependencies": { + "path-key": "^3.1.0", + "shebang-command": "^2.0.0", + "which": "^2.0.1" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/csstype": { + "version": "3.2.3", + "resolved": "https://registry.npmjs.org/csstype/-/csstype-3.2.3.tgz", + "integrity": "sha512-z1HGKcYy2xA8AGQfwrn0PAy+PB7X/GSj3UVJW9qKyn43xWa+gl5nXmU4qqLMRzWVLFC8KusUX8T/0kCiOYpAIQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/debug": { + "version": "4.4.3", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.3.tgz", + "integrity": "sha512-RGwwWnwQvkVfavKVt22FGLw+xYSdzARwm0ru6DhTVA3umU5hZc28V3kO4stgYryrTlLpuvgI9GiijltAjNbcqA==", + "dev": true, + "license": "MIT", + "dependencies": { + "ms": "^2.1.3" + }, + "engines": { + "node": ">=6.0" + }, + "peerDependenciesMeta": { + "supports-color": { + "optional": true + } + } + }, + "node_modules/deep-is": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.4.tgz", + "integrity": "sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/electron-to-chromium": { + "version": "1.5.267", + "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.5.267.tgz", + "integrity": "sha512-0Drusm6MVRXSOJpGbaSVgcQsuB4hEkMpHXaVstcPmhu5LIedxs1xNK/nIxmQIU/RPC0+1/o0AVZfBTkTNJOdUw==", + "dev": true, + "license": "ISC" + }, + "node_modules/esbuild": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.25.12.tgz", + "integrity": "sha512-bbPBYYrtZbkt6Os6FiTLCTFxvq4tt3JKall1vRwshA3fdVztsLAatFaZobhkBC8/BrPetoa0oksYoKXoG4ryJg==", + "dev": true, + "hasInstallScript": true, + "license": "MIT", + "bin": { + "esbuild": "bin/esbuild" + }, + "engines": { + "node": ">=18" + }, + "optionalDependencies": { + "@esbuild/aix-ppc64": "0.25.12", + "@esbuild/android-arm": "0.25.12", + "@esbuild/android-arm64": "0.25.12", + "@esbuild/android-x64": "0.25.12", + "@esbuild/darwin-arm64": "0.25.12", + "@esbuild/darwin-x64": "0.25.12", + "@esbuild/freebsd-arm64": "0.25.12", + "@esbuild/freebsd-x64": "0.25.12", + "@esbuild/linux-arm": "0.25.12", + "@esbuild/linux-arm64": "0.25.12", + "@esbuild/linux-ia32": "0.25.12", + "@esbuild/linux-loong64": "0.25.12", + "@esbuild/linux-mips64el": "0.25.12", + "@esbuild/linux-ppc64": "0.25.12", + "@esbuild/linux-riscv64": "0.25.12", + "@esbuild/linux-s390x": "0.25.12", + "@esbuild/linux-x64": "0.25.12", + "@esbuild/netbsd-arm64": "0.25.12", + "@esbuild/netbsd-x64": "0.25.12", + "@esbuild/openbsd-arm64": "0.25.12", + "@esbuild/openbsd-x64": "0.25.12", + "@esbuild/openharmony-arm64": "0.25.12", + "@esbuild/sunos-x64": "0.25.12", + "@esbuild/win32-arm64": "0.25.12", + "@esbuild/win32-ia32": "0.25.12", + "@esbuild/win32-x64": "0.25.12" + } + }, + "node_modules/escalade": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.2.0.tgz", + "integrity": "sha512-WUj2qlxaQtO4g6Pq5c29GTcWGDyd8itL8zTlipgECz3JesAiiOKotd8JU6otB3PACgG6xkJUyVhboMS+bje/jA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/escape-string-regexp": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz", + "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/eslint": { + "version": "9.39.1", + "resolved": "https://registry.npmjs.org/eslint/-/eslint-9.39.1.tgz", + "integrity": "sha512-BhHmn2yNOFA9H9JmmIVKJmd288g9hrVRDkdoIgRCRuSySRUHH7r/DI6aAXW9T1WwUuY3DFgrcaqB+deURBLR5g==", + "dev": true, + "license": "MIT", + "peer": true, + "dependencies": { + "@eslint-community/eslint-utils": "^4.8.0", + "@eslint-community/regexpp": "^4.12.1", + "@eslint/config-array": "^0.21.1", + "@eslint/config-helpers": "^0.4.2", + "@eslint/core": "^0.17.0", + "@eslint/eslintrc": "^3.3.1", + "@eslint/js": "9.39.1", + "@eslint/plugin-kit": "^0.4.1", + "@humanfs/node": "^0.16.6", + "@humanwhocodes/module-importer": "^1.0.1", + "@humanwhocodes/retry": "^0.4.2", + "@types/estree": "^1.0.6", + "ajv": "^6.12.4", + "chalk": "^4.0.0", + "cross-spawn": "^7.0.6", + "debug": "^4.3.2", + "escape-string-regexp": "^4.0.0", + "eslint-scope": "^8.4.0", + "eslint-visitor-keys": "^4.2.1", + "espree": "^10.4.0", + "esquery": "^1.5.0", + "esutils": "^2.0.2", + "fast-deep-equal": "^3.1.3", + "file-entry-cache": "^8.0.0", + "find-up": "^5.0.0", + "glob-parent": "^6.0.2", + "ignore": "^5.2.0", + "imurmurhash": "^0.1.4", + "is-glob": "^4.0.0", + "json-stable-stringify-without-jsonify": "^1.0.1", + "lodash.merge": "^4.6.2", + "minimatch": "^3.1.2", + "natural-compare": "^1.4.0", + "optionator": "^0.9.3" + }, + "bin": { + "eslint": "bin/eslint.js" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "url": "https://eslint.org/donate" + }, + "peerDependencies": { + "jiti": "*" + }, + "peerDependenciesMeta": { + "jiti": { + "optional": true + } + } + }, + "node_modules/eslint-plugin-react-hooks": { + "version": "7.0.1", + "resolved": "https://registry.npmjs.org/eslint-plugin-react-hooks/-/eslint-plugin-react-hooks-7.0.1.tgz", + "integrity": "sha512-O0d0m04evaNzEPoSW+59Mezf8Qt0InfgGIBJnpC0h3NH/WjUAR7BIKUfysC6todmtiZ/A0oUVS8Gce0WhBrHsA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/core": "^7.24.4", + "@babel/parser": "^7.24.4", + "hermes-parser": "^0.25.1", + "zod": "^3.25.0 || ^4.0.0", + "zod-validation-error": "^3.5.0 || ^4.0.0" + }, + "engines": { + "node": ">=18" + }, + "peerDependencies": { + "eslint": "^3.0.0 || ^4.0.0 || ^5.0.0 || ^6.0.0 || ^7.0.0 || ^8.0.0-0 || ^9.0.0" + } + }, + "node_modules/eslint-plugin-react-refresh": { + "version": "0.4.24", + "resolved": "https://registry.npmjs.org/eslint-plugin-react-refresh/-/eslint-plugin-react-refresh-0.4.24.tgz", + "integrity": "sha512-nLHIW7TEq3aLrEYWpVaJ1dRgFR+wLDPN8e8FpYAql/bMV2oBEfC37K0gLEGgv9fy66juNShSMV8OkTqzltcG/w==", + "dev": true, + "license": "MIT", + "peerDependencies": { + "eslint": ">=8.40" + } + }, + "node_modules/eslint-scope": { + "version": "8.4.0", + "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-8.4.0.tgz", + "integrity": "sha512-sNXOfKCn74rt8RICKMvJS7XKV/Xk9kA7DyJr8mJik3S7Cwgy3qlkkmyS2uQB3jiJg6VNdZd/pDBJu0nvG2NlTg==", + "dev": true, + "license": "BSD-2-Clause", + "dependencies": { + "esrecurse": "^4.3.0", + "estraverse": "^5.2.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/eslint-visitor-keys": { + "version": "4.2.1", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-4.2.1.tgz", + "integrity": "sha512-Uhdk5sfqcee/9H/rCOJikYz67o0a2Tw2hGRPOG2Y1R2dg7brRe1uG0yaNQDHu+TO/uQPF/5eCapvYSmHUjt7JQ==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/espree": { + "version": "10.4.0", + "resolved": "https://registry.npmjs.org/espree/-/espree-10.4.0.tgz", + "integrity": "sha512-j6PAQ2uUr79PZhBjP5C5fhl8e39FmRnOjsD5lGnWrFU8i2G776tBK7+nP8KuQUTTyAZUwfQqXAgrVH5MbH9CYQ==", + "dev": true, + "license": "BSD-2-Clause", + "dependencies": { + "acorn": "^8.15.0", + "acorn-jsx": "^5.3.2", + "eslint-visitor-keys": "^4.2.1" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/esquery": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.6.0.tgz", + "integrity": "sha512-ca9pw9fomFcKPvFLXhBKUK90ZvGibiGOvRJNbjljY7s7uq/5YO4BOzcYtJqExdx99rF6aAcnRxHmcUHcz6sQsg==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "estraverse": "^5.1.0" + }, + "engines": { + "node": ">=0.10" + } + }, + "node_modules/esrecurse": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.3.0.tgz", + "integrity": "sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==", + "dev": true, + "license": "BSD-2-Clause", + "dependencies": { + "estraverse": "^5.2.0" + }, + "engines": { + "node": ">=4.0" + } + }, + "node_modules/estraverse": { + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz", + "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==", + "dev": true, + "license": "BSD-2-Clause", + "engines": { + "node": ">=4.0" + } + }, + "node_modules/esutils": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz", + "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", + "dev": true, + "license": "BSD-2-Clause", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/fast-deep-equal": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz", + "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==", + "dev": true, + "license": "MIT" + }, + "node_modules/fast-json-stable-stringify": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", + "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==", + "dev": true, + "license": "MIT" + }, + "node_modules/fast-levenshtein": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz", + "integrity": "sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw==", + "dev": true, + "license": "MIT" + }, + "node_modules/fdir": { + "version": "6.5.0", + "resolved": "https://registry.npmjs.org/fdir/-/fdir-6.5.0.tgz", + "integrity": "sha512-tIbYtZbucOs0BRGqPJkshJUYdL+SDH7dVM8gjy+ERp3WAUjLEFJE+02kanyHtwjWOnwrKYBiwAmM0p4kLJAnXg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=12.0.0" + }, + "peerDependencies": { + "picomatch": "^3 || ^4" + }, + "peerDependenciesMeta": { + "picomatch": { + "optional": true + } + } + }, + "node_modules/file-entry-cache": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-8.0.0.tgz", + "integrity": "sha512-XXTUwCvisa5oacNGRP9SfNtYBNAMi+RPwBFmblZEF7N7swHYQS6/Zfk7SRwx4D5j3CH211YNRco1DEMNVfZCnQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "flat-cache": "^4.0.0" + }, + "engines": { + "node": ">=16.0.0" + } + }, + "node_modules/find-up": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz", + "integrity": "sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==", + "dev": true, + "license": "MIT", + "dependencies": { + "locate-path": "^6.0.0", + "path-exists": "^4.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/flat-cache": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-4.0.1.tgz", + "integrity": "sha512-f7ccFPK3SXFHpx15UIGyRJ/FJQctuKZ0zVuN3frBo4HnK3cay9VEW0R6yPYFHC0AgqhukPzKjq22t5DmAyqGyw==", + "dev": true, + "license": "MIT", + "dependencies": { + "flatted": "^3.2.9", + "keyv": "^4.5.4" + }, + "engines": { + "node": ">=16" + } + }, + "node_modules/flatted": { + "version": "3.3.3", + "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.3.3.tgz", + "integrity": "sha512-GX+ysw4PBCz0PzosHDepZGANEuFCMLrnRTiEy9McGjmkCQYwRq4A/X786G/fjM/+OjsWSU1ZrY5qyARZmO/uwg==", + "dev": true, + "license": "ISC" + }, + "node_modules/fsevents": { + "version": "2.3.3", + "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.3.tgz", + "integrity": "sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==", + "dev": true, + "hasInstallScript": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": "^8.16.0 || ^10.6.0 || >=11.0.0" + } + }, + "node_modules/gensync": { + "version": "1.0.0-beta.2", + "resolved": "https://registry.npmjs.org/gensync/-/gensync-1.0.0-beta.2.tgz", + "integrity": "sha512-3hN7NaskYvMDLQY55gnW3NQ+mesEAepTqlg+VEbj7zzqEMBVNhzcGYYeqFo/TlYz6eQiFcp1HcsCZO+nGgS8zg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/glob-parent": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-6.0.2.tgz", + "integrity": "sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A==", + "dev": true, + "license": "ISC", + "dependencies": { + "is-glob": "^4.0.3" + }, + "engines": { + "node": ">=10.13.0" + } + }, + "node_modules/globals": { + "version": "16.5.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-16.5.0.tgz", + "integrity": "sha512-c/c15i26VrJ4IRt5Z89DnIzCGDn9EcebibhAOjw5ibqEHsE1wLUgkPn9RDmNcUKyU87GeaL633nyJ+pplFR2ZQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=18" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/hermes-estree": { + "version": "0.25.1", + "resolved": "https://registry.npmjs.org/hermes-estree/-/hermes-estree-0.25.1.tgz", + "integrity": "sha512-0wUoCcLp+5Ev5pDW2OriHC2MJCbwLwuRx+gAqMTOkGKJJiBCLjtrvy4PWUGn6MIVefecRpzoOZ/UV6iGdOr+Cw==", + "dev": true, + "license": "MIT" + }, + "node_modules/hermes-parser": { + "version": "0.25.1", + "resolved": "https://registry.npmjs.org/hermes-parser/-/hermes-parser-0.25.1.tgz", + "integrity": "sha512-6pEjquH3rqaI6cYAXYPcz9MS4rY6R4ngRgrgfDshRptUZIc3lw0MCIJIGDj9++mfySOuPTHB4nrSW99BCvOPIA==", + "dev": true, + "license": "MIT", + "dependencies": { + "hermes-estree": "0.25.1" + } + }, + "node_modules/ignore": { + "version": "5.3.2", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.3.2.tgz", + "integrity": "sha512-hsBTNUqQTDwkWtcdYI2i06Y/nUBEsNEDJKjWdigLvegy8kDuJAS8uRlpkkcQpyEXL0Z/pjDy5HBmMjRCJ2gq+g==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 4" + } + }, + "node_modules/import-fresh": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.3.1.tgz", + "integrity": "sha512-TR3KfrTZTYLPB6jUjfx6MF9WcWrHL9su5TObK4ZkYgBdWKPOFoSoQIdEuTuR82pmtxH2spWG9h6etwfr1pLBqQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "parent-module": "^1.0.0", + "resolve-from": "^4.0.0" + }, + "engines": { + "node": ">=6" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/imurmurhash": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", + "integrity": "sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.8.19" + } + }, + "node_modules/is-extglob": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", + "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-glob": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz", + "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==", + "dev": true, + "license": "MIT", + "dependencies": { + "is-extglob": "^2.1.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/isexe": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", + "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==", + "dev": true, + "license": "ISC" + }, + "node_modules/js-tokens": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz", + "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/js-yaml": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.1.tgz", + "integrity": "sha512-qQKT4zQxXl8lLwBtHMWwaTcGfFOZviOJet3Oy/xmGk2gZH677CJM9EvtfdSkgWcATZhj/55JZ0rmy3myCT5lsA==", + "dev": true, + "license": "MIT", + "dependencies": { + "argparse": "^2.0.1" + }, + "bin": { + "js-yaml": "bin/js-yaml.js" + } + }, + "node_modules/jsesc": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-3.1.0.tgz", + "integrity": "sha512-/sM3dO2FOzXjKQhJuo0Q173wf2KOo8t4I8vHy6lF9poUp7bKT0/NHE8fPX23PwfhnykfqnC2xRxOnVw5XuGIaA==", + "dev": true, + "license": "MIT", + "bin": { + "jsesc": "bin/jsesc" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/json-buffer": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.1.tgz", + "integrity": "sha512-4bV5BfR2mqfQTJm+V5tPPdf+ZpuhiIvTuAB5g8kcrXOZpTT/QwwVRWBywX1ozr6lEuPdbHxwaJlm9G6mI2sfSQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/json-schema-traverse": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz", + "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==", + "dev": true, + "license": "MIT" + }, + "node_modules/json-stable-stringify-without-jsonify": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz", + "integrity": "sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw==", + "dev": true, + "license": "MIT" + }, + "node_modules/json5": { + "version": "2.2.3", + "resolved": "https://registry.npmjs.org/json5/-/json5-2.2.3.tgz", + "integrity": "sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg==", + "dev": true, + "license": "MIT", + "bin": { + "json5": "lib/cli.js" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/keyv": { + "version": "4.5.4", + "resolved": "https://registry.npmjs.org/keyv/-/keyv-4.5.4.tgz", + "integrity": "sha512-oxVHkHR/EJf2CNXnWxRLW6mg7JyCCUcG0DtEGmL2ctUo1PNTin1PUil+r/+4r5MpVgC/fn1kjsx7mjSujKqIpw==", + "dev": true, + "license": "MIT", + "dependencies": { + "json-buffer": "3.0.1" + } + }, + "node_modules/levn": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/levn/-/levn-0.4.1.tgz", + "integrity": "sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "prelude-ls": "^1.2.1", + "type-check": "~0.4.0" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/locate-path": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz", + "integrity": "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==", + "dev": true, + "license": "MIT", + "dependencies": { + "p-locate": "^5.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/lodash.merge": { + "version": "4.6.2", + "resolved": "https://registry.npmjs.org/lodash.merge/-/lodash.merge-4.6.2.tgz", + "integrity": "sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/lru-cache": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-5.1.1.tgz", + "integrity": "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==", + "dev": true, + "license": "ISC", + "dependencies": { + "yallist": "^3.0.2" + } + }, + "node_modules/minimatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "dev": true, + "license": "ISC", + "dependencies": { + "brace-expansion": "^1.1.7" + }, + "engines": { + "node": "*" + } + }, + "node_modules/ms": { + "version": "2.1.3", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", + "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", + "dev": true, + "license": "MIT" + }, + "node_modules/nanoid": { + "version": "3.3.11", + "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.11.tgz", + "integrity": "sha512-N8SpfPUnUp1bK+PMYW8qSWdl9U+wwNWI4QKxOYDy9JAro3WMX7p2OeVRF9v+347pnakNevPmiHhNmZ2HbFA76w==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "license": "MIT", + "bin": { + "nanoid": "bin/nanoid.cjs" + }, + "engines": { + "node": "^10 || ^12 || ^13.7 || ^14 || >=15.0.1" + } + }, + "node_modules/natural-compare": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", + "integrity": "sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==", + "dev": true, + "license": "MIT" + }, + "node_modules/node-releases": { + "version": "2.0.27", + "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-2.0.27.tgz", + "integrity": "sha512-nmh3lCkYZ3grZvqcCH+fjmQ7X+H0OeZgP40OierEaAptX4XofMh5kwNbWh7lBduUzCcV/8kZ+NDLCwm2iorIlA==", + "dev": true, + "license": "MIT" + }, + "node_modules/optionator": { + "version": "0.9.4", + "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.4.tgz", + "integrity": "sha512-6IpQ7mKUxRcZNLIObR0hz7lxsapSSIYNZJwXPGeF0mTVqGKFIXj1DQcMoT22S3ROcLyY/rz0PWaWZ9ayWmad9g==", + "dev": true, + "license": "MIT", + "dependencies": { + "deep-is": "^0.1.3", + "fast-levenshtein": "^2.0.6", + "levn": "^0.4.1", + "prelude-ls": "^1.2.1", + "type-check": "^0.4.0", + "word-wrap": "^1.2.5" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/p-limit": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz", + "integrity": "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "yocto-queue": "^0.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/p-locate": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-5.0.0.tgz", + "integrity": "sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==", + "dev": true, + "license": "MIT", + "dependencies": { + "p-limit": "^3.0.2" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/parent-module": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/parent-module/-/parent-module-1.0.1.tgz", + "integrity": "sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==", + "dev": true, + "license": "MIT", + "dependencies": { + "callsites": "^3.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/path-exists": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", + "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/path-key": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", + "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/picocolors": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.1.1.tgz", + "integrity": "sha512-xceH2snhtb5M9liqDsmEw56le376mTZkEX/jEb/RxNFyegNul7eNslCXP9FDj/Lcu0X8KEyMceP2ntpaHrDEVA==", + "dev": true, + "license": "ISC" + }, + "node_modules/picomatch": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-4.0.3.tgz", + "integrity": "sha512-5gTmgEY/sqK6gFXLIsQNH19lWb4ebPDLA4SdLP7dsWkIXHWlG66oPuVvXSGFPppYZz8ZDZq0dYYrbHfBCVUb1Q==", + "dev": true, + "license": "MIT", + "peer": true, + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/sponsors/jonschlinkert" + } + }, + "node_modules/postcss": { + "version": "8.5.6", + "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.5.6.tgz", + "integrity": "sha512-3Ybi1tAuwAP9s0r1UQ2J4n5Y0G05bJkpUIO0/bI9MhwmD70S5aTWbXGBwxHrelT+XM1k6dM0pk+SwNkpTRN7Pg==", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/postcss/" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/postcss" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "license": "MIT", + "dependencies": { + "nanoid": "^3.3.11", + "picocolors": "^1.1.1", + "source-map-js": "^1.2.1" + }, + "engines": { + "node": "^10 || ^12 || >=14" + } + }, + "node_modules/prelude-ls": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.2.1.tgz", + "integrity": "sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/punycode": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.3.1.tgz", + "integrity": "sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/react": { + "version": "19.2.1", + "resolved": "https://registry.npmjs.org/react/-/react-19.2.1.tgz", + "integrity": "sha512-DGrYcCWK7tvYMnWh79yrPHt+vdx9tY+1gPZa7nJQtO/p8bLTDaHp4dzwEhQB7pZ4Xe3ok4XKuEPrVuc+wlpkmw==", + "license": "MIT", + "peer": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/react-dom": { + "version": "19.2.1", + "resolved": "https://registry.npmjs.org/react-dom/-/react-dom-19.2.1.tgz", + "integrity": "sha512-ibrK8llX2a4eOskq1mXKu/TGZj9qzomO+sNfO98M6d9zIPOEhlBkMkBUBLd1vgS0gQsLDBzA+8jJBVXDnfHmJg==", + "license": "MIT", + "dependencies": { + "scheduler": "^0.27.0" + }, + "peerDependencies": { + "react": "^19.2.1" + } + }, + "node_modules/react-refresh": { + "version": "0.18.0", + "resolved": "https://registry.npmjs.org/react-refresh/-/react-refresh-0.18.0.tgz", + "integrity": "sha512-QgT5//D3jfjJb6Gsjxv0Slpj23ip+HtOpnNgnb2S5zU3CB26G/IDPGoy4RJB42wzFE46DRsstbW6tKHoKbhAxw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/resolve-from": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz", + "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=4" + } + }, + "node_modules/rollup": { + "version": "4.53.3", + "resolved": "https://registry.npmjs.org/rollup/-/rollup-4.53.3.tgz", + "integrity": "sha512-w8GmOxZfBmKknvdXU1sdM9NHcoQejwF/4mNgj2JuEEdRaHwwF12K7e9eXn1nLZ07ad+du76mkVsyeb2rKGllsA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/estree": "1.0.8" + }, + "bin": { + "rollup": "dist/bin/rollup" + }, + "engines": { + "node": ">=18.0.0", + "npm": ">=8.0.0" + }, + "optionalDependencies": { + "@rollup/rollup-android-arm-eabi": "4.53.3", + "@rollup/rollup-android-arm64": "4.53.3", + "@rollup/rollup-darwin-arm64": "4.53.3", + "@rollup/rollup-darwin-x64": "4.53.3", + "@rollup/rollup-freebsd-arm64": "4.53.3", + "@rollup/rollup-freebsd-x64": "4.53.3", + "@rollup/rollup-linux-arm-gnueabihf": "4.53.3", + "@rollup/rollup-linux-arm-musleabihf": "4.53.3", + "@rollup/rollup-linux-arm64-gnu": "4.53.3", + "@rollup/rollup-linux-arm64-musl": "4.53.3", + "@rollup/rollup-linux-loong64-gnu": "4.53.3", + "@rollup/rollup-linux-ppc64-gnu": "4.53.3", + "@rollup/rollup-linux-riscv64-gnu": "4.53.3", + "@rollup/rollup-linux-riscv64-musl": "4.53.3", + "@rollup/rollup-linux-s390x-gnu": "4.53.3", + "@rollup/rollup-linux-x64-gnu": "4.53.3", + "@rollup/rollup-linux-x64-musl": "4.53.3", + "@rollup/rollup-openharmony-arm64": "4.53.3", + "@rollup/rollup-win32-arm64-msvc": "4.53.3", + "@rollup/rollup-win32-ia32-msvc": "4.53.3", + "@rollup/rollup-win32-x64-gnu": "4.53.3", + "@rollup/rollup-win32-x64-msvc": "4.53.3", + "fsevents": "~2.3.2" + } + }, + "node_modules/scheduler": { + "version": "0.27.0", + "resolved": "https://registry.npmjs.org/scheduler/-/scheduler-0.27.0.tgz", + "integrity": "sha512-eNv+WrVbKu1f3vbYJT/xtiF5syA5HPIMtf9IgY/nKg0sWqzAUEvqY/xm7OcZc/qafLx/iO9FgOmeSAp4v5ti/Q==", + "license": "MIT" + }, + "node_modules/semver": { + "version": "6.3.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", + "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==", + "dev": true, + "license": "ISC", + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/shebang-command": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", + "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", + "dev": true, + "license": "MIT", + "dependencies": { + "shebang-regex": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/shebang-regex": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz", + "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/source-map-js": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/source-map-js/-/source-map-js-1.2.1.tgz", + "integrity": "sha512-UXWMKhLOwVKb728IUtQPXxfYU+usdybtUrK/8uGE8CQMvrhOpwvzDBwj0QhSL7MQc7vIsISBG8VQ8+IDQxpfQA==", + "dev": true, + "license": "BSD-3-Clause", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/strip-json-comments": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-3.1.1.tgz", + "integrity": "sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "license": "MIT", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/tinyglobby": { + "version": "0.2.15", + "resolved": "https://registry.npmjs.org/tinyglobby/-/tinyglobby-0.2.15.tgz", + "integrity": "sha512-j2Zq4NyQYG5XMST4cbs02Ak8iJUdxRM0XI5QyxXuZOzKOINmWurp3smXu3y5wDcJrptwpSjgXHzIQxR0omXljQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "fdir": "^6.5.0", + "picomatch": "^4.0.3" + }, + "engines": { + "node": ">=12.0.0" + }, + "funding": { + "url": "https://github.com/sponsors/SuperchupuDev" + } + }, + "node_modules/ts-api-utils": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/ts-api-utils/-/ts-api-utils-2.1.0.tgz", + "integrity": "sha512-CUgTZL1irw8u29bzrOD/nH85jqyc74D6SshFgujOIA7osm2Rz7dYH77agkx7H4FBNxDq7Cjf+IjaX/8zwFW+ZQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=18.12" + }, + "peerDependencies": { + "typescript": ">=4.8.4" + } + }, + "node_modules/type-check": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.4.0.tgz", + "integrity": "sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew==", + "dev": true, + "license": "MIT", + "dependencies": { + "prelude-ls": "^1.2.1" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/typescript": { + "version": "5.9.3", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.9.3.tgz", + "integrity": "sha512-jl1vZzPDinLr9eUt3J/t7V6FgNEw9QjvBPdysz9KfQDD41fQrC2Y4vKQdiaUpFT4bXlb1RHhLpp8wtm6M5TgSw==", + "dev": true, + "license": "Apache-2.0", + "peer": true, + "bin": { + "tsc": "bin/tsc", + "tsserver": "bin/tsserver" + }, + "engines": { + "node": ">=14.17" + } + }, + "node_modules/typescript-eslint": { + "version": "8.49.0", + "resolved": "https://registry.npmjs.org/typescript-eslint/-/typescript-eslint-8.49.0.tgz", + "integrity": "sha512-zRSVH1WXD0uXczCXw+nsdjGPUdx4dfrs5VQoHnUWmv1U3oNlAKv4FUNdLDhVUg+gYn+a5hUESqch//Rv5wVhrg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@typescript-eslint/eslint-plugin": "8.49.0", + "@typescript-eslint/parser": "8.49.0", + "@typescript-eslint/typescript-estree": "8.49.0", + "@typescript-eslint/utils": "8.49.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "eslint": "^8.57.0 || ^9.0.0", + "typescript": ">=4.8.4 <6.0.0" + } + }, + "node_modules/undici-types": { + "version": "7.16.0", + "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-7.16.0.tgz", + "integrity": "sha512-Zz+aZWSj8LE6zoxD+xrjh4VfkIG8Ya6LvYkZqtUQGJPZjYl53ypCaUwWqo7eI0x66KBGeRo+mlBEkMSeSZ38Nw==", + "dev": true, + "license": "MIT" + }, + "node_modules/update-browserslist-db": { + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/update-browserslist-db/-/update-browserslist-db-1.2.2.tgz", + "integrity": "sha512-E85pfNzMQ9jpKkA7+TJAi4TJN+tBCuWh5rUcS/sv6cFi+1q9LYDwDI5dpUL0u/73EElyQ8d3TEaeW4sPedBqYA==", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/browserslist" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/browserslist" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "license": "MIT", + "dependencies": { + "escalade": "^3.2.0", + "picocolors": "^1.1.1" + }, + "bin": { + "update-browserslist-db": "cli.js" + }, + "peerDependencies": { + "browserslist": ">= 4.21.0" + } + }, + "node_modules/uri-js": { + "version": "4.4.1", + "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.4.1.tgz", + "integrity": "sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==", + "dev": true, + "license": "BSD-2-Clause", + "dependencies": { + "punycode": "^2.1.0" + } + }, + "node_modules/vite": { + "version": "7.2.7", + "resolved": "https://registry.npmjs.org/vite/-/vite-7.2.7.tgz", + "integrity": "sha512-ITcnkFeR3+fI8P1wMgItjGrR10170d8auB4EpMLPqmx6uxElH3a/hHGQabSHKdqd4FXWO1nFIp9rRn7JQ34ACQ==", + "dev": true, + "license": "MIT", + "peer": true, + "dependencies": { + "esbuild": "^0.25.0", + "fdir": "^6.5.0", + "picomatch": "^4.0.3", + "postcss": "^8.5.6", + "rollup": "^4.43.0", + "tinyglobby": "^0.2.15" + }, + "bin": { + "vite": "bin/vite.js" + }, + "engines": { + "node": "^20.19.0 || >=22.12.0" + }, + "funding": { + "url": "https://github.com/vitejs/vite?sponsor=1" + }, + "optionalDependencies": { + "fsevents": "~2.3.3" + }, + "peerDependencies": { + "@types/node": "^20.19.0 || >=22.12.0", + "jiti": ">=1.21.0", + "less": "^4.0.0", + "lightningcss": "^1.21.0", + "sass": "^1.70.0", + "sass-embedded": "^1.70.0", + "stylus": ">=0.54.8", + "sugarss": "^5.0.0", + "terser": "^5.16.0", + "tsx": "^4.8.1", + "yaml": "^2.4.2" + }, + "peerDependenciesMeta": { + "@types/node": { + "optional": true + }, + "jiti": { + "optional": true + }, + "less": { + "optional": true + }, + "lightningcss": { + "optional": true + }, + "sass": { + "optional": true + }, + "sass-embedded": { + "optional": true + }, + "stylus": { + "optional": true + }, + "sugarss": { + "optional": true + }, + "terser": { + "optional": true + }, + "tsx": { + "optional": true + }, + "yaml": { + "optional": true + } + } + }, + "node_modules/which": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", + "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", + "dev": true, + "license": "ISC", + "dependencies": { + "isexe": "^2.0.0" + }, + "bin": { + "node-which": "bin/node-which" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/word-wrap": { + "version": "1.2.5", + "resolved": "https://registry.npmjs.org/word-wrap/-/word-wrap-1.2.5.tgz", + "integrity": "sha512-BN22B5eaMMI9UMtjrGd5g5eCYPpCPDUy0FJXbYsaT5zYxjFOckS53SQDE3pWkVoWpHXVb3BrYcEN4Twa55B5cA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/yallist": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz", + "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==", + "dev": true, + "license": "ISC" + }, + "node_modules/yocto-queue": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-0.1.0.tgz", + "integrity": "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/zod": { + "version": "4.1.13", + "resolved": "https://registry.npmjs.org/zod/-/zod-4.1.13.tgz", + "integrity": "sha512-AvvthqfqrAhNH9dnfmrfKzX5upOdjUVJYFqNSlkmGf64gRaTzlPwz99IHYnVs28qYAybvAlBV+H7pn0saFY4Ig==", + "dev": true, + "license": "MIT", + "peer": true, + "funding": { + "url": "https://github.com/sponsors/colinhacks" + } + }, + "node_modules/zod-validation-error": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/zod-validation-error/-/zod-validation-error-4.0.2.tgz", + "integrity": "sha512-Q6/nZLe6jxuU80qb/4uJ4t5v2VEZ44lzQjPDhYJNztRQ4wyWc6VF3D3Kb/fAuPetZQnhS3hnajCf9CsWesghLQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=18.0.0" + }, + "peerDependencies": { + "zod": "^3.25.0 || ^4.0.0" + } + } + } +} diff --git a/mengyamonitor-frontend/package.json b/mengyamonitor-frontend/package.json new file mode 100644 index 0000000..17acb5b --- /dev/null +++ b/mengyamonitor-frontend/package.json @@ -0,0 +1,30 @@ +{ + "name": "mengyamonitor-frontend", + "private": true, + "version": "0.0.0", + "type": "module", + "scripts": { + "dev": "vite", + "build": "tsc -b && vite build", + "lint": "eslint .", + "preview": "vite preview" + }, + "dependencies": { + "react": "^19.2.0", + "react-dom": "^19.2.0" + }, + "devDependencies": { + "@eslint/js": "^9.39.1", + "@types/node": "^24.10.1", + "@types/react": "^19.2.5", + "@types/react-dom": "^19.2.3", + "@vitejs/plugin-react": "^5.1.1", + "eslint": "^9.39.1", + "eslint-plugin-react-hooks": "^7.0.1", + "eslint-plugin-react-refresh": "^0.4.24", + "globals": "^16.5.0", + "typescript": "~5.9.3", + "typescript-eslint": "^8.46.4", + "vite": "^7.2.4" + } +} diff --git a/mengyamonitor-frontend/public/logo.png b/mengyamonitor-frontend/public/logo.png new file mode 100644 index 0000000..eec06de Binary files /dev/null and b/mengyamonitor-frontend/public/logo.png differ diff --git a/mengyamonitor-frontend/public/logo2.png b/mengyamonitor-frontend/public/logo2.png new file mode 100644 index 0000000..8132e3e Binary files /dev/null and b/mengyamonitor-frontend/public/logo2.png differ diff --git a/mengyamonitor-frontend/src/App.css b/mengyamonitor-frontend/src/App.css new file mode 100644 index 0000000..b8bc260 --- /dev/null +++ b/mengyamonitor-frontend/src/App.css @@ -0,0 +1,178 @@ +.app { + width: 100%; + max-width: 100%; + margin: 0 auto; + padding: 1rem; + min-height: 100vh; +} + +.app-header { + display: flex; + justify-content: space-between; + align-items: center; + margin-bottom: 1rem; + padding: 0.5rem 1rem; + background: var(--surface-color); + border-radius: var(--radius-md); + box-shadow: var(--shadow-sm); + border: 1px solid var(--border-color); +} + +.app-header h1 { + margin: 0; + font-size: 0.75rem; + color: var(--text-light); + display: flex; + align-items: center; + gap: 0.25rem; + font-weight: 400; +} + +.app-logo { + width: 16px; + height: 16px; + border-radius: 4px; + display: inline-block; +} + +.btn-add { + background: var(--primary-color); + color: white; + border: none; + padding: 0.5rem; + width: 32px; + height: 32px; + border-radius: var(--radius-md); + font-size: 1.25rem; + font-weight: 600; + cursor: pointer; + transition: all 0.2s ease; + box-shadow: var(--shadow-sm); + display: flex; + align-items: center; + justify-content: center; + line-height: 1; +} + +.btn-add:hover { + background: var(--primary-hover); + transform: translateY(-1px); + box-shadow: var(--shadow-md); +} + +.add-form { + background: var(--surface-color); + padding: 1.5rem; + border-radius: var(--radius-md); + margin-bottom: 1rem; + box-shadow: var(--shadow-sm); + display: flex; + gap: 1rem; + flex-wrap: wrap; + border: 1px solid var(--border-color); + animation: slideDown 0.3s ease-out; +} + +@keyframes slideDown { + from { + opacity: 0; + transform: translateY(-10px); + } + to { + opacity: 1; + transform: translateY(0); + } +} + +.add-form input { + flex: 1; + min-width: 250px; + padding: 0.75rem 1rem; + border: 1px solid var(--border-color); + border-radius: var(--radius-md); + font-size: 0.875rem; + transition: all 0.2s; + background: var(--bg-color); + color: var(--text-main); +} + +.add-form input:focus { + outline: none; + border-color: var(--primary-color); + box-shadow: 0 0 0 3px rgba(134, 239, 172, 0.1); + background: var(--surface-color); +} + +.btn-submit { + background: var(--primary-color); + color: white; + border: none; + padding: 0.75rem 2rem; + border-radius: var(--radius-md); + font-size: 0.875rem; + font-weight: 600; + cursor: pointer; + transition: all 0.2s; +} + +.btn-submit:hover { + background: var(--primary-hover); +} + +.server-grid { + display: grid; + grid-template-columns: repeat(auto-fill, minmax(350px, 1fr)); + gap: 1.5rem; +} + +.empty-state { + grid-column: 1 / -1; + text-align: center; + padding: 4rem 2rem; + background: var(--surface-color); + border-radius: var(--radius-lg); + box-shadow: var(--shadow-sm); + border: 1px dashed var(--border-color); +} + +.empty-state p { + margin: 0.5rem 0; + font-size: 1.125rem; + color: var(--text-secondary); +} + +.empty-state .hint { + color: var(--text-light); + font-size: 0.875rem; +} + +@media (max-width: 768px) { + .app { + padding: 1rem; + } + + .app-header { + flex-direction: column; + gap: 1rem; + align-items: stretch; + text-align: center; + padding: 1rem; + } + + .app-header h1 { + justify-content: center; + } + + .server-grid { + grid-template-columns: 1fr; + } + + .add-form { + flex-direction: column; + padding: 1.5rem; + } + + .add-form input { + min-width: 100%; + } +} diff --git a/mengyamonitor-frontend/src/App.tsx b/mengyamonitor-frontend/src/App.tsx new file mode 100644 index 0000000..891f866 --- /dev/null +++ b/mengyamonitor-frontend/src/App.tsx @@ -0,0 +1,130 @@ +import { useState, useEffect } from 'react'; +import type { ServerConfig } from './types'; +import { loadServers, saveServers, removeServer } from './utils/storage'; +import { useServerMonitor } from './hooks/useServerMonitor'; +import { ServerCard } from './components/ServerCard/ServerCard'; +import { ServerDetail } from './components/ServerDetail/ServerDetail'; +import './App.css'; + +function App() { + const [servers, setServers] = useState([]); + const [showAddForm, setShowAddForm] = useState(false); + const [selectedServerId, setSelectedServerId] = useState(null); + const [newServerForm, setNewServerForm] = useState({ name: '', url: '' }); + + const statuses = useServerMonitor(servers, 2000); + + useEffect(() => { + const loaded = loadServers(); + setServers(loaded); + }, []); + + const handleAddServer = () => { + if (!newServerForm.name || !newServerForm.url) { + alert('请填写服务器名称和地址'); + return; + } + + const newServer: ServerConfig = { + id: Date.now().toString(), + name: newServerForm.name, + url: newServerForm.url, + enabled: true, + }; + + const updated = [...servers, newServer]; + setServers(updated); + saveServers(updated); + setNewServerForm({ name: '', url: '' }); + setShowAddForm(false); + }; + + const handleRemoveServer = (serverId: string) => { + if (confirm('确定要移除这个服务器吗?')) { + const updated = servers.filter(s => s.id !== serverId); + setServers(updated); + removeServer(serverId); + } + }; + + const handleShowDetail = (serverId: string) => { + setSelectedServerId(serverId); + }; + + const selectedStatus = selectedServerId ? statuses[selectedServerId] : null; + const selectedServer = servers.find(s => s.id === selectedServerId); + + return ( +
+
+

+ 萌芽监控面板 + 萌芽监控面板 +

+ +
+ + {showAddForm && ( +
+ setNewServerForm({ ...newServerForm, name: e.target.value })} + /> + setNewServerForm({ ...newServerForm, url: e.target.value })} + /> + +
+ )} + +
+ {servers.length === 0 ? ( +
+

还没有添加任何服务器

+

点击右上角的"添加服务器"按钮开始使用

+
+ ) : ( + servers.map((server) => { + const status = statuses[server.id]; + // Calculate storage usage (max of all mounts) + const storageUsage = status?.metrics?.storage?.reduce((max, s) => Math.max(max, s.usedPercent), 0) || 0; + + return ( + + ); + }) + )} +
+ + {selectedStatus?.metrics && selectedServer && ( + setSelectedServerId(null)} + /> + )} +
+ ); +} + +export default App; diff --git a/mengyamonitor-frontend/src/api/monitor.ts b/mengyamonitor-frontend/src/api/monitor.ts new file mode 100644 index 0000000..0821b37 --- /dev/null +++ b/mengyamonitor-frontend/src/api/monitor.ts @@ -0,0 +1,101 @@ +import type { ServerMetrics } from '../types'; + +export const fetchServerMetrics = async (serverUrl: string): Promise => { + // 测量客户端到服务器的延迟(使用健康检查端点) + const clientToServerLatency = await measureLatency(`${serverUrl}/api/health`); + + // 并行请求各个端点 + const endpoints = [ + '/api/metrics/cpu', + '/api/metrics/memory', + '/api/metrics/storage', + '/api/metrics/gpu', + '/api/metrics/network', + '/api/metrics/system', + '/api/metrics/docker', + '/api/metrics/latency', + ]; + + const results = await Promise.all( + endpoints.map(endpoint => + fetch(`${serverUrl}${endpoint}`, { + method: 'GET', + headers: { + 'Content-Type': 'application/json', + }, + }) + .then(res => { + if (!res.ok) throw new Error(`Failed to fetch ${endpoint}`); + return res.json(); + }) + .catch(err => { + console.error(`Error fetching ${endpoint}:`, err); + return null; + }) + ) + ); + + // 合并所有结果 + const [cpuRes, memRes, storageRes, gpuRes, networkRes, systemRes, dockerRes, latencyRes] = results; + + const system = systemRes?.data || {}; + const docker = dockerRes?.data || {}; + const latency = latencyRes?.data || {}; + + // 将 docker 数据合并到 system.dockerStats + return { + hostname: system.hostname || 'Unknown', + timestamp: new Date().toISOString(), + cpu: cpuRes?.data || {}, + memory: memRes?.data || {}, + storage: storageRes?.data || [], + gpu: gpuRes?.data || [], + network: networkRes?.data || [], + system: { + ...system, + dockerStats: docker + }, + os: system.os || { kernel: '', distro: '', architecture: '' }, + uptimeSeconds: system.uptimeSeconds || 0, + latency: { + clientToServer: clientToServerLatency, + external: latency.external || {}, + }, + }; +}; + +// 测量延迟 +async function measureLatency(url: string): Promise { + try { + const startTime = performance.now(); + const response = await fetch(url, { + method: 'GET', + headers: { + 'Content-Type': 'application/json', + }, + }); + const endTime = performance.now(); + + if (response.ok) { + return Math.round(endTime - startTime); + } + return -1; // 表示失败 + } catch { + return -1; // 表示超时或失败 + } +} + +export const checkServerHealth = async (serverUrl: string): Promise => { + try { + const url = `${serverUrl}/api/health`; + const response = await fetch(url, { + method: 'GET', + headers: { + 'Content-Type': 'application/json', + }, + }); + return response.ok; + } catch { + return false; + } +}; diff --git a/mengyamonitor-frontend/src/assets/react.svg b/mengyamonitor-frontend/src/assets/react.svg new file mode 100644 index 0000000..6c87de9 --- /dev/null +++ b/mengyamonitor-frontend/src/assets/react.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/mengyamonitor-frontend/src/components/Common/CircularProgress.css b/mengyamonitor-frontend/src/components/Common/CircularProgress.css new file mode 100644 index 0000000..95634c9 --- /dev/null +++ b/mengyamonitor-frontend/src/components/Common/CircularProgress.css @@ -0,0 +1,41 @@ +.circular-progress { + position: relative; + display: inline-block; +} + +.circular-progress svg { + transform: rotate(-90deg); +} + +.progress-ring-bg { + fill: none; + stroke: #e2e8f0; + transition: stroke 0.3s ease; +} + +.progress-ring-circle { + fill: none; + stroke-linecap: round; + transition: stroke-dashoffset 0.5s ease; +} + +.progress-value { + font-weight: 700; + fill: var(--text-main); + transform: rotate(90deg); + transform-origin: center; +} + +.progress-label { + fill: var(--text-secondary); + font-weight: 500; + transform: rotate(90deg); + transform-origin: center; +} + +.progress-sublabel { + fill: var(--text-light); + font-weight: 400; + transform: rotate(90deg); + transform-origin: center; +} diff --git a/mengyamonitor-frontend/src/components/Common/CircularProgress.tsx b/mengyamonitor-frontend/src/components/Common/CircularProgress.tsx new file mode 100644 index 0000000..186cf71 --- /dev/null +++ b/mengyamonitor-frontend/src/components/Common/CircularProgress.tsx @@ -0,0 +1,83 @@ +import './CircularProgress.css'; + +interface CircularProgressProps { + value: number; + label: string; + color?: string; + size?: number; + subLabel?: string; +} + +export const CircularProgress = ({ + value, + label, + color = 'var(--primary-color)', + size = 150, + subLabel +}: CircularProgressProps) => { + const radius = size * 0.36; // 36% of size + const circumference = 2 * Math.PI * radius; + const offset = circumference - (value / 100) * circumference; + const center = size / 2; + + // 根据尺寸动态计算字体大小 + const valueFontSize = size * 0.12; // 12% of size + const subLabelFontSize = size * 0.08; // 8% of size + const labelFontSize = size * 0.09; // 9% of size + + return ( +
+ + + + {subLabel ? ( + + {subLabel} + + ) : ( + + {Math.round(value)}% + + )} + + {label} + + +
+ ); +}; diff --git a/mengyamonitor-frontend/src/components/ServerCard/ServerCard.css b/mengyamonitor-frontend/src/components/ServerCard/ServerCard.css new file mode 100644 index 0000000..a6c690a --- /dev/null +++ b/mengyamonitor-frontend/src/components/ServerCard/ServerCard.css @@ -0,0 +1,381 @@ +.server-card { + background: var(--surface-color); + border-radius: var(--radius-lg); + padding: 1.5rem; + box-shadow: var(--shadow-sm); + transition: all 0.3s ease; + border: 1px solid var(--border-color); + display: flex; + flex-direction: column; + height: 100%; +} + +.server-card:hover { + box-shadow: var(--shadow-lg); + transform: translateY(-4px); + border-color: var(--primary-color); +} + +.server-card.online { + border-top: 4px solid var(--success-color); +} + +.server-card.offline { + border-top: 4px solid var(--text-light); + opacity: 0.9; +} + +.card-header { + display: flex; + justify-content: space-between; + align-items: flex-start; + margin-bottom: 0.5rem; +} + +.server-info { + display: flex; + align-items: center; + gap: 0.75rem; +} + +.status-indicator { + width: 10px; + height: 10px; + border-radius: 50%; + position: relative; +} + +.status-online { + background: var(--success-color); + box-shadow: 0 0 0 4px rgba(134, 239, 172, 0.2); +} + +.status-online::after { + content: ''; + position: absolute; + top: -4px; + left: -4px; + right: -4px; + bottom: -4px; + border-radius: 50%; + animation: pulse 2s infinite; + border: 1px solid var(--success-color); +} + +@keyframes pulse { + 0% { transform: scale(1); opacity: 0.5; } + 70% { transform: scale(1.5); opacity: 0; } + 100% { transform: scale(1); opacity: 0; } +} + +.status-offline { + background: var(--text-light); +} + +.server-name-group { + display: flex; + flex-direction: column; + gap: 0.25rem; +} + +.server-name { + margin: 0; + font-size: 1.125rem; + font-weight: 600; + color: var(--text-main); +} + +.server-hostname { + font-size: 0.75rem; + color: var(--text-light); + font-weight: 400; +} + +.btn-remove { + background: transparent; + border: none; + font-size: 1.5rem; + color: var(--text-light); + cursor: pointer; + padding: 0; + width: 24px; + height: 24px; + line-height: 24px; + border-radius: 50%; + display: flex; + align-items: center; + justify-content: center; + transition: all 0.2s; +} + +.btn-remove:hover { + background: var(--danger-color); + color: white; +} + +/* Main Metrics */ +.card-main-metrics { + flex: 1; + display: flex; + flex-direction: column; + gap: 1rem; +} + +.metrics-grid-main { + display: flex; + justify-content: space-around; + align-items: center; + padding: 0.5rem 0; +} + +.circular-progress-small { + display: flex; + align-items: center; + justify-content: center; +} + +.circular-progress-small .progress-ring-bg { + fill: none; + stroke: #e2e8f0; + stroke-width: 6; +} + +.circular-progress-small .progress-ring-circle { + fill: none; + stroke-width: 6; + stroke-linecap: round; + transition: stroke-dashoffset 0.5s ease; + transform: rotate(-90deg); + transform-origin: center; +} + +.circular-progress-small .progress-value-small { + font-size: 14px; + font-weight: 600; + fill: var(--text-main); +} + +.circular-progress-small .progress-label-small { + font-size: 10px; + fill: var(--text-secondary); +} + +/* Info Grid */ +.card-info-grid { + display: grid; + grid-template-columns: 1fr 1fr; + gap: 0.75rem; + padding: 0.75rem; + background: var(--bg-color); + border-radius: var(--radius-md); +} + +.info-section { + display: flex; + flex-direction: column; + gap: 0.5rem; +} + +.info-row { + display: flex; + justify-content: space-between; + align-items: center; + font-size: 0.75rem; + writing-mode: horizontal-tb; + text-orientation: mixed; +} + +.info-label { + color: var(--text-light); + font-weight: 500; + writing-mode: horizontal-tb; + text-orientation: mixed; + white-space: nowrap; +} + +.info-value { + color: var(--text-main); + font-weight: 600; + font-family: monospace; + font-size: 0.7rem; + max-width: 120px; + overflow: hidden; + text-overflow: ellipsis; + white-space: nowrap; + text-align: right; +} + +/* Performance Grid */ +.card-performance-grid { + display: grid; + grid-template-columns: 1fr 1fr; + gap: 0.75rem; + padding: 0.75rem; + background: var(--bg-color); + border-radius: var(--radius-md); +} + +.performance-section { + display: flex; + flex-direction: column; + gap: 0.5rem; +} + +.performance-header { + margin-bottom: 0.25rem; +} + +.performance-label { + font-size: 0.75rem; + font-weight: 600; + color: var(--text-secondary); +} + +.performance-row { + display: flex; + flex-direction: column; + gap: 0.4rem; +} + +.performance-item { + display: flex; + justify-content: space-between; + align-items: center; + font-size: 0.7rem; +} + +.performance-item-label { + color: var(--text-light); + font-weight: 500; +} + +.performance-item-value { + color: var(--primary-color); + font-weight: 600; + font-family: monospace; + text-align: right; +} + +/* Footer Info */ +.card-footer-info { + display: flex; + justify-content: space-between; + align-items: center; + padding: 0.75rem; + background: var(--bg-color); + border-radius: var(--radius-md); + margin-top: 0.5rem; +} + +.footer-row { + display: flex; + align-items: center; + gap: 0.5rem; + font-size: 0.75rem; +} + +.footer-label { + color: var(--text-light); + font-weight: 500; +} + +.footer-value { + color: var(--text-main); + font-weight: 600; + font-family: monospace; +} + + +/* Metrics List */ +.metrics-list { + display: flex; + flex-direction: column; + gap: 1rem; + margin-bottom: 1.5rem; +} + +.metric-item { + width: 100%; +} + +.metric-header { + display: flex; + justify-content: space-between; + margin-bottom: 0.25rem; + font-size: 0.875rem; +} + +.metric-label { + color: var(--text-secondary); + font-weight: 500; +} + +.metric-value { + font-weight: 600; + color: var(--text-main); +} + +.progress-bar-bg { + width: 100%; + height: 8px; + background: var(--bg-color); + border-radius: 4px; + overflow: hidden; +} + +.progress-bar-fill { + height: 100%; + border-radius: 4px; + transition: width 0.5s ease; +} + +.uptime-info { + display: flex; + justify-content: space-between; + font-size: 0.875rem; + padding-top: 0.5rem; + border-top: 1px solid var(--border-color); +} + +.uptime-label { + color: var(--text-light); +} + +.uptime-value { + color: var(--text-secondary); + font-family: monospace; +} + +.offline-state { + flex: 1; + display: flex; + align-items: center; + justify-content: center; + background: var(--bg-color); + border-radius: var(--radius-md); + margin-bottom: 1.5rem; + color: var(--text-light); +} + +.card-footer { + margin-top: auto; +} + +.btn-detail { + width: 100%; + background: var(--surface-hover); + border: 1px solid var(--border-color); + padding: 0.75rem; + border-radius: var(--radius-md); + font-size: 0.875rem; + font-weight: 600; + color: var(--text-secondary); + cursor: pointer; + transition: all 0.2s; +} + +.btn-detail:hover { + background: var(--primary-color); + color: white; + border-color: var(--primary-color); +} diff --git a/mengyamonitor-frontend/src/components/ServerCard/ServerCard.tsx b/mengyamonitor-frontend/src/components/ServerCard/ServerCard.tsx new file mode 100644 index 0000000..e574fb3 --- /dev/null +++ b/mengyamonitor-frontend/src/components/ServerCard/ServerCard.tsx @@ -0,0 +1,281 @@ +import type { ServerConfig, ServerMetrics } from '../../types'; +import { formatUptime, formatBytes } from '../../utils/format'; +import './ServerCard.css'; + +const CircularProgress = ({ value, label, color, size = 60 }: { value: number; label: string; color: string; size?: number }) => { + const radius = size * 0.36; + const circumference = 2 * Math.PI * radius; + const offset = circumference - (value / 100) * circumference; + + return ( +
+ + + + + {Math.round(value)}% + + + {label} + + +
+ ); +}; + +const LoadProgress = ({ value, maxValue = 1.0, size = 80 }: { value: number; maxValue?: number; size?: number }) => { + const percentage = Math.min((value / maxValue) * 100, 100); + const radius = size * 0.36; + const circumference = 2 * Math.PI * radius; + const offset = circumference - (percentage / 100) * circumference; + const color = percentage > 80 ? '#f87171' : percentage > 50 ? '#fde047' : '#86efac'; + + return ( +
+ + + + + {value.toFixed(2)} + + + 负载 + + +
+ ); +}; + +export interface ServerCardProps { + server: ServerConfig; + online: boolean; + metrics?: ServerMetrics; + cpuUsage?: number; + memoryUsage?: number; + storageUsage?: number; + uptime?: number; + onDetail: (serverId: string) => void; + onRemove: (serverId: string) => void; +} + +export const ServerCard = ({ + server, + online, + metrics, + cpuUsage = 0, + memoryUsage = 0, + storageUsage = 0, + uptime, + onDetail, + onRemove, +}: ServerCardProps) => { + // 从 URL 提取 IP 地址 + const extractIP = (url: string): string => { + try { + const urlObj = new URL(url); + return urlObj.hostname; + } catch { + return url; + } + }; + + // 获取主 IP 地址(从网络接口) + const getMainIP = (): string => { + if (metrics?.network && metrics.network.length > 0) { + const mainInterface = metrics.network.find(iface => + iface.ipAddress && iface.ipAddress !== 'N/A' && !iface.ipAddress.startsWith('127.') + ); + return mainInterface?.ipAddress || extractIP(server.url); + } + return extractIP(server.url); + }; + + const mainIP = getMainIP(); + const loadAverage = metrics?.cpu?.loadAverages?.[0] || 0; + const cpuCores = metrics?.cpu?.cores || 1; + const temperature = metrics?.system?.temperature || metrics?.cpu?.temperature || 0; + const hostname = metrics?.hostname || 'Unknown'; + const osDistro = metrics?.os?.distro || 'Unknown'; + const architecture = metrics?.os?.architecture || 'Unknown'; + const diskReadSpeed = metrics?.system?.diskReadSpeed || 0; + const diskWriteSpeed = metrics?.system?.diskWriteSpeed || 0; + const networkRxSpeed = metrics?.system?.networkRxSpeed || 0; + const networkTxSpeed = metrics?.system?.networkTxSpeed || 0; + const latency = metrics?.latency?.clientToServer || -1; + const processCount = metrics?.system?.processCount || 0; + const packageCount = metrics?.system?.packageCount || 0; + + // 延迟颜色判断 + const getLatencyColor = (latency: number): string => { + if (latency < 0) return '#ef4444'; // 超时/失败 - 红色 + if (latency < 50) return '#86efac'; // 低延迟 - 绿色 + if (latency < 200) return '#fde047'; // 中延迟 - 黄色 + return '#f87171'; // 高延迟 - 红色 + }; + + const latencyColor = getLatencyColor(latency); + + return ( +
+
+
+
+
+

{server.name}

+ {online && hostname !== 'Unknown' && ( + {hostname} + )} +
+
+ +
+ + {online && metrics ? ( + <> +
+
+ {loadAverage > 0 && ( + + )} + 80 ? '#f87171' : '#86efac'} + size={80} + /> + 80 ? '#fde047' : '#6ee7b7'} + size={80} + /> + 90 ? '#f87171' : '#a7f3d0'} + size={80} + /> +
+ +
+
+
+ 系统 + {osDistro} +
+
+ 架构 + {architecture} +
+
+ IP + {mainIP} +
+
+ +
+
+ 温度 + {temperature > 0 ? `${temperature.toFixed(1)}°C` : 'N/A'} +
+
+ 进程 + {processCount} +
+
+ 软件包 + {packageCount} +
+
+
+ +
+
+
+ 磁盘 +
+
+ + 读取 + {diskReadSpeed > 0 ? formatBytes(diskReadSpeed * 1024 * 1024) + '/s' : '0 B/s'} + + + 写入 + {diskWriteSpeed > 0 ? formatBytes(diskWriteSpeed * 1024 * 1024) + '/s' : '0 B/s'} + +
+
+ +
+
+ 网络 +
+
+ + 下载 + {networkRxSpeed > 0 ? formatBytes(networkRxSpeed * 1024 * 1024) + '/s' : '0 B/s'} + + + 上传 + {networkTxSpeed > 0 ? formatBytes(networkTxSpeed * 1024 * 1024) + '/s' : '0 B/s'} + +
+
+
+ +
+
+ 运行时间 + {uptime !== undefined ? formatUptime(uptime) : 'N/A'} +
+
+ 延迟 + + {latency > 0 ? `${latency} ms` : latency === -1 ? '超时' : 'N/A'} + +
+
+
+ + ) : ( +
+

服务器连接断开

+
+ )} + +
+ +
+
+ ); +}; diff --git a/mengyamonitor-frontend/src/components/ServerDetail/ServerDetail.css b/mengyamonitor-frontend/src/components/ServerDetail/ServerDetail.css new file mode 100644 index 0000000..a26e541 --- /dev/null +++ b/mengyamonitor-frontend/src/components/ServerDetail/ServerDetail.css @@ -0,0 +1,1088 @@ +.modal-overlay { + position: fixed; + top: 0; + left: 0; + right: 0; + bottom: 0; + background: rgba(0, 0, 0, 0.75); + backdrop-filter: blur(8px); + display: flex; + align-items: center; + justify-content: center; + z-index: 1000; + padding: 20px; + animation: fadeIn 0.3s cubic-bezier(0.4, 0, 0.2, 1); +} + +@keyframes fadeIn { + from { + opacity: 0; + } + to { + opacity: 1; + } +} + +.modal-content { + background: var(--surface-color); + border-radius: 24px; + width: 100%; + max-width: 1200px; + max-height: 90vh; + overflow: hidden; + box-shadow: var(--shadow-lg); + border: 1px solid var(--border-color); + animation: slideUp 0.4s cubic-bezier(0.4, 0, 0.2, 1); + display: flex; + flex-direction: column; +} + +@keyframes slideUp { + from { transform: translateY(20px); opacity: 0; } + to { transform: translateY(0); opacity: 1; } +} + +.modal-header { + display: flex; + justify-content: space-between; + align-items: center; + padding: 24px 32px; + border-bottom: 1px solid var(--border-color); + position: sticky; + top: 0; + background: var(--surface-color); + z-index: 10; +} + +.modal-header h2 { + margin: 0; + font-size: 20px; + font-weight: 700; + color: var(--text-main); + letter-spacing: -0.2px; +} + +.btn-close { + background: var(--surface-hover); + border: 1px solid var(--border-color); + font-size: 20px; + color: var(--text-secondary); + cursor: pointer; + padding: 0; + width: 36px; + height: 36px; + border-radius: 10px; + display: flex; + align-items: center; + justify-content: center; + transition: all 0.3s ease; +} + +.btn-close:hover { + background: var(--bg-color); + border-color: var(--border-color); + color: var(--text-main); +} + +.modal-body { + padding: 24px; + overflow-y: auto; + flex: 1; +} + +.modal-body::-webkit-scrollbar { + width: 8px; +} + +.modal-body::-webkit-scrollbar-track { + background: rgba(255, 255, 255, 0.03); +} + +.modal-body::-webkit-scrollbar-thumb { + background: rgba(255, 255, 255, 0.15); + border-radius: 10px; +} + +.modal-body::-webkit-scrollbar-thumb:hover { + background: rgba(255, 255, 255, 0.25); +} + +.detail-section { + margin-bottom: 14px; + background: var(--surface-color); + border-radius: 12px; + padding: 14px; + border: 1px solid var(--border-color); + transition: all 0.3s ease; +} + +.detail-section:hover { + background: var(--surface-hover); + border-color: var(--border-color); +} + +.detail-section:last-child { + margin-bottom: 0; +} + +.detail-section h3 { + margin: 0 0 12px 0; + font-size: 15px; + font-weight: 600; + color: var(--text-main); + display: flex; + align-items: center; + gap: 8px; +} + +.detail-section h3::before { + content: ''; + width: 4px; + height: 20px; + background: linear-gradient(135deg, #86efac 0%, #a7f3d0 100%); + border-radius: 2px; +} + +.detail-grid { + display: grid; + grid-template-columns: repeat(auto-fit, minmax(240px, 1fr)); + gap: 12px; +} + +.cpu-section-content .detail-grid, +.memory-section-content .detail-grid { + grid-template-columns: repeat(2, 1fr); + width: 100%; +} + +.detail-item { + display: flex; + flex-direction: column; + gap: 6px; + padding: 12px 14px; + background: var(--surface-hover); + border-radius: 10px; + border: 1px solid var(--border-color); + transition: all 0.3s cubic-bezier(0.4, 0, 0.2, 1); +} + +.detail-item:hover { + background: var(--bg-color); + border-color: var(--border-color); +} + +.item-label { + font-size: 12px; + color: var(--text-secondary); + font-weight: 600; + letter-spacing: 0.5px; +} + +.item-value { + font-size: 15px; + color: var(--text-main); + font-weight: 700; + font-family: 'SF Mono', 'Consolas', 'Monaco', monospace; + letter-spacing: -0.2px; +} + +.item-value.highlight { + background: linear-gradient(135deg, #86efac 0%, #a7f3d0 100%); + -webkit-background-clip: text; + background-clip: text; + -webkit-text-fill-color: transparent; + font-size: 20px; +} + +/* System Info Styles */ +.system-info-container { + display: flex; + flex-direction: column; + gap: 14px; +} + +.system-info-group { + display: flex; + flex-direction: column; + gap: 8px; +} + +.info-group-title { + margin: 0; + font-size: 12px; + font-weight: 700; + color: var(--text-secondary); + text-transform: uppercase; + letter-spacing: 1px; + padding-bottom: 6px; + border-bottom: 1px solid var(--border-color); +} + +.system-info-grid { + display: grid; + grid-template-columns: repeat(auto-fit, minmax(200px, 1fr)); + gap: 8px; +} + +.system-info-item { + display: flex; + align-items: flex-start; + padding: 10px 12px; + background: #ffffff; + border-radius: 8px; + border: 1px solid var(--border-color); + transition: all 0.3s cubic-bezier(0.4, 0, 0.2, 1); + position: relative; + overflow: hidden; +} + +.system-info-item::before { + content: ''; + position: absolute; + top: 0; + left: 0; + width: 3px; + height: 100%; + background: linear-gradient(180deg, #86efac 0%, #a7f3d0 100%); + opacity: 0; + transition: opacity 0.3s ease; +} + +.system-info-item:hover { + transform: translateY(-2px); + box-shadow: 0 6px 20px rgba(0, 0, 0, 0.08); + border-color: rgba(134, 239, 172, 0.3); + background: #ffffff; +} + +.system-info-item:hover::before { + opacity: 1; +} + +.system-info-item-full { + grid-column: 1 / -1; +} + +.system-info-item-highlight { + background: #ffffff; + border-color: rgba(134, 239, 172, 0.2); +} + +.system-info-item-highlight:hover { + background: #ffffff; + border-color: rgba(134, 239, 172, 0.4); +} + +.system-info-item-temperature { + background: linear-gradient(135deg, rgba(248, 113, 113, 0.05) 0%, rgba(253, 224, 71, 0.05) 100%); + border-color: rgba(248, 113, 113, 0.2); +} + +.system-info-item-temperature:hover { + background: linear-gradient(135deg, rgba(248, 113, 113, 0.1) 0%, rgba(253, 224, 71, 0.1) 100%); + border-color: rgba(248, 113, 113, 0.4); +} + +.system-info-content { + flex: 1; + display: flex; + flex-direction: column; + gap: 4px; + min-width: 0; + width: 100%; +} + +.system-info-label { + font-size: 10px; + color: var(--text-secondary); + font-weight: 600; + text-transform: uppercase; + letter-spacing: 0.5px; +} + +.system-info-value { + font-size: 13px; + color: var(--text-main); + font-weight: 700; + word-break: break-word; + line-height: 1.3; +} + +.system-info-value.highlight { + background: linear-gradient(135deg, #86efac 0%, #a7f3d0 100%); + -webkit-background-clip: text; + background-clip: text; + -webkit-text-fill-color: transparent; + font-size: 16px; +} + +.system-info-value-long { + font-size: 12px; + font-family: 'SF Mono', 'Consolas', 'Monaco', monospace; + font-weight: 500; + line-height: 1.5; + color: var(--text-secondary); +} + +.system-info-sub { + font-size: 11px; + color: var(--text-secondary); + font-weight: 500; +} + +.temperature-value { + color: #f87171; + font-size: 16px; + font-weight: 700; +} + +@media (max-width: 768px) { + .system-info-grid { + grid-template-columns: 1fr; + } + + .system-info-item-full { + grid-column: 1; + } +} + +/* GPU Items */ +.gpu-item { + margin-bottom: 1.5rem; + padding: 1.5rem; + background: var(--surface-hover); + border-radius: var(--radius-lg); + border: 1px solid var(--border-color); +} + +.gpu-item:last-child { + margin-bottom: 0; +} + +.gpu-header { + display: flex; + justify-content: space-between; + align-items: center; + margin-bottom: 1rem; +} + +.gpu-header strong { + color: var(--text-main); + font-size: 1rem; +} + +.gpu-status { + padding: 4px 12px; + border-radius: 12px; + font-size: 0.75rem; + font-weight: 600; + text-transform: uppercase; +} + +.gpu-status.ok { + background: rgba(134, 239, 172, 0.1); + color: var(--success-color); +} + +.gpu-status.error { + background: rgba(248, 113, 113, 0.1); + color: var(--danger-color); +} + +.gpu-bar-container { + display: flex; + align-items: center; + gap: 1rem; + margin-bottom: 1rem; +} + +.gpu-bar-label { + font-size: 0.875rem; + color: var(--text-secondary); + font-weight: 500; + min-width: 60px; +} + +.gpu-bar-bg { + flex: 1; + height: 8px; + background: var(--bg-color); + border-radius: 4px; + overflow: hidden; +} + +.gpu-bar { + height: 100%; + background: linear-gradient(90deg, #a7f3d0, #d1fae5); + border-radius: 4px; + transition: width 0.5s ease; +} + +.gpu-bar-value { + font-size: 0.875rem; + font-weight: 600; + color: var(--text-main); + min-width: 50px; + text-align: right; +} + +/* Core Usage Circular */ +.per-core-section { + margin-top: 1.5rem; + padding: 1.5rem; + background: var(--surface-hover); + border-radius: var(--radius-lg); + border: 1px solid var(--border-color); +} + +.per-core-section h4 { + margin: 0 0 1rem 0; + font-size: 1rem; + color: var(--text-main); + font-weight: 600; +} + +.core-grid-circular { + display: grid; + grid-template-columns: repeat(auto-fill, minmax(max(160px, 17%), 1fr)); + gap: 1.5rem; + justify-items: center; +} + +.core-item-circular { + display: flex; + flex-direction: column; + align-items: center; + gap: 0.5rem; +} + +/* CPU & Memory Section Circular */ +.cpu-section-content, +.memory-section-content { + display: flex; + gap: 1rem; + align-items: center; + background: var(--surface-hover); + padding: 1rem; + border-radius: var(--radius-lg); + border: 1px solid var(--border-color); +} + +.cpu-circular-group { + display: flex; + gap: 1.5rem; + flex-shrink: 0; +} + +.cpu-circular, +.cpu-load-circular, +.memory-circular { + flex-shrink: 0; +} + +/* Storage Circular */ +.storage-grid-circular { + display: grid; + grid-template-columns: repeat(auto-fill, minmax(180px, 1fr)); + gap: 1.5rem; +} + +.storage-item-circular { + display: flex; + flex-direction: column; + align-items: center; + background: var(--surface-hover); + padding: 1rem; + border-radius: var(--radius-md); + border: 1px solid var(--border-color); +} + +.storage-details-mini { + margin-top: 0.5rem; + text-align: center; + font-size: 0.75rem; + color: var(--text-secondary); +} + +@media (max-width: 768px) { + .modal-content { + max-width: 100%; + max-height: 100vh; + border-radius: 0; + } + + .detail-grid { + grid-template-columns: 1fr; + } + + .cpu-section-content, + .memory-section-content { + flex-direction: column; + text-align: center; + } + + .cpu-circular-group { + flex-direction: column; + align-items: center; + } + + .gpu-bar-container { + flex-wrap: wrap; + gap: 0.5rem; + } + + .gpu-bar-bg { + min-width: 100%; + order: 3; + } +} + +/* Docker Containers Section */ +.containers-section { + margin-top: 1rem; + padding: 1rem; + background: #f8f9fa; + border-radius: 8px; +} + +.containers-section h4 { + margin: 0 0 1rem 0; + font-size: 15px; + color: #2c3e50; + font-weight: 600; +} + +.containers-grid { + display: grid; + grid-template-columns: repeat(auto-fill, minmax(350px, 1fr)); + gap: 12px; +} + +/* Docker Images Section */ +.docker-images-section { + margin-top: 1rem; + padding: 1rem; + background: #f8f9fa; + border-radius: 8px; +} + +.docker-images-section h4 { + margin: 0 0 1rem 0; + font-size: 15px; + color: #2c3e50; + font-weight: 600; +} + +/* Docker Containers and Images Names */ +.docker-containers-section { + margin-top: 16px; + padding: 16px; + background: var(--surface-hover); + border-radius: 12px; + border: 1px solid var(--border-color); +} + +.docker-containers-section h4 { + margin: 0 0 12px 0; + font-size: 14px; + color: var(--text-main); + font-weight: 600; +} + +.docker-names-list { + display: flex; + flex-wrap: wrap; + gap: 8px; +} + +.docker-name-tag { + display: inline-block; + padding: 6px 12px; + border-radius: 8px; + font-size: 12px; + font-weight: 500; + font-family: 'SF Mono', 'Consolas', 'Monaco', monospace; + transition: all 0.3s ease; +} + +.docker-name-tag.running { + background: linear-gradient(135deg, rgba(134, 239, 172, 0.1) 0%, rgba(167, 243, 208, 0.1) 100%); + color: #059669; + border: 1px solid rgba(134, 239, 172, 0.3); +} + +.docker-name-tag.stopped { + background: linear-gradient(135deg, rgba(148, 163, 184, 0.1) 0%, rgba(203, 213, 225, 0.1) 100%); + color: #64748b; + border: 1px solid rgba(148, 163, 184, 0.3); +} + +.docker-name-tag.image { + background: linear-gradient(135deg, rgba(134, 239, 172, 0.08) 0%, rgba(167, 243, 208, 0.08) 100%); + color: var(--text-main); + border: 1px solid rgba(134, 239, 172, 0.2); +} + +.docker-name-tag:hover { + transform: translateY(-1px); + box-shadow: 0 2px 8px rgba(0, 0, 0, 0.1); +} + +/* System Logs Section */ +.logs-container { + background: #0a0c10; + color: rgba(255, 255, 255, 0.8); + border-radius: 12px; + padding: 20px; + font-family: 'SF Mono', 'Consolas', 'Monaco', monospace; + font-size: 13px; + max-height: 400px; + overflow-y: auto; + border: 1px solid rgba(255, 255, 255, 0.1); + line-height: 1.7; +} + +.logs-container::-webkit-scrollbar { + width: 6px; +} + +.logs-container::-webkit-scrollbar-thumb { + background: rgba(255, 255, 255, 0.2); + border-radius: 10px; +} + +.log-line { + padding: 4px 0; + white-space: pre-wrap; + word-wrap: break-word; + border-bottom: 1px solid rgba(255, 255, 255, 0.03); + transition: all 0.2s ease; +} + +.log-line:hover { + background: rgba(255, 255, 255, 0.05); + color: rgba(255, 255, 255, 1); +} + +.log-line:last-child { + border-bottom: none; +} + +/* Process List Styles */ +.process-list { + display: flex; + flex-direction: column; + gap: 8px; +} + +.process-item { + display: flex; + gap: 12px; + padding: 12px 14px; + background: #ffffff; + border-radius: 12px; + border: 1px solid var(--border-color); + transition: all 0.3s cubic-bezier(0.4, 0, 0.2, 1); + position: relative; + overflow: hidden; +} + +.process-item::before { + content: ''; + position: absolute; + top: 0; + left: 0; + right: 0; + height: 3px; + background: linear-gradient(90deg, #86efac 0%, #a7f3d0 50%, #d1fae5 100%); + opacity: 0; + transition: opacity 0.3s ease; +} + +.process-item:hover { + transform: translateY(-2px); + box-shadow: 0 8px 24px rgba(0, 0, 0, 0.08); + border-color: rgba(134, 239, 172, 0.3); + background: #ffffff; +} + +.process-item:hover::before { + opacity: 1; +} + +.process-rank { + display: flex; + align-items: center; + justify-content: center; + min-width: 32px; + height: 32px; + background: linear-gradient(135deg, #86efac 0%, #a7f3d0 100%); + color: white; + font-weight: 700; + font-size: 14px; + border-radius: 8px; + box-shadow: 0 2px 8px rgba(134, 239, 172, 0.3); + flex-shrink: 0; +} + +.process-item:nth-child(1) .process-rank { + background: linear-gradient(135deg, #fde047 0%, #facc15 100%); + box-shadow: 0 4px 12px rgba(253, 224, 71, 0.3); +} + +.process-item:nth-child(2) .process-rank { + background: linear-gradient(135deg, #a7f3d0 0%, #86efac 100%); + box-shadow: 0 4px 12px rgba(167, 243, 208, 0.3); +} + +.process-item:nth-child(3) .process-rank { + background: linear-gradient(135deg, #d1fae5 0%, #a7f3d0 100%); + box-shadow: 0 4px 12px rgba(209, 250, 229, 0.3); +} + +.process-content { + flex: 1; + display: flex; + flex-direction: column; + gap: 10px; +} + +.process-header { + display: flex; + align-items: center; + justify-content: space-between; +} + +.process-name-group { + display: flex; + flex-direction: column; + gap: 3px; +} + +.process-name { + font-size: 14px; + font-weight: 700; + color: var(--text-main); + letter-spacing: -0.3px; +} + +.process-pid { + font-size: 11px; + color: var(--text-secondary); + font-weight: 500; + font-family: 'SF Mono', 'Consolas', 'Monaco', monospace; + background: var(--surface-hover); + padding: 3px 8px; + border-radius: 5px; + display: inline-block; + width: fit-content; +} + +.process-metrics { + display: flex; + flex-direction: column; + gap: 8px; +} + +.process-metric-item { + display: flex; + flex-direction: column; + gap: 4px; +} + +.metric-header { + display: flex; + justify-content: space-between; + align-items: center; +} + +.metric-label { + font-size: 11px; + color: var(--text-secondary); + font-weight: 600; + text-transform: uppercase; + letter-spacing: 0.5px; +} + +.metric-value { + font-size: 12px; + font-weight: 700; + color: var(--text-main); + font-family: 'SF Mono', 'Consolas', 'Monaco', monospace; +} + +.metric-bar-container { + width: 100%; + height: 6px; + background: var(--surface-hover); + border-radius: 4px; + overflow: hidden; + position: relative; +} + +.metric-bar { + height: 100%; + border-radius: 4px; + transition: width 0.6s cubic-bezier(0.4, 0, 0.2, 1); + position: relative; + overflow: hidden; +} + +.metric-bar::after { + content: ''; + position: absolute; + top: 0; + left: 0; + right: 0; + bottom: 0; + background: linear-gradient(90deg, + transparent 0%, + rgba(255, 255, 255, 0.3) 50%, + transparent 100%); + animation: shimmer 2s infinite; +} + +@keyframes shimmer { + 0% { transform: translateX(-100%); } + 100% { transform: translateX(100%); } +} + +.cpu-bar { + background: linear-gradient(90deg, #86efac 0%, #a7f3d0 50%, #d1fae5 100%); + box-shadow: 0 0 8px rgba(134, 239, 172, 0.4); +} + +.memory-bar { + background: linear-gradient(90deg, #6ee7b7 0%, #86efac 50%, #a7f3d0 100%); + box-shadow: 0 0 8px rgba(110, 231, 183, 0.4); +} + +.process-command { + display: flex; + align-items: center; + gap: 6px; + padding: 6px 10px; + background: var(--surface-hover); + border-radius: 6px; + border: 1px solid var(--border-color); + font-size: 11px; + color: var(--text-secondary); + font-family: 'SF Mono', 'Consolas', 'Monaco', monospace; + overflow: hidden; + position: relative; +} + +.command-icon { + font-size: 12px; + flex-shrink: 0; + opacity: 0.7; +} + +.command-text { + flex: 1; + overflow: hidden; + text-overflow: ellipsis; + white-space: nowrap; + color: var(--text-main); +} + +/* Network Interface Styles */ +.network-list { + display: flex; + flex-direction: column; + gap: 10px; +} + +.network-item { + padding: 12px 14px; + background: #ffffff; + border-radius: 10px; + border: 1px solid var(--border-color); + transition: all 0.3s cubic-bezier(0.4, 0, 0.2, 1); + position: relative; + overflow: hidden; +} + +.network-item::before { + content: ''; + position: absolute; + top: 0; + left: 0; + right: 0; + height: 3px; + background: linear-gradient(90deg, #86efac 0%, #a7f3d0 50%, #d1fae5 100%); + opacity: 0; + transition: opacity 0.3s ease; +} + +.network-item:hover { + transform: translateY(-2px); + box-shadow: 0 8px 24px rgba(0, 0, 0, 0.08); + border-color: rgba(134, 239, 172, 0.3); + background: #ffffff; +} + +.network-item:hover::before { + opacity: 1; +} + +.network-header { + margin-bottom: 10px; + padding-bottom: 10px; + border-bottom: 1px solid var(--border-color); +} + +.network-name-group { + display: flex; + align-items: center; + gap: 12px; +} + +.network-icon { + font-size: 20px; + width: 36px; + height: 36px; + display: flex; + align-items: center; + justify-content: center; + background: linear-gradient(135deg, #86efac 0%, #a7f3d0 100%); + border-radius: 8px; + flex-shrink: 0; +} + +.network-name-info { + display: flex; + flex-direction: column; + gap: 4px; + flex: 1; +} + +.network-name { + font-size: 15px; + font-weight: 700; + color: var(--text-main); + letter-spacing: -0.3px; +} + +.network-ip { + font-size: 12px; + font-weight: 500; + font-family: 'SF Mono', 'Consolas', 'Monaco', monospace; + padding: 4px 10px; + border-radius: 6px; + display: inline-block; + width: fit-content; + transition: all 0.3s ease; +} + +.network-ip.has-ip { + background: linear-gradient(135deg, rgba(134, 239, 172, 0.1) 0%, rgba(167, 243, 208, 0.1) 100%); + color: #86efac; + border: 1px solid rgba(134, 239, 172, 0.2); +} + +.network-ip.no-ip { + background: var(--surface-hover); + color: var(--text-secondary); + border: 1px solid var(--border-color); +} + +.network-details { + display: flex; + flex-direction: column; + gap: 10px; +} + +.network-detail-row { + display: grid; + grid-template-columns: repeat(auto-fit, minmax(200px, 1fr)); + gap: 10px; +} + +.network-detail-item { + display: flex; + flex-direction: column; + gap: 4px; + padding: 10px 12px; + background: var(--surface-hover); + border-radius: 8px; + border: 1px solid var(--border-color); +} + +.network-detail-label { + font-size: 11px; + color: var(--text-secondary); + font-weight: 600; + text-transform: uppercase; + letter-spacing: 0.5px; +} + +.network-detail-value { + font-size: 13px; + font-weight: 700; + color: var(--text-main); + font-family: 'SF Mono', 'Consolas', 'Monaco', monospace; +} + +.network-traffic { + display: grid; + grid-template-columns: repeat(auto-fit, minmax(200px, 1fr)); + gap: 10px; +} + +.traffic-item { + padding: 10px 12px; + background: var(--surface-hover); + border-radius: 8px; + border: 1px solid var(--border-color); + transition: all 0.3s ease; +} + +.traffic-item:hover { + background: var(--bg-color); + border-color: var(--border-color); +} + +.traffic-header { + display: flex; + align-items: center; + justify-content: space-between; + gap: 8px; +} + +.traffic-label { + font-size: 11px; + color: var(--text-secondary); + font-weight: 600; +} + +.traffic-value { + font-size: 13px; + font-weight: 700; + color: var(--text-main); + font-family: 'SF Mono', 'Consolas', 'Monaco', monospace; +} + +@media (max-width: 768px) { + .containers-grid { + grid-template-columns: 1fr; + } + + .process-item { + flex-direction: column; + gap: 10px; + } + + .process-rank { + align-self: flex-start; + } + + .process-metrics { + gap: 8px; + } + + .network-detail-row, + .network-traffic { + grid-template-columns: 1fr; + } +} diff --git a/mengyamonitor-frontend/src/components/ServerDetail/ServerDetail.tsx b/mengyamonitor-frontend/src/components/ServerDetail/ServerDetail.tsx new file mode 100644 index 0000000..c9f43c7 --- /dev/null +++ b/mengyamonitor-frontend/src/components/ServerDetail/ServerDetail.tsx @@ -0,0 +1,521 @@ +import type { ServerMetrics } from '../../types'; +import { formatBytes, formatUptime, formatPercent } from '../../utils/format'; +import { CircularProgress } from '../Common/CircularProgress'; +import './ServerDetail.css'; + +interface ServerDetailProps { + metrics: ServerMetrics; + serverName: string; + onClose: () => void; +} + +export const ServerDetail = ({ metrics, serverName, onClose }: ServerDetailProps) => { + return ( +
+
e.stopPropagation()}> +
+

{serverName} - 详细信息

+ +
+ +
+
+

系统信息

+
+ {/* 系统基本信息 */} +
+

基本信息

+
+
+
+ 主机名 + {metrics.hostname} +
+
+
+
+ 操作系统 + {metrics.os.distro} +
+
+ +
+
+ 架构 + {metrics.os.architecture} +
+
+ +
+
+ 内核版本 + {metrics.os.kernel} +
+
+ +
+
+ + {/* 运行时信息 */} +
+

运行时信息

+
+
+
+ 运行时间 + {formatUptime(metrics.uptimeSeconds)} +
+
+
+
+ 进程数量 + {metrics.system.processCount} +
+
+ {metrics.system.packageCount > 0 && ( +
+
+ 已安装软件包 + {metrics.system.packageCount} ({metrics.system.packageManager}) +
+
+ )} +
+
+ + {/* 性能指标 */} +
+

性能指标

+
+
+
+ 磁盘读取 + {metrics.system.diskReadSpeed.toFixed(2)} MB/s +
+
+
+
+ 磁盘写入 + {metrics.system.diskWriteSpeed.toFixed(2)} MB/s +
+
+ {metrics.system.temperature && metrics.system.temperature > 0 && ( +
+
+ 系统温度 + {metrics.system.temperature.toFixed(1)}°C +
+
+ )} + {(metrics.system.networkRxSpeed !== undefined || metrics.system.networkTxSpeed !== undefined) && ( + <> +
+
+ 网络下载 + {(metrics.system.networkRxSpeed || 0).toFixed(2)} MB/s +
+
+
+
+ 网络上传 + {(metrics.system.networkTxSpeed || 0).toFixed(2)} MB/s +
+
+ + )} +
+
+ + {/* 延迟信息 */} + {metrics.latency && ( +
+

延迟检测

+
+
+
+ 客户端到服务器 + + {metrics.latency.clientToServer >= 0 ? `${metrics.latency.clientToServer} ms` : '超时'} + +
+
+ {metrics.latency.external && ( + <> +
+
+ 百度 (baidu.com) + {metrics.latency.external['baidu.com'] || '超时'} +
+
+
+
+ 谷歌 (google.com) + {metrics.latency.external['google.com'] || '超时'} +
+
+
+
+ GitHub (github.com) + {metrics.latency.external['github.com'] || '超时'} +
+
+ + )} +
+
+ )} +
+
+ +
+

CPU

+
+
+
+ 80 ? '#f87171' : '#86efac'} + subLabel={formatPercent(metrics.cpu.usagePercent)} + /> +
+ {metrics.cpu.loadAverages && metrics.cpu.loadAverages.length > 0 && metrics.cpu.cores > 0 && ( +
+ 0.8 ? '#fde047' : '#a7f3d0'} + subLabel={metrics.cpu.loadAverages[0].toFixed(2)} + /> +
+ )} +
+
+
+ 型号 + {metrics.cpu.model} +
+
+ 核心数 + {metrics.cpu.cores} +
+
+ 使用率 + {formatPercent(metrics.cpu.usagePercent)} +
+ {metrics.cpu.temperature && metrics.cpu.temperature > 0 && ( +
+ 温度 + {metrics.cpu.temperature.toFixed(1)}°C +
+ )} + {metrics.cpu.loadAverages && metrics.cpu.loadAverages.length > 0 && ( +
+ 负载平均值 + {metrics.cpu.loadAverages.join(' / ')} +
+ )} +
+
+ + {metrics.cpu.perCoreUsage && metrics.cpu.perCoreUsage.length > 0 && ( +
+

各核心使用率

+
+ {metrics.cpu.perCoreUsage.map((core) => ( +
+ 80 ? '#f87171' : '#86efac'} + subLabel={formatPercent(core.percent)} + /> +
+ ))} +
+
+ )} +
+ +
+

内存

+
+
+ 80 ? '#fde047' : '#6ee7b7'} + subLabel={formatPercent(metrics.memory.usedPercent)} + /> +
+
+
+ 总容量 + {formatBytes(metrics.memory.totalBytes)} +
+
+ 已使用 + {formatBytes(metrics.memory.usedBytes)} +
+
+ 可用 + {formatBytes(metrics.memory.freeBytes)} +
+
+ 使用率 + {formatPercent(metrics.memory.usedPercent)} +
+
+
+
+ +
+

存储

+
+ {metrics.storage.map((disk, index) => ( +
+ 90 ? '#f87171' : '#a7f3d0'} + subLabel={formatPercent(disk.usedPercent)} + /> +
+
总: {formatBytes(disk.totalBytes)}
+
用: {formatBytes(disk.usedBytes)}
+
+
+ ))} +
+
+ + {metrics.gpu && metrics.gpu.length > 0 && metrics.gpu[0].status !== 'not_available' && ( +
+

GPU

+ {metrics.gpu.map((gpu, index) => ( +
+
+ {gpu.name} + {gpu.status} +
+
+
利用率
+
+
+
+
{formatPercent(gpu.utilizationPercent)}
+
+
+
+ 显存总量 + {gpu.memoryTotalMB} MB +
+
+ 显存使用 + {gpu.memoryUsedMB} MB +
+ {gpu.temperature && gpu.temperature > 0 && ( +
+ 温度 + {gpu.temperature.toFixed(1)}°C +
+ )} +
+
+ ))} +
+ )} + + {metrics.network && metrics.network.length > 0 && ( +
+

网络接口

+
+ {metrics.network.map((iface, index) => ( +
+
+
+
+ {iface.name} + {iface.ipAddress && ( + + {iface.ipAddress} + + )} +
+
+
+
+ {(iface.macAddress || iface.ipAddress) && ( +
+ {iface.macAddress && ( +
+ MAC 地址 + {iface.macAddress} +
+ )} + {iface.ipAddress && ( +
+ IP 地址 + {iface.ipAddress} +
+ )} +
+ )} +
+
+
+ 接收流量 + {formatBytes(iface.rxBytes)} +
+
+
+
+ 发送流量 + {formatBytes(iface.txBytes)} +
+
+
+
+
+ ))} +
+
+ )} + + {metrics.system.topProcesses && metrics.system.topProcesses.length > 0 && ( +
+

Top 5 进程

+
+ {metrics.system.topProcesses.map((proc, index) => ( +
+
{index + 1}
+
+
+
+ {proc.name} + PID: {proc.pid} +
+
+
+
+
+ CPU + {proc.cpu}% +
+
+
+
+
+
+
+ 内存 + {proc.memory}% ({proc.memoryMB.toFixed(1)} MB) +
+
+
+
+
+
+
+
+ ))} +
+
+ )} + + {metrics.system.dockerStats && metrics.system.dockerStats.available && ( +
+

Docker 信息

+ {metrics.system.dockerStats.version && ( +
+
+ Docker 版本 + {metrics.system.dockerStats.version} +
+
+ 运行中的容器 + {metrics.system.dockerStats.running} +
+
+ 停止的容器 + {metrics.system.dockerStats.stopped} +
+
+ 镜像数量 + {metrics.system.dockerStats.imageCount} +
+
+ )} + + {metrics.system.dockerStats.runningNames && metrics.system.dockerStats.runningNames.length > 0 && ( +
+

运行中的容器

+
+ {metrics.system.dockerStats.runningNames.map((name, index) => ( + + {name} + + ))} +
+
+ )} + + {metrics.system.dockerStats.stoppedNames && metrics.system.dockerStats.stoppedNames.length > 0 && ( +
+

停止的容器

+
+ {metrics.system.dockerStats.stoppedNames.map((name, index) => ( + + {name} + + ))} +
+
+ )} + + {metrics.system.dockerStats.imageNames && metrics.system.dockerStats.imageNames.length > 0 && ( +
+

镜像列表

+
+ {metrics.system.dockerStats.imageNames.map((name, index) => ( + + {name} + + ))} +
+
+ )} +
+ )} + + {metrics.system.systemLogs && metrics.system.systemLogs.length > 0 && ( +
+

系统日志

+
+ {metrics.system.systemLogs.map((log, index) => ( +
+ {log} +
+ ))} +
+
+ )} +
+
+
+ ); +}; diff --git a/mengyamonitor-frontend/src/hooks/useServerMonitor.ts b/mengyamonitor-frontend/src/hooks/useServerMonitor.ts new file mode 100644 index 0000000..36dad51 --- /dev/null +++ b/mengyamonitor-frontend/src/hooks/useServerMonitor.ts @@ -0,0 +1,58 @@ +import { useState, useEffect, useCallback } from 'react'; +import type { ServerConfig, ServerStatus } from '../types'; +import { fetchServerMetrics } from '../api/monitor'; + +export const useServerMonitor = (servers: ServerConfig[], interval: number = 2000) => { + const [statuses, setStatuses] = useState>({}); + + const fetchMetrics = useCallback(async (server: ServerConfig) => { + if (!server.enabled) { + return; + } + + try { + const metrics = await fetchServerMetrics(server.url); + setStatuses(prev => ({ + ...prev, + [server.id]: { + serverId: server.id, + online: true, + metrics, + lastUpdate: Date.now(), + }, + })); + } catch (error) { + setStatuses(prev => ({ + ...prev, + [server.id]: { + serverId: server.id, + online: false, + error: error instanceof Error ? error.message : 'Unknown error', + lastUpdate: Date.now(), + }, + })); + } + }, []); + + useEffect(() => { + // Initial fetch + servers.forEach(server => { + if (server.enabled) { + fetchMetrics(server); + } + }); + + // Set up polling + const timer = setInterval(() => { + servers.forEach(server => { + if (server.enabled) { + fetchMetrics(server); + } + }); + }, interval); + + return () => clearInterval(timer); + }, [servers, interval, fetchMetrics]); + + return statuses; +}; diff --git a/mengyamonitor-frontend/src/index.css b/mengyamonitor-frontend/src/index.css new file mode 100644 index 0000000..7e46452 --- /dev/null +++ b/mengyamonitor-frontend/src/index.css @@ -0,0 +1,74 @@ +* { + box-sizing: border-box; +} + +/* 隐藏所有滚动条 */ +* { + scrollbar-width: none; /* Firefox */ + -ms-overflow-style: none; /* IE and Edge */ +} + +*::-webkit-scrollbar { + display: none; /* Chrome, Safari, Opera */ +} + +:root { + /* Theme Colors - 淡绿色/淡黄绿色渐变 */ + --primary-color: #86efac; + --primary-hover: #6ee7b7; + --secondary-color: #a7f3d0; + --warning-color: #fde047; + --danger-color: #f87171; + --success-color: #86efac; + + /* Backgrounds - 纯白色 */ + --bg-color: #ffffff; + --surface-color: #ffffff; + --surface-hover: #ffffff; + + /* Text */ + --text-main: #0f172a; + --text-secondary: #64748b; + --text-light: #94a3b8; + + /* Borders & Shadows - 灰白色边框 */ + --border-color: #e2e8f0; + --shadow-sm: 0 1px 2px 0 rgb(0 0 0 / 0.05); + --shadow-md: 0 4px 6px -1px rgb(0 0 0 / 0.1), 0 2px 4px -2px rgb(0 0 0 / 0.1); + --shadow-lg: 0 10px 15px -3px rgb(0 0 0 / 0.1), 0 4px 6px -4px rgb(0 0 0 / 0.1); + + /* Radius */ + --radius-sm: 0.375rem; + --radius-md: 0.5rem; + --radius-lg: 1rem; + + font-family: 'Inter', system-ui, -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, Oxygen, Ubuntu, Cantarell, 'Open Sans', 'Helvetica Neue', sans-serif; + line-height: 1.5; + font-weight: 400; + + color-scheme: light; + color: var(--text-main); + background-color: var(--bg-color); + + font-synthesis: none; + text-rendering: optimizeLegibility; + -webkit-font-smoothing: antialiased; + -moz-osx-font-smoothing: grayscale; +} + +body { + margin: 0; + min-width: 320px; + min-height: 100vh; + background-color: var(--bg-color); +} + +h1, h2, h3, h4, h5, h6 { + color: var(--text-main); + font-weight: 600; + line-height: 1.2; +} + +button { + font-family: inherit; +} diff --git a/mengyamonitor-frontend/src/main.tsx b/mengyamonitor-frontend/src/main.tsx new file mode 100644 index 0000000..bef5202 --- /dev/null +++ b/mengyamonitor-frontend/src/main.tsx @@ -0,0 +1,10 @@ +import { StrictMode } from 'react' +import { createRoot } from 'react-dom/client' +import './index.css' +import App from './App.tsx' + +createRoot(document.getElementById('root')!).render( + + + , +) diff --git a/mengyamonitor-frontend/src/types/index.ts b/mengyamonitor-frontend/src/types/index.ts new file mode 100644 index 0000000..92b98b7 --- /dev/null +++ b/mengyamonitor-frontend/src/types/index.ts @@ -0,0 +1,125 @@ +export interface ServerConfig { + id: string; + name: string; + url: string; + enabled: boolean; +} + +export interface CPUMetrics { + model: string; + cores: number; + usagePercent: number; + loadAverages?: number[]; + temperature?: number; + perCoreUsage?: CoreUsage[]; +} + +export interface CoreUsage { + core: number; + percent: number; +} + +export interface MemoryMetrics { + totalBytes: number; + usedBytes: number; + freeBytes: number; + usedPercent: number; +} + +export interface StorageMetrics { + mount: string; + totalBytes: number; + usedBytes: number; + freeBytes: number; + usedPercent: number; +} + +export interface GPUMetrics { + name: string; + memoryTotalMB: number; + memoryUsedMB: number; + utilizationPercent: number; + temperature?: number; + status: string; +} + +export interface NetworkInterface { + name: string; + ipAddress: string; + macAddress: string; + rxBytes: number; + txBytes: number; + rxSpeed: number; + txSpeed: number; +} + +export interface SystemStats { + processCount: number; + packageCount: number; + packageManager: string; + temperature?: number; + diskReadSpeed: number; + diskWriteSpeed: number; + networkRxSpeed?: number; + networkTxSpeed?: number; + topProcesses: ProcessInfo[]; + dockerStats: DockerStats; + systemLogs?: string[]; +} + +export interface ProcessInfo { + pid: number; + name: string; + cpu: number; + memory: number; + memoryMB: number; + command: string; +} + +export interface DockerStats { + available: boolean; + version?: string; + running: number; + stopped: number; + imageCount: number; + runningNames?: string[]; + stoppedNames?: string[]; + imageNames?: string[]; +} + +export interface OSInfo { + kernel: string; + distro: string; + architecture: string; +} + +export interface LatencyInfo { + clientToServer: number; + external: { + 'baidu.com'?: string; + 'google.com'?: string; + 'github.com'?: string; + }; +} + +export interface ServerMetrics { + hostname: string; + timestamp: string; + cpu: CPUMetrics; + memory: MemoryMetrics; + storage: StorageMetrics[]; + gpu: GPUMetrics[]; + network: NetworkInterface[]; + system: SystemStats; + os: OSInfo; + uptimeSeconds: number; + latency?: LatencyInfo; +} + +export interface ServerStatus { + serverId: string; + online: boolean; + metrics?: ServerMetrics; + error?: string; + lastUpdate: number; +} diff --git a/mengyamonitor-frontend/src/utils/format.ts b/mengyamonitor-frontend/src/utils/format.ts new file mode 100644 index 0000000..bc3dfa4 --- /dev/null +++ b/mengyamonitor-frontend/src/utils/format.ts @@ -0,0 +1,45 @@ +export const formatBytes = (bytes: number): string => { + if (bytes === 0 || bytes === null || bytes === undefined || isNaN(bytes) || !isFinite(bytes)) { + return '0 B'; + } + + // 处理负数 + if (bytes < 0) { + bytes = 0; + } + + const k = 1024; + const sizes = ['B', 'KB', 'MB', 'GB', 'TB']; + + // 处理小于 1 的情况 + if (bytes < 1) { + return bytes.toFixed(2) + ' ' + sizes[0]; + } + + const i = Math.floor(Math.log(bytes) / Math.log(k)); + // 确保索引在有效范围内 + const sizeIndex = Math.max(0, Math.min(i, sizes.length - 1)); + const value = Math.round((bytes / Math.pow(k, sizeIndex)) * 100) / 100; + + // 确保 sizes[sizeIndex] 存在 + const unit = sizes[sizeIndex] || sizes[0]; + + return value + ' ' + unit; +}; + +export const formatUptime = (seconds: number): string => { + const days = Math.floor(seconds / 86400); + const hours = Math.floor((seconds % 86400) / 3600); + const minutes = Math.floor((seconds % 3600) / 60); + + const parts: string[] = []; + if (days > 0) parts.push(`${days}天`); + if (hours > 0) parts.push(`${hours}小时`); + if (minutes > 0) parts.push(`${minutes}分钟`); + + return parts.length > 0 ? parts.join(' ') : '刚刚启动'; +}; + +export const formatPercent = (value: number): string => { + return `${Math.round(value * 10) / 10}%`; +}; diff --git a/mengyamonitor-frontend/src/utils/storage.ts b/mengyamonitor-frontend/src/utils/storage.ts new file mode 100644 index 0000000..01014eb --- /dev/null +++ b/mengyamonitor-frontend/src/utils/storage.ts @@ -0,0 +1,53 @@ +import type { ServerConfig } from '../types'; + +const STORAGE_KEY = 'mengya_monitor_servers'; + +export const loadServers = (): ServerConfig[] => { + try { + const stored = localStorage.getItem(STORAGE_KEY); + if (stored) { + const parsed = JSON.parse(stored); + // Ensure it's an array + if (Array.isArray(parsed)) { + return parsed; + } + } + } catch (error) { + console.error('Failed to load servers:', error); + } + return []; +}; + +export const saveServers = (servers: ServerConfig[]): void => { + try { + localStorage.setItem(STORAGE_KEY, JSON.stringify(servers)); + } catch (error) { + console.error('Failed to save servers:', error); + } +}; + +export const addServer = (server: Omit): ServerConfig => { + const servers = loadServers(); + const newServer: ServerConfig = { + ...server, + id: Date.now().toString(), + }; + servers.push(newServer); + saveServers(servers); + return newServer; +}; + +export const updateServer = (id: string, updates: Partial): void => { + const servers = loadServers(); + const index = servers.findIndex(s => s.id === id); + if (index !== -1) { + servers[index] = { ...servers[index], ...updates }; + saveServers(servers); + } +}; + +export const removeServer = (id: string): void => { + const servers = loadServers(); + const filtered = servers.filter(s => s.id !== id); + saveServers(filtered); +}; diff --git a/mengyamonitor-frontend/tsconfig.app.json b/mengyamonitor-frontend/tsconfig.app.json new file mode 100644 index 0000000..a9b5a59 --- /dev/null +++ b/mengyamonitor-frontend/tsconfig.app.json @@ -0,0 +1,28 @@ +{ + "compilerOptions": { + "tsBuildInfoFile": "./node_modules/.tmp/tsconfig.app.tsbuildinfo", + "target": "ES2022", + "useDefineForClassFields": true, + "lib": ["ES2022", "DOM", "DOM.Iterable"], + "module": "ESNext", + "types": ["vite/client"], + "skipLibCheck": true, + + /* Bundler mode */ + "moduleResolution": "bundler", + "allowImportingTsExtensions": true, + "verbatimModuleSyntax": true, + "moduleDetection": "force", + "noEmit": true, + "jsx": "react-jsx", + + /* Linting */ + "strict": true, + "noUnusedLocals": true, + "noUnusedParameters": true, + "erasableSyntaxOnly": true, + "noFallthroughCasesInSwitch": true, + "noUncheckedSideEffectImports": true + }, + "include": ["src"] +} diff --git a/mengyamonitor-frontend/tsconfig.json b/mengyamonitor-frontend/tsconfig.json new file mode 100644 index 0000000..1ffef60 --- /dev/null +++ b/mengyamonitor-frontend/tsconfig.json @@ -0,0 +1,7 @@ +{ + "files": [], + "references": [ + { "path": "./tsconfig.app.json" }, + { "path": "./tsconfig.node.json" } + ] +} diff --git a/mengyamonitor-frontend/tsconfig.node.json b/mengyamonitor-frontend/tsconfig.node.json new file mode 100644 index 0000000..8a67f62 --- /dev/null +++ b/mengyamonitor-frontend/tsconfig.node.json @@ -0,0 +1,26 @@ +{ + "compilerOptions": { + "tsBuildInfoFile": "./node_modules/.tmp/tsconfig.node.tsbuildinfo", + "target": "ES2023", + "lib": ["ES2023"], + "module": "ESNext", + "types": ["node"], + "skipLibCheck": true, + + /* Bundler mode */ + "moduleResolution": "bundler", + "allowImportingTsExtensions": true, + "verbatimModuleSyntax": true, + "moduleDetection": "force", + "noEmit": true, + + /* Linting */ + "strict": true, + "noUnusedLocals": true, + "noUnusedParameters": true, + "erasableSyntaxOnly": true, + "noFallthroughCasesInSwitch": true, + "noUncheckedSideEffectImports": true + }, + "include": ["vite.config.ts"] +} diff --git a/mengyamonitor-frontend/vite.config.ts b/mengyamonitor-frontend/vite.config.ts new file mode 100644 index 0000000..a4807b9 --- /dev/null +++ b/mengyamonitor-frontend/vite.config.ts @@ -0,0 +1,11 @@ +import { defineConfig } from 'vite' +import react from '@vitejs/plugin-react' + +// https://vite.dev/config/ +export default defineConfig({ + plugins: [react()], + server: { + port: 2929, + host: true, + }, +}) diff --git a/需求.txt b/需求.txt new file mode 100644 index 0000000..18f5df8 --- /dev/null +++ b/需求.txt @@ -0,0 +1,8 @@ +1.实现一个服务器监控面板,目前初步支持Linux服务器,监控服务器基本信息,比如:cpu/GPU,内存,储存,操作系统等 +2.采用企业级前后端分离架构,前端使用React框架,后端使用golang原版自带的net库 +3.原理十分简单,后端go获取Linux相关硬件信息,并封装成相关后端json API路由,前端每隔1秒或者2秒获取后端信息 +4.对外访问使用默认端口:2929 后端默认端口为9292 +5.由于是监控面板 前端可以一次性对接多个服务器后端卡片式展示相关信息,前端面板可以展示一些基本信息,用户可以点击详情按钮查看更多详细信息 +6.前端项目已经初始化 前后端代码架构必须分类整齐符合标准方便我阅读修改 +7.注意前端相当于一个只是客户端用于请求可以配置后端服务器,这两个是分开的,前端可以打包部署在用户电脑上比如做成软件app,而后端构建打包后放到各个服务器上 +8.前端页面风格偏白色柔和风 \ No newline at end of file