fix: 修复链路追踪密钥显示和默认选中逻辑

1. 修复密钥脱敏显示问题:先解密再脱敏,避免显示加密后的 base64 数据
2. 优化详情默认选中逻辑:优先显示最后一个有效结果(成功/失败),而非未执行/跳过
This commit is contained in:
fawney19
2025-12-12 18:15:46 +08:00
parent 9629e9be0d
commit ca60202636
2 changed files with 37 additions and 20 deletions

View File

@@ -552,8 +552,8 @@ const isCapabilityUsed = (cap: string): boolean => {
return activeCapabilities.value.includes(cap) return activeCapabilities.value.includes(cap)
} }
// 检查某个能力是否被 Key 支持 // 检查某个能力是否被 Key 支持(保留以备后用)
const isCapabilitySupported = (cap: string): boolean => { const _isCapabilitySupported = (cap: string): boolean => {
return keyCapabilities.value.includes(cap) return keyCapabilities.value.includes(cap)
} }
@@ -653,7 +653,20 @@ watch(groupedTimeline, (newGroups) => {
return return
} }
// 默认选择最后一个组 // 查找最后一个有效结果的组failed 优先于 skipped/available
// 从后往前找第一个 failed 的组
for (let i = newGroups.length - 1; i >= 0; i--) {
const group = newGroups[i]
if (group.primaryStatus === 'failed') {
selectedGroupIndex.value = i
// 选中最后一个失败的尝试
const failedIdx = group.allAttempts.findLastIndex((a: CandidateRecord) => a.status === 'failed')
selectedAttemptIndex.value = failedIdx >= 0 ? failedIdx : group.allAttempts.length - 1
return
}
}
// 都没有则选择最后一个组
selectedGroupIndex.value = newGroups.length - 1 selectedGroupIndex.value = newGroups.length - 1
selectedAttemptIndex.value = newGroups[newGroups.length - 1].allAttempts.length - 1 selectedAttemptIndex.value = newGroups[newGroups.length - 1].allAttempts.length - 1
}, { immediate: true }) }, { immediate: true })
@@ -702,8 +715,8 @@ const getStatusLabel = (status: string) => {
return labels[status] || status return labels[status] || status
} }
// 获取状态徽章样式 // 获取状态徽章样式(保留以备后用)
const getStatusBadgeVariant = (status: string): any => { const _getStatusBadgeVariant = (status: string): string => {
const variants: Record<string, string> = { const variants: Record<string, string> = {
available: 'outline', available: 'outline',
pending: 'secondary', pending: 'secondary',

View File

@@ -14,6 +14,7 @@ from src.api.base.admin_adapter import AdminApiAdapter
from src.api.base.pipeline import ApiRequestPipeline from src.api.base.pipeline import ApiRequestPipeline
from src.database import get_db from src.database import get_db
from src.models.database import Provider, ProviderEndpoint, ProviderAPIKey from src.models.database import Provider, ProviderEndpoint, ProviderAPIKey
from src.core.crypto import crypto_service
from src.services.request.candidate import RequestCandidateService from src.services.request.candidate import RequestCandidateService
router = APIRouter(prefix="/api/admin/monitoring/trace", tags=["Admin - Monitoring: Trace"]) router = APIRouter(prefix="/api/admin/monitoring/trace", tags=["Admin - Monitoring: Trace"])
@@ -171,22 +172,25 @@ class AdminGetRequestTraceAdapter(AdminApiAdapter):
for k in keys: for k in keys:
key_map[k.id] = k.name key_map[k.id] = k.name
key_capabilities_map[k.id] = k.capabilities key_capabilities_map[k.id] = k.capabilities
# 生成脱敏预览:保留前缀和最后4位 # 生成脱敏预览:先解密再脱敏
api_key = k.api_key or "" try:
if len(api_key) > 8: decrypted_key = crypto_service.decrypt(k.api_key)
# 检测常见前缀模式 if len(decrypted_key) > 8:
prefix_end = 0 # 检测常见前缀模式
for prefix in ["sk-", "key-", "api-", "ak-"]: prefix_end = 0
if api_key.lower().startswith(prefix): for prefix in ["sk-", "key-", "api-", "ak-"]:
prefix_end = len(prefix) if decrypted_key.lower().startswith(prefix):
break prefix_end = len(prefix)
if prefix_end > 0: break
key_preview_map[k.id] = f"{api_key[:prefix_end]}***{api_key[-4:]}" if prefix_end > 0:
key_preview_map[k.id] = f"{decrypted_key[:prefix_end]}***{decrypted_key[-4:]}"
else:
key_preview_map[k.id] = f"{decrypted_key[:4]}***{decrypted_key[-4:]}"
elif len(decrypted_key) > 4:
key_preview_map[k.id] = f"***{decrypted_key[-4:]}"
else: else:
key_preview_map[k.id] = f"{api_key[:3]}***{api_key[-4:]}" key_preview_map[k.id] = "***"
elif len(api_key) > 4: except Exception:
key_preview_map[k.id] = f"***{api_key[-4:]}"
else:
key_preview_map[k.id] = "***" key_preview_map[k.id] = "***"
# 构建 candidate 响应列表 # 构建 candidate 响应列表