feat: differentiate fetch models error messages by failure type

Distinguish between missing API key, missing endpoint, auth failure,
unsupported provider (404/405), and timeout errors instead of showing
a generic failure toast for all cases.
This commit is contained in:
Jason
2026-04-03 23:40:49 +08:00
parent f200feebe4
commit 84998aa217
9 changed files with 119 additions and 18 deletions
@@ -33,7 +33,11 @@ import {
copilotGetModelsForAccount,
} from "@/lib/api/copilot";
import type { CopilotModel } from "@/lib/api/copilot";
import { fetchModelsForConfig, type FetchedModel } from "@/lib/api/model-fetch";
import {
fetchModelsForConfig,
showFetchModelsError,
type FetchedModel,
} from "@/lib/api/model-fetch";
import type {
ProviderCategory,
ClaudeApiFormat,
@@ -186,7 +190,10 @@ export function ClaudeFormFields({
const handleFetchModels = useCallback(() => {
if (!baseUrl || !apiKey) {
toast.error(t("providerForm.fetchModelsFailed"));
showFetchModelsError(null, t, {
hasApiKey: !!apiKey,
hasBaseUrl: !!baseUrl,
});
return;
}
setIsFetchingModels(true);
@@ -203,7 +210,7 @@ export function ClaudeFormFields({
})
.catch((err) => {
console.warn("[ModelFetch] Failed:", err);
toast.error(t("providerForm.fetchModelsFailed"));
showFetchModelsError(err, t);
})
.finally(() => setIsFetchingModels(false));
}, [baseUrl, apiKey, isFullUrl, t]);
@@ -5,7 +5,11 @@ import { toast } from "sonner";
import { Download, Loader2 } from "lucide-react";
import EndpointSpeedTest from "./EndpointSpeedTest";
import { ApiKeySection, EndpointField, ModelInputWithFetch } from "./shared";
import { fetchModelsForConfig, type FetchedModel } from "@/lib/api/model-fetch";
import {
fetchModelsForConfig,
showFetchModelsError,
type FetchedModel,
} from "@/lib/api/model-fetch";
import type { ProviderCategory } from "@/types";
interface EndpointCandidate {
@@ -75,7 +79,10 @@ export function CodexFormFields({
const handleFetchModels = useCallback(() => {
if (!codexBaseUrl || !codexApiKey) {
toast.error(t("providerForm.fetchModelsFailed"));
showFetchModelsError(null, t, {
hasApiKey: !!codexApiKey,
hasBaseUrl: !!codexBaseUrl,
});
return;
}
setIsFetchingModels(true);
@@ -92,7 +99,7 @@ export function CodexFormFields({
})
.catch((err) => {
console.warn("[ModelFetch] Failed:", err);
toast.error(t("providerForm.fetchModelsFailed"));
showFetchModelsError(err, t);
})
.finally(() => setIsFetchingModels(false));
}, [codexBaseUrl, codexApiKey, isFullUrl, t]);
@@ -6,7 +6,11 @@ import { Button } from "@/components/ui/button";
import { toast } from "sonner";
import EndpointSpeedTest from "./EndpointSpeedTest";
import { ApiKeySection, EndpointField, ModelInputWithFetch } from "./shared";
import { fetchModelsForConfig, type FetchedModel } from "@/lib/api/model-fetch";
import {
fetchModelsForConfig,
showFetchModelsError,
type FetchedModel,
} from "@/lib/api/model-fetch";
import type { ProviderCategory } from "@/types";
interface EndpointCandidate {
@@ -74,7 +78,10 @@ export function GeminiFormFields({
const handleFetchModels = useCallback(() => {
if (!baseUrl || !apiKey) {
toast.error(t("providerForm.fetchModelsFailed"));
showFetchModelsError(null, t, {
hasApiKey: !!apiKey,
hasBaseUrl: !!baseUrl,
});
return;
}
setIsFetchingModels(true);
@@ -91,7 +98,7 @@ export function GeminiFormFields({
})
.catch((err) => {
console.warn("[ModelFetch] Failed:", err);
toast.error(t("providerForm.fetchModelsFailed"));
showFetchModelsError(err, t);
})
.finally(() => setIsFetchingModels(false));
}, [baseUrl, apiKey, t]);
@@ -35,7 +35,11 @@ import {
} from "@/components/ui/dropdown-menu";
import { Checkbox } from "@/components/ui/checkbox";
import { ApiKeySection } from "./shared";
import { fetchModelsForConfig, type FetchedModel } from "@/lib/api/model-fetch";
import {
fetchModelsForConfig,
showFetchModelsError,
type FetchedModel,
} from "@/lib/api/model-fetch";
import { openclawApiProtocols } from "@/config/openclawProviderPresets";
import type { ProviderCategory, OpenClawModel } from "@/types";
@@ -129,7 +133,10 @@ export function OpenClawFormFields({
// Fetch models from API
const handleFetchModels = useCallback(() => {
if (!baseUrl || !apiKey) {
toast.error(t("providerForm.fetchModelsFailed"));
showFetchModelsError(null, t, {
hasApiKey: !!apiKey,
hasBaseUrl: !!baseUrl,
});
return;
}
setIsFetchingModels(true);
@@ -146,7 +153,7 @@ export function OpenClawFormFields({
})
.catch((err) => {
console.warn("[ModelFetch] Failed:", err);
toast.error(t("providerForm.fetchModelsFailed"));
showFetchModelsError(err, t);
})
.finally(() => setIsFetchingModels(false));
}, [baseUrl, apiKey, t]);
@@ -28,7 +28,11 @@ import {
DropdownMenuTrigger,
} from "@/components/ui/dropdown-menu";
import { ApiKeySection } from "./shared";
import { fetchModelsForConfig, type FetchedModel } from "@/lib/api/model-fetch";
import {
fetchModelsForConfig,
showFetchModelsError,
type FetchedModel,
} from "@/lib/api/model-fetch";
import { opencodeNpmPackages } from "@/config/opencodeProviderPresets";
import { cn } from "@/lib/utils";
import {
@@ -247,7 +251,10 @@ export function OpenCodeFormFields({
const handleFetchModels = useCallback(() => {
if (!baseUrl || !apiKey) {
toast.error(t("providerForm.fetchModelsFailed"));
showFetchModelsError(null, t, {
hasApiKey: !!apiKey,
hasBaseUrl: !!baseUrl,
});
return;
}
setIsFetchingModels(true);
@@ -264,7 +271,7 @@ export function OpenCodeFormFields({
})
.catch((err) => {
console.warn("[ModelFetch] Failed:", err);
toast.error(t("providerForm.fetchModelsFailed"));
showFetchModelsError(err, t);
})
.finally(() => setIsFetchingModels(false));
}, [baseUrl, apiKey, t]);
+7 -1
View File
@@ -795,7 +795,13 @@
"fetchingModels": "Fetching...",
"fetchModelsSuccess": "Found {{count}} models",
"fetchModelsFailed": "Failed to fetch models",
"fetchModelsEmpty": "No models found"
"fetchModelsEmpty": "No models found",
"fetchModelsNeedApiKey": "Please fill in API Key first",
"fetchModelsNeedEndpoint": "Please fill in API endpoint first",
"fetchModelsNeedConfig": "Please fill in API endpoint and API Key first",
"fetchModelsAuthFailed": "API Key is invalid or lacks permission",
"fetchModelsNotSupported": "This provider does not support fetching model list",
"fetchModelsTimeout": "Request timed out, please check network connection"
},
"copilot": {
"authSection": "GitHub Copilot Authentication",
+7 -1
View File
@@ -795,7 +795,13 @@
"fetchingModels": "取得中...",
"fetchModelsSuccess": "{{count}}件のモデルを取得",
"fetchModelsFailed": "モデル一覧の取得に失敗しました",
"fetchModelsEmpty": "モデルが見つかりません"
"fetchModelsEmpty": "モデルが見つかりません",
"fetchModelsNeedApiKey": "先に API Key を入力してください",
"fetchModelsNeedEndpoint": "先に API エンドポイントを入力してください",
"fetchModelsNeedConfig": "先に API エンドポイントと API Key を入力してください",
"fetchModelsAuthFailed": "API Key が無効か、権限がありません",
"fetchModelsNotSupported": "このプロバイダーはモデル一覧の取得に対応していません",
"fetchModelsTimeout": "リクエストがタイムアウトしました。ネットワーク接続を確認してください"
},
"copilot": {
"authSection": "GitHub Copilot 認証",
+7 -1
View File
@@ -795,7 +795,13 @@
"fetchingModels": "正在获取...",
"fetchModelsSuccess": "获取到 {{count}} 个模型",
"fetchModelsFailed": "获取模型列表失败",
"fetchModelsEmpty": "未找到可用模型"
"fetchModelsEmpty": "未找到可用模型",
"fetchModelsNeedApiKey": "请先填写 API Key",
"fetchModelsNeedEndpoint": "请先填写 API 端点",
"fetchModelsNeedConfig": "请先填写 API 端点和 API Key",
"fetchModelsAuthFailed": "API Key 无效或无权限",
"fetchModelsNotSupported": "该供应商不支持获取模型列表",
"fetchModelsTimeout": "请求超时,请检查网络连接"
},
"copilot": {
"authSection": "GitHub Copilot 认证",
+48
View File
@@ -1,4 +1,6 @@
import { invoke } from "@tauri-apps/api/core";
import type { TFunction } from "i18next";
import { toast } from "sonner";
export interface FetchedModel {
id: string;
@@ -18,3 +20,49 @@ export async function fetchModelsForConfig(
): Promise<FetchedModel[]> {
return invoke("fetch_models_for_config", { baseUrl, apiKey, isFullUrl });
}
/**
* 根据错误类型显示对应的 toast 提示
*/
export function showFetchModelsError(
err: unknown,
t: TFunction,
opts?: { hasApiKey: boolean; hasBaseUrl: boolean },
): void {
// 前端预检:缺少必填字段
if (opts && !opts.hasBaseUrl && !opts.hasApiKey) {
toast.error(t("providerForm.fetchModelsNeedConfig"));
return;
}
if (opts && !opts.hasApiKey) {
toast.error(t("providerForm.fetchModelsNeedApiKey"));
return;
}
if (opts && !opts.hasBaseUrl) {
toast.error(t("providerForm.fetchModelsNeedEndpoint"));
return;
}
// 解析后端错误字符串
const msg = String(err);
if (msg.includes("HTTP 401") || msg.includes("HTTP 403")) {
toast.error(t("providerForm.fetchModelsAuthFailed"));
return;
}
if (msg.includes("HTTP 404") || msg.includes("HTTP 405")) {
toast.error(t("providerForm.fetchModelsNotSupported"));
return;
}
if (msg.includes("timeout") || msg.includes("timed out")) {
toast.error(t("providerForm.fetchModelsTimeout"));
return;
}
if (msg.includes("Failed to parse")) {
toast.error(t("providerForm.fetchModelsNotSupported"));
return;
}
// 通用兜底
toast.error(t("providerForm.fetchModelsFailed"));
}