import { useMemo } from "react"; import type { AppType } from "@/lib/api"; import type { ProviderPreset } from "@/config/providerPresets"; import type { CodexProviderPreset } from "@/config/codexProviderPresets"; import type { ProviderMeta, EndpointCandidate } from "@/types"; type PresetEntry = { id: string; preset: ProviderPreset | CodexProviderPreset; }; interface UseSpeedTestEndpointsProps { appType: AppType; selectedPresetId: string | null; presetEntries: PresetEntry[]; baseUrl: string; codexBaseUrl: string; initialData?: { settingsConfig?: Record; meta?: ProviderMeta; }; } /** * 收集端点测速弹窗的初始端点列表 * * 收集来源: * 1. 编辑模式:从 meta.custom_endpoints 读取已保存的端点(优先) * 2. 当前选中的 Base URL * 3. 编辑模式下的初始数据 URL * 4. 预设中的 endpointCandidates */ export function useSpeedTestEndpoints({ appType, selectedPresetId, presetEntries, baseUrl, codexBaseUrl, initialData, }: UseSpeedTestEndpointsProps) { const claudeEndpoints = useMemo(() => { if (appType !== "claude") return []; const map = new Map(); // 所有端点都标记为 isCustom: true,给用户完全的管理自由 const add = (url?: string) => { if (!url) return; const sanitized = url.trim().replace(/\/+$/, ""); if (!sanitized || map.has(sanitized)) return; map.set(sanitized, { url: sanitized, isCustom: true }); }; // 1. 编辑模式:从 meta.custom_endpoints 读取已保存的端点(优先) if (initialData?.meta?.custom_endpoints) { const customEndpoints = initialData.meta.custom_endpoints; for (const url of Object.keys(customEndpoints)) { add(url); } } // 2. 当前 Base URL if (baseUrl) { add(baseUrl); } // 3. 编辑模式:初始数据中的 URL if (initialData && typeof initialData.settingsConfig === "object") { const configEnv = initialData.settingsConfig as { env?: { ANTHROPIC_BASE_URL?: string }; }; const envUrl = configEnv.env?.ANTHROPIC_BASE_URL; if (typeof envUrl === "string") { add(envUrl); } } // 4. 预设中的 endpointCandidates(也允许用户删除) if (selectedPresetId && selectedPresetId !== "custom") { const entry = presetEntries.find((item) => item.id === selectedPresetId); if (entry) { const preset = entry.preset as ProviderPreset; // 添加预设自己的 baseUrl const presetEnv = preset.settingsConfig as { env?: { ANTHROPIC_BASE_URL?: string }; }; if (presetEnv.env?.ANTHROPIC_BASE_URL) { add(presetEnv.env.ANTHROPIC_BASE_URL); } // 添加预设的候选端点 if (preset.endpointCandidates) { preset.endpointCandidates.forEach((url) => add(url)); } } } return Array.from(map.values()); }, [appType, baseUrl, initialData, selectedPresetId, presetEntries]); const codexEndpoints = useMemo(() => { if (appType !== "codex") return []; const map = new Map(); // 所有端点都标记为 isCustom: true,给用户完全的管理自由 const add = (url?: string) => { if (!url) return; const sanitized = url.trim().replace(/\/+$/, ""); if (!sanitized || map.has(sanitized)) return; map.set(sanitized, { url: sanitized, isCustom: true }); }; // 1. 编辑模式:从 meta.custom_endpoints 读取已保存的端点(优先) if (initialData?.meta?.custom_endpoints) { const customEndpoints = initialData.meta.custom_endpoints; for (const url of Object.keys(customEndpoints)) { add(url); } } // 2. 当前 Codex Base URL if (codexBaseUrl) { add(codexBaseUrl); } // 3. 编辑模式:初始数据中的 URL const initialCodexConfig = initialData?.settingsConfig as | { config?: string; } | undefined; const configStr = initialCodexConfig?.config ?? ""; // 从 TOML 中提取 base_url const match = /base_url\s*=\s*["']([^"']+)["']/i.exec(configStr); if (match?.[1]) { add(match[1]); } // 4. 预设中的 endpointCandidates(也允许用户删除) if (selectedPresetId && selectedPresetId !== "custom") { const entry = presetEntries.find((item) => item.id === selectedPresetId); if (entry) { const preset = entry.preset as CodexProviderPreset; // 添加预设自己的 baseUrl const presetConfig = preset.config || ""; const presetMatch = /base_url\s*=\s*["']([^"']+)["']/i.exec( presetConfig, ); if (presetMatch?.[1]) { add(presetMatch[1]); } // 添加预设的候选端点 if (preset.endpointCandidates) { preset.endpointCandidates.forEach((url) => add(url)); } } } return Array.from(map.values()); }, [appType, codexBaseUrl, initialData, selectedPresetId, presetEntries]); return appType === "codex" ? codexEndpoints : claudeEndpoints; }