hwh
2024-08-23 6c000a9f933f43a072d52790efc22bbcc8238eee
日志文件
1个文件已修改
4个文件已添加
881 ■■■■■ 已修改文件
.gitignore 2 ●●● 补丁 | 查看 | 原始文档 | blame | 历史
Web/src/views/system/log/difflog/index.vue 147 ●●●●● 补丁 | 查看 | 原始文档 | blame | 历史
Web/src/views/system/log/exlog/index.vue 279 ●●●●● 补丁 | 查看 | 原始文档 | blame | 历史
Web/src/views/system/log/oplog/index.vue 270 ●●●●● 补丁 | 查看 | 原始文档 | blame | 历史
Web/src/views/system/log/vislog/index.vue 183 ●●●●● 补丁 | 查看 | 原始文档 | blame | 历史
.gitignore
@@ -141,7 +141,6 @@
# AxoCover is a Code Coverage Tool
.axoCover/*
!.axoCover/settings.json
# Coverlet is a free, cross platform Code Coverage Tool
coverage*.json
coverage*.xml
@@ -200,6 +199,7 @@
**/[Pp]ackages/*
# except build/, which is used as an MSBuild target.
!**/[Pp]ackages/build/
!**/system/log/
# Uncomment if necessary however generally it will be regenerated when needed
#!**/[Pp]ackages/repositories.config
# NuGet v3's project.json files produces more ignorable files
Web/src/views/system/log/difflog/index.vue
New file
@@ -0,0 +1,147 @@
<template>
    <div class="sys-difflog-container">
        <el-card shadow="hover" :body-style="{ paddingBottom: '0' }">
            <el-form :model="state.queryParams" ref="queryForm" :inline="true">
                <el-form-item label="开始时间">
                    <el-date-picker v-model="state.queryParams.startTime" type="datetime" placeholder="开始时间" value-format="YYYY-MM-DD HH:mm:ss" :shortcuts="shortcuts" />
                </el-form-item>
                <el-form-item label="结束时间">
                    <el-date-picker v-model="state.queryParams.endTime" type="datetime" placeholder="结束时间" value-format="YYYY-MM-DD HH:mm:ss" :shortcuts="shortcuts" />
                </el-form-item>
                <el-form-item>
                    <el-button-group>
                        <el-button type="primary" icon="ele-Search" @click="handleQuery" v-auth="'sysDifflog:page'"> 查询 </el-button>
                        <el-button icon="ele-Refresh" @click="resetQuery"> 重置 </el-button>
                    </el-button-group>
                </el-form-item>
                <el-form-item>
                    <el-button icon="ele-DeleteFilled" type="danger" @click="clearLog" v-auth="'sysDifflog:clear'"> 清空 </el-button>
                </el-form-item>
            </el-form>
        </el-card>
        <el-card class="full-table" shadow="hover" style="margin-top: 5px">
            <el-table :data="state.logData" style="width: 100%" v-loading="state.loading" border>
                <el-table-column type="index" label="序号" width="55" align="center" />
                <el-table-column prop="diffType" label="差异操作" header-align="center" show-overflow-tooltip />
                <el-table-column prop="sql" label="Sql语句" header-align="center" show-overflow-tooltip />
                <el-table-column prop="parameters" label="参数" header-align="center" show-overflow-tooltip />
                <el-table-column prop="elapsed" label="耗时(ms)" header-align="center" show-overflow-tooltip />
                <el-table-column prop="message" label="日志消息" header-align="center" show-overflow-tooltip />
                <el-table-column prop="beforeData" label="操作前记录" header-align="center" show-overflow-tooltip />
                <el-table-column prop="afterData" label="操作后记录" header-align="center" show-overflow-tooltip />
                <el-table-column prop="businessData" label="业务对象" header-align="center" show-overflow-tooltip />
                <el-table-column prop="createTime" label="操作时间" align="center" show-overflow-tooltip />
            </el-table>
            <el-pagination
                v-model:currentPage="state.tableParams.page"
                v-model:page-size="state.tableParams.pageSize"
                :total="state.tableParams.total"
                :page-sizes="[10, 20, 50, 100]"
                size="small"
                background
                @size-change="handleSizeChange"
                @current-change="handleCurrentChange"
                layout="total, sizes, prev, pager, next, jumper"
            />
        </el-card>
    </div>
