mirror of
https://github.com/whyour/qinglong.git
synced 2026-04-29 00:45:11 +08:00
Add statistics panel: backend models, services, APIs and frontend page
Agent-Logs-Url: https://github.com/whyour/qinglong/sessions/3db54913-03d2-4721-b720-8ccbf8d0f00e Co-authored-by: whyour <22700758+whyour@users.noreply.github.com>
This commit is contained in:
parent
0995808309
commit
34bc18cb25
|
|
@ -3,6 +3,7 @@ import { Container } from 'typedi';
|
||||||
import { Logger } from 'winston';
|
import { Logger } from 'winston';
|
||||||
import CronService from '../services/cron';
|
import CronService from '../services/cron';
|
||||||
import CronViewService from '../services/cronView';
|
import CronViewService from '../services/cronView';
|
||||||
|
import CronStatsService from '../services/cronStats';
|
||||||
import { celebrate, Joi } from 'celebrate';
|
import { celebrate, Joi } from 'celebrate';
|
||||||
import { commonCronSchema } from '../validation/schedule';
|
import { commonCronSchema } from '../validation/schedule';
|
||||||
|
|
||||||
|
|
@ -141,6 +142,58 @@ export default (app: Router) => {
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
|
||||||
|
route.get(
|
||||||
|
'/stats',
|
||||||
|
async (req: Request, res: Response, next: NextFunction) => {
|
||||||
|
try {
|
||||||
|
const cronStatsService = Container.get(CronStatsService);
|
||||||
|
const data = await cronStatsService.stats();
|
||||||
|
return res.send({ code: 200, data });
|
||||||
|
} catch (e) {
|
||||||
|
return next(e);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
|
route.get(
|
||||||
|
'/stats/trend',
|
||||||
|
async (req: Request, res: Response, next: NextFunction) => {
|
||||||
|
try {
|
||||||
|
const cronStatsService = Container.get(CronStatsService);
|
||||||
|
const data = await cronStatsService.trend();
|
||||||
|
return res.send({ code: 200, data });
|
||||||
|
} catch (e) {
|
||||||
|
return next(e);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
|
route.get(
|
||||||
|
'/stats/top-duration',
|
||||||
|
async (req: Request, res: Response, next: NextFunction) => {
|
||||||
|
try {
|
||||||
|
const cronStatsService = Container.get(CronStatsService);
|
||||||
|
const data = await cronStatsService.topDuration();
|
||||||
|
return res.send({ code: 200, data });
|
||||||
|
} catch (e) {
|
||||||
|
return next(e);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
|
route.get(
|
||||||
|
'/stats/top-count',
|
||||||
|
async (req: Request, res: Response, next: NextFunction) => {
|
||||||
|
try {
|
||||||
|
const cronStatsService = Container.get(CronStatsService);
|
||||||
|
const data = await cronStatsService.topCount();
|
||||||
|
return res.send({ code: 200, data });
|
||||||
|
} catch (e) {
|
||||||
|
return next(e);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
route.get('/', async (req: Request, res: Response, next: NextFunction) => {
|
route.get('/', async (req: Request, res: Response, next: NextFunction) => {
|
||||||
const logger: Logger = Container.get('logger');
|
const logger: Logger = Container.get('logger');
|
||||||
try {
|
try {
|
||||||
|
|
|
||||||
31
back/data/cronLog.ts
Normal file
31
back/data/cronLog.ts
Normal file
|
|
@ -0,0 +1,31 @@
|
||||||
|
import { sequelize } from '.';
|
||||||
|
import { DataTypes, Model } from 'sequelize';
|
||||||
|
|
||||||
|
export class CronLog {
|
||||||
|
id?: number;
|
||||||
|
cron_id: number;
|
||||||
|
cron_name: string;
|
||||||
|
start_time: number;
|
||||||
|
duration: number;
|
||||||
|
|
||||||
|
constructor(options: CronLog) {
|
||||||
|
this.cron_id = options.cron_id;
|
||||||
|
this.cron_name = options.cron_name;
|
||||||
|
this.start_time = options.start_time;
|
||||||
|
this.duration = options.duration;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface CronLogInstance extends Model<CronLog, CronLog>, CronLog {}
|
||||||
|
export const CronLogModel = sequelize.define<CronLogInstance>(
|
||||||
|
'CronLog',
|
||||||
|
{
|
||||||
|
cron_id: DataTypes.NUMBER,
|
||||||
|
cron_name: DataTypes.STRING,
|
||||||
|
start_time: DataTypes.NUMBER,
|
||||||
|
duration: DataTypes.NUMBER,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
indexes: [{ fields: ['cron_id'] }, { fields: ['start_time'] }],
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
@ -6,6 +6,7 @@ import { AppModel } from '../data/open';
|
||||||
import { SystemModel } from '../data/system';
|
import { SystemModel } from '../data/system';
|
||||||
import { SubscriptionModel } from '../data/subscription';
|
import { SubscriptionModel } from '../data/subscription';
|
||||||
import { CrontabViewModel } from '../data/cronView';
|
import { CrontabViewModel } from '../data/cronView';
|
||||||
|
import { CronLogModel } from '../data/cronLog';
|
||||||
import { sequelize } from '../data';
|
import { sequelize } from '../data';
|
||||||
|
|
||||||
export default async () => {
|
export default async () => {
|
||||||
|
|
@ -17,6 +18,7 @@ export default async () => {
|
||||||
await EnvModel.sync();
|
await EnvModel.sync();
|
||||||
await SubscriptionModel.sync();
|
await SubscriptionModel.sync();
|
||||||
await CrontabViewModel.sync();
|
await CrontabViewModel.sync();
|
||||||
|
await CronLogModel.sync();
|
||||||
|
|
||||||
// 初始化新增字段
|
// 初始化新增字段
|
||||||
const migrations = [
|
const migrations = [
|
||||||
|
|
|
||||||
|
|
@ -2,6 +2,7 @@ import { Service, Inject } from 'typedi';
|
||||||
import winston from 'winston';
|
import winston from 'winston';
|
||||||
import config from '../config';
|
import config from '../config';
|
||||||
import { Crontab, CrontabModel, CrontabStatus } from '../data/cron';
|
import { Crontab, CrontabModel, CrontabStatus } from '../data/cron';
|
||||||
|
import { CronLog, CronLogModel } from '../data/cronLog';
|
||||||
import { exec, execSync } from 'child_process';
|
import { exec, execSync } from 'child_process';
|
||||||
import fs from 'fs/promises';
|
import fs from 'fs/promises';
|
||||||
import CronExpressionParser from 'cron-parser';
|
import CronExpressionParser from 'cron-parser';
|
||||||
|
|
@ -176,6 +177,17 @@ export default class CronService {
|
||||||
{ ...pickBy(options, (v) => v === 0 || !!v) },
|
{ ...pickBy(options, (v) => v === 0 || !!v) },
|
||||||
{ where: { id } },
|
{ where: { id } },
|
||||||
);
|
);
|
||||||
|
|
||||||
|
if (status === CrontabStatus.idle && last_running_time > 0) {
|
||||||
|
await CronLogModel.create(
|
||||||
|
new CronLog({
|
||||||
|
cron_id: id,
|
||||||
|
cron_name: cron.name || cron.command || '',
|
||||||
|
start_time: last_execution_time,
|
||||||
|
duration: last_running_time,
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
164
back/services/cronStats.ts
Normal file
164
back/services/cronStats.ts
Normal file
|
|
@ -0,0 +1,164 @@
|
||||||
|
import { Service, Inject } from 'typedi';
|
||||||
|
import winston from 'winston';
|
||||||
|
import { CrontabModel, CrontabStatus } from '../data/cron';
|
||||||
|
import { CronLogModel } from '../data/cronLog';
|
||||||
|
import { Op } from 'sequelize';
|
||||||
|
import dayjs from 'dayjs';
|
||||||
|
|
||||||
|
@Service()
|
||||||
|
export default class CronStatsService {
|
||||||
|
constructor(@Inject('logger') private logger: winston.Logger) {}
|
||||||
|
|
||||||
|
public async stats() {
|
||||||
|
const todayStart = dayjs().startOf('day').unix();
|
||||||
|
const todayEnd = dayjs().endOf('day').unix();
|
||||||
|
|
||||||
|
const [allCrons, todayLogs] = await Promise.all([
|
||||||
|
CrontabModel.findAll({ where: {} }),
|
||||||
|
CronLogModel.findAll({
|
||||||
|
where: {
|
||||||
|
start_time: { [Op.between]: [todayStart, todayEnd] },
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
]);
|
||||||
|
|
||||||
|
const total = allCrons.length;
|
||||||
|
const enabled = allCrons.filter((c) => c.isDisabled !== 1).length;
|
||||||
|
const disabled = allCrons.filter((c) => c.isDisabled === 1).length;
|
||||||
|
|
||||||
|
const todayCount = todayLogs.length;
|
||||||
|
const todayTotalDuration = todayLogs.reduce(
|
||||||
|
(sum, l) => sum + (l.duration || 0),
|
||||||
|
0,
|
||||||
|
);
|
||||||
|
const todayAvgDuration =
|
||||||
|
todayCount > 0 ? Math.round(todayTotalDuration / todayCount) : 0;
|
||||||
|
|
||||||
|
return {
|
||||||
|
total,
|
||||||
|
enabled,
|
||||||
|
disabled,
|
||||||
|
today: {
|
||||||
|
count: todayCount,
|
||||||
|
avgDuration: todayAvgDuration,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
public async trend() {
|
||||||
|
const days = 7;
|
||||||
|
const result: Array<{
|
||||||
|
date: string;
|
||||||
|
count: number;
|
||||||
|
}> = [];
|
||||||
|
|
||||||
|
for (let i = days - 1; i >= 0; i--) {
|
||||||
|
const dayStart = dayjs().subtract(i, 'day').startOf('day').unix();
|
||||||
|
const dayEnd = dayjs().subtract(i, 'day').endOf('day').unix();
|
||||||
|
const date = dayjs().subtract(i, 'day').format('MM-DD');
|
||||||
|
|
||||||
|
const logs = await CronLogModel.findAll({
|
||||||
|
where: {
|
||||||
|
start_time: { [Op.between]: [dayStart, dayEnd] },
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
result.push({
|
||||||
|
date,
|
||||||
|
count: logs.length,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
public async topDuration(limit = 5) {
|
||||||
|
const todayStart = dayjs().startOf('day').unix();
|
||||||
|
const todayEnd = dayjs().endOf('day').unix();
|
||||||
|
|
||||||
|
const logs = await CronLogModel.findAll({
|
||||||
|
where: {
|
||||||
|
start_time: { [Op.between]: [todayStart, todayEnd] },
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const grouped: Record<
|
||||||
|
number,
|
||||||
|
{ cron_id: number; cron_name: string; durations: number[] }
|
||||||
|
> = {};
|
||||||
|
|
||||||
|
for (const log of logs) {
|
||||||
|
if (!grouped[log.cron_id]) {
|
||||||
|
grouped[log.cron_id] = {
|
||||||
|
cron_id: log.cron_id,
|
||||||
|
cron_name: log.cron_name,
|
||||||
|
durations: [],
|
||||||
|
};
|
||||||
|
}
|
||||||
|
grouped[log.cron_id].durations.push(log.duration);
|
||||||
|
}
|
||||||
|
|
||||||
|
const result = Object.values(grouped)
|
||||||
|
.map((g) => {
|
||||||
|
const avgDuration = Math.round(
|
||||||
|
g.durations.reduce((a, b) => a + b, 0) / g.durations.length,
|
||||||
|
);
|
||||||
|
const maxDuration = Math.max(...g.durations);
|
||||||
|
return {
|
||||||
|
cron_id: g.cron_id,
|
||||||
|
cron_name: g.cron_name,
|
||||||
|
count: g.durations.length,
|
||||||
|
avgDuration,
|
||||||
|
maxDuration,
|
||||||
|
};
|
||||||
|
})
|
||||||
|
.sort((a, b) => b.avgDuration - a.avgDuration)
|
||||||
|
.slice(0, limit);
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
public async topCount(limit = 5) {
|
||||||
|
const todayStart = dayjs().startOf('day').unix();
|
||||||
|
const todayEnd = dayjs().endOf('day').unix();
|
||||||
|
|
||||||
|
const logs = await CronLogModel.findAll({
|
||||||
|
where: {
|
||||||
|
start_time: { [Op.between]: [todayStart, todayEnd] },
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const grouped: Record<
|
||||||
|
number,
|
||||||
|
{ cron_id: number; cron_name: string; durations: number[] }
|
||||||
|
> = {};
|
||||||
|
|
||||||
|
for (const log of logs) {
|
||||||
|
if (!grouped[log.cron_id]) {
|
||||||
|
grouped[log.cron_id] = {
|
||||||
|
cron_id: log.cron_id,
|
||||||
|
cron_name: log.cron_name,
|
||||||
|
durations: [],
|
||||||
|
};
|
||||||
|
}
|
||||||
|
grouped[log.cron_id].durations.push(log.duration);
|
||||||
|
}
|
||||||
|
|
||||||
|
const result = Object.values(grouped)
|
||||||
|
.map((g) => {
|
||||||
|
const avgDuration = Math.round(
|
||||||
|
g.durations.reduce((a, b) => a + b, 0) / g.durations.length,
|
||||||
|
);
|
||||||
|
return {
|
||||||
|
cron_id: g.cron_id,
|
||||||
|
cron_name: g.cron_name,
|
||||||
|
count: g.durations.length,
|
||||||
|
avgDuration,
|
||||||
|
};
|
||||||
|
})
|
||||||
|
.sort((a, b) => b.count - a.count)
|
||||||
|
.slice(0, limit);
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -1,5 +1,5 @@
|
||||||
import intl from 'react-intl-universal';
|
import intl from 'react-intl-universal';
|
||||||
import { SettingOutlined } from '@ant-design/icons';
|
import { BarChartOutlined, SettingOutlined } from '@ant-design/icons';
|
||||||
import IconFont from '@/components/iconfont';
|
import IconFont from '@/components/iconfont';
|
||||||
import { BasicLayoutProps } from '@ant-design/pro-layout';
|
import { BasicLayoutProps } from '@ant-design/pro-layout';
|
||||||
|
|
||||||
|
|
@ -30,6 +30,12 @@ export default {
|
||||||
icon: <IconFont type="ql-icon-crontab" />,
|
icon: <IconFont type="ql-icon-crontab" />,
|
||||||
component: '@/pages/crontab/index',
|
component: '@/pages/crontab/index',
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
path: '/statistics',
|
||||||
|
name: intl.get('统计面板'),
|
||||||
|
icon: <BarChartOutlined />,
|
||||||
|
component: '@/pages/statistics/index',
|
||||||
|
},
|
||||||
{
|
{
|
||||||
path: '/subscription',
|
path: '/subscription',
|
||||||
name: intl.get('订阅管理'),
|
name: intl.get('订阅管理'),
|
||||||
|
|
|
||||||
|
|
@ -18,6 +18,25 @@
|
||||||
"青龙": "Qinglong",
|
"青龙": "Qinglong",
|
||||||
"返回首页": "Return to Home",
|
"返回首页": "Return to Home",
|
||||||
"保存": "Save",
|
"保存": "Save",
|
||||||
|
"统计面板": "Statistics",
|
||||||
|
"总体概览": "Overview",
|
||||||
|
"总任务数量": "Total Tasks",
|
||||||
|
"启用任务数": "Enabled Tasks",
|
||||||
|
"禁用任务数": "Disabled Tasks",
|
||||||
|
"今日总执行次数": "Today's Executions",
|
||||||
|
"今日平均耗时(秒)": "Today's Avg Duration (s)",
|
||||||
|
"近7日执行趋势": "7-Day Execution Trend",
|
||||||
|
"今日平均耗时 Top 5": "Top 5 Slowest Today",
|
||||||
|
"今日执行次数 Top 5": "Top 5 Most Frequent Today",
|
||||||
|
"排名": "Rank",
|
||||||
|
"任务名称": "Task Name",
|
||||||
|
"平均耗时(秒)": "Avg Duration (s)",
|
||||||
|
"最长单次(秒)": "Max Duration (s)",
|
||||||
|
"今日执行次数": "Today's Count",
|
||||||
|
"今日暂无执行记录": "No execution records today",
|
||||||
|
"暂无数据": "No data",
|
||||||
|
"次": "times",
|
||||||
|
"刷新": "Refresh",
|
||||||
"日志": "Log",
|
"日志": "Log",
|
||||||
"脚本": "Script",
|
"脚本": "Script",
|
||||||
"确认保存文件": "Confirm to Save File",
|
"确认保存文件": "Confirm to Save File",
|
||||||
|
|
|
||||||
|
|
@ -18,6 +18,25 @@
|
||||||
"青龙": "青龙",
|
"青龙": "青龙",
|
||||||
"返回首页": "返回首页",
|
"返回首页": "返回首页",
|
||||||
"保存": "保存",
|
"保存": "保存",
|
||||||
|
"统计面板": "统计面板",
|
||||||
|
"总体概览": "总体概览",
|
||||||
|
"总任务数量": "总任务数量",
|
||||||
|
"启用任务数": "启用任务数",
|
||||||
|
"禁用任务数": "禁用任务数",
|
||||||
|
"今日总执行次数": "今日总执行次数",
|
||||||
|
"今日平均耗时(秒)": "今日平均耗时(秒)",
|
||||||
|
"近7日执行趋势": "近7日执行趋势",
|
||||||
|
"今日平均耗时 Top 5": "今日平均耗时 Top 5",
|
||||||
|
"今日执行次数 Top 5": "今日执行次数 Top 5",
|
||||||
|
"排名": "排名",
|
||||||
|
"任务名称": "任务名称",
|
||||||
|
"平均耗时(秒)": "平均耗时(秒)",
|
||||||
|
"最长单次(秒)": "最长单次(秒)",
|
||||||
|
"今日执行次数": "今日执行次数",
|
||||||
|
"今日暂无执行记录": "今日暂无执行记录",
|
||||||
|
"暂无数据": "暂无数据",
|
||||||
|
"次": "次",
|
||||||
|
"刷新": "刷新",
|
||||||
"日志": "日志",
|
"日志": "日志",
|
||||||
"脚本": "脚本",
|
"脚本": "脚本",
|
||||||
"确认保存文件": "确认保存文件",
|
"确认保存文件": "确认保存文件",
|
||||||
|
|
|
||||||
17
src/pages/statistics/index.less
Normal file
17
src/pages/statistics/index.less
Normal file
|
|
@ -0,0 +1,17 @@
|
||||||
|
.stats-section {
|
||||||
|
margin-bottom: 16px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.trend-chart-wrapper {
|
||||||
|
width: 100%;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
.trend-chart-empty {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
height: 200px;
|
||||||
|
color: #999;
|
||||||
|
font-size: 14px;
|
||||||
|
}
|
||||||
402
src/pages/statistics/index.tsx
Normal file
402
src/pages/statistics/index.tsx
Normal file
|
|
@ -0,0 +1,402 @@
|
||||||
|
import { SharedContext } from '@/layouts';
|
||||||
|
import config from '@/utils/config';
|
||||||
|
import { request } from '@/utils/http';
|
||||||
|
import { BarChartOutlined, ReloadOutlined } from '@ant-design/icons';
|
||||||
|
import { PageContainer } from '@ant-design/pro-layout';
|
||||||
|
import { useOutletContext } from '@umijs/max';
|
||||||
|
import {
|
||||||
|
Button,
|
||||||
|
Card,
|
||||||
|
Col,
|
||||||
|
Row,
|
||||||
|
Statistic,
|
||||||
|
Table,
|
||||||
|
Tooltip,
|
||||||
|
Typography,
|
||||||
|
} from 'antd';
|
||||||
|
import { ColumnProps } from 'antd/lib/table';
|
||||||
|
import React, { useEffect, useState } from 'react';
|
||||||
|
import intl from 'react-intl-universal';
|
||||||
|
import './index.less';
|
||||||
|
|
||||||
|
const { Title } = Typography;
|
||||||
|
|
||||||
|
interface StatsData {
|
||||||
|
total: number;
|
||||||
|
enabled: number;
|
||||||
|
disabled: number;
|
||||||
|
today: {
|
||||||
|
count: number;
|
||||||
|
avgDuration: number;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
interface TrendItem {
|
||||||
|
date: string;
|
||||||
|
count: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface TopDurationItem {
|
||||||
|
cron_id: number;
|
||||||
|
cron_name: string;
|
||||||
|
count: number;
|
||||||
|
avgDuration: number;
|
||||||
|
maxDuration: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface TopCountItem {
|
||||||
|
cron_id: number;
|
||||||
|
cron_name: string;
|
||||||
|
count: number;
|
||||||
|
avgDuration: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
const TrendChart = ({ data }: { data: TrendItem[] }) => {
|
||||||
|
if (!data || data.length === 0) {
|
||||||
|
return (
|
||||||
|
<div className="trend-chart-empty">
|
||||||
|
{intl.get('暂无数据')}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const width = 600;
|
||||||
|
const height = 200;
|
||||||
|
const paddingLeft = 40;
|
||||||
|
const paddingRight = 20;
|
||||||
|
const paddingTop = 20;
|
||||||
|
const paddingBottom = 40;
|
||||||
|
|
||||||
|
const chartWidth = width - paddingLeft - paddingRight;
|
||||||
|
const chartHeight = height - paddingTop - paddingBottom;
|
||||||
|
|
||||||
|
const maxCount = Math.max(...data.map((d) => d.count), 1);
|
||||||
|
|
||||||
|
const points = data.map((d, i) => ({
|
||||||
|
x: paddingLeft + (i / (data.length - 1)) * chartWidth,
|
||||||
|
y: paddingTop + chartHeight - (d.count / maxCount) * chartHeight,
|
||||||
|
...d,
|
||||||
|
}));
|
||||||
|
|
||||||
|
const pathD = points
|
||||||
|
.map((p, i) => `${i === 0 ? 'M' : 'L'} ${p.x.toFixed(1)} ${p.y.toFixed(1)}`)
|
||||||
|
.join(' ');
|
||||||
|
|
||||||
|
const areaD =
|
||||||
|
pathD +
|
||||||
|
` L ${points[points.length - 1].x.toFixed(1)} ${(paddingTop + chartHeight).toFixed(1)}` +
|
||||||
|
` L ${points[0].x.toFixed(1)} ${(paddingTop + chartHeight).toFixed(1)} Z`;
|
||||||
|
|
||||||
|
const yTicks = [0, Math.ceil(maxCount / 2), maxCount];
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="trend-chart-wrapper">
|
||||||
|
<svg
|
||||||
|
viewBox={`0 0 ${width} ${height}`}
|
||||||
|
preserveAspectRatio="xMidYMid meet"
|
||||||
|
style={{ width: '100%', height: 200 }}
|
||||||
|
>
|
||||||
|
{/* Grid lines */}
|
||||||
|
{yTicks.map((tick) => {
|
||||||
|
const y =
|
||||||
|
paddingTop + chartHeight - (tick / maxCount) * chartHeight;
|
||||||
|
return (
|
||||||
|
<g key={tick}>
|
||||||
|
<line
|
||||||
|
x1={paddingLeft}
|
||||||
|
y1={y}
|
||||||
|
x2={paddingLeft + chartWidth}
|
||||||
|
y2={y}
|
||||||
|
stroke="#f0f0f0"
|
||||||
|
strokeWidth={1}
|
||||||
|
/>
|
||||||
|
<text
|
||||||
|
x={paddingLeft - 6}
|
||||||
|
y={y + 4}
|
||||||
|
textAnchor="end"
|
||||||
|
fontSize={10}
|
||||||
|
fill="#999"
|
||||||
|
>
|
||||||
|
{tick}
|
||||||
|
</text>
|
||||||
|
</g>
|
||||||
|
);
|
||||||
|
})}
|
||||||
|
|
||||||
|
{/* Area fill */}
|
||||||
|
<path d={areaD} fill="rgba(24, 144, 255, 0.1)" />
|
||||||
|
|
||||||
|
{/* Line */}
|
||||||
|
<path
|
||||||
|
d={pathD}
|
||||||
|
fill="none"
|
||||||
|
stroke="#1890ff"
|
||||||
|
strokeWidth={2}
|
||||||
|
strokeLinejoin="round"
|
||||||
|
strokeLinecap="round"
|
||||||
|
/>
|
||||||
|
|
||||||
|
{/* Points */}
|
||||||
|
{points.map((p, i) => (
|
||||||
|
<Tooltip
|
||||||
|
key={i}
|
||||||
|
title={`${p.date}: ${p.count} ${intl.get('次')}`}
|
||||||
|
>
|
||||||
|
<circle
|
||||||
|
cx={p.x}
|
||||||
|
cy={p.y}
|
||||||
|
r={4}
|
||||||
|
fill="#1890ff"
|
||||||
|
stroke="#fff"
|
||||||
|
strokeWidth={2}
|
||||||
|
style={{ cursor: 'pointer' }}
|
||||||
|
/>
|
||||||
|
</Tooltip>
|
||||||
|
))}
|
||||||
|
|
||||||
|
{/* X axis labels */}
|
||||||
|
{points.map((p, i) => (
|
||||||
|
<text
|
||||||
|
key={i}
|
||||||
|
x={p.x}
|
||||||
|
y={height - 8}
|
||||||
|
textAnchor="middle"
|
||||||
|
fontSize={10}
|
||||||
|
fill="#999"
|
||||||
|
>
|
||||||
|
{p.date}
|
||||||
|
</text>
|
||||||
|
))}
|
||||||
|
|
||||||
|
{/* Axes */}
|
||||||
|
<line
|
||||||
|
x1={paddingLeft}
|
||||||
|
y1={paddingTop}
|
||||||
|
x2={paddingLeft}
|
||||||
|
y2={paddingTop + chartHeight}
|
||||||
|
stroke="#e8e8e8"
|
||||||
|
strokeWidth={1}
|
||||||
|
/>
|
||||||
|
<line
|
||||||
|
x1={paddingLeft}
|
||||||
|
y1={paddingTop + chartHeight}
|
||||||
|
x2={paddingLeft + chartWidth}
|
||||||
|
y2={paddingTop + chartHeight}
|
||||||
|
stroke="#e8e8e8"
|
||||||
|
strokeWidth={1}
|
||||||
|
/>
|
||||||
|
</svg>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
const Statistics = () => {
|
||||||
|
const { headerStyle, isPhone } = useOutletContext<SharedContext>();
|
||||||
|
const [stats, setStats] = useState<StatsData | null>(null);
|
||||||
|
const [trend, setTrend] = useState<TrendItem[]>([]);
|
||||||
|
const [topDuration, setTopDuration] = useState<TopDurationItem[]>([]);
|
||||||
|
const [topCount, setTopCount] = useState<TopCountItem[]>([]);
|
||||||
|
const [loading, setLoading] = useState(true);
|
||||||
|
|
||||||
|
const loadAll = async () => {
|
||||||
|
setLoading(true);
|
||||||
|
try {
|
||||||
|
const [
|
||||||
|
statsRes,
|
||||||
|
trendRes,
|
||||||
|
topDurationRes,
|
||||||
|
topCountRes,
|
||||||
|
] = await Promise.all([
|
||||||
|
request.get(`${config.apiPrefix}crons/stats`),
|
||||||
|
request.get(`${config.apiPrefix}crons/stats/trend`),
|
||||||
|
request.get(`${config.apiPrefix}crons/stats/top-duration`),
|
||||||
|
request.get(`${config.apiPrefix}crons/stats/top-count`),
|
||||||
|
]);
|
||||||
|
if (statsRes.code === 200) setStats(statsRes.data);
|
||||||
|
if (trendRes.code === 200) setTrend(trendRes.data);
|
||||||
|
if (topDurationRes.code === 200) setTopDuration(topDurationRes.data);
|
||||||
|
if (topCountRes.code === 200) setTopCount(topCountRes.data);
|
||||||
|
} catch (e) {
|
||||||
|
console.error(e);
|
||||||
|
} finally {
|
||||||
|
setLoading(false);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
loadAll();
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const topDurationColumns: ColumnProps<TopDurationItem>[] = [
|
||||||
|
{
|
||||||
|
title: intl.get('排名'),
|
||||||
|
key: 'rank',
|
||||||
|
width: 60,
|
||||||
|
render: (_: any, __: any, index: number) => index + 1,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: intl.get('任务名称'),
|
||||||
|
dataIndex: 'cron_name',
|
||||||
|
key: 'cron_name',
|
||||||
|
ellipsis: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: intl.get('平均耗时(秒)'),
|
||||||
|
dataIndex: 'avgDuration',
|
||||||
|
key: 'avgDuration',
|
||||||
|
width: 120,
|
||||||
|
render: (v: number) => `${v}s`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: intl.get('最长单次(秒)'),
|
||||||
|
dataIndex: 'maxDuration',
|
||||||
|
key: 'maxDuration',
|
||||||
|
width: 120,
|
||||||
|
render: (v: number) => `${v}s`,
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
const topCountColumns: ColumnProps<TopCountItem>[] = [
|
||||||
|
{
|
||||||
|
title: intl.get('排名'),
|
||||||
|
key: 'rank',
|
||||||
|
width: 60,
|
||||||
|
render: (_: any, __: any, index: number) => index + 1,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: intl.get('任务名称'),
|
||||||
|
dataIndex: 'cron_name',
|
||||||
|
key: 'cron_name',
|
||||||
|
ellipsis: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: intl.get('今日执行次数'),
|
||||||
|
dataIndex: 'count',
|
||||||
|
key: 'count',
|
||||||
|
width: 120,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: intl.get('平均耗时(秒)'),
|
||||||
|
dataIndex: 'avgDuration',
|
||||||
|
key: 'avgDuration',
|
||||||
|
width: 120,
|
||||||
|
render: (v: number) => `${v}s`,
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
return (
|
||||||
|
<PageContainer
|
||||||
|
header={{
|
||||||
|
style: headerStyle,
|
||||||
|
}}
|
||||||
|
title={
|
||||||
|
<span>
|
||||||
|
<BarChartOutlined style={{ marginRight: 8 }} />
|
||||||
|
{intl.get('统计面板')}
|
||||||
|
</span>
|
||||||
|
}
|
||||||
|
extra={[
|
||||||
|
<Button
|
||||||
|
key="refresh"
|
||||||
|
icon={<ReloadOutlined />}
|
||||||
|
loading={loading}
|
||||||
|
onClick={loadAll}
|
||||||
|
>
|
||||||
|
{intl.get('刷新')}
|
||||||
|
</Button>,
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
{/* Section 1: Overview Cards */}
|
||||||
|
<Card
|
||||||
|
className="stats-section"
|
||||||
|
title={intl.get('总体概览')}
|
||||||
|
loading={loading}
|
||||||
|
>
|
||||||
|
<Row gutter={[16, 16]}>
|
||||||
|
<Col xs={12} sm={8} md={6} lg={4}>
|
||||||
|
<Statistic
|
||||||
|
title={intl.get('总任务数量')}
|
||||||
|
value={stats?.total ?? '-'}
|
||||||
|
/>
|
||||||
|
</Col>
|
||||||
|
<Col xs={12} sm={8} md={6} lg={4}>
|
||||||
|
<Statistic
|
||||||
|
title={intl.get('启用任务数')}
|
||||||
|
value={stats?.enabled ?? '-'}
|
||||||
|
valueStyle={{ color: '#52c41a' }}
|
||||||
|
/>
|
||||||
|
</Col>
|
||||||
|
<Col xs={12} sm={8} md={6} lg={4}>
|
||||||
|
<Statistic
|
||||||
|
title={intl.get('禁用任务数')}
|
||||||
|
value={stats?.disabled ?? '-'}
|
||||||
|
valueStyle={{ color: '#d9d9d9' }}
|
||||||
|
/>
|
||||||
|
</Col>
|
||||||
|
<Col xs={12} sm={8} md={6} lg={4}>
|
||||||
|
<Statistic
|
||||||
|
title={intl.get('今日总执行次数')}
|
||||||
|
value={stats?.today?.count ?? '-'}
|
||||||
|
valueStyle={{ color: '#1890ff' }}
|
||||||
|
/>
|
||||||
|
</Col>
|
||||||
|
<Col xs={12} sm={8} md={6} lg={4}>
|
||||||
|
<Statistic
|
||||||
|
title={intl.get('今日平均耗时(秒)')}
|
||||||
|
value={stats?.today?.avgDuration ?? '-'}
|
||||||
|
suffix="s"
|
||||||
|
valueStyle={{ color: '#faad14' }}
|
||||||
|
/>
|
||||||
|
</Col>
|
||||||
|
</Row>
|
||||||
|
</Card>
|
||||||
|
|
||||||
|
{/* Section 2: 7-day Trend */}
|
||||||
|
<Card
|
||||||
|
className="stats-section"
|
||||||
|
title={intl.get('近7日执行趋势')}
|
||||||
|
loading={loading}
|
||||||
|
>
|
||||||
|
<TrendChart data={trend} />
|
||||||
|
</Card>
|
||||||
|
|
||||||
|
{/* Section 3 & 4: Top Tables */}
|
||||||
|
<Row gutter={[16, 16]}>
|
||||||
|
<Col xs={24} lg={12}>
|
||||||
|
<Card
|
||||||
|
className="stats-section"
|
||||||
|
title={intl.get('今日平均耗时 Top 5')}
|
||||||
|
loading={loading}
|
||||||
|
>
|
||||||
|
<Table
|
||||||
|
dataSource={topDuration}
|
||||||
|
columns={topDurationColumns}
|
||||||
|
rowKey="cron_id"
|
||||||
|
pagination={false}
|
||||||
|
size="small"
|
||||||
|
locale={{ emptyText: intl.get('今日暂无执行记录') }}
|
||||||
|
/>
|
||||||
|
</Card>
|
||||||
|
</Col>
|
||||||
|
<Col xs={24} lg={12}>
|
||||||
|
<Card
|
||||||
|
className="stats-section"
|
||||||
|
title={intl.get('今日执行次数 Top 5')}
|
||||||
|
loading={loading}
|
||||||
|
>
|
||||||
|
<Table
|
||||||
|
dataSource={topCount}
|
||||||
|
columns={topCountColumns}
|
||||||
|
rowKey="cron_id"
|
||||||
|
pagination={false}
|
||||||
|
size="small"
|
||||||
|
locale={{ emptyText: intl.get('今日暂无执行记录') }}
|
||||||
|
/>
|
||||||
|
</Card>
|
||||||
|
</Col>
|
||||||
|
</Row>
|
||||||
|
</PageContainer>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default Statistics;
|
||||||
|
|
@ -504,6 +504,7 @@ export default {
|
||||||
'/login': intl.get('登录'),
|
'/login': intl.get('登录'),
|
||||||
'/initialization': intl.get('初始化'),
|
'/initialization': intl.get('初始化'),
|
||||||
'/crontab': intl.get('定时任务'),
|
'/crontab': intl.get('定时任务'),
|
||||||
|
'/statistics': intl.get('统计面板'),
|
||||||
'/env': intl.get('环境变量'),
|
'/env': intl.get('环境变量'),
|
||||||
'/subscription': intl.get('订阅管理'),
|
'/subscription': intl.get('订阅管理'),
|
||||||
'/config': intl.get('配置文件'),
|
'/config': intl.get('配置文件'),
|
||||||
|
|
|
||||||
Loading…
Reference in New Issue
Block a user