mirror of
				https://github.com/ChatGPTNextWeb/ChatGPT-Next-Web.git
				synced 2025-10-25 19:33:42 +08:00 
			
		
		
		
	Compare commits
	
		
			31 Commits
		
	
	
		
			6305-bugth
			...
			a9d0632bf5
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | a9d0632bf5 | ||
|  | c30ddfbb07 | ||
|  | a2f0149786 | ||
|  | 03d36f96ed | ||
|  | 705dffc664 | ||
|  | 02f7e6de98 | ||
|  | 843dc52efa | ||
|  | 3809375694 | ||
|  | 1b0de25986 | ||
|  | 865c45dd29 | ||
|  | 1f5d8e6d9c | ||
|  | c9ef6d58ed | ||
|  | 2d7229d2b8 | ||
|  | 11b37c15bd | ||
|  | 1d0038f17d | ||
|  | 619fa519c0 | ||
|  | c261ebc82c | ||
|  | f7c747c65f | ||
|  | 48469bd8ca | ||
|  | 5a5e887f2b | ||
|  | b6f5d75656 | ||
|  | 0d41a17ef6 | ||
|  | f7cde17919 | ||
|  | 570cbb34b6 | ||
|  | 7aa9ae0a3e | ||
|  | ad6666eeaf | ||
|  | a2c4e468a0 | ||
|  | 0a25a1a8cb | ||
|  | b709ee3983 | ||
|  | bc53c17a8c | ||
|  | 16c16887ae | 
							
								
								
									
										30
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										30
									
								
								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,8 +22,7 @@ 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) / [iOS APP](https://apps.apple.com/us/app/nextchat-ai/id6743085599) / [Web App Demo](https://app.nextchat.dev) / [Desktop App](https://github.com/Yidadaa/ChatGPT-Next-Web/releases) / [Enterprise Edition](#enterprise-edition)  | ||||||
| [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) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| [saas-url]: https://nextchat.club?utm_source=readme | [saas-url]: https://nextchat.club?utm_source=readme | ||||||
| @@ -41,29 +40,12 @@ English / [简体中文](./README_CN.md) | |||||||
|  |  | ||||||
| </div> | </div> | ||||||
|  |  | ||||||
| ## 👋 Hey, NextChat is going to develop a native app! | ## 🥳 Cheer for NextChat iOS Version Online! | ||||||
|  | > [👉 Click Here to Install Now](https://apps.apple.com/us/app/nextchat-ai/id6743085599) | ||||||
|  |  | ||||||
| > 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! | > [❤️ Source Code Coming Soon](https://github.com/ChatGPTNextWeb/NextChat-iOS) | ||||||
|  |  | ||||||
|  |  | ||||||
| ✨ 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 |  | ||||||
|   |  | ||||||
| <img src="https://github.com/user-attachments/assets/f3952210-3af1-4dc0-9b81-40eaa4847d9a"/> |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|   |   | ||||||
| ## 🫣 NextChat Support MCP  !  | ## 🫣 NextChat Support MCP  !  | ||||||
|   | |||||||
| @@ -40,6 +40,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[]; | ||||||
|   | |||||||
| @@ -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, | ||||||
|               }; |               }; | ||||||
|             } |             } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -56,7 +56,7 @@ export interface OpenAIListModelResponse { | |||||||
|  |  | ||||||
| export interface RequestPayload { | export interface RequestPayload { | ||||||
|   messages: { |   messages: { | ||||||
|     role: "system" | "user" | "assistant"; |     role: "developer" | "system" | "user" | "assistant"; | ||||||
|     content: string | MultimodalContent[]; |     content: string | MultimodalContent[]; | ||||||
|   }[]; |   }[]; | ||||||
|   stream?: boolean; |   stream?: boolean; | ||||||
| @@ -198,7 +198,8 @@ export class ChatGPTApi implements LLMApi { | |||||||
|     const isDalle3 = _isDalle3(options.config.model); |     const isDalle3 = _isDalle3(options.config.model); | ||||||
|     const isO1OrO3 = |     const isO1OrO3 = | ||||||
|       options.config.model.startsWith("o1") || |       options.config.model.startsWith("o1") || | ||||||
|       options.config.model.startsWith("o3"); |       options.config.model.startsWith("o3") || | ||||||
|  |       options.config.model.startsWith("o4-mini"); | ||||||
|     if (isDalle3) { |     if (isDalle3) { | ||||||
|       const prompt = getMessageTextContent( |       const prompt = getMessageTextContent( | ||||||
|         options.messages.slice(-1)?.pop() as any, |         options.messages.slice(-1)?.pop() as any, | ||||||
| @@ -237,13 +238,21 @@ export class ChatGPTApi implements LLMApi { | |||||||
|         // Please do not ask me why not send max_tokens, no reason, this param is just shit, I dont want to explain anymore. |         // Please do not ask me why not send max_tokens, no reason, this param is just shit, I dont want to explain anymore. | ||||||
|       }; |       }; | ||||||
|  |  | ||||||
|       // O1 使用 max_completion_tokens 控制token数 (https://platform.openai.com/docs/guides/reasoning#controlling-costs) |  | ||||||
|       if (isO1OrO3) { |       if (isO1OrO3) { | ||||||
|  |         // by default the o1/o3 models will not attempt to produce output that includes markdown formatting | ||||||
|  |         // manually add "Formatting re-enabled" developer message to encourage markdown inclusion in model responses | ||||||
|  |         // (https://learn.microsoft.com/en-us/azure/ai-services/openai/how-to/reasoning?tabs=python-secure#markdown-output) | ||||||
|  |         requestPayload["messages"].unshift({ | ||||||
|  |           role: "developer", | ||||||
|  |           content: "Formatting re-enabled", | ||||||
|  |         }); | ||||||
|  |  | ||||||
|  |         // o1/o3 uses max_completion_tokens to control the number of tokens (https://platform.openai.com/docs/guides/reasoning#controlling-costs) | ||||||
|         requestPayload["max_completion_tokens"] = modelConfig.max_tokens; |         requestPayload["max_completion_tokens"] = modelConfig.max_tokens; | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       // add max_tokens to vision model |       // add max_tokens to vision model | ||||||
|       if (visionModel) { |       if (visionModel && !isO1OrO3) { | ||||||
|         requestPayload["max_tokens"] = Math.max(modelConfig.max_tokens, 4000); |         requestPayload["max_tokens"] = Math.max(modelConfig.max_tokens, 4000); | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -221,7 +221,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 = { | ||||||
| @@ -412,6 +417,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 +466,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/, | ||||||
| @@ -464,6 +478,8 @@ export const VISION_MODEL_REGEXES = [ | |||||||
|   /^dall-e-3$/, // Matches exactly "dall-e-3" |   /^dall-e-3$/, // Matches exactly "dall-e-3" | ||||||
|   /glm-4v/, |   /glm-4v/, | ||||||
|   /vl/i, |   /vl/i, | ||||||
|  |   /o3/, | ||||||
|  |   /o4-mini/, | ||||||
| ]; | ]; | ||||||
|  |  | ||||||
| export const EXCLUDE_VISION_MODEL_REGEXES = [/claude-3-5-haiku-20241022/]; | export const EXCLUDE_VISION_MODEL_REGEXES = [/claude-3-5-haiku-20241022/]; | ||||||
| @@ -480,6 +496,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", | ||||||
| @@ -494,6 +518,8 @@ const openaiModels = [ | |||||||
|   "o1-mini", |   "o1-mini", | ||||||
|   "o1-preview", |   "o1-preview", | ||||||
|   "o3-mini", |   "o3-mini", | ||||||
|  |   "o3", | ||||||
|  |   "o4-mini", | ||||||
| ]; | ]; | ||||||
|  |  | ||||||
| const googleModels = [ | const googleModels = [ | ||||||
| @@ -520,6 +546,7 @@ const googleModels = [ | |||||||
|   "gemini-2.0-flash-thinking-exp-01-21", |   "gemini-2.0-flash-thinking-exp-01-21", | ||||||
|   "gemini-2.0-pro-exp", |   "gemini-2.0-pro-exp", | ||||||
|   "gemini-2.0-pro-exp-02-05", |   "gemini-2.0-pro-exp-02-05", | ||||||
|  |   "gemini-2.5-pro-preview-06-05", | ||||||
| ]; | ]; | ||||||
|  |  | ||||||
| const anthropicModels = [ | const anthropicModels = [ | ||||||
| @@ -570,6 +597,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 = [ | ||||||
| @@ -603,6 +633,18 @@ const xAIModes = [ | |||||||
|   "grok-2-vision-1212", |   "grok-2-vision-1212", | ||||||
|   "grok-2-vision", |   "grok-2-vision", | ||||||
|   "grok-2-vision-latest", |   "grok-2-vision-latest", | ||||||
|  |   "grok-3-mini-fast-beta", | ||||||
|  |   "grok-3-mini-fast", | ||||||
|  |   "grok-3-mini-fast-latest", | ||||||
|  |   "grok-3-mini-beta", | ||||||
|  |   "grok-3-mini", | ||||||
|  |   "grok-3-mini-latest", | ||||||
|  |   "grok-3-fast-beta", | ||||||
|  |   "grok-3-fast", | ||||||
|  |   "grok-3-fast-latest", | ||||||
|  |   "grok-3-beta", | ||||||
|  |   "grok-3", | ||||||
|  |   "grok-3-latest", | ||||||
| ]; | ]; | ||||||
|  |  | ||||||
| const chatglmModels = [ | const chatglmModels = [ | ||||||
|   | |||||||
| @@ -159,10 +159,10 @@ export const usePromptStore = createPersistStore( | |||||||
|       fetch(PROMPT_URL) |       fetch(PROMPT_URL) | ||||||
|         .then((res) => res.json()) |         .then((res) => res.json()) | ||||||
|         .then((res) => { |         .then((res) => { | ||||||
|           let fetchPrompts = [res.en, res.tw, res.cn]; |           const lang = getLang(); | ||||||
|           if (getLang() === "cn") { |           const fetchPrompts = [res[lang], res.en, res.tw, res.cn].filter( | ||||||
|             fetchPrompts = fetchPrompts.reverse(); |             Boolean, | ||||||
|           } |           ); | ||||||
|           const builtinPrompts = fetchPrompts.map((promptList: PromptList) => { |           const builtinPrompts = fetchPrompts.map((promptList: PromptList) => { | ||||||
|             return promptList.map( |             return promptList.map( | ||||||
|               ([title, content]) => |               ([title, content]) => | ||||||
| @@ -180,8 +180,9 @@ export const usePromptStore = createPersistStore( | |||||||
|           const allPromptsForSearch = builtinPrompts |           const allPromptsForSearch = builtinPrompts | ||||||
|             .reduce((pre, cur) => pre.concat(cur), []) |             .reduce((pre, cur) => pre.concat(cur), []) | ||||||
|             .filter((v) => !!v.title && !!v.content); |             .filter((v) => !!v.title && !!v.content); | ||||||
|           SearchService.count.builtin = |           SearchService.count.builtin = Object.values(res) | ||||||
|             res.en.length + res.cn.length + res.tw.length; |             .filter(Array.isArray) | ||||||
|  |             .reduce((total, promptList) => total + promptList.length, 0); | ||||||
|           SearchService.init(allPromptsForSearch, userPrompts); |           SearchService.init(allPromptsForSearch, userPrompts); | ||||||
|         }); |         }); | ||||||
|     }, |     }, | ||||||
|   | |||||||
| @@ -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)) { | ||||||
|   | |||||||
| @@ -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