mirror of
https://github.com/DrizzleTime/Foxel.git
synced 2026-05-12 02:20:28 +08:00
156 lines
4.7 KiB
TypeScript
156 lines
4.7 KiB
TypeScript
import { memo, useState, useEffect, useCallback } from 'react';
|
|
import { Table, message, Tag, Input, Select, Button, Space, Modal, DatePicker } from 'antd';
|
|
import PageCard from '../components/PageCard';
|
|
import { logsApi, type LogItem, type PaginatedLogs } from '../api/logs';
|
|
import { format, formatISO } from 'date-fns';
|
|
|
|
const { RangePicker } = DatePicker;
|
|
|
|
const LOG_LEVELS = ['API', 'INFO', 'WARNING', 'ERROR'];
|
|
|
|
const LogsPage = memo(function LogsPage() {
|
|
const [loading, setLoading] = useState(false);
|
|
const [data, setData] = useState<PaginatedLogs | null>(null);
|
|
const [filters, setFilters] = useState({
|
|
page: 1,
|
|
page_size: 20,
|
|
level: '',
|
|
source: '',
|
|
start_time: '',
|
|
end_time: '',
|
|
});
|
|
const [selectedLog, setSelectedLog] = useState<LogItem | null>(null);
|
|
|
|
const fetchList = useCallback(async () => {
|
|
setLoading(true);
|
|
try {
|
|
const params = { ...filters };
|
|
if (!params.start_time) delete (params as any).start_time;
|
|
if (!params.end_time) delete (params as any).end_time;
|
|
|
|
const res = await logsApi.list(params);
|
|
setData(res);
|
|
} catch (e: any) {
|
|
message.error(e.message || '加载失败');
|
|
} finally {
|
|
setLoading(false);
|
|
}
|
|
}, [filters]);
|
|
|
|
useEffect(() => {
|
|
fetchList();
|
|
}, [fetchList]);
|
|
|
|
const handleClearLogs = () => {
|
|
Modal.confirm({
|
|
title: '确认清理日志?',
|
|
content: '该操作将删除选定时间范围内的所有日志,且不可恢复。',
|
|
onOk: async () => {
|
|
try {
|
|
const params = { start_time: filters.start_time, end_time: filters.end_time };
|
|
if (!params.start_time) delete (params as any).start_time;
|
|
if (!params.end_time) delete (params as any).end_time;
|
|
const res = await logsApi.clear(params);
|
|
message.success(`成功清理 ${res.deleted_count} 条日志`);
|
|
fetchList();
|
|
} catch (e: any) {
|
|
message.error(e.message || '清理失败');
|
|
}
|
|
},
|
|
});
|
|
};
|
|
|
|
const columns = [
|
|
{
|
|
title: '时间',
|
|
dataIndex: 'timestamp',
|
|
width: 180,
|
|
render: (ts: string) => format(new Date(ts), 'yyyy-MM-dd HH:mm:ss'),
|
|
},
|
|
{
|
|
title: '级别',
|
|
dataIndex: 'level',
|
|
width: 100,
|
|
render: (level: string) => {
|
|
const color = { API: 'blue', INFO: 'green', WARNING: 'orange', ERROR: 'red' }[level] || 'default';
|
|
return <Tag color={color}>{level}</Tag>;
|
|
},
|
|
},
|
|
{ title: '来源', dataIndex: 'source', width: 180 },
|
|
{ title: '消息', dataIndex: 'message', ellipsis: true },
|
|
{
|
|
title: '操作',
|
|
width: 100,
|
|
render: (_: any, rec: LogItem) => (
|
|
<Button size="small" onClick={() => setSelectedLog(rec)}>详情</Button>
|
|
),
|
|
},
|
|
];
|
|
|
|
return (
|
|
<PageCard
|
|
title="系统日志"
|
|
extra={
|
|
<Space>
|
|
<RangePicker
|
|
showTime
|
|
onChange={dates => {
|
|
setFilters(f => ({
|
|
...f,
|
|
start_time: dates?.[0] ? formatISO(dates[0].toDate()) : '',
|
|
end_time: dates?.[1] ? formatISO(dates[1].toDate()) : '',
|
|
page: 1,
|
|
}));
|
|
}}
|
|
/>
|
|
<Select
|
|
style={{ width: 120 }}
|
|
placeholder="级别"
|
|
allowClear
|
|
value={filters.level || undefined}
|
|
onChange={level => setFilters(f => ({ ...f, level: level || '', page: 1 }))}
|
|
options={LOG_LEVELS.map(l => ({ value: l, label: l }))}
|
|
/>
|
|
<Input.Search
|
|
style={{ width: 240 }}
|
|
placeholder="搜索来源"
|
|
onSearch={source => setFilters(f => ({ ...f, source, page: 1 }))}
|
|
allowClear
|
|
/>
|
|
<Button onClick={fetchList} loading={loading}>刷新</Button>
|
|
<Button danger onClick={handleClearLogs}>清理</Button>
|
|
</Space>
|
|
}
|
|
>
|
|
<Table
|
|
rowKey="id"
|
|
dataSource={data?.items}
|
|
columns={columns}
|
|
loading={loading}
|
|
pagination={{
|
|
current: filters.page,
|
|
pageSize: filters.page_size,
|
|
total: data?.total,
|
|
showSizeChanger: true,
|
|
onChange: (page, pageSize) => setFilters(f => ({ ...f, page, page_size: pageSize })),
|
|
}}
|
|
/>
|
|
<Modal
|
|
title="日志详情"
|
|
open={!!selectedLog}
|
|
onCancel={() => setSelectedLog(null)}
|
|
footer={null}
|
|
width={800}
|
|
>
|
|
{selectedLog && (
|
|
<pre style={{ maxHeight: '60vh', overflow: 'auto', background: 'var(--ant-color-fill-tertiary, #f5f5f5)', padding: 12 }}>
|
|
{JSON.stringify(selectedLog.details, null, 2)}
|
|
</pre>
|
|
)}
|
|
</Modal>
|
|
</PageCard>
|
|
);
|
|
});
|
|
|
|
export default LogsPage;
|