refactor: operatelog
parent
4a692ed28c
commit
f09a1a4b8e
|
@ -1,10 +1,40 @@
|
||||||
import request from '@/config/axios'
|
import request from '@/config/axios'
|
||||||
|
|
||||||
|
export type OperateLogVO = {
|
||||||
|
id: number
|
||||||
|
userNickname: string
|
||||||
|
traceId: string
|
||||||
|
userId: number
|
||||||
|
module: string
|
||||||
|
name: string
|
||||||
|
type: number
|
||||||
|
content: string
|
||||||
|
exts: object
|
||||||
|
requestMethod: string
|
||||||
|
requestUrl: string
|
||||||
|
userIp: string
|
||||||
|
userAgent: string
|
||||||
|
javaMethod: string
|
||||||
|
javaMethodArgs: string
|
||||||
|
startTime: string
|
||||||
|
duration: number
|
||||||
|
resultCode: number
|
||||||
|
resultMsg: string
|
||||||
|
resultData: string
|
||||||
|
}
|
||||||
|
export interface OperateLogPageReqVO extends BasePage {
|
||||||
|
module?: string
|
||||||
|
userNickname?: string
|
||||||
|
type?: number
|
||||||
|
success?: boolean
|
||||||
|
startTime?: string[]
|
||||||
|
}
|
||||||
|
|
||||||
// 查询操作日志列表
|
// 查询操作日志列表
|
||||||
export const getOperateLogPageApi = (params) => {
|
export const getOperateLogPageApi = (params: OperateLogPageReqVO) => {
|
||||||
return request.get({ url: '/system/operate-log/page', params })
|
return request.get({ url: '/system/operate-log/page', params })
|
||||||
}
|
}
|
||||||
// 导出操作日志
|
// 导出操作日志
|
||||||
export const exportOperateLogApi = (params) => {
|
export const exportOperateLogApi = (params: OperateLogPageReqVO) => {
|
||||||
return request.download({ url: '/system/operate-log/export', params })
|
return request.download({ url: '/system/operate-log/export', params })
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,22 +0,0 @@
|
||||||
export type OperateLogVO = {
|
|
||||||
id: number
|
|
||||||
userNickname: string
|
|
||||||
traceId: string
|
|
||||||
userId: number
|
|
||||||
module: string
|
|
||||||
name: string
|
|
||||||
type: number
|
|
||||||
content: string
|
|
||||||
exts: object
|
|
||||||
requestMethod: string
|
|
||||||
requestUrl: string
|
|
||||||
userIp: string
|
|
||||||
userAgent: string
|
|
||||||
javaMethod: string
|
|
||||||
javaMethodArgs: string
|
|
||||||
startTime: string
|
|
||||||
duration: number
|
|
||||||
resultCode: number
|
|
||||||
resultMsg: string
|
|
||||||
resultData: string
|
|
||||||
}
|
|
|
@ -20,6 +20,7 @@ export type VxeCrudSchema = {
|
||||||
primaryType?: VxeColumnPropTypes.Type
|
primaryType?: VxeColumnPropTypes.Type
|
||||||
// 是否开启操作栏插槽
|
// 是否开启操作栏插槽
|
||||||
action?: boolean
|
action?: boolean
|
||||||
|
actionWidth?: string
|
||||||
columns: VxeCrudColumns[]
|
columns: VxeCrudColumns[]
|
||||||
}
|
}
|
||||||
type VxeCrudColumns = Omit<VxeTableColumn, 'children'> & {
|
type VxeCrudColumns = Omit<VxeTableColumn, 'children'> & {
|
||||||
|
@ -204,7 +205,7 @@ const filterTableSchema = (crudSchema: VxeCrudSchema): VxeGridPropTypes.Columns
|
||||||
const tableSchemaItem = {
|
const tableSchemaItem = {
|
||||||
title: t('table.action'),
|
title: t('table.action'),
|
||||||
field: 'actionbtns',
|
field: 'actionbtns',
|
||||||
width: '240px',
|
width: crudSchema.actionWidth ? crudSchema.actionWidth : '240px',
|
||||||
slots: {
|
slots: {
|
||||||
default: 'actionbtns_default'
|
default: 'actionbtns_default'
|
||||||
}
|
}
|
||||||
|
|
|
@ -10,11 +10,6 @@ import { DeptVO } from '@/api/system/dept/types'
|
||||||
import { useMessage } from '@/hooks/web/useMessage'
|
import { useMessage } from '@/hooks/web/useMessage'
|
||||||
import { getListSimpleUsersApi } from '@/api/system/user'
|
import { getListSimpleUsersApi } from '@/api/system/user'
|
||||||
const message = useMessage()
|
const message = useMessage()
|
||||||
interface Tree {
|
|
||||||
id: number
|
|
||||||
name: string
|
|
||||||
children?: Tree[]
|
|
||||||
}
|
|
||||||
|
|
||||||
const defaultProps = {
|
const defaultProps = {
|
||||||
children: 'children',
|
children: 'children',
|
||||||
|
|
|
@ -9,6 +9,7 @@ const crudSchemas = reactive<VxeCrudSchema>({
|
||||||
primaryKey: 'id',
|
primaryKey: 'id',
|
||||||
primaryType: 'seq',
|
primaryType: 'seq',
|
||||||
action: true,
|
action: true,
|
||||||
|
actionWidth: '80px',
|
||||||
columns: [
|
columns: [
|
||||||
{
|
{
|
||||||
title: '日志类型',
|
title: '日志类型',
|
||||||
|
|
|
@ -1,97 +1,85 @@
|
||||||
|
<template>
|
||||||
|
<ContentWrap>
|
||||||
|
<!-- 列表 -->
|
||||||
|
<vxe-grid ref="xGrid" v-bind="gridOptions" class="xtable-scrollbar">
|
||||||
|
<!-- 操作:新增 -->
|
||||||
|
<template #toolbar_buttons>
|
||||||
|
<XButton
|
||||||
|
type="warning"
|
||||||
|
preIcon="ep:download"
|
||||||
|
:title="t('action.export')"
|
||||||
|
v-hasPermi="['system:operate-log:export']"
|
||||||
|
@click="handleExport()"
|
||||||
|
/>
|
||||||
|
</template>
|
||||||
|
<template #duration="{ row }">
|
||||||
|
<span>{{ row.duration + 'ms' }}</span>
|
||||||
|
</template>
|
||||||
|
<template #resultCode="{ row }">
|
||||||
|
<span>{{ row.resultCode === 0 ? '成功' : '失败' }}</span>
|
||||||
|
</template>
|
||||||
|
<template #actionbtns_default="{ row }">
|
||||||
|
<!-- 操作:详情 -->
|
||||||
|
<XTextButton preIcon="ep:view" :title="t('action.detail')" @click="handleDetail(row)" />
|
||||||
|
</template>
|
||||||
|
</vxe-grid>
|
||||||
|
</ContentWrap>
|
||||||
|
<!-- 弹窗 -->
|
||||||
|
<XModal id="postModel" v-model="dialogVisible" :title="dialogTitle">
|
||||||
|
<template #default>
|
||||||
|
<!-- 对话框(详情) -->
|
||||||
|
<Descriptions :schema="allSchemas.detailSchema" :data="detailRef">
|
||||||
|
<template #resultCode="{ row }">
|
||||||
|
<span>{{ row.resultCode === 0 ? '成功' : '失败' }}</span>
|
||||||
|
</template>
|
||||||
|
<template #duration="{ row }">
|
||||||
|
<span>{{ row.duration + 'ms' }}</span>
|
||||||
|
</template>
|
||||||
|
</Descriptions>
|
||||||
|
</template>
|
||||||
|
<template #footer>
|
||||||
|
<!-- 按钮:关闭 -->
|
||||||
|
<XButton :loading="actionLoading" :title="t('dialog.close')" @click="dialogVisible = false" />
|
||||||
|
</template>
|
||||||
|
</XModal>
|
||||||
|
</template>
|
||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import dayjs from 'dayjs'
|
// 全局相关的 import
|
||||||
import { useTable } from '@/hooks/web/useTable'
|
|
||||||
import { allSchemas } from './operatelog.data'
|
|
||||||
import { DICT_TYPE } from '@/utils/dict'
|
|
||||||
import { useI18n } from '@/hooks/web/useI18n'
|
|
||||||
import type { OperateLogVO } from '@/api/system/operatelog/types'
|
|
||||||
import * as OperateLogApi from '@/api/system/operatelog'
|
|
||||||
import { ref } from 'vue'
|
import { ref } from 'vue'
|
||||||
|
import { useI18n } from '@/hooks/web/useI18n'
|
||||||
|
import { useVxeGrid } from '@/hooks/web/useVxeGrid'
|
||||||
|
import { VxeGridInstance } from 'vxe-table'
|
||||||
|
// 业务相关的 import
|
||||||
|
import * as OperateLogApi from '@/api/system/operatelog'
|
||||||
|
import { allSchemas } from './operatelog.data'
|
||||||
|
import download from '@/utils/download'
|
||||||
|
|
||||||
const { t } = useI18n() // 国际化
|
const { t } = useI18n() // 国际化
|
||||||
// ========== 列表相关 ==========
|
// 列表相关的变量
|
||||||
const { register, tableObject, methods } = useTable<OperateLogVO>({
|
const xGrid = ref<VxeGridInstance>() // 列表 Grid Ref
|
||||||
getListApi: OperateLogApi.getOperateLogPageApi,
|
const { gridOptions } = useVxeGrid<OperateLogApi.OperateLogVO>({
|
||||||
exportListApi: OperateLogApi.exportOperateLogApi
|
allSchemas: allSchemas,
|
||||||
|
getListApi: OperateLogApi.getOperateLogPageApi
|
||||||
})
|
})
|
||||||
// ========== 详情相关 ==========
|
// 弹窗相关的变量
|
||||||
const detailRef = ref() // 详情 Ref
|
|
||||||
const dialogVisible = ref(false) // 是否显示弹出层
|
const dialogVisible = ref(false) // 是否显示弹出层
|
||||||
const dialogTitle = ref(t('action.detail')) // 弹出层标题
|
const dialogTitle = ref('edit') // 弹出层标题
|
||||||
const { getList, setSearchParams, exportList } = methods
|
const actionLoading = ref(false) // 按钮 Loading
|
||||||
|
const detailRef = ref() // 详情 Ref
|
||||||
// 详情
|
// 详情
|
||||||
const handleDetail = (row: OperateLogVO) => {
|
const handleDetail = (row: OperateLogApi.OperateLogVO) => {
|
||||||
// 设置数据
|
// 设置数据
|
||||||
detailRef.value = row
|
detailRef.value = row
|
||||||
dialogVisible.value = true
|
dialogVisible.value = true
|
||||||
}
|
}
|
||||||
getList()
|
|
||||||
|
// 导出操作
|
||||||
|
const handleExport = async () => {
|
||||||
|
const queryParams = Object.assign(
|
||||||
|
{},
|
||||||
|
JSON.parse(JSON.stringify(xGrid.value?.getRefMaps().refForm.value.data))
|
||||||
|
)
|
||||||
|
const res = await OperateLogApi.exportOperateLogApi(queryParams)
|
||||||
|
download.excel(res, '岗位列表.xls')
|
||||||
|
}
|
||||||
</script>
|
</script>
|
||||||
<template>
|
|
||||||
<ContentWrap>
|
|
||||||
<Search :schema="allSchemas.searchSchema" @search="setSearchParams" @reset="setSearchParams" />
|
|
||||||
</ContentWrap>
|
|
||||||
<ContentWrap>
|
|
||||||
<!-- 操作工具栏 -->
|
|
||||||
<div class="mb-10px">
|
|
||||||
<el-button
|
|
||||||
type="warning"
|
|
||||||
v-hasPermi="['system:operate-log:export']"
|
|
||||||
:loading="tableObject.exportLoading"
|
|
||||||
@click="exportList('操作日志.xls')"
|
|
||||||
>
|
|
||||||
<Icon icon="ep:download" class="mr-5px" /> {{ t('action.export') }}
|
|
||||||
</el-button>
|
|
||||||
</div>
|
|
||||||
<Table
|
|
||||||
:columns="allSchemas.tableColumns"
|
|
||||||
:selection="false"
|
|
||||||
:data="tableObject.tableList"
|
|
||||||
:loading="tableObject.loading"
|
|
||||||
:pagination="{
|
|
||||||
total: tableObject.total
|
|
||||||
}"
|
|
||||||
v-model:pageSize="tableObject.pageSize"
|
|
||||||
v-model:currentPage="tableObject.currentPage"
|
|
||||||
@register="register"
|
|
||||||
>
|
|
||||||
<template #type="{ row }">
|
|
||||||
<DictTag :type="DICT_TYPE.SYSTEM_OPERATE_TYPE" :value="row.type" />
|
|
||||||
</template>
|
|
||||||
<template #duration="{ row }">
|
|
||||||
<span>{{ row.duration + 'ms' }}</span>
|
|
||||||
</template>
|
|
||||||
<template #resultCode="{ row }">
|
|
||||||
<span>{{ row.resultCode === 0 ? '成功' : '失败' }}</span>
|
|
||||||
</template>
|
|
||||||
<template #startTime="{ row }">
|
|
||||||
<span>{{ dayjs(row.startTime).format('YYYY-MM-DD HH:mm:ss') }}</span>
|
|
||||||
</template>
|
|
||||||
<template #action="{ row }">
|
|
||||||
<el-button link type="primary" @click="handleDetail(row)">
|
|
||||||
<Icon icon="ep:view" class="mr-1px" /> {{ t('action.detail') }}
|
|
||||||
</el-button>
|
|
||||||
</template>
|
|
||||||
</Table>
|
|
||||||
</ContentWrap>
|
|
||||||
<Dialog v-model="dialogVisible" :title="dialogTitle">
|
|
||||||
<!-- 对话框(详情) -->
|
|
||||||
<Descriptions :schema="allSchemas.detailSchema" :data="detailRef">
|
|
||||||
<template #resultCode="{ row }">
|
|
||||||
<span>{{ row.resultCode === 0 ? '成功' : '失败' }}</span>
|
|
||||||
</template>
|
|
||||||
<template #type="{ row }">
|
|
||||||
<DictTag :type="DICT_TYPE.SYSTEM_OPERATE_TYPE" :value="row.type" />
|
|
||||||
</template>
|
|
||||||
<template #duration="{ row }">
|
|
||||||
<span>{{ row.duration + 'ms' }}</span>
|
|
||||||
</template>
|
|
||||||
<template #startTime="{ row }">
|
|
||||||
<span>{{ dayjs(row.startTime).format('YYYY-MM-DD HH:mm:ss') }}</span>
|
|
||||||
</template>
|
|
||||||
</Descriptions>
|
|
||||||
<!-- 操作按钮 -->
|
|
||||||
<template #footer>
|
|
||||||
<el-button @click="dialogVisible = false">{{ t('dialog.close') }}</el-button>
|
|
||||||
</template>
|
|
||||||
</Dialog>
|
|
||||||
</template>
|
|
||||||
|
|
|
@ -1,113 +1,84 @@
|
||||||
import { reactive } from 'vue'
|
import { reactive } from 'vue'
|
||||||
import { DICT_TYPE } from '@/utils/dict'
|
import { DICT_TYPE } from '@/utils/dict'
|
||||||
import { useI18n } from '@/hooks/web/useI18n'
|
import { VxeCrudSchema, useVxeCrudSchemas } from '@/hooks/web/useVxeCrudSchemas'
|
||||||
import { CrudSchema, useCrudSchemas } from '@/hooks/web/useCrudSchemas'
|
|
||||||
const { t } = useI18n() // 国际化
|
const crudSchemas = reactive<VxeCrudSchema>({
|
||||||
const crudSchemas = reactive<CrudSchema[]>([
|
primaryKey: 'id',
|
||||||
{
|
primaryType: 'seq',
|
||||||
label: t('common.index'),
|
action: true,
|
||||||
field: 'id',
|
actionWidth: '80px',
|
||||||
type: 'index',
|
columns: [
|
||||||
form: {
|
{
|
||||||
show: false
|
title: '操作模块',
|
||||||
}
|
field: 'module',
|
||||||
},
|
isSearch: true
|
||||||
{
|
},
|
||||||
label: '操作模块',
|
{
|
||||||
field: 'module',
|
title: '操作名',
|
||||||
search: {
|
field: 'name'
|
||||||
show: true
|
},
|
||||||
}
|
{
|
||||||
},
|
title: '操作类型',
|
||||||
{
|
field: 'type',
|
||||||
label: '操作名',
|
dictType: DICT_TYPE.SYSTEM_OPERATE_TYPE,
|
||||||
field: 'name'
|
isSearch: true
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
label: '操作类型',
|
title: '请求方法名',
|
||||||
field: 'type',
|
field: 'requestMethod'
|
||||||
dictType: DICT_TYPE.SYSTEM_OPERATE_TYPE,
|
},
|
||||||
search: {
|
{
|
||||||
show: true
|
title: '请求地址',
|
||||||
}
|
field: 'requestUrl'
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
label: '请求方法名',
|
title: '操作人员',
|
||||||
field: 'requestMethod'
|
field: 'userNickname',
|
||||||
},
|
isSearch: true
|
||||||
{
|
},
|
||||||
label: '请求地址',
|
{
|
||||||
field: 'requestUrl'
|
title: '操作明细',
|
||||||
},
|
field: 'content',
|
||||||
{
|
isTable: false
|
||||||
label: '操作人员',
|
},
|
||||||
field: 'userNickname',
|
{
|
||||||
search: {
|
title: '用户 IP',
|
||||||
show: true
|
field: 'userIp',
|
||||||
}
|
isTable: false
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
label: '操作明细',
|
title: 'userAgent',
|
||||||
field: 'content',
|
field: 'userAgent'
|
||||||
table: {
|
},
|
||||||
show: false
|
{
|
||||||
}
|
title: '操作结果',
|
||||||
},
|
field: 'resultCode',
|
||||||
{
|
table: {
|
||||||
label: '用户 IP',
|
slots: {
|
||||||
field: 'userIp',
|
default: 'resultCode'
|
||||||
table: {
|
}
|
||||||
show: false
|
}
|
||||||
}
|
},
|
||||||
},
|
{
|
||||||
{
|
title: '操作日期',
|
||||||
label: 'userAgent',
|
field: 'startTime',
|
||||||
field: 'userAgent'
|
formatter: 'formatDate',
|
||||||
},
|
isForm: false,
|
||||||
{
|
search: {
|
||||||
label: '操作结果',
|
itemRender: {
|
||||||
field: 'resultCode',
|
name: 'XDataTimePicker'
|
||||||
search: {
|
}
|
||||||
show: true,
|
}
|
||||||
component: 'Select',
|
},
|
||||||
componentProps: {
|
{
|
||||||
options: [
|
title: '执行时长',
|
||||||
{ label: '成功', value: true },
|
field: 'duration',
|
||||||
{ label: '失败', value: false }
|
table: {
|
||||||
]
|
slots: {
|
||||||
|
default: 'duration'
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
]
|
||||||
{
|
})
|
||||||
label: '操作日期',
|
export const { allSchemas } = useVxeCrudSchemas(crudSchemas)
|
||||||
field: 'startTime',
|
|
||||||
form: {
|
|
||||||
show: false
|
|
||||||
},
|
|
||||||
search: {
|
|
||||||
show: true,
|
|
||||||
component: 'DatePicker',
|
|
||||||
componentProps: {
|
|
||||||
type: 'daterange',
|
|
||||||
valueFormat: 'YYYY-MM-DD HH:mm:ss',
|
|
||||||
defaultTime: [new Date(2000, 1, 1, 0, 0, 0), new Date(2000, 2, 1, 23, 59, 59)]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: '执行时长',
|
|
||||||
field: 'duration'
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: t('table.action'),
|
|
||||||
field: 'action',
|
|
||||||
width: '120px',
|
|
||||||
form: {
|
|
||||||
show: false
|
|
||||||
},
|
|
||||||
detail: {
|
|
||||||
show: false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
])
|
|
||||||
export const { allSchemas } = useCrudSchemas(crudSchemas)
|
|
||||||
|
|
|
@ -39,11 +39,6 @@ import { getAccessToken, getTenantId } from '@/utils/auth'
|
||||||
import { useMessage } from '@/hooks/web/useMessage'
|
import { useMessage } from '@/hooks/web/useMessage'
|
||||||
|
|
||||||
const message = useMessage()
|
const message = useMessage()
|
||||||
interface Tree {
|
|
||||||
id: number
|
|
||||||
name: string
|
|
||||||
children?: Tree[]
|
|
||||||
}
|
|
||||||
|
|
||||||
const defaultProps = {
|
const defaultProps = {
|
||||||
children: 'children',
|
children: 'children',
|
||||||
|
|
Loading…
Reference in New Issue