2025-12-10 20:52:44 +08:00
|
|
|
|
<template>
|
|
|
|
|
|
<Dialog
|
|
|
|
|
|
:model-value="internalOpen"
|
|
|
|
|
|
:title="isEditMode ? '编辑 API 端点' : '添加 API 端点'"
|
|
|
|
|
|
:description="isEditMode ? `修改 ${provider?.display_name} 的端点配置` : '为提供商添加新的 API 端点'"
|
|
|
|
|
|
:icon="isEditMode ? SquarePen : Link"
|
|
|
|
|
|
size="xl"
|
|
|
|
|
|
@update:model-value="handleDialogUpdate"
|
|
|
|
|
|
>
|
2025-12-12 16:15:36 +08:00
|
|
|
|
<form
|
|
|
|
|
|
class="space-y-6"
|
|
|
|
|
|
@submit.prevent="handleSubmit"
|
|
|
|
|
|
>
|
2025-12-10 20:52:44 +08:00
|
|
|
|
<!-- API 配置 -->
|
|
|
|
|
|
<div class="space-y-4">
|
2025-12-12 16:15:36 +08:00
|
|
|
|
<h3
|
|
|
|
|
|
v-if="isEditMode"
|
|
|
|
|
|
class="text-sm font-medium"
|
|
|
|
|
|
>
|
|
|
|
|
|
API 配置
|
|
|
|
|
|
</h3>
|
2025-12-10 20:52:44 +08:00
|
|
|
|
|
|
|
|
|
|
<div class="grid grid-cols-2 gap-4">
|
|
|
|
|
|
<!-- API 格式 -->
|
|
|
|
|
|
<div class="space-y-2">
|
|
|
|
|
|
<Label for="api_format">API 格式 *</Label>
|
|
|
|
|
|
<template v-if="isEditMode">
|
|
|
|
|
|
<Input
|
|
|
|
|
|
id="api_format"
|
|
|
|
|
|
v-model="form.api_format"
|
|
|
|
|
|
disabled
|
|
|
|
|
|
class="bg-muted"
|
|
|
|
|
|
/>
|
2025-12-12 16:15:36 +08:00
|
|
|
|
<p class="text-xs text-muted-foreground">
|
|
|
|
|
|
API 格式创建后不可修改
|
|
|
|
|
|
</p>
|
2025-12-10 20:52:44 +08:00
|
|
|
|
</template>
|
|
|
|
|
|
<template v-else>
|
2025-12-12 16:15:36 +08:00
|
|
|
|
<Select
|
|
|
|
|
|
v-model="form.api_format"
|
|
|
|
|
|
v-model:open="selectOpen"
|
|
|
|
|
|
required
|
|
|
|
|
|
>
|
2025-12-10 20:52:44 +08:00
|
|
|
|
<SelectTrigger>
|
|
|
|
|
|
<SelectValue placeholder="请选择 API 格式" />
|
|
|
|
|
|
</SelectTrigger>
|
|
|
|
|
|
<SelectContent>
|
|
|
|
|
|
<SelectItem
|
|
|
|
|
|
v-for="format in apiFormats"
|
|
|
|
|
|
:key="format.value"
|
|
|
|
|
|
:value="format.value"
|
|
|
|
|
|
>
|
|
|
|
|
|
{{ format.label }}
|
|
|
|
|
|
</SelectItem>
|
|
|
|
|
|
</SelectContent>
|
|
|
|
|
|
</Select>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<!-- API URL -->
|
|
|
|
|
|
<div class="space-y-2">
|
|
|
|
|
|
<Label for="base_url">API URL *</Label>
|
|
|
|
|
|
<Input
|
|
|
|
|
|
id="base_url"
|
|
|
|
|
|
v-model="form.base_url"
|
|
|
|
|
|
placeholder="https://api.example.com"
|
|
|
|
|
|
required
|
|
|
|
|
|
/>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<!-- 自定义路径 -->
|
|
|
|
|
|
<div class="space-y-2">
|
|
|
|
|
|
<Label for="custom_path">自定义请求路径(可选)</Label>
|
|
|
|
|
|
<Input
|
|
|
|
|
|
id="custom_path"
|
|
|
|
|
|
v-model="form.custom_path"
|
|
|
|
|
|
:placeholder="defaultPathPlaceholder"
|
|
|
|
|
|
/>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<!-- 请求配置 -->
|
|
|
|
|
|
<div class="space-y-4">
|
2025-12-12 16:15:36 +08:00
|
|
|
|
<h3 class="text-sm font-medium">
|
|
|
|
|
|
请求配置
|
|
|
|
|
|
</h3>
|
2025-12-10 20:52:44 +08:00
|
|
|
|
|
|
|
|
|
|
<div class="grid grid-cols-3 gap-4">
|
|
|
|
|
|
<div class="space-y-2">
|
|
|
|
|
|
<Label for="timeout">超时(秒)</Label>
|
|
|
|
|
|
<Input
|
|
|
|
|
|
id="timeout"
|
|
|
|
|
|
v-model.number="form.timeout"
|
|
|
|
|
|
type="number"
|
|
|
|
|
|
placeholder="300"
|
|
|
|
|
|
/>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<div class="space-y-2">
|
|
|
|
|
|
<Label for="max_retries">最大重试</Label>
|
|
|
|
|
|
<Input
|
|
|
|
|
|
id="max_retries"
|
|
|
|
|
|
v-model.number="form.max_retries"
|
|
|
|
|
|
type="number"
|
|
|
|
|
|
placeholder="3"
|
|
|
|
|
|
/>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<div class="space-y-2">
|
|
|
|
|
|
<Label for="max_concurrent">最大并发</Label>
|
|
|
|
|
|
<Input
|
|
|
|
|
|
id="max_concurrent"
|
|
|
|
|
|
:model-value="form.max_concurrent ?? ''"
|
|
|
|
|
|
type="number"
|
|
|
|
|
|
placeholder="无限制"
|
|
|
|
|
|
@update:model-value="(v) => form.max_concurrent = parseNumberInput(v)"
|
|
|
|
|
|
/>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<div class="grid grid-cols-2 gap-4">
|
|
|
|
|
|
<div class="space-y-2">
|
|
|
|
|
|
<Label for="rate_limit">速率限制(请求/分钟)</Label>
|
|
|
|
|
|
<Input
|
|
|
|
|
|
id="rate_limit"
|
|
|
|
|
|
:model-value="form.rate_limit ?? ''"
|
|
|
|
|
|
type="number"
|
|
|
|
|
|
placeholder="无限制"
|
|
|
|
|
|
@update:model-value="(v) => form.rate_limit = parseNumberInput(v)"
|
|
|
|
|
|
/>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</form>
|
|
|
|
|
|
|
|
|
|
|
|
<template #footer>
|
|
|
|
|
|
<Button
|
|
|
|
|
|
type="button"
|
|
|
|
|
|
variant="outline"
|
|
|
|
|
|
:disabled="loading"
|
2025-12-12 16:15:36 +08:00
|
|
|
|
@click="handleCancel"
|
2025-12-10 20:52:44 +08:00
|
|
|
|
>
|
|
|
|
|
|
取消
|
|
|
|
|
|
</Button>
|
|
|
|
|
|
<Button
|
|
|
|
|
|
:disabled="loading || !form.base_url || (!isEditMode && !form.api_format)"
|
2025-12-12 16:15:36 +08:00
|
|
|
|
@click="handleSubmit"
|
2025-12-10 20:52:44 +08:00
|
|
|
|
>
|
|
|
|
|
|
{{ loading ? (isEditMode ? '保存中...' : '创建中...') : (isEditMode ? '保存修改' : '创建') }}
|
|
|
|
|
|
</Button>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
</Dialog>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
|
|
|
|
|
|
<script setup lang="ts">
|
|
|
|
|
|
import { ref, computed, onMounted } from 'vue'
|
|
|
|
|
|
import { Dialog } from '@/components/ui'
|
|
|
|
|
|
import Button from '@/components/ui/button.vue'
|
|
|
|
|
|
import Input from '@/components/ui/input.vue'
|
|
|
|
|
|
import Label from '@/components/ui/label.vue'
|
|
|
|
|
|
import Select from '@/components/ui/select.vue'
|
|
|
|
|
|
import SelectTrigger from '@/components/ui/select-trigger.vue'
|
|
|
|
|
|
import SelectValue from '@/components/ui/select-value.vue'
|
|
|
|
|
|
import SelectContent from '@/components/ui/select-content.vue'
|
|
|
|
|
|
import SelectItem from '@/components/ui/select-item.vue'
|
|
|
|
|
|
import { Link, SquarePen } from 'lucide-vue-next'
|
|
|
|
|
|
import { useToast } from '@/composables/useToast'
|
|
|
|
|
|
import { useFormDialog } from '@/composables/useFormDialog'
|
|
|
|
|
|
import { parseNumberInput } from '@/utils/form'
|
|
|
|
|
|
import {
|
|
|
|
|
|
createEndpoint,
|
|
|
|
|
|
updateEndpoint,
|
|
|
|
|
|
type ProviderEndpoint,
|
|
|
|
|
|
type ProviderWithEndpointsSummary
|
|
|
|
|
|
} from '@/api/endpoints'
|
|
|
|
|
|
import { adminApi } from '@/api/admin'
|
|
|
|
|
|
|
|
|
|
|
|
const props = defineProps<{
|
|
|
|
|
|
modelValue: boolean
|
|
|
|
|
|
provider: ProviderWithEndpointsSummary | null
|
|
|
|
|
|
endpoint?: ProviderEndpoint | null // 编辑模式时传入
|
|
|
|
|
|
}>()
|
|
|
|
|
|
|
|
|
|
|
|
const emit = defineEmits<{
|
|
|
|
|
|
'update:modelValue': [value: boolean]
|
|
|
|
|
|
'endpoint-created': []
|
|
|
|
|
|
'endpoint-updated': []
|
|
|
|
|
|
}>()
|
|
|
|
|
|
|
|
|
|
|
|
const { success, error: showError } = useToast()
|
|
|
|
|
|
const loading = ref(false)
|
|
|
|
|
|
const selectOpen = ref(false)
|
|
|
|
|
|
|
|
|
|
|
|
// 内部状态
|
|
|
|
|
|
const internalOpen = computed(() => props.modelValue)
|
|
|
|
|
|
|
|
|
|
|
|
// 表单数据
|
|
|
|
|
|
const form = ref({
|
|
|
|
|
|
api_format: '',
|
|
|
|
|
|
base_url: '',
|
|
|
|
|
|
custom_path: '',
|
|
|
|
|
|
timeout: 300,
|
|
|
|
|
|
max_retries: 3,
|
|
|
|
|
|
max_concurrent: undefined as number | undefined,
|
|
|
|
|
|
rate_limit: undefined as number | undefined,
|
|
|
|
|
|
is_active: true
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
// API 格式列表
|
|
|
|
|
|
const apiFormats = ref<Array<{ value: string; label: string; default_path: string; aliases: string[] }>>([])
|
|
|
|
|
|
|
|
|
|
|
|
// 加载API格式列表
|
|
|
|
|
|
const loadApiFormats = async () => {
|
|
|
|
|
|
try {
|
|
|
|
|
|
const response = await adminApi.getApiFormats()
|
|
|
|
|
|
apiFormats.value = response.formats
|
|
|
|
|
|
} catch (error) {
|
|
|
|
|
|
console.error('加载API格式失败:', error)
|
|
|
|
|
|
if (!isEditMode.value) {
|
|
|
|
|
|
showError('加载API格式失败', '错误')
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 根据选择的 API 格式计算默认路径
|
|
|
|
|
|
const defaultPath = computed(() => {
|
|
|
|
|
|
const format = apiFormats.value.find(f => f.value === form.value.api_format)
|
|
|
|
|
|
return format?.default_path || '/'
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
// 动态 placeholder
|
|
|
|
|
|
const defaultPathPlaceholder = computed(() => {
|
|
|
|
|
|
return `留空使用默认路径:${defaultPath.value}`
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
// 组件挂载时加载API格式
|
|
|
|
|
|
onMounted(() => {
|
|
|
|
|
|
loadApiFormats()
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
// 重置表单
|
|
|
|
|
|
function resetForm() {
|
|
|
|
|
|
form.value = {
|
|
|
|
|
|
api_format: '',
|
|
|
|
|
|
base_url: '',
|
|
|
|
|
|
custom_path: '',
|
|
|
|
|
|
timeout: 300,
|
|
|
|
|
|
max_retries: 3,
|
|
|
|
|
|
max_concurrent: undefined,
|
|
|
|
|
|
rate_limit: undefined,
|
|
|
|
|
|
is_active: true
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 加载端点数据(编辑模式)
|
|
|
|
|
|
function loadEndpointData() {
|
|
|
|
|
|
if (!props.endpoint) return
|
|
|
|
|
|
|
|
|
|
|
|
form.value = {
|
|
|
|
|
|
api_format: props.endpoint.api_format,
|
|
|
|
|
|
base_url: props.endpoint.base_url,
|
|
|
|
|
|
custom_path: props.endpoint.custom_path || '',
|
|
|
|
|
|
timeout: props.endpoint.timeout,
|
|
|
|
|
|
max_retries: props.endpoint.max_retries,
|
|
|
|
|
|
max_concurrent: props.endpoint.max_concurrent || undefined,
|
|
|
|
|
|
rate_limit: props.endpoint.rate_limit || undefined,
|
|
|
|
|
|
is_active: props.endpoint.is_active
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 使用 useFormDialog 统一处理对话框逻辑
|
|
|
|
|
|
const { isEditMode, handleDialogUpdate, handleCancel } = useFormDialog({
|
|
|
|
|
|
isOpen: () => props.modelValue,
|
|
|
|
|
|
entity: () => props.endpoint,
|
|
|
|
|
|
isLoading: loading,
|
|
|
|
|
|
onClose: () => emit('update:modelValue', false),
|
|
|
|
|
|
loadData: loadEndpointData,
|
|
|
|
|
|
resetForm,
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
// 提交表单
|
|
|
|
|
|
const handleSubmit = async () => {
|
|
|
|
|
|
if (!props.provider && !props.endpoint) return
|
|
|
|
|
|
|
|
|
|
|
|
loading.value = true
|
|
|
|
|
|
try {
|
|
|
|
|
|
if (isEditMode.value && props.endpoint) {
|
|
|
|
|
|
// 更新端点
|
|
|
|
|
|
await updateEndpoint(props.endpoint.id, {
|
|
|
|
|
|
base_url: form.value.base_url,
|
|
|
|
|
|
custom_path: form.value.custom_path || undefined,
|
|
|
|
|
|
timeout: form.value.timeout,
|
|
|
|
|
|
max_retries: form.value.max_retries,
|
|
|
|
|
|
max_concurrent: form.value.max_concurrent,
|
|
|
|
|
|
rate_limit: form.value.rate_limit,
|
|
|
|
|
|
is_active: form.value.is_active
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
success('端点已更新', '保存成功')
|
|
|
|
|
|
emit('endpoint-updated')
|
|
|
|
|
|
} else if (props.provider) {
|
|
|
|
|
|
// 创建端点
|
|
|
|
|
|
await createEndpoint(props.provider.id, {
|
|
|
|
|
|
provider_id: props.provider.id,
|
|
|
|
|
|
api_format: form.value.api_format,
|
|
|
|
|
|
base_url: form.value.base_url,
|
|
|
|
|
|
custom_path: form.value.custom_path || undefined,
|
|
|
|
|
|
timeout: form.value.timeout,
|
|
|
|
|
|
max_retries: form.value.max_retries,
|
|
|
|
|
|
max_concurrent: form.value.max_concurrent,
|
|
|
|
|
|
rate_limit: form.value.rate_limit,
|
|
|
|
|
|
is_active: form.value.is_active
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
success('端点创建成功', '成功')
|
|
|
|
|
|
emit('endpoint-created')
|
|
|
|
|
|
resetForm()
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
emit('update:modelValue', false)
|
|
|
|
|
|
} catch (error: any) {
|
|
|
|
|
|
const action = isEditMode.value ? '更新' : '创建'
|
|
|
|
|
|
showError(error.response?.data?.detail || `${action}端点失败`, '错误')
|
|
|
|
|
|
} finally {
|
|
|
|
|
|
loading.value = false
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
</script>
|