</template>
<script lang="ts" setup name="sysDiffLog">
import { onMounted, reactive } from 'vue';
import { ElMessage } from 'element-plus';
import { getAPI } from '/@/utils/axios-utils';
import { SysLogDiffApi } from '/@/api-services/api';
import { SysLogDiff } from '/@/api-services/models';
const state = reactive({
    loading: false,
    queryParams: {
        startTime: undefined,
        endTime: undefined,
    },
    tableParams: {
        page: 1,
        pageSize: 10,
        total: 0 as any,
    },
    logData: [] as Array<SysLogDiff>,
});
onMounted(async () => {
    handleQuery();
});
// 查询操作
const handleQuery = async () => {
    if (state.queryParams.startTime == null) state.queryParams.startTime = undefined;
    if (state.queryParams.endTime == null) state.queryParams.endTime = undefined;
    state.loading = true;
    let params = Object.assign(state.queryParams, state.tableParams);
    var res = await getAPI(SysLogDiffApi).apiSysLogDiffPagePost(params);
    state.logData = res.data.result?.items ?? [];
    state.tableParams.total = res.data.result?.total;
    state.loading = false;
};
// 重置操作
const resetQuery = () => {
    state.queryParams.startTime = undefined;
    state.queryParams.endTime = undefined;
    handleQuery();
};
// 清空日志
const clearLog = async () => {
    state.loading = true;
    await getAPI(SysLogDiffApi).apiSysLogDiffClearPost();
    state.loading = false;
    ElMessage.success('清空成功');
    handleQuery();
};
// 改变页面容量
const handleSizeChange = (val: number) => {
    state.tableParams.pageSize = val;
    handleQuery();
};
// 改变页码序号
const handleCurrentChange = (val: number) => {
    state.tableParams.page = val;
    handleQuery();
};
const shortcuts = [
    {
        text: '今天',
        value: new Date(),
    },
    {
        text: '昨天',
        value: () => {
            const date = new Date();
            date.setTime(date.getTime() - 3600 * 1000 * 24);
            return date;
        },
    },
    {
        text: '上周',
        value: () => {
            const date = new Date();
            date.setTime(date.getTime() - 3600 * 1000 * 24 * 7);
            return date;
        },
    },
];
</script>
<style lang="scss" scoped>
.el-popper {
    max-width: 60%;
}
</style>
Web/src/views/system/log/exlog/index.vue
New file
@@ -0,0 +1,279 @@
<template>
    <div class="sys-exlog-container" v-loading="state.loading">
        <el-card shadow="hover" :body-style="{ paddingBottom: '0' }">
            <el-form :model="state.queryParams" ref="queryForm" :inline="true">
                <el-form-item label="开始时间">
                    <el-date-picker v-model="state.queryParams.startTime" type="datetime" placeholder="开始时间" value-format="YYYY-MM-DD HH:mm:ss" :shortcuts="shortcuts" />
                </el-form-item>
                <el-form-item label="结束时间">
                    <el-date-picker v-model="state.queryParams.endTime" type="datetime" placeholder="结束时间" value-format="YYYY-MM-DD HH:mm:ss" :shortcuts="shortcuts" />
                </el-form-item>
                <el-form-item label="模块名称">
                    <el-input v-model="state.queryParams.controllerName" placeholder="模块名称" clearable />
                </el-form-item>
                <el-form-item label="方法名称">
                    <el-input v-model="state.queryParams.actionName" placeholder="方法名称" clearable />
                </el-form-item>
                <el-form-item label="账号名称">
                    <el-input v-model="state.queryParams.account" placeholder="账号名称" clearable />
                </el-form-item>
                <el-form-item label="状态">
                    <el-select v-model="state.queryParams.status" placeholder="状态" clearable>
                        <el-option label="成功" :value="200" />
                        <el-option label="失败" :value="400" />
                    </el-select>
                </el-form-item>
                <el-form-item label="耗时">
                    <el-input v-model="state.queryParams.elapsed" placeholder="耗时>?MS" clearable />
                </el-form-item>
                <el-form-item label="IP地址">
                    <el-input v-model="state.queryParams.remoteIp" placeholder="IP地址" clearable />
                </el-form-item>
                <el-form-item>
                    <el-button-group>
                        <el-button type="primary" icon="ele-Search" @click="handleQuery" v-auth="'sysExlog:page'"> 查询 </el-button>
                        <el-button icon="ele-Refresh" @click="resetQuery"> 重置 </el-button>
                    </el-button-group>
                </el-form-item>
                <el-form-item>
                    <el-button icon="ele-DeleteFilled" type="danger" @click="clearLog" v-auth="'sysExlog:clear'"> 清空 </el-button>
                    <el-button icon="ele-FolderOpened" @click="exportLog" v-auth="'sysExlog:export'"> 导出 </el-button>
                </el-form-item>
            </el-form>
        </el-card>
        <el-card class="full-table" shadow="hover" style="margin-top: 5px">
            <el-table :data="state.logData" @sort-change="sortChange" style="width: 100%" border :row-class-name="tableRowClassName">
                <el-table-column type="index" label="序号" width="55" align="center" />
                <el-table-column prop="controllerName" label="模块名称" width="100" header-align="center" show-overflow-tooltip />
                <el-table-column prop="displayTitle" label="显示名称" width="150" header-align="center" show-overflow-tooltip />
                <el-table-column prop="actionName" label="方法名称" width="100" header-align="center" show-overflow-tooltip />
                <el-table-column prop="httpMethod" label="请求方式" width="90" align="center" show-overflow-tooltip />
                <el-table-column prop="requestUrl" label="请求地址" width="300" header-align="center" show-overflow-tooltip />
                <!-- <el-table-column prop="requestParam" label="请求参数" show-overflow-tooltip />
                <el-table-column prop="returnResult" label="返回结果" show-overflow-tooltip /> -->
                <el-table-column prop="logLevel" label="级别" width="70" align="center" show-overflow-tooltip>
                    <template #default="scope">
                        <el-tag v-if="scope.row.logLevel === 1">调试</el-tag>
                        <el-tag v-else-if="scope.row.logLevel === 2">消息</el-tag>
                        <el-tag v-else-if="scope.row.logLevel === 3">警告</el-tag>
                        <el-tag v-else-if="scope.row.logLevel === 4">错误</el-tag>
                        <el-tag v-else>其他</el-tag>
                    </template>
                </el-table-column>
                <el-table-column prop="eventId" label="事件Id" width="70" align="center" show-overflow-tooltip />
                <el-table-column prop="threadId" label="线程Id" sortable="custom" width="90" align="center" show-overflow-tooltip />
                <el-table-column prop="traceId" label="请求跟踪Id" width="150" header-align="center" sortable="custom" show-overflow-tooltip />
                <el-table-column prop="account" label="账号名称" width="100" align="center" show-overflow-tooltip />
                <el-table-column prop="realName" label="真实姓名" width="100" align="center" show-overflow-tooltip />
                <el-table-column prop="remoteIp" label="IP地址" width="120" align="center" show-overflow-tooltip />
                <el-table-column prop="location" label="登录地点" width="150" align="center" show-overflow-tooltip />
                <el-table-column prop="longitude" label="经度" min-width="100" align="center" show-overflow-tooltip />
                <el-table-column prop="latitude" label="纬度" min-width="100" align="center" show-overflow-tooltip />
                <el-table-column prop="browser" label="浏览器" width="160" align="center" show-overflow-tooltip />
                <el-table-column prop="os" label="操作系统" width="120" align="center" show-overflow-tooltip />
                <el-table-column prop="status" label="状态" width="70" align="center" show-overflow-tooltip>
                    <template #default="scope">
                        <el-tag type="success" v-if="scope.row.status === '200'">成功</el-tag>
                        <el-tag type="danger" v-else>失败</el-tag>
                    </template>
                </el-table-column>
                <el-table-column prop="elapsed" label="耗时(ms)" width="90" align="center" show-overflow-tooltip />
                <el-table-column prop="exception" label="异常对象" width="150" header-align="center" show-overflow-tooltip />
                <!-- <el-table-column prop="message" label="日志消息" width="160" fixed="right" show-overflow-tooltip /> -->
                <el-table-column prop="logDateTime" label="日志时间" width="160" align="center" fixed="right" show-overflow-tooltip />
                <el-table-column label="操作" width="80" align="center" fixed="right" show-overflow-tooltip>
                    <template #default="scope">
                        <el-button icon="ele-InfoFilled" size="small" text type="primary" @click="viewDetail(scope.row)" v-auth="'sysOplog:page'">详情 </el-button>
                    </template>
                </el-table-column>
            </el-table>
            <el-pagination
                v-model:currentPage="state.tableParams.page"
                v-model:page-size="state.tableParams.pageSize"
                :total="state.tableParams.total"
                :page-sizes="[10, 20, 50, 100]"
                size="small"
                background
                @size-change="handleSizeChange"
                @current-change="handleCurrentChange"
                layout="total, sizes, prev, pager, next, jumper"
            />
        </el-card>
        <el-dialog v-model="state.dialogVisible" draggable fullscreen>
            <template #header>
                <div style="color: #fff">
                    <el-icon size="16" style="margin-right: 3px; display: inline; vertical-align: middle"> <ele-Document /> </el-icon>
                    <span> 日志详情 </span>
                </div>
            </template>
            <pre v-loading="state.loadingDetail">{{ state.content }}</pre>
        </el-dialog>
    </div>
