290 lines
8.8 KiB
Markdown
290 lines
8.8 KiB
Markdown
# P0 功能实施计划
|
||
|
||
> 计划日期:2026-05-06
|
||
> 基于:产品化评估与规划.md 中的 P0 优先级
|
||
|
||
---
|
||
|
||
## 总览
|
||
|
||
| 功能 | 依赖链 | 工作量估计 |
|
||
|------|--------|-----------|
|
||
| 告警中心 | media-server → agent → 后端 | 最大 |
|
||
| 人脸库管理 | agent → 后端 | 中 |
|
||
| 视频监控 | agent → 后端 | 中 |
|
||
| Dashboard 改造 | 后端(依赖以上三项) | 小 |
|
||
|
||
---
|
||
|
||
## 一、告警中心(最大的缺口)
|
||
|
||
### 1.1 当前问题
|
||
|
||
media-server 的 alarm 节点触发告警后,直接发送给第三方(HTTP 回调、MinIO 等),**不在本地保存告警记录**。agent 无法查询历史告警,后端自然也无法展示。
|
||
|
||
### 1.2 架构设计
|
||
|
||
```
|
||
media-server alarm 节点触发
|
||
├── 截图/视频片段上传 MinIO(已有)
|
||
├── HTTP 回调通知第三方(已有)
|
||
└── 写入本地告警日志(新增)──→ agent 查询 ──→ 后端收集展示
|
||
```
|
||
|
||
### 1.3 告警记录格式
|
||
|
||
```json
|
||
{
|
||
"id": "alarm_20260506_143021_a1b2c3",
|
||
"timestamp": "2026-05-06T14:30:21+08:00",
|
||
"device_id": "d12a4719c91641df",
|
||
"channel": "cam1",
|
||
"rule_name": "intrusion_zone_a",
|
||
"rule_type": "intrusion",
|
||
"object_label": "person",
|
||
"confidence": 0.87,
|
||
"snapshot_url": "http://minio:9000/alarms/xxx.jpg",
|
||
"clip_url": "http://minio:9000/clips/xxx.mp4",
|
||
"duration_ms": 3500
|
||
}
|
||
```
|
||
|
||
### 1.4 任务分解
|
||
|
||
#### Task A1:media-server 增加本地告警存储(C++)
|
||
|
||
**文件**:`plugins/alarm/alarm_node.cpp`
|
||
|
||
在 alarm 节点触发告警动作时,追加写入本地 SQLite 数据库:
|
||
|
||
```
|
||
表结构:
|
||
CREATE TABLE alarm_records (
|
||
id TEXT PRIMARY KEY,
|
||
timestamp TEXT NOT NULL,
|
||
channel TEXT NOT NULL,
|
||
rule_name TEXT NOT NULL,
|
||
rule_type TEXT NOT NULL,
|
||
object_label TEXT,
|
||
confidence REAL,
|
||
snapshot_url TEXT,
|
||
clip_url TEXT,
|
||
duration_ms INTEGER,
|
||
created_at TEXT NOT NULL
|
||
);
|
||
```
|
||
|
||
位置:`${modelsDir}/../resources/alarms/alarm_records.db`
|
||
|
||
#### Task A2:agent 增加告警查询端点(Go)
|
||
|
||
**文件**:`agent/internal/httpapi/alarms.go`(新建)
|
||
|
||
```
|
||
GET /v1/alarms/recent?limit=50&since=2026-05-06T00:00:00
|
||
```
|
||
|
||
返回最近 N 条告警记录。可选时间范围过滤。
|
||
|
||
#### Task A3:后端定时收集设备告警(Go)
|
||
|
||
**文件**:`internal/service/alarm_collector.go`(新建)
|
||
|
||
- RegistryService 增加 goroutine,每 10 秒遍历在线设备
|
||
- 对每个设备调用 `agent.Do("GET", dev.IP, dev.AgentPort, "/v1/alarms/recent?limit=100", nil)`
|
||
- 新告警写入后端 SQLite(`alarm_records` 表)
|
||
- 前端 SSE 推送新告警
|
||
|
||
**数据库**:`internal/storage/migrate.go` 新增表
|
||
|
||
```sql
|
||
CREATE TABLE IF NOT EXISTS alarm_records (
|
||
id TEXT PRIMARY KEY,
|
||
device_id TEXT NOT NULL,
|
||
channel TEXT NOT NULL,
|
||
timestamp TEXT NOT NULL,
|
||
rule_name TEXT NOT NULL,
|
||
rule_type TEXT NOT NULL,
|
||
object_label TEXT,
|
||
confidence REAL,
|
||
snapshot_url TEXT,
|
||
clip_url TEXT,
|
||
duration_ms INTEGER,
|
||
collected_at TEXT NOT NULL
|
||
);
|
||
```
|
||
|
||
#### Task A4:告警中心页面(Go + HTML)
|
||
|
||
**路由**:`GET /ui/alarms`
|
||
|
||
**布局**:
|
||
- 上半部分:告警筛选(设备/通道/规则类型/时间范围)
|
||
- 下半部分:告警列表(时间、设备、通道、规则、对象标签、截图缩略图、视频链接)
|
||
- 点击进入告警详情(大图、视频片段播放)
|
||
|
||
#### Task A5:Dashboard 告警信息流(HTML + JS)
|
||
|
||
在 Dashboard 顶部增加一个告警信息流卡片,SSE 实时推送最近 5 条告警。
|
||
|
||
---
|
||
|
||
## 二、人脸库管理
|
||
|
||
### 2.1 当前问题
|
||
|
||
只能整文件上传/替换 `.db`,无法增删改单个人物。
|
||
|
||
### 2.2 架构设计
|
||
|
||
```
|
||
后端人脸库页面 ──CRUD──▶ agent /v1/face-gallery/persons/* ──▶ media-server face_gallery.db
|
||
```
|
||
|
||
### 2.3 任务分解
|
||
|
||
#### Task B1:agent 增加人脸库 CRUD API(Go)
|
||
|
||
**文件**:`agent/internal/httpapi/face_gallery.go`(修改)
|
||
|
||
新增端点:
|
||
- `GET /v1/face-gallery/persons` — 返回人物列表 {id, name, photo_count, created_at}
|
||
- `POST /v1/face-gallery/persons` — 新增人物 {name, photo_bytes},agent 调用 media-server 接口提取 embedding 并写入 DB
|
||
- `DELETE /v1/face-gallery/persons/{id}` — 删除人物及其 embeddings
|
||
- `PUT /v1/face-gallery/persons/{id}/name` — 修改人物姓名
|
||
- `POST /v1/face-gallery/persons/{id}/photo` — 追加人脸照片,提取 embedding
|
||
|
||
#### Task B2:后端人脸库管理页面(Go + HTML)
|
||
|
||
**路由**:`GET /ui/face-gallery`
|
||
|
||
**布局**:
|
||
- 人员列表(姓名、照片缩略图、注册时间、操作按钮)
|
||
- 搜索框(按姓名搜索)
|
||
- 新增按钮 → 弹出表单(姓名 + 上传照片)
|
||
- 编辑/删除确认
|
||
|
||
---
|
||
|
||
## 三、视频监控
|
||
|
||
### 3.1 当前问题
|
||
|
||
后台看不到摄像头画面。只能通过 `http://设备IP:9000/hls_player.html` 在设备上查看。
|
||
|
||
### 3.2 架构设计
|
||
|
||
```
|
||
后端视频监控页 ──iframe──▶ agent 返回 HLS 播放页面 URL
|
||
└──img──▶ agent /v1/preview/snapshot?channel=X 返回 JPEG
|
||
```
|
||
|
||
### 3.3 任务分解
|
||
|
||
#### Task C1:agent 增加预览端点(Go)
|
||
|
||
**文件**:`agent/internal/httpapi/preview.go`(新建)
|
||
|
||
- `GET /v1/preview/channels` — 返回各通道信息 {name, hls_url, rtsp_url, resolution, fps}
|
||
- `GET /v1/preview/snapshot/:channel` — 从 media-server 请求当前帧,返回 JPEG
|
||
|
||
HLS URL 格式:代理转发到 media-server 的 `/hls/{channel}/index.m3u8`
|
||
|
||
#### Task C2:后端视频监控页面(Go + HTML)
|
||
|
||
**路由**:`GET /ui/monitor`
|
||
|
||
**布局**:
|
||
- 左侧:设备通道列表(可多选)
|
||
- 右侧:多路画面网格(4/9/16 宫格)
|
||
- 每个画面:HLS 播放器 iframe + 通道名称标签
|
||
- 单击画面可全屏
|
||
|
||
---
|
||
|
||
## 四、Dashboard 改造
|
||
|
||
### 4.1 当前问题
|
||
|
||
只有静态统计(设备数、任务数),缺乏实时信息。
|
||
|
||
### 4.2 目标布局
|
||
|
||
```
|
||
┌─────────────────────────────────────┐
|
||
│ 实时告警(5条滚动) │ ← 告警中心对接
|
||
├──────────┬──────────┬───────────────┤
|
||
│ 设备状态 │ NPU/CPU │ 今日告警统计 │ ← agent metrics
|
||
│ 在线 3/3 │ ████░░░░ │ 告警: 12 │
|
||
├──────────┴──────────┴───────────────┤
|
||
│ 关键通道预览(1-4路 snapshot) │ ← 视频监控对接
|
||
├─────────────────────────────────────┤
|
||
│ 最近任务 │ ← 已有
|
||
└─────────────────────────────────────┘
|
||
```
|
||
|
||
### 4.3 任务分解
|
||
|
||
#### Task D1:Dashboard 实时数据 API(Go)
|
||
|
||
**文件**:`internal/web/ui.go`(修改 `pageDashboard`)
|
||
|
||
```go
|
||
func (u *UI) pageDashboard(w http.ResponseWriter, r *http.Request) {
|
||
u.ensureDevicesLoaded()
|
||
data := ...现有逻辑...
|
||
|
||
// 新增:最近告警
|
||
data.RecentAlarms = alarmService.GetRecent(5)
|
||
|
||
// 新增:设备 metrics(聚合)
|
||
data.DeviceMetrics = collectDeviceMetrics(u.agent, u.registry)
|
||
|
||
u.render(w, r, "dashboard", data)
|
||
}
|
||
```
|
||
|
||
#### Task D2:Dashboard 模板改造(HTML)
|
||
|
||
**文件**:`internal/web/ui/templates/dashboard.html`
|
||
|
||
- 告警信息流(SSE 实时更新)
|
||
- 设备状态卡片(在线数、CPU、内存)
|
||
- 快照预览(定时刷新 img src)
|
||
- 最近任务(已有,保留)
|
||
|
||
---
|
||
|
||
## 五、实施顺序
|
||
|
||
```
|
||
第一阶段(打通告警链路):
|
||
Task A1(media-server)→ Task A2(agent)→ Task A3(后端收集)→ Task A4(告警页面)
|
||
|
||
第二阶段(人脸库 + 视频):
|
||
Task B1 + C1(agent)→ Task B2 + C2(后端页面)
|
||
|
||
第三阶段(Dashboard整合):
|
||
Task A5 + D1 + D2(Dashboard 改造)
|
||
```
|
||
|
||
---
|
||
|
||
## 六、涉及文件清单
|
||
|
||
| 项目 | 文件 | 操作 |
|
||
|------|------|------|
|
||
| media-server | `plugins/alarm/alarm_node.cpp` | 修改:追加本地告警存储 |
|
||
| agent | `internal/httpapi/alarms.go` | 新建:告警查询端点 |
|
||
| agent | `internal/httpapi/face_gallery.go` | 修改:人脸库 CRUD |
|
||
| agent | `internal/httpapi/preview.go` | 新建:视频预览端点 |
|
||
| agent | `internal/httpapi/server.go` | 修改:注册新路由 |
|
||
| 后端 | `internal/service/alarm_collector.go` | 新建:告警收集服务 |
|
||
| 后端 | `internal/storage/migrate.go` | 修改:alarm_records 表 |
|
||
| 后端 | `internal/web/ui/templates/alarms.html` | 新建:告警中心页面 |
|
||
| 后端 | `internal/web/ui/templates/face_gallery.html` | 新建:人脸库管理页面 |
|
||
| 后端 | `internal/web/ui/templates/monitor.html` | 新建:视频监控页面 |
|
||
| 后端 | `internal/web/ui/templates/dashboard.html` | 修改:实时化改造 |
|
||
| 后端 | `internal/web/ui.go` | 修改:新页面 handler + 路由 |
|
||
| 后端 | `cmd/managerd/main.go` | 修改:注入新服务 |
|