2026-05-01 21:27:39 +08:00
|
|
|
|
<template>
|
|
|
|
|
|
<div>
|
|
|
|
|
|
<div class="page-header">
|
|
|
|
|
|
<div>
|
|
|
|
|
|
<h2>安全中心</h2>
|
|
|
|
|
|
<p class="subtitle">管理登录安全、应用密钥和子账号风险控制。</p>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<el-button @click="$router.push('/accounts')">子账号管理</el-button>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<el-row :gutter="16" style="margin-bottom: 16px">
|
|
|
|
|
|
<el-col :xs="24" :md="8">
|
|
|
|
|
|
<el-card shadow="hover" class="summary-card">
|
|
|
|
|
|
<el-statistic title="当前账号" :value="auth.user?.nickname || auth.user?.username || '-'" />
|
|
|
|
|
|
</el-card>
|
|
|
|
|
|
</el-col>
|
|
|
|
|
|
<el-col :xs="24" :md="8">
|
|
|
|
|
|
<el-card shadow="hover" class="summary-card">
|
|
|
|
|
|
<el-statistic title="应用数量" :value="apps.length" />
|
|
|
|
|
|
</el-card>
|
|
|
|
|
|
</el-col>
|
|
|
|
|
|
<el-col :xs="24" :md="8">
|
|
|
|
|
|
<el-card shadow="hover" class="summary-card">
|
|
|
|
|
|
<el-statistic title="子账号数量" :value="subAccountCount" />
|
|
|
|
|
|
</el-card>
|
|
|
|
|
|
</el-col>
|
|
|
|
|
|
</el-row>
|
|
|
|
|
|
|
|
|
|
|
|
<el-card style="margin-bottom: 16px">
|
|
|
|
|
|
<template #header>账号保护建议</template>
|
|
|
|
|
|
<el-space wrap>
|
|
|
|
|
|
<el-tag type="success">强密码</el-tag>
|
|
|
|
|
|
<el-tag type="warning">邮箱验证</el-tag>
|
|
|
|
|
|
<el-tag type="info">子账号最小权限</el-tag>
|
|
|
|
|
|
<el-tag type="info">密钥定期轮换</el-tag>
|
|
|
|
|
|
</el-space>
|
|
|
|
|
|
<el-divider />
|
|
|
|
|
|
<el-button type="primary" plain @click="$router.push('/forgot-password')">重置当前账号密码</el-button>
|
|
|
|
|
|
</el-card>
|
|
|
|
|
|
|
2026-05-19 15:12:56 +08:00
|
|
|
|
<el-card style="margin-bottom: 16px">
|
|
|
|
|
|
<template #header>私有化部署迁移</template>
|
|
|
|
|
|
<p style="color: #606266; margin: 0 0 16px;">
|
|
|
|
|
|
将当前账号及应用数据迁移至私有化部署环境。迁移密钥仅在生成时显示一次,请及时保存。
|
|
|
|
|
|
</p>
|
|
|
|
|
|
<el-button type="warning" plain @click="openMigrateDialog">生成迁移密钥</el-button>
|
|
|
|
|
|
</el-card>
|
|
|
|
|
|
|
2026-05-01 21:27:39 +08:00
|
|
|
|
<el-card>
|
|
|
|
|
|
<template #header>应用密钥管理</template>
|
|
|
|
|
|
<el-table :data="apps" v-loading="loading" border stripe>
|
|
|
|
|
|
<el-table-column prop="name" label="应用名称" min-width="160" />
|
|
|
|
|
|
<el-table-column prop="packageName" label="包名" min-width="180" />
|
|
|
|
|
|
<el-table-column prop="appKey" label="AppKey" min-width="220" show-overflow-tooltip />
|
|
|
|
|
|
<el-table-column prop="createdAt" label="创建时间" width="180">
|
|
|
|
|
|
<template #default="{ row }">{{ fmt(row.createdAt) }}</template>
|
|
|
|
|
|
</el-table-column>
|
|
|
|
|
|
<el-table-column label="操作" width="220" fixed="right">
|
|
|
|
|
|
<template #default="{ row }">
|
|
|
|
|
|
<el-button link type="primary" @click="openSecretDialog(row, 'REVEAL_SECRET')">查看密钥</el-button>
|
|
|
|
|
|
<el-button link type="warning" @click="openSecretDialog(row, 'RESET_SECRET')">重置密钥</el-button>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
</el-table-column>
|
|
|
|
|
|
</el-table>
|
|
|
|
|
|
</el-card>
|
|
|
|
|
|
|
|
|
|
|
|
<el-dialog
|
|
|
|
|
|
v-model="showDialog"
|
|
|
|
|
|
:title="dialogMode === 'REVEAL_SECRET' ? '查看 AppSecret' : '重置 AppSecret'"
|
|
|
|
|
|
width="460px"
|
|
|
|
|
|
@closed="closeDialog"
|
|
|
|
|
|
>
|
|
|
|
|
|
<div v-if="!codeSent">
|
|
|
|
|
|
<p class="dialog-text">
|
|
|
|
|
|
{{ dialogMode === 'REVEAL_SECRET'
|
|
|
|
|
|
? '系统将向租户邮箱发送验证码,验证后可查看密钥。'
|
|
|
|
|
|
: '系统将向租户邮箱发送验证码,验证后会立即重置密钥。' }}
|
|
|
|
|
|
</p>
|
|
|
|
|
|
<el-button type="primary" :loading="sendingCode" @click="sendVerifyCode">发送验证码</el-button>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div v-else>
|
|
|
|
|
|
<p class="dialog-text">请输入邮箱验证码:</p>
|
|
|
|
|
|
<el-input v-model="verifyCode" maxlength="6" placeholder="6位验证码" />
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<template #footer>
|
|
|
|
|
|
<el-button @click="showDialog = false">取消</el-button>
|
|
|
|
|
|
<el-button v-if="codeSent" type="primary" :loading="submitting" @click="submitVerify">
|
|
|
|
|
|
确认
|
|
|
|
|
|
</el-button>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
</el-dialog>
|
|
|
|
|
|
|
2026-05-19 15:12:56 +08:00
|
|
|
|
<!-- Migration: step 1 — request email code -->
|
|
|
|
|
|
<el-dialog v-model="showMigrateDialog" title="私有化部署迁移" width="460px" @closed="closeMigrateDialog">
|
|
|
|
|
|
<div v-if="!migrateCodeSent">
|
|
|
|
|
|
<p class="dialog-text">点击发送验证码,系统将向您的注册邮箱发送一次性验证码,验证通过后生成迁移密钥。</p>
|
|
|
|
|
|
<el-button type="primary" :loading="migrateSendingCode" @click="sendMigrateCode">发送验证码</el-button>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div v-else>
|
|
|
|
|
|
<p class="dialog-text">请输入邮箱验证码:</p>
|
|
|
|
|
|
<el-input v-model="migrateCode" maxlength="6" placeholder="6位验证码" />
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<template #footer>
|
|
|
|
|
|
<el-button @click="showMigrateDialog = false">取消</el-button>
|
|
|
|
|
|
<el-button v-if="migrateCodeSent" type="primary" :loading="migrateSubmitting" @click="submitMigrateCode">
|
|
|
|
|
|
生成密钥
|
|
|
|
|
|
</el-button>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
</el-dialog>
|
|
|
|
|
|
|
|
|
|
|
|
<!-- Migration: step 2 — display one-time key -->
|
|
|
|
|
|
<el-dialog
|
|
|
|
|
|
v-model="showMigrateKey"
|
|
|
|
|
|
title="迁移密钥"
|
|
|
|
|
|
width="480px"
|
|
|
|
|
|
:close-on-click-modal="false"
|
|
|
|
|
|
:close-on-press-escape="false"
|
|
|
|
|
|
@closed="migrateKeyValue = ''"
|
|
|
|
|
|
>
|
|
|
|
|
|
<el-alert type="warning" :closable="false" show-icon style="margin-bottom: 16px;">
|
|
|
|
|
|
<template #title>密钥只显示一次,关闭后无法再次查看</template>
|
|
|
|
|
|
<template #default>如果忘记,请重新生成新的迁移密钥。</template>
|
|
|
|
|
|
</el-alert>
|
|
|
|
|
|
<div class="migrate-key-box">{{ migrateKeyValue }}</div>
|
|
|
|
|
|
<el-button type="primary" plain size="small" style="margin-top: 12px;" @click="copyMigrateKey">
|
|
|
|
|
|
复制密钥
|
|
|
|
|
|
</el-button>
|
|
|
|
|
|
<template #footer>
|
|
|
|
|
|
<el-button type="danger" @click="showMigrateKey = false">我已保存,关闭</el-button>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
</el-dialog>
|
|
|
|
|
|
|
2026-05-01 21:27:39 +08:00
|
|
|
|
<el-dialog v-model="showResult" title="AppSecret" width="420px">
|
|
|
|
|
|
<el-alert type="success" :closable="false" show-icon>
|
|
|
|
|
|
<template #title>操作已完成</template>
|
|
|
|
|
|
<template #default>
|
|
|
|
|
|
<div class="secret-box">{{ secretResult || '无结果' }}</div>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
</el-alert>
|
|
|
|
|
|
<template #footer>
|
|
|
|
|
|
<el-button type="primary" @click="showResult = false">知道了</el-button>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
</el-dialog>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
|
|
|
|
|
|
<script setup lang="ts">
|
|
|
|
|
|
import { onMounted, ref } from 'vue'
|
|
|
|
|
|
import { ElMessage } from 'element-plus'
|
|
|
|
|
|
import { accountApi } from '@/api/account'
|
|
|
|
|
|
import { appApi, type App } from '@/api/app'
|
2026-05-19 15:12:56 +08:00
|
|
|
|
import { migrateApi } from '@/api/migrate'
|
2026-05-01 21:27:39 +08:00
|
|
|
|
import { useAuthStore } from '@/stores/auth'
|
|
|
|
|
|
|
|
|
|
|
|
const auth = useAuthStore()
|
|
|
|
|
|
const apps = ref<App[]>([])
|
|
|
|
|
|
const loading = ref(false)
|
|
|
|
|
|
const subAccountCount = ref(0)
|
|
|
|
|
|
|
|
|
|
|
|
const showDialog = ref(false)
|
|
|
|
|
|
const showResult = ref(false)
|
|
|
|
|
|
const sendingCode = ref(false)
|
|
|
|
|
|
const submitting = ref(false)
|
|
|
|
|
|
const codeSent = ref(false)
|
|
|
|
|
|
const verifyCode = ref('')
|
|
|
|
|
|
const selectedApp = ref<App | null>(null)
|
|
|
|
|
|
const dialogMode = ref<'REVEAL_SECRET' | 'RESET_SECRET'>('REVEAL_SECRET')
|
|
|
|
|
|
const secretResult = ref('')
|
|
|
|
|
|
|
|
|
|
|
|
async function loadData() {
|
|
|
|
|
|
loading.value = true
|
|
|
|
|
|
try {
|
|
|
|
|
|
const [appsRes, subRes] = await Promise.all([appApi.list(), accountApi.list()])
|
|
|
|
|
|
apps.value = appsRes.data.data
|
|
|
|
|
|
subAccountCount.value = subRes.data.data.length
|
|
|
|
|
|
} finally {
|
|
|
|
|
|
loading.value = false
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
function openSecretDialog(app: App, mode: 'REVEAL_SECRET' | 'RESET_SECRET') {
|
|
|
|
|
|
selectedApp.value = app
|
|
|
|
|
|
dialogMode.value = mode
|
|
|
|
|
|
showDialog.value = true
|
|
|
|
|
|
codeSent.value = false
|
|
|
|
|
|
verifyCode.value = ''
|
|
|
|
|
|
secretResult.value = ''
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
async function sendVerifyCode() {
|
|
|
|
|
|
if (!selectedApp.value) return
|
|
|
|
|
|
sendingCode.value = true
|
|
|
|
|
|
try {
|
2026-05-09 14:53:43 +08:00
|
|
|
|
await appApi.requestSecretVerify(selectedApp.value.appKey, dialogMode.value)
|
2026-05-01 21:27:39 +08:00
|
|
|
|
codeSent.value = true
|
|
|
|
|
|
ElMessage.success('验证码已发送到邮箱')
|
|
|
|
|
|
} finally {
|
|
|
|
|
|
sendingCode.value = false
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
async function submitVerify() {
|
|
|
|
|
|
if (!selectedApp.value) return
|
|
|
|
|
|
if (!verifyCode.value.trim()) {
|
|
|
|
|
|
ElMessage.warning('请输入验证码')
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
submitting.value = true
|
|
|
|
|
|
try {
|
|
|
|
|
|
if (dialogMode.value === 'REVEAL_SECRET') {
|
2026-05-09 14:53:43 +08:00
|
|
|
|
const res = await appApi.revealSecret(selectedApp.value.appKey, verifyCode.value.trim())
|
2026-05-01 21:27:39 +08:00
|
|
|
|
secretResult.value = res.data.data.appSecret
|
|
|
|
|
|
} else {
|
2026-05-09 14:53:43 +08:00
|
|
|
|
const res = await appApi.resetSecret(selectedApp.value.appKey, verifyCode.value.trim())
|
2026-05-01 21:27:39 +08:00
|
|
|
|
secretResult.value = res.data.data.appSecret
|
|
|
|
|
|
}
|
|
|
|
|
|
showDialog.value = false
|
|
|
|
|
|
showResult.value = true
|
|
|
|
|
|
ElMessage.success(dialogMode.value === 'REVEAL_SECRET' ? '密钥已查看' : '密钥已重置')
|
|
|
|
|
|
} finally {
|
|
|
|
|
|
submitting.value = false
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
function closeDialog() {
|
|
|
|
|
|
selectedApp.value = null
|
|
|
|
|
|
verifyCode.value = ''
|
|
|
|
|
|
codeSent.value = false
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-05-19 15:12:56 +08:00
|
|
|
|
// ── Migration ────────────────────────────────────────────────────────────────
|
|
|
|
|
|
|
|
|
|
|
|
const showMigrateDialog = ref(false)
|
|
|
|
|
|
const showMigrateKey = ref(false)
|
|
|
|
|
|
const migrateCodeSent = ref(false)
|
|
|
|
|
|
const migrateSendingCode = ref(false)
|
|
|
|
|
|
const migrateSubmitting = ref(false)
|
|
|
|
|
|
const migrateCode = ref('')
|
|
|
|
|
|
const migrateKeyValue = ref('')
|
|
|
|
|
|
|
|
|
|
|
|
function openMigrateDialog() {
|
|
|
|
|
|
migrateCodeSent.value = false
|
|
|
|
|
|
migrateCode.value = ''
|
|
|
|
|
|
showMigrateDialog.value = true
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
function closeMigrateDialog() {
|
|
|
|
|
|
migrateCode.value = ''
|
|
|
|
|
|
migrateCodeSent.value = false
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
async function sendMigrateCode() {
|
|
|
|
|
|
migrateSendingCode.value = true
|
|
|
|
|
|
try {
|
|
|
|
|
|
await migrateApi.requestCode()
|
|
|
|
|
|
migrateCodeSent.value = true
|
|
|
|
|
|
ElMessage.success('验证码已发送到邮箱')
|
|
|
|
|
|
} finally {
|
|
|
|
|
|
migrateSendingCode.value = false
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
async function submitMigrateCode() {
|
|
|
|
|
|
if (!migrateCode.value.trim()) {
|
|
|
|
|
|
ElMessage.warning('请输入验证码')
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
migrateSubmitting.value = true
|
|
|
|
|
|
try {
|
|
|
|
|
|
const res = await migrateApi.generateKey(migrateCode.value.trim())
|
|
|
|
|
|
migrateKeyValue.value = res.data.data.migrationKey
|
|
|
|
|
|
showMigrateDialog.value = false
|
|
|
|
|
|
showMigrateKey.value = true
|
|
|
|
|
|
} finally {
|
|
|
|
|
|
migrateSubmitting.value = false
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
async function copyMigrateKey() {
|
|
|
|
|
|
try {
|
|
|
|
|
|
await navigator.clipboard.writeText(migrateKeyValue.value)
|
|
|
|
|
|
ElMessage.success('已复制到剪贴板')
|
|
|
|
|
|
} catch {
|
|
|
|
|
|
ElMessage.warning('复制失败,请手动选择并复制')
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-05-01 21:27:39 +08:00
|
|
|
|
function fmt(value: string) {
|
|
|
|
|
|
return value ? new Date(value).toLocaleString('zh-CN') : '-'
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
onMounted(loadData)
|
|
|
|
|
|
</script>
|
|
|
|
|
|
|
|
|
|
|
|
<style scoped>
|
|
|
|
|
|
.page-header {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
justify-content: space-between;
|
|
|
|
|
|
align-items: flex-start;
|
|
|
|
|
|
gap: 16px;
|
|
|
|
|
|
margin-bottom: 16px;
|
|
|
|
|
|
}
|
|
|
|
|
|
.subtitle {
|
|
|
|
|
|
margin: 6px 0 0;
|
|
|
|
|
|
color: #606266;
|
|
|
|
|
|
}
|
|
|
|
|
|
.summary-card {
|
|
|
|
|
|
min-height: 110px;
|
|
|
|
|
|
}
|
|
|
|
|
|
.dialog-text {
|
|
|
|
|
|
color: #606266;
|
|
|
|
|
|
margin: 0 0 16px;
|
|
|
|
|
|
}
|
|
|
|
|
|
.secret-box {
|
|
|
|
|
|
margin-top: 8px;
|
|
|
|
|
|
font-family: monospace;
|
|
|
|
|
|
word-break: break-all;
|
|
|
|
|
|
}
|
2026-05-19 15:12:56 +08:00
|
|
|
|
.migrate-key-box {
|
|
|
|
|
|
font-family: monospace;
|
|
|
|
|
|
font-size: 14px;
|
|
|
|
|
|
word-break: break-all;
|
|
|
|
|
|
background: #f5f7fa;
|
|
|
|
|
|
border: 1px solid #dcdfe6;
|
|
|
|
|
|
border-radius: 4px;
|
|
|
|
|
|
padding: 12px 16px;
|
|
|
|
|
|
line-height: 1.6;
|
|
|
|
|
|
user-select: all;
|
|
|
|
|
|
}
|
2026-05-01 21:27:39 +08:00
|
|
|
|
</style>
|