mirror of
				https://github.com/ChatGPTNextWeb/ChatGPT-Next-Web.git
				synced 2025-11-04 16:23:41 +08:00 
			
		
		
		
	Compare commits
	
		
			1 Commits
		
	
	
		
			c0ec019082
			...
			Leizhenpen
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 
						 | 
					b95b1ac6f3 | 
@@ -81,7 +81,3 @@ SILICONFLOW_API_KEY=
 | 
			
		||||
 | 
			
		||||
### siliconflow Api url (optional)
 | 
			
		||||
SILICONFLOW_URL=
 | 
			
		||||
 | 
			
		||||
HUAWEI_URL=
 | 
			
		||||
 | 
			
		||||
HUAWEI_API_KEY=
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										21
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										21
									
								
								README.md
									
									
									
									
									
								
							@@ -7,7 +7,7 @@
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
<h1 align="center">NextChat</h1>
 | 
			
		||||
<h1 align="center">NextChat (ChatGPT Next Web)</h1>
 | 
			
		||||
 | 
			
		||||
English / [简体中文](./README_CN.md)
 | 
			
		||||
 | 
			
		||||
@@ -22,6 +22,7 @@ English / [简体中文](./README_CN.md)
 | 
			
		||||
[![MacOS][MacOS-image]][download-url]
 | 
			
		||||
[![Linux][Linux-image]][download-url]
 | 
			
		||||
 | 
			
		||||
