/** * AI 运行总览 Dashboard 页面。 * * - 顶部:门店筛选 + 刷新 * - 第一行:4 个统计卡片(今日调用、成功率、Token 消耗、平均延迟) * - 第二行:7 天趋势表格 + App 调用占比表格 * - 第三行:Token 预算进度条 + App 健康状态 * - 第四行:告警列表 */ import React, { useEffect, useState, useCallback } from "react"; import { Card, Row, Col, Statistic, Table, Tag, Badge, Progress, Select, Button, message, Typography, Space, } from "antd"; import { ReloadOutlined } from "@ant-design/icons"; import type { ColumnsType } from "antd/es/table"; import { getDashboard, type DashboardResponse, type DailyTrend, type AppDistItem, type AlertItem, type AppHealthItem, } from "../api/adminAI"; const { Title } = Typography; const ALERT_STATUS_COLOR: Record = { failed: "red", timeout: "orange", circuit_open: "volcano", }; const ALERT_MGMT_COLOR: Record = { pending: "warning", acknowledged: "success", ignored: "default", }; const HEALTH_STATUS: Record = { success: "success", failed: "error", timeout: "warning", circuit_open: "error", }; function fmtTime(raw: string | null): string { if (!raw) return "—"; const d = new Date(raw); return Number.isNaN(d.getTime()) ? raw : d.toLocaleString("zh-CN"); } // ---- 表格列定义 ---- const trendColumns: ColumnsType = [ { title: "日期", dataIndex: "date", key: "date", width: 120 }, { title: "调用量", dataIndex: "calls", key: "calls", align: "right" }, { title: "成功率", dataIndex: "success_rate", key: "success_rate", align: "right", render: (v: number) => `${(v * 100).toFixed(1)}%`, }, ]; const distColumns: ColumnsType = [ { title: "App 类型", dataIndex: "app_type", key: "app_type" }, { title: "调用次数", dataIndex: "count", key: "count", align: "right" }, { title: "占比", dataIndex: "percentage", key: "percentage", align: "right", render: (v: number) => `${(v * 100).toFixed(1)}%`, }, ]; const alertColumns: ColumnsType = [ { title: "ID", dataIndex: "id", key: "id", width: 80 }, { title: "App", dataIndex: "app_type", key: "app_type", width: 160 }, { title: "状态", dataIndex: "status", key: "status", width: 110, render: (v: string) => {v}, }, { title: "告警状态", dataIndex: "alert_status", key: "alert_status", width: 110, render: (v: string | null) => v ? {v} : "—", }, { title: "错误信息", dataIndex: "error_message", key: "error_message", ellipsis: true, render: (v: string | null) => v ?? "—", }, { title: "时间", dataIndex: "created_at", key: "created_at", width: 170, render: fmtTime }, ]; // ---- 页面组件 ---- const AIDashboard: React.FC = () => { const [data, setData] = useState(null); const [loading, setLoading] = useState(false); const [siteId, setSiteId] = useState(undefined); const load = useCallback(async () => { setLoading(true); try { const res = await getDashboard(siteId); setData(res); } catch { message.error("加载 Dashboard 失败"); } finally { setLoading(false); } }, [siteId]); useEffect(() => { load(); }, [load]); return (
{/* 顶部:门店筛选 + 刷新 */} AI 运行总览