2025-12-10 20:52:44 +08:00
|
|
|
|
<template>
|
|
|
|
|
|
<Card class="overflow-hidden">
|
|
|
|
|
|
<!-- 标题头部 -->
|
|
|
|
|
|
<div class="p-4 border-b border-border/60">
|
|
|
|
|
|
<div class="flex items-center justify-between">
|
|
|
|
|
|
<h3 class="text-sm font-semibold flex items-center gap-2">
|
|
|
|
|
|
模型列表
|
|
|
|
|
|
</h3>
|
2025-12-12 16:15:36 +08:00
|
|
|
|
<Button
|
|
|
|
|
|
variant="outline"
|
|
|
|
|
|
size="sm"
|
|
|
|
|
|
class="h-8"
|
|
|
|
|
|
@click="openBatchAssignDialog"
|
|
|
|
|
|
>
|
2025-12-10 20:52:44 +08:00
|
|
|
|
<Layers class="w-3.5 h-3.5 mr-1.5" />
|
|
|
|
|
|
关联模型
|
|
|
|
|
|
</Button>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<!-- 加载状态 -->
|
2025-12-12 16:15:36 +08:00
|
|
|
|
<div
|
|
|
|
|
|
v-if="loading"
|
|
|
|
|
|
class="flex items-center justify-center py-12"
|
|
|
|
|
|
>
|
|
|
|
|
|
<div class="animate-spin rounded-full h-8 w-8 border-b-2 border-primary" />
|
2025-12-10 20:52:44 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<!-- 模型列表 -->
|
2025-12-12 16:15:36 +08:00
|
|
|
|
<div
|
|
|
|
|
|
v-else-if="models.length > 0"
|
|
|
|
|
|
class="overflow-hidden"
|
|
|
|
|
|
>
|
2025-12-10 20:52:44 +08:00
|
|
|
|
<table class="w-full text-sm table-fixed">
|
|
|
|
|
|
<thead class="bg-muted/50 text-xs uppercase tracking-wide text-muted-foreground">
|
|
|
|
|
|
<tr>
|
2025-12-12 16:15:36 +08:00
|
|
|
|
<th class="text-left px-4 py-3 font-semibold w-[40%]">
|
|
|
|
|
|
模型
|
|
|
|
|
|
</th>
|
|
|
|
|
|
<th class="text-left px-4 py-3 font-semibold w-[15%]">
|
|
|
|
|
|
能力
|
|
|
|
|
|
</th>
|
|
|
|
|
|
<th class="text-left px-4 py-3 font-semibold w-[25%]">
|
|
|
|
|
|
价格 ($/M)
|
|
|
|
|
|
</th>
|
|
|
|
|
|
<th class="text-center px-4 py-3 font-semibold w-[20%]">
|
|
|
|
|
|
操作
|
|
|
|
|
|
</th>
|
2025-12-10 20:52:44 +08:00
|
|
|
|
</tr>
|
|
|
|
|
|
</thead>
|
|
|
|
|
|
<tbody>
|
|
|
|
|
|
<tr
|
|
|
|
|
|
v-for="model in sortedModels"
|
|
|
|
|
|
:key="model.id"
|
|
|
|
|
|
class="border-b border-border/40 last:border-b-0 hover:bg-muted/30 transition-colors"
|
|
|
|
|
|
>
|
|
|
|
|
|
<td class="align-top px-4 py-3">
|
|
|
|
|
|
<div class="flex items-center gap-2.5">
|
|
|
|
|
|
<!-- 状态指示灯 -->
|
|
|
|
|
|
<div
|
|
|
|
|
|
class="w-2 h-2 rounded-full shrink-0"
|
|
|
|
|
|
:class="getStatusIndicatorClass(model)"
|
|
|
|
|
|
:title="getStatusTitle(model)"
|
2025-12-12 16:15:36 +08:00
|
|
|
|
/>
|
2025-12-10 20:52:44 +08:00
|
|
|
|
<!-- 模型信息 -->
|
|
|
|
|
|
<div class="text-left flex-1 min-w-0">
|
|
|
|
|
|
<span class="font-semibold text-sm">
|
|
|
|
|
|
{{ model.global_model_display_name || model.provider_model_name }}
|
|
|
|
|
|
</span>
|
|
|
|
|
|
<div class="text-xs text-muted-foreground mt-1 flex items-center gap-1">
|
|
|
|
|
|
<span class="font-mono truncate">{{ model.provider_model_name }}</span>
|
|
|
|
|
|
<button
|
|
|
|
|
|
class="p-0.5 hover:bg-muted rounded transition-colors shrink-0"
|
|
|
|
|
|
title="复制模型 ID"
|
2025-12-12 16:15:36 +08:00
|
|
|
|
@click.stop="copyModelId(model.provider_model_name)"
|
2025-12-10 20:52:44 +08:00
|
|
|
|
>
|
|
|
|
|
|
<Copy class="w-3 h-3" />
|
|
|
|
|
|
</button>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</td>
|
|
|
|
|
|
<td class="align-top px-4 py-3">
|
2025-12-12 16:15:36 +08:00
|
|
|
|
<div
|
|
|
|
|
|
v-if="hasAnyCapability(model)"
|
|
|
|
|
|
class="grid grid-cols-3 gap-1 w-fit"
|
|
|
|
|
|
>
|
|
|
|
|
|
<Zap
|
|
|
|
|
|
v-if="model.effective_supports_streaming ?? model.supports_streaming"
|
|
|
|
|
|
class="w-4 h-4 text-muted-foreground"
|
|
|
|
|
|
title="流式输出"
|
|
|
|
|
|
/>
|
|
|
|
|
|
<Image
|
|
|
|
|
|
v-if="model.effective_supports_image_generation ?? model.supports_image_generation"
|
|
|
|
|
|
class="w-4 h-4 text-muted-foreground"
|
|
|
|
|
|
title="图像生成"
|
|
|
|
|
|
/>
|
|
|
|
|
|
<Eye
|
|
|
|
|
|
v-if="model.effective_supports_vision ?? model.supports_vision"
|
|
|
|
|
|
class="w-4 h-4 text-muted-foreground"
|
|
|
|
|
|
title="视觉理解"
|
|
|
|
|
|
/>
|
|
|
|
|
|
<Wrench
|
|
|
|
|
|
v-if="model.effective_supports_function_calling ?? model.supports_function_calling"
|
|
|
|
|
|
class="w-4 h-4 text-muted-foreground"
|
|
|
|
|
|
title="工具调用"
|
|
|
|
|
|
/>
|
|
|
|
|
|
<Brain
|
|
|
|
|
|
v-if="model.effective_supports_extended_thinking ?? model.supports_extended_thinking"
|
|
|
|
|
|
class="w-4 h-4 text-muted-foreground"
|
|
|
|
|
|
title="深度思考"
|
|
|
|
|
|
/>
|
2025-12-10 20:52:44 +08:00
|
|
|
|
</div>
|
2025-12-12 16:15:36 +08:00
|
|
|
|
<span
|
|
|
|
|
|
v-else
|
|
|
|
|
|
class="text-xs text-muted-foreground"
|
|
|
|
|
|
>—</span>
|
2025-12-10 20:52:44 +08:00
|
|
|
|
</td>
|
|
|
|
|
|
<td class="align-top px-4 py-3 text-xs whitespace-nowrap">
|
2025-12-12 16:15:36 +08:00
|
|
|
|
<div
|
|
|
|
|
|
class="grid gap-1"
|
|
|
|
|
|
style="grid-template-columns: auto 1fr;"
|
|
|
|
|
|
>
|
2025-12-10 20:52:44 +08:00
|
|
|
|
<!-- 按 Token 计费 -->
|
|
|
|
|
|
<template v-if="hasTokenPricing(model)">
|
|
|
|
|
|
<span class="text-muted-foreground text-right">输入/输出:</span>
|
|
|
|
|
|
<span class="font-mono font-semibold">
|
|
|
|
|
|
${{ formatPrice(model.effective_input_price) }}/${{ formatPrice(model.effective_output_price) }}
|
|
|
|
|
|
</span>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
<template v-if="getEffectiveCachePrice(model, 'creation') > 0 || getEffectiveCachePrice(model, 'read') > 0">
|
|
|
|
|
|
<span class="text-muted-foreground text-right">缓存:</span>
|
|
|
|
|
|
<span class="font-mono font-semibold">
|
|
|
|
|
|
${{ formatPrice(getEffectiveCachePrice(model, 'creation')) }}/${{ formatPrice(getEffectiveCachePrice(model, 'read')) }}
|
|
|
|
|
|
</span>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
<!-- 1h 缓存价格 -->
|
|
|
|
|
|
<template v-if="get1hCachePrice(model) > 0">
|
|
|
|
|
|
<span class="text-muted-foreground text-right">1h 缓存:</span>
|
|
|
|
|
|
<span class="font-mono font-semibold">
|
|
|
|
|
|
${{ formatPrice(get1hCachePrice(model)) }}
|
|
|
|
|
|
</span>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
<!-- 按次计费 -->
|
|
|
|
|
|
<template v-if="hasRequestPricing(model)">
|
|
|
|
|
|
<span class="text-muted-foreground text-right">按次:</span>
|
|
|
|
|
|
<span class="font-mono font-semibold">
|
|
|
|
|
|
${{ formatPrice(model.effective_price_per_request ?? model.price_per_request) }}/次
|
|
|
|
|
|
</span>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
<!-- 无计费配置 -->
|
|
|
|
|
|
<template v-if="!hasTokenPricing(model) && !hasRequestPricing(model)">
|
|
|
|
|
|
<span class="text-muted-foreground">—</span>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</td>
|
|
|
|
|
|
<td class="align-top px-4 py-3">
|
|
|
|
|
|
<div class="flex justify-center gap-1.5">
|
2025-12-25 00:02:56 +08:00
|
|
|
|
<Button
|
|
|
|
|
|
variant="ghost"
|
|
|
|
|
|
size="icon"
|
|
|
|
|
|
class="h-8 w-8"
|
|
|
|
|
|
title="测试模型"
|
|
|
|
|
|
:disabled="testingModelId === model.id"
|
|
|
|
|
|
@click="testModelConnection(model)"
|
|
|
|
|
|
>
|
|
|
|
|
|
<Loader2 v-if="testingModelId === model.id" class="w-3.5 h-3.5 animate-spin" />
|
|
|
|
|
|
<Play v-else class="w-3.5 h-3.5" />
|
|
|
|
|
|
</Button>
|
2025-12-10 20:52:44 +08:00
|
|
|
|
<Button
|
|
|
|
|
|
variant="ghost"
|
|
|
|
|
|
size="icon"
|
|
|
|
|
|
class="h-8 w-8"
|
|
|
|
|
|
title="编辑"
|
2025-12-12 16:15:36 +08:00
|
|
|
|
@click="editModel(model)"
|
2025-12-10 20:52:44 +08:00
|
|
|
|
>
|
|
|
|
|
|
<Edit class="w-3.5 h-3.5" />
|
|
|
|
|
|
</Button>
|
|
|
|
|
|
<Button
|
|
|
|
|
|
variant="ghost"
|
|
|
|
|
|
size="icon"
|
|
|
|
|
|
class="h-8 w-8"
|
|
|
|
|
|
:disabled="togglingModelId === model.id"
|
|
|
|
|
|
:title="model.is_active ? '点击停用' : '点击启用'"
|
2025-12-12 16:15:36 +08:00
|
|
|
|
@click="toggleModelActive(model)"
|
2025-12-10 20:52:44 +08:00
|
|
|
|
>
|
|
|
|
|
|
<Power class="w-3.5 h-3.5" />
|
|
|
|
|
|
</Button>
|
|
|
|
|
|
<Button
|
|
|
|
|
|
variant="ghost"
|
|
|
|
|
|
size="icon"
|
|
|
|
|
|
class="h-8 w-8 text-destructive hover:text-destructive"
|
|
|
|
|
|
title="删除"
|
2025-12-12 16:15:36 +08:00
|
|
|
|
@click="deleteModel(model)"
|
2025-12-10 20:52:44 +08:00
|
|
|
|
>
|
|
|
|
|
|
<Trash2 class="w-3.5 h-3.5" />
|
|
|
|
|
|
</Button>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</td>
|
|
|
|
|
|
</tr>
|
|
|
|
|
|
</tbody>
|
|
|
|
|
|
</table>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<!-- 空状态 -->
|
2025-12-12 16:15:36 +08:00
|
|
|
|
<div
|
|
|
|
|
|
v-else
|
|
|
|
|
|
class="p-8 text-center text-muted-foreground"
|
|
|
|
|
|
>
|
2025-12-10 20:52:44 +08:00
|
|
|
|
<Box class="w-12 h-12 mx-auto mb-3 opacity-50" />
|
2025-12-12 16:15:36 +08:00
|
|
|
|
<p class="text-sm">
|
|
|
|
|
|
暂无模型
|
|
|
|
|
|
</p>
|
|
|
|
|
|
<p class="text-xs mt-1">
|
|
|
|
|
|
请前往"模型目录"页面添加模型
|
|
|
|
|
|
</p>
|
2025-12-10 20:52:44 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</Card>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
|
|
|
|
|
|
<script setup lang="ts">
|
|
|
|
|
|
import { ref, computed, onMounted } from 'vue'
|
2025-12-25 00:02:56 +08:00
|
|
|
|
import { Box, Edit, Trash2, Layers, Eye, Wrench, Zap, Brain, Power, Copy, Image, Loader2, Play } from 'lucide-vue-next'
|
2025-12-10 20:52:44 +08:00
|
|
|
|
import Card from '@/components/ui/card.vue'
|
|
|
|
|
|
import Button from '@/components/ui/button.vue'
|
|
|
|
|
|
import { useToast } from '@/composables/useToast'
|
2025-12-25 00:02:56 +08:00
|
|
|
|
import { getProviderModels, testModel, type Model } from '@/api/endpoints'
|
2025-12-10 20:52:44 +08:00
|
|
|
|
import { updateModel } from '@/api/endpoints/models'
|
2025-12-25 19:36:29 +08:00
|
|
|
|
import { parseTestModelError } from '@/utils/errorParser'
|
2025-12-10 20:52:44 +08:00
|
|
|
|
|
|
|
|
|
|
const props = defineProps<{
|
|
|
|
|
|
provider: any
|
|
|
|
|
|
}>()
|
|
|
|
|
|
|
|
|
|
|
|
const emit = defineEmits<{
|
2025-12-12 20:22:27 +08:00
|
|
|
|
'editModel': [model: Model]
|
|
|
|
|
|
'deleteModel': [model: Model]
|
|
|
|
|
|
'batchAssign': []
|
2025-12-10 20:52:44 +08:00
|
|
|
|
}>()
|
|
|
|
|
|
|
|
|
|
|
|
const { error: showError, success: showSuccess } = useToast()
|
|
|
|
|
|
|
|
|
|
|
|
// 状态
|
|
|
|
|
|
const loading = ref(false)
|
|
|
|
|
|
const models = ref<Model[]>([])
|
|
|
|
|
|
const togglingModelId = ref<string | null>(null)
|
2025-12-25 00:02:56 +08:00
|
|
|
|
const testingModelId = ref<string | null>(null)
|
2025-12-10 20:52:44 +08:00
|
|
|
|
|
|
|
|
|
|
// 按名称排序的模型列表
|
|
|
|
|
|
const sortedModels = computed(() => {
|
|
|
|
|
|
return [...models.value].sort((a, b) => {
|
|
|
|
|
|
const nameA = (a.global_model_display_name || a.provider_model_name || '').toLowerCase()
|
|
|
|
|
|
const nameB = (b.global_model_display_name || b.provider_model_name || '').toLowerCase()
|
|
|
|
|
|
return nameA.localeCompare(nameB)
|
|
|
|
|
|
})
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
// 复制模型 ID 到剪贴板
|
|
|
|
|
|
async function copyModelId(modelId: string) {
|
|
|
|
|
|
try {
|
|
|
|
|
|
await navigator.clipboard.writeText(modelId)
|
|
|
|
|
|
showSuccess('已复制到剪贴板')
|
|
|
|
|
|
} catch {
|
|
|
|
|
|
showError('复制失败', '错误')
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 加载模型
|
|
|
|
|
|
async function loadModels() {
|
|
|
|
|
|
try {
|
|
|
|
|
|
loading.value = true
|
|
|
|
|
|
models.value = await getProviderModels(props.provider.id)
|
|
|
|
|
|
} catch (err: any) {
|
|
|
|
|
|
showError(err.response?.data?.detail || '加载失败', '错误')
|
|
|
|
|
|
} finally {
|
|
|
|
|
|
loading.value = false
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 格式化价格显示
|
|
|
|
|
|
function formatPrice(price: number | null | undefined): string {
|
|
|
|
|
|
if (price === null || price === undefined) return '-'
|
|
|
|
|
|
// 如果是整数或小数点后只有1-2位,直接显示
|
|
|
|
|
|
if (price >= 0.01 || price === 0) {
|
|
|
|
|
|
return price.toFixed(2)
|
|
|
|
|
|
}
|
|
|
|
|
|
// 对于非常小的数字,使用科学计数法
|
|
|
|
|
|
if (price < 0.0001) {
|
|
|
|
|
|
return price.toExponential(2)
|
|
|
|
|
|
}
|
|
|
|
|
|
// 其他情况保留4位小数
|
|
|
|
|
|
return price.toFixed(4)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 检查模型是否有任何能力
|
|
|
|
|
|
function hasAnyCapability(model: Model): boolean {
|
|
|
|
|
|
return !!(
|
|
|
|
|
|
(model.effective_supports_vision ?? model.supports_vision) ||
|
|
|
|
|
|
(model.effective_supports_function_calling ?? model.supports_function_calling) ||
|
|
|
|
|
|
(model.effective_supports_streaming ?? model.supports_streaming) ||
|
|
|
|
|
|
(model.effective_supports_extended_thinking ?? model.supports_extended_thinking) ||
|
|
|
|
|
|
(model.effective_supports_image_generation ?? model.supports_image_generation)
|
|
|
|
|
|
)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 检查是否有按 Token 计费
|
|
|
|
|
|
function hasTokenPricing(model: Model): boolean {
|
|
|
|
|
|
const inputPrice = model.effective_input_price
|
|
|
|
|
|
const outputPrice = model.effective_output_price
|
|
|
|
|
|
return (inputPrice != null && inputPrice > 0) || (outputPrice != null && outputPrice > 0)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 获取有效的缓存价格(从 effective_tiered_pricing 或 tiered_pricing 中提取)
|
|
|
|
|
|
function getEffectiveCachePrice(model: Model, type: 'creation' | 'read'): number {
|
|
|
|
|
|
const tiered = model.effective_tiered_pricing || model.tiered_pricing
|
|
|
|
|
|
if (!tiered?.tiers?.length) return 0
|
|
|
|
|
|
const firstTier = tiered.tiers[0]
|
|
|
|
|
|
if (type === 'creation') {
|
|
|
|
|
|
return firstTier.cache_creation_price_per_1m || 0
|
|
|
|
|
|
}
|
|
|
|
|
|
return firstTier.cache_read_price_per_1m || 0
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 获取 1h 缓存价格
|
|
|
|
|
|
function get1hCachePrice(model: Model): number {
|
|
|
|
|
|
const tiered = model.effective_tiered_pricing || model.tiered_pricing
|
|
|
|
|
|
if (!tiered?.tiers?.length) return 0
|
|
|
|
|
|
const firstTier = tiered.tiers[0]
|
|
|
|
|
|
const ttl1h = firstTier.cache_ttl_pricing?.find(t => t.ttl_minutes === 60)
|
|
|
|
|
|
return ttl1h?.cache_creation_price_per_1m || 0
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 检查是否有按次计费
|
|
|
|
|
|
function hasRequestPricing(model: Model): boolean {
|
|
|
|
|
|
const requestPrice = model.effective_price_per_request ?? model.price_per_request
|
|
|
|
|
|
return requestPrice != null && requestPrice > 0
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 获取状态指示灯样式
|
|
|
|
|
|
function getStatusIndicatorClass(model: Model): string {
|
|
|
|
|
|
if (!model.is_active) {
|
|
|
|
|
|
// 已停用 - 灰色
|
|
|
|
|
|
return 'bg-gray-400 dark:bg-gray-600'
|
|
|
|
|
|
}
|
|
|
|
|
|
if (model.is_available) {
|
|
|
|
|
|
// 活跃且可用 - 绿色
|
|
|
|
|
|
return 'bg-green-500 dark:bg-green-400'
|
|
|
|
|
|
}
|
|
|
|
|
|
// 活跃但不可用 - 红色
|
|
|
|
|
|
return 'bg-red-500 dark:bg-red-400'
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 获取状态提示文本
|
|
|
|
|
|
function getStatusTitle(model: Model): string {
|
|
|
|
|
|
if (!model.is_active) {
|
|
|
|
|
|
return '已停用'
|
|
|
|
|
|
}
|
|
|
|
|
|
if (model.is_available) {
|
|
|
|
|
|
return '活跃且可用'
|
|
|
|
|
|
}
|
|
|
|
|
|
return '活跃但不可用'
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 编辑模型
|
|
|
|
|
|
function editModel(model: Model) {
|
2025-12-12 20:22:27 +08:00
|
|
|
|
emit('editModel', model)
|
2025-12-10 20:52:44 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 删除模型
|
|
|
|
|
|
function deleteModel(model: Model) {
|
2025-12-12 20:22:27 +08:00
|
|
|
|
emit('deleteModel', model)
|
2025-12-10 20:52:44 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 打开批量关联对话框
|
|
|
|
|
|
function openBatchAssignDialog() {
|
2025-12-12 20:22:27 +08:00
|
|
|
|
emit('batchAssign')
|
2025-12-10 20:52:44 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 切换模型启用状态
|
|
|
|
|
|
async function toggleModelActive(model: Model) {
|
|
|
|
|
|
if (togglingModelId.value) return
|
|
|
|
|
|
|
|
|
|
|
|
togglingModelId.value = model.id
|
|
|
|
|
|
try {
|
|
|
|
|
|
const newStatus = !model.is_active
|
|
|
|
|
|
await updateModel(props.provider.id, model.id, { is_active: newStatus })
|
|
|
|
|
|
model.is_active = newStatus
|
|
|
|
|
|
showSuccess(newStatus ? '模型已启用' : '模型已停用')
|
|
|
|
|
|
} catch (err: any) {
|
|
|
|
|
|
showError(err.response?.data?.detail || '操作失败', '错误')
|
|
|
|
|
|
} finally {
|
|
|
|
|
|
togglingModelId.value = null
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-25 00:02:56 +08:00
|
|
|
|
// 测试模型连接性
|
|
|
|
|
|
async function testModelConnection(model: Model) {
|
|
|
|
|
|
if (testingModelId.value) return
|
|
|
|
|
|
|
|
|
|
|
|
testingModelId.value = model.id
|
|
|
|
|
|
try {
|
|
|
|
|
|
const result = await testModel({
|
|
|
|
|
|
provider_id: props.provider.id,
|
|
|
|
|
|
model_name: model.provider_model_name,
|
|
|
|
|
|
message: "hello"
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
if (result.success) {
|
|
|
|
|
|
showSuccess(`模型 "${model.provider_model_name}" 测试成功`)
|
|
|
|
|
|
|
|
|
|
|
|
// 如果有响应内容,可以显示更多信息
|
|
|
|
|
|
if (result.data?.response?.choices?.[0]?.message?.content) {
|
|
|
|
|
|
const content = result.data.response.choices[0].message.content
|
|
|
|
|
|
showSuccess(`测试成功,响应: ${content.substring(0, 100)}${content.length > 100 ? '...' : ''}`)
|
|
|
|
|
|
} else if (result.data?.content_preview) {
|
|
|
|
|
|
showSuccess(`流式测试成功,预览: ${result.data.content_preview}`)
|
|
|
|
|
|
}
|
|
|
|
|
|
} else {
|
2025-12-25 19:36:29 +08:00
|
|
|
|
showError(`模型测试失败: ${parseTestModelError(result)}`)
|
2025-12-25 00:02:56 +08:00
|
|
|
|
}
|
|
|
|
|
|
} catch (err: any) {
|
|
|
|
|
|
const errorMsg = err.response?.data?.detail || err.message || '测试请求失败'
|
|
|
|
|
|
showError(`模型测试失败: ${errorMsg}`)
|
|
|
|
|
|
} finally {
|
|
|
|
|
|
testingModelId.value = null
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-10 20:52:44 +08:00
|
|
|
|
onMounted(() => {
|
|
|
|
|
|
loadModels()
|
|
|
|
|
|
})
|
|
|
|
|
|
</script>
|