</template>
<script lang="ts" setup name="sysExLog">
import { onMounted, reactive } from 'vue';
import { ElMessage } from 'element-plus';
import { downloadByData, getFileName } from '/@/utils/download';
import { getAPI } from '/@/utils/axios-utils';
import { SysLogExApi } from '/@/api-services/api';
import { SysLogEx } from '/@/api-services/models';
const state = reactive({
    loading: false,
    loadingDetail: false,
    queryParams: {
        startTime: undefined,
        endTime: undefined,
        status: undefined,
        controllerName: undefined,
        actionName: undefined,
        account: undefined,
        elapsed: undefined,
        remoteIp: undefined,
    },
    tableParams: {
        page: 1,
        pageSize: 20,
        field: 'createTime', // 默认的排序字段
        order: 'descending', // 排序方向
        descStr: 'descending', // 降序排序的关键字符
        total: 0 as any,
    },
    logData: [] as Array<SysLogEx>,
    dialogVisible: false,
    content: '',
});
onMounted(async () => {
    handleQuery();
});
// 查询操作
const handleQuery = async () => {
    if (state.queryParams.startTime == null) state.queryParams.startTime = undefined;
    if (state.queryParams.endTime == null) state.queryParams.endTime = undefined;
    if (state.queryParams.status == null) state.queryParams.status = undefined;
    if (state.queryParams.controllerName == null) state.queryParams.controllerName = undefined;
    if (state.queryParams.actionName == null) state.queryParams.actionName = undefined;
    if (state.queryParams.account == null) state.queryParams.account = undefined;
    if (state.queryParams.elapsed == null) state.queryParams.elapsed = undefined;
    if (state.queryParams.remoteIp == null) state.queryParams.remoteIp = undefined;
    state.loading = true;
    let params = Object.assign(state.queryParams, state.tableParams);
    var res = await getAPI(SysLogExApi).apiSysLogExPagePost(params);
    state.logData = res.data.result?.items ?? [];
    state.tableParams.total = res.data.result?.total;
    state.loading = false;
};
// 重置操作
const resetQuery = () => {
    state.queryParams.startTime = undefined;
    state.queryParams.endTime = undefined;
    state.queryParams.status = undefined;
    state.queryParams.controllerName = undefined;
    state.queryParams.actionName = undefined;
    state.queryParams.account = undefined;
    state.queryParams.elapsed = undefined;
    state.queryParams.remoteIp = undefined;
    handleQuery();
};
// 清空日志
const clearLog = async () => {
    state.loading = true;
    await getAPI(SysLogExApi).apiSysLogExClearPost();
    state.loading = false;
    ElMessage.success('清空成功');
    handleQuery();
};
// 导出日志
const exportLog = async () => {
    state.loading = true;
    var res = await getAPI(SysLogExApi).apiSysLogExExportPost(state.queryParams, { responseType: 'blob' });
    state.loading = false;
    var fileName = getFileName(res.headers);
    downloadByData(res.data as any, fileName);
};
// 改变页面容量
const handleSizeChange = (val: number) => {
    state.tableParams.pageSize = val;
    handleQuery();
};
// 改变页码序号
const handleCurrentChange = (val: number) => {
    state.tableParams.page = val;
    handleQuery();
};
// 查看详情
const viewDetail = async (row: any) => {
    state.content = '';
    state.dialogVisible = true;
    state.loadingDetail = true;
    var res = await getAPI(SysLogExApi).apiSysLogExDetailIdGet(row.id);
    row.message = res.data.result?.message ?? '';
    state.content = row.message;
    state.loadingDetail = false;
};
// 设置行颜色
const tableRowClassName = (row: any) => {
    return row.row.exception != null ? 'warning-row' : '';
};
const shortcuts = [
    {
        text: '今天',
        value: new Date(),
    },
    {
        text: '昨天',
        value: () => {
            const date = new Date();
            date.setTime(date.getTime() - 3600 * 1000 * 24);
            return date;
        },
    },
    {
        text: '上周',
        value: () => {
            const date = new Date();
            date.setTime(date.getTime() - 3600 * 1000 * 24 * 7);
            return date;
        },
    },
];
// 列排序
const sortChange = (column: any) => {
    state.tableParams.field = column.prop;
    state.tableParams.order = column.order;
    handleQuery();
};
</script>
<style lang="scss" scoped>
.el-popper {
    max-width: 60%;
}
pre {
    white-space: break-spaces;
    line-height: 20px;
}
.el-table .warning-row {
    --el-table-tr-bg-color: var(--el-color-warning-light-9);
}
.el-table .success-row {
    --el-table-tr-bg-color: var(--el-color-success-light-9);
}
</style>
Web/src/views/system/log/oplog/index.vue
New file
@@ -0,0 +1,270 @@
<template>
    <div class="sys-oplog-container" v-loading="state.loading">
        <el-card shadow="hover" :body-style="{ paddingBottom: '0' }">
            <el-form :model="state.queryParams" ref="queryForm" :inline="true">
                <el-form-item label="开始时间">
                    <el-date-picker v-model="state.queryParams.startTime" type="datetime" placeholder="开始时间" value-format="YYYY-MM-DD HH:mm:ss" :shortcuts="shortcuts" />
                </el-form-item>
                <el-form-item label="结束时间">
                    <el-date-picker v-model="state.queryParams.endTime" type="datetime" placeholder="结束时间" value-format="YYYY-MM-DD HH:mm:ss" :shortcuts="shortcuts" />
                </el-form-item>
                <el-form-item label="模块名称">
                    <el-input v-model="state.queryParams.controllerName" placeholder="模块名称" clearable />
                </el-form-item>
                <el-form-item label="方法名称">
                    <el-input v-model="state.queryParams.actionName" placeholder="方法名称" clearable />
                </el-form-item>
                <el-form-item label="账号名称">
                    <el-input v-model="state.queryParams.account" placeholder="账号名称" clearable />
                </el-form-item>
                <el-form-item label="耗时">
                    <el-input v-model="state.queryParams.elapsed" placeholder="耗时>?MS" clearable />
                </el-form-item>
                <el-form-item label="IP地址">
                    <el-input v-model="state.queryParams.remoteIp" placeholder="IP地址" clearable />
                </el-form-item>
                <el-form-item>
                    <el-button-group>
                        <el-button type="primary" icon="ele-Search" @click="handleQuery" v-auth="'sysOplog:page'"> 查询 </el-button>
                        <el-button icon="ele-Refresh" @click="resetQuery"> 重置 </el-button>
                    </el-button-group>
                </el-form-item>
                <el-form-item>
                    <el-button icon="ele-DeleteFilled" type="danger" @click="clearLog" v-auth="'sysOplog:clear'"> 清空 </el-button>
                    <el-button icon="ele-FolderOpened" @click="exportLog" v-auth="'sysOplog:export'"> 导出 </el-button>
                </el-form-item>
            </el-form>
        </el-card>
        <el-card class="full-table" shadow="hover" style="margin-top: 5px">
            <el-table :data="state.logData" @sort-change="sortChange" style="width: 100%" border :row-class-name="tableRowClassName">
                <el-table-column type="index" label="序号" width="55" align="center" />
                <el-table-column prop="controllerName" label="模块名称" min-width="120" header-align="center" show-overflow-tooltip />
                <el-table-column prop="displayTitle" label="显示名称" width="170" header-align="center" show-overflow-tooltip />
                <el-table-column prop="actionName" label="方法名称" width="150" header-align="center" show-overflow-tooltip />
                <el-table-column prop="httpMethod" label="请求方式" width="90" align="center" show-overflow-tooltip />
                <el-table-column prop="requestUrl" label="请求地址" width="300" header-align="center" show-overflow-tooltip />
                <!-- <el-table-column prop="requestParam" label="请求参数" show-overflow-tooltip />
                <el-table-column prop="returnResult" label="返回结果" show-overflow-tooltip /> -->
                <el-table-column prop="logLevel" label="级别" width="70" align="center" show-overflow-tooltip>
                    <template #default="scope">
                        <el-tag v-if="scope.row.logLevel === 1">调试</el-tag>
                        <el-tag v-else-if="scope.row.logLevel === 2">消息</el-tag>
                        <el-tag v-else-if="scope.row.logLevel === 3">警告</el-tag>
                        <el-tag v-else-if="scope.row.logLevel === 4">错误</el-tag>
                        <el-tag v-else>其他</el-tag>
                    </template>
                </el-table-column>
                <el-table-column prop="eventId" label="事件Id" width="70" align="center" show-overflow-tooltip />
                <el-table-column prop="threadId" label="线程Id" sortable="custom" width="90" align="center" show-overflow-tooltip />
                <el-table-column prop="traceId" label="请求跟踪Id" width="150" header-align="center" sortable="custom" show-overflow-tooltip />
                <el-table-column prop="account" label="账号名称" width="100" align="center" show-overflow-tooltip />
                <el-table-column prop="realName" label="真实姓名" width="100" align="center" show-overflow-tooltip />
                <el-table-column prop="remoteIp" label="IP地址" min-width="120" align="center" show-overflow-tooltip />
                <el-table-column prop="location" label="登录地点" min-width="150" align="center" show-overflow-tooltip />
                <el-table-column prop="longitude" label="经度" min-width="100" align="center" show-overflow-tooltip />
                <el-table-column prop="latitude" label="纬度" min-width="100" align="center" show-overflow-tooltip />
                <el-table-column prop="browser" label="浏览器" min-width="150" align="center" show-overflow-tooltip />
                <el-table-column prop="os" label="操作系统" width="120" align="center" show-overflow-tooltip />
                <el-table-column prop="status" label="状态" width="70" align="center" show-overflow-tooltip>
                    <template #default="scope">
                        <el-tag type="success" v-if="scope.row.status === '200'">成功</el-tag>
                        <el-tag type="danger" v-else>失败</el-tag>
                    </template>
                </el-table-column>
                <el-table-column prop="elapsed" label="耗时(ms)" width="90" align="center" show-overflow-tooltip />
                <!-- <el-table-column prop="exception" label="异常对象" width="150" show-overflow-tooltip /> -->
                <!-- <el-table-column prop="message" label="日志消息" width="160" fixed="right" show-overflow-tooltip /> -->
                <el-table-column prop="logDateTime" label="日志时间" width="160" align="center" fixed="right" show-overflow-tooltip />
                <el-table-column label="操作" width="80" align="center" fixed="right" show-overflow-tooltip>
                    <template #default="scope">
                        <el-button icon="ele-InfoFilled" size="small" text type="primary" @click="viewDetail(scope.row)" v-auth="'sysOplog:page'">详情 </el-button>
                    </template>
                </el-table-column>
            </el-table>
            <el-pagination
                v-model:currentPage="state.tableParams.page"
                v-model:page-size="state.tableParams.pageSize"
                :total="state.tableParams.total"
                :page-sizes="[10, 20, 50, 100]"
                size="small"
                background
                @size-change="handleSizeChange"
                @current-change="handleCurrentChange"
                layout="total, sizes, prev, pager, next, jumper"
            />
        </el-card>
        <el-dialog v-model="state.dialogVisible" draggable fullscreen>
            <template #header>
                <div style="color: #fff">
                    <el-icon size="16" style="margin-right: 3px; display: inline; vertical-align: middle"> <ele-Document /> </el-icon>
                    <span> 日志详情 </span>
                </div>
            </template>
            <pre v-loading="state.loadingDetail">{{ state.content }}</pre>
        </el-dialog>
    </div>
