mirror of
https://github.com/Ed1s0nZ/CyberStrikeAI.git
synced 2026-05-17 21:44:43 +02:00
Compare commits
14 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 5404d95db7 | |||
| 32d0e98cfb | |||
| e4b1e10a42 | |||
| 870715fc8f | |||
| 772a04b715 | |||
| 2455bde7ab | |||
| dbdfc18d57 | |||
| 82daad3b56 | |||
| 9eee820096 | |||
| fae912b79c | |||
| 9b48daf795 | |||
| bfbb8b31d3 | |||
| 8b2dfea884 | |||
| 7447e82c39 |
+1
-1
@@ -10,7 +10,7 @@
|
|||||||
# ============================================
|
# ============================================
|
||||||
|
|
||||||
# 前端显示的版本号(可选,不填则显示默认版本)
|
# 前端显示的版本号(可选,不填则显示默认版本)
|
||||||
version: "v1.3.4"
|
version: "v1.3.6"
|
||||||
|
|
||||||
# 服务器配置
|
# 服务器配置
|
||||||
server:
|
server:
|
||||||
|
|||||||
@@ -694,6 +694,18 @@ func setupRoutes(
|
|||||||
}
|
}
|
||||||
app.knowledgeHandler.Search(c)
|
app.knowledgeHandler.Search(c)
|
||||||
})
|
})
|
||||||
|
knowledgeRoutes.GET("/stats", func(c *gin.Context) {
|
||||||
|
if app.knowledgeHandler == nil {
|
||||||
|
c.JSON(http.StatusOK, gin.H{
|
||||||
|
"enabled": false,
|
||||||
|
"total_categories": 0,
|
||||||
|
"total_items": 0,
|
||||||
|
"message": "知识库功能未启用,请前往系统设置启用知识检索功能",
|
||||||
|
})
|
||||||
|
return
|
||||||
|
}
|
||||||
|
app.knowledgeHandler.GetStats(c)
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
// 漏洞管理
|
// 漏洞管理
|
||||||
|
|||||||
@@ -15,11 +15,11 @@ import (
|
|||||||
|
|
||||||
// KnowledgeHandler 知识库处理器
|
// KnowledgeHandler 知识库处理器
|
||||||
type KnowledgeHandler struct {
|
type KnowledgeHandler struct {
|
||||||
manager *knowledge.Manager
|
manager *knowledge.Manager
|
||||||
retriever *knowledge.Retriever
|
retriever *knowledge.Retriever
|
||||||
indexer *knowledge.Indexer
|
indexer *knowledge.Indexer
|
||||||
db *database.DB
|
db *database.DB
|
||||||
logger *zap.Logger
|
logger *zap.Logger
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewKnowledgeHandler 创建新的知识库处理器
|
// NewKnowledgeHandler 创建新的知识库处理器
|
||||||
@@ -55,7 +55,7 @@ func (h *KnowledgeHandler) GetCategories(c *gin.Context) {
|
|||||||
func (h *KnowledgeHandler) GetItems(c *gin.Context) {
|
func (h *KnowledgeHandler) GetItems(c *gin.Context) {
|
||||||
category := c.Query("category")
|
category := c.Query("category")
|
||||||
searchKeyword := c.Query("search") // 搜索关键字
|
searchKeyword := c.Query("search") // 搜索关键字
|
||||||
|
|
||||||
// 如果提供了搜索关键字,执行关键字搜索(在所有数据中搜索)
|
// 如果提供了搜索关键字,执行关键字搜索(在所有数据中搜索)
|
||||||
if searchKeyword != "" {
|
if searchKeyword != "" {
|
||||||
items, err := h.manager.SearchItemsByKeyword(searchKeyword, category)
|
items, err := h.manager.SearchItemsByKeyword(searchKeyword, category)
|
||||||
@@ -102,10 +102,10 @@ func (h *KnowledgeHandler) GetItems(c *gin.Context) {
|
|||||||
})
|
})
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
// 分页模式:categoryPage=true 表示按分类分页,否则按项分页(向后兼容)
|
// 分页模式:categoryPage=true 表示按分类分页,否则按项分页(向后兼容)
|
||||||
categoryPageMode := c.Query("categoryPage") != "false" // 默认使用分类分页
|
categoryPageMode := c.Query("categoryPage") != "false" // 默认使用分类分页
|
||||||
|
|
||||||
// 分页参数
|
// 分页参数
|
||||||
limit := 50 // 默认每页50条(分类分页时为分类数,项分页时为项数)
|
limit := 50 // 默认每页50条(分类分页时为分类数,项分页时为项数)
|
||||||
offset := 0
|
offset := 0
|
||||||
@@ -192,9 +192,9 @@ func (h *KnowledgeHandler) GetItems(c *gin.Context) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
c.JSON(http.StatusOK, gin.H{
|
c.JSON(http.StatusOK, gin.H{
|
||||||
"items": items,
|
"items": items,
|
||||||
"total": total,
|
"total": total,
|
||||||
"limit": limit,
|
"limit": limit,
|
||||||
"offset": offset,
|
"offset": offset,
|
||||||
})
|
})
|
||||||
} else {
|
} else {
|
||||||
@@ -207,9 +207,9 @@ func (h *KnowledgeHandler) GetItems(c *gin.Context) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
c.JSON(http.StatusOK, gin.H{
|
c.JSON(http.StatusOK, gin.H{
|
||||||
"items": items,
|
"items": items,
|
||||||
"total": total,
|
"total": total,
|
||||||
"limit": limit,
|
"limit": limit,
|
||||||
"offset": offset,
|
"offset": offset,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
@@ -341,12 +341,12 @@ func (h *KnowledgeHandler) ScanKnowledgeBase(c *gin.Context) {
|
|||||||
consecutiveFailures := 0
|
consecutiveFailures := 0
|
||||||
var firstFailureItemID string
|
var firstFailureItemID string
|
||||||
var firstFailureError error
|
var firstFailureError error
|
||||||
|
|
||||||
for i, itemID := range itemsToIndex {
|
for i, itemID := range itemsToIndex {
|
||||||
if err := h.indexer.IndexItem(ctx, itemID); err != nil {
|
if err := h.indexer.IndexItem(ctx, itemID); err != nil {
|
||||||
failedCount++
|
failedCount++
|
||||||
consecutiveFailures++
|
consecutiveFailures++
|
||||||
|
|
||||||
// 只在第一个失败时记录详细日志
|
// 只在第一个失败时记录详细日志
|
||||||
if consecutiveFailures == 1 {
|
if consecutiveFailures == 1 {
|
||||||
firstFailureItemID = itemID
|
firstFailureItemID = itemID
|
||||||
@@ -357,7 +357,7 @@ func (h *KnowledgeHandler) ScanKnowledgeBase(c *gin.Context) {
|
|||||||
zap.Error(err),
|
zap.Error(err),
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
// 如果连续失败2次,立即停止增量索引
|
// 如果连续失败2次,立即停止增量索引
|
||||||
if consecutiveFailures >= 2 {
|
if consecutiveFailures >= 2 {
|
||||||
h.logger.Error("连续索引失败次数过多,立即停止增量索引",
|
h.logger.Error("连续索引失败次数过多,立即停止增量索引",
|
||||||
@@ -371,14 +371,14 @@ func (h *KnowledgeHandler) ScanKnowledgeBase(c *gin.Context) {
|
|||||||
}
|
}
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
// 成功时重置连续失败计数
|
// 成功时重置连续失败计数
|
||||||
if consecutiveFailures > 0 {
|
if consecutiveFailures > 0 {
|
||||||
consecutiveFailures = 0
|
consecutiveFailures = 0
|
||||||
firstFailureItemID = ""
|
firstFailureItemID = ""
|
||||||
firstFailureError = nil
|
firstFailureError = nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// 减少进度日志频率
|
// 减少进度日志频率
|
||||||
if (i+1)%10 == 0 || i+1 == len(itemsToIndex) {
|
if (i+1)%10 == 0 || i+1 == len(itemsToIndex) {
|
||||||
h.logger.Info("索引进度", zap.Int("current", i+1), zap.Int("total", len(itemsToIndex)), zap.Int("failed", failedCount))
|
h.logger.Info("索引进度", zap.Int("current", i+1), zap.Int("total", len(itemsToIndex)), zap.Int("failed", failedCount))
|
||||||
@@ -388,7 +388,7 @@ func (h *KnowledgeHandler) ScanKnowledgeBase(c *gin.Context) {
|
|||||||
}()
|
}()
|
||||||
|
|
||||||
c.JSON(http.StatusOK, gin.H{
|
c.JSON(http.StatusOK, gin.H{
|
||||||
"message": fmt.Sprintf("扫描完成,开始索引 %d 个新添加或更新的知识项", len(itemsToIndex)),
|
"message": fmt.Sprintf("扫描完成,开始索引 %d 个新添加或更新的知识项", len(itemsToIndex)),
|
||||||
"items_to_index": len(itemsToIndex),
|
"items_to_index": len(itemsToIndex),
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
@@ -470,10 +470,25 @@ func (h *KnowledgeHandler) Search(c *gin.Context) {
|
|||||||
c.JSON(http.StatusOK, gin.H{"results": results})
|
c.JSON(http.StatusOK, gin.H{"results": results})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// GetStats 获取知识库统计信息
|
||||||
|
func (h *KnowledgeHandler) GetStats(c *gin.Context) {
|
||||||
|
totalCategories, totalItems, err := h.manager.GetStats()
|
||||||
|
if err != nil {
|
||||||
|
h.logger.Error("获取知识库统计信息失败", zap.Error(err))
|
||||||
|
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
c.JSON(http.StatusOK, gin.H{
|
||||||
|
"enabled": true,
|
||||||
|
"total_categories": totalCategories,
|
||||||
|
"total_items": totalItems,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
// 辅助函数:解析整数
|
// 辅助函数:解析整数
|
||||||
func parseInt(s string) (int, error) {
|
func parseInt(s string) (int, error) {
|
||||||
var result int
|
var result int
|
||||||
_, err := fmt.Sscanf(s, "%d", &result)
|
_, err := fmt.Sscanf(s, "%d", &result)
|
||||||
return result, err
|
return result, err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -153,6 +153,25 @@ func (m *Manager) GetCategories() ([]string, error) {
|
|||||||
return categories, nil
|
return categories, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// GetStats 获取知识库统计信息
|
||||||
|
func (m *Manager) GetStats() (int, int, error) {
|
||||||
|
// 获取分类总数
|
||||||
|
categories, err := m.GetCategories()
|
||||||
|
if err != nil {
|
||||||
|
return 0, 0, fmt.Errorf("获取分类失败: %w", err)
|
||||||
|
}
|
||||||
|
totalCategories := len(categories)
|
||||||
|
|
||||||
|
// 获取知识项总数
|
||||||
|
var totalItems int
|
||||||
|
err = m.db.QueryRow("SELECT COUNT(*) FROM knowledge_base_items").Scan(&totalItems)
|
||||||
|
if err != nil {
|
||||||
|
return totalCategories, 0, fmt.Errorf("获取知识项总数失败: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return totalCategories, totalItems, nil
|
||||||
|
}
|
||||||
|
|
||||||
// GetCategoriesWithItems 按分类分页获取知识项(每个分类包含其下的所有知识项)
|
// GetCategoriesWithItems 按分类分页获取知识项(每个分类包含其下的所有知识项)
|
||||||
// limit: 每页分类数量(0表示不限制)
|
// limit: 每页分类数量(0表示不限制)
|
||||||
// offset: 偏移量(按分类偏移)
|
// offset: 偏移量(按分类偏移)
|
||||||
@@ -359,7 +378,7 @@ func (m *Manager) SearchItemsByKeyword(keyword string, category string) ([]*Know
|
|||||||
// SQLite的LIKE不区分大小写,使用COLLATE NOCASE或LOWER()函数
|
// SQLite的LIKE不区分大小写,使用COLLATE NOCASE或LOWER()函数
|
||||||
// 使用%keyword%进行模糊匹配
|
// 使用%keyword%进行模糊匹配
|
||||||
searchPattern := "%" + keyword + "%"
|
searchPattern := "%" + keyword + "%"
|
||||||
|
|
||||||
query = `
|
query = `
|
||||||
SELECT id, category, title, file_path, created_at, updated_at
|
SELECT id, category, title, file_path, created_at, updated_at
|
||||||
FROM knowledge_base_items
|
FROM knowledge_base_items
|
||||||
|
|||||||
@@ -1,5 +1,8 @@
|
|||||||
# Python HTTP helpers leveraged by tools like api-fuzzer, dnslog, http-intruder, http-framework-test
|
# Python HTTP helpers leveraged by tools like api-fuzzer, dnslog, http-intruder, http-framework-test
|
||||||
requests>=2.32.3
|
requests>=2.32.3
|
||||||
|
httpx>=0.27.0
|
||||||
|
charset-normalizer>=3.3.2
|
||||||
|
chardet>=5.2.0
|
||||||
|
|
||||||
# dirsearch:用 python3 -m dirsearch 时由本依赖提供(含 defusedxml 等)
|
# dirsearch:用 python3 -m dirsearch 时由本依赖提供(含 defusedxml 等)
|
||||||
dirsearch>=0.4.3
|
dirsearch>=0.4.3
|
||||||
|
|||||||
@@ -1,52 +0,0 @@
|
|||||||
name: "list-files"
|
|
||||||
command: "ls"
|
|
||||||
enabled: true
|
|
||||||
short_description: "列出目录文件工具"
|
|
||||||
description: |
|
|
||||||
列出服务器上指定目录中的文件。
|
|
||||||
|
|
||||||
**主要功能:**
|
|
||||||
- 列出文件
|
|
||||||
- 显示详细信息
|
|
||||||
- 递归列出
|
|
||||||
|
|
||||||
**使用场景:**
|
|
||||||
- 目录浏览
|
|
||||||
- 文件查找
|
|
||||||
- 系统检查
|
|
||||||
parameters:
|
|
||||||
- name: "directory"
|
|
||||||
type: "string"
|
|
||||||
description: "要列出的目录(相对于服务器基础目录)"
|
|
||||||
required: false
|
|
||||||
default: "."
|
|
||||||
position: 0
|
|
||||||
format: "positional"
|
|
||||||
- name: "long_format"
|
|
||||||
type: "bool"
|
|
||||||
description: "显示详细信息(长格式)"
|
|
||||||
required: false
|
|
||||||
flag: "-l"
|
|
||||||
format: "flag"
|
|
||||||
default: false
|
|
||||||
- name: "recursive"
|
|
||||||
type: "bool"
|
|
||||||
description: "递归列出"
|
|
||||||
required: false
|
|
||||||
flag: "-R"
|
|
||||||
format: "flag"
|
|
||||||
default: false
|
|
||||||
- name: "additional_args"
|
|
||||||
type: "string"
|
|
||||||
description: |
|
|
||||||
额外的list-files参数。用于传递未在参数列表中定义的list-files选项。
|
|
||||||
|
|
||||||
**示例值:**
|
|
||||||
- 根据工具特性添加常用参数示例
|
|
||||||
|
|
||||||
**注意事项:**
|
|
||||||
- 多个参数用空格分隔
|
|
||||||
- 确保参数格式正确,避免命令注入
|
|
||||||
- 此参数会直接追加到命令末尾
|
|
||||||
required: false
|
|
||||||
format: "positional"
|
|
||||||
+702
-179
File diff suppressed because it is too large
Load Diff
+228
-10
@@ -14,6 +14,12 @@ async function refreshDashboard() {
|
|||||||
if (barEl) barEl.style.width = '0%';
|
if (barEl) barEl.style.width = '0%';
|
||||||
});
|
});
|
||||||
setDashboardOverviewPlaceholder('…');
|
setDashboardOverviewPlaceholder('…');
|
||||||
|
setEl('dashboard-kpi-tools-calls', '…');
|
||||||
|
setEl('dashboard-kpi-success-rate', '…');
|
||||||
|
var chartPlaceholder = document.getElementById('dashboard-tools-pie-placeholder');
|
||||||
|
if (chartPlaceholder) { chartPlaceholder.style.display = 'block'; chartPlaceholder.textContent = '加载中…'; }
|
||||||
|
var barChartEl = document.getElementById('dashboard-tools-bar-chart');
|
||||||
|
if (barChartEl) { barChartEl.style.display = 'none'; barChartEl.innerHTML = ''; }
|
||||||
|
|
||||||
if (typeof apiFetch === 'undefined') {
|
if (typeof apiFetch === 'undefined') {
|
||||||
if (runningEl) runningEl.textContent = '-';
|
if (runningEl) runningEl.textContent = '-';
|
||||||
@@ -23,11 +29,12 @@ async function refreshDashboard() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const [tasksRes, vulnRes, batchRes, monitorRes, skillsRes] = await Promise.all([
|
const [tasksRes, vulnRes, batchRes, monitorRes, knowledgeRes, skillsRes] = await Promise.all([
|
||||||
apiFetch('/api/agent-loop/tasks').then(r => r.ok ? r.json() : null).catch(() => null),
|
apiFetch('/api/agent-loop/tasks').then(r => r.ok ? r.json() : null).catch(() => null),
|
||||||
apiFetch('/api/vulnerabilities/stats').then(r => r.ok ? r.json() : null).catch(() => null),
|
apiFetch('/api/vulnerabilities/stats').then(r => r.ok ? r.json() : null).catch(() => null),
|
||||||
apiFetch('/api/batch-tasks?limit=500&page=1').then(r => r.ok ? r.json() : null).catch(() => null),
|
apiFetch('/api/batch-tasks?limit=500&page=1').then(r => r.ok ? r.json() : null).catch(() => null),
|
||||||
apiFetch('/api/monitor/stats').then(r => r.ok ? r.json() : null).catch(() => null),
|
apiFetch('/api/monitor/stats').then(r => r.ok ? r.json() : null).catch(() => null),
|
||||||
|
apiFetch('/api/knowledge/stats').then(r => r.ok ? r.json() : null).catch(() => null),
|
||||||
apiFetch('/api/skills/stats').then(r => r.ok ? r.json() : null).catch(() => null)
|
apiFetch('/api/skills/stats').then(r => r.ok ? r.json() : null).catch(() => null)
|
||||||
]);
|
]);
|
||||||
|
|
||||||
@@ -56,7 +63,7 @@ async function refreshDashboard() {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
// 批量任务队列:按状态统计
|
// 批量任务队列:按状态统计(优化版)
|
||||||
if (batchRes && Array.isArray(batchRes.queues)) {
|
if (batchRes && Array.isArray(batchRes.queues)) {
|
||||||
const queues = batchRes.queues;
|
const queues = batchRes.queues;
|
||||||
let pending = 0, running = 0, done = 0;
|
let pending = 0, running = 0, done = 0;
|
||||||
@@ -66,44 +73,122 @@ async function refreshDashboard() {
|
|||||||
else if (s === 'running') running++;
|
else if (s === 'running') running++;
|
||||||
else if (s === 'completed' || s === 'cancelled') done++;
|
else if (s === 'completed' || s === 'cancelled') done++;
|
||||||
});
|
});
|
||||||
|
const total = pending + running + done;
|
||||||
setEl('dashboard-batch-pending', String(pending));
|
setEl('dashboard-batch-pending', String(pending));
|
||||||
setEl('dashboard-batch-running', String(running));
|
setEl('dashboard-batch-running', String(running));
|
||||||
setEl('dashboard-batch-done', String(done));
|
setEl('dashboard-batch-done', String(done));
|
||||||
|
setEl('dashboard-batch-total', total > 0 ? `共 ${total} 个` : '暂无任务');
|
||||||
|
|
||||||
|
// 更新进度条
|
||||||
|
if (total > 0) {
|
||||||
|
const pendingPct = (pending / total * 100).toFixed(1);
|
||||||
|
const runningPct = (running / total * 100).toFixed(1);
|
||||||
|
const donePct = (done / total * 100).toFixed(1);
|
||||||
|
updateProgressBar('dashboard-batch-progress-pending', pendingPct);
|
||||||
|
updateProgressBar('dashboard-batch-progress-running', runningPct);
|
||||||
|
updateProgressBar('dashboard-batch-progress-done', donePct);
|
||||||
|
} else {
|
||||||
|
updateProgressBar('dashboard-batch-progress-pending', '0');
|
||||||
|
updateProgressBar('dashboard-batch-progress-running', '0');
|
||||||
|
updateProgressBar('dashboard-batch-progress-done', '0');
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
setEl('dashboard-batch-pending', '-');
|
setEl('dashboard-batch-pending', '-');
|
||||||
setEl('dashboard-batch-running', '-');
|
setEl('dashboard-batch-running', '-');
|
||||||
setEl('dashboard-batch-done', '-');
|
setEl('dashboard-batch-done', '-');
|
||||||
|
setEl('dashboard-batch-total', '-');
|
||||||
|
updateProgressBar('dashboard-batch-progress-pending', '0');
|
||||||
|
updateProgressBar('dashboard-batch-progress-running', '0');
|
||||||
|
updateProgressBar('dashboard-batch-progress-done', '0');
|
||||||
}
|
}
|
||||||
|
|
||||||
// 工具调用:monitor/stats 为 { toolName: { TotalCalls, ... } }
|
// 工具调用:monitor/stats 为 { toolName: { totalCalls, successCalls, failedCalls, ... } }(优化版)
|
||||||
if (monitorRes && typeof monitorRes === 'object') {
|
if (monitorRes && typeof monitorRes === 'object') {
|
||||||
const names = Object.keys(monitorRes);
|
const names = Object.keys(monitorRes);
|
||||||
let totalCalls = 0;
|
let totalCalls = 0, totalSuccess = 0, totalFailed = 0;
|
||||||
names.forEach(k => {
|
names.forEach(k => {
|
||||||
const v = monitorRes[k];
|
const v = monitorRes[k];
|
||||||
const n = v && (v.totalCalls ?? v.TotalCalls);
|
const n = v && (v.totalCalls ?? v.TotalCalls);
|
||||||
if (typeof n === 'number') totalCalls += n;
|
if (typeof n === 'number') totalCalls += n;
|
||||||
|
const s = v && (v.successCalls ?? v.SuccessCalls);
|
||||||
|
if (typeof s === 'number') totalSuccess += s;
|
||||||
|
const f = v && (v.failedCalls ?? v.FailedCalls);
|
||||||
|
if (typeof f === 'number') totalFailed += f;
|
||||||
});
|
});
|
||||||
setEl('dashboard-tools-count', String(names.length));
|
setEl('dashboard-tools-count', String(names.length));
|
||||||
setEl('dashboard-tools-calls', String(totalCalls));
|
setEl('dashboard-tools-calls', formatNumber(totalCalls));
|
||||||
|
setEl('dashboard-kpi-tools-calls', String(totalCalls));
|
||||||
|
var rateStr = totalCalls > 0 ? ((totalSuccess / totalCalls) * 100).toFixed(1) + '%' : '-';
|
||||||
|
setEl('dashboard-kpi-success-rate', rateStr);
|
||||||
|
setEl('dashboard-tools-success-rate', rateStr !== '-' ? `成功率 ${rateStr}` : '-');
|
||||||
|
renderDashboardToolsBar(monitorRes);
|
||||||
} else {
|
} else {
|
||||||
setEl('dashboard-tools-count', '-');
|
setEl('dashboard-tools-count', '-');
|
||||||
setEl('dashboard-tools-calls', '-');
|
setEl('dashboard-tools-calls', '-');
|
||||||
|
setEl('dashboard-kpi-tools-calls', '-');
|
||||||
|
setEl('dashboard-kpi-success-rate', '-');
|
||||||
|
setEl('dashboard-tools-success-rate', '-');
|
||||||
|
renderDashboardToolsBar(null);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Skills:{ total_skills, total_calls, ... }
|
// 知识:{ enabled, total_categories, total_items, ... }(优化版)
|
||||||
|
const knowledgeItemsEl = document.getElementById('dashboard-knowledge-items');
|
||||||
|
const knowledgeCategoriesEl = document.getElementById('dashboard-knowledge-categories');
|
||||||
|
if (knowledgeRes && typeof knowledgeRes === 'object') {
|
||||||
|
if (knowledgeRes.enabled === false) {
|
||||||
|
if (knowledgeItemsEl) knowledgeItemsEl.textContent = '未启用';
|
||||||
|
if (knowledgeCategoriesEl) knowledgeCategoriesEl.textContent = '-';
|
||||||
|
} else {
|
||||||
|
const categories = knowledgeRes.total_categories ?? 0;
|
||||||
|
const items = knowledgeRes.total_items ?? 0;
|
||||||
|
if (knowledgeItemsEl) knowledgeItemsEl.textContent = formatNumber(items);
|
||||||
|
if (knowledgeCategoriesEl) knowledgeCategoriesEl.textContent = formatNumber(categories);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (knowledgeItemsEl) knowledgeItemsEl.textContent = '-';
|
||||||
|
if (knowledgeCategoriesEl) knowledgeCategoriesEl.textContent = '-';
|
||||||
|
}
|
||||||
|
|
||||||
|
// Skills:{ total_skills, total_calls, ... }(优化版)
|
||||||
if (skillsRes && typeof skillsRes === 'object') {
|
if (skillsRes && typeof skillsRes === 'object') {
|
||||||
setEl('dashboard-skills-count', String(skillsRes.total_skills ?? '-'));
|
const totalSkills = skillsRes.total_skills ?? 0;
|
||||||
setEl('dashboard-skills-calls', String(skillsRes.total_calls ?? '-'));
|
const totalCalls = skillsRes.total_calls ?? 0;
|
||||||
|
setEl('dashboard-skills-count', formatNumber(totalSkills));
|
||||||
|
setEl('dashboard-skills-calls', formatNumber(totalCalls));
|
||||||
|
|
||||||
|
// 设置状态标签
|
||||||
|
const statusEl = document.getElementById('dashboard-skills-status');
|
||||||
|
if (statusEl) {
|
||||||
|
if (totalCalls === 0) {
|
||||||
|
statusEl.textContent = '待使用';
|
||||||
|
statusEl.style.background = 'rgba(0, 0, 0, 0.05)';
|
||||||
|
statusEl.style.color = 'var(--text-secondary)';
|
||||||
|
} else if (totalCalls < 10) {
|
||||||
|
statusEl.textContent = '活跃';
|
||||||
|
statusEl.style.background = 'rgba(16, 185, 129, 0.1)';
|
||||||
|
statusEl.style.color = '#10b981';
|
||||||
|
} else {
|
||||||
|
statusEl.textContent = '高频';
|
||||||
|
statusEl.style.background = 'rgba(59, 130, 246, 0.1)';
|
||||||
|
statusEl.style.color = '#3b82f6';
|
||||||
|
}
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
setEl('dashboard-skills-count', '-');
|
setEl('dashboard-skills-count', '-');
|
||||||
setEl('dashboard-skills-calls', '-');
|
setEl('dashboard-skills-calls', '-');
|
||||||
|
const statusEl = document.getElementById('dashboard-skills-status');
|
||||||
|
if (statusEl) statusEl.textContent = '-';
|
||||||
}
|
}
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
console.warn('仪表盘拉取统计失败', e);
|
console.warn('仪表盘拉取统计失败', e);
|
||||||
if (runningEl) runningEl.textContent = '-';
|
if (runningEl) runningEl.textContent = '-';
|
||||||
if (vulnTotalEl) vulnTotalEl.textContent = '-';
|
if (vulnTotalEl) vulnTotalEl.textContent = '-';
|
||||||
setDashboardOverviewPlaceholder('-');
|
setDashboardOverviewPlaceholder('-');
|
||||||
|
setEl('dashboard-kpi-success-rate', '-');
|
||||||
|
setEl('dashboard-kpi-tools-calls', '-');
|
||||||
|
renderDashboardToolsBar(null);
|
||||||
|
var ph = document.getElementById('dashboard-tools-pie-placeholder');
|
||||||
|
if (ph) { ph.style.display = 'block'; ph.textContent = '暂无调用数据'; }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -113,6 +198,139 @@ function setEl(id, text) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function setDashboardOverviewPlaceholder(t) {
|
function setDashboardOverviewPlaceholder(t) {
|
||||||
['dashboard-batch-pending', 'dashboard-batch-running', 'dashboard-batch-done',
|
['dashboard-batch-pending', 'dashboard-batch-running', 'dashboard-batch-done', 'dashboard-batch-total',
|
||||||
'dashboard-tools-count', 'dashboard-tools-calls', 'dashboard-skills-count', 'dashboard-skills-calls'].forEach(id => setEl(id, t));
|
'dashboard-tools-count', 'dashboard-tools-calls', 'dashboard-tools-success-rate',
|
||||||
|
'dashboard-skills-count', 'dashboard-skills-calls', 'dashboard-skills-status',
|
||||||
|
'dashboard-knowledge-items', 'dashboard-knowledge-categories'].forEach(id => setEl(id, t));
|
||||||
|
updateProgressBar('dashboard-batch-progress-pending', '0');
|
||||||
|
updateProgressBar('dashboard-batch-progress-running', '0');
|
||||||
|
updateProgressBar('dashboard-batch-progress-done', '0');
|
||||||
|
}
|
||||||
|
|
||||||
|
// 格式化数字,添加千位分隔符
|
||||||
|
function formatNumber(num) {
|
||||||
|
if (typeof num !== 'number' || isNaN(num)) return '-';
|
||||||
|
if (num === 0) return '0';
|
||||||
|
return num.toLocaleString('zh-CN');
|
||||||
|
}
|
||||||
|
|
||||||
|
// 更新进度条宽度
|
||||||
|
function updateProgressBar(id, percentage) {
|
||||||
|
const el = document.getElementById(id);
|
||||||
|
if (el) {
|
||||||
|
const pct = parseFloat(percentage) || 0;
|
||||||
|
el.style.width = Math.max(0, Math.min(100, pct)) + '%';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Top 30 工具执行次数柱状图颜色(30 色不重复,柔和、易区分)
|
||||||
|
var DASHBOARD_BAR_COLORS = [
|
||||||
|
'#93c5fd', '#a78bfa', '#6ee7b7', '#fde047', '#fda4af',
|
||||||
|
'#7dd3fc', '#a5b4fc', '#5eead4', '#fdba74', '#e9d5ff',
|
||||||
|
'#67e8f9', '#c4b5fd', '#86efac', '#fcd34d', '#f9a8d4',
|
||||||
|
'#bae6fd', '#c7d2fe', '#99f6e4', '#fed7aa', '#ddd6fe',
|
||||||
|
'#22d3ee', '#8b5cf6', '#4ade80', '#fbbf24', '#fb7185',
|
||||||
|
'#38bdf8', '#818cf8', '#2dd4bf', '#fb923c', '#e0e7ff'
|
||||||
|
];
|
||||||
|
|
||||||
|
function esc(s) {
|
||||||
|
if (typeof s !== 'string') return '';
|
||||||
|
return s.replace(/&/g, '&').replace(/</g, '<').replace(/"/g, '"');
|
||||||
|
}
|
||||||
|
|
||||||
|
function renderDashboardToolsBar(monitorRes) {
|
||||||
|
const placeholder = document.getElementById('dashboard-tools-pie-placeholder');
|
||||||
|
const barChartEl = document.getElementById('dashboard-tools-bar-chart');
|
||||||
|
if (!placeholder || !barChartEl) return;
|
||||||
|
|
||||||
|
if (!monitorRes || typeof monitorRes !== 'object') {
|
||||||
|
placeholder.style.display = 'block';
|
||||||
|
barChartEl.style.display = 'none';
|
||||||
|
barChartEl.innerHTML = '';
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const entries = Object.keys(monitorRes).map(function (k) {
|
||||||
|
const v = monitorRes[k];
|
||||||
|
const totalCalls = v && (v.totalCalls ?? v.TotalCalls);
|
||||||
|
return { name: k, totalCalls: typeof totalCalls === 'number' ? totalCalls : 0 };
|
||||||
|
}).filter(function (e) { return e.totalCalls > 0; })
|
||||||
|
.sort(function (a, b) { return b.totalCalls - a.totalCalls; })
|
||||||
|
.slice(0, 30);
|
||||||
|
|
||||||
|
if (entries.length === 0) {
|
||||||
|
placeholder.style.display = 'block';
|
||||||
|
barChartEl.style.display = 'none';
|
||||||
|
barChartEl.innerHTML = '';
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
placeholder.style.display = 'none';
|
||||||
|
barChartEl.style.display = 'block';
|
||||||
|
|
||||||
|
const maxCalls = Math.max.apply(null, entries.map(function (e) { return e.totalCalls; }));
|
||||||
|
var html = '';
|
||||||
|
entries.forEach(function (e, i) {
|
||||||
|
var pct = maxCalls > 0 ? (e.totalCalls / maxCalls) * 100 : 0;
|
||||||
|
var label = e.name.length > 12 ? e.name.slice(0, 10) + '…' : e.name;
|
||||||
|
var color = DASHBOARD_BAR_COLORS[i % DASHBOARD_BAR_COLORS.length];
|
||||||
|
var fullName = esc(e.name);
|
||||||
|
html += '<div class="dashboard-tools-bar-item" data-tooltip="' + fullName + '">';
|
||||||
|
html += '<span class="dashboard-tools-bar-label">' + esc(label) + '</span>';
|
||||||
|
html += '<div class="dashboard-tools-bar-track"><div class="dashboard-tools-bar-fill" style="width:' + pct + '%;background:' + color + '"></div></div>';
|
||||||
|
html += '<span class="dashboard-tools-bar-value">' + e.totalCalls + '</span>';
|
||||||
|
html += '</div>';
|
||||||
|
});
|
||||||
|
barChartEl.innerHTML = html;
|
||||||
|
attachDashboardBarTooltips(barChartEl);
|
||||||
|
}
|
||||||
|
|
||||||
|
var dashboardBarTooltipEl = null;
|
||||||
|
var dashboardBarTooltipTimer = null;
|
||||||
|
|
||||||
|
function attachDashboardBarTooltips(barChartEl) {
|
||||||
|
if (!barChartEl) return;
|
||||||
|
if (!dashboardBarTooltipEl) {
|
||||||
|
dashboardBarTooltipEl = document.createElement('div');
|
||||||
|
dashboardBarTooltipEl.className = 'dashboard-tools-bar-tooltip';
|
||||||
|
dashboardBarTooltipEl.setAttribute('role', 'tooltip');
|
||||||
|
document.body.appendChild(dashboardBarTooltipEl);
|
||||||
|
}
|
||||||
|
barChartEl.removeEventListener('mouseover', dashboardBarTooltipOnOver);
|
||||||
|
barChartEl.removeEventListener('mouseout', dashboardBarTooltipOnOut);
|
||||||
|
barChartEl.addEventListener('mouseover', dashboardBarTooltipOnOver);
|
||||||
|
barChartEl.addEventListener('mouseout', dashboardBarTooltipOnOut);
|
||||||
|
}
|
||||||
|
|
||||||
|
function dashboardBarTooltipOnOver(ev) {
|
||||||
|
var item = ev.target && ev.target.closest && ev.target.closest('.dashboard-tools-bar-item');
|
||||||
|
if (!item || !dashboardBarTooltipEl) return;
|
||||||
|
var text = item.getAttribute('data-tooltip');
|
||||||
|
if (!text) return;
|
||||||
|
clearTimeout(dashboardBarTooltipTimer);
|
||||||
|
dashboardBarTooltipTimer = setTimeout(function () {
|
||||||
|
dashboardBarTooltipEl.textContent = text;
|
||||||
|
dashboardBarTooltipEl.style.display = 'block';
|
||||||
|
requestAnimationFrame(function () {
|
||||||
|
var rect = item.getBoundingClientRect();
|
||||||
|
var ttRect = dashboardBarTooltipEl.getBoundingClientRect();
|
||||||
|
var x = rect.left + (rect.width / 2) - (ttRect.width / 2);
|
||||||
|
var y = rect.top - ttRect.height - 6;
|
||||||
|
if (y < 8) y = rect.bottom + 6;
|
||||||
|
var pad = 8;
|
||||||
|
if (x < pad) x = pad;
|
||||||
|
if (x + ttRect.width > window.innerWidth - pad) x = window.innerWidth - ttRect.width - pad;
|
||||||
|
dashboardBarTooltipEl.style.left = x + 'px';
|
||||||
|
dashboardBarTooltipEl.style.top = y + 'px';
|
||||||
|
});
|
||||||
|
}, 180);
|
||||||
|
}
|
||||||
|
|
||||||
|
function dashboardBarTooltipOnOut(ev) {
|
||||||
|
var item = ev.target && ev.target.closest && ev.target.closest('.dashboard-tools-bar-item');
|
||||||
|
var related = ev.relatedTarget && ev.relatedTarget.closest && ev.relatedTarget.closest('.dashboard-tools-bar-item');
|
||||||
|
if (item && item === related) return;
|
||||||
|
clearTimeout(dashboardBarTooltipTimer);
|
||||||
|
dashboardBarTooltipTimer = null;
|
||||||
|
if (dashboardBarTooltipEl) dashboardBarTooltipEl.style.display = 'none';
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
// 页面路由管理
|
// 页面路由管理
|
||||||
let currentPage = 'chat';
|
let currentPage = 'dashboard';
|
||||||
|
|
||||||
// 初始化路由
|
// 初始化路由
|
||||||
function initRouter() {
|
function initRouter() {
|
||||||
@@ -32,8 +32,8 @@ function initRouter() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 默认显示对话页面
|
// 默认显示仪表盘
|
||||||
switchPage('chat');
|
switchPage('dashboard');
|
||||||
}
|
}
|
||||||
|
|
||||||
// 切换页面
|
// 切换页面
|
||||||
|
|||||||
+147
-97
@@ -204,7 +204,7 @@
|
|||||||
<!-- 内容区域 -->
|
<!-- 内容区域 -->
|
||||||
<div class="content-area">
|
<div class="content-area">
|
||||||
<!-- 仪表盘页面 -->
|
<!-- 仪表盘页面 -->
|
||||||
<div id="page-dashboard" class="page">
|
<div id="page-dashboard" class="page active">
|
||||||
<div class="dashboard-page">
|
<div class="dashboard-page">
|
||||||
<div class="page-header">
|
<div class="page-header">
|
||||||
<h2>仪表盘</h2>
|
<h2>仪表盘</h2>
|
||||||
@@ -213,114 +213,164 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="dashboard-content">
|
<div class="dashboard-content">
|
||||||
<div class="dashboard-cards" id="dashboard-cards">
|
<!-- 第一行:核心 KPI(仪表盘最佳实践:关键指标置顶) -->
|
||||||
<div class="dashboard-card dashboard-card-tasks" onclick="switchPage('tasks')">
|
<div class="dashboard-kpi-row" id="dashboard-cards">
|
||||||
<div class="dashboard-card-glow"></div>
|
<div class="dashboard-kpi-card" role="button" tabindex="0" onclick="switchPage('tasks')" onkeydown="if(event.key==='Enter'||event.key===' ') { event.preventDefault(); switchPage('tasks'); }" title="点击查看任务管理"> <div class="dashboard-kpi-value" id="dashboard-running-tasks">-</div><div class="dashboard-kpi-label">运行中任务</div></div>
|
||||||
<div class="dashboard-card-icon">
|
<div class="dashboard-kpi-card" role="button" tabindex="0" onclick="switchPage('vulnerabilities')" onkeydown="if(event.key==='Enter'||event.key===' ') { event.preventDefault(); switchPage('vulnerabilities'); }" title="点击查看漏洞管理"><div class="dashboard-kpi-value" id="dashboard-vuln-total">-</div><div class="dashboard-kpi-label">漏洞总数</div></div>
|
||||||
<svg width="32" height="32" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M9 11l3 3L22 4"></path><path d="M21 12v7a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2V5a2 2 0 0 1 2-2h11"></path></svg>
|
<div class="dashboard-kpi-card" role="button" tabindex="0" onclick="switchPage('mcp-monitor')" onkeydown="if(event.key==='Enter'||event.key===' ') { event.preventDefault(); switchPage('mcp-monitor'); }" title="点击查看 MCP 监控"><div class="dashboard-kpi-value" id="dashboard-kpi-tools-calls">-</div><div class="dashboard-kpi-label">工具调用次数</div></div>
|
||||||
</div>
|
<div class="dashboard-kpi-card" role="button" tabindex="0" onclick="switchPage('mcp-monitor')" onkeydown="if(event.key==='Enter'||event.key===' ') { event.preventDefault(); switchPage('mcp-monitor'); }" title="点击查看 MCP 监控"><div class="dashboard-kpi-value" id="dashboard-kpi-success-rate">-</div><div class="dashboard-kpi-label">工具执行成功率</div></div>
|
||||||
<div class="dashboard-card-body">
|
|
||||||
<div class="dashboard-card-value" id="dashboard-running-tasks">-</div>
|
|
||||||
<div class="dashboard-card-label">运行中任务</div>
|
|
||||||
<div class="dashboard-card-hint">点击查看任务管理</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="dashboard-card dashboard-card-vulns" onclick="switchPage('vulnerabilities')">
|
|
||||||
<div class="dashboard-card-glow"></div>
|
|
||||||
<div class="dashboard-card-icon">
|
|
||||||
<svg width="32" height="32" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M12 22s8-4 8-10V5l-8-3-8 3v7c0 6 8 10 8 10z"></path></svg>
|
|
||||||
</div>
|
|
||||||
<div class="dashboard-card-body">
|
|
||||||
<div class="dashboard-card-value" id="dashboard-vuln-total">-</div>
|
|
||||||
<div class="dashboard-card-label">漏洞总数</div>
|
|
||||||
<div class="dashboard-card-hint">点击查看漏洞管理</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="dashboard-card dashboard-card-chat" onclick="switchPage('chat')">
|
|
||||||
<div class="dashboard-card-glow"></div>
|
|
||||||
<div class="dashboard-card-icon">
|
|
||||||
<svg width="32" height="32" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M21 15a2 2 0 0 1-2 2H7l-4 4V5a2 2 0 0 1 2-2h14a2 2 0 0 1 2 2z"></path></svg>
|
|
||||||
</div>
|
|
||||||
<div class="dashboard-card-body">
|
|
||||||
<div class="dashboard-card-cta">开始对话</div>
|
|
||||||
<div class="dashboard-card-desc">与 AI 进行安全测试</div>
|
|
||||||
<div class="dashboard-card-hint">立即开始</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
<div class="dashboard-section dashboard-section-chart">
|
<!-- 两列主内容区 -->
|
||||||
<h3 class="dashboard-section-title"><span class="dashboard-section-dot"></span>漏洞严重程度分布</h3>
|
<div class="dashboard-grid">
|
||||||
<div class="dashboard-chart-wrap">
|
<div class="dashboard-main">
|
||||||
<div class="dashboard-stacked-bar" id="dashboard-stacked-bar">
|
<section class="dashboard-section dashboard-section-chart">
|
||||||
<span class="dashboard-bar-seg seg-critical" id="dashboard-bar-critical" style="width: 0%"></span>
|
<h3 class="dashboard-section-title">漏洞严重程度分布</h3>
|
||||||
<span class="dashboard-bar-seg seg-high" id="dashboard-bar-high" style="width: 0%"></span>
|
<div class="dashboard-chart-wrap">
|
||||||
<span class="dashboard-bar-seg seg-medium" id="dashboard-bar-medium" style="width: 0%"></span>
|
<div class="dashboard-stacked-bar" id="dashboard-stacked-bar">
|
||||||
<span class="dashboard-bar-seg seg-low" id="dashboard-bar-low" style="width: 0%"></span>
|
<span class="dashboard-bar-seg seg-critical" id="dashboard-bar-critical" style="width: 0%"></span>
|
||||||
<span class="dashboard-bar-seg seg-info" id="dashboard-bar-info" style="width: 0%"></span>
|
<span class="dashboard-bar-seg seg-high" id="dashboard-bar-high" style="width: 0%"></span>
|
||||||
</div>
|
<span class="dashboard-bar-seg seg-medium" id="dashboard-bar-medium" style="width: 0%"></span>
|
||||||
<div class="dashboard-legend" id="dashboard-vuln-bars">
|
<span class="dashboard-bar-seg seg-low" id="dashboard-bar-low" style="width: 0%"></span>
|
||||||
<div class="dashboard-legend-item"><span class="dashboard-legend-dot critical"></span><span class="dashboard-legend-label">严重</span><span class="dashboard-legend-value" id="dashboard-severity-critical">0</span></div>
|
<span class="dashboard-bar-seg seg-info" id="dashboard-bar-info" style="width: 0%"></span>
|
||||||
<div class="dashboard-legend-item"><span class="dashboard-legend-dot high"></span><span class="dashboard-legend-label">高危</span><span class="dashboard-legend-value" id="dashboard-severity-high">0</span></div>
|
</div>
|
||||||
<div class="dashboard-legend-item"><span class="dashboard-legend-dot medium"></span><span class="dashboard-legend-label">中危</span><span class="dashboard-legend-value" id="dashboard-severity-medium">0</span></div>
|
<div class="dashboard-legend" id="dashboard-vuln-bars">
|
||||||
<div class="dashboard-legend-item"><span class="dashboard-legend-dot low"></span><span class="dashboard-legend-label">低危</span><span class="dashboard-legend-value" id="dashboard-severity-low">0</span></div>
|
<div class="dashboard-legend-item"><span class="dashboard-legend-dot critical"></span><span class="dashboard-legend-label">严重</span><span class="dashboard-legend-value" id="dashboard-severity-critical">0</span></div>
|
||||||
<div class="dashboard-legend-item"><span class="dashboard-legend-dot info"></span><span class="dashboard-legend-label">信息</span><span class="dashboard-legend-value" id="dashboard-severity-info">0</span></div>
|
<div class="dashboard-legend-item"><span class="dashboard-legend-dot high"></span><span class="dashboard-legend-label">高危</span><span class="dashboard-legend-value" id="dashboard-severity-high">0</span></div>
|
||||||
</div>
|
<div class="dashboard-legend-item"><span class="dashboard-legend-dot medium"></span><span class="dashboard-legend-label">中危</span><span class="dashboard-legend-value" id="dashboard-severity-medium">0</span></div>
|
||||||
|
<div class="dashboard-legend-item"><span class="dashboard-legend-dot low"></span><span class="dashboard-legend-label">低危</span><span class="dashboard-legend-value" id="dashboard-severity-low">0</span></div>
|
||||||
|
<div class="dashboard-legend-item"><span class="dashboard-legend-dot info"></span><span class="dashboard-legend-label">信息</span><span class="dashboard-legend-value" id="dashboard-severity-info">0</span></div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
<section class="dashboard-section dashboard-section-overview">
|
||||||
|
<h3 class="dashboard-section-title">运行概览</h3>
|
||||||
|
<div class="dashboard-overview-list">
|
||||||
|
<div class="dashboard-overview-item dashboard-overview-item-batch" role="button" tabindex="0" onclick="switchPage('tasks')" onkeydown="if(event.key==='Enter'||event.key===' ') { event.preventDefault(); switchPage('tasks'); }">
|
||||||
|
<span class="dashboard-overview-icon dashboard-overview-icon-batch" aria-hidden="true"><svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><rect x="3" y="3" width="7" height="7"/><rect x="14" y="3" width="7" height="7"/><rect x="14" y="14" width="7" height="7"/><rect x="3" y="14" width="7" height="7"/></svg></span>
|
||||||
|
<div class="dashboard-overview-content">
|
||||||
|
<div class="dashboard-overview-header">
|
||||||
|
<span class="dashboard-overview-label">批量任务队列</span>
|
||||||
|
<span class="dashboard-overview-total" id="dashboard-batch-total">-</span>
|
||||||
|
</div>
|
||||||
|
<div class="dashboard-overview-stats">
|
||||||
|
<span class="dashboard-overview-stat dashboard-overview-stat-pending">
|
||||||
|
<span class="dashboard-overview-stat-badge badge-pending"></span>
|
||||||
|
<span class="dashboard-overview-stat-value" id="dashboard-batch-pending">-</span>
|
||||||
|
<span class="dashboard-overview-stat-label">待执行</span>
|
||||||
|
</span>
|
||||||
|
<span class="dashboard-overview-stat dashboard-overview-stat-running">
|
||||||
|
<span class="dashboard-overview-stat-badge badge-running"></span>
|
||||||
|
<span class="dashboard-overview-stat-value" id="dashboard-batch-running">-</span>
|
||||||
|
<span class="dashboard-overview-stat-label">执行中</span>
|
||||||
|
</span>
|
||||||
|
<span class="dashboard-overview-stat dashboard-overview-stat-done">
|
||||||
|
<span class="dashboard-overview-stat-badge badge-done"></span>
|
||||||
|
<span class="dashboard-overview-stat-value" id="dashboard-batch-done">-</span>
|
||||||
|
<span class="dashboard-overview-stat-label">已完成</span>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div class="dashboard-overview-progress">
|
||||||
|
<div class="dashboard-overview-progress-bar">
|
||||||
|
<div class="dashboard-overview-progress-segment dashboard-overview-progress-pending" id="dashboard-batch-progress-pending" style="width: 0%"></div>
|
||||||
|
<div class="dashboard-overview-progress-segment dashboard-overview-progress-running" id="dashboard-batch-progress-running" style="width: 0%"></div>
|
||||||
|
<div class="dashboard-overview-progress-segment dashboard-overview-progress-done" id="dashboard-batch-progress-done" style="width: 0%"></div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="dashboard-overview-item dashboard-overview-item-tools" role="button" tabindex="0" onclick="switchPage('mcp-monitor')" onkeydown="if(event.key==='Enter'||event.key===' ') { event.preventDefault(); switchPage('mcp-monitor'); }">
|
||||||
|
<span class="dashboard-overview-icon dashboard-overview-icon-tools" aria-hidden="true"><svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="M14.7 6.3a1 1 0 0 0 0 1.4l1.6 1.6a1 1 0 0 0 1.4 0l3.77-3.77a6 6 0 0 1-7.94 7.94l-6.91 6.91a2.12 2.12 0 0 1-3-3l6.91-6.91a6 6 0 0 1 7.94-7.94l-3.76 3.76z"/></svg></span>
|
||||||
|
<div class="dashboard-overview-content">
|
||||||
|
<div class="dashboard-overview-header">
|
||||||
|
<span class="dashboard-overview-label">工具调用</span>
|
||||||
|
<span class="dashboard-overview-success-rate" id="dashboard-tools-success-rate">-</span>
|
||||||
|
</div>
|
||||||
|
<div class="dashboard-overview-value-group">
|
||||||
|
<span class="dashboard-overview-value-large" id="dashboard-tools-calls">-</span>
|
||||||
|
<span class="dashboard-overview-value-unit">次调用</span>
|
||||||
|
<span class="dashboard-overview-value-separator">·</span>
|
||||||
|
<span class="dashboard-overview-value-normal" id="dashboard-tools-count">-</span>
|
||||||
|
<span class="dashboard-overview-value-unit">个工具</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="dashboard-overview-item dashboard-overview-item-knowledge" role="button" tabindex="0" onclick="switchPage('knowledge-management')" onkeydown="if(event.key==='Enter'||event.key===' ') { event.preventDefault(); switchPage('knowledge-management'); }">
|
||||||
|
<span class="dashboard-overview-icon dashboard-overview-icon-knowledge" aria-hidden="true"><svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="M4 19.5A2.5 2.5 0 0 1 6.5 17H20"></path><path d="M6.5 2H20v20H6.5A2.5 2.5 0 0 1 4 19.5v-15A2.5 2.5 0 0 1 6.5 2z"></path></svg></span>
|
||||||
|
<div class="dashboard-overview-content">
|
||||||
|
<div class="dashboard-overview-header">
|
||||||
|
<span class="dashboard-overview-label">知识</span>
|
||||||
|
</div>
|
||||||
|
<div class="dashboard-overview-value-group">
|
||||||
|
<span class="dashboard-overview-value-large" id="dashboard-knowledge-items">-</span>
|
||||||
|
<span class="dashboard-overview-value-unit">项知识</span>
|
||||||
|
<span class="dashboard-overview-value-separator">·</span>
|
||||||
|
<span class="dashboard-overview-value-normal" id="dashboard-knowledge-categories">-</span>
|
||||||
|
<span class="dashboard-overview-value-unit">个分类</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="dashboard-overview-item dashboard-overview-item-skills" role="button" tabindex="0" onclick="switchPage('skills-monitor')" onkeydown="if(event.key==='Enter'||event.key===' ') { event.preventDefault(); switchPage('skills-monitor'); }">
|
||||||
|
<span class="dashboard-overview-icon dashboard-overview-icon-skills" aria-hidden="true"><svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="M14 2H6a2 2 0 0 0-2 2v16a2 2 0 0 0 2 2h12a2 2 0 0 0 2-2V8z"/><polyline points="14 2 14 8 20 8"/><line x1="16" y1="13" x2="8" y2="13"/><line x1="16" y1="17" x2="8" y2="17"/></svg></span>
|
||||||
|
<div class="dashboard-overview-content">
|
||||||
|
<div class="dashboard-overview-header">
|
||||||
|
<span class="dashboard-overview-label">Skills</span>
|
||||||
|
<span class="dashboard-overview-status" id="dashboard-skills-status">-</span>
|
||||||
|
</div>
|
||||||
|
<div class="dashboard-overview-value-group">
|
||||||
|
<span class="dashboard-overview-value-large" id="dashboard-skills-calls">-</span>
|
||||||
|
<span class="dashboard-overview-value-unit">次调用</span>
|
||||||
|
<span class="dashboard-overview-value-separator">·</span>
|
||||||
|
<span class="dashboard-overview-value-normal" id="dashboard-skills-count">-</span>
|
||||||
|
<span class="dashboard-overview-value-unit">个 Skill</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
<section class="dashboard-section dashboard-section-quick dashboard-quick-inline">
|
||||||
|
<h3 class="dashboard-section-title">快捷入口</h3>
|
||||||
|
<div class="dashboard-quick-links dashboard-quick-links-row">
|
||||||
|
<a class="dashboard-quick-link" onclick="switchPage('chat')"><span class="dashboard-quick-icon"><svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="M21 15a2 2 0 0 1-2 2H7l-4 4V5a2 2 0 0 1 2-2h14a2 2 0 0 1 2 2z"></path></svg></span><span>对话</span></a>
|
||||||
|
<a class="dashboard-quick-link" onclick="switchPage('tasks')"><span class="dashboard-quick-icon"><svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="M9 11l3 3L22 4"></path><path d="M21 12v7a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2V5a2 2 0 0 1 2-2h11"></path></svg></span><span>任务管理</span></a>
|
||||||
|
<a class="dashboard-quick-link" onclick="switchPage('vulnerabilities')"><span class="dashboard-quick-icon"><svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="M12 22s8-4 8-10V5l-8-3-8 3v7c0 6 8 10 8 10z"></path></svg></span><span>漏洞管理</span></a>
|
||||||
|
<a class="dashboard-quick-link" onclick="switchPage('mcp-management')"><span class="dashboard-quick-icon"><svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="M13 2L3 14h9l-1 8 10-12h-9l1-8z"></path></svg></span><span>MCP 管理</span></a>
|
||||||
|
<a class="dashboard-quick-link" onclick="switchPage('knowledge-management')"><span class="dashboard-quick-icon"><svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="M2 3h6a4 4 0 0 1 4 4v14a3 3 0 0 0-3-3H2z"></path><path d="M22 3h-6a4 4 0 0 0-4 4v14a3 3 0 0 1 3-3h7z"></path></svg></span><span>知识管理</span></a>
|
||||||
|
<a class="dashboard-quick-link" onclick="switchPage('skills-management')"><span class="dashboard-quick-icon"><svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="M14.5 2H6a2 2 0 0 0-2 2v16a2 2 0 0 0 2 2h12a2 2 0 0 0 2-2V7.5L14.5 2z"></path><polyline points="14 2 14 8 20 8"></polyline></svg></span><span>Skills 管理</span></a>
|
||||||
|
<a class="dashboard-quick-link" onclick="switchPage('roles-management')"><span class="dashboard-quick-icon"><svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="M17 21v-2a4 4 0 0 0-4-4H5a4 4 0 0 0-4 4v2"></path><circle cx="9" cy="7" r="4"></circle><path d="M23 21v-2a4 4 0 0 0-3-3.87"></path><path d="M16 3.13a4 4 0 0 1 0 7.75"></path></svg></span><span>角色管理</span></a>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
<div class="dashboard-side">
|
||||||
<div class="dashboard-section dashboard-section-overview">
|
<section class="dashboard-section dashboard-section-tools">
|
||||||
<h3 class="dashboard-section-title"><span class="dashboard-section-dot"></span>运行概览</h3>
|
<h3 class="dashboard-section-title">工具执行次数</h3>
|
||||||
<div class="dashboard-overview-grid">
|
<div class="dashboard-tools-chart-wrap">
|
||||||
<div class="dashboard-overview-item" onclick="switchPage('tasks')">
|
<div class="dashboard-tools-chart-placeholder" id="dashboard-tools-pie-placeholder">暂无数据</div>
|
||||||
<span class="dashboard-overview-label">批量任务队列</span>
|
<div class="dashboard-tools-bar-chart" id="dashboard-tools-bar-chart"></div>
|
||||||
<span class="dashboard-overview-value"><span id="dashboard-batch-pending">-</span> 待执行 / <span id="dashboard-batch-running">-</span> 执行中 / <span id="dashboard-batch-done">-</span> 已完成</span>
|
</div>
|
||||||
</div>
|
</section>
|
||||||
<div class="dashboard-overview-item" onclick="switchPage('mcp-monitor')">
|
|
||||||
<span class="dashboard-overview-label">工具调用</span>
|
|
||||||
<span class="dashboard-overview-value"><span id="dashboard-tools-count">-</span> 个工具,共 <span id="dashboard-tools-calls">-</span> 次调用</span>
|
|
||||||
</div>
|
|
||||||
<div class="dashboard-overview-item" onclick="switchPage('skills-monitor')">
|
|
||||||
<span class="dashboard-overview-label">Skills</span>
|
|
||||||
<span class="dashboard-overview-value"><span id="dashboard-skills-count">-</span> 个 Skill,共 <span id="dashboard-skills-calls">-</span> 次调用</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="dashboard-section">
|
|
||||||
<h3 class="dashboard-section-title"><span class="dashboard-section-dot"></span>快捷入口</h3>
|
|
||||||
<div class="dashboard-quick-links">
|
|
||||||
<a class="dashboard-quick-link" onclick="switchPage('tasks')">
|
|
||||||
<span class="dashboard-quick-icon"><svg width="18" height="18" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="M9 11l3 3L22 4"></path><path d="M21 12v7a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2V5a2 2 0 0 1 2-2h11"></path></svg></span>
|
|
||||||
<span>任务管理</span>
|
|
||||||
</a>
|
|
||||||
<a class="dashboard-quick-link" onclick="switchPage('vulnerabilities')">
|
|
||||||
<span class="dashboard-quick-icon"><svg width="18" height="18" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="M12 22s8-4 8-10V5l-8-3-8 3v7c0 6 8 10 8 10z"></path></svg></span>
|
|
||||||
<span>漏洞管理</span>
|
|
||||||
</a>
|
|
||||||
<a class="dashboard-quick-link" onclick="switchPage('chat')">
|
|
||||||
<span class="dashboard-quick-icon"><svg width="18" height="18" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="M21 15a2 2 0 0 1-2 2H7l-4 4V5a2 2 0 0 1 2-2h14a2 2 0 0 1 2 2z"></path></svg></span>
|
|
||||||
<span>对话</span>
|
|
||||||
</a>
|
|
||||||
<a class="dashboard-quick-link" onclick="switchPage('mcp-monitor')">
|
|
||||||
<span class="dashboard-quick-icon"><svg width="18" height="18" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="M13 2L3 14h9l-1 8 10-12h-9l1-8z"></path></svg></span>
|
|
||||||
<span>MCP 监控</span>
|
|
||||||
</a>
|
|
||||||
<a class="dashboard-quick-link" onclick="switchPage('skills-monitor')">
|
|
||||||
<span class="dashboard-quick-icon"><svg width="18" height="18" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="M14.5 2H6a2 2 0 0 0-2 2v16a2 2 0 0 0 2 2h12a2 2 0 0 0 2-2V7.5L14.5 2z"></path><polyline points="14 2 14 8 20 8"></polyline></svg></span>
|
|
||||||
<span>Skills 监控</span>
|
|
||||||
</a>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="dashboard-cta-block">
|
<div class="dashboard-cta-block">
|
||||||
<div class="dashboard-cta-inner">
|
<div class="dashboard-cta-content">
|
||||||
<p class="dashboard-cta-text">准备好开始安全测试?</p>
|
<div class="dashboard-cta-icon" aria-hidden="true">
|
||||||
<button class="dashboard-cta-btn" onclick="switchPage('chat')">前往对话 →</button>
|
<svg width="28" height="28" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.75" stroke-linecap="round" stroke-linejoin="round"><path d="M21 15a2 2 0 0 1-2 2H7l-4 4V5a2 2 0 0 1 2-2h14a2 2 0 0 1 2 2z"></path></svg>
|
||||||
|
</div>
|
||||||
|
<div class="dashboard-cta-copy">
|
||||||
|
<p class="dashboard-cta-text">开始你的安全之旅</p>
|
||||||
|
<p class="dashboard-cta-sub">在对话中描述目标,AI 将协助执行扫描与漏洞分析</p>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
<button class="dashboard-cta-btn" onclick="switchPage('chat')">
|
||||||
|
<span>前往对话</span>
|
||||||
|
<span class="dashboard-cta-btn-arrow" aria-hidden="true"><svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5"><path d="M5 12h14M12 5l7 7-7 7"/></svg></span>
|
||||||
|
</button>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<!-- 对话页面 -->
|
<!-- 对话页面 -->
|
||||||
<div id="page-chat" class="page active">
|
<div id="page-chat" class="page">
|
||||||
<div class="chat-page-layout">
|
<div class="chat-page-layout">
|
||||||
<!-- 历史对话侧边栏 -->
|
<!-- 历史对话侧边栏 -->
|
||||||
<aside class="conversation-sidebar">
|
<aside class="conversation-sidebar">
|
||||||
|
|||||||
Reference in New Issue
Block a user