[NextChatAI](https://nextchat.dev/chat?utm_source=readme) / [Web App Demo](https://app.nextchat.dev) / [Desktop App](https://github.com/Yidadaa/ChatGPT-Next-Web/releases) 
 | 
			
		||||
[NextChatAI](https://nextchat.club?utm_source=readme) / [Web App Demo](https://app.nextchat.dev) / [Desktop App](https://github.com/Yidadaa/ChatGPT-Next-Web/releases) / [Discord](https://discord.gg/YCkeafCafC) / [Enterprise Edition](#enterprise-edition) / [Twitter](https://twitter.com/NextChatDev)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@@ -40,6 +41,24 @@ English / [简体中文](./README_CN.md)
 | 
			
		||||
 | 
			
		||||
</div>
 | 
			
		||||
 | 
			
		||||
## 👋 Hey, NextChat is going to develop a native app!
 | 
			
		||||
 | 
			
		||||
> This week we are going to start working on iOS and Android APP, and we want to find some reliable friends to do it together!
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
✨ Several key points:
 | 
			
		||||
 | 
			
		||||
- Starting from 0, you are a veteran
 | 
			
		||||
- Completely open source, not hidden
 | 
			
		||||
- Native development, pursuing the ultimate experience
 | 
			
		||||
 | 
			
		||||
Will you come and do something together? 😎
 | 
			
		||||
 | 
			
		||||
https://github.com/ChatGPTNextWeb/NextChat/issues/6269
 | 
			
		||||
 | 
			
		||||
#Seeking for talents is thirsty #lack of people
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
## 🥳 Cheer for DeepSeek, China's AI star!
 | 
			
		||||
 > Purpose-Built UI for DeepSeek Reasoner Model
 | 
			
		||||
 
 | 
			
		||||
 
 | 
			
		||||
@@ -15,7 +15,6 @@ import { handle as siliconflowHandler } from "../../siliconflow";
 | 
			
		||||
import { handle as xaiHandler } from "../../xai";
 | 
			
		||||
import { handle as chatglmHandler } from "../../glm";
 | 
			
		||||
import { handle as proxyHandler } from "../../proxy";
 | 
			
		||||
import { handle as huaweiHandler } from "../../huawei";
 | 
			
		||||
 | 
			
		||||
async function handle(
 | 
			
		||||
  req: NextRequest,
 | 
			
		||||
@@ -53,8 +52,6 @@ async function handle(
 | 
			
		||||
      return siliconflowHandler(req, { params });
 | 
			
		||||
    case ApiPath.OpenAI:
 | 
			
		||||
      return openaiHandler(req, { params });
 | 
			
		||||
    case ApiPath.Huawei:
 | 
			
		||||
      return huaweiHandler(req, { params });
 | 
			
		||||
    default:
 | 
			
		||||
      return proxyHandler(req, { params });
 | 
			
		||||
  }
 | 
			
		||||
 
 | 
			
		||||
@@ -1,176 +0,0 @@
 | 
			
		||||
import { getServerSideConfig } from "@/app/config/server";
 | 
			
		||||
import {
 | 
			
		||||
  HUAWEI_BASE_URL,
 | 
			
		||||
  ApiPath,
 | 
			
		||||
  ModelProvider,
 | 
			
		||||
  Huawei,
 | 
			
		||||
} from "@/app/constant";
 | 
			
		||||
import { prettyObject } from "@/app/utils/format";
 | 
			
		||||
import { NextRequest, NextResponse } from "next/server";
 | 
			
		||||
import { auth } from "@/app/api/auth";
 | 
			
		||||
 | 
			
		||||
const serverConfig = getServerSideConfig();
 | 
			
		||||
 | 
			
		||||
export async function handle(
 | 
			
		||||
  req: NextRequest,
 | 
			
		||||
  { params }: { params: { path: string[] } },
 | 
			
		||||
) {
 | 
			
		||||
  console.log("[Huawei Route] params ", params);
 | 
			
		||||
 | 
			
		||||
  if (req.method === "OPTIONS") {
 | 
			
		||||
    return NextResponse.json({ body: "OK" }, { status: 200 });
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  const authResult = auth(req, ModelProvider.Huawei);
 | 
			
		||||
  if (authResult.error) {
 | 
			
		||||
    return NextResponse.json(authResult, {
 | 
			
		||||
      status: 401,
 | 
			
		||||
    });
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  try {
 | 
			
		||||
    const response = await request(req);
 | 
			
		||||
    return response;
 | 
			
		||||
  } catch (e) {
 | 
			
		||||
    console.error("[Huawei] ", e);
 | 
			
		||||
    return NextResponse.json(prettyObject(e));
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
async function request(req: NextRequest) {
 | 
			
		||||
  const controller = new AbortController();
 | 
			
		||||
 | 
			
		||||
  let path = `${req.nextUrl.pathname}`.replaceAll(ApiPath.Huawei, "");
 | 
			
		||||
  const bodyText = await req.text();
 | 
			
		||||
  const body = JSON.parse(bodyText);
 | 
			
		||||
  let modelName = body.model as string;
 | 
			
		||||
 | 
			
		||||
  // 先用原始 modelName 获取 charUrl
 | 
			
		||||
  let baseUrl: string;
 | 
			
		||||
  let endpoint = "";
 | 
			
		||||
  if (modelName === "DeepSeek-V3") {
 | 
			
		||||
    endpoint = "271c9332-4aa6-4ff5-95b3-0cf8bd94c394";
 | 
			
		||||
  }
 | 
			
		||||
  if (modelName === "DeepSeek-R1") {
 | 
			
		||||
    endpoint = "8a062fd4-7367-4ab4-a936-5eeb8fb821c4";
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
  let charUrl = HUAWEI_BASE_URL.concat("/")
 | 
			
		||||
    .concat(endpoint)
 | 
			
		||||
    .concat("/v1/chat/completions")
 | 
			
		||||
    .replace(/(?<!:)\/+/g, "/"); // 只替换不在 :// 后面的多个斜杠
 | 
			
		||||
  console.log(`current charUrl name:${charUrl}`);
 | 
			
		||||
  baseUrl = charUrl;
 | 
			
		||||
 | 
			
		||||
  // 处理请求体:1. 移除 system role 消息 2. 修改 model 名称格式
 | 
			
		||||
  const modifiedBody = {
 | 
			
		||||
    messages: body.messages
 | 
			
		||||
      .map((msg: any) => ({
 | 
			
		||||
        role: msg.role,
 | 
			
		||||
        content: msg.content,
 | 
			
		||||
      }))
 | 
			
		||||
      .filter((msg: any) => msg.role !== "system"),
 | 
			
		||||
    model: modelName.replace(/^(DeepSeek-(?:R1|V3)).*$/, "$1"), // 只保留 DeepSeek-R1 或 DeepSeek-V3
 | 
			
		||||
    stream: body.stream,
 | 
			
		||||
    temperature: body.temperature,
 | 
			
		||||
    presence_penalty: body.presence_penalty,
 | 
			
		||||
    frequency_penalty: body.frequency_penalty,
 | 
			
		||||
    top_p: body.top_p,
 | 
			
		||||
  };
 | 
			
		||||
  const modifiedBodyText = JSON.stringify(modifiedBody);
 | 
			
		||||
  console.log("Modified request body:", modifiedBodyText);
 | 
			
		||||
 | 
			
		||||
  // if(!baseUrl){
 | 
			
		||||
  //     baseUrl = HUAWEI_BASE_URL
 | 
			
		||||
  // }
 | 
			
		||||
  // baseUrl = Huawei.ChatPath(modelName) || serverConfig.huaweiUrl || HUAWEI_BASE_URL;
 | 
			
		||||
  console.log(
 | 
			
		||||
    `current model name:${modelName},current api path:${baseUrl}.........`,
 | 
			
		||||
  );
 | 
			
		||||
  if (!baseUrl.startsWith("http")) {
 | 
			
		||||
    baseUrl = `https://${baseUrl}`;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  if (baseUrl.endsWith("/")) {
 | 
			
		||||
    baseUrl = baseUrl.slice(0, -1);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  console.log("[Proxy] ", path);
 | 
			
		||||
  console.log("[Base Url]", baseUrl);
 | 
			
		||||
 | 
			
		||||
  const timeoutId = setTimeout(
 | 
			
		||||
    () => {
 | 
			
		||||
      controller.abort();
 | 
			
		||||
    },
 | 
			
		||||
    10 * 60 * 1000,
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
  // 如果 baseUrl 来自 Huawei.ChatPath,则不需要再拼接 path
 | 
			
		||||
  let fetchUrl = baseUrl.includes(HUAWEI_BASE_URL)
 | 
			
		||||
    ? baseUrl
 | 
			
		||||
    : `${baseUrl}${path}`;
 | 
			
		||||
 | 
			
		||||
  const headers: Record<string, string> = {
 | 
			
		||||
    "Content-Type": "application/json",
 | 
			
		||||
    Authorization: req.headers.get("Authorization") ?? "",
 | 
			
		||||
    "X-Forwarded-For": req.headers.get("X-Forwarded-For") ?? "",
 | 
			
		||||
    "X-Real-IP": req.headers.get("X-Real-IP") ?? "",
 | 
			
		||||
    "User-Agent": req.headers.get("User-Agent") ?? "",
 | 
			
		||||
  };
 | 
			
		||||
  console.debug(`headers.Authorization:${headers.Authorization}`);
 | 
			
		||||
  console.debug("serverConfig.huaweiApiKey: *****");
 | 
			
		||||
  // 如果没有 Authorization header,使用系统配置的 API key
 | 
			
		||||
 | 
			
		||||
  headers.Authorization = `Bearer ${serverConfig.huaweiApiKey}`;
 | 
			
		||||
 | 
			
		||||
  // #1815 try to refuse some request to some models
 | 
			
		||||
  // if (serverConfig.customModels) {
 | 
			
		||||
  //     try {
 | 
			
		||||
  //         const jsonBody = JSON.parse(bodyText);  // 直接使用已解析的 body
 | 
			
		||||
  //
 | 
			
		||||
  //         if (
 | 
			
		||||
  //             isModelNotavailableInServer(
 | 
			
		||||
  //                 serverConfig.customModels,
 | 
			
		||||
  //                 jsonBody?.model as string,
 | 
			
		||||
  //                 ServiceProvider.Huawei as string,
 | 
			
		||||
  //             )
 | 
			
		||||
  //         ) {
 | 
			
		||||
  //             return NextResponse.json(
 | 
			
		||||
  //                 {
 | 
			
		||||
  //                     error: true,
 | 
			
		||||
  //                     message: `you are not allowed to use ${jsonBody?.model} model`,
 | 
			
		||||
  //                 },
 | 
			
		||||
  //                 {
 | 
			
		||||
  //                     status: 403,
 | 
			
		||||
  //                 },
 | 
			
		||||
  //             );
 | 
			
		||||
  //         }
 | 
			
		||||
  //     } catch (e) {
 | 
			
		||||
  //         console.error(`[Huawei] filter`, e);
 | 
			
		||||
  //     }
 | 
			
		||||
  // }
 | 
			
		||||
  try {
 | 
			
		||||
    const res = await fetch(fetchUrl, {
 | 
			
		||||
      headers,
 | 
			
		||||
      method: req.method,
 | 
			
		||||
      body: modifiedBodyText,
 | 
			
		||||
      redirect: "manual",
 | 
			
		||||
      // @ts-ignore
 | 
			
		||||
      duplex: "half",
 | 
			
		||||
      signal: controller.signal,
 | 
			
		||||
    });
 | 
			
		||||
    const newHeaders = new Headers(res.headers);
 | 
			
		||||
    newHeaders.delete("www-authenticate");
 | 
			
		||||
    // to disable nginx buffering
 | 
			
		||||
    newHeaders.set("X-Accel-Buffering", "no");
 | 
			
		||||
 | 
			
		||||
    return new Response(res.body, {
 | 
			
		||||
      status: res.status,
 | 
			
		||||
      statusText: res.statusText,
 | 
			
		||||
      headers: newHeaders,
 | 
			
		||||
    });
 | 
			
		||||
  } finally {
 | 
			
		||||
    clearTimeout(timeoutId);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
export { Huawei };
 | 
			
		||||
@@ -24,7 +24,6 @@ import { DeepSeekApi } from "./platforms/deepseek";
 | 
			
		||||
import { XAIApi } from "./platforms/xai";
 | 
			
		||||
import { ChatGLMApi } from "./platforms/glm";
 | 
			
		||||
import { SiliconflowApi } from "./platforms/siliconflow";
 | 
			
		||||
import { HuaweiApi } from "./platforms/huawei";
 | 
			
		||||
 | 
			
		||||
export const ROLES = ["system", "user", "assistant"] as const;
 | 
			
		||||
export type MessageRole = (typeof ROLES)[number];
 | 
			
		||||
@@ -41,11 +40,6 @@ export interface MultimodalContent {
 | 
			
		||||
  };
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export interface MultimodalContentForAlibaba {
 | 
			
		||||
  text?: string;
 | 
			
		||||
  image?: string;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export interface RequestMessage {
 | 
			
		||||
  role: MessageRole;
 | 
			
		||||
  content: string | MultimodalContent[];
 | 
			
		||||
@@ -174,9 +168,6 @@ export class ClientApi {
 | 
			
		||||
      case ModelProvider.SiliconFlow:
 | 
			
		||||
        this.llm = new SiliconflowApi();
 | 
			
		||||
        break;
 | 
			
		||||
      case ModelProvider.Huawei:
 | 
			
		||||
        this.llm = new HuaweiApi();
 | 
			
		||||
        break;
 | 
			
		||||
      default:
 | 
			
		||||
        this.llm = new ChatGPTApi();
 | 
			
		||||
    }
 | 
			
		||||
@@ -269,7 +260,6 @@ export function getHeaders(ignoreHeaders: boolean = false) {
 | 
			
		||||
    const isChatGLM = modelConfig.providerName === ServiceProvider.ChatGLM;
 | 
			
		||||
    const isSiliconFlow =
 | 
			
		||||
      modelConfig.providerName === ServiceProvider.SiliconFlow;
 | 
			
		||||
    const isHuawei = modelConfig.providerName == ServiceProvider.Huawei;
 | 
			
		||||
    const isEnabledAccessControl = accessStore.enabledAccessControl();
 | 
			
		||||
    const apiKey = isGoogle
 | 
			
		||||
      ? accessStore.googleApiKey
 | 
			
		||||
@@ -295,8 +285,6 @@ export function getHeaders(ignoreHeaders: boolean = false) {
 | 
			
		||||
      ? accessStore.iflytekApiKey && accessStore.iflytekApiSecret
 | 
			
		||||
        ? accessStore.iflytekApiKey + ":" + accessStore.iflytekApiSecret
 | 
			
		||||
        : ""
 | 
			
		||||
      : isHuawei
 | 
			
		||||
      ? accessStore.huaweiApiKey
 | 
			
		||||
      : accessStore.openaiApiKey;
 | 
			
		||||
    return {
 | 
			
		||||
      isGoogle,
 | 
			
		||||
@@ -311,7 +299,6 @@ export function getHeaders(ignoreHeaders: boolean = false) {
 | 
			
		||||
      isXAI,
 | 
			
		||||
      isChatGLM,
 | 
			
		||||
      isSiliconFlow,
 | 
			
		||||
      isHuawei,
 | 
			
		||||
      apiKey,
 | 
			
		||||
      isEnabledAccessControl,
 | 
			
		||||
    };
 | 
			
		||||
@@ -340,7 +327,6 @@ export function getHeaders(ignoreHeaders: boolean = false) {
 | 
			
		||||
    isXAI,
 | 
			
		||||
    isChatGLM,
 | 
			
		||||
    isSiliconFlow,
 | 
			
		||||
    isHuawei: boolean,
 | 
			
		||||
    apiKey,
 | 
			
		||||
    isEnabledAccessControl,
 | 
			
		||||
  } = getConfig();
 | 
			
		||||
@@ -391,8 +377,6 @@ export function getClientApi(provider: ServiceProvider): ClientApi {
 | 
			
		||||
      return new ClientApi(ModelProvider.ChatGLM);
 | 
			
		||||
    case ServiceProvider.SiliconFlow:
 | 
			
		||||
      return new ClientApi(ModelProvider.SiliconFlow);
 | 
			
		||||
    case ServiceProvider.Huawei:
 | 
			
		||||
      return new ClientApi(ModelProvider.Huawei);
 | 
			
		||||
    default:
 | 
			
		||||
      return new ClientApi(ModelProvider.GPT);
 | 
			
		||||
  }
 | 
			
		||||
 
 | 
			
		||||
@@ -7,10 +7,7 @@ import {
 | 
			
		||||
  ChatMessageTool,
 | 
			
		||||
  usePluginStore,
 | 
			
		||||
} from "@/app/store";
 | 
			
		||||
import {
 | 
			
		||||
  preProcessImageContentForAlibabaDashScope,
 | 
			
		||||
  streamWithThink,
 | 
			
		||||
} from "@/app/utils/chat";
 | 
			
		||||
import { streamWithThink } from "@/app/utils/chat";
 | 
			
		||||
import {
 | 
			
		||||
  ChatOptions,
 | 
			
		||||
  getHeaders,
 | 
			
		||||
@@ -18,14 +15,12 @@ import {
 | 
			
		||||
  LLMModel,
 | 
			
		||||
  SpeechOptions,
 | 
			
		||||
  MultimodalContent,
 | 
			
		||||
  MultimodalContentForAlibaba,
 | 
			
		||||
} from "../api";
 | 
			
		||||
import { getClientConfig } from "@/app/config/client";
 | 
			
		||||
import {
 | 
			
		||||
  getMessageTextContent,
 | 
			
		||||
  getMessageTextContentWithoutThinking,
 | 
			
		||||
  getTimeoutMSByModel,
 | 
			
		||||
  isVisionModel,
 | 
			
		||||
} from "@/app/utils";
 | 
			
		||||
import { fetch } from "@/app/utils/stream";
 | 
			
		||||
 | 
			
		||||
@@ -94,6 +89,14 @@ export class QwenApi implements LLMApi {
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  async chat(options: ChatOptions) {
 | 
			
		||||
    const messages = options.messages.map((v) => ({
 | 
			
		||||
      role: v.role,
 | 
			
		||||
      content:
 | 
			
		||||
        v.role === "assistant"
 | 
			
		||||
          ? getMessageTextContentWithoutThinking(v)
 | 
			
		||||
          : getMessageTextContent(v),
 | 
			
		||||
    }));
 | 
			
		||||
 | 
			
		||||
    const modelConfig = {
 | 
			
		||||
      ...useAppConfig.getState().modelConfig,
 | 
			
		||||
      ...useChatStore.getState().currentSession().mask.modelConfig,
 | 
			
		||||
@@ -102,21 +105,6 @@ export class QwenApi implements LLMApi {
 | 
			
		||||
      },
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    const visionModel = isVisionModel(options.config.model);
 | 
			
		||||
 | 
			
		||||
    const messages: ChatOptions["messages"] = [];
 | 
			
		||||
    for (const v of options.messages) {
 | 
			
		||||
      const content = (
 | 
			
		||||
        visionModel
 | 
			
		||||
          ? await preProcessImageContentForAlibabaDashScope(v.content)
 | 
			
		||||
          : v.role === "assistant"
 | 
			
		||||
          ? getMessageTextContentWithoutThinking(v)
 | 
			
		||||
          : getMessageTextContent(v)
 | 
			
		||||
      ) as any;
 | 
			
		||||
 | 
			
		||||
      messages.push({ role: v.role, content });
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    const shouldStream = !!options.config.stream;
 | 
			
		||||
    const requestPayload: RequestPayload = {
 | 
			
		||||
      model: modelConfig.model,
 | 
			
		||||
@@ -141,7 +129,7 @@ export class QwenApi implements LLMApi {
 | 
			
		||||
        "X-DashScope-SSE": shouldStream ? "enable" : "disable",
 | 
			
		||||
      };
 | 
			
		||||
 | 
			
		||||
      const chatPath = this.path(Alibaba.ChatPath(modelConfig.model));
 | 
			
		||||
      const chatPath = this.path(Alibaba.ChatPath);
 | 
			
		||||
      const chatPayload = {
 | 
			
		||||
        method: "POST",
 | 
			
		||||
        body: JSON.stringify(requestPayload),
 | 
			
		||||
@@ -174,7 +162,7 @@ export class QwenApi implements LLMApi {
 | 
			
		||||
            const json = JSON.parse(text);
 | 
			
		||||
            const choices = json.output.choices as Array<{
 | 
			
		||||
              message: {
 | 
			
		||||
                content: string | null | MultimodalContentForAlibaba[];
 | 
			
		||||
                content: string | null;
 | 
			
		||||
                tool_calls: ChatMessageTool[];
 | 
			
		||||
                reasoning_content: string | null;
 | 
			
		||||
              };
 | 
			
		||||
@@ -224,9 +212,7 @@ export class QwenApi implements LLMApi {
 | 
			
		||||
            } else if (content && content.length > 0) {
 | 
			
		||||
              return {
 | 
			
		||||
                isThinking: false,
 | 
			
		||||
                content: Array.isArray(content)
 | 
			
		||||
                  ? content.map((item) => item.text).join(",")
 | 
			
		||||
                  : content,
 | 
			
		||||
                content: content,
 | 
			
		||||
              };
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -75,25 +75,6 @@ export class DeepSeekApi implements LLMApi {
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // 检测并修复消息顺序,确保除system外的第一个消息是user
 | 
			
		||||
    const filteredMessages: ChatOptions["messages"] = [];
 | 
			
		||||
    let hasFoundFirstUser = false;
 | 
			
		||||
 | 
			
		||||
    for (const msg of messages) {
 | 
			
		||||
      if (msg.role === "system") {
 | 
			
		||||
        // Keep all system messages
 | 
			
		||||
        filteredMessages.push(msg);
 | 
			
		||||
      } else if (msg.role === "user") {
 | 
			
		||||
        // User message directly added
 | 
			
		||||
        filteredMessages.push(msg);
 | 
			
		||||
        hasFoundFirstUser = true;
 | 
			
		||||
      } else if (hasFoundFirstUser) {
 | 
			
		||||
        // After finding the first user message, all subsequent non-system messages are retained.
 | 
			
		||||
        filteredMessages.push(msg);
 | 
			
		||||
      }
 | 
			
		||||
      // If hasFoundFirstUser is false and it is not a system message, it will be skipped.
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    const modelConfig = {
 | 
			
		||||
      ...useAppConfig.getState().modelConfig,
 | 
			
		||||
      ...useChatStore.getState().currentSession().mask.modelConfig,
 | 
			
		||||
@@ -104,7 +85,7 @@ export class DeepSeekApi implements LLMApi {
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    const requestPayload: RequestPayload = {
 | 
			
		||||
      messages: filteredMessages,
 | 
			
		||||
      messages,
 | 
			
		||||
      stream: options.config.stream,
 | 
			
		||||
      model: modelConfig.model,
 | 
			
		||||
      temperature: modelConfig.temperature,
 | 
			
		||||
 
 | 
			
		||||
@@ -1,200 +0,0 @@
 | 
			
		||||
"use client";
 | 
			
		||||
 | 
			
		||||
import { ApiPath, HUAWEI_BASE_URL, Huawei } from "@/app/constant";
 | 
			
		||||
import {
 | 
			
		||||
  useAccessStore,
 | 
			
		||||
  useAppConfig,
 | 
			
		||||
  useChatStore,
 | 
			
		||||
  usePluginStore,
 | 
			
		||||
  ChatMessageTool,
 | 
			
		||||
} from "@/app/store";
 | 
			
		||||
import {
 | 
			
		||||
  ChatOptions,
 | 
			
		||||
  getHeaders,
 | 
			
		||||
  LLMApi,
 | 
			
		||||
  LLMModel,
 | 
			
		||||
  MultimodalContent,
 | 
			
		||||
  SpeechOptions,
 | 
			
		||||
} from "../api";
 | 
			
		||||
import { getClientConfig } from "@/app/config/client";
 | 
			
		||||
import { getTimeoutMSByModel } from "@/app/utils";
 | 
			
		||||
import { streamWithThink } from "@/app/utils/chat";
 | 
			
		||||
import { fetch } from "@/app/utils/stream";
 | 
			
		||||
 | 
			
		||||
interface RequestPayloadForHuawei {
 | 
			
		||||
  messages: {
 | 
			
		||||
    role: "system" | "user" | "assistant";
 | 
			
		||||
    content: string | MultimodalContent[];
 | 
			
		||||
  }[];
 | 
			
		||||
  stream?: boolean;
 | 
			
		||||
  model: string;
 | 
			
		||||
  temperature: number;
 | 
			
		||||
  presence_penalty: number;
 | 
			
		||||
  frequency_penalty: number;
 | 
			
		||||
  top_p: number;
 | 
			
		||||
  max_tokens?: number;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export class HuaweiApi implements LLMApi {
 | 
			
		||||
  path(path: string): string {
 | 
			
		||||
    const accessStore = useAccessStore.getState();
 | 
			
		||||
 | 
			
		||||
    let baseUrl = "";
 | 
			
		||||
 | 
			
		||||
    if (accessStore.useCustomConfig) {
 | 
			
		||||
      baseUrl = accessStore.huaweiUrl;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (baseUrl.length === 0) {
 | 
			
		||||
      const isApp = !!getClientConfig()?.isApp;
 | 
			
		||||
      baseUrl = isApp ? HUAWEI_BASE_URL : ApiPath.Huawei;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (baseUrl.endsWith("/")) {
 | 
			
		||||
      baseUrl = baseUrl.slice(0, baseUrl.length - 1);
 | 
			
		||||
    }
 | 
			
		||||
    if (!baseUrl.startsWith("http") && !baseUrl.startsWith(ApiPath.Huawei)) {
 | 
			
		||||
      baseUrl = "https://" + baseUrl;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    console.log("[Proxy Endpoint] ", baseUrl, path);
 | 
			
		||||
    return [baseUrl, path].join("/");
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  extractMessage(res: any) {
 | 
			
		||||
    return res.choices?.at(0)?.message?.content ?? "";
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  speech(options: SpeechOptions): Promise<ArrayBuffer> {
 | 
			
		||||
    throw new Error("Method not implemented.");
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  async chat(options: ChatOptions) {
 | 
			
		||||
    const messages = options.messages.map((v) => ({
 | 
			
		||||
      role: v.role,
 | 
			
		||||
      content: v.content,
 | 
			
		||||
    }));
 | 
			
		||||
 | 
			
		||||
    const modelConfig = {
 | 
			
		||||
      ...useAppConfig.getState().modelConfig,
 | 
			
		||||
      ...useChatStore.getState().currentSession().mask.modelConfig,
 | 
			
		||||
      ...{
 | 
			
		||||
        model: options.config.model,
 | 
			
		||||
      },
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    const requestPayload: RequestPayloadForHuawei = {
 | 
			
		||||
      messages,
 | 
			
		||||
      stream: options.config.stream,
 | 
			
		||||
      model: modelConfig.model,
 | 
			
		||||
      temperature: modelConfig.temperature,
 | 
			
		||||
      presence_penalty: modelConfig.presence_penalty,
 | 
			
		||||
      frequency_penalty: modelConfig.frequency_penalty,
 | 
			
		||||
      top_p: modelConfig.top_p,
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    const shouldStream = !!options.config.stream;
 | 
			
		||||
    const controller = new AbortController();
 | 
			
		||||
    options.onController?.(controller);
 | 
			
		||||
 | 
			
		||||
    try {
 | 
			
		||||
      const chatPath = this.path(Huawei.ChatPath);
 | 
			
		||||
      const chatPayload = {
 | 
			
		||||
        method: "POST",
 | 
			
		||||
        body: JSON.stringify(requestPayload),
 | 
			
		||||
        signal: controller.signal,
 | 
			
		||||
        headers: getHeaders(),
 | 
			
		||||
      };
 | 
			
		||||
 | 
			
		||||
      const requestTimeoutId = setTimeout(
 | 
			
		||||
        () => controller.abort(),
 | 
			
		||||
        getTimeoutMSByModel(options.config.model),
 | 
			
		||||
      );
 | 
			
		||||
 | 
			
		||||
      if (shouldStream) {
 | 
			
		||||
        const [tools, funcs] = usePluginStore
 | 
			
		||||
          .getState()
 | 
			
		||||
          .getAsTools(
 | 
			
		||||
            useChatStore.getState().currentSession().mask?.plugin || [],
 | 
			
		||||
          );
 | 
			
		||||
 | 
			
		||||
        return streamWithThink(
 | 
			
		||||
          chatPath,
 | 
			
		||||
          requestPayload,
 | 
			
		||||
          getHeaders(),
 | 
			
		||||
          tools as any[],
 | 
			
		||||
          funcs,
 | 
			
		||||
          controller,
 | 
			
		||||
          // parseSSE
 | 
			
		||||
          (text: string, runTools: ChatMessageTool[]) => {
 | 
			
		||||
            const json = JSON.parse(text);
 | 
			
		||||
            const choices = json.choices as Array<{
 | 
			
		||||
              delta: {
 | 
			
		||||
                content: string;
 | 
			
		||||
                tool_calls: ChatMessageTool[];
 | 
			
		||||
              };
 | 
			
		||||
            }>;
 | 
			
		||||
            const tool_calls = choices[0]?.delta?.tool_calls;
 | 
			
		||||
            if (tool_calls?.length > 0) {
 | 
			
		||||
              const index = tool_calls[0]?.index;
 | 
			
		||||
              const id = tool_calls[0]?.id;
 | 
			
		||||
              const args = tool_calls[0]?.function?.arguments;
 | 
			
		||||
              if (id) {
 | 
			
		||||
                runTools.push({
 | 
			
		||||
                  id,
 | 
			
		||||
                  type: tool_calls[0]?.type,
 | 
			
		||||
                  function: {
 | 
			
		||||
                    name: tool_calls[0]?.function?.name as string,
 | 
			
		||||
                    arguments: args,
 | 
			
		||||
                  },
 | 
			
		||||
                });
 | 
			
		||||
              } else {
 | 
			
		||||
                // @ts-ignore
 | 
			
		||||
                runTools[index]["function"]["arguments"] += args;
 | 
			
		||||
              }
 | 
			
		||||
            }
 | 
			
		||||
            return {
 | 
			
		||||
              isThinking: false,
 | 
			
		||||
              content: choices[0]?.delta?.content || "",
 | 
			
		||||
            };
 | 
			
		||||
          },
 | 
			
		||||
          // processToolMessage
 | 
			
		||||
          (
 | 
			
		||||
            payload: RequestPayloadForHuawei,
 | 
			
		||||
            toolCallMessage: any,
 | 
			
		||||
            toolCallResult: any[],
 | 
			
		||||
          ) => {
 | 
			
		||||
            payload?.messages?.splice(
 | 
			
		||||
              payload?.messages?.length,
 | 
			
		||||
              0,
 | 
			
		||||
              toolCallMessage,
 | 
			
		||||
              ...toolCallResult,
 | 
			
		||||
            );
 | 
			
		||||
          },
 | 
			
		||||
          options,
 | 
			
		||||
        );
 | 
			
		||||
      } else {
 | 
			
		||||
        const res = await fetch(chatPath, chatPayload);
 | 
			
		||||
        clearTimeout(requestTimeoutId);
 | 
			
		||||
 | 
			
		||||
        const resJson = await res.json();
 | 
			
		||||
        const message = this.extractMessage(resJson);
 | 
			
		||||
        options.onFinish(message, res);
 | 
			
		||||
      }
 | 
			
		||||
    } catch (e) {
 | 
			
		||||
      console.log("[Request] failed to make a chat request", e);
 | 
			
		||||
      options.onError?.(e as Error);
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  async usage() {
 | 
			
		||||
    return {
 | 
			
		||||
      used: 0,
 | 
			
		||||
      total: 0,
 | 
			
		||||
    };
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  async models(): Promise<LLMModel[]> {
 | 
			
		||||
    return [];
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
@@ -66,11 +66,11 @@ export function Avatar(props: { model?: ModelType; avatar?: string }) {
 | 
			
		||||
      LlmIcon = BotIconGemma;
 | 
			
		||||
    } else if (modelName.startsWith("claude")) {
 | 
			
		||||
      LlmIcon = BotIconClaude;
 | 
			
		||||
    } else if (modelName.includes("llama")) {
 | 
			
		||||
    } else if (modelName.toLowerCase().includes("llama")) {
 | 
			
		||||
      LlmIcon = BotIconMeta;
 | 
			
		||||
    } else if (modelName.startsWith("mixtral") || modelName.startsWith("codestral")) {
 | 
			
		||||
    } else if (modelName.startsWith("mixtral")) {
 | 
			
		||||
      LlmIcon = BotIconMistral;
 | 
			
		||||
    } else if (modelName.includes("deepseek")) {
 | 
			
		||||
    } else if (modelName.toLowerCase().includes("deepseek")) {
 | 
			
		||||
      LlmIcon = BotIconDeepseek;
 | 
			
		||||
    } else if (modelName.startsWith("moonshot")) {
 | 
			
		||||
      LlmIcon = BotIconMoonshot;
 | 
			
		||||
@@ -85,7 +85,7 @@ export function Avatar(props: { model?: ModelType; avatar?: string }) {
 | 
			
		||||
    } else if (modelName.startsWith("doubao") || modelName.startsWith("ep-")) {
 | 
			
		||||
      LlmIcon = BotIconDoubao;
 | 
			
		||||
    } else if (
 | 
			
		||||
      modelName.includes("glm") ||
 | 
			
		||||
      modelName.toLowerCase().includes("glm") ||
 | 
			
		||||
      modelName.startsWith("cogview-") ||
 | 
			
		||||
      modelName.startsWith("cogvideox-")
 | 
			
		||||
    ) {
 | 
			
		||||
 
 | 
			
		||||
@@ -75,7 +75,6 @@ import {
 | 
			
		||||
  ChatGLM,
 | 
			
		||||
  DeepSeek,
 | 
			
		||||
  SiliconFlow,
 | 
			
		||||
  Huawei,
 | 
			
		||||
} from "../constant";
 | 
			
		||||
import { Prompt, SearchService, usePromptStore } from "../store/prompt";
 | 
			
		||||
import { ErrorBoundary } from "./error";
 | 
			
		||||
@@ -1458,46 +1457,6 @@ export function Settings() {
 | 
			
		||||
      </ListItem>
 | 
			
		||||
    </>
 | 
			
		||||
  );
 | 
			
		||||
  const huaweiConfigComponent = accessStore.provider ===
 | 
			
		||||
    ServiceProvider.Huawei && (
 | 
			
		||||
    <>
 | 
			
		||||
      <ListItem
 | 
			
		||||
        title={Locale.Settings.Access.Huawei.Endpoint.Title}
 | 
			
		||||
        subTitle={
 | 
			
		||||
          Locale.Settings.Access.Huawei.Endpoint.SubTitle +
 | 
			
		||||
          Huawei.ExampleEndpoint
 | 
			
		||||
        }
 | 
			
		||||
      >
 | 
			
		||||
        <input
 | 
			
		||||
          aria-label={Locale.Settings.Access.Huawei.Endpoint.Title}
 | 
			
		||||
          type="text"
 | 
			
		||||
          value={accessStore.huaweiUrl}
 | 
			
		||||
          placeholder={Huawei.ExampleEndpoint}
 | 
			
		||||
          onChange={(e) =>
 | 
			
		||||
            accessStore.update(
 | 
			
		||||
              (access) => (access.huaweiUrl = e.currentTarget.value),
 | 
			
		||||
            )
 | 
			
		||||
          }
 | 
			
		||||
        ></input>
 | 
			
		||||
      </ListItem>
 | 
			
		||||
      <ListItem
 | 
			
		||||
        title={Locale.Settings.Access.Huawei.ApiKey.Title}
 | 
			
		||||
        subTitle={Locale.Settings.Access.Huawei.ApiKey.SubTitle}
 | 
			
		||||
      >
 | 
			
		||||
        <PasswordInput
 | 
			
		||||
          aria-label={Locale.Settings.Access.Huawei.ApiKey.Title}
 | 
			
		||||
          value={accessStore.huaweiApiKey}
 | 
			
		||||
          type="text"
 | 
			
		||||
          placeholder={Locale.Settings.Access.Huawei.ApiKey.Placeholder}
 | 
			
		||||
          onChange={(e) => {
 | 
			
		||||
            accessStore.update(
 | 
			
		||||
              (access) => (access.huaweiApiKey = e.currentTarget.value),
 | 
			
		||||
            );
 | 
			
		||||
          }}
 | 
			
		||||
        />
 | 
			
		||||
      </ListItem>
 | 
			
		||||
    </>
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <ErrorBoundary>
 | 
			
		||||
@@ -1863,7 +1822,6 @@ export function Settings() {
 | 
			
		||||
                  {XAIConfigComponent}
 | 
			
		||||
                  {chatglmConfigComponent}
 | 
			
		||||
                  {siliconflowConfigComponent}
 | 
			
		||||
                  {huaweiConfigComponent}
 | 
			
		||||
                </>
 | 
			
		||||
              )}
 | 
			
		||||
            </>
 | 
			
		||||
 
 | 
			
		||||
@@ -88,10 +88,6 @@ declare global {
 | 
			
		||||
      SILICONFLOW_URL?: string;
 | 
			
		||||
      SILICONFLOW_API_KEY?: string;
 | 
			
		||||
 | 
			
		||||
      //huaweionly
 | 
			
		||||
      HUAWEI_URL?: string;
 | 
			
		||||
      HUAWEI_API_KEY?: string;
 | 
			
		||||
 | 
			
		||||
      // custom template for preprocessing user input
 | 
			
		||||
      DEFAULT_INPUT_TEMPLATE?: string;
 | 
			
		||||
 | 
			
		||||
@@ -167,7 +163,6 @@ export const getServerSideConfig = () => {
 | 
			
		||||
  const isXAI = !!process.env.XAI_API_KEY;
 | 
			
		||||
  const isChatGLM = !!process.env.CHATGLM_API_KEY;
 | 
			
		||||
  const isSiliconFlow = !!process.env.SILICONFLOW_API_KEY;
 | 
			
		||||
  const isHuawei = !!process.env.HUAWEI_API_KEY;
 | 
			
		||||
  // const apiKeyEnvVar = process.env.OPENAI_API_KEY ?? "";
 | 
			
		||||
  // const apiKeys = apiKeyEnvVar.split(",").map((v) => v.trim());
 | 
			
		||||
  // const randomIndex = Math.floor(Math.random() * apiKeys.length);
 | 
			
		||||
@@ -238,10 +233,6 @@ export const getServerSideConfig = () => {
 | 
			
		||||
    xaiUrl: process.env.XAI_URL,
 | 
			
		||||
    xaiApiKey: getApiKey(process.env.XAI_API_KEY),
 | 
			
		||||
 | 
			
		||||
    isHuawei,
 | 
			
		||||
    huaweiUrl: process.env.HUAWEI_URL,
 | 
			
		||||
    huaweiApiKey: getApiKey(process.env.HUAWEI_API_KEY),
 | 
			
		||||
 | 
			
		||||
    isChatGLM,
 | 
			
		||||
    chatglmUrl: process.env.CHATGLM_URL,
 | 
			
		||||
    chatglmApiKey: getApiKey(process.env.CHATGLM_API_KEY),
 | 
			
		||||
 
 | 
			
		||||
@@ -36,9 +36,6 @@ export const CHATGLM_BASE_URL = "https://open.bigmodel.cn";
 | 
			
		||||
 | 
			
		||||
export const SILICONFLOW_BASE_URL = "https://api.siliconflow.cn";
 | 
			
		||||
 | 
			
		||||
export const HUAWEI_BASE_URL =
 | 
			
		||||
  "https://maas-cn-southwest-2.modelarts-maas.com/v1/infers";
 | 
			
		||||
 | 
			
		||||
export const CACHE_URL_PREFIX = "/api/cache";
 | 
			
		||||
export const UPLOAD_URL = `${CACHE_URL_PREFIX}/upload`;
 | 
			
		||||
 | 
			
		||||
@@ -75,7 +72,6 @@ export enum ApiPath {
 | 
			
		||||
  ChatGLM = "/api/chatglm",
 | 
			
		||||
  DeepSeek = "/api/deepseek",
 | 
			
		||||
  SiliconFlow = "/api/siliconflow",
 | 
			
		||||
  Huawei = "/api/huawei",
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export enum SlotID {
 | 
			
		||||
@@ -134,7 +130,6 @@ export enum ServiceProvider {
 | 
			
		||||
  ChatGLM = "ChatGLM",
 | 
			
		||||
  DeepSeek = "DeepSeek",
 | 
			
		||||
  SiliconFlow = "SiliconFlow",
 | 
			
		||||
  Huawei = "Huawei",
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Google API safety settings, see https://ai.google.dev/gemini-api/docs/safety-settings
 | 
			
		||||
@@ -161,7 +156,6 @@ export enum ModelProvider {
 | 
			
		||||
  ChatGLM = "ChatGLM",
 | 
			
		||||
  DeepSeek = "DeepSeek",
 | 
			
		||||
  SiliconFlow = "SiliconFlow",
 | 
			
		||||
  Huawei = "Huawei",
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export const Stability = {
 | 
			
		||||
@@ -220,11 +214,6 @@ export const Baidu = {
 | 
			
		||||
  },
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
export const Huawei = {
 | 
			
		||||
  ExampleEndpoint: HUAWEI_BASE_URL,
 | 
			
		||||
  ChatPath: "/v1/chat/completions",
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
export const ByteDance = {
 | 
			
		||||
  ExampleEndpoint: "https://ark.cn-beijing.volces.com/api/",
 | 
			
		||||
  ChatPath: "api/v3/chat/completions",
 | 
			
		||||
@@ -232,12 +221,7 @@ export const ByteDance = {
 | 
			
		||||
 | 
			
		||||
export const Alibaba = {
 | 
			
		||||
  ExampleEndpoint: ALIBABA_BASE_URL,
 | 
			
		||||
  ChatPath: (modelName: string) => {
 | 
			
		||||
    if (modelName.includes("vl") || modelName.includes("omni")) {
 | 
			
		||||
      return "v1/services/aigc/multimodal-generation/generation";
 | 
			
		||||
    }
 | 
			
		||||
    return `v1/services/aigc/text-generation/generation`;
 | 
			
		||||
  },
 | 
			
		||||
  ChatPath: "v1/services/aigc/text-generation/generation",
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
export const Tencent = {
 | 
			
		||||
@@ -428,14 +412,6 @@ export const KnowledgeCutOffDate: Record<string, string> = {
 | 
			
		||||
  "gpt-4-turbo": "2023-12",
 | 
			
		||||
  "gpt-4-turbo-2024-04-09": "2023-12",
 | 
			
		||||
  "gpt-4-turbo-preview": "2023-12",
 | 
			
		||||
  "gpt-4.1": "2024-06",
 | 
			
		||||
  "gpt-4.1-2025-04-14": "2024-06",
 | 
			
		||||
  "gpt-4.1-mini": "2024-06",
 | 
			
		||||
  "gpt-4.1-mini-2025-04-14": "2024-06",
 | 
			
		||||
  "gpt-4.1-nano": "2024-06",
 | 
			
		||||
  "gpt-4.1-nano-2025-04-14": "2024-06",
 | 
			
		||||
  "gpt-4.5-preview": "2023-10",
 | 
			
		||||
  "gpt-4.5-preview-2025-02-27": "2023-10",
 | 
			
		||||
  "gpt-4o": "2023-10",
 | 
			
		||||
  "gpt-4o-2024-05-13": "2023-10",
 | 
			
		||||
  "gpt-4o-2024-08-06": "2023-10",
 | 
			
		||||
@@ -477,7 +453,6 @@ export const DEFAULT_TTS_VOICES = [
 | 
			
		||||
export const VISION_MODEL_REGEXES = [
 | 
			
		||||
  /vision/,
 | 
			
		||||
  /gpt-4o/,
 | 
			
		||||
  /gpt-4\.1/,
 | 
			
		||||
  /claude-3/,
 | 
			
		||||
  /gemini-1\.5/,
 | 
			
		||||
  /gemini-exp/,
 | 
			
		||||
@@ -505,14 +480,6 @@ const openaiModels = [
 | 
			
		||||
  "gpt-4-32k-0613",
 | 
			
		||||
  "gpt-4-turbo",
 | 
			
		||||
  "gpt-4-turbo-preview",
 | 
			
		||||
  "gpt-4.1",
 | 
			
		||||
  "gpt-4.1-2025-04-14",
 | 
			
		||||
  "gpt-4.1-mini",
 | 
			
		||||
  "gpt-4.1-mini-2025-04-14",
 | 
			
		||||
  "gpt-4.1-nano",
 | 
			
		||||
  "gpt-4.1-nano-2025-04-14",
 | 
			
		||||
  "gpt-4.5-preview",
 | 
			
		||||
  "gpt-4.5-preview-2025-02-27",
 | 
			
		||||
  "gpt-4o",
 | 
			
		||||
  "gpt-4o-2024-05-13",
 | 
			
		||||
  "gpt-4o-2024-08-06",
 | 
			
		||||
@@ -568,8 +535,6 @@ const anthropicModels = [
 | 
			
		||||
  "claude-3-5-sonnet-20240620",
 | 
			
		||||
  "claude-3-5-sonnet-20241022",
 | 
			
		||||
  "claude-3-5-sonnet-latest",
 | 
			
		||||
  "claude-3-7-sonnet-20250219",
 | 
			
		||||
  "claude-3-7-sonnet-latest",
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
const baiduModels = [
 | 
			
		||||
@@ -593,11 +558,6 @@ const bytedanceModels = [
 | 
			
		||||
  "Doubao-pro-4k",
 | 
			
		||||
  "Doubao-pro-32k",
 | 
			
		||||
  "Doubao-pro-128k",
 | 
			
		||||
  "deepseek-r1-250120",
 | 
			
		||||
  "deepseek-v3-241226",
 | 
			
		||||
  "deepseek-v3-250324",
 | 
			
		||||
  "deepseek-r1-distill-qwen-7b-250120",
 | 
			
		||||
  "deepseek-r1-distill-qwen-32b-250120",
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
const alibabaModes = [
 | 
			
		||||
@@ -608,9 +568,6 @@ const alibabaModes = [
 | 
			
		||||
  "qwen-max-0403",
 | 
			
		||||
  "qwen-max-0107",
 | 
			
		||||
  "qwen-max-longcontext",
 | 
			
		||||
  "qwen-omni-turbo",
 | 
			
		||||
  "qwen-vl-plus",
 | 
			
		||||
  "qwen-vl-max",
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
const tencentModels = [
 | 
			
		||||
@@ -683,11 +640,6 @@ const siliconflowModels = [
 | 
			
		||||
  "Pro/deepseek-ai/DeepSeek-V3",
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
const huaweiModels = [
 | 
			
		||||
  "DeepSeek-R1",
 | 
			
		||||
  "DeepSeek-V3",
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
let seq = 1000; // 内置的模型序号生成器从1000开始
 | 
			
		||||
export const DEFAULT_MODELS = [
 | 
			
		||||
  ...openaiModels.map((name) => ({
 | 
			
		||||
@@ -844,17 +796,6 @@ export const DEFAULT_MODELS = [
 | 
			
		||||
      sorted: 14,
 | 
			
		||||
    },
 | 
			
		||||
  })),
 | 
			
		||||
  ...huaweiModels.map((name) => ({
 | 
			
		||||
    name,
 | 
			
		||||
    available: true,
 | 
			
		||||
    sorted: seq++,
 | 
			
		||||
    provider: {
 | 
			
		||||
      id: "huawei",
 | 
			
		||||
      providerName: "Huawei",
 | 
			
		||||
      providerType: "Huawei",
 | 
			
		||||
      sorted: 15,
 | 
			
		||||
    },
 | 
			
		||||
  })),
 | 
			
		||||
] as const;
 | 
			
		||||
 | 
			
		||||
export const CHAT_PAGE_SIZE = 15;
 | 
			
		||||
 
 | 
			
		||||
@@ -420,22 +420,6 @@ const ar: PartialLocaleType = {
 | 
			
		||||
        Title: "اسم النموذج المخصص",
 | 
			
		||||
        SubTitle: "أضف خيارات نموذج مخصص، مفصولة بفواصل إنجليزية",
 | 
			
		||||
      },
 | 
			
		||||
      Huawei: {
 | 
			
		||||
        ApiKey: {
 | 
			
		||||
          Title: "مفتاح API",
 | 
			
		||||
          SubTitle: "استخدم مفتاح API الخاص بـ HUAWEI",
 | 
			
		||||
          Placeholder: "مفتاح HUAWEI",
 | 
			
		||||
        },
 | 
			
		||||
        SecretKey: {
 | 
			
		||||
          Title: "المفتاح السري",
 | 
			
		||||
          SubTitle: "استخدم مفتاح HUAWEI السري الخاص بك",
 | 
			
		||||
          Placeholder: "المفتاح السري HUAWEI",
 | 
			
		||||
        },
 | 
			
		||||
        Endpoint: {
 | 
			
		||||
          Title: "عنوان الواجهة",
 | 
			
		||||
          SubTitle: "لا يدعم التكوين المخصص .env",
 | 
			
		||||
        },
 | 
			
		||||
      },
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    Model: "النموذج",
 | 
			
		||||
 
 | 
			
		||||
@@ -428,22 +428,6 @@ const bn: PartialLocaleType = {
 | 
			
		||||
        SubTitle:
 | 
			
		||||
          "স্বনির্ধারিত মডেল বিকল্পগুলি যুক্ত করুন, ইংরেজি কমা দ্বারা আলাদা করুন",
 | 
			
		||||
      },
 | 
			
		||||
      Huawei: {
 | 
			
		||||
        ApiKey: {
 | 
			
		||||
          Title: "এপিআই কী",
 | 
			
		||||
          SubTitle: "আপনার HUAWEI এপিআই কী ব্যবহার করুন",
 | 
			
		||||
          Placeholder: "HUAWEI কী",
 | 
			
		||||
        },
 | 
			
		||||
        SecretKey: {
 | 
			
		||||
          Title: "গোপন কী",
 | 
			
		||||
          SubTitle: "আপনার HUAWEI গোপন কী ব্যবহার করুন",
 | 
			
		||||
          Placeholder: "HUAWEI গোপন কী",
 | 
			
		||||
        },
 | 
			
		||||
        Endpoint: {
 | 
			
		||||
          Title: "এন্ডপয়েন্ট ঠিকানা",
 | 
			
		||||
          SubTitle: "কাস্টম কনফিগারেশনের জন্য .env-এ যান",
 | 
			
		||||
        },
 | 
			
		||||
      },
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    Model: "মডেল (model)",
 | 
			
		||||
 
 | 
			
		||||
@@ -538,22 +538,6 @@ const cn = {
 | 
			
		||||
        Title: "自定义模型名",
 | 
			
		||||
        SubTitle: "增加自定义模型可选项,使用英文逗号隔开",
 | 
			
		||||
      },
 | 
			
		||||
      Huawei: {
 | 
			
		||||
        ApiKey: {
 | 
			
		||||
          Title: "API Key",
 | 
			
		||||
          SubTitle: "使用自定义华为API Key",
 | 
			
		||||
          Placeholder: "HUAWEI Key",
 | 
			
		||||
        },
 | 
			
		||||
        SecretKey: {
 | 
			
		||||
          Title: "Secret Key",
 | 
			
		||||
          SubTitle: "使用自定义HUAWEI Secret Key",
 | 
			
		||||
          Placeholder: "HUAWEI Secret Key",
 | 
			
		||||
        },
 | 
			
		||||
        Endpoint: {
 | 
			
		||||
          Title: "接口地址",
 | 
			
		||||
          SubTitle: "不支持自定义前往.env配置",
 | 
			
		||||
        },
 | 
			
		||||
      },
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    Model: "模型 (model)",
 | 
			
		||||
 
 | 
			
		||||
@@ -427,22 +427,6 @@ const cs: PartialLocaleType = {
 | 
			
		||||
        Title: "Vlastní názvy modelů",
 | 
			
		||||
        SubTitle: "Přidejte možnosti vlastních modelů, oddělené čárkami",
 | 
			
		||||
      },
 | 
			
		||||
      Huawei: {
 | 
			
		||||
        ApiKey: {
 | 
			
		||||
          Title: "API Klíč",
 | 
			
		||||
          SubTitle: "Použijte svůj HUAWEI API klíč",
 | 
			
		||||
          Placeholder: "HUAWEI Klíč",
 | 
			
		||||
        },
 | 
			
		||||
        SecretKey: {
 | 
			
		||||
          Title: "Tajný Klíč",
 | 
			
		||||
          SubTitle: "Použijte svůj HUAWEI Tajný klíč",
 | 
			
		||||
          Placeholder: "HUAWEI Tajný Klíč",
 | 
			
		||||
        },
 | 
			
		||||
        Endpoint: {
 | 
			
		||||
          Title: "Adresa rozhraní",
 | 
			
		||||
          SubTitle: "Nepodporuje vlastní konfiguraci .env",
 | 
			
		||||
        },
 | 
			
		||||
      },
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    Model: "Model (model)",
 | 
			
		||||
 
 | 
			
		||||
@@ -498,22 +498,6 @@ const da: PartialLocaleType = {
 | 
			
		||||
        Title: "Egne modelnavne",
 | 
			
		||||
        SubTitle: "Skriv komma-adskilte navne",
 | 
			
		||||
      },
 | 
			
		||||
      Huawei: {
 | 
			
		||||
        ApiKey: {
 | 
			
		||||
          Title: "API-nøgle",
 | 
			
		||||
          SubTitle: "Brug din egen HUAWEI API-nøgle",
 | 
			
		||||
          Placeholder: "HUAWEI-nøgle",
 | 
			
		||||
        },
 | 
			
		||||
        SecretKey: {
 | 
			
		||||
          Title: "Hemmelig nøgle",
 | 
			
		||||
          SubTitle: "Brug din egen HUAWEI hemmelige nøgle",
 | 
			
		||||
          Placeholder: "HUAWEI hemmelig nøgle",
 | 
			
		||||
        },
 | 
			
		||||
        Endpoint: {
 | 
			
		||||
          Title: "Grænsefladeadresse",
 | 
			
		||||
          SubTitle: "Understøtter ikke tilpasset .env-konfiguration",
 | 
			
		||||
        },
 | 
			
		||||
      },
 | 
			
		||||
      Google: {
 | 
			
		||||
        ApiKey: {
 | 
			
		||||
          Title: "Google-nøgle",
 | 
			
		||||
 
 | 
			
		||||
@@ -439,22 +439,6 @@ const de: PartialLocaleType = {
 | 
			
		||||
        SubTitle:
 | 
			
		||||
          "Fügen Sie benutzerdefinierte Modelloptionen hinzu, getrennt durch Kommas",
 | 
			
		||||
      },
 | 
			
		||||
      Huawei: {
 | 
			
		||||
        ApiKey: {
 | 
			
		||||
          Title: "API-Schlüssel",
 | 
			
		||||
          SubTitle: "Verwenden Sie Ihren eigenen HUAWEI API-Schlüssel",
 | 
			
		||||
          Placeholder: "HUAWEI-Schlüssel",
 | 
			
		||||
        },
 | 
			
		||||
        SecretKey: {
 | 
			
		||||
          Title: "Geheimer Schlüssel",
 | 
			
		||||
          SubTitle: "Verwenden Sie Ihren eigenen HUAWEI geheimen Schlüssel",
 | 
			
		||||
          Placeholder: "HUAWEI geheimer Schlüssel",
 | 
			
		||||
        },
 | 
			
		||||
        Endpoint: {
 | 
			
		||||
          Title: "Schnittstellenadresse",
 | 
			
		||||
          SubTitle: "Unterstützt keine benutzerdefinierte .env-Konfiguration",
 | 
			
		||||
        },
 | 
			
		||||
      },
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    Model: "Modell",
 | 
			
		||||
 
 | 
			
		||||
@@ -522,22 +522,6 @@ const en: LocaleType = {
 | 
			
		||||
        Title: "Custom Models",
 | 
			
		||||
        SubTitle: "Custom model options, seperated by comma",
 | 
			
		||||
      },
 | 
			
		||||
      Huawei: {
 | 
			
		||||
        ApiKey: {
 | 
			
		||||
          Title: "API Key",
 | 
			
		||||
          SubTitle: "Use your own HUAWEI API key",
 | 
			
		||||
          Placeholder: "HUAWEI Key",
 | 
			
		||||
        },
 | 
			
		||||
        SecretKey: {
 | 
			
		||||
          Title: "Secret Key",
 | 
			
		||||
          SubTitle: "Use your own HUAWEI Secret key",
 | 
			
		||||
          Placeholder: "HUAWEI Secret Key",
 | 
			
		||||
        },
 | 
			
		||||
        Endpoint: {
 | 
			
		||||
          Title: "Endpoint Address",
 | 
			
		||||
          SubTitle: "Does not support custom .env configuration",
 | 
			
		||||
        },
 | 
			
		||||
      },
 | 
			
		||||
      Google: {
 | 
			
		||||
        ApiKey: {
 | 
			
		||||
          Title: "API Key",
 | 
			
		||||
 
 | 
			
		||||
@@ -441,22 +441,6 @@ const es: PartialLocaleType = {
 | 
			
		||||
        SubTitle:
 | 
			
		||||
          "Agrega opciones de modelos personalizados, separados por comas",
 | 
			
		||||
      },
 | 
			
		||||
      Huawei: {
 | 
			
		||||
        ApiKey: {
 | 
			
		||||
          Title: "Clave API",
 | 
			
		||||
          SubTitle: "Utiliza tu propia clave API de HUAWEI",
 | 
			
		||||
          Placeholder: "Clave HUAWEI",
 | 
			
		||||
        },
 | 
			
		||||
        SecretKey: {
 | 
			
		||||
          Title: "Clave Secreta",
 | 
			
		||||
          SubTitle: "Utiliza tu propia clave secreta de HUAWEI",
 | 
			
		||||
          Placeholder: "Clave secreta HUAWEI",
 | 
			
		||||
        },
 | 
			
		||||
        Endpoint: {
 | 
			
		||||
          Title: "Dirección del Endpoint",
 | 
			
		||||
          SubTitle: "No admite configuración personalizada .env",
 | 
			
		||||
        },
 | 
			
		||||
      },
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    Model: "Modelo (model)",
 | 
			
		||||
 
 | 
			
		||||
@@ -440,23 +440,6 @@ const fr: PartialLocaleType = {
 | 
			
		||||
        SubTitle:
 | 
			
		||||
          "Ajouter des options de modèles personnalisés, séparées par des virgules",
 | 
			
		||||
      },
 | 
			
		||||
      Huawei: {
 | 
			
		||||
        ApiKey: {
 | 
			
		||||
          Title: "Clé API",
 | 
			
		||||
          SubTitle: "Utilisez votre propre clé API HUAWEI",
 | 
			
		||||
          Placeholder: "Clé HUAWEI",
 | 
			
		||||
        },
 | 
			
		||||
        SecretKey: {
 | 
			
		||||
          Title: "Clé secrète",
 | 
			
		||||
          SubTitle: "Utilisez votre propre clé secrète HUAWEI",
 | 
			
		||||
          Placeholder: "Clé secrète HUAWEI",
 | 
			
		||||
        },
 | 
			
		||||
        Endpoint: {
 | 
			
		||||
          Title: "Adresse de l'endpoint",
 | 
			
		||||
          SubTitle:
 | 
			
		||||
            "Ne prend pas en charge la configuration personnalisée .env",
 | 
			
		||||
        },
 | 
			
		||||
      },
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    Model: "Modèle",
 | 
			
		||||
 
 | 
			
		||||
@@ -428,22 +428,6 @@ const id: PartialLocaleType = {
 | 
			
		||||
        Title: "Nama Model Kustom",
 | 
			
		||||
        SubTitle: "Tambahkan opsi model kustom, pisahkan dengan koma",
 | 
			
		||||
      },
 | 
			
		||||
      Huawei: {
 | 
			
		||||
        ApiKey: {
 | 
			
		||||
          Title: "Kunci API",
 | 
			
		||||
          SubTitle: "Gunakan kunci API HUAWEI Anda sendiri",
 | 
			
		||||
          Placeholder: "Kunci HUAWEI",
 | 
			
		||||
        },
 | 
			
		||||
        SecretKey: {
 | 
			
		||||
          Title: "Kunci Rahasia",
 | 
			
		||||
          SubTitle: "Gunakan kunci rahasia HUAWEI Anda sendiri",
 | 
			
		||||
          Placeholder: "Kunci Rahasia HUAWEI",
 | 
			
		||||
        },
 | 
			
		||||
        Endpoint: {
 | 
			
		||||
          Title: "Alamat Endpoint",
 | 
			
		||||
          SubTitle: "Tidak mendukung konfigurasi .env kustom",
 | 
			
		||||
        },
 | 
			
		||||
      },
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    Model: "Model",
 | 
			
		||||
 
 | 
			
		||||
@@ -441,22 +441,6 @@ const it: PartialLocaleType = {
 | 
			
		||||
        SubTitle:
 | 
			
		||||
          "Aggiungi opzioni di modelli personalizzati, separati da virgole",
 | 
			
		||||
      },
 | 
			
		||||
      Huawei: {
 | 
			
		||||
        ApiKey: {
 | 
			
		||||
          Title: "Chiave API",
 | 
			
		||||
          SubTitle: "Usa la tua chiave API HUAWEI",
 | 
			
		||||
          Placeholder: "Chiave HUAWEI",
 | 
			
		||||
        },
 | 
			
		||||
        SecretKey: {
 | 
			
		||||
          Title: "Chiave Segreta",
 | 
			
		||||
          SubTitle: "Usa la tua chiave segreta HUAWEI",
 | 
			
		||||
          Placeholder: "Chiave segreta HUAWEI",
 | 
			
		||||
        },
 | 
			
		||||
        Endpoint: {
 | 
			
		||||
          Title: "Indirizzo dell'Endpoint",
 | 
			
		||||
          SubTitle: "Non supporta configurazioni personalizzate .env",
 | 
			
		||||
        },
 | 
			
		||||
      },
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    Model: "Modello (model)",
 | 
			
		||||
 
 | 
			
		||||
@@ -424,22 +424,6 @@ const jp: PartialLocaleType = {
 | 
			
		||||
        Title: "カスタムモデル名",
 | 
			
		||||
        SubTitle: "カスタムモデルの選択肢を追加、英語のカンマで区切る",
 | 
			
		||||
      },
 | 
			
		||||
      Huawei: {
 | 
			
		||||
        ApiKey: {
 | 
			
		||||
          Title: "APIキー",
 | 
			
		||||
          SubTitle: "自分のHUAWEI APIキーを使用してください",
 | 
			
		||||
          Placeholder: "HUAWEIキー",
 | 
			
		||||
        },
 | 
			
		||||
        SecretKey: {
 | 
			
		||||
          Title: "秘密キー",
 | 
			
		||||
          SubTitle: "自分のHUAWEI秘密キーを使用してください",
 | 
			
		||||
          Placeholder: "HUAWEI秘密キー",
 | 
			
		||||
        },
 | 
			
		||||
        Endpoint: {
 | 
			
		||||
          Title: "エンドポイントアドレス",
 | 
			
		||||
          SubTitle: "カスタム.env構成はサポートされていません",
 | 
			
		||||
        },
 | 
			
		||||
      },
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    Model: "モデル (model)",
 | 
			
		||||
 
 | 
			
		||||
@@ -421,22 +421,6 @@ const ko: PartialLocaleType = {
 | 
			
		||||
        Title: "커스텀 모델 이름",
 | 
			
		||||
        SubTitle: "커스텀 모델 옵션 추가, 영어 쉼표로 구분",
 | 
			
		||||
      },
 | 
			
		||||
      Huawei: {
 | 
			
		||||
        ApiKey: {
 | 
			
		||||
          Title: "API 키",
 | 
			
		||||
          SubTitle: "자신의 HUAWEI API 키를 사용하세요",
 | 
			
		||||
          Placeholder: "HUAWEI 키",
 | 
			
		||||
        },
 | 
			
		||||
        SecretKey: {
 | 
			
		||||
          Title: "비밀 키",
 | 
			
		||||
          SubTitle: "자신의 HUAWEI 비밀 키를 사용하세요",
 | 
			
		||||
          Placeholder: "HUAWEI 비밀 키",
 | 
			
		||||
        },
 | 
			
		||||
        Endpoint: {
 | 
			
		||||
          Title: "엔드포인트 주소",
 | 
			
		||||
          SubTitle: "사용자 정의 .env 구성을 지원하지 않습니다",
 | 
			
		||||
        },
 | 
			
		||||
      },
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    Model: "모델 (model)",
 | 
			
		||||
 
 | 
			
		||||
@@ -433,22 +433,6 @@ const no: PartialLocaleType = {
 | 
			
		||||
        Title: "Egendefinert modellnavn",
 | 
			
		||||
        SubTitle: "Legg til egendefinerte modellalternativer, skill med komma",
 | 
			
		||||
      },
 | 
			
		||||
      Huawei: {
 | 
			
		||||
        ApiKey: {
 | 
			
		||||
          Title: "API-nøkkel",
 | 
			
		||||
          SubTitle: "Bruk din egen HUAWEI API-nøkkel",
 | 
			
		||||
          Placeholder: "HUAWEI-nøkkel",
 | 
			
		||||
        },
 | 
			
		||||
        SecretKey: {
 | 
			
		||||
          Title: "Hemmelig nøkkel",
 | 
			
		||||
          SubTitle: "Bruk din egen HUAWEI hemmelige nøkkel",
 | 
			
		||||
          Placeholder: "HUAWEI hemmelig nøkkel",
 | 
			
		||||
        },
 | 
			
		||||
        Endpoint: {
 | 
			
		||||
          Title: "Grensesnittadresse",
 | 
			
		||||
          SubTitle: "Støtter ikke tilpasset .env-konfigurasjon",
 | 
			
		||||
        },
 | 
			
		||||
      },
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    Model: "Modell",
 | 
			
		||||
 
 | 
			
		||||
@@ -363,22 +363,6 @@ const pt: PartialLocaleType = {
 | 
			
		||||
        Title: "Modelos Personalizados",
 | 
			
		||||
        SubTitle: "Opções de modelo personalizado, separados por vírgula",
 | 
			
		||||
      },
 | 
			
		||||
      Huawei: {
 | 
			
		||||
        ApiKey: {
 | 
			
		||||
          Title: "Chave API",
 | 
			
		||||
          SubTitle: "Use sua própria chave API HUAWEI",
 | 
			
		||||
          Placeholder: "Chave HUAWEI",
 | 
			
		||||
        },
 | 
			
		||||
        SecretKey: {
 | 
			
		||||
          Title: "Chave Secreta",
 | 
			
		||||
          SubTitle: "Use sua própria chave secreta HUAWEI",
 | 
			
		||||
          Placeholder: "Chave Secreta HUAWEI",
 | 
			
		||||
        },
 | 
			
		||||
        Endpoint: {
 | 
			
		||||
          Title: "Endereço do Endpoint",
 | 
			
		||||
          SubTitle: "Não suporta configuração personalizada .env",
 | 
			
		||||
        },
 | 
			
		||||
      },
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    Model: "Modelo",
 | 
			
		||||
 
 | 
			
		||||
@@ -431,22 +431,6 @@ const ru: PartialLocaleType = {
 | 
			
		||||
        SubTitle:
 | 
			
		||||
          "Добавьте варианты пользовательских моделей, разделяя запятыми",
 | 
			
		||||
      },
 | 
			
		||||
      Huawei: {
 | 
			
		||||
        ApiKey: {
 | 
			
		||||
          Title: "API Ключ",
 | 
			
		||||
          SubTitle: "Используйте свой HUAWEI API ключ",
 | 
			
		||||
          Placeholder: "HUAWEI Ключ",
 | 
			
		||||
        },
 | 
			
		||||
        SecretKey: {
 | 
			
		||||
          Title: "Секретный Ключ",
 | 
			
		||||
          SubTitle: "Используйте свой HUAWEI Секретный ключ",
 | 
			
		||||
          Placeholder: "HUAWEI Секретный Ключ",
 | 
			
		||||
        },
 | 
			
		||||
        Endpoint: {
 | 
			
		||||
          Title: "Адрес интерфейса",
 | 
			
		||||
          SubTitle: "Не поддерживает собственную конфигурацию .env",
 | 
			
		||||
        },
 | 
			
		||||
      },
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    Model: "Модель",
 | 
			
		||||
 
 | 
			
		||||
@@ -363,22 +363,6 @@ const sk: PartialLocaleType = {
 | 
			
		||||
        Title: "Vlastné modely",
 | 
			
		||||
        SubTitle: "Možnosti vlastného modelu, oddelené čiarkou",
 | 
			
		||||
      },
 | 
			
		||||
      Huawei: {
 | 
			
		||||
        ApiKey: {
 | 
			
		||||
          Title: "API kľúč",
 | 
			
		||||
          SubTitle: "Použite svoj vlastný HUAWEI API kľúč",
 | 
			
		||||
          Placeholder: "HUAWEI kľúč",
 | 
			
		||||
        },
 | 
			
		||||
        SecretKey: {
 | 
			
		||||
          Title: "Tajný kľúč",
 | 
			
		||||
          SubTitle: "Použite svoj vlastný HUAWEI tajný kľúč",
 | 
			
		||||
          Placeholder: "HUAWEI tajný kľúč",
 | 
			
		||||
        },
 | 
			
		||||
        Endpoint: {
 | 
			
		||||
          Title: "Adresa rozhrania",
 | 
			
		||||
          SubTitle: "Nepodporuje vlastnú konfiguráciu .env",
 | 
			
		||||
        },
 | 
			
		||||
      },
 | 
			
		||||
      Google: {
 | 
			
		||||
        ApiKey: {
 | 
			
		||||
          Title: "API kľúč",
 | 
			
		||||
 
 | 
			
		||||
@@ -431,22 +431,6 @@ const tr: PartialLocaleType = {
 | 
			
		||||
        SubTitle:
 | 
			
		||||
          "Özelleştirilmiş model seçenekleri ekleyin, İngilizce virgül ile ayırın",
 | 
			
		||||
      },
 | 
			
		||||
      Huawei: {
 | 
			
		||||
        ApiKey: {
 | 
			
		||||
          Title: "API Anahtarı",
 | 
			
		||||
          SubTitle: "Kendi Huawei API Anahtarınızı kullanın",
 | 
			
		||||
          Placeholder: "HUAWEI Anahtarı",
 | 
			
		||||
        },
 | 
			
		||||
        SecretKey: {
 | 
			
		||||
          Title: "Gizli Anahtar",
 | 
			
		||||
          SubTitle: "Kendi HUAWEI Gizli Anahtarınızı kullanın",
 | 
			
		||||
          Placeholder: "HUAWEI Gizli Anahtarı",
 | 
			
		||||
        },
 | 
			
		||||
        Endpoint: {
 | 
			
		||||
          Title: "Arayüz Adresi",
 | 
			
		||||
          SubTitle: "Özelleştirilmiş .env yapılandırmasına desteklenmez",
 | 
			
		||||
        },
 | 
			
		||||
      },
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    Model: "Model (model)",
 | 
			
		||||
 
 | 
			
		||||
@@ -386,22 +386,6 @@ const tw = {
 | 
			
		||||
        Title: "自訂模型名稱",
 | 
			
		||||
        SubTitle: "增加自訂模型可選擇項目,使用英文逗號隔開",
 | 
			
		||||
      },
 | 
			
		||||
      Huawei: {
 | 
			
		||||
        ApiKey: {
 | 
			
		||||
          Title: "API金鑰",
 | 
			
		||||
          SubTitle: "使用您的HUAWEI API金鑰",
 | 
			
		||||
          Placeholder: "HUAWEI金鑰",
 | 
			
		||||
        },
 | 
			
		||||
        SecretKey: {
 | 
			
		||||
          Title: "密鑰",
 | 
			
		||||
          SubTitle: "使用您的HUAWEI密鑰",
 | 
			
		||||
          Placeholder: "HUAWEI密鑰",
 | 
			
		||||
        },
 | 
			
		||||
        Endpoint: {
 | 
			
		||||
          Title: "端點地址",
 | 
			
		||||
          SubTitle: "不支援自訂的.env配置",
 | 
			
		||||
        },
 | 
			
		||||
      },
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    Model: "模型 (model)",
 | 
			
		||||
 
 | 
			
		||||
@@ -427,22 +427,6 @@ const vi: PartialLocaleType = {
 | 
			
		||||
        SubTitle:
 | 
			
		||||
          "Thêm tùy chọn mô hình tùy chỉnh, sử dụng dấu phẩy để phân cách",
 | 
			
		||||
      },
 | 
			
		||||
      Huawei: {
 | 
			
		||||
        ApiKey: {
 | 
			
		||||
          Title: "Khóa API",
 | 
			
		||||
          SubTitle: "Sử dụng khóa API HUAWEI của bạn",
 | 
			
		||||
          Placeholder: "Khóa HUAWEI",
 | 
			
		||||
        },
 | 
			
		||||
        SecretKey: {
 | 
			
		||||
          Title: "Khóa bí mật",
 | 
			
		||||
          SubTitle: "Sử dụng khóa bí mật HUAWEI của bạn",
 | 
			
		||||
          Placeholder: "Khóa bí mật HUAWEI",
 | 
			
		||||
        },
 | 
			
		||||
        Endpoint: {
 | 
			
		||||
          Title: "Địa chỉ điểm cuối",
 | 
			
		||||
          SubTitle: "Không hỗ trợ cấu hình .env tùy chỉnh",
 | 
			
		||||
        },
 | 
			
		||||
      },
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    Model: "Mô hình (model)",
 | 
			
		||||
 
 | 
			
		||||
@@ -17,7 +17,6 @@ import {
 | 
			
		||||
  XAI_BASE_URL,
 | 
			
		||||
  CHATGLM_BASE_URL,
 | 
			
		||||
  SILICONFLOW_BASE_URL,
 | 
			
		||||
  HUAWEI_BASE_URL,
 | 
			
		||||
} from "../constant";
 | 
			
		||||
import { getHeaders } from "../client/api";
 | 
			
		||||
import { getClientConfig } from "../config/client";
 | 
			
		||||
@@ -56,8 +55,6 @@ const DEFAULT_XAI_URL = isApp ? XAI_BASE_URL : ApiPath.XAI;
 | 
			
		||||
 | 
			
		||||
const DEFAULT_CHATGLM_URL = isApp ? CHATGLM_BASE_URL : ApiPath.ChatGLM;
 | 
			
		||||
 | 
			
		||||
const DEFAULT_HUAWEI_URL = isApp ? HUAWEI_BASE_URL : ApiPath.Huawei;
 | 
			
		||||
 | 
			
		||||
const DEFAULT_SILICONFLOW_URL = isApp
 | 
			
		||||
  ? SILICONFLOW_BASE_URL
 | 
			
		||||
  : ApiPath.SiliconFlow;
 | 
			
		||||
@@ -134,9 +131,6 @@ const DEFAULT_ACCESS_STATE = {
 | 
			
		||||
  // siliconflow
 | 
			
		||||
  siliconflowUrl: DEFAULT_SILICONFLOW_URL,
 | 
			
		||||
  siliconflowApiKey: "",
 | 
			
		||||
  // huawei
 | 
			
		||||
  huaweiUrl: DEFAULT_HUAWEI_URL,
 | 
			
		||||
  huaweiApiKey: "",
 | 
			
		||||
 | 
			
		||||
  // server config
 | 
			
		||||
  needCode: true,
 | 
			
		||||
@@ -225,9 +219,6 @@ export const useAccessStore = createPersistStore(
 | 
			
		||||
      return ensure(get(), ["siliconflowApiKey"]);
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    isValidHuawei() {
 | 
			
		||||
      return ensure(get(), ["huaweiApiKey"]);
 | 
			
		||||
    },
 | 
			
		||||
    isAuthorized() {
 | 
			
		||||
      this.fetch();
 | 
			
		||||
 | 
			
		||||
@@ -247,7 +238,6 @@ export const useAccessStore = createPersistStore(
 | 
			
		||||
        this.isValidXAI() ||
 | 
			
		||||
        this.isValidChatGLM() ||
 | 
			
		||||
        this.isValidSiliconFlow() ||
 | 
			
		||||
        this.isValidHuawei() ||
 | 
			
		||||
        !this.enabledAccessControl() ||
 | 
			
		||||
        (this.enabledAccessControl() && ensure(get(), ["accessCode"]))
 | 
			
		||||
      );
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@ import {
 | 
			
		||||
  UPLOAD_URL,
 | 
			
		||||
  REQUEST_TIMEOUT_MS,
 | 
			
		||||
} from "@/app/constant";
 | 
			
		||||
import { MultimodalContent, RequestMessage } from "@/app/client/api";
 | 
			
		||||
import { RequestMessage } from "@/app/client/api";
 | 
			
		||||
import Locale from "@/app/locales";
 | 
			
		||||
import {
 | 
			
		||||
  EventStreamContentType,
 | 
			
		||||
@@ -70,9 +70,8 @@ export function compressImage(file: Blob, maxSize: number): Promise<string> {
 | 
			
		||||
  });
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export async function preProcessImageContentBase(
 | 
			
		||||
export async function preProcessImageContent(
 | 
			
		||||
  content: RequestMessage["content"],
 | 
			
		||||
  transformImageUrl: (url: string) => Promise<{ [key: string]: any }>,
 | 
			
		||||
) {
 | 
			
		||||
  if (typeof content === "string") {
 | 
			
		||||
    return content;
 | 
			
		||||
@@ -82,7 +81,7 @@ export async function preProcessImageContentBase(
 | 
			
		||||
    if (part?.type == "image_url" && part?.image_url?.url) {
 | 
			
		||||
      try {
 | 
			
		||||
        const url = await cacheImageToBase64Image(part?.image_url?.url);
 | 
			
		||||
        result.push(await transformImageUrl(url));
 | 
			
		||||
        result.push({ type: part.type, image_url: { url } });
 | 
			
		||||
      } catch (error) {
 | 
			
		||||
        console.error("Error processing image URL:", error);
 | 
			
		||||
      }
 | 
			
		||||
@@ -93,23 +92,6 @@ export async function preProcessImageContentBase(
 | 
			
		||||
  return result;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export async function preProcessImageContent(
 | 
			
		||||
  content: RequestMessage["content"],
 | 
			
		||||
) {
 | 
			
		||||
  return preProcessImageContentBase(content, async (url) => ({
 | 
			
		||||
    type: "image_url",
 | 
			
		||||
    image_url: { url },
 | 
			
		||||
  })) as Promise<MultimodalContent[] | string>;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export async function preProcessImageContentForAlibabaDashScope(
 | 
			
		||||
  content: RequestMessage["content"],
 | 
			
		||||
) {
 | 
			
		||||
  return preProcessImageContentBase(content, async (url) => ({
 | 
			
		||||
    image: url,
 | 
			
		||||
  }));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
const imageCaches: Record<string, string> = {};
 | 
			
		||||
export function cacheImageToBase64Image(imageUrl: string) {
 | 
			
		||||
  if (imageUrl.includes(CACHE_URL_PREFIX)) {
 | 
			
		||||
 
 | 
			
		||||
@@ -17,40 +17,6 @@ services:
 | 
			
		||||
      - ENABLE_BALANCE_QUERY=$ENABLE_BALANCE_QUERY
 | 
			
		||||
      - DISABLE_FAST_LINK=$DISABLE_FAST_LINK
 | 
			
		||||
      - OPENAI_SB=$OPENAI_SB
 | 
			
		||||
      - SILICONFLOW_API_KEY=$SILICONFLOW_API_KEY
 | 
			
		||||
      - SILICONFLOW_URL=$SILICONFLOW_URL
 | 
			
		||||
      - AZURE_URL=$AZURE_URL
 | 
			
		||||
      - AZURE_API_KEY=$AZURE_API_KEY
 | 
			
		||||
      - AZURE_API_VERSION=$AZURE_API_VERSION
 | 
			
		||||
      - GOOGLE_URL=$GOOGLE_URL
 | 
			
		||||
      - GTM_ID=$GTM_ID
 | 
			
		||||
      - ANTHROPIC_URL=$ANTHROPIC_URL
 | 
			
		||||
      - ANTHROPIC_API_KEY=$ANTHROPIC_API_KEY
 | 
			
		||||
      - ANTHROPIC_API_VERSION=$ANTHROPIC_API_VERSION
 | 
			
		||||
      - BAIDU_URL=$BAIDU_URL
 | 
			
		||||
      - BAIDU_API_KEY=$BAIDU_API_KEY
 | 
			
		||||
      - BAIDU_SECRET_KEY=$BAIDU_SECRET_KEY
 | 
			
		||||
      - BYTEDANCE_URL=$BYTEDANCE_URL
 | 
			
		||||
      - BYTEDANCE_API_KEY=$BYTEDANCE_API_KEY
 | 
			
		||||
      - ALIBABA_URL=$ALIBABA_URL
 | 
			
		||||
      - ALIBABA_API_KEY=$ALIBABA_API_KEY
 | 
			
		||||
      - TENCENT_URL=$TENCENT_URL
 | 
			
		||||
      - TENCENT_SECRET_KEY=$TENCENT_SECRET_KEY
 | 
			
		||||
      - TENCENT_SECRET_ID=$TENCENT_SECRET_ID
 | 
			
		||||
      - MOONSHOT_URL=$MOONSHOT_URL
 | 
			
		||||
      - MOONSHOT_API_KEY=$MOONSHOT_API_KEY
 | 
			
		||||
      - IFLYTEK_URL=$IFLYTEK_URL
 | 
			
		||||
      - IFLYTEK_API_KEY=$IFLYTEK_API_KEY
 | 
			
		||||
      - IFLYTEK_API_SECRET=$IFLYTEK_API_SECRET
 | 
			
		||||
      - DEEPSEEK_URL=$DEEPSEEK_URL
 | 
			
		||||
      - DEEPSEEK_API_KEY=$DEEPSEEK_API_KEY
 | 
			
		||||
      - XAI_URL=$XAI_URL
 | 
			
		||||
      - XAI_API_KEY=$XAI_API_KEY
 | 
			
		||||
      - CHATGLM_URL=$CHATGLM_URL
 | 
			
		||||
      - CHATGLM_API_KEY=$CHATGLM_API_KEY
 | 
			
		||||
      - DEFAULT_INPUT_TEMPLATE=$DEFAULT_INPUT_TEMPLATE
 | 
			
		||||
      - HUAWEI_API_KEY=$HUAWEI_API_KEY
 | 
			
		||||
      - HUAWEI_URL=$HUAWEI_URL
 | 
			
		||||
 | 
			
		||||
  chatgpt-next-web-proxy:
 | 
			
		||||
    profiles: [ "proxy" ]
 | 
			
		||||
@@ -70,37 +36,3 @@ services:
 | 
			
		||||
      - ENABLE_BALANCE_QUERY=$ENABLE_BALANCE_QUERY
 | 
			
		||||
      - DISABLE_FAST_LINK=$DISABLE_FAST_LINK
 | 
			
		||||
      - OPENAI_SB=$OPENAI_SB
 | 
			
		||||
      - SILICONFLOW_API_KEY=$SILICONFLOW_API_KEY
 | 
			
		||||
      - SILICONFLOW_URL=$SILICONFLOW_URL
 | 
			
		||||
      - AZURE_URL=$AZURE_URL
 | 
			
		||||
      - AZURE_API_KEY=$AZURE_API_KEY
 | 
			
		||||
      - AZURE_API_VERSION=$AZURE_API_VERSION
 | 
			
		||||
      - GOOGLE_URL=$GOOGLE_URL
 | 
			
		||||
      - GTM_ID=$GTM_ID
 | 
			
		||||
      - ANTHROPIC_URL=$ANTHROPIC_URL
 | 
			
		||||
      - ANTHROPIC_API_KEY=$ANTHROPIC_API_KEY
 | 
			
		||||
      - ANTHROPIC_API_VERSION=$ANTHROPIC_API_VERSION
 | 
			
		||||
      - BAIDU_URL=$BAIDU_URL
 | 
			
		||||
      - BAIDU_API_KEY=$BAIDU_API_KEY
 | 
			
		||||
      - BAIDU_SECRET_KEY=$BAIDU_SECRET_KEY
 | 
			
		||||
      - BYTEDANCE_URL=$BYTEDANCE_URL
 | 
			
		||||
      - BYTEDANCE_API_KEY=$BYTEDANCE_API_KEY
 | 
			
		||||
      - ALIBABA_URL=$ALIBABA_URL
 | 
			
		||||
      - ALIBABA_API_KEY=$ALIBABA_API_KEY
 | 
			
		||||
      - TENCENT_URL=$TENCENT_URL
 | 
			
		||||
      - TENCENT_SECRET_KEY=$TENCENT_SECRET_KEY
 | 
			
		||||
      - TENCENT_SECRET_ID=$TENCENT_SECRET_ID
 | 
			
		||||
      - MOONSHOT_URL=$MOONSHOT_URL
 | 
			
		||||
      - MOONSHOT_API_KEY=$MOONSHOT_API_KEY
 | 
			
		||||
      - IFLYTEK_URL=$IFLYTEK_URL
 | 
			
		||||
      - IFLYTEK_API_KEY=$IFLYTEK_API_KEY
 | 
			
		||||
      - IFLYTEK_API_SECRET=$IFLYTEK_API_SECRET
 | 
			
		||||
      - DEEPSEEK_URL=$DEEPSEEK_URL
 | 
			
		||||
      - DEEPSEEK_API_KEY=$DEEPSEEK_API_KEY
 | 
			
		||||
      - XAI_URL=$XAI_URL
 | 
			
		||||
      - XAI_API_KEY=$XAI_API_KEY
 | 
			
		||||
      - CHATGLM_URL=$CHATGLM_URL
 | 
			
		||||
      - CHATGLM_API_KEY=$CHATGLM_API_KEY
 | 
			
		||||
      - DEFAULT_INPUT_TEMPLATE=$DEFAULT_INPUT_TEMPLATE
 | 
			
		||||
      - HUAWEI_API_KEY=$HUAWEI_API_KEY
 | 
			
		||||
      - HUAWEI_URL=$HUAWEI_URL
 | 
			
		||||
 
 | 
			
		||||
@@ -15,8 +15,6 @@ const config: Config = {
 | 
			
		||||
  moduleNameMapper: {
 | 
			
		||||
    "^@/(.*)$": "<rootDir>/$1",
 | 
			
		||||
  },
 | 
			
		||||
  extensionsToTreatAsEsm: [".ts", ".tsx"],
 | 
			
		||||
  injectGlobals: true,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
// createJestConfig is exported this way to ensure that next/jest can load the Next.js config which is async
 | 
			
		||||
 
 | 
			
		||||
@@ -1,22 +1,24 @@
 | 
			
		||||
// Learn more: https://github.com/testing-library/jest-dom
 | 
			
		||||
import "@testing-library/jest-dom";
 | 
			
		||||
import { jest } from "@jest/globals";
 | 
			
		||||
 | 
			
		||||
global.fetch = jest.fn(() =>
 | 
			
		||||
  Promise.resolve({
 | 
			
		||||
    ok: true,
 | 
			
		||||
    status: 200,
 | 
			
		||||
    json: () => Promise.resolve([]),
 | 
			
		||||
    json: () => Promise.resolve({}),
 | 
			
		||||
    headers: new Headers(),
 | 
			
		||||
    redirected: false,
 | 
			
		||||
    statusText: "OK",
 | 
			
		||||
    type: "basic",
 | 
			
		||||
    url: "",
 | 
			
		||||
    clone: function () {
 | 
			
		||||
      return this;
 | 
			
		||||
    },
 | 
			
		||||
    body: null,
 | 
			
		||||
    bodyUsed: false,
 | 
			
		||||
    arrayBuffer: () => Promise.resolve(new ArrayBuffer(0)),
 | 
			
		||||
    blob: () => Promise.resolve(new Blob()),
 | 
			
		||||
    formData: () => Promise.resolve(new FormData()),
 | 
			
		||||
    text: () => Promise.resolve(""),
 | 
			
		||||
  } as Response),
 | 
			
		||||
  }),
 | 
			
		||||
);
 | 
			
		||||
 
 | 
			
		||||
@@ -17,8 +17,8 @@
 | 
			
		||||
    "prompts": "node ./scripts/fetch-prompts.mjs",
 | 
			
		||||
    "prepare": "husky install",
 | 
			
		||||
    "proxy-dev": "sh ./scripts/init-proxy.sh && proxychains -f ./scripts/proxychains.conf yarn dev",
 | 
			
		||||
    "test": "node --no-warnings --experimental-vm-modules $(yarn bin jest) --watch",
 | 
			
		||||
    "test:ci": "node --no-warnings --experimental-vm-modules $(yarn bin jest) --ci"
 | 
			
		||||
    "test": "jest --watch",
 | 
			
		||||
    "test:ci": "jest --ci"
 | 
			
		||||
  },
 | 
			
		||||
  "dependencies": {
 | 
			
		||||
    "@fortaine/fetch-event-source": "^3.0.6",
 | 
			
		||||
 
 | 
			
		||||
@@ -1,4 +1,3 @@
 | 
			
		||||
import { jest } from "@jest/globals";
 | 
			
		||||
import { isVisionModel } from "../app/utils";
 | 
			
		||||
 | 
			
		||||
describe("isVisionModel", () => {
 | 
			
		||||
 
 | 
			
		||||
		Reference in New Issue
	
	Block a user