Files
X-Financial/web/src/views/scripts/LogsView.js
2026-05-28 22:49:58 +08:00

291 lines
8.2 KiB
JavaScript

import { computed, onBeforeUnmount, onMounted, ref, watch } from 'vue'
import { useRouter } from 'vue-router'
import EnterpriseListPage from '../../components/shared/EnterpriseListPage.vue'
import { useSystemState } from '../../composables/useSystemState.js'
import { useToast } from '../../composables/useToast.js'
import { fetchSystemLogEntries } from '../../services/systemLogs.js'
import { AGENT_RUN_POLL_INTERVAL_MS } from '../../utils/agentRunMonitor.js'
import { isManagerUser } from '../../utils/accessControl.js'
function formatDateTime(value) {
if (!value) {
return '未结束'
}
const date = new Date(value)
if (Number.isNaN(date.getTime())) {
return String(value)
}
return date.toLocaleString('zh-CN', { hour12: false })
}
function formatSummary(summary) {
const text = String(summary || '').trim()
if (!text) {
return '暂无摘要。'
}
if (text.length <= 64) {
return text
}
return `${text.slice(0, 64)}...`
}
function resolveSystemLevelTone(level) {
if (level === 'ERROR' || level === 'CRITICAL') {
return 'danger'
}
if (level === 'WARNING' || level === 'WARN') {
return 'warning'
}
if (level === 'INFO') {
return 'info'
}
return 'muted'
}
function resolveSystemOutcomeTone(outcome) {
if (outcome === '失败') {
return 'danger'
}
if (outcome === '异常' || outcome === '告警') {
return 'warning'
}
if (outcome === '成功') {
return 'success'
}
return 'muted'
}
export default {
name: 'LogsView',
components: {
EnterpriseListPage
},
emits: ['summary-change'],
setup(_, { emit }) {
const router = useRouter()
const { currentUser } = useSystemState()
const { toast } = useToast()
const systemLogLoading = ref(false)
const systemSearchKeyword = ref('')
const systemLevelFilter = ref('')
const systemEventTypeFilter = ref('')
const systemLogEntries = ref([])
const openFilterKey = ref('')
const currentPage = ref(1)
const pageSize = ref(10)
const pageSizes = [10, 20, 50]
const pageSizeOptions = pageSizes.map((size) => ({ label: `${size} 条/页`, value: size }))
let pollTimer = 0
const isAdmin = computed(() => isManagerUser(currentUser.value))
const systemLevelOptions = computed(() =>
Array.from(new Set(systemLogEntries.value.map((entry) => entry.level).filter(Boolean)))
)
const systemEventTypeOptions = computed(() =>
Array.from(new Set(systemLogEntries.value.map((entry) => entry.event_type).filter(Boolean)))
)
const systemLevelFilterOptions = computed(() => [
{ label: '全部级别', value: '' },
...systemLevelOptions.value.map((level) => ({ label: level, value: level }))
])
const systemEventTypeFilterOptions = computed(() => [
{ label: '全部类型', value: '' },
...systemEventTypeOptions.value.map((eventType) => ({ label: eventType, value: eventType }))
])
const systemLevelFilterLabel = computed(() =>
systemLevelFilterOptions.value.find((item) => item.value === systemLevelFilter.value)?.label || '全部级别'
)
const systemEventTypeFilterLabel = computed(() =>
systemEventTypeFilterOptions.value.find((item) => item.value === systemEventTypeFilter.value)?.label || '全部类型'
)
const hasActiveFilters = computed(() =>
Boolean(systemSearchKeyword.value.trim() || systemLevelFilter.value || systemEventTypeFilter.value)
)
const filteredSystemLogEntries = computed(() => {
const keyword = systemSearchKeyword.value.trim().toLowerCase()
return systemLogEntries.value.filter((entry) => {
if (systemLevelFilter.value && entry.level !== systemLevelFilter.value) {
return false
}
if (systemEventTypeFilter.value && entry.event_type !== systemEventTypeFilter.value) {
return false
}
if (!keyword) {
return true
}
const haystack = [
entry.summary,
entry.message,
entry.logger,
entry.request_id,
entry.path,
entry.event_type,
entry.outcome,
entry.source_file
]
.filter(Boolean)
.join(' ')
.toLowerCase()
return haystack.includes(keyword)
})
})
const totalCount = computed(() => filteredSystemLogEntries.value.length)
const errorCount = computed(() =>
filteredSystemLogEntries.value.filter((entry) => ['ERROR', 'CRITICAL'].includes(entry.level)).length
)
const warningCount = computed(() =>
filteredSystemLogEntries.value.filter((entry) => ['WARNING', 'WARN'].includes(entry.level)).length
)
const infoCount = computed(() =>
filteredSystemLogEntries.value.filter((entry) => entry.level === 'INFO').length
)
const totalPages = computed(() => Math.max(1, Math.ceil(totalCount.value / pageSize.value)))
const visiblePageItems = computed(() => {
if (totalPages.value <= 6) {
return Array.from({ length: totalPages.value }, (_, index) => index + 1)
}
return [1, 2, 3, 4, 5, 'ellipsis', totalPages.value]
})
const visibleSystemLogEntries = computed(() => {
const start = (currentPage.value - 1) * pageSize.value
return filteredSystemLogEntries.value.slice(start, start + pageSize.value)
})
function changePageSize(size) {
pageSize.value = size
currentPage.value = 1
}
function toggleFilter(key) {
openFilterKey.value = openFilterKey.value === key ? '' : key
}
function selectLevelFilter(value) {
systemLevelFilter.value = value
openFilterKey.value = ''
}
function selectEventTypeFilter(value) {
systemEventTypeFilter.value = value
openFilterKey.value = ''
}
function resetFilters() {
systemSearchKeyword.value = ''
systemLevelFilter.value = ''
systemEventTypeFilter.value = ''
openFilterKey.value = ''
}
async function loadSystemLogs(showToast = false) {
if (!isAdmin.value) {
return
}
systemLogLoading.value = true
try {
const payload = await fetchSystemLogEntries(300)
systemLogEntries.value = Array.isArray(payload) ? payload : []
} catch (error) {
if (showToast) {
toast(error.message || '系统日志加载失败。')
}
} finally {
systemLogLoading.value = false
}
}
function selectSystemLog(entryId) {
router.push({
name: 'app-log-detail',
params: { logKind: 'system', logId: entryId }
})
}
function startPolling() {
stopPolling()
pollTimer = window.setInterval(() => {
loadSystemLogs(false)
}, AGENT_RUN_POLL_INTERVAL_MS)
}
function stopPolling() {
if (pollTimer) {
window.clearInterval(pollTimer)
pollTimer = 0
}
}
watch(
[systemSearchKeyword, systemLevelFilter, systemEventTypeFilter],
() => {
currentPage.value = 1
}
)
watch(totalPages, (value) => {
if (currentPage.value > value) {
currentPage.value = value
}
})
watch(
() => [totalCount.value, errorCount.value, warningCount.value, infoCount.value],
([total, errors, warnings, info]) => {
emit('summary-change', { total, errors, warnings, info })
},
{ immediate: true }
)
onMounted(async () => {
await loadSystemLogs(false)
startPolling()
})
onBeforeUnmount(() => {
stopPolling()
})
return {
changePageSize,
currentPage,
filteredSystemLogEntries,
formatDateTime,
formatSummary,
hasActiveFilters,
isAdmin,
loadSystemLogs,
openFilterKey,
pageSize,
pageSizeOptions,
resetFilters,
resolveSystemLevelTone,
resolveSystemOutcomeTone,
selectEventTypeFilter,
selectLevelFilter,
selectSystemLog,
toggleFilter,
systemEventTypeFilter,
systemEventTypeFilterLabel,
systemEventTypeFilterOptions,
systemEventTypeOptions,
systemLevelFilter,
systemLevelFilterLabel,
systemLevelFilterOptions,
systemLevelOptions,
systemLogEntries,
systemLogLoading,
systemSearchKeyword,
totalCount,
totalPages,
visiblePageItems,
visibleSystemLogEntries
}
}
}