- .env.example: API基础路径改为相对路径 /api/v1,支持代理转发 - README.md: 完善项目结构与启动说明文档 - docker-compose.yml: 新增Docker编排配置,支持容器化部署 - docker/: 新增Docker部署相关文档与配置 - server_start.sh: 重构启动脚本,添加容器环境检测、隔离虚拟环境路径、环境变量覆盖机制 - deps.py: 完善API依赖注入,增强权限验证逻辑 - admin_secret.py: 优化管理员密钥加密存储与验证 - config.py: 扩展配置管理,支持多环境变量绑定 - security.py: 增强安全模块,完善加密与认证机制 - db/base.py: 优化数据库基础架构与连接管理 - main.py: 更新应用入口,整合新模块路由 - models/: 完善系统模型配置,支持模型设置持久化 - repositories/settings.py: 优化设置仓储层,增强数据持久化 - services/settings.py: 重构设置服务,精简代码结构 - router.py: 更新API路由配置 - endpoints/knowledge.py: 新增知识库API端点 - schemas/knowledge.py: 新增知识库数据模型 - services/knowledge.py: 新增知识库业务逻辑 - storage/knowledge/.index.json: 知识库索引存储 - api.js: 完善API服务层,增强错误处理 - bootstrap.js: 优化前端初始化与引导流程 - useSetupView.js / useSystemState.js: 重构组合式函数 - TopBar.vue: 优化顶部导航栏组件 - SettingsView.vue: 重构设置页面UI,增强用户体验 - SetupView.vue / SetupRouteView.vue: 完善引导流程页面 - PoliciesView.vue: 优化策略视图组件 - vite.config.js: 更新Vite构建配置 - web_start.sh: 完善前端启动脚本 - views/scripts/: 优化各业务视图JS逻辑 - settings-view.css: 重构设置页面样式 - setup-view.css: 完善引导页样式 - policies-view.css: 优化策略页样式 - test_auth_service.py: 完善认证服务测试 - test_settings_persistence.py: 增强设置持久化测试 - document/: 新增开发文档与工作日志
718 lines
22 KiB
JavaScript
718 lines
22 KiB
JavaScript
import { computed, onMounted, ref } from 'vue'
|
|
|
|
import { useSystemState } from '../../composables/useSystemState.js'
|
|
import { fetchSettings, saveSettings, testModelConnectivity } from '../../services/settings.js'
|
|
import { useToast } from '../../composables/useToast.js'
|
|
|
|
const SETTINGS_STORAGE_KEY = 'x-financial-settings-draft'
|
|
const CURRENT_YEAR = new Date().getFullYear()
|
|
const CUSTOM_OPENAI_PROVIDER = 'Custom OpenAI Compatible'
|
|
const MODEL_SECRET_MASK = '********'
|
|
|
|
const SECTION_DEFINITIONS = [
|
|
{
|
|
id: 'profile',
|
|
label: '企业信息',
|
|
title: '系统基本信息',
|
|
desc: '公司名称、品牌与版权',
|
|
longDesc: '统一维护企业名称、系统显示名称和版权信息,保存后会同步更新当前系统品牌展示。',
|
|
actionLabel: '保存企业信息'
|
|
},
|
|
{
|
|
id: 'admin',
|
|
label: '管理员安全',
|
|
title: '管理员账号与安全策略',
|
|
desc: '账号、密码与登录安全',
|
|
longDesc: '集中管理管理员账号、邮箱和登录安全策略。密码仅在当前输入时可见,不会写入浏览器草稿。',
|
|
actionLabel: '保存安全设置'
|
|
},
|
|
{
|
|
id: 'llm',
|
|
label: '大语言模型',
|
|
title: '模型接入配置',
|
|
desc: '主模型、备份模型与多模态模型',
|
|
longDesc: '集中维护主模型、备份模型、VLM 模型和 Embedding 模型的接入参数,供 AI 助手和识别链路调用。',
|
|
actionLabel: '保存模型配置'
|
|
},
|
|
{
|
|
id: 'logs',
|
|
label: '日志策略',
|
|
title: '日志与审计策略',
|
|
desc: '日志级别、留存与脱敏',
|
|
longDesc: '定义系统日志级别、留存周期和审计策略,保证问题排查和合规审计可追溯。',
|
|
actionLabel: '保存日志策略'
|
|
},
|
|
{
|
|
id: 'mail',
|
|
label: '邮箱设置',
|
|
title: '邮箱通知配置',
|
|
desc: 'SMTP 与通知投递策略',
|
|
longDesc: '维护系统邮件发送配置和通知投递策略,审批、告警和摘要邮件都会依赖这里的设置。',
|
|
actionLabel: '保存邮箱配置'
|
|
}
|
|
]
|
|
|
|
const LOG_LEVELS = ['DEBUG', 'INFO', 'WARN', 'ERROR']
|
|
|
|
const PROVIDER_OPTIONS = [
|
|
'MiniMax',
|
|
'GLM',
|
|
'Kimi',
|
|
'Ali',
|
|
'Codex',
|
|
'Claude',
|
|
'Gemini',
|
|
CUSTOM_OPENAI_PROVIDER
|
|
]
|
|
|
|
const PROVIDER_ENDPOINTS = {
|
|
MiniMax: 'https://api.minimaxi.com/v1',
|
|
GLM: 'https://open.bigmodel.cn/api/paas/v4/',
|
|
Kimi: 'https://api.moonshot.ai/v1',
|
|
Ali: 'https://dashscope.aliyuncs.com/compatible-mode/v1',
|
|
Codex: 'https://api.openai.com/v1',
|
|
Claude: 'https://api.anthropic.com/v1/',
|
|
Gemini: 'https://generativelanguage.googleapis.com/v1beta/openai/',
|
|
[CUSTOM_OPENAI_PROVIDER]: ''
|
|
}
|
|
|
|
const LEGACY_PROVIDER_MAP = {
|
|
'OpenAI Compatible': 'Codex',
|
|
'Azure OpenAI': CUSTOM_OPENAI_PROVIDER,
|
|
Ollama: CUSTOM_OPENAI_PROVIDER,
|
|
'自定义网关': CUSTOM_OPENAI_PROVIDER
|
|
}
|
|
|
|
const MODEL_TEST_CONFIGS = {
|
|
main: {
|
|
label: '主模型',
|
|
providerKey: 'mainProvider',
|
|
modelKey: 'mainModel',
|
|
endpointKey: 'mainEndpoint',
|
|
apiKeyKey: 'mainApiKey',
|
|
capability: 'chat'
|
|
},
|
|
backup: {
|
|
label: '备份模型',
|
|
providerKey: 'backupProvider',
|
|
modelKey: 'backupModel',
|
|
endpointKey: 'backupEndpoint',
|
|
apiKeyKey: 'backupApiKey',
|
|
capability: 'chat'
|
|
},
|
|
vlm: {
|
|
label: 'VLM 模型',
|
|
providerKey: 'vlmProvider',
|
|
modelKey: 'vlmModel',
|
|
endpointKey: 'vlmEndpoint',
|
|
apiKeyKey: 'vlmApiKey',
|
|
capability: 'chat'
|
|
},
|
|
embedding: {
|
|
label: 'Embedding 模型',
|
|
providerKey: 'embeddingProvider',
|
|
modelKey: 'embeddingModel',
|
|
endpointKey: 'embeddingEndpoint',
|
|
apiKeyKey: 'embeddingApiKey',
|
|
capability: 'embedding'
|
|
}
|
|
}
|
|
|
|
const MODEL_API_KEY_CONFIGS = Object.values(MODEL_TEST_CONFIGS)
|
|
|
|
function normalizeValue(value) {
|
|
return String(value ?? '').trim()
|
|
}
|
|
|
|
function normalizeProviderValue(value, fallback = 'Codex') {
|
|
const normalized = normalizeValue(value)
|
|
|
|
if (PROVIDER_OPTIONS.includes(normalized)) {
|
|
return normalized
|
|
}
|
|
|
|
if (LEGACY_PROVIDER_MAP[normalized]) {
|
|
return LEGACY_PROVIDER_MAP[normalized]
|
|
}
|
|
|
|
return fallback
|
|
}
|
|
|
|
function getProviderEndpoint(provider) {
|
|
return PROVIDER_ENDPOINTS[provider] ?? ''
|
|
}
|
|
|
|
function buildDefaultState(companyProfile, currentUser) {
|
|
const companyName = normalizeValue(companyProfile?.name) || 'X-Financial'
|
|
const companyCode = normalizeValue(companyProfile?.code) || 'XF-001'
|
|
const adminEmail =
|
|
normalizeValue(companyProfile?.adminEmail) ||
|
|
normalizeValue(currentUser?.email) ||
|
|
'admin@example.com'
|
|
const adminAccount = normalizeValue(currentUser?.username) || 'superadmin'
|
|
|
|
return {
|
|
companyForm: {
|
|
companyName,
|
|
displayName: companyName,
|
|
companyCode,
|
|
recordNumber: '',
|
|
copyright: `Copyright © 2024-${CURRENT_YEAR} ${companyName}. All Rights Reserved.`
|
|
},
|
|
adminForm: {
|
|
adminAccount,
|
|
adminEmail,
|
|
newPassword: '',
|
|
confirmPassword: '',
|
|
sessionTimeout: Number(import.meta.env.VITE_AUTH_IDLE_TIMEOUT_MINUTES || 30),
|
|
noticeEmail: adminEmail,
|
|
mfaEnabled: true,
|
|
strongPassword: true,
|
|
loginAlertEnabled: true,
|
|
adminPasswordConfigured: false
|
|
},
|
|
llmForm: {
|
|
mainProvider: 'Codex',
|
|
mainModel: 'codex-mini-latest',
|
|
mainEndpoint: getProviderEndpoint('Codex'),
|
|
mainApiKey: '',
|
|
mainApiKeyConfigured: false,
|
|
backupProvider: 'GLM',
|
|
backupModel: 'glm-5.1',
|
|
backupEndpoint: getProviderEndpoint('GLM'),
|
|
backupApiKey: '',
|
|
backupApiKeyConfigured: false,
|
|
vlmProvider: 'Gemini',
|
|
vlmModel: 'gemini-2.5-flash',
|
|
vlmEndpoint: getProviderEndpoint('Gemini'),
|
|
vlmApiKey: '',
|
|
vlmApiKeyConfigured: false,
|
|
embeddingProvider: 'GLM',
|
|
embeddingModel: 'Embedding-3',
|
|
embeddingEndpoint: getProviderEndpoint('GLM'),
|
|
embeddingApiKey: '',
|
|
embeddingApiKeyConfigured: false
|
|
},
|
|
logForm: {
|
|
level: 'INFO',
|
|
retentionDays: 180,
|
|
archiveCycle: 'weekly',
|
|
logPath: 'server/logs/app.log',
|
|
alertEmail: adminEmail,
|
|
operationAudit: true,
|
|
loginAudit: true,
|
|
maskSensitive: true
|
|
},
|
|
mailForm: {
|
|
smtpHost: 'smtp.exmail.qq.com',
|
|
port: 465,
|
|
encryption: 'SSL/TLS',
|
|
senderName: companyName,
|
|
senderAddress: adminEmail,
|
|
username: adminEmail,
|
|
password: '',
|
|
passwordConfigured: false,
|
|
alertEnabled: true,
|
|
digestEnabled: false,
|
|
digestTime: '09:00',
|
|
defaultReceiver: adminEmail
|
|
}
|
|
}
|
|
}
|
|
|
|
function readStoredSettings() {
|
|
if (typeof window === 'undefined') {
|
|
return null
|
|
}
|
|
|
|
const raw = window.sessionStorage.getItem(SETTINGS_STORAGE_KEY)
|
|
if (!raw) {
|
|
return null
|
|
}
|
|
|
|
try {
|
|
return JSON.parse(raw)
|
|
} catch {
|
|
return null
|
|
}
|
|
}
|
|
|
|
function mergeState(baseState, overrideState) {
|
|
const mergedLlmForm = { ...baseState.llmForm, ...(overrideState?.llmForm || {}) }
|
|
|
|
mergedLlmForm.mainProvider = normalizeProviderValue(mergedLlmForm.mainProvider, baseState.llmForm.mainProvider)
|
|
mergedLlmForm.backupProvider = normalizeProviderValue(mergedLlmForm.backupProvider, baseState.llmForm.backupProvider)
|
|
mergedLlmForm.vlmProvider = normalizeProviderValue(mergedLlmForm.vlmProvider, baseState.llmForm.vlmProvider)
|
|
mergedLlmForm.embeddingProvider = normalizeProviderValue(
|
|
mergedLlmForm.embeddingProvider,
|
|
baseState.llmForm.embeddingProvider
|
|
)
|
|
|
|
return {
|
|
companyForm: { ...baseState.companyForm, ...(overrideState?.companyForm || {}) },
|
|
adminForm: { ...baseState.adminForm, ...(overrideState?.adminForm || {}) },
|
|
llmForm: mergedLlmForm,
|
|
logForm: { ...baseState.logForm, ...(overrideState?.logForm || {}) },
|
|
mailForm: { ...baseState.mailForm, ...(overrideState?.mailForm || {}) }
|
|
}
|
|
}
|
|
|
|
function sanitizeForStorage(state) {
|
|
return {
|
|
companyForm: { ...state.companyForm },
|
|
adminForm: {
|
|
...state.adminForm,
|
|
newPassword: '',
|
|
confirmPassword: ''
|
|
},
|
|
llmForm: {
|
|
...state.llmForm,
|
|
mainApiKey: '',
|
|
backupApiKey: '',
|
|
vlmApiKey: '',
|
|
embeddingApiKey: ''
|
|
},
|
|
logForm: { ...state.logForm },
|
|
mailForm: {
|
|
...state.mailForm,
|
|
password: ''
|
|
}
|
|
}
|
|
}
|
|
|
|
function getModelConfiguredKey(apiKeyKey) {
|
|
return `${apiKeyKey}Configured`
|
|
}
|
|
|
|
function isModelSecretMask(value) {
|
|
return value === MODEL_SECRET_MASK
|
|
}
|
|
|
|
function maskConfiguredModelSecrets(state) {
|
|
for (const config of MODEL_API_KEY_CONFIGS) {
|
|
const configuredKey = getModelConfiguredKey(config.apiKeyKey)
|
|
|
|
if (state.llmForm[configuredKey] && !normalizeValue(state.llmForm[config.apiKeyKey])) {
|
|
state.llmForm[config.apiKeyKey] = MODEL_SECRET_MASK
|
|
}
|
|
}
|
|
|
|
return state
|
|
}
|
|
|
|
function buildLlmPayload(llmForm) {
|
|
const payload = { ...llmForm }
|
|
|
|
for (const config of MODEL_API_KEY_CONFIGS) {
|
|
if (isModelSecretMask(payload[config.apiKeyKey])) {
|
|
payload[config.apiKeyKey] = ''
|
|
}
|
|
}
|
|
|
|
return payload
|
|
}
|
|
|
|
function persistSettings(state) {
|
|
if (typeof window === 'undefined') {
|
|
return
|
|
}
|
|
|
|
window.sessionStorage.setItem(SETTINGS_STORAGE_KEY, JSON.stringify(sanitizeForStorage(state)))
|
|
}
|
|
|
|
function isModelConfigReady(provider, model, endpoint) {
|
|
return Boolean(normalizeValue(provider) && normalizeValue(model) && normalizeValue(endpoint))
|
|
}
|
|
|
|
function computeSectionStatus(state) {
|
|
return {
|
|
profile: Boolean(
|
|
normalizeValue(state.companyForm.companyName) &&
|
|
normalizeValue(state.companyForm.displayName) &&
|
|
normalizeValue(state.companyForm.copyright)
|
|
),
|
|
admin: Boolean(
|
|
normalizeValue(state.adminForm.adminAccount) &&
|
|
normalizeValue(state.adminForm.adminEmail) &&
|
|
Number(state.adminForm.sessionTimeout) >= 5
|
|
),
|
|
llm: Boolean(
|
|
isModelConfigReady(state.llmForm.mainProvider, state.llmForm.mainModel, state.llmForm.mainEndpoint) &&
|
|
isModelConfigReady(state.llmForm.backupProvider, state.llmForm.backupModel, state.llmForm.backupEndpoint) &&
|
|
isModelConfigReady(state.llmForm.vlmProvider, state.llmForm.vlmModel, state.llmForm.vlmEndpoint) &&
|
|
isModelConfigReady(
|
|
state.llmForm.embeddingProvider,
|
|
state.llmForm.embeddingModel,
|
|
state.llmForm.embeddingEndpoint
|
|
)
|
|
),
|
|
logs: Boolean(
|
|
normalizeValue(state.logForm.level) &&
|
|
Number(state.logForm.retentionDays) > 0 &&
|
|
normalizeValue(state.logForm.logPath)
|
|
),
|
|
mail: Boolean(
|
|
normalizeValue(state.mailForm.smtpHost) &&
|
|
Number(state.mailForm.port) > 0 &&
|
|
normalizeValue(state.mailForm.senderAddress) &&
|
|
normalizeValue(state.mailForm.username)
|
|
)
|
|
}
|
|
}
|
|
|
|
export default {
|
|
name: 'SettingsView',
|
|
setup() {
|
|
const { toast } = useToast()
|
|
const { companyProfile, currentUser, updateCompanyProfilePreview } = useSystemState()
|
|
|
|
const buildResolvedDefaults = () => buildDefaultState(companyProfile.value, currentUser.value)
|
|
const pageState = ref(mergeState(buildResolvedDefaults(), readStoredSettings()))
|
|
const activeSection = ref('profile')
|
|
const modelTestState = ref({
|
|
main: { status: 'idle', message: '' },
|
|
backup: { status: 'idle', message: '' },
|
|
vlm: { status: 'idle', message: '' },
|
|
embedding: { status: 'idle', message: '' }
|
|
})
|
|
|
|
const sections = SECTION_DEFINITIONS
|
|
const logLevels = LOG_LEVELS
|
|
const providerOptions = PROVIDER_OPTIONS
|
|
|
|
const sectionStatus = computed(() => computeSectionStatus(pageState.value))
|
|
const completedSectionCount = computed(() => Object.values(sectionStatus.value).filter(Boolean).length)
|
|
const activeSectionConfig = computed(
|
|
() => sections.find((section) => section.id === activeSection.value) || sections[0]
|
|
)
|
|
|
|
function updateBrandPreviewFromState(state) {
|
|
updateCompanyProfilePreview({
|
|
name: normalizeValue(state.companyForm.displayName),
|
|
code: normalizeValue(state.companyForm.companyCode),
|
|
adminEmail: normalizeValue(state.adminForm.adminEmail)
|
|
})
|
|
}
|
|
|
|
function applyLoadedSnapshot(snapshot, options = {}) {
|
|
const {
|
|
mergeDraft = false,
|
|
preserveModelApiKeys = false,
|
|
preserveAdminPasswords = false,
|
|
preserveMailPassword = false
|
|
} = options
|
|
|
|
const currentState = pageState.value
|
|
let nextState = mergeState(buildResolvedDefaults(), snapshot)
|
|
|
|
if (mergeDraft) {
|
|
nextState = mergeState(nextState, readStoredSettings())
|
|
}
|
|
|
|
if (preserveModelApiKeys) {
|
|
nextState.llmForm.mainApiKey = currentState.llmForm.mainApiKey
|
|
nextState.llmForm.backupApiKey = currentState.llmForm.backupApiKey
|
|
nextState.llmForm.vlmApiKey = currentState.llmForm.vlmApiKey
|
|
nextState.llmForm.embeddingApiKey = currentState.llmForm.embeddingApiKey
|
|
}
|
|
|
|
if (preserveAdminPasswords) {
|
|
nextState.adminForm.newPassword = currentState.adminForm.newPassword
|
|
nextState.adminForm.confirmPassword = currentState.adminForm.confirmPassword
|
|
}
|
|
|
|
if (preserveMailPassword) {
|
|
nextState.mailForm.password = currentState.mailForm.password
|
|
}
|
|
|
|
pageState.value = maskConfiguredModelSecrets(nextState)
|
|
persistSettings(pageState.value)
|
|
updateBrandPreviewFromState(pageState.value)
|
|
}
|
|
|
|
async function loadSettingsSnapshot() {
|
|
try {
|
|
const snapshot = await fetchSettings()
|
|
applyLoadedSnapshot(snapshot, { mergeDraft: true })
|
|
} catch (error) {
|
|
persistSettings(pageState.value)
|
|
updateBrandPreviewFromState(pageState.value)
|
|
toast(error.message || '无法加载已保存设置,继续使用当前会话草稿。')
|
|
}
|
|
}
|
|
|
|
function buildSettingsPayload() {
|
|
return {
|
|
companyForm: { ...pageState.value.companyForm },
|
|
adminForm: { ...pageState.value.adminForm },
|
|
llmForm: buildLlmPayload(pageState.value.llmForm),
|
|
logForm: { ...pageState.value.logForm },
|
|
mailForm: { ...pageState.value.mailForm }
|
|
}
|
|
}
|
|
|
|
async function persistRemoteSettings(successMessage, options = {}) {
|
|
try {
|
|
const snapshot = await saveSettings(buildSettingsPayload())
|
|
applyLoadedSnapshot(snapshot, options)
|
|
toast(successMessage)
|
|
return true
|
|
} catch (error) {
|
|
toast(error.message || '设置保存失败,请稍后重试。')
|
|
return false
|
|
}
|
|
}
|
|
|
|
function activateSection(sectionId) {
|
|
activeSection.value = sectionId
|
|
}
|
|
|
|
function toggleBoolean(formKey, field) {
|
|
pageState.value[formKey][field] = !pageState.value[formKey][field]
|
|
}
|
|
|
|
function applyProviderPreset(testKey) {
|
|
const config = MODEL_TEST_CONFIGS[testKey]
|
|
const llmForm = pageState.value.llmForm
|
|
const provider = normalizeProviderValue(llmForm[config.providerKey], CUSTOM_OPENAI_PROVIDER)
|
|
|
|
llmForm[config.providerKey] = provider
|
|
llmForm[config.endpointKey] = getProviderEndpoint(provider)
|
|
}
|
|
|
|
function getModelTestState(testKey) {
|
|
return modelTestState.value[testKey] || { status: 'idle', message: '' }
|
|
}
|
|
|
|
function isModelTesting(testKey) {
|
|
return getModelTestState(testKey).status === 'testing'
|
|
}
|
|
|
|
function buildModelTestPayload(testKey) {
|
|
const config = MODEL_TEST_CONFIGS[testKey]
|
|
const llmForm = pageState.value.llmForm
|
|
const apiKey = llmForm[config.apiKeyKey]
|
|
|
|
return {
|
|
provider: llmForm[config.providerKey],
|
|
model: llmForm[config.modelKey],
|
|
endpoint: llmForm[config.endpointKey],
|
|
api_key: isModelSecretMask(apiKey) ? '' : apiKey,
|
|
capability: config.capability,
|
|
slot: testKey
|
|
}
|
|
}
|
|
|
|
function clearModelSecretMask(testKey) {
|
|
const config = MODEL_TEST_CONFIGS[testKey]
|
|
|
|
if (isModelSecretMask(pageState.value.llmForm[config.apiKeyKey])) {
|
|
pageState.value.llmForm[config.apiKeyKey] = ''
|
|
}
|
|
}
|
|
|
|
async function testModelConnection(testKey) {
|
|
const config = MODEL_TEST_CONFIGS[testKey]
|
|
const payload = buildModelTestPayload(testKey)
|
|
|
|
if (!isModelConfigReady(payload.provider, payload.model, payload.endpoint)) {
|
|
const message = `请先完整填写${config.label}的供应商、模型名称和接口地址。`
|
|
modelTestState.value[testKey] = { status: 'error', message }
|
|
toast(message)
|
|
return
|
|
}
|
|
|
|
modelTestState.value[testKey] = { status: 'testing', message: '正在测试模型连通性...' }
|
|
|
|
try {
|
|
const result = await testModelConnectivity(payload)
|
|
|
|
modelTestState.value[testKey] = {
|
|
status: result.ok ? 'success' : 'error',
|
|
message: result.detail || (result.ok ? '模型连接成功。' : '模型连接失败。')
|
|
}
|
|
|
|
toast(modelTestState.value[testKey].message)
|
|
} catch (error) {
|
|
const message = error.message || '模型测试请求失败,请确认 FastAPI 已启动。'
|
|
modelTestState.value[testKey] = { status: 'error', message }
|
|
toast(message)
|
|
}
|
|
}
|
|
|
|
async function saveProfileSection() {
|
|
const companyForm = pageState.value.companyForm
|
|
|
|
if (!normalizeValue(companyForm.companyName)) {
|
|
toast('请输入企业名称。')
|
|
return
|
|
}
|
|
|
|
if (!normalizeValue(companyForm.displayName)) {
|
|
toast('请输入系统显示名称。')
|
|
return
|
|
}
|
|
|
|
if (!normalizeValue(companyForm.copyright)) {
|
|
toast('请输入版权信息。')
|
|
return
|
|
}
|
|
|
|
pageState.value.mailForm.senderName = normalizeValue(companyForm.displayName)
|
|
await persistRemoteSettings('企业信息已保存并应用到当前系统。', {
|
|
preserveModelApiKeys: true,
|
|
preserveAdminPasswords: true,
|
|
preserveMailPassword: true
|
|
})
|
|
}
|
|
|
|
async function saveAdminSection() {
|
|
const adminForm = pageState.value.adminForm
|
|
|
|
if (!normalizeValue(adminForm.adminAccount)) {
|
|
toast('请输入管理员账号。')
|
|
return
|
|
}
|
|
|
|
if (!normalizeValue(adminForm.adminEmail)) {
|
|
toast('请输入管理员邮箱。')
|
|
return
|
|
}
|
|
|
|
if (Number(adminForm.sessionTimeout) < 5) {
|
|
toast('会话超时时间不能少于 5 分钟。')
|
|
return
|
|
}
|
|
|
|
if (adminForm.newPassword) {
|
|
if (adminForm.newPassword.length < 5) {
|
|
toast('管理员密码至少需要 5 位。')
|
|
return
|
|
}
|
|
|
|
if (adminForm.newPassword !== adminForm.confirmPassword) {
|
|
toast('两次输入的管理员密码不一致。')
|
|
return
|
|
}
|
|
}
|
|
|
|
await persistRemoteSettings('管理员安全设置已保存。', {
|
|
preserveModelApiKeys: true,
|
|
preserveAdminPasswords: false,
|
|
preserveMailPassword: true
|
|
})
|
|
}
|
|
|
|
async function saveLlmSection() {
|
|
const llmForm = pageState.value.llmForm
|
|
const modelConfigs = [
|
|
['主模型', llmForm.mainProvider, llmForm.mainModel, llmForm.mainEndpoint],
|
|
['备份模型', llmForm.backupProvider, llmForm.backupModel, llmForm.backupEndpoint],
|
|
['VLM 模型', llmForm.vlmProvider, llmForm.vlmModel, llmForm.vlmEndpoint],
|
|
['Embedding 模型', llmForm.embeddingProvider, llmForm.embeddingModel, llmForm.embeddingEndpoint]
|
|
]
|
|
|
|
for (const [label, provider, model, endpoint] of modelConfigs) {
|
|
if (!isModelConfigReady(provider, model, endpoint)) {
|
|
toast(`请完整填写${label}的供应商、模型名称和接口地址。`)
|
|
return
|
|
}
|
|
}
|
|
|
|
await persistRemoteSettings('模型配置已保存。', {
|
|
preserveModelApiKeys: true,
|
|
preserveAdminPasswords: true,
|
|
preserveMailPassword: true
|
|
})
|
|
}
|
|
|
|
async function saveLogsSection() {
|
|
const logForm = pageState.value.logForm
|
|
|
|
if (!normalizeValue(logForm.level) || Number(logForm.retentionDays) <= 0) {
|
|
toast('请填写有效的日志级别和留存天数。')
|
|
return
|
|
}
|
|
|
|
if (!normalizeValue(logForm.logPath)) {
|
|
toast('请输入日志路径。')
|
|
return
|
|
}
|
|
|
|
await persistRemoteSettings('日志策略已保存。', {
|
|
preserveModelApiKeys: true,
|
|
preserveAdminPasswords: true,
|
|
preserveMailPassword: true
|
|
})
|
|
}
|
|
|
|
async function saveMailSection() {
|
|
const mailForm = pageState.value.mailForm
|
|
|
|
if (!normalizeValue(mailForm.smtpHost) || Number(mailForm.port) <= 0) {
|
|
toast('请填写有效的 SMTP Host 和端口。')
|
|
return
|
|
}
|
|
|
|
if (!normalizeValue(mailForm.senderAddress) || !normalizeValue(mailForm.username)) {
|
|
toast('请填写发件人邮箱和 SMTP 登录账号。')
|
|
return
|
|
}
|
|
|
|
await persistRemoteSettings('邮箱配置已保存。', {
|
|
preserveModelApiKeys: true,
|
|
preserveAdminPasswords: true,
|
|
preserveMailPassword: false
|
|
})
|
|
}
|
|
|
|
async function saveActiveSection() {
|
|
if (activeSection.value === 'profile') {
|
|
await saveProfileSection()
|
|
return
|
|
}
|
|
|
|
if (activeSection.value === 'admin') {
|
|
await saveAdminSection()
|
|
return
|
|
}
|
|
|
|
if (activeSection.value === 'llm') {
|
|
await saveLlmSection()
|
|
return
|
|
}
|
|
|
|
if (activeSection.value === 'logs') {
|
|
await saveLogsSection()
|
|
return
|
|
}
|
|
|
|
await saveMailSection()
|
|
}
|
|
|
|
onMounted(() => {
|
|
loadSettingsSnapshot()
|
|
})
|
|
|
|
return {
|
|
activeSection,
|
|
activeSectionConfig,
|
|
activateSection,
|
|
applyProviderPreset,
|
|
clearModelSecretMask,
|
|
completedSectionCount,
|
|
getModelTestState,
|
|
isModelTesting,
|
|
logLevels,
|
|
modelTestState,
|
|
pageState,
|
|
providerOptions,
|
|
saveActiveSection,
|
|
sectionStatus,
|
|
sections,
|
|
testModelConnection,
|
|
toggleBoolean
|
|
}
|
|
}
|
|
}
|