mirror of
				https://github.com/ChatGPTNextWeb/ChatGPT-Next-Web.git
				synced 2025-10-26 11:53:43 +08:00 
			
		
		
		
	Compare commits
	
		
			20 Commits
		
	
	
		
			6305-bugth
			...
			7e3371008b
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 7e3371008b | ||
|  | 11b37c15bd | ||
|  | 1d0038f17d | ||
|  | 619fa519c0 | ||
|  | 48469bd8ca | ||
|  | 5a5e887f2b | ||
|  | b6f5d75656 | ||
|  | 0d41a17ef6 | ||
|  | f7cde17919 | ||
|  | 570cbb34b6 | ||
|  | 7aa9ae0a3e | ||
|  | ad6666eeaf | ||
|  | a2c4e468a0 | ||
|  | 0a25a1a8cb | ||
|  | b709ee3983 | ||
|  | ebf84a3b2c | ||
|  | fd796521d4 | ||
|  | 6a25bdcf2d | ||
|  | a899e9aceb | ||
|  | c0826d2275 | 
| @@ -81,3 +81,9 @@ SILICONFLOW_API_KEY= | |||||||
|  |  | ||||||
| ### siliconflow Api url (optional) | ### siliconflow Api url (optional) | ||||||
| SILICONFLOW_URL= | SILICONFLOW_URL= | ||||||
|  |  | ||||||
|  | ### ppio Api key (optional) | ||||||
|  | PPIO_API_KEY= | ||||||
|  |  | ||||||
|  | ### ppio Api url (optional) | ||||||
|  | PPIO_URL= | ||||||
|   | |||||||
							
								
								
									
										29
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										29
									
								
								README.md
									
									
									
									
									
								
							| @@ -7,7 +7,7 @@ | |||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| <h1 align="center">NextChat (ChatGPT Next Web)</h1> | <h1 align="center">NextChat</h1> | ||||||