</template>
<script lang="ts" setup name="sysOpLog">
import { onMounted, reactive } from 'vue';
import { ElMessage } from 'element-plus';
import { downloadByData, getFileName } from '/@/utils/download';
import { getAPI } from '/@/utils/axios-utils';
import { SysLogOpApi } from '/@/api-services/api';
import { SysLogOp } from '/@/api-services/models';
const state = reactive({
    loading: false,
    loadingDetail: false,
    queryParams: {
        startTime: undefined,
        endTime: undefined,
        controllerName: undefined,
        actionName: undefined,
        account: undefined,
        elapsed: undefined,
        remoteIp: undefined,
    },
    tableParams: {
        page: 1,
        pageSize: 20,
        field: 'createTime', // 默认的排序字段
        order: 'descending', // 排序方向
        descStr: 'descending', // 降序排序的关键字符
        total: 0 as any,
    },
    logData: [] as Array<SysLogOp>,
    dialogVisible: false,
    content: '',
});
onMounted(async () => {
    handleQuery();
});
// 查询操作
const handleQuery = async () => {
    if (state.queryParams.startTime == null) state.queryParams.startTime = undefined;
    if (state.queryParams.endTime == null) state.queryParams.endTime = undefined;
    if (state.queryParams.controllerName == null) state.queryParams.controllerName = undefined;
    if (state.queryParams.actionName == null) state.queryParams.actionName = undefined;
    if (state.queryParams.account == null) state.queryParams.account = undefined;
    if (state.queryParams.elapsed == null) state.queryParams.elapsed = undefined;
    if (state.queryParams.remoteIp == null) state.queryParams.remoteIp = undefined;
    state.loading = true;
    let params = Object.assign(state.queryParams, state.tableParams);
    var res = await getAPI(SysLogOpApi).apiSysLogOpPagePost(params);
    state.logData = res.data.result?.items ?? [];
    state.tableParams.total = res.data.result?.total;
    state.loading = false;
};
// 重置操作
const resetQuery = () => {
    state.queryParams.startTime = undefined;
    state.queryParams.endTime = undefined;
    state.queryParams.controllerName = undefined;
    state.queryParams.actionName = undefined;
    state.queryParams.account = undefined;
    state.queryParams.elapsed = undefined;
    state.queryParams.remoteIp = undefined;
    handleQuery();
};
// 清空日志
const clearLog = async () => {
    state.loading = true;
    await getAPI(SysLogOpApi).apiSysLogOpClearPost();
    state.loading = false;
    ElMessage.success('清空成功');
    handleQuery();
};
// 导出日志
const exportLog = async () => {
    state.loading = true;
    var res = await getAPI(SysLogOpApi).apiSysLogOpExportPost(state.queryParams, { responseType: 'blob' });
    state.loading = false;
    var fileName = getFileName(res.headers);
    downloadByData(res.data as any, fileName);
};
// 改变页面容量
const handleSizeChange = (val: number) => {
    state.tableParams.pageSize = val;
    handleQuery();
};
// 改变页码序号
const handleCurrentChange = (val: number) => {
    state.tableParams.page = val;
    handleQuery();
};
// 查看详情
const viewDetail = async (row: any) => {
    state.content = '';
    state.dialogVisible = true;
    state.loadingDetail = true;
    var res = await getAPI(SysLogOpApi).apiSysLogOpDetailIdGet(row.id);
    row.message = res.data.result?.message ?? '';
    state.content = row.message;
    state.loadingDetail = false;
};
// 设置行颜色
const tableRowClassName = (row: any) => {
    return row.row.exception != null ? 'warning-row' : '';
};
const shortcuts = [
    {
        text: '今天',
        value: new Date(),
    },
    {
        text: '昨天',
        value: () => {
            const date = new Date();
            date.setTime(date.getTime() - 3600 * 1000 * 24);
            return date;
        },
    },
    {
        text: '上周',
        value: () => {
            const date = new Date();
            date.setTime(date.getTime() - 3600 * 1000 * 24 * 7);
            return date;
        },
    },
];
// 列排序
const sortChange = (column: any) => {
    state.tableParams.field = column.prop;
    state.tableParams.order = column.order;
    handleQuery();
};
</script>
<style lang="scss" scoped>
.el-popper {
    max-width: 60%;
}
pre {
    white-space: break-spaces;
    line-height: 20px;
}
.el-table .warning-row {
    --el-table-tr-bg-color: var(--el-color-warning-light-9);
}
.el-table .success-row {
    --el-table-tr-bg-color: var(--el-color-success-light-9);
}
</style>
Web/src/views/system/log/vislog/index.vue
New file
@@ -0,0 +1,183 @@
<template>
    <div class="sys-vislog-container">
        <el-card shadow="hover" :body-style="{ paddingBottom: '0' }">
            <el-form :model="state.queryParams" ref="queryForm" :inline="true">
                <el-form-item label="开始时间">
                    <el-date-picker v-model="state.queryParams.startTime" type="datetime" placeholder="开始时间" value-format="YYYY-MM-DD HH:mm:ss" :shortcuts="shortcuts" />
                </el-form-item>
                <el-form-item label="结束时间" prop="code">
                    <el-date-picker v-model="state.queryParams.endTime" type="datetime" placeholder="结束时间" value-format="YYYY-MM-DD HH:mm:ss" :shortcuts="shortcuts" />
                </el-form-item>
                <el-form-item label="方法名称">
                    <el-input v-model="state.queryParams.actionName" placeholder="方法名称" clearable />
                </el-form-item>
                <el-form-item label="账号名称">
                    <el-input v-model="state.queryParams.account" placeholder="账号名称" clearable />
                </el-form-item>
                <el-form-item label="状态">
                    <el-select v-model="state.queryParams.status" placeholder="状态" clearable>
                        <el-option label="成功" :value="200" />
                        <el-option label="失败" :value="400" />
                    </el-select>
                </el-form-item>
                <el-form-item label="耗时">
                    <el-input v-model="state.queryParams.elapsed" placeholder="耗时>?MS" clearable />
                </el-form-item>
                <el-form-item label="IP地址">
                    <el-input v-model="state.queryParams.remoteIp" placeholder="IP地址" clearable />
                </el-form-item>
                <el-form-item>
                    <el-button-group>
                        <el-button type="primary" icon="ele-Search" @click="handleQuery" v-auth="'sysVislog:page'"> 查询 </el-button>
                        <el-button icon="ele-Refresh" @click="resetQuery"> 重置 </el-button>
                    </el-button-group>
                </el-form-item>
                <el-form-item>
                    <el-button icon="ele-DeleteFilled" type="danger" @click="clearLog" v-auth="'sysVislog:clear'" disabled> 清空 </el-button>
                </el-form-item>
            </el-form>
        </el-card>
        <el-card class="full-table" shadow="hover" style="margin-top: 5px">
            <el-table :data="state.logData" style="width: 100%" v-loading="state.loading" border>
                <el-table-column type="index" label="序号" width="55" align="center" />
                <el-table-column prop="displayTitle" label="显示名称" width="150" align="center" show-overflow-tooltip />
                <el-table-column prop="actionName" label="方法名称" width="150" header-align="center" show-overflow-tooltip />
                <el-table-column prop="account" label="账号名称" width="100" align="center" show-overflow-tooltip />
                <el-table-column prop="realName" label="真实姓名" width="100" align="center" show-overflow-tooltip />
                <el-table-column prop="remoteIp" label="IP地址" min-width="120" align="center" show-overflow-tooltip />
                <el-table-column prop="location" label="登录地点" min-width="150" align="center" show-overflow-tooltip />
                <el-table-column prop="longitude" label="经度" min-width="100" align="center" show-overflow-tooltip />
                <el-table-column prop="latitude" label="纬度" min-width="100" align="center" show-overflow-tooltip />
                <el-table-column prop="browser" label="浏览器" min-width="150" align="center" show-overflow-tooltip />
                <el-table-column prop="os" label="操作系统" width="120" align="center" show-overflow-tooltip />
                <el-table-column prop="status" label="状态" width="70" align="center" show-overflow-tooltip>
                    <template #default="scope">
                        <el-tag type="success" v-if="scope.row.status === '200'">成功</el-tag>
                        <el-tag type="danger" v-else>失败</el-tag>
                    </template>
                </el-table-column>
                <el-table-column prop="elapsed" label="耗时(ms)" width="90" align="center" show-overflow-tooltip />
                <el-table-column prop="logDateTime" label="日志时间" width="160" align="center" fixed="right" show-overflow-tooltip />
            </el-table>
            <el-pagination
                v-model:currentPage="state.tableParams.page"
                v-model:page-size="state.tableParams.pageSize"
                :total="state.tableParams.total"
                :page-sizes="[10, 20, 50, 100]"
                size="small"
                background
                @size-change="handleSizeChange"
                @current-change="handleCurrentChange"
                layout="total, sizes, prev, pager, next, jumper"
            />
        </el-card>
    </div>
