feat(ops): 优化警报规则和设置的成功提示信息
- 添加警报规则保存成功提示:"警报规则保存成功" - 添加警报规则删除成功提示:"警报规则删除成功" - 添加运维监控设置保存成功提示:"运维监控设置保存成功" - 替换通用的"操作成功"提示为具体的业务提示 - 失败时显示后端返回的详细错误信息 相关文件: - frontend/src/i18n/locales/zh.ts - frontend/src/views/admin/ops/components/OpsAlertRulesCard.vue - frontend/src/views/admin/ops/components/OpsSettingsDialog.vue
This commit is contained in:
@@ -136,7 +136,7 @@ async function save() {
|
||||
draft.value = null
|
||||
editingId.value = null
|
||||
await load()
|
||||
appStore.showSuccess(t('common.success'))
|
||||
appStore.showSuccess(t('admin.ops.alertRules.saveSuccess'))
|
||||
} catch (err: any) {
|
||||
console.error('[OpsAlertRulesCard] Failed to save rule', err)
|
||||
appStore.showError(err?.response?.data?.detail || t('admin.ops.alertRules.saveFailed'))
|
||||
@@ -160,7 +160,7 @@ async function confirmDelete() {
|
||||
showDeleteConfirm.value = false
|
||||
pendingDelete.value = null
|
||||
await load()
|
||||
appStore.showSuccess(t('common.success'))
|
||||
appStore.showSuccess(t('admin.ops.alertRules.deleteSuccess'))
|
||||
} catch (err: any) {
|
||||
console.error('[OpsAlertRulesCard] Failed to delete rule', err)
|
||||
appStore.showError(err?.response?.data?.detail || t('admin.ops.alertRules.deleteFailed'))
|
||||
|
||||
395
frontend/src/views/admin/ops/components/OpsSettingsDialog.vue
Normal file
395
frontend/src/views/admin/ops/components/OpsSettingsDialog.vue
Normal file
@@ -0,0 +1,395 @@
|
||||
<script setup lang="ts">
|
||||
import { ref, computed, watch } from 'vue'
|
||||
import { useI18n } from 'vue-i18n'
|
||||
import { useAppStore } from '@/stores/app'
|
||||
import { opsAPI } from '@/api/admin/ops'
|
||||
import BaseDialog from '@/components/common/BaseDialog.vue'
|
||||
import Select from '@/components/common/Select.vue'
|
||||
import Toggle from '@/components/common/Toggle.vue'
|
||||
import type { OpsAlertRuntimeSettings, EmailNotificationConfig, AlertSeverity, OpsAdvancedSettings } from '../types'
|
||||
|
||||
const { t } = useI18n()
|
||||
const appStore = useAppStore()
|
||||
|
||||
const props = defineProps<{
|
||||
show: boolean
|
||||
}>()
|
||||
|
||||
const emit = defineEmits<{
|
||||
close: []
|
||||
saved: []
|
||||
}>()
|
||||
|
||||
const loading = ref(false)
|
||||
const saving = ref(false)
|
||||
|
||||
// 运行时设置
|
||||
const runtimeSettings = ref<OpsAlertRuntimeSettings | null>(null)
|
||||
// 邮件通知配置
|
||||
const emailConfig = ref<EmailNotificationConfig | null>(null)
|
||||
// 高级设置
|
||||
const advancedSettings = ref<OpsAdvancedSettings | null>(null)
|
||||
|
||||
// 加载所有配置
|
||||
async function loadAllSettings() {
|
||||
loading.value = true
|
||||
try {
|
||||
const [runtime, email, advanced] = await Promise.all([
|
||||
opsAPI.getAlertRuntimeSettings(),
|
||||
opsAPI.getEmailNotificationConfig(),
|
||||
opsAPI.getAdvancedSettings()
|
||||
])
|
||||
runtimeSettings.value = runtime
|
||||
emailConfig.value = email
|
||||
advancedSettings.value = advanced
|
||||
} catch (err: any) {
|
||||
console.error('[OpsSettingsDialog] Failed to load settings', err)
|
||||
appStore.showError(err?.response?.data?.detail || t('admin.ops.settings.loadFailed'))
|
||||
} finally {
|
||||
loading.value = false
|
||||
}
|
||||
}
|
||||
|
||||
// 监听弹窗打开
|
||||
watch(() => props.show, (show) => {
|
||||
if (show) {
|
||||
loadAllSettings()
|
||||
}
|
||||
})
|
||||
|
||||
// 邮件输入
|
||||
const alertRecipientInput = ref('')
|
||||
const reportRecipientInput = ref('')
|
||||
|
||||
// 严重级别选项
|
||||
const severityOptions: Array<{ value: AlertSeverity | ''; label: string }> = [
|
||||
{ value: '', label: t('admin.ops.email.minSeverityAll') },
|
||||
{ value: 'critical', label: t('common.critical') },
|
||||
{ value: 'warning', label: t('common.warning') },
|
||||
{ value: 'info', label: t('common.info') }
|
||||
]
|
||||
|
||||
// 验证邮箱
|
||||
function isValidEmailAddress(email: string): boolean {
|
||||
return /^[^\s@]+@[^\s@]+\.[^\s@]+$/.test(email)
|
||||
}
|
||||
|
||||
// 添加收件人
|
||||
function addRecipient(target: 'alert' | 'report') {
|
||||
if (!emailConfig.value) return
|
||||
const raw = (target === 'alert' ? alertRecipientInput.value : reportRecipientInput.value).trim()
|
||||
if (!raw) return
|
||||
|
||||
if (!isValidEmailAddress(raw)) {
|
||||
appStore.showError(t('common.invalidEmail'))
|
||||
return
|
||||
}
|
||||
|
||||
const normalized = raw.toLowerCase()
|
||||
const list = target === 'alert' ? emailConfig.value.alert.recipients : emailConfig.value.report.recipients
|
||||
if (!list.includes(normalized)) {
|
||||
list.push(normalized)
|
||||
}
|
||||
if (target === 'alert') alertRecipientInput.value = ''
|
||||
else reportRecipientInput.value = ''
|
||||
}
|
||||
|
||||
// 移除收件人
|
||||
function removeRecipient(target: 'alert' | 'report', email: string) {
|
||||
if (!emailConfig.value) return
|
||||
const list = target === 'alert' ? emailConfig.value.alert.recipients : emailConfig.value.report.recipients
|
||||
const idx = list.indexOf(email)
|
||||
if (idx >= 0) list.splice(idx, 1)
|
||||
}
|
||||
|
||||
// 验证
|
||||
const validation = computed(() => {
|
||||
const errors: string[] = []
|
||||
|
||||
// 验证运行时设置
|
||||
if (runtimeSettings.value) {
|
||||
const evalSeconds = runtimeSettings.value.evaluation_interval_seconds
|
||||
if (!Number.isFinite(evalSeconds) || evalSeconds < 1 || evalSeconds > 86400) {
|
||||
errors.push(t('admin.ops.runtime.validation.evalIntervalRange'))
|
||||
}
|
||||
}
|
||||
|
||||
// 验证邮件配置
|
||||
if (emailConfig.value) {
|
||||
if (emailConfig.value.alert.enabled && emailConfig.value.alert.recipients.length === 0) {
|
||||
errors.push(t('admin.ops.email.validation.alertRecipientsRequired'))
|
||||
}
|
||||
if (emailConfig.value.report.enabled && emailConfig.value.report.recipients.length === 0) {
|
||||
errors.push(t('admin.ops.email.validation.reportRecipientsRequired'))
|
||||
}
|
||||
}
|
||||
|
||||
// 验证高级设置
|
||||
if (advancedSettings.value) {
|
||||
const { error_log_retention_days, minute_metrics_retention_days, hourly_metrics_retention_days } = advancedSettings.value.data_retention
|
||||
if (error_log_retention_days < 1 || error_log_retention_days > 365) {
|
||||
errors.push(t('admin.ops.settings.validation.retentionDaysRange'))
|
||||
}
|
||||
if (minute_metrics_retention_days < 1 || minute_metrics_retention_days > 365) {
|
||||
errors.push(t('admin.ops.settings.validation.retentionDaysRange'))
|
||||
}
|
||||
if (hourly_metrics_retention_days < 1 || hourly_metrics_retention_days > 365) {
|
||||
errors.push(t('admin.ops.settings.validation.retentionDaysRange'))
|
||||
}
|
||||
}
|
||||
|
||||
return { valid: errors.length === 0, errors }
|
||||
})
|
||||
|
||||
// 保存所有配置
|
||||
async function saveAllSettings() {
|
||||
if (!validation.value.valid) {
|
||||
appStore.showError(validation.value.errors[0])
|
||||
return
|
||||
}
|
||||
|
||||
saving.value = true
|
||||
try {
|
||||
await Promise.all([
|
||||
runtimeSettings.value ? opsAPI.updateAlertRuntimeSettings(runtimeSettings.value) : Promise.resolve(),
|
||||
emailConfig.value ? opsAPI.updateEmailNotificationConfig(emailConfig.value) : Promise.resolve(),
|
||||
advancedSettings.value ? opsAPI.updateAdvancedSettings(advancedSettings.value) : Promise.resolve()
|
||||
])
|
||||
appStore.showSuccess(t('admin.ops.settings.saveSuccess'))
|
||||
emit('saved')
|
||||
emit('close')
|
||||
} catch (err: any) {
|
||||
console.error('[OpsSettingsDialog] Failed to save settings', err)
|
||||
appStore.showError(err?.response?.data?.detail || t('admin.ops.settings.saveFailed'))
|
||||
} finally {
|
||||
saving.value = false
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<BaseDialog :show="show" :title="t('admin.ops.settings.title')" width="extra-wide" @close="emit('close')">
|
||||
<div v-if="loading" class="py-10 text-center text-sm text-gray-500">
|
||||
{{ t('common.loading') }}
|
||||
</div>
|
||||
|
||||
<div v-else-if="runtimeSettings && emailConfig && advancedSettings" class="space-y-6">
|
||||
<!-- 验证错误 -->
|
||||
<div v-if="!validation.valid" class="rounded-lg border border-amber-200 bg-amber-50 p-3 text-xs text-amber-800 dark:border-amber-900/50 dark:bg-amber-900/20 dark:text-amber-200">
|
||||
<div class="font-bold">{{ t('admin.ops.settings.validation.title') }}</div>
|
||||
<ul class="mt-1 list-disc space-y-1 pl-4">
|
||||
<li v-for="msg in validation.errors" :key="msg">{{ msg }}</li>
|
||||
</ul>
|
||||
</div>
|
||||
|
||||
<!-- 数据采集频率 -->
|
||||
<div class="rounded-2xl bg-gray-50 p-4 dark:bg-dark-700/50">
|
||||
<h4 class="mb-3 text-sm font-semibold text-gray-900 dark:text-white">{{ t('admin.ops.settings.dataCollection') }}</h4>
|
||||
<div>
|
||||
<label class="input-label">{{ t('admin.ops.settings.evaluationInterval') }}</label>
|
||||
<input
|
||||
v-model.number="runtimeSettings.evaluation_interval_seconds"
|
||||
type="number"
|
||||
min="1"
|
||||
max="86400"
|
||||
class="input"
|
||||
/>
|
||||
<p class="mt-1 text-xs text-gray-500">{{ t('admin.ops.settings.evaluationIntervalHint') }}</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- 预警配置 -->
|
||||
<div class="rounded-2xl bg-gray-50 p-4 dark:bg-dark-700/50">
|
||||
<h4 class="mb-3 text-sm font-semibold text-gray-900 dark:text-white">{{ t('admin.ops.settings.alertConfig') }}</h4>
|
||||
|
||||
<div class="space-y-4">
|
||||
<div class="flex items-center justify-between">
|
||||
<div>
|
||||
<label class="font-medium text-gray-900 dark:text-white">{{ t('admin.ops.settings.enableAlert') }}</label>
|
||||
</div>
|
||||
<Toggle v-model="emailConfig.alert.enabled" />
|
||||
</div>
|
||||
|
||||
<div v-if="emailConfig.alert.enabled">
|
||||
<label class="input-label">{{ t('admin.ops.settings.alertRecipients') }}</label>
|
||||
<div class="flex gap-2">
|
||||
<input
|
||||
v-model="alertRecipientInput"
|
||||
type="email"
|
||||
class="input"
|
||||
:placeholder="t('admin.ops.settings.emailPlaceholder')"
|
||||
@keydown.enter.prevent="addRecipient('alert')"
|
||||
/>
|
||||
<button class="btn btn-secondary whitespace-nowrap" type="button" @click="addRecipient('alert')">
|
||||
{{ t('common.add') }}
|
||||
</button>
|
||||
</div>
|
||||
<div class="mt-2 flex flex-wrap gap-2">
|
||||
<span
|
||||
v-for="email in emailConfig.alert.recipients"
|
||||
:key="email"
|
||||
class="inline-flex items-center gap-2 rounded-full bg-blue-100 px-3 py-1 text-xs font-medium text-blue-700 dark:bg-blue-900/30 dark:text-blue-400"
|
||||
>
|
||||
{{ email }}
|
||||
<button type="button" class="text-blue-700/80 hover:text-blue-900" @click="removeRecipient('alert', email)">×</button>
|
||||
</span>
|
||||
</div>
|
||||
<p class="mt-2 text-xs text-gray-500 dark:text-gray-400">
|
||||
{{ t('admin.ops.settings.recipientsHint') }}
|
||||
</p>
|
||||
</div>
|
||||
|
||||
<div v-if="emailConfig.alert.enabled">
|
||||
<label class="input-label">{{ t('admin.ops.settings.minSeverity') }}</label>
|
||||
<Select v-model="emailConfig.alert.min_severity" :options="severityOptions" />
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- 评估报告配置 -->
|
||||
<div class="rounded-2xl bg-gray-50 p-4 dark:bg-dark-700/50">
|
||||
<h4 class="mb-3 text-sm font-semibold text-gray-900 dark:text-white">{{ t('admin.ops.settings.reportConfig') }}</h4>
|
||||
|
||||
<div class="space-y-4">
|
||||
<div class="flex items-center justify-between">
|
||||
<div>
|
||||
<label class="font-medium text-gray-900 dark:text-white">{{ t('admin.ops.settings.enableReport') }}</label>
|
||||
</div>
|
||||
<Toggle v-model="emailConfig.report.enabled" />
|
||||
</div>
|
||||
|
||||
<div v-if="emailConfig.report.enabled">
|
||||
<label class="input-label">{{ t('admin.ops.settings.reportRecipients') }}</label>
|
||||
<div class="flex gap-2">
|
||||
<input
|
||||
v-model="reportRecipientInput"
|
||||
type="email"
|
||||
class="input"
|
||||
:placeholder="t('admin.ops.settings.emailPlaceholder')"
|
||||
@keydown.enter.prevent="addRecipient('report')"
|
||||
/>
|
||||
<button class="btn btn-secondary whitespace-nowrap" type="button" @click="addRecipient('report')">
|
||||
{{ t('common.add') }}
|
||||
</button>
|
||||
</div>
|
||||
<div class="mt-2 flex flex-wrap gap-2">
|
||||
<span
|
||||
v-for="email in emailConfig.report.recipients"
|
||||
:key="email"
|
||||
class="inline-flex items-center gap-2 rounded-full bg-blue-100 px-3 py-1 text-xs font-medium text-blue-700 dark:bg-blue-900/30 dark:text-blue-400"
|
||||
>
|
||||
{{ email }}
|
||||
<button type="button" class="text-blue-700/80 hover:text-blue-900" @click="removeRecipient('report', email)">×</button>
|
||||
</span>
|
||||
</div>
|
||||
<p class="mt-2 text-xs text-gray-500 dark:text-gray-400">
|
||||
{{ t('admin.ops.settings.recipientsHint') }}
|
||||
</p>
|
||||
</div>
|
||||
|
||||
<div v-if="emailConfig.report.enabled" class="grid grid-cols-1 gap-4 md:grid-cols-2">
|
||||
<div class="flex items-center justify-between">
|
||||
<label class="text-sm font-medium text-gray-700 dark:text-gray-300">{{ t('admin.ops.settings.dailySummary') }}</label>
|
||||
<Toggle v-model="emailConfig.report.daily_summary_enabled" />
|
||||
</div>
|
||||
<div v-if="emailConfig.report.daily_summary_enabled">
|
||||
<input v-model="emailConfig.report.daily_summary_schedule" type="text" class="input" placeholder="0 9 * * *" />
|
||||
</div>
|
||||
<div class="flex items-center justify-between">
|
||||
<label class="text-sm font-medium text-gray-700 dark:text-gray-300">{{ t('admin.ops.settings.weeklySummary') }}</label>
|
||||
<Toggle v-model="emailConfig.report.weekly_summary_enabled" />
|
||||
</div>
|
||||
<div v-if="emailConfig.report.weekly_summary_enabled">
|
||||
<input v-model="emailConfig.report.weekly_summary_schedule" type="text" class="input" placeholder="0 9 * * 1" />
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- 高级设置 -->
|
||||
<details class="rounded-2xl bg-gray-50 dark:bg-dark-700/50">
|
||||
<summary class="cursor-pointer p-4 text-sm font-semibold text-gray-900 dark:text-white">
|
||||
{{ t('admin.ops.settings.advancedSettings') }}
|
||||
</summary>
|
||||
<div class="space-y-4 px-4 pb-4">
|
||||
<!-- 数据保留策略 -->
|
||||
<div class="space-y-3">
|
||||
<h5 class="text-xs font-semibold text-gray-700 dark:text-gray-300">{{ t('admin.ops.settings.dataRetention') }}</h5>
|
||||
|
||||
<div class="flex items-center justify-between">
|
||||
<label class="text-sm font-medium text-gray-700 dark:text-gray-300">{{ t('admin.ops.settings.enableCleanup') }}</label>
|
||||
<Toggle v-model="advancedSettings.data_retention.cleanup_enabled" />
|
||||
</div>
|
||||
|
||||
<div v-if="advancedSettings.data_retention.cleanup_enabled">
|
||||
<label class="input-label">{{ t('admin.ops.settings.cleanupSchedule') }}</label>
|
||||
<input
|
||||
v-model="advancedSettings.data_retention.cleanup_schedule"
|
||||
type="text"
|
||||
class="input"
|
||||
placeholder="0 2 * * *"
|
||||
/>
|
||||
<p class="mt-1 text-xs text-gray-500">{{ t('admin.ops.settings.cleanupScheduleHint') }}</p>
|
||||
</div>
|
||||
|
||||
<div class="grid grid-cols-1 gap-4 md:grid-cols-3">
|
||||
<div>
|
||||
<label class="input-label">{{ t('admin.ops.settings.errorLogRetentionDays') }}</label>
|
||||
<input
|
||||
v-model.number="advancedSettings.data_retention.error_log_retention_days"
|
||||
type="number"
|
||||
min="1"
|
||||
max="365"
|
||||
class="input"
|
||||
/>
|
||||
</div>
|
||||
<div>
|
||||
<label class="input-label">{{ t('admin.ops.settings.minuteMetricsRetentionDays') }}</label>
|
||||
<input
|
||||
v-model.number="advancedSettings.data_retention.minute_metrics_retention_days"
|
||||
type="number"
|
||||
min="1"
|
||||
max="365"
|
||||
class="input"
|
||||
/>
|
||||
</div>
|
||||
<div>
|
||||
<label class="input-label">{{ t('admin.ops.settings.hourlyMetricsRetentionDays') }}</label>
|
||||
<input
|
||||
v-model.number="advancedSettings.data_retention.hourly_metrics_retention_days"
|
||||
type="number"
|
||||
min="1"
|
||||
max="365"
|
||||
class="input"
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
<p class="text-xs text-gray-500">{{ t('admin.ops.settings.retentionDaysHint') }}</p>
|
||||
</div>
|
||||
|
||||
<!-- 预聚合任务 -->
|
||||
<div class="space-y-3">
|
||||
<h5 class="text-xs font-semibold text-gray-700 dark:text-gray-300">{{ t('admin.ops.settings.aggregation') }}</h5>
|
||||
|
||||
<div class="flex items-center justify-between">
|
||||
<div>
|
||||
<label class="text-sm font-medium text-gray-700 dark:text-gray-300">{{ t('admin.ops.settings.enableAggregation') }}</label>
|
||||
<p class="mt-1 text-xs text-gray-500">{{ t('admin.ops.settings.aggregationHint') }}</p>
|
||||
</div>
|
||||
<Toggle v-model="advancedSettings.aggregation.aggregation_enabled" />
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</details>
|
||||
</div>
|
||||
|
||||
<template #footer>
|
||||
<div class="flex justify-end gap-2">
|
||||
<button class="btn btn-secondary" @click="emit('close')">{{ t('common.cancel') }}</button>
|
||||
<button class="btn btn-primary" :disabled="saving || !validation.valid" @click="saveAllSettings">
|
||||
{{ saving ? t('common.saving') : t('common.save') }}
|
||||
</button>
|
||||
</div>
|
||||
</template>
|
||||
</BaseDialog>
|
||||
</template>
|
||||
Reference in New Issue
Block a user