|  |  | ||||||
| English / [简体中文](./README_CN.md) | English / [简体中文](./README_CN.md) | ||||||
|  |  | ||||||
| @@ -22,7 +22,6 @@ English / [简体中文](./README_CN.md) | |||||||
| [![MacOS][MacOS-image]][download-url] | [![MacOS][MacOS-image]][download-url] | ||||||
| [![Linux][Linux-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) | [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) | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -41,24 +40,6 @@ English / [简体中文](./README_CN.md) | |||||||
|  |  | ||||||
| </div> | </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! | ## 🥳 Cheer for DeepSeek, China's AI star! | ||||||
|  > Purpose-Built UI for DeepSeek Reasoner Model |  > Purpose-Built UI for DeepSeek Reasoner Model | ||||||
|   |   | ||||||
| @@ -379,6 +360,14 @@ SiliconFlow API Key. | |||||||
|  |  | ||||||
| SiliconFlow API URL. | SiliconFlow API URL. | ||||||
|  |  | ||||||
|  | ### `PPIO_API_KEY` (optional) | ||||||
|  |  | ||||||
|  | PPIO API Key. | ||||||
|  |  | ||||||
|  | ### `PPIO_URL` (optional) | ||||||
|  |  | ||||||
|  | PPIO API URL. | ||||||
|  |  | ||||||
| ## Requirements | ## Requirements | ||||||
|  |  | ||||||
| NodeJS >= 18, Docker >= 20 | NodeJS >= 18, Docker >= 20 | ||||||
|   | |||||||
| @@ -275,6 +275,14 @@ SiliconFlow API Key. | |||||||
|  |  | ||||||
| SiliconFlow API URL. | SiliconFlow API URL. | ||||||
|  |  | ||||||
|  | ### `PPIO_API_KEY` (optional) | ||||||
|  |  | ||||||
|  | PPIO API Key. | ||||||
|  |  | ||||||
|  | ### `PPIO_URL` (optional) | ||||||
|  |  | ||||||
|  | PPIO API URL. | ||||||
|  |  | ||||||
| ## 开发 | ## 开发 | ||||||
|  |  | ||||||
| 点击下方按钮,开始二次开发: | 点击下方按钮,开始二次开发: | ||||||
|   | |||||||
| @@ -12,6 +12,7 @@ import { handle as stabilityHandler } from "../../stability"; | |||||||
| import { handle as iflytekHandler } from "../../iflytek"; | import { handle as iflytekHandler } from "../../iflytek"; | ||||||
| import { handle as deepseekHandler } from "../../deepseek"; | import { handle as deepseekHandler } from "../../deepseek"; | ||||||
| import { handle as siliconflowHandler } from "../../siliconflow"; | import { handle as siliconflowHandler } from "../../siliconflow"; | ||||||
|  | import { handle as ppioHandler } from "../../ppio"; | ||||||
| import { handle as xaiHandler } from "../../xai"; | import { handle as xaiHandler } from "../../xai"; | ||||||
| import { handle as chatglmHandler } from "../../glm"; | import { handle as chatglmHandler } from "../../glm"; | ||||||
| import { handle as proxyHandler } from "../../proxy"; | import { handle as proxyHandler } from "../../proxy"; | ||||||
| @@ -50,6 +51,8 @@ async function handle( | |||||||
|       return chatglmHandler(req, { params }); |       return chatglmHandler(req, { params }); | ||||||
|     case ApiPath.SiliconFlow: |     case ApiPath.SiliconFlow: | ||||||
|       return siliconflowHandler(req, { params }); |       return siliconflowHandler(req, { params }); | ||||||
|  |     case ApiPath.PPIO: | ||||||
|  |       return ppioHandler(req, { params }); | ||||||
|     case ApiPath.OpenAI: |     case ApiPath.OpenAI: | ||||||
|       return openaiHandler(req, { params }); |       return openaiHandler(req, { params }); | ||||||
|     default: |     default: | ||||||
|   | |||||||
| @@ -104,6 +104,9 @@ export function auth(req: NextRequest, modelProvider: ModelProvider) { | |||||||
|       case ModelProvider.SiliconFlow: |       case ModelProvider.SiliconFlow: | ||||||
|         systemApiKey = serverConfig.siliconFlowApiKey; |         systemApiKey = serverConfig.siliconFlowApiKey; | ||||||
|         break; |         break; | ||||||
|  |       case ModelProvider.PPIO: | ||||||
|  |         systemApiKey = serverConfig.ppioApiKey; | ||||||
|  |         break; | ||||||
|       case ModelProvider.GPT: |       case ModelProvider.GPT: | ||||||
|       default: |       default: | ||||||
|         if (req.nextUrl.pathname.includes("azure/deployments")) { |         if (req.nextUrl.pathname.includes("azure/deployments")) { | ||||||
|   | |||||||
							
								
								
									
										128
									
								
								app/api/ppio.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										128
									
								
								app/api/ppio.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,128 @@ | |||||||
|  | import { getServerSideConfig } from "@/app/config/server"; | ||||||
|  | import { | ||||||
|  |   PPIO_BASE_URL, | ||||||
|  |   ApiPath, | ||||||
|  |   ModelProvider, | ||||||
|  |   ServiceProvider, | ||||||
|  | } from "@/app/constant"; | ||||||
|  | import { prettyObject } from "@/app/utils/format"; | ||||||
|  | import { NextRequest, NextResponse } from "next/server"; | ||||||
|  | import { auth } from "@/app/api/auth"; | ||||||
|  | import { isModelNotavailableInServer } from "@/app/utils/model"; | ||||||
|  |  | ||||||
|  | const serverConfig = getServerSideConfig(); | ||||||
|  |  | ||||||
|  | export async function handle( | ||||||
|  |   req: NextRequest, | ||||||
|  |   { params }: { params: { path: string[] } }, | ||||||
|  | ) { | ||||||
|  |   console.log("[PPIO Route] params ", params); | ||||||
|  |  | ||||||
|  |   if (req.method === "OPTIONS") { | ||||||
|  |     return NextResponse.json({ body: "OK" }, { status: 200 }); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   const authResult = auth(req, ModelProvider.PPIO); | ||||||
|  |   if (authResult.error) { | ||||||
|  |     return NextResponse.json(authResult, { | ||||||
|  |       status: 401, | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   try { | ||||||
|  |     const response = await request(req); | ||||||
|  |     return response; | ||||||
|  |   } catch (e) { | ||||||
|  |     console.error("[PPIO] ", e); | ||||||
|  |     return NextResponse.json(prettyObject(e)); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | async function request(req: NextRequest) { | ||||||
|  |   const controller = new AbortController(); | ||||||
|  |  | ||||||
|  |   // alibaba use base url or just remove the path | ||||||
|  |   let path = `${req.nextUrl.pathname}`.replaceAll(ApiPath.PPIO, ""); | ||||||
|  |  | ||||||
|  |   let baseUrl = serverConfig.ppioUrl || PPIO_BASE_URL; | ||||||
|  |  | ||||||
|  |   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, | ||||||
|  |   ); | ||||||
|  |  | ||||||
|  |   const fetchUrl = `${baseUrl}${path}`; | ||||||
|  |   const fetchOptions: RequestInit = { | ||||||
|  |     headers: { | ||||||
|  |       "Content-Type": "application/json", | ||||||
|  |       Authorization: req.headers.get("Authorization") ?? "", | ||||||
|  |     }, | ||||||
|  |     method: req.method, | ||||||
|  |     body: req.body, | ||||||
|  |     redirect: "manual", | ||||||
|  |     // @ts-ignore | ||||||
|  |     duplex: "half", | ||||||
|  |     signal: controller.signal, | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   // #1815 try to refuse some request to some models | ||||||
|  |   if (serverConfig.customModels && req.body) { | ||||||
|  |     try { | ||||||
|  |       const clonedBody = await req.text(); | ||||||
|  |       fetchOptions.body = clonedBody; | ||||||
|  |  | ||||||
|  |       const jsonBody = JSON.parse(clonedBody) as { model?: string }; | ||||||
|  |  | ||||||
|  |       // not undefined and is false | ||||||
|  |       if ( | ||||||
|  |         isModelNotavailableInServer( | ||||||
|  |           serverConfig.customModels, | ||||||
|  |           jsonBody?.model as string, | ||||||
|  |           ServiceProvider.PPIO as string, | ||||||
|  |         ) | ||||||
|  |       ) { | ||||||
|  |         return NextResponse.json( | ||||||
|  |           { | ||||||
|  |             error: true, | ||||||
|  |             message: `you are not allowed to use ${jsonBody?.model} model`, | ||||||
|  |           }, | ||||||
|  |           { | ||||||
|  |             status: 403, | ||||||
|  |           }, | ||||||
|  |         ); | ||||||
|  |       } | ||||||
|  |     } catch (e) { | ||||||
|  |       console.error(`[PPIO] filter`, e); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   try { | ||||||
|  |     const res = await fetch(fetchUrl, fetchOptions); | ||||||
|  |  | ||||||
|  |     // to prevent browser prompt for credentials | ||||||
|  |     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); | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -24,6 +24,7 @@ import { DeepSeekApi } from "./platforms/deepseek"; | |||||||
| import { XAIApi } from "./platforms/xai"; | import { XAIApi } from "./platforms/xai"; | ||||||
| import { ChatGLMApi } from "./platforms/glm"; | import { ChatGLMApi } from "./platforms/glm"; | ||||||
| import { SiliconflowApi } from "./platforms/siliconflow"; | import { SiliconflowApi } from "./platforms/siliconflow"; | ||||||
|  | import { PPIOApi } from "./platforms/ppio"; | ||||||
|  |  | ||||||
| export const ROLES = ["system", "user", "assistant"] as const; | export const ROLES = ["system", "user", "assistant"] as const; | ||||||
| export type MessageRole = (typeof ROLES)[number]; | export type MessageRole = (typeof ROLES)[number]; | ||||||
| @@ -40,6 +41,11 @@ export interface MultimodalContent { | |||||||
|   }; |   }; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | export interface MultimodalContentForAlibaba { | ||||||
|  |   text?: string; | ||||||
|  |   image?: string; | ||||||
|  | } | ||||||
|  |  | ||||||
| export interface RequestMessage { | export interface RequestMessage { | ||||||
|   role: MessageRole; |   role: MessageRole; | ||||||
|   content: string | MultimodalContent[]; |   content: string | MultimodalContent[]; | ||||||
| @@ -168,6 +174,9 @@ export class ClientApi { | |||||||
|       case ModelProvider.SiliconFlow: |       case ModelProvider.SiliconFlow: | ||||||
|         this.llm = new SiliconflowApi(); |         this.llm = new SiliconflowApi(); | ||||||
|         break; |         break; | ||||||
|  |       case ModelProvider.PPIO: | ||||||
|  |         this.llm = new PPIOApi(); | ||||||
|  |         break; | ||||||
|       default: |       default: | ||||||
|         this.llm = new ChatGPTApi(); |         this.llm = new ChatGPTApi(); | ||||||
|     } |     } | ||||||
| @@ -260,6 +269,7 @@ export function getHeaders(ignoreHeaders: boolean = false) { | |||||||
|     const isChatGLM = modelConfig.providerName === ServiceProvider.ChatGLM; |     const isChatGLM = modelConfig.providerName === ServiceProvider.ChatGLM; | ||||||
|     const isSiliconFlow = |     const isSiliconFlow = | ||||||
|       modelConfig.providerName === ServiceProvider.SiliconFlow; |       modelConfig.providerName === ServiceProvider.SiliconFlow; | ||||||
|  |     const isPPIO = modelConfig.providerName === ServiceProvider.PPIO; | ||||||
|     const isEnabledAccessControl = accessStore.enabledAccessControl(); |     const isEnabledAccessControl = accessStore.enabledAccessControl(); | ||||||
|     const apiKey = isGoogle |     const apiKey = isGoogle | ||||||
|       ? accessStore.googleApiKey |       ? accessStore.googleApiKey | ||||||
| @@ -281,6 +291,8 @@ export function getHeaders(ignoreHeaders: boolean = false) { | |||||||
|       ? accessStore.chatglmApiKey |       ? accessStore.chatglmApiKey | ||||||
|       : isSiliconFlow |       : isSiliconFlow | ||||||
|       ? accessStore.siliconflowApiKey |       ? accessStore.siliconflowApiKey | ||||||
|  |       : isPPIO | ||||||
|  |       ? accessStore.ppioApiKey | ||||||
|       : isIflytek |       : isIflytek | ||||||
|       ? accessStore.iflytekApiKey && accessStore.iflytekApiSecret |       ? accessStore.iflytekApiKey && accessStore.iflytekApiSecret | ||||||
|         ? accessStore.iflytekApiKey + ":" + accessStore.iflytekApiSecret |         ? accessStore.iflytekApiKey + ":" + accessStore.iflytekApiSecret | ||||||
| @@ -299,6 +311,7 @@ export function getHeaders(ignoreHeaders: boolean = false) { | |||||||
|       isXAI, |       isXAI, | ||||||
|       isChatGLM, |       isChatGLM, | ||||||
|       isSiliconFlow, |       isSiliconFlow, | ||||||
|  |       isPPIO, | ||||||
|       apiKey, |       apiKey, | ||||||
|       isEnabledAccessControl, |       isEnabledAccessControl, | ||||||
|     }; |     }; | ||||||
| @@ -327,6 +340,7 @@ export function getHeaders(ignoreHeaders: boolean = false) { | |||||||
|     isXAI, |     isXAI, | ||||||
|     isChatGLM, |     isChatGLM, | ||||||
|     isSiliconFlow, |     isSiliconFlow, | ||||||
|  |     isPPIO, | ||||||
|     apiKey, |     apiKey, | ||||||
|     isEnabledAccessControl, |     isEnabledAccessControl, | ||||||
|   } = getConfig(); |   } = getConfig(); | ||||||
| @@ -377,6 +391,8 @@ export function getClientApi(provider: ServiceProvider): ClientApi { | |||||||
|       return new ClientApi(ModelProvider.ChatGLM); |       return new ClientApi(ModelProvider.ChatGLM); | ||||||
|     case ServiceProvider.SiliconFlow: |     case ServiceProvider.SiliconFlow: | ||||||
|       return new ClientApi(ModelProvider.SiliconFlow); |       return new ClientApi(ModelProvider.SiliconFlow); | ||||||
|  |     case ServiceProvider.PPIO: | ||||||
|  |       return new ClientApi(ModelProvider.PPIO); | ||||||
|     default: |     default: | ||||||
|       return new ClientApi(ModelProvider.GPT); |       return new ClientApi(ModelProvider.GPT); | ||||||
|   } |   } | ||||||
|   | |||||||
| @@ -7,7 +7,10 @@ import { | |||||||
|   ChatMessageTool, |   ChatMessageTool, | ||||||
|   usePluginStore, |   usePluginStore, | ||||||
| } from "@/app/store"; | } from "@/app/store"; | ||||||
| import { streamWithThink } from "@/app/utils/chat"; | import { | ||||||
|  |   preProcessImageContentForAlibabaDashScope, | ||||||
|  |   streamWithThink, | ||||||
|  | } from "@/app/utils/chat"; | ||||||
| import { | import { | ||||||
|   ChatOptions, |   ChatOptions, | ||||||
|   getHeaders, |   getHeaders, | ||||||
| @@ -15,12 +18,14 @@ import { | |||||||
|   LLMModel, |   LLMModel, | ||||||
|   SpeechOptions, |   SpeechOptions, | ||||||
|   MultimodalContent, |   MultimodalContent, | ||||||
|  |   MultimodalContentForAlibaba, | ||||||
| } from "../api"; | } from "../api"; | ||||||
| import { getClientConfig } from "@/app/config/client"; | import { getClientConfig } from "@/app/config/client"; | ||||||
| import { | import { | ||||||
|   getMessageTextContent, |   getMessageTextContent, | ||||||
|   getMessageTextContentWithoutThinking, |   getMessageTextContentWithoutThinking, | ||||||
|   getTimeoutMSByModel, |   getTimeoutMSByModel, | ||||||
|  |   isVisionModel, | ||||||
| } from "@/app/utils"; | } from "@/app/utils"; | ||||||
| import { fetch } from "@/app/utils/stream"; | import { fetch } from "@/app/utils/stream"; | ||||||
|  |  | ||||||
| @@ -89,14 +94,6 @@ export class QwenApi implements LLMApi { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   async chat(options: ChatOptions) { |   async chat(options: ChatOptions) { | ||||||
|     const messages = options.messages.map((v) => ({ |  | ||||||
|       role: v.role, |  | ||||||
|       content: |  | ||||||
|         v.role === "assistant" |  | ||||||
|           ? getMessageTextContentWithoutThinking(v) |  | ||||||
|           : getMessageTextContent(v), |  | ||||||
|     })); |  | ||||||
|  |  | ||||||
|     const modelConfig = { |     const modelConfig = { | ||||||
|       ...useAppConfig.getState().modelConfig, |       ...useAppConfig.getState().modelConfig, | ||||||
|       ...useChatStore.getState().currentSession().mask.modelConfig, |       ...useChatStore.getState().currentSession().mask.modelConfig, | ||||||
| @@ -105,6 +102,21 @@ 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 shouldStream = !!options.config.stream; | ||||||
|     const requestPayload: RequestPayload = { |     const requestPayload: RequestPayload = { | ||||||
|       model: modelConfig.model, |       model: modelConfig.model, | ||||||
| @@ -129,7 +141,7 @@ export class QwenApi implements LLMApi { | |||||||
|         "X-DashScope-SSE": shouldStream ? "enable" : "disable", |         "X-DashScope-SSE": shouldStream ? "enable" : "disable", | ||||||
|       }; |       }; | ||||||
|  |  | ||||||
|       const chatPath = this.path(Alibaba.ChatPath); |       const chatPath = this.path(Alibaba.ChatPath(modelConfig.model)); | ||||||
|       const chatPayload = { |       const chatPayload = { | ||||||
|         method: "POST", |         method: "POST", | ||||||
|         body: JSON.stringify(requestPayload), |         body: JSON.stringify(requestPayload), | ||||||
| @@ -162,7 +174,7 @@ export class QwenApi implements LLMApi { | |||||||
|             const json = JSON.parse(text); |             const json = JSON.parse(text); | ||||||
|             const choices = json.output.choices as Array<{ |             const choices = json.output.choices as Array<{ | ||||||
|               message: { |               message: { | ||||||
|                 content: string | null; |                 content: string | null | MultimodalContentForAlibaba[]; | ||||||
|                 tool_calls: ChatMessageTool[]; |                 tool_calls: ChatMessageTool[]; | ||||||
|                 reasoning_content: string | null; |                 reasoning_content: string | null; | ||||||
|               }; |               }; | ||||||
| @@ -212,7 +224,9 @@ export class QwenApi implements LLMApi { | |||||||
|             } else if (content && content.length > 0) { |             } else if (content && content.length > 0) { | ||||||
|               return { |               return { | ||||||
|                 isThinking: false, |                 isThinking: false, | ||||||
|                 content: content, |                 content: Array.isArray(content) | ||||||
|  |                   ? content.map((item) => item.text).join(",") | ||||||
|  |                   : content, | ||||||
|               }; |               }; | ||||||
|             } |             } | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										279
									
								
								app/client/platforms/ppio.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										279
									
								
								app/client/platforms/ppio.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,279 @@ | |||||||
|  | "use client"; | ||||||
|  | // azure and openai, using same models. so using same LLMApi. | ||||||
|  | import { ApiPath, PPIO_BASE_URL, PPIO, DEFAULT_MODELS } from "@/app/constant"; | ||||||
|  | import { | ||||||
|  |   useAccessStore, | ||||||
|  |   useAppConfig, | ||||||
|  |   useChatStore, | ||||||
|  |   ChatMessageTool, | ||||||
|  |   usePluginStore, | ||||||
|  | } from "@/app/store"; | ||||||
|  | import { preProcessImageContent, streamWithThink } from "@/app/utils/chat"; | ||||||
|  | import { | ||||||
|  |   ChatOptions, | ||||||
|  |   getHeaders, | ||||||
|  |   LLMApi, | ||||||
|  |   LLMModel, | ||||||
|  |   SpeechOptions, | ||||||
|  | } from "../api"; | ||||||
|  | import { getClientConfig } from "@/app/config/client"; | ||||||
|  | import { | ||||||
|  |   getMessageTextContent, | ||||||
|  |   getMessageTextContentWithoutThinking, | ||||||
|  |   isVisionModel, | ||||||
|  |   getTimeoutMSByModel, | ||||||
|  | } from "@/app/utils"; | ||||||
|  | import { RequestPayload } from "./openai"; | ||||||
|  |  | ||||||
|  | import { fetch } from "@/app/utils/stream"; | ||||||
|  | export interface PPIOListModelResponse { | ||||||
|  |   object: string; | ||||||
|  |   data: Array<{ | ||||||
|  |     id: string; | ||||||
|  |     object: string; | ||||||
|  |     root: string; | ||||||
|  |   }>; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export class PPIOApi implements LLMApi { | ||||||
|  |   private disableListModels = false; | ||||||
|  |  | ||||||
|  |   path(path: string): string { | ||||||
|  |     const accessStore = useAccessStore.getState(); | ||||||
|  |  | ||||||
|  |     let baseUrl = ""; | ||||||
|  |  | ||||||
|  |     if (accessStore.useCustomConfig) { | ||||||
|  |       baseUrl = accessStore.ppioUrl; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if (baseUrl.length === 0) { | ||||||
|  |       const isApp = !!getClientConfig()?.isApp; | ||||||
|  |       const apiPath = ApiPath.PPIO; | ||||||
|  |       baseUrl = isApp ? PPIO_BASE_URL : apiPath; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if (baseUrl.endsWith("/")) { | ||||||
|  |       baseUrl = baseUrl.slice(0, baseUrl.length - 1); | ||||||
|  |     } | ||||||
|  |     if (!baseUrl.startsWith("http") && !baseUrl.startsWith(ApiPath.PPIO)) { | ||||||
|  |       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 visionModel = isVisionModel(options.config.model); | ||||||
|  |     const messages: ChatOptions["messages"] = []; | ||||||
|  |     for (const v of options.messages) { | ||||||
|  |       if (v.role === "assistant") { | ||||||
|  |         const content = getMessageTextContentWithoutThinking(v); | ||||||
|  |         messages.push({ role: v.role, content }); | ||||||
|  |       } else { | ||||||
|  |         const content = visionModel | ||||||
|  |           ? await preProcessImageContent(v.content) | ||||||
|  |           : getMessageTextContent(v); | ||||||
|  |         messages.push({ role: v.role, content }); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     const modelConfig = { | ||||||
|  |       ...useAppConfig.getState().modelConfig, | ||||||
|  |       ...useChatStore.getState().currentSession().mask.modelConfig, | ||||||
|  |       ...{ | ||||||
|  |         model: options.config.model, | ||||||
|  |         providerName: options.config.providerName, | ||||||
|  |       }, | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     const requestPayload: RequestPayload = { | ||||||
|  |       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, | ||||||
|  |       // max_tokens: Math.max(modelConfig.max_tokens, 1024), | ||||||
|  |       // Please do not ask me why not send max_tokens, no reason, this param is just shit, I dont want to explain anymore. | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     console.log("[Request] openai payload: ", requestPayload); | ||||||
|  |  | ||||||
|  |     const shouldStream = !!options.config.stream; | ||||||
|  |     const controller = new AbortController(); | ||||||
|  |     options.onController?.(controller); | ||||||
|  |  | ||||||
|  |     try { | ||||||
|  |       const chatPath = this.path(PPIO.ChatPath); | ||||||
|  |       const chatPayload = { | ||||||
|  |         method: "POST", | ||||||
|  |         body: JSON.stringify(requestPayload), | ||||||
|  |         signal: controller.signal, | ||||||
|  |         headers: getHeaders(), | ||||||
|  |       }; | ||||||
|  |  | ||||||
|  |       // console.log(chatPayload); | ||||||
|  |  | ||||||
|  |       // Use extended timeout for thinking models as they typically require more processing time | ||||||
|  |       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[]) => { | ||||||
|  |             // console.log("parseSSE", text, runTools); | ||||||
|  |             const json = JSON.parse(text); | ||||||
|  |             const choices = json.choices as Array<{ | ||||||
|  |               delta: { | ||||||
|  |                 content: string | null; | ||||||
|  |                 tool_calls: ChatMessageTool[]; | ||||||
|  |                 reasoning_content: string | null; | ||||||
|  |               }; | ||||||
|  |             }>; | ||||||
|  |             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; | ||||||
|  |               } | ||||||
|  |             } | ||||||
|  |             const reasoning = choices[0]?.delta?.reasoning_content; | ||||||
|  |             const content = choices[0]?.delta?.content; | ||||||
|  |  | ||||||
|  |             // Skip if both content and reasoning_content are empty or null | ||||||
|  |             if ( | ||||||
|  |               (!reasoning || reasoning.length === 0) && | ||||||
|  |               (!content || content.length === 0) | ||||||
|  |             ) { | ||||||
|  |               return { | ||||||
|  |                 isThinking: false, | ||||||
|  |                 content: "", | ||||||
|  |               }; | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             if (reasoning && reasoning.length > 0) { | ||||||
|  |               return { | ||||||
|  |                 isThinking: true, | ||||||
|  |                 content: reasoning, | ||||||
|  |               }; | ||||||
|  |             } else if (content && content.length > 0) { | ||||||
|  |               return { | ||||||
|  |                 isThinking: false, | ||||||
|  |                 content: content, | ||||||
|  |               }; | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             return { | ||||||
|  |               isThinking: false, | ||||||
|  |               content: "", | ||||||
|  |             }; | ||||||
|  |           }, | ||||||
|  |           // processToolMessage, include tool_calls message and tool call results | ||||||
|  |           ( | ||||||
|  |             requestPayload: RequestPayload, | ||||||
|  |             toolCallMessage: any, | ||||||
|  |             toolCallResult: any[], | ||||||
|  |           ) => { | ||||||
|  |             // @ts-ignore | ||||||
|  |             requestPayload?.messages?.splice( | ||||||
|  |               // @ts-ignore | ||||||
|  |               requestPayload?.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[]> { | ||||||
|  |     if (this.disableListModels) { | ||||||
|  |       return DEFAULT_MODELS.slice(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     const res = await fetch(this.path(PPIO.ListModelPath), { | ||||||
|  |       method: "GET", | ||||||
|  |       headers: { | ||||||
|  |         ...getHeaders(), | ||||||
|  |       }, | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const resJson = (await res.json()) as PPIOListModelResponse; | ||||||
|  |     const chatModels = resJson.data; | ||||||
|  |     console.log("[Models]", chatModels); | ||||||
|  |  | ||||||
|  |     if (!chatModels) { | ||||||
|  |       return []; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     let seq = 1000; | ||||||
|  |     return chatModels.map((m) => ({ | ||||||
|  |       name: m.id, | ||||||
|  |       available: true, | ||||||
|  |       sorted: seq++, | ||||||
|  |       provider: { | ||||||
|  |         id: "ppio", | ||||||
|  |         providerName: "PPIO", | ||||||
|  |         providerType: "ppio", | ||||||
|  |         sorted: 15, | ||||||
|  |       }, | ||||||
|  |     })); | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -75,6 +75,7 @@ import { | |||||||
|   ChatGLM, |   ChatGLM, | ||||||
|   DeepSeek, |   DeepSeek, | ||||||
|   SiliconFlow, |   SiliconFlow, | ||||||
|  |   PPIO, | ||||||
| } from "../constant"; | } from "../constant"; | ||||||
| import { Prompt, SearchService, usePromptStore } from "../store/prompt"; | import { Prompt, SearchService, usePromptStore } from "../store/prompt"; | ||||||
| import { ErrorBoundary } from "./error"; | import { ErrorBoundary } from "./error"; | ||||||
| @@ -1359,6 +1360,44 @@ export function Settings() { | |||||||
|       </ListItem> |       </ListItem> | ||||||
|     </> |     </> | ||||||
|   ); |   ); | ||||||
|  |   const ppioConfigComponent = accessStore.provider === ServiceProvider.PPIO && ( | ||||||
|  |     <> | ||||||
|  |       <ListItem | ||||||
|  |         title={Locale.Settings.Access.PPIO.Endpoint.Title} | ||||||
|  |         subTitle={ | ||||||
|  |           Locale.Settings.Access.PPIO.Endpoint.SubTitle + PPIO.ExampleEndpoint | ||||||
|  |         } | ||||||
|  |       > | ||||||
|  |         <input | ||||||
|  |           aria-label={Locale.Settings.Access.PPIO.Endpoint.Title} | ||||||
|  |           type="text" | ||||||
|  |           value={accessStore.ppioUrl} | ||||||
|  |           placeholder={PPIO.ExampleEndpoint} | ||||||
|  |           onChange={(e) => | ||||||
|  |             accessStore.update( | ||||||
|  |               (access) => (access.ppioUrl = e.currentTarget.value), | ||||||
|  |             ) | ||||||
|  |           } | ||||||
|  |         ></input> | ||||||
|  |       </ListItem> | ||||||
|  |       <ListItem | ||||||
|  |         title={Locale.Settings.Access.PPIO.ApiKey.Title} | ||||||
|  |         subTitle={Locale.Settings.Access.PPIO.ApiKey.SubTitle} | ||||||
|  |       > | ||||||
|  |         <PasswordInput | ||||||
|  |           aria-label={Locale.Settings.Access.PPIO.ApiKey.Title} | ||||||
|  |           value={accessStore.ppioApiKey} | ||||||
|  |           type="text" | ||||||
|  |           placeholder={Locale.Settings.Access.PPIO.ApiKey.Placeholder} | ||||||
|  |           onChange={(e) => { | ||||||
|  |             accessStore.update( | ||||||
|  |               (access) => (access.ppioApiKey = e.currentTarget.value), | ||||||
|  |             ); | ||||||
|  |           }} | ||||||
|  |         /> | ||||||
|  |       </ListItem> | ||||||
|  |     </> | ||||||
|  |   ); | ||||||
|  |  | ||||||
|   const stabilityConfigComponent = accessStore.provider === |   const stabilityConfigComponent = accessStore.provider === | ||||||
|     ServiceProvider.Stability && ( |     ServiceProvider.Stability && ( | ||||||
| @@ -1822,6 +1861,7 @@ export function Settings() { | |||||||
|                   {XAIConfigComponent} |                   {XAIConfigComponent} | ||||||
|                   {chatglmConfigComponent} |                   {chatglmConfigComponent} | ||||||
|                   {siliconflowConfigComponent} |                   {siliconflowConfigComponent} | ||||||
|  |                   {ppioConfigComponent} | ||||||
|                 </> |                 </> | ||||||
|               )} |               )} | ||||||
|             </> |             </> | ||||||
|   | |||||||
| @@ -88,6 +88,10 @@ declare global { | |||||||
|       SILICONFLOW_URL?: string; |       SILICONFLOW_URL?: string; | ||||||
|       SILICONFLOW_API_KEY?: string; |       SILICONFLOW_API_KEY?: string; | ||||||
|  |  | ||||||
|  |       // ppio only | ||||||
|  |       PPIO_URL?: string; | ||||||
|  |       PPIO_API_KEY?: string; | ||||||
|  |  | ||||||
|       // custom template for preprocessing user input |       // custom template for preprocessing user input | ||||||
|       DEFAULT_INPUT_TEMPLATE?: string; |       DEFAULT_INPUT_TEMPLATE?: string; | ||||||
|  |  | ||||||
| @@ -163,6 +167,7 @@ export const getServerSideConfig = () => { | |||||||
|   const isXAI = !!process.env.XAI_API_KEY; |   const isXAI = !!process.env.XAI_API_KEY; | ||||||
|   const isChatGLM = !!process.env.CHATGLM_API_KEY; |   const isChatGLM = !!process.env.CHATGLM_API_KEY; | ||||||
|   const isSiliconFlow = !!process.env.SILICONFLOW_API_KEY; |   const isSiliconFlow = !!process.env.SILICONFLOW_API_KEY; | ||||||
|  |   const isPPIO = !!process.env.PPIO_API_KEY; | ||||||
|   // const apiKeyEnvVar = process.env.OPENAI_API_KEY ?? ""; |   // const apiKeyEnvVar = process.env.OPENAI_API_KEY ?? ""; | ||||||
|   // const apiKeys = apiKeyEnvVar.split(",").map((v) => v.trim()); |   // const apiKeys = apiKeyEnvVar.split(",").map((v) => v.trim()); | ||||||
|   // const randomIndex = Math.floor(Math.random() * apiKeys.length); |   // const randomIndex = Math.floor(Math.random() * apiKeys.length); | ||||||
| @@ -246,6 +251,10 @@ export const getServerSideConfig = () => { | |||||||
|     siliconFlowUrl: process.env.SILICONFLOW_URL, |     siliconFlowUrl: process.env.SILICONFLOW_URL, | ||||||
|     siliconFlowApiKey: getApiKey(process.env.SILICONFLOW_API_KEY), |     siliconFlowApiKey: getApiKey(process.env.SILICONFLOW_API_KEY), | ||||||
|  |  | ||||||
|  |     isPPIO, | ||||||
|  |     ppioUrl: process.env.PPIO_URL, | ||||||
|  |     ppioApiKey: getApiKey(process.env.PPIO_API_KEY), | ||||||
|  |  | ||||||
|     gtmId: process.env.GTM_ID, |     gtmId: process.env.GTM_ID, | ||||||
|     gaId: process.env.GA_ID || DEFAULT_GA_ID, |     gaId: process.env.GA_ID || DEFAULT_GA_ID, | ||||||
|  |  | ||||||
|   | |||||||
| @@ -36,6 +36,8 @@ export const CHATGLM_BASE_URL = "https://open.bigmodel.cn"; | |||||||
|  |  | ||||||
| export const SILICONFLOW_BASE_URL = "https://api.siliconflow.cn"; | export const SILICONFLOW_BASE_URL = "https://api.siliconflow.cn"; | ||||||
|  |  | ||||||
|  | export const PPIO_BASE_URL = "https://api.ppinfra.com/v3/openai"; | ||||||
|  |  | ||||||
| export const CACHE_URL_PREFIX = "/api/cache"; | export const CACHE_URL_PREFIX = "/api/cache"; | ||||||
| export const UPLOAD_URL = `${CACHE_URL_PREFIX}/upload`; | export const UPLOAD_URL = `${CACHE_URL_PREFIX}/upload`; | ||||||
|  |  | ||||||
| @@ -72,6 +74,7 @@ export enum ApiPath { | |||||||
|   ChatGLM = "/api/chatglm", |   ChatGLM = "/api/chatglm", | ||||||
|   DeepSeek = "/api/deepseek", |   DeepSeek = "/api/deepseek", | ||||||
|   SiliconFlow = "/api/siliconflow", |   SiliconFlow = "/api/siliconflow", | ||||||
|  |   PPIO = "/api/ppio", | ||||||
| } | } | ||||||
|  |  | ||||||
| export enum SlotID { | export enum SlotID { | ||||||
| @@ -130,6 +133,7 @@ export enum ServiceProvider { | |||||||
|   ChatGLM = "ChatGLM", |   ChatGLM = "ChatGLM", | ||||||
|   DeepSeek = "DeepSeek", |   DeepSeek = "DeepSeek", | ||||||
|   SiliconFlow = "SiliconFlow", |   SiliconFlow = "SiliconFlow", | ||||||
|  |   PPIO = "PPIO", | ||||||
| } | } | ||||||
|  |  | ||||||
| // Google API safety settings, see https://ai.google.dev/gemini-api/docs/safety-settings | // Google API safety settings, see https://ai.google.dev/gemini-api/docs/safety-settings | ||||||
| @@ -156,6 +160,7 @@ export enum ModelProvider { | |||||||
|   ChatGLM = "ChatGLM", |   ChatGLM = "ChatGLM", | ||||||
|   DeepSeek = "DeepSeek", |   DeepSeek = "DeepSeek", | ||||||
|   SiliconFlow = "SiliconFlow", |   SiliconFlow = "SiliconFlow", | ||||||
|  |   PPIO = "PPIO", | ||||||
| } | } | ||||||
|  |  | ||||||
| export const Stability = { | export const Stability = { | ||||||
| @@ -221,7 +226,12 @@ export const ByteDance = { | |||||||
|  |  | ||||||
| export const Alibaba = { | export const Alibaba = { | ||||||
|   ExampleEndpoint: ALIBABA_BASE_URL, |   ExampleEndpoint: ALIBABA_BASE_URL, | ||||||
|   ChatPath: "v1/services/aigc/text-generation/generation", |   ChatPath: (modelName: string) => { | ||||||
|  |     if (modelName.includes("vl") || modelName.includes("omni")) { | ||||||
|  |       return "v1/services/aigc/multimodal-generation/generation"; | ||||||
|  |     } | ||||||
|  |     return `v1/services/aigc/text-generation/generation`; | ||||||
|  |   }, | ||||||
| }; | }; | ||||||
|  |  | ||||||
| export const Tencent = { | export const Tencent = { | ||||||
| @@ -261,6 +271,12 @@ export const SiliconFlow = { | |||||||
|   ListModelPath: "v1/models?&sub_type=chat", |   ListModelPath: "v1/models?&sub_type=chat", | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  | export const PPIO = { | ||||||
|  |   ExampleEndpoint: PPIO_BASE_URL, | ||||||
|  |   ChatPath: "chat/completions", | ||||||
|  |   ListModelPath: "models", | ||||||
|  | }; | ||||||
|  |  | ||||||
| export const DEFAULT_INPUT_TEMPLATE = `{{input}}`; // input / time / model / lang | export const DEFAULT_INPUT_TEMPLATE = `{{input}}`; // input / time / model / lang | ||||||
| // export const DEFAULT_SYSTEM_TEMPLATE = ` | // export const DEFAULT_SYSTEM_TEMPLATE = ` | ||||||
| // You are ChatGPT, a large language model trained by {{ServiceProvider}}. | // You are ChatGPT, a large language model trained by {{ServiceProvider}}. | ||||||
| @@ -412,6 +428,14 @@ export const KnowledgeCutOffDate: Record<string, string> = { | |||||||
|   "gpt-4-turbo": "2023-12", |   "gpt-4-turbo": "2023-12", | ||||||
|   "gpt-4-turbo-2024-04-09": "2023-12", |   "gpt-4-turbo-2024-04-09": "2023-12", | ||||||
|   "gpt-4-turbo-preview": "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": "2023-10", | ||||||
|   "gpt-4o-2024-05-13": "2023-10", |   "gpt-4o-2024-05-13": "2023-10", | ||||||
|   "gpt-4o-2024-08-06": "2023-10", |   "gpt-4o-2024-08-06": "2023-10", | ||||||
| @@ -453,6 +477,7 @@ export const DEFAULT_TTS_VOICES = [ | |||||||
| export const VISION_MODEL_REGEXES = [ | export const VISION_MODEL_REGEXES = [ | ||||||
|   /vision/, |   /vision/, | ||||||
|   /gpt-4o/, |   /gpt-4o/, | ||||||
|  |   /gpt-4\.1/, | ||||||
|   /claude-3/, |   /claude-3/, | ||||||
|   /gemini-1\.5/, |   /gemini-1\.5/, | ||||||
|   /gemini-exp/, |   /gemini-exp/, | ||||||
| @@ -480,6 +505,14 @@ const openaiModels = [ | |||||||
|   "gpt-4-32k-0613", |   "gpt-4-32k-0613", | ||||||
|   "gpt-4-turbo", |   "gpt-4-turbo", | ||||||
|   "gpt-4-turbo-preview", |   "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", | ||||||
|   "gpt-4o-2024-05-13", |   "gpt-4o-2024-05-13", | ||||||
|   "gpt-4o-2024-08-06", |   "gpt-4o-2024-08-06", | ||||||
| @@ -570,6 +603,9 @@ const alibabaModes = [ | |||||||
|   "qwen-max-0403", |   "qwen-max-0403", | ||||||
|   "qwen-max-0107", |   "qwen-max-0107", | ||||||
|   "qwen-max-longcontext", |   "qwen-max-longcontext", | ||||||
|  |   "qwen-omni-turbo", | ||||||
|  |   "qwen-vl-plus", | ||||||
|  |   "qwen-vl-max", | ||||||
| ]; | ]; | ||||||
|  |  | ||||||
| const tencentModels = [ | const tencentModels = [ | ||||||
| @@ -642,6 +678,28 @@ const siliconflowModels = [ | |||||||
|   "Pro/deepseek-ai/DeepSeek-V3", |   "Pro/deepseek-ai/DeepSeek-V3", | ||||||
| ]; | ]; | ||||||
|  |  | ||||||
|  | const ppioModels = [ | ||||||
|  |   "deepseek/deepseek-r1/community", | ||||||
|  |   "deepseek/deepseek-v3/community", | ||||||
|  |   "deepseek/deepseek-r1", | ||||||
|  |   "deepseek/deepseek-v3", | ||||||
|  |   "deepseek/deepseek-r1-distill-llama-70b", | ||||||
|  |   "deepseek/deepseek-r1-distill-qwen-32b", | ||||||
|  |   "deepseek/deepseek-r1-distill-qwen-14b", | ||||||
|  |   "deepseek/deepseek-r1-distill-llama-8b", | ||||||
|  |   "qwen/qwen-2.5-72b-instruct", | ||||||
|  |   "qwen/qwen2.5-32b-instruct", | ||||||
|  |   "qwen/qwen-2-vl-72b-instruct", | ||||||
|  |   "meta-llama/llama-3.2-3b-instruct", | ||||||
|  |   "meta-llama/llama-3.1-70b-instruct", | ||||||
|  |   "meta-llama/llama-3.1-8b-instruct", | ||||||
|  |   "baichuan/baichuan2-13b-chat", | ||||||
|  |   "01-ai/yi-1.5-34b-chat", | ||||||
|  |   "01-ai/yi-1.5-9b-chat", | ||||||
|  |   "thudm/glm-4-9b-chat", | ||||||
|  |   "qwen/qwen-2-7b-instruct", | ||||||
|  | ]; | ||||||
|  |  | ||||||
| let seq = 1000; // 内置的模型序号生成器从1000开始 | let seq = 1000; // 内置的模型序号生成器从1000开始 | ||||||
| export const DEFAULT_MODELS = [ | export const DEFAULT_MODELS = [ | ||||||
|   ...openaiModels.map((name) => ({ |   ...openaiModels.map((name) => ({ | ||||||
| @@ -798,6 +856,17 @@ export const DEFAULT_MODELS = [ | |||||||
|       sorted: 14, |       sorted: 14, | ||||||
|     }, |     }, | ||||||
|   })), |   })), | ||||||
|  |   ...ppioModels.map((name) => ({ | ||||||
|  |     name, | ||||||
|  |     available: true, | ||||||
|  |     sorted: seq++, | ||||||
|  |     provider: { | ||||||
|  |       id: "ppio", | ||||||
|  |       providerName: "PPIO", | ||||||
|  |       providerType: "ppio", | ||||||
|  |       sorted: 15, | ||||||
|  |     }, | ||||||
|  |   })), | ||||||
| ] as const; | ] as const; | ||||||
|  |  | ||||||
| export const CHAT_PAGE_SIZE = 15; | export const CHAT_PAGE_SIZE = 15; | ||||||
|   | |||||||
| @@ -507,6 +507,17 @@ const cn = { | |||||||
|           SubTitle: "样例:", |           SubTitle: "样例:", | ||||||
|         }, |         }, | ||||||
|       }, |       }, | ||||||
|  |       PPIO: { | ||||||
|  |         ApiKey: { | ||||||
|  |           Title: "接口密钥", | ||||||
|  |           SubTitle: "使用自定义PPIO API Key", | ||||||
|  |           Placeholder: "PPIO API Key", | ||||||
|  |         }, | ||||||
|  |         Endpoint: { | ||||||
|  |           Title: "接口地址", | ||||||
|  |           SubTitle: "样例:", | ||||||
|  |         }, | ||||||
|  |       }, | ||||||
|       Stability: { |       Stability: { | ||||||
|         ApiKey: { |         ApiKey: { | ||||||
|           Title: "接口密钥", |           Title: "接口密钥", | ||||||
|   | |||||||
| @@ -491,6 +491,17 @@ const en: LocaleType = { | |||||||
|           SubTitle: "Example: ", |           SubTitle: "Example: ", | ||||||
|         }, |         }, | ||||||
|       }, |       }, | ||||||
|  |       PPIO: { | ||||||
|  |         ApiKey: { | ||||||
|  |           Title: "PPIO API Key", | ||||||
|  |           SubTitle: "Use a custom PPIO API Key", | ||||||
|  |           Placeholder: "PPIO API Key", | ||||||
|  |         }, | ||||||
|  |         Endpoint: { | ||||||
|  |           Title: "Endpoint Address", | ||||||
|  |           SubTitle: "Example: ", | ||||||
|  |         }, | ||||||
|  |       }, | ||||||
|       Stability: { |       Stability: { | ||||||
|         ApiKey: { |         ApiKey: { | ||||||
|           Title: "Stability API Key", |           Title: "Stability API Key", | ||||||
|   | |||||||
| @@ -17,6 +17,7 @@ import { | |||||||
|   XAI_BASE_URL, |   XAI_BASE_URL, | ||||||
|   CHATGLM_BASE_URL, |   CHATGLM_BASE_URL, | ||||||
|   SILICONFLOW_BASE_URL, |   SILICONFLOW_BASE_URL, | ||||||
|  |   PPIO_BASE_URL, | ||||||
| } from "../constant"; | } from "../constant"; | ||||||
| import { getHeaders } from "../client/api"; | import { getHeaders } from "../client/api"; | ||||||
| import { getClientConfig } from "../config/client"; | import { getClientConfig } from "../config/client"; | ||||||
| @@ -59,6 +60,8 @@ const DEFAULT_SILICONFLOW_URL = isApp | |||||||
|   ? SILICONFLOW_BASE_URL |   ? SILICONFLOW_BASE_URL | ||||||
|   : ApiPath.SiliconFlow; |   : ApiPath.SiliconFlow; | ||||||
|  |  | ||||||
|  | const DEFAULT_PPIO_URL = isApp ? PPIO_BASE_URL : ApiPath.PPIO; | ||||||
|  |  | ||||||
| const DEFAULT_ACCESS_STATE = { | const DEFAULT_ACCESS_STATE = { | ||||||
|   accessCode: "", |   accessCode: "", | ||||||
|   useCustomConfig: false, |   useCustomConfig: false, | ||||||
| @@ -132,6 +135,10 @@ const DEFAULT_ACCESS_STATE = { | |||||||
|   siliconflowUrl: DEFAULT_SILICONFLOW_URL, |   siliconflowUrl: DEFAULT_SILICONFLOW_URL, | ||||||
|   siliconflowApiKey: "", |   siliconflowApiKey: "", | ||||||
|  |  | ||||||
|  |   // ppio | ||||||
|  |   ppioUrl: DEFAULT_PPIO_URL, | ||||||
|  |   ppioApiKey: "", | ||||||
|  |  | ||||||
|   // server config |   // server config | ||||||
|   needCode: true, |   needCode: true, | ||||||
|   hideUserApiKey: false, |   hideUserApiKey: false, | ||||||
| @@ -219,6 +226,10 @@ export const useAccessStore = createPersistStore( | |||||||
|       return ensure(get(), ["siliconflowApiKey"]); |       return ensure(get(), ["siliconflowApiKey"]); | ||||||
|     }, |     }, | ||||||
|  |  | ||||||
|  |     isValidPPIO() { | ||||||
|  |       return ensure(get(), ["ppioApiKey"]); | ||||||
|  |     }, | ||||||
|  |  | ||||||
|     isAuthorized() { |     isAuthorized() { | ||||||
|       this.fetch(); |       this.fetch(); | ||||||
|  |  | ||||||
| @@ -238,6 +249,7 @@ export const useAccessStore = createPersistStore( | |||||||
|         this.isValidXAI() || |         this.isValidXAI() || | ||||||
|         this.isValidChatGLM() || |         this.isValidChatGLM() || | ||||||
|         this.isValidSiliconFlow() || |         this.isValidSiliconFlow() || | ||||||
|  |         this.isValidPPIO() || | ||||||
|         !this.enabledAccessControl() || |         !this.enabledAccessControl() || | ||||||
|         (this.enabledAccessControl() && ensure(get(), ["accessCode"])) |         (this.enabledAccessControl() && ensure(get(), ["accessCode"])) | ||||||
|       ); |       ); | ||||||
|   | |||||||
| @@ -714,6 +714,12 @@ export const useChatStore = createPersistStore( | |||||||
|             }, |             }, | ||||||
|             onFinish(message, responseRes) { |             onFinish(message, responseRes) { | ||||||
|               if (responseRes?.status === 200) { |               if (responseRes?.status === 200) { | ||||||
|  |                 // deal with <think> and </think> tags | ||||||
|  |                 if (message.startsWith("<think>")) { | ||||||
|  |                   message = message | ||||||
|  |                     .slice(message.indexOf("</think>") + 8) | ||||||
|  |                     .trim(); | ||||||
|  |                 } | ||||||
|                 get().updateTargetSession( |                 get().updateTargetSession( | ||||||
|                   session, |                   session, | ||||||
|                   (session) => |                   (session) => | ||||||
|   | |||||||
| @@ -3,7 +3,7 @@ import { | |||||||
|   UPLOAD_URL, |   UPLOAD_URL, | ||||||
|   REQUEST_TIMEOUT_MS, |   REQUEST_TIMEOUT_MS, | ||||||
| } from "@/app/constant"; | } from "@/app/constant"; | ||||||
| import { RequestMessage } from "@/app/client/api"; | import { MultimodalContent, RequestMessage } from "@/app/client/api"; | ||||||
| import Locale from "@/app/locales"; | import Locale from "@/app/locales"; | ||||||
| import { | import { | ||||||
|   EventStreamContentType, |   EventStreamContentType, | ||||||
| @@ -70,8 +70,9 @@ export function compressImage(file: Blob, maxSize: number): Promise<string> { | |||||||
|   }); |   }); | ||||||
| } | } | ||||||
|  |  | ||||||
| export async function preProcessImageContent( | export async function preProcessImageContentBase( | ||||||
|   content: RequestMessage["content"], |   content: RequestMessage["content"], | ||||||
|  |   transformImageUrl: (url: string) => Promise<{ [key: string]: any }>, | ||||||
| ) { | ) { | ||||||
|   if (typeof content === "string") { |   if (typeof content === "string") { | ||||||
|     return content; |     return content; | ||||||
| @@ -81,7 +82,7 @@ export async function preProcessImageContent( | |||||||
|     if (part?.type == "image_url" && part?.image_url?.url) { |     if (part?.type == "image_url" && part?.image_url?.url) { | ||||||
|       try { |       try { | ||||||
|         const url = await cacheImageToBase64Image(part?.image_url?.url); |         const url = await cacheImageToBase64Image(part?.image_url?.url); | ||||||
|         result.push({ type: part.type, image_url: { url } }); |         result.push(await transformImageUrl(url)); | ||||||
|       } catch (error) { |       } catch (error) { | ||||||
|         console.error("Error processing image URL:", error); |         console.error("Error processing image URL:", error); | ||||||
|       } |       } | ||||||
| @@ -92,6 +93,23 @@ export async function preProcessImageContent( | |||||||
|   return result; |   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> = {}; | const imageCaches: Record<string, string> = {}; | ||||||
| export function cacheImageToBase64Image(imageUrl: string) { | export function cacheImageToBase64Image(imageUrl: string) { | ||||||
|   if (imageUrl.includes(CACHE_URL_PREFIX)) { |   if (imageUrl.includes(CACHE_URL_PREFIX)) { | ||||||
| @@ -615,7 +633,8 @@ export function streamWithThink( | |||||||
|               if (chunk.content.includes("\n\n")) { |               if (chunk.content.includes("\n\n")) { | ||||||
|                 const lines = chunk.content.split("\n\n"); |                 const lines = chunk.content.split("\n\n"); | ||||||
|                 remainText += lines.join("\n\n> "); |                 remainText += lines.join("\n\n> "); | ||||||
|               } else { |               } else if (chunk.content != "\n") { | ||||||
|  |                 // deal with single newline after <think> tag | ||||||
|                 remainText += chunk.content; |                 remainText += chunk.content; | ||||||
|               } |               } | ||||||
|             } |             } | ||||||
|   | |||||||
| @@ -15,6 +15,8 @@ const config: Config = { | |||||||
|   moduleNameMapper: { |   moduleNameMapper: { | ||||||
|     "^@/(.*)$": "<rootDir>/$1", |     "^@/(.*)$": "<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 | // createJestConfig is exported this way to ensure that next/jest can load the Next.js config which is async | ||||||
|   | |||||||
| @@ -1,24 +1,22 @@ | |||||||
| // Learn more: https://github.com/testing-library/jest-dom | // Learn more: https://github.com/testing-library/jest-dom | ||||||
| import "@testing-library/jest-dom"; | import "@testing-library/jest-dom"; | ||||||
|  | import { jest } from "@jest/globals"; | ||||||
|  |  | ||||||
| global.fetch = jest.fn(() => | global.fetch = jest.fn(() => | ||||||
|   Promise.resolve({ |   Promise.resolve({ | ||||||
|     ok: true, |     ok: true, | ||||||
|     status: 200, |     status: 200, | ||||||
|     json: () => Promise.resolve({}), |     json: () => Promise.resolve([]), | ||||||
|     headers: new Headers(), |     headers: new Headers(), | ||||||
|     redirected: false, |     redirected: false, | ||||||
|     statusText: "OK", |     statusText: "OK", | ||||||
|     type: "basic", |     type: "basic", | ||||||
|     url: "", |     url: "", | ||||||
|     clone: function () { |  | ||||||
|       return this; |  | ||||||
|     }, |  | ||||||
|     body: null, |     body: null, | ||||||
|     bodyUsed: false, |     bodyUsed: false, | ||||||
|     arrayBuffer: () => Promise.resolve(new ArrayBuffer(0)), |     arrayBuffer: () => Promise.resolve(new ArrayBuffer(0)), | ||||||
|     blob: () => Promise.resolve(new Blob()), |     blob: () => Promise.resolve(new Blob()), | ||||||
|     formData: () => Promise.resolve(new FormData()), |     formData: () => Promise.resolve(new FormData()), | ||||||
|     text: () => Promise.resolve(""), |     text: () => Promise.resolve(""), | ||||||
|   }), |   } as Response), | ||||||
| ); | ); | ||||||
|   | |||||||
| @@ -17,8 +17,8 @@ | |||||||
|     "prompts": "node ./scripts/fetch-prompts.mjs", |     "prompts": "node ./scripts/fetch-prompts.mjs", | ||||||
|     "prepare": "husky install", |     "prepare": "husky install", | ||||||
|     "proxy-dev": "sh ./scripts/init-proxy.sh && proxychains -f ./scripts/proxychains.conf yarn dev", |     "proxy-dev": "sh ./scripts/init-proxy.sh && proxychains -f ./scripts/proxychains.conf yarn dev", | ||||||
|     "test": "jest --watch", |     "test": "node --no-warnings --experimental-vm-modules $(yarn bin jest) --watch", | ||||||
|     "test:ci": "jest --ci" |     "test:ci": "node --no-warnings --experimental-vm-modules $(yarn bin jest) --ci" | ||||||
|   }, |   }, | ||||||
|   "dependencies": { |   "dependencies": { | ||||||
|     "@fortaine/fetch-event-source": "^3.0.6", |     "@fortaine/fetch-event-source": "^3.0.6", | ||||||
|   | |||||||
| @@ -1,3 +1,4 @@ | |||||||
|  | import { jest } from "@jest/globals"; | ||||||
| import { isVisionModel } from "../app/utils"; | import { isVisionModel } from "../app/utils"; | ||||||
|  |  | ||||||
| describe("isVisionModel", () => { | describe("isVisionModel", () => { | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user