</template>
<script lang="ts" setup name="sysVisLog">
import { onMounted, reactive } from 'vue';
import { ElMessage } from 'element-plus';
import { getAPI } from '/@/utils/axios-utils';
import { SysLogVisApi } from '/@/api-services/api';
import { SysLogVis } from '/@/api-services/models';
const state = reactive({
    loading: false,
    queryParams: {
        startTime: undefined,
        endTime: undefined,
        status: undefined,
        actionName: undefined,
        account: undefined,
        elapsed: undefined,
        remoteIp: undefined,
    },
    tableParams: {
        page: 1,
        pageSize: 20,
        total: 0 as any,
    },
    logData: [] as Array<SysLogVis>,
});
onMounted(async () => {
    handleQuery();
});
// 查询操作
const handleQuery = async () => {
    if (state.queryParams.startTime == null) state.queryParams.startTime = undefined;
    if (state.queryParams.endTime == null) state.queryParams.endTime = undefined;
    if (state.queryParams.status == null) state.queryParams.status = undefined;
    if (state.queryParams.actionName == null) state.queryParams.actionName = undefined;
    if (state.queryParams.account == null) state.queryParams.account = undefined;
    if (state.queryParams.elapsed == null) state.queryParams.elapsed = undefined;
    if (state.queryParams.remoteIp == null) state.queryParams.remoteIp = undefined;
    state.loading = true;
    let params = Object.assign(state.queryParams, state.tableParams);
    var res = await getAPI(SysLogVisApi).apiSysLogVisPagePost(params);
    state.logData = res.data.result?.items ?? [];
    state.tableParams.total = res.data.result?.total;
    state.loading = false;
};
// 重置操作
const resetQuery = () => {
    state.queryParams.startTime = undefined;
    state.queryParams.endTime = undefined;
    state.queryParams.status = undefined;
    state.queryParams.actionName = undefined;
    state.queryParams.account = undefined;
    state.queryParams.elapsed = undefined;
    state.queryParams.remoteIp = undefined;
    handleQuery();
};
// 清空日志
const clearLog = async () => {
    state.loading = true;
    await getAPI(SysLogVisApi).apiSysLogVisClearPost();
    state.loading = false;
    ElMessage.success('清空成功');
    handleQuery();
};
// 改变页面容量
const handleSizeChange = (val: number) => {
    state.tableParams.pageSize = val;
    handleQuery();
};
// 改变页码序号
const handleCurrentChange = (val: number) => {
    state.tableParams.page = val;
    handleQuery();
};
const shortcuts = [
    {
        text: '今天',
        value: new Date(),
    },
    {
        text: '昨天',
        value: () => {
            const date = new Date();
            date.setTime(date.getTime() - 3600 * 1000 * 24);
            return date;
        },
    },
    {
        text: '上周',
        value: () => {
            const date = new Date();
            date.setTime(date.getTime() - 3600 * 1000 * 24 * 7);
            return date;
        },
    },
];
</script>