@hopemyl619/deepseek 0.1.2 → 0.2.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/index.js +23 -52
- package/dist/index.js.map +1 -1
- package/package.json +3 -2
- package/src/deepseek-provider.ts +23 -52
- package/dist/index.d.ts +0 -184
package/dist/index.js
CHANGED
|
@@ -1,5 +1,27 @@
|
|
|
1
1
|
// src/deepseek-provider.ts
|
|
2
|
-
import {
|
|
2
|
+
import { createDeepSeek as createOfficialProvider } from "@ai-sdk/deepseek";
|
|
3
|
+
import { loadApiKey } from "@ai-sdk/provider-utils";
|
|
4
|
+
async function createProvider(options = {}) {
|
|
5
|
+
const officialProvider = createOfficialProvider({
|
|
6
|
+
apiKey: options.apiKey,
|
|
7
|
+
baseURL: options.baseURL,
|
|
8
|
+
headers: options.headers
|
|
9
|
+
});
|
|
10
|
+
const provider = {
|
|
11
|
+
id: "deepseek",
|
|
12
|
+
name: "DeepSeek AI",
|
|
13
|
+
async initialize() {
|
|
14
|
+
await loadApiKey({
|
|
15
|
+
apiKey: options.apiKey,
|
|
16
|
+
environmentVariableName: "DEEPSEEK_API_KEY",
|
|
17
|
+
description: "DeepSeek API key"
|
|
18
|
+
});
|
|
19
|
+
},
|
|
20
|
+
chat: officialProvider.chat
|
|
21
|
+
};
|
|
22
|
+
return provider;
|
|
23
|
+
}
|
|
24
|
+
var deepseek = createProvider();
|
|
3
25
|
|
|
4
26
|
// src/deepseek-chat-settings.ts
|
|
5
27
|
function validateChatSettings(settings = {}) {
|
|
@@ -523,57 +545,6 @@ var DeepSeekChatLanguageModel = class {
|
|
|
523
545
|
};
|
|
524
546
|
}
|
|
525
547
|
};
|
|
526
|
-
|
|
527
|
-
// src/deepseek-provider.ts
|
|
528
|
-
async function createProvider(options = {}) {
|
|
529
|
-
const baseURL = withoutTrailingSlash(
|
|
530
|
-
options.baseURL || "https://api.deepseek.com/v1"
|
|
531
|
-
) ?? "https://api.deepseek.com/v1";
|
|
532
|
-
const getHeaders = () => {
|
|
533
|
-
const apiKeyResult = loadApiKey({
|
|
534
|
-
apiKey: options.apiKey,
|
|
535
|
-
environmentVariableName: "DEEPSEEK_API_KEY",
|
|
536
|
-
description: "DeepSeek API key"
|
|
537
|
-
});
|
|
538
|
-
const apiKey = apiKeyResult ?? "";
|
|
539
|
-
const authHeader = "Bearer " + apiKey;
|
|
540
|
-
const result = {
|
|
541
|
-
Authorization: authHeader
|
|
542
|
-
};
|
|
543
|
-
if (options.headers) {
|
|
544
|
-
for (const [key, value] of Object.entries(options.headers)) {
|
|
545
|
-
if (typeof value === "string") {
|
|
546
|
-
result[key] = value;
|
|
547
|
-
}
|
|
548
|
-
}
|
|
549
|
-
}
|
|
550
|
-
return result;
|
|
551
|
-
};
|
|
552
|
-
const createChatModel = (modelId, settings = {}) => {
|
|
553
|
-
return new DeepSeekChatLanguageModel(modelId, settings, {
|
|
554
|
-
baseURL,
|
|
555
|
-
headers: getHeaders
|
|
556
|
-
});
|
|
557
|
-
};
|
|
558
|
-
const provider = {
|
|
559
|
-
id: "deepseek",
|
|
560
|
-
name: "DeepSeek AI",
|
|
561
|
-
async initialize() {
|
|
562
|
-
try {
|
|
563
|
-
await loadApiKey({
|
|
564
|
-
apiKey: options.apiKey,
|
|
565
|
-
environmentVariableName: "DEEPSEEK_API_KEY",
|
|
566
|
-
description: "DeepSeek API key"
|
|
567
|
-
});
|
|
568
|
-
} catch (error) {
|
|
569
|
-
throw new Error(`DeepSeek provider initialization failed: ${error}`);
|
|
570
|
-
}
|
|
571
|
-
},
|
|
572
|
-
chat: createChatModel
|
|
573
|
-
};
|
|
574
|
-
return provider;
|
|
575
|
-
}
|
|
576
|
-
var deepseek = createProvider();
|
|
577
548
|
export {
|
|
578
549
|
ChineseParamsPreprocessor,
|
|
579
550
|
DeepSeekChatLanguageModel,
|
package/dist/index.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/deepseek-provider.ts","../src/deepseek-chat-settings.ts","../src/utils/chinese-params-preprocessor.ts","../src/utils/request-transformer.ts","../src/utils/response-transformer.ts","../src/utils/stream-parser.ts","../src/deepseek-chat-language-model.ts"],"sourcesContent":["// DeepSeek Provider 工厂\n\nimport { loadApiKey, withoutTrailingSlash } from '@ai-sdk/provider-utils';\nimport { DeepSeekChatLanguageModel } from './deepseek-chat-language-model';\nimport type { DeepSeekProviderSettings, DeepSeekChatSettings } from './types';\n\nexport async function createProvider(options: DeepSeekProviderSettings = {}) {\n const baseURL = withoutTrailingSlash(\n options.baseURL || 'https://api.deepseek.com/v1'\n ) ?? 'https://api.deepseek.com/v1';\n\n const getHeaders = (): Record<string, string> => {\n const apiKeyResult = loadApiKey({\n apiKey: options.apiKey,\n environmentVariableName: 'DEEPSEEK_API_KEY',\n description: 'DeepSeek API key',\n });\n // loadApiKey 可能返回 undefined,使用空字符串作为默认值并确保类型为 string\n const apiKey = (apiKeyResult ?? '') as string;\n const authHeader: string = 'Bearer ' + apiKey;\n const result: Record<string, string> = {\n Authorization: authHeader,\n };\n if (options.headers) {\n for (const [key, value] of Object.entries(options.headers)) {\n if (typeof value === 'string') {\n result[key] = value;\n }\n }\n }\n return result;\n };\n\n const createChatModel = (\n modelId: string,\n settings: DeepSeekChatSettings = {}\n ) => {\n return new DeepSeekChatLanguageModel(modelId, settings, {\n baseURL,\n headers: getHeaders,\n });\n };\n\n const provider = {\n id: 'deepseek',\n name: 'DeepSeek AI',\n async initialize() {\n // 初始化逻辑:验证配置\n try {\n await loadApiKey({\n apiKey: options.apiKey,\n environmentVariableName: 'DEEPSEEK_API_KEY',\n description: 'DeepSeek API key',\n });\n } catch (error) {\n throw new Error(`DeepSeek provider initialization failed: ${error}`);\n }\n },\n chat: createChatModel,\n };\n\n return provider;\n}\n\n// 默认导出\nexport const deepseek = createProvider();","// DeepSeek Chat 模型配置\n\nimport type { DeepSeekChatSettings } from './types';\n\nexport function validateChatSettings(settings: DeepSeekChatSettings = {}): Required<DeepSeekChatSettings> {\n return {\n temperature: settings.temperature ?? 0.7,\n topP: settings.topP ?? 1.0,\n maxTokens: settings.maxTokens ?? 4096,\n presencePenalty: settings.presencePenalty ?? 0,\n frequencyPenalty: settings.frequencyPenalty ?? 0,\n };\n}","// 中文参数预处理器\n\nimport type { DeepSeekRequest } from '../types';\n\nexport class ChineseParamsPreprocessor {\n private cityMap: Record<string, string> = {\n // 中国城市\n '北京': 'Beijing',\n '上海': 'Shanghai',\n '广州': 'Guangzhou',\n '深圳': 'Shenzhen',\n '杭州': 'Hangzhou',\n '南京': 'Nanjing',\n '成都': 'Chengdu',\n '武汉': 'Wuhan',\n '西安': 'Xi\\'an',\n '重庆': 'Chongqing',\n '天津': 'Tianjin',\n '苏州': 'Suzhou',\n '长沙': 'Changsha',\n '青岛': 'Qingdao',\n '大连': 'Dalian',\n '厦门': 'Xiamen',\n '宁波': 'Ningbo',\n '无锡': 'Wuxi',\n '佛山': 'Foshan',\n '东莞': 'Dongguan',\n '合肥': 'Hefei',\n '郑州': 'Zhengzhou',\n '福州': 'Fuzhou',\n '济南': 'Jinan',\n '昆明': 'Kunming',\n '沈阳': 'Shenyang',\n '长春': 'Changchun',\n '哈尔滨': 'Harbin',\n '石家庄': 'Shijiazhuang',\n '太原': 'Taiyuan',\n '南昌': 'Nanchang',\n '南宁': 'Nanning',\n '贵阳': 'Guiyang',\n '兰州': 'Lanzhou',\n '海口': 'Haikou',\n '呼和浩特': 'Hohhot',\n '银川': 'Yinchuan',\n '西宁': 'Xining',\n '乌鲁木齐': 'Urumqi',\n '拉萨': 'Lhasa',\n // 国际城市\n '伦敦': 'London',\n '纽约': 'New York',\n '东京': 'Tokyo',\n '巴黎': 'Paris',\n '柏林': 'Berlin',\n '悉尼': 'Sydney',\n '洛杉矶': 'Los Angeles',\n '多伦多': 'Toronto',\n '新加坡': 'Singapore',\n '香港': 'Hong Kong',\n '台北': 'Taipei',\n '首尔': 'Seoul',\n '曼谷': 'Bangkok',\n '迪拜': 'Dubai',\n '莫斯科': 'Moscow',\n '旧金山': 'San Francisco',\n '波士顿': 'Boston',\n '芝加哥': 'Chicago',\n '华盛顿': 'Washington',\n '休斯顿': 'Houston',\n '费城': 'Philadelphia',\n '凤凰城': 'Phoenix',\n '圣迭戈': 'San Diego',\n '达拉斯': 'Dallas',\n '圣何塞': 'San Jose',\n '奥斯汀': 'Austin',\n '温哥华': 'Vancouver',\n '蒙特利尔': 'Montreal',\n '卡尔加里': 'Calgary',\n '渥太华': 'Ottawa',\n '魁北克': 'Quebec',\n '埃德蒙顿': 'Edmonton',\n '温尼伯': 'Winnipeg',\n '哈利法克斯': 'Halifax',\n '维多利亚': 'Victoria',\n };\n\n process(request: DeepSeekRequest): DeepSeekRequest {\n const processed = JSON.parse(JSON.stringify(request));\n\n // 处理消息内容\n if (processed.messages) {\n processed.messages = this.processMessages(processed.messages);\n }\n\n // 处理工具参数\n if (processed.tools) {\n processed.tools = this.processTools(processed.tools);\n }\n\n return processed;\n }\n\n private processMessages(messages: any[]): any[] {\n return messages.map(msg => {\n if (msg.content && typeof msg.content === 'string') {\n msg.content = this.replaceCityNames(msg.content);\n } else if (Array.isArray(msg.content)) {\n msg.content = msg.content.map((part: any) => {\n if (part.type === 'text' && part.text) {\n part.text = this.replaceCityNames(part.text);\n }\n return part;\n });\n }\n return msg;\n });\n }\n\n private processTools(tools: any[]): any[] {\n return tools.map(tool => {\n if (tool.function?.parameters?.properties) {\n this.processProperties(tool.function.parameters.properties);\n }\n if (tool.function?.description) {\n tool.function.description = this.replaceCityNames(tool.function.description);\n }\n return tool;\n });\n }\n\n private processProperties(properties: any): void {\n for (const key in properties) {\n if (properties[key].description) {\n properties[key].description = this.replaceCityNames(properties[key].description);\n }\n if (properties[key].example) {\n const exampleStr = JSON.stringify(properties[key].example);\n properties[key].example = JSON.parse(this.replaceCityNames(exampleStr));\n }\n }\n }\n\n private replaceCityNames(text: string): string {\n let result = text;\n for (const [chinese, english] of Object.entries(this.cityMap)) {\n // 使用正则表达式替换,避免替换部分匹配的词\n const regex = new RegExp(`\\\\b${chinese}\\\\b`, 'g');\n result = result.replace(regex, english);\n }\n return result;\n }\n\n // 添加自定义城市映射\n addCityMapping(chinese: string, english: string): void {\n this.cityMap[chinese] = english;\n }\n\n // 批量添加城市映射\n addCityMappings(mappings: Record<string, string>): void {\n Object.assign(this.cityMap, mappings);\n }\n}","// 请求转换器\n\nimport type {\n LanguageModelV1CallOptions,\n LanguageModelV1Prompt,\n} from '@ai-sdk/provider';\nimport type { DeepSeekRequest } from '../types';\nimport { ChineseParamsPreprocessor } from './chinese-params-preprocessor';\n\nexport class RequestTransformer {\n private preprocessor: ChineseParamsPreprocessor;\n\n constructor() {\n this.preprocessor = new ChineseParamsPreprocessor();\n }\n\n transform(modelId: string, options: LanguageModelV1CallOptions): DeepSeekRequest {\n const transformed: DeepSeekRequest = {\n model: modelId,\n messages: this.transformMessages(options.prompt),\n stream: false,\n };\n\n // 添加温度参数\n if (options.temperature !== undefined) {\n transformed.temperature = options.temperature;\n }\n\n // 添加 top_p 参数\n if (options.topP !== undefined) {\n transformed.top_p = options.topP;\n }\n\n // 添加 max_tokens 参数\n if (options.maxTokens !== undefined) {\n transformed.max_tokens = options.maxTokens;\n }\n\n // 添加 presence_penalty 参数\n if (options.presencePenalty !== undefined) {\n transformed.presence_penalty = options.presencePenalty;\n }\n\n // 添加 frequency_penalty 参数\n if (options.frequencyPenalty !== undefined) {\n transformed.frequency_penalty = options.frequencyPenalty;\n }\n\n // 添加工具配置\n const mode = options.mode;\n let toolsArray: Array<any> = [];\n let toolChoice: any;\n\n // 只有 'regular' 模式支持工具\n if (mode && mode.type === 'regular') {\n toolsArray = mode.tools ?? [];\n toolChoice = mode.toolChoice;\n }\n\n if (toolsArray.length > 0) {\n transformed.tools = this.transformTools(toolsArray);\n transformed.tool_choice = this.transformToolChoice(toolChoice);\n }\n\n // 预处理中文参数\n return this.preprocessor.process(transformed);\n }\n\n private transformMessages(messages: LanguageModelV1Prompt): Array<any> {\n return messages.map((msg) => {\n const transformed: any = {\n role: msg.role,\n };\n\n // 处理内容\n if (msg.content && Array.isArray(msg.content)) {\n const contentParts: Array<any> = [];\n let hasToolCalls = false;\n let toolCallsArray: Array<any> = [];\n\n for (const part of msg.content) {\n // 处理文本内容\n if (part.type === 'text') {\n contentParts.push({ type: 'text', text: part.text });\n }\n // 处理图像内容\n else if (part.type === 'image') {\n // V1 image part has 'image' property, convert to data for DeepSeek API\n const imageData = part.image;\n if (typeof imageData === 'string') {\n contentParts.push({ type: 'image_url', image_url: { url: imageData } });\n } else if (imageData instanceof URL) {\n contentParts.push({ type: 'image_url', image_url: { url: imageData.toString() } });\n }\n }\n // 处理工具调用 (V1 格式)\n else if (part.type === 'tool-call') {\n hasToolCalls = true;\n toolCallsArray.push({\n id: part.toolCallId,\n type: 'function',\n function: {\n name: part.toolName,\n arguments: typeof part.args === 'string' ? part.args : JSON.stringify(part.args),\n },\n });\n }\n // 处理工具结果 (V1 格式)\n else if (part.type === 'tool-result') {\n // Tool results in V1 don't get added to content directly\n // They are already part of the prompt structure\n }\n // 处理推理内容\n else if (part.type === 'reasoning' || part.type === 'redacted-reasoning') {\n // DeepSeek API doesn't support reasoning content directly\n // We'll add it as text if there's a text field\n if ('text' in part && part.text) {\n contentParts.push({ type: 'text', text: part.text });\n }\n }\n // 处理文件内容\n else if (part.type === 'file') {\n contentParts.push({\n type: 'text',\n text: `[File: ${part.filename || 'unnamed'}]`,\n });\n }\n // 其他未知类型\n else {\n contentParts.push(part);\n }\n }\n\n // 设置内容\n if (contentParts.length > 0) {\n transformed.content = contentParts;\n }\n\n // 添加工具调用\n if (hasToolCalls && toolCallsArray.length > 0) {\n transformed.tool_calls = toolCallsArray;\n }\n } else if (typeof msg.content === 'string') {\n // 系统消息等内容为字符串的情况\n transformed.content = msg.content;\n }\n\n return transformed;\n });\n }\n\n private transformTools(tools: Array<any>): Array<any> {\n return tools.map(tool => ({\n type: 'function',\n function: {\n name: tool.name,\n description: tool.description,\n parameters: tool.parameters,\n },\n }));\n }\n\n private transformToolChoice(toolChoice: any): 'auto' | 'none' | 'required' | string {\n if (toolChoice === undefined || toolChoice === 'auto') {\n return 'auto';\n }\n if (toolChoice === 'none') {\n return 'none';\n }\n if (toolChoice === 'required') {\n return 'required';\n }\n if (typeof toolChoice === 'object' && toolChoice.type === 'tool') {\n return JSON.stringify({\n type: 'function',\n function: {\n name: toolChoice.toolName,\n },\n });\n }\n return 'auto';\n }\n}","// 响应转换器 - V1 API 兼容版本\n\nimport type {\n LanguageModelV1,\n LanguageModelV1FinishReason,\n LanguageModelV1CallWarning,\n LanguageModelV1FunctionToolCall,\n} from '@ai-sdk/provider';\nimport type { DeepSeekResponse } from '../types';\n\n// V1 API 返回结果类型\nexport interface V1GenerateResult {\n text?: string;\n reasoning?: string;\n usage: {\n promptTokens: number;\n completionTokens: number;\n };\n finishReason: LanguageModelV1FinishReason;\n toolCalls?: LanguageModelV1FunctionToolCall[];\n rawCall: {\n rawPrompt: unknown;\n rawSettings: Record<string, unknown>;\n };\n warnings?: LanguageModelV1CallWarning[];\n}\n\nexport class ResponseTransformer {\n transform(apiResponse: DeepSeekResponse): V1GenerateResult {\n const choice = apiResponse.choices?.[0];\n const message = choice?.message || {};\n\n // 修复 finish_reason\n const finishReason = this.fixFinishReason(\n choice?.finish_reason,\n message.tool_calls\n );\n\n return {\n text: message.content || '',\n finishReason: finishReason as LanguageModelV1FinishReason,\n usage: {\n promptTokens: apiResponse.usage?.prompt_tokens || 0,\n completionTokens: apiResponse.usage?.completion_tokens || 0,\n },\n rawCall: {\n rawPrompt: '',\n rawSettings: {},\n },\n toolCalls: this.transformToolCalls(message.tool_calls),\n warnings: [],\n };\n }\n\n private fixFinishReason(\n originalFinishReason: string | undefined,\n toolCalls: any[] | undefined\n ): string {\n // 如果有 tool_calls 但 finish_reason 是 \"stop\",修复为 \"tool-calls\"\n if (toolCalls && toolCalls.length > 0 && originalFinishReason === 'stop') {\n return 'tool-calls';\n }\n\n // 映射其他 finish_reason 值\n const finishReasonMap: Record<string, string> = {\n 'stop': 'stop',\n 'length': 'length',\n 'content_filter': 'content-filter',\n 'tool_calls': 'tool-calls',\n 'function_call': 'tool-calls',\n };\n\n return finishReasonMap[originalFinishReason || ''] || 'stop';\n }\n\n private transformToolCalls(toolCalls: any[] | undefined): LanguageModelV1FunctionToolCall[] {\n if (!toolCalls || toolCalls.length === 0) {\n return [];\n }\n\n return toolCalls.map((tc: any) => ({\n toolCallType: 'function' as const,\n toolCallId: tc.id,\n toolName: tc.function.name,\n args: typeof tc.function.arguments === 'string'\n ? tc.function.arguments\n : JSON.stringify(tc.function.arguments),\n }));\n }\n}","// 流解析器\n\nimport type { LanguageModelV1StreamPart } from '@ai-sdk/provider';\nimport type { DeepSeekStreamResponse, DeepSeekUsage } from '../types';\n\ninterface DeepSeekStreamChunk {\n id: string;\n object: string;\n created: number;\n model: string;\n choices: Array<{\n index: number;\n delta: {\n role?: string;\n content?: string;\n tool_calls?: Array<{\n id: string;\n type: 'function';\n function: {\n name: string;\n arguments: string;\n };\n }>;\n };\n finish_reason?: string;\n }>;\n usage?: DeepSeekUsage;\n}\n\nexport class StreamParser {\n async *parse(body: ReadableStream<Uint8Array>): AsyncIterable<LanguageModelV1StreamPart> {\n const reader = body.getReader();\n const decoder = new TextDecoder('utf-8');\n let buffer = '';\n let partialJson = '';\n\n try {\n while (true) {\n const { done, value } = await reader.read();\n if (done) break;\n\n buffer += decoder.decode(value, { stream: true });\n const lines = buffer.split('\\n');\n buffer = lines.pop() || '';\n\n for (const line of lines) {\n if (line.trim() === '') continue;\n if (line.startsWith(':')) continue; // SSE 注释\n\n if (line.startsWith('data: ')) {\n const data = line.slice(6).trim();\n if (data === '[DONE]') {\n yield* this.processPartialJson(partialJson);\n partialJson = '';\n return;\n }\n\n try {\n // 尝试解析 JSON\n const parsed = JSON.parse(data);\n yield* this.processPartialJson(partialJson);\n partialJson = '';\n yield* this.processChunk(parsed);\n } catch (e) {\n // JSON 解析失败,可能是部分数据\n partialJson += data;\n }\n }\n }\n }\n\n // 处理剩余的部分 JSON\n yield* this.processPartialJson(partialJson);\n } finally {\n reader.releaseLock();\n }\n }\n\n private async *processPartialJson(partialJson: string): AsyncIterable<LanguageModelV1StreamPart> {\n if (!partialJson.trim()) return;\n\n try {\n const parsed = JSON.parse(partialJson);\n yield* this.processChunk(parsed);\n } catch (e) {\n // 仍然无法解析,跳过\n console.warn('Failed to parse partial JSON:', partialJson);\n }\n }\n\n private async *processChunk(chunk: DeepSeekStreamChunk): AsyncIterable<LanguageModelV1StreamPart> {\n const delta = chunk.choices?.[0]?.delta;\n\n // 工具调用增量 - JSON in content 格式\n if (delta?.content) {\n const toolCallData = this.tryParseToolCall(delta.content);\n if (toolCallData) {\n yield {\n type: 'tool-call-delta',\n toolCallType: 'function',\n toolCallId: toolCallData.id || `tool-${Date.now()}-${Math.random().toString(36).substr(2, 9)}`,\n toolName: toolCallData.name,\n argsTextDelta: JSON.stringify(toolCallData.arguments || {}),\n } as LanguageModelV1StreamPart;\n } else {\n // 普通文本增量\n yield {\n type: 'text-delta',\n textDelta: delta.content,\n } as LanguageModelV1StreamPart;\n }\n }\n\n // 工具调用增量 - 标准 tool_calls 格式\n if (delta?.tool_calls && delta.tool_calls.length > 0) {\n for (const toolCall of delta.tool_calls) {\n yield {\n type: 'tool-call-delta',\n toolCallType: 'function',\n toolCallId: toolCall.id,\n toolName: toolCall.function?.name || '',\n argsTextDelta: toolCall.function?.arguments || '',\n } as LanguageModelV1StreamPart;\n }\n }\n\n // 完成信号\n const finishReason = chunk.choices?.[0]?.finish_reason;\n if (finishReason) {\n // V1 API: usage 在 finish 类型的 part 中\n const usage = chunk.usage\n ? {\n promptTokens: chunk.usage.prompt_tokens,\n completionTokens: chunk.usage.completion_tokens,\n }\n : { promptTokens: 0, completionTokens: 0 };\n\n yield {\n type: 'finish',\n finishReason: this.mapFinishReason(finishReason),\n usage,\n } as LanguageModelV1StreamPart;\n }\n }\n\n private mapFinishReason(reason: string): 'stop' | 'length' | 'content-filter' | 'tool-calls' | 'error' | 'other' | 'unknown' {\n const reasonMap: Record<string, 'stop' | 'length' | 'content-filter' | 'tool-calls' | 'error' | 'other' | 'unknown'> = {\n 'stop': 'stop',\n 'length': 'length',\n 'content_filter': 'content-filter',\n 'tool_calls': 'tool-calls',\n 'function_call': 'tool-calls',\n 'error': 'error',\n };\n return reasonMap[reason] || 'unknown';\n }\n\n private tryParseToolCall(content: string): { id?: string; name: string; arguments: Record<string, unknown> } | null {\n try {\n const parsed = JSON.parse(content);\n // 检查是否是工具调用格式: { name: \"...\", arguments: {...} }\n if (parsed && typeof parsed.name === 'string' && (parsed.arguments === undefined || typeof parsed.arguments === 'object')) {\n return {\n id: parsed.id,\n name: parsed.name,\n arguments: parsed.arguments || {},\n };\n }\n } catch {\n // JSON 解析失败,不是工具调用\n }\n return null;\n }\n}\n\n\n","// DeepSeek Chat 模型实现\n\nimport type {\n LanguageModelV1,\n LanguageModelV1CallOptions,\n} from '@ai-sdk/provider';\nimport type { DeepSeekChatSettings, DeepSeekResponse } from './types';\nimport { validateChatSettings } from './deepseek-chat-settings';\nimport { RequestTransformer } from './utils/request-transformer';\nimport { ResponseTransformer } from './utils/response-transformer';\nimport { StreamParser } from './utils/stream-parser';\n\nexport class DeepSeekChatLanguageModel implements LanguageModelV1 {\n readonly specificationVersion = 'v1' as const;\n readonly provider = 'deepseek' as const;\n readonly modelId: string;\n readonly settings: Required<DeepSeekChatSettings>;\n readonly defaultObjectGenerationMode: 'json' | 'tool' | undefined = undefined;\n\n private baseURL: string;\n private headers: () => Record<string, string>;\n private requestTransformer: RequestTransformer;\n private responseTransformer: ResponseTransformer;\n private streamParser: StreamParser;\n\n constructor(\n modelId: string,\n settings: DeepSeekChatSettings = {},\n config: {\n baseURL: string;\n headers: () => Record<string, string>;\n }\n ) {\n this.modelId = modelId;\n this.settings = validateChatSettings(settings);\n this.baseURL = config.baseURL;\n this.headers = config.headers;\n this.requestTransformer = new RequestTransformer();\n this.responseTransformer = new ResponseTransformer();\n this.streamParser = new StreamParser();\n }\n\n async doGenerate(options: LanguageModelV1CallOptions) {\n const requestBody = this.requestTransformer.transform(this.modelId, options);\n\n const response = await fetch(`${this.baseURL}/chat/completions`, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n ...this.headers(),\n },\n body: JSON.stringify(requestBody),\n });\n\n if (!response.ok) {\n const error = await response.text();\n throw new Error(`DeepSeek API error: ${response.status} - ${error}`);\n }\n\n const apiResponse = (await response.json()) as DeepSeekResponse;\n return this.responseTransformer.transform(apiResponse);\n }\n\n async doStream(options: LanguageModelV1CallOptions) {\n const requestBody = {\n ...this.requestTransformer.transform(this.modelId, options),\n stream: true,\n };\n\n const response = await fetch(`${this.baseURL}/chat/completions`, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n 'Accept': 'text/event-stream',\n ...this.headers(),\n },\n body: JSON.stringify(requestBody),\n });\n\n if (!response.ok) {\n const error = await response.text();\n throw new Error(`DeepSeek API error: ${response.status} - ${error}`);\n }\n\n if (!response.body) {\n throw new Error('Response body is null');\n }\n\n return {\n stream: this.streamParser.parse(response.body) as any,\n rawCall: {\n rawPrompt: requestBody as unknown,\n rawSettings: {},\n },\n warnings: [],\n };\n }\n}"],"mappings":";AAEA,SAAS,YAAY,4BAA4B;;;ACE1C,SAAS,qBAAqB,WAAiC,CAAC,GAAmC;AACxG,SAAO;AAAA,IACL,aAAa,SAAS,eAAe;AAAA,IACrC,MAAM,SAAS,QAAQ;AAAA,IACvB,WAAW,SAAS,aAAa;AAAA,IACjC,iBAAiB,SAAS,mBAAmB;AAAA,IAC7C,kBAAkB,SAAS,oBAAoB;AAAA,EACjD;AACF;;;ACRO,IAAM,4BAAN,MAAgC;AAAA,EAC7B,UAAkC;AAAA;AAAA,IAExC,gBAAM;AAAA,IACN,gBAAM;AAAA,IACN,gBAAM;AAAA,IACN,gBAAM;AAAA,IACN,gBAAM;AAAA,IACN,gBAAM;AAAA,IACN,gBAAM;AAAA,IACN,gBAAM;AAAA,IACN,gBAAM;AAAA,IACN,gBAAM;AAAA,IACN,gBAAM;AAAA,IACN,gBAAM;AAAA,IACN,gBAAM;AAAA,IACN,gBAAM;AAAA,IACN,gBAAM;AAAA,IACN,gBAAM;AAAA,IACN,gBAAM;AAAA,IACN,gBAAM;AAAA,IACN,gBAAM;AAAA,IACN,gBAAM;AAAA,IACN,gBAAM;AAAA,IACN,gBAAM;AAAA,IACN,gBAAM;AAAA,IACN,gBAAM;AAAA,IACN,gBAAM;AAAA,IACN,gBAAM;AAAA,IACN,gBAAM;AAAA,IACN,sBAAO;AAAA,IACP,sBAAO;AAAA,IACP,gBAAM;AAAA,IACN,gBAAM;AAAA,IACN,gBAAM;AAAA,IACN,gBAAM;AAAA,IACN,gBAAM;AAAA,IACN,gBAAM;AAAA,IACN,4BAAQ;AAAA,IACR,gBAAM;AAAA,IACN,gBAAM;AAAA,IACN,4BAAQ;AAAA,IACR,gBAAM;AAAA;AAAA,IAEN,gBAAM;AAAA,IACN,gBAAM;AAAA,IACN,gBAAM;AAAA,IACN,gBAAM;AAAA,IACN,gBAAM;AAAA,IACN,gBAAM;AAAA,IACN,sBAAO;AAAA,IACP,sBAAO;AAAA,IACP,sBAAO;AAAA,IACP,gBAAM;AAAA,IACN,gBAAM;AAAA,IACN,gBAAM;AAAA,IACN,gBAAM;AAAA,IACN,gBAAM;AAAA,IACN,sBAAO;AAAA,IACP,sBAAO;AAAA,IACP,sBAAO;AAAA,IACP,sBAAO;AAAA,IACP,sBAAO;AAAA,IACP,sBAAO;AAAA,IACP,gBAAM;AAAA,IACN,sBAAO;AAAA,IACP,sBAAO;AAAA,IACP,sBAAO;AAAA,IACP,sBAAO;AAAA,IACP,sBAAO;AAAA,IACP,sBAAO;AAAA,IACP,4BAAQ;AAAA,IACR,4BAAQ;AAAA,IACR,sBAAO;AAAA,IACP,sBAAO;AAAA,IACP,4BAAQ;AAAA,IACR,sBAAO;AAAA,IACP,kCAAS;AAAA,IACT,4BAAQ;AAAA,EACV;AAAA,EAEA,QAAQ,SAA2C;AACjD,UAAM,YAAY,KAAK,MAAM,KAAK,UAAU,OAAO,CAAC;AAGpD,QAAI,UAAU,UAAU;AACtB,gBAAU,WAAW,KAAK,gBAAgB,UAAU,QAAQ;AAAA,IAC9D;AAGA,QAAI,UAAU,OAAO;AACnB,gBAAU,QAAQ,KAAK,aAAa,UAAU,KAAK;AAAA,IACrD;AAEA,WAAO;AAAA,EACT;AAAA,EAEQ,gBAAgB,UAAwB;AAC9C,WAAO,SAAS,IAAI,SAAO;AACzB,UAAI,IAAI,WAAW,OAAO,IAAI,YAAY,UAAU;AAClD,YAAI,UAAU,KAAK,iBAAiB,IAAI,OAAO;AAAA,MACjD,WAAW,MAAM,QAAQ,IAAI,OAAO,GAAG;AACrC,YAAI,UAAU,IAAI,QAAQ,IAAI,CAAC,SAAc;AAC3C,cAAI,KAAK,SAAS,UAAU,KAAK,MAAM;AACrC,iBAAK,OAAO,KAAK,iBAAiB,KAAK,IAAI;AAAA,UAC7C;AACA,iBAAO;AAAA,QACT,CAAC;AAAA,MACH;AACA,aAAO;AAAA,IACT,CAAC;AAAA,EACH;AAAA,EAEQ,aAAa,OAAqB;AACxC,WAAO,MAAM,IAAI,UAAQ;AACvB,UAAI,KAAK,UAAU,YAAY,YAAY;AACzC,aAAK,kBAAkB,KAAK,SAAS,WAAW,UAAU;AAAA,MAC5D;AACA,UAAI,KAAK,UAAU,aAAa;AAC9B,aAAK,SAAS,cAAc,KAAK,iBAAiB,KAAK,SAAS,WAAW;AAAA,MAC7E;AACA,aAAO;AAAA,IACT,CAAC;AAAA,EACH;AAAA,EAEQ,kBAAkB,YAAuB;AAC/C,eAAW,OAAO,YAAY;AAC5B,UAAI,WAAW,GAAG,EAAE,aAAa;AAC/B,mBAAW,GAAG,EAAE,cAAc,KAAK,iBAAiB,WAAW,GAAG,EAAE,WAAW;AAAA,MACjF;AACA,UAAI,WAAW,GAAG,EAAE,SAAS;AAC3B,cAAM,aAAa,KAAK,UAAU,WAAW,GAAG,EAAE,OAAO;AACzD,mBAAW,GAAG,EAAE,UAAU,KAAK,MAAM,KAAK,iBAAiB,UAAU,CAAC;AAAA,MACxE;AAAA,IACF;AAAA,EACF;AAAA,EAEQ,iBAAiB,MAAsB;AAC7C,QAAI,SAAS;AACb,eAAW,CAAC,SAAS,OAAO,KAAK,OAAO,QAAQ,KAAK,OAAO,GAAG;AAE7D,YAAM,QAAQ,IAAI,OAAO,MAAM,OAAO,OAAO,GAAG;AAChD,eAAS,OAAO,QAAQ,OAAO,OAAO;AAAA,IACxC;AACA,WAAO;AAAA,EACT;AAAA;AAAA,EAGA,eAAe,SAAiB,SAAuB;AACrD,SAAK,QAAQ,OAAO,IAAI;AAAA,EAC1B;AAAA;AAAA,EAGA,gBAAgB,UAAwC;AACtD,WAAO,OAAO,KAAK,SAAS,QAAQ;AAAA,EACtC;AACF;;;ACvJO,IAAM,qBAAN,MAAyB;AAAA,EACtB;AAAA,EAER,cAAc;AACZ,SAAK,eAAe,IAAI,0BAA0B;AAAA,EACpD;AAAA,EAEA,UAAU,SAAiB,SAAsD;AAC/E,UAAM,cAA+B;AAAA,MACnC,OAAO;AAAA,MACP,UAAU,KAAK,kBAAkB,QAAQ,MAAM;AAAA,MAC/C,QAAQ;AAAA,IACV;AAGA,QAAI,QAAQ,gBAAgB,QAAW;AACrC,kBAAY,cAAc,QAAQ;AAAA,IACpC;AAGA,QAAI,QAAQ,SAAS,QAAW;AAC9B,kBAAY,QAAQ,QAAQ;AAAA,IAC9B;AAGA,QAAI,QAAQ,cAAc,QAAW;AACnC,kBAAY,aAAa,QAAQ;AAAA,IACnC;AAGA,QAAI,QAAQ,oBAAoB,QAAW;AACzC,kBAAY,mBAAmB,QAAQ;AAAA,IACzC;AAGA,QAAI,QAAQ,qBAAqB,QAAW;AAC1C,kBAAY,oBAAoB,QAAQ;AAAA,IAC1C;AAGA,UAAM,OAAO,QAAQ;AACrB,QAAI,aAAyB,CAAC;AAC9B,QAAI;AAGJ,QAAI,QAAQ,KAAK,SAAS,WAAW;AACnC,mBAAa,KAAK,SAAS,CAAC;AAC5B,mBAAa,KAAK;AAAA,IACpB;AAEA,QAAI,WAAW,SAAS,GAAG;AACzB,kBAAY,QAAQ,KAAK,eAAe,UAAU;AAClD,kBAAY,cAAc,KAAK,oBAAoB,UAAU;AAAA,IAC/D;AAGA,WAAO,KAAK,aAAa,QAAQ,WAAW;AAAA,EAC9C;AAAA,EAEQ,kBAAkB,UAA6C;AACrE,WAAO,SAAS,IAAI,CAAC,QAAQ;AAC3B,YAAM,cAAmB;AAAA,QACvB,MAAM,IAAI;AAAA,MACZ;AAGA,UAAI,IAAI,WAAW,MAAM,QAAQ,IAAI,OAAO,GAAG;AAC7C,cAAM,eAA2B,CAAC;AAClC,YAAI,eAAe;AACnB,YAAI,iBAA6B,CAAC;AAElC,mBAAW,QAAQ,IAAI,SAAS;AAE9B,cAAI,KAAK,SAAS,QAAQ;AACxB,yBAAa,KAAK,EAAE,MAAM,QAAQ,MAAM,KAAK,KAAK,CAAC;AAAA,UACrD,WAES,KAAK,SAAS,SAAS;AAE9B,kBAAM,YAAY,KAAK;AACvB,gBAAI,OAAO,cAAc,UAAU;AACjC,2BAAa,KAAK,EAAE,MAAM,aAAa,WAAW,EAAE,KAAK,UAAU,EAAE,CAAC;AAAA,YACxE,WAAW,qBAAqB,KAAK;AACnC,2BAAa,KAAK,EAAE,MAAM,aAAa,WAAW,EAAE,KAAK,UAAU,SAAS,EAAE,EAAE,CAAC;AAAA,YACnF;AAAA,UACF,WAES,KAAK,SAAS,aAAa;AAClC,2BAAe;AACf,2BAAe,KAAK;AAAA,cAClB,IAAI,KAAK;AAAA,cACT,MAAM;AAAA,cACN,UAAU;AAAA,gBACR,MAAM,KAAK;AAAA,gBACX,WAAW,OAAO,KAAK,SAAS,WAAW,KAAK,OAAO,KAAK,UAAU,KAAK,IAAI;AAAA,cACjF;AAAA,YACF,CAAC;AAAA,UACH,WAES,KAAK,SAAS,eAAe;AAAA,UAGtC,WAES,KAAK,SAAS,eAAe,KAAK,SAAS,sBAAsB;AAGxE,gBAAI,UAAU,QAAQ,KAAK,MAAM;AAC/B,2BAAa,KAAK,EAAE,MAAM,QAAQ,MAAM,KAAK,KAAK,CAAC;AAAA,YACrD;AAAA,UACF,WAES,KAAK,SAAS,QAAQ;AAC7B,yBAAa,KAAK;AAAA,cAChB,MAAM;AAAA,cACN,MAAM,UAAU,KAAK,YAAY,SAAS;AAAA,YAC5C,CAAC;AAAA,UACH,OAEK;AACH,yBAAa,KAAK,IAAI;AAAA,UACxB;AAAA,QACF;AAGA,YAAI,aAAa,SAAS,GAAG;AAC3B,sBAAY,UAAU;AAAA,QACxB;AAGA,YAAI,gBAAgB,eAAe,SAAS,GAAG;AAC7C,sBAAY,aAAa;AAAA,QAC3B;AAAA,MACF,WAAW,OAAO,IAAI,YAAY,UAAU;AAE1C,oBAAY,UAAU,IAAI;AAAA,MAC5B;AAEA,aAAO;AAAA,IACT,CAAC;AAAA,EACH;AAAA,EAEQ,eAAe,OAA+B;AACpD,WAAO,MAAM,IAAI,WAAS;AAAA,MACxB,MAAM;AAAA,MACN,UAAU;AAAA,QACR,MAAM,KAAK;AAAA,QACX,aAAa,KAAK;AAAA,QAClB,YAAY,KAAK;AAAA,MACnB;AAAA,IACF,EAAE;AAAA,EACJ;AAAA,EAEQ,oBAAoB,YAAwD;AAClF,QAAI,eAAe,UAAa,eAAe,QAAQ;AACrD,aAAO;AAAA,IACT;AACA,QAAI,eAAe,QAAQ;AACzB,aAAO;AAAA,IACT;AACA,QAAI,eAAe,YAAY;AAC7B,aAAO;AAAA,IACT;AACA,QAAI,OAAO,eAAe,YAAY,WAAW,SAAS,QAAQ;AAChE,aAAO,KAAK,UAAU;AAAA,QACpB,MAAM;AAAA,QACN,UAAU;AAAA,UACR,MAAM,WAAW;AAAA,QACnB;AAAA,MACF,CAAC;AAAA,IACH;AACA,WAAO;AAAA,EACT;AACF;;;AC3JO,IAAM,sBAAN,MAA0B;AAAA,EAC/B,UAAU,aAAiD;AACzD,UAAM,SAAS,YAAY,UAAU,CAAC;AACtC,UAAM,UAAU,QAAQ,WAAW,CAAC;AAGpC,UAAM,eAAe,KAAK;AAAA,MACxB,QAAQ;AAAA,MACR,QAAQ;AAAA,IACV;AAEA,WAAO;AAAA,MACL,MAAM,QAAQ,WAAW;AAAA,MACzB;AAAA,MACA,OAAO;AAAA,QACL,cAAc,YAAY,OAAO,iBAAiB;AAAA,QAClD,kBAAkB,YAAY,OAAO,qBAAqB;AAAA,MAC5D;AAAA,MACA,SAAS;AAAA,QACP,WAAW;AAAA,QACX,aAAa,CAAC;AAAA,MAChB;AAAA,MACA,WAAW,KAAK,mBAAmB,QAAQ,UAAU;AAAA,MACrD,UAAU,CAAC;AAAA,IACb;AAAA,EACF;AAAA,EAEQ,gBACN,sBACA,WACQ;AAER,QAAI,aAAa,UAAU,SAAS,KAAK,yBAAyB,QAAQ;AACxE,aAAO;AAAA,IACT;AAGA,UAAM,kBAA0C;AAAA,MAC9C,QAAQ;AAAA,MACR,UAAU;AAAA,MACV,kBAAkB;AAAA,MAClB,cAAc;AAAA,MACd,iBAAiB;AAAA,IACnB;AAEA,WAAO,gBAAgB,wBAAwB,EAAE,KAAK;AAAA,EACxD;AAAA,EAEQ,mBAAmB,WAAiE;AAC1F,QAAI,CAAC,aAAa,UAAU,WAAW,GAAG;AACxC,aAAO,CAAC;AAAA,IACV;AAEA,WAAO,UAAU,IAAI,CAAC,QAAa;AAAA,MACjC,cAAc;AAAA,MACd,YAAY,GAAG;AAAA,MACf,UAAU,GAAG,SAAS;AAAA,MACtB,MAAM,OAAO,GAAG,SAAS,cAAc,WACnC,GAAG,SAAS,YACZ,KAAK,UAAU,GAAG,SAAS,SAAS;AAAA,IAC1C,EAAE;AAAA,EACJ;AACF;;;AC5DO,IAAM,eAAN,MAAmB;AAAA,EACxB,OAAO,MAAM,MAA4E;AACvF,UAAM,SAAS,KAAK,UAAU;AAC9B,UAAM,UAAU,IAAI,YAAY,OAAO;AACvC,QAAI,SAAS;AACb,QAAI,cAAc;AAElB,QAAI;AACF,aAAO,MAAM;AACX,cAAM,EAAE,MAAM,MAAM,IAAI,MAAM,OAAO,KAAK;AAC1C,YAAI,KAAM;AAEV,kBAAU,QAAQ,OAAO,OAAO,EAAE,QAAQ,KAAK,CAAC;AAChD,cAAM,QAAQ,OAAO,MAAM,IAAI;AAC/B,iBAAS,MAAM,IAAI,KAAK;AAExB,mBAAW,QAAQ,OAAO;AACxB,cAAI,KAAK,KAAK,MAAM,GAAI;AACxB,cAAI,KAAK,WAAW,GAAG,EAAG;AAE1B,cAAI,KAAK,WAAW,QAAQ,GAAG;AAC7B,kBAAM,OAAO,KAAK,MAAM,CAAC,EAAE,KAAK;AAChC,gBAAI,SAAS,UAAU;AACrB,qBAAO,KAAK,mBAAmB,WAAW;AAC1C,4BAAc;AACd;AAAA,YACF;AAEA,gBAAI;AAEF,oBAAM,SAAS,KAAK,MAAM,IAAI;AAC9B,qBAAO,KAAK,mBAAmB,WAAW;AAC1C,4BAAc;AACd,qBAAO,KAAK,aAAa,MAAM;AAAA,YACjC,SAAS,GAAG;AAEV,6BAAe;AAAA,YACjB;AAAA,UACF;AAAA,QACF;AAAA,MACF;AAGA,aAAO,KAAK,mBAAmB,WAAW;AAAA,IAC5C,UAAE;AACA,aAAO,YAAY;AAAA,IACrB;AAAA,EACF;AAAA,EAEA,OAAe,mBAAmB,aAA+D;AAC/F,QAAI,CAAC,YAAY,KAAK,EAAG;AAEzB,QAAI;AACF,YAAM,SAAS,KAAK,MAAM,WAAW;AACrC,aAAO,KAAK,aAAa,MAAM;AAAA,IACjC,SAAS,GAAG;AAEV,cAAQ,KAAK,iCAAiC,WAAW;AAAA,IAC3D;AAAA,EACF;AAAA,EAEA,OAAe,aAAa,OAAsE;AAChG,UAAM,QAAQ,MAAM,UAAU,CAAC,GAAG;AAGlC,QAAI,OAAO,SAAS;AAClB,YAAM,eAAe,KAAK,iBAAiB,MAAM,OAAO;AACxD,UAAI,cAAc;AAChB,cAAM;AAAA,UACJ,MAAM;AAAA,UACN,cAAc;AAAA,UACd,YAAY,aAAa,MAAM,QAAQ,KAAK,IAAI,CAAC,IAAI,KAAK,OAAO,EAAE,SAAS,EAAE,EAAE,OAAO,GAAG,CAAC,CAAC;AAAA,UAC5F,UAAU,aAAa;AAAA,UACvB,eAAe,KAAK,UAAU,aAAa,aAAa,CAAC,CAAC;AAAA,QAC5D;AAAA,MACF,OAAO;AAEL,cAAM;AAAA,UACJ,MAAM;AAAA,UACN,WAAW,MAAM;AAAA,QACnB;AAAA,MACF;AAAA,IACF;AAGA,QAAI,OAAO,cAAc,MAAM,WAAW,SAAS,GAAG;AACpD,iBAAW,YAAY,MAAM,YAAY;AACvC,cAAM;AAAA,UACJ,MAAM;AAAA,UACN,cAAc;AAAA,UACd,YAAY,SAAS;AAAA,UACrB,UAAU,SAAS,UAAU,QAAQ;AAAA,UACrC,eAAe,SAAS,UAAU,aAAa;AAAA,QACjD;AAAA,MACF;AAAA,IACF;AAGA,UAAM,eAAe,MAAM,UAAU,CAAC,GAAG;AACzC,QAAI,cAAc;AAEhB,YAAM,QAAQ,MAAM,QAChB;AAAA,QACE,cAAc,MAAM,MAAM;AAAA,QAC1B,kBAAkB,MAAM,MAAM;AAAA,MAChC,IACA,EAAE,cAAc,GAAG,kBAAkB,EAAE;AAE3C,YAAM;AAAA,QACJ,MAAM;AAAA,QACN,cAAc,KAAK,gBAAgB,YAAY;AAAA,QAC/C;AAAA,MACF;AAAA,IACF;AAAA,EACF;AAAA,EAEQ,gBAAgB,QAAqG;AAC3H,UAAM,YAAiH;AAAA,MACrH,QAAQ;AAAA,MACR,UAAU;AAAA,MACV,kBAAkB;AAAA,MAClB,cAAc;AAAA,MACd,iBAAiB;AAAA,MACjB,SAAS;AAAA,IACX;AACA,WAAO,UAAU,MAAM,KAAK;AAAA,EAC9B;AAAA,EAEQ,iBAAiB,SAA2F;AAClH,QAAI;AACF,YAAM,SAAS,KAAK,MAAM,OAAO;AAEjC,UAAI,UAAU,OAAO,OAAO,SAAS,aAAa,OAAO,cAAc,UAAa,OAAO,OAAO,cAAc,WAAW;AACzH,eAAO;AAAA,UACL,IAAI,OAAO;AAAA,UACX,MAAM,OAAO;AAAA,UACb,WAAW,OAAO,aAAa,CAAC;AAAA,QAClC;AAAA,MACF;AAAA,IACF,QAAQ;AAAA,IAER;AACA,WAAO;AAAA,EACT;AACF;;;ACjKO,IAAM,4BAAN,MAA2D;AAAA,EACvD,uBAAuB;AAAA,EACvB,WAAW;AAAA,EACX;AAAA,EACA;AAAA,EACA,8BAA2D;AAAA,EAE5D;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EAER,YACE,SACA,WAAiC,CAAC,GAClC,QAIA;AACA,SAAK,UAAU;AACf,SAAK,WAAW,qBAAqB,QAAQ;AAC7C,SAAK,UAAU,OAAO;AACtB,SAAK,UAAU,OAAO;AACtB,SAAK,qBAAqB,IAAI,mBAAmB;AACjD,SAAK,sBAAsB,IAAI,oBAAoB;AACnD,SAAK,eAAe,IAAI,aAAa;AAAA,EACvC;AAAA,EAEA,MAAM,WAAW,SAAqC;AACpD,UAAM,cAAc,KAAK,mBAAmB,UAAU,KAAK,SAAS,OAAO;AAE3E,UAAM,WAAW,MAAM,MAAM,GAAG,KAAK,OAAO,qBAAqB;AAAA,MAC/D,QAAQ;AAAA,MACR,SAAS;AAAA,QACP,gBAAgB;AAAA,QAChB,GAAG,KAAK,QAAQ;AAAA,MAClB;AAAA,MACA,MAAM,KAAK,UAAU,WAAW;AAAA,IAClC,CAAC;AAED,QAAI,CAAC,SAAS,IAAI;AAChB,YAAM,QAAQ,MAAM,SAAS,KAAK;AAClC,YAAM,IAAI,MAAM,uBAAuB,SAAS,MAAM,MAAM,KAAK,EAAE;AAAA,IACrE;AAEA,UAAM,cAAe,MAAM,SAAS,KAAK;AACzC,WAAO,KAAK,oBAAoB,UAAU,WAAW;AAAA,EACvD;AAAA,EAEA,MAAM,SAAS,SAAqC;AAClD,UAAM,cAAc;AAAA,MAClB,GAAG,KAAK,mBAAmB,UAAU,KAAK,SAAS,OAAO;AAAA,MAC1D,QAAQ;AAAA,IACV;AAEA,UAAM,WAAW,MAAM,MAAM,GAAG,KAAK,OAAO,qBAAqB;AAAA,MAC/D,QAAQ;AAAA,MACR,SAAS;AAAA,QACP,gBAAgB;AAAA,QAChB,UAAU;AAAA,QACV,GAAG,KAAK,QAAQ;AAAA,MAClB;AAAA,MACA,MAAM,KAAK,UAAU,WAAW;AAAA,IAClC,CAAC;AAED,QAAI,CAAC,SAAS,IAAI;AAChB,YAAM,QAAQ,MAAM,SAAS,KAAK;AAClC,YAAM,IAAI,MAAM,uBAAuB,SAAS,MAAM,MAAM,KAAK,EAAE;AAAA,IACrE;AAEA,QAAI,CAAC,SAAS,MAAM;AAClB,YAAM,IAAI,MAAM,uBAAuB;AAAA,IACzC;AAEA,WAAO;AAAA,MACL,QAAQ,KAAK,aAAa,MAAM,SAAS,IAAI;AAAA,MAC7C,SAAS;AAAA,QACP,WAAW;AAAA,QACX,aAAa,CAAC;AAAA,MAChB;AAAA,MACA,UAAU,CAAC;AAAA,IACb;AAAA,EACF;AACF;;;AN3FA,eAAsB,eAAe,UAAoC,CAAC,GAAG;AAC3E,QAAM,UAAU;AAAA,IACd,QAAQ,WAAW;AAAA,EACrB,KAAK;AAEL,QAAM,aAAa,MAA8B;AAC/C,UAAM,eAAe,WAAW;AAAA,MAC9B,QAAQ,QAAQ;AAAA,MAChB,yBAAyB;AAAA,MACzB,aAAa;AAAA,IACf,CAAC;AAED,UAAM,SAAU,gBAAgB;AAChC,UAAM,aAAqB,YAAY;AACvC,UAAM,SAAiC;AAAA,MACrC,eAAe;AAAA,IACjB;AACA,QAAI,QAAQ,SAAS;AACnB,iBAAW,CAAC,KAAK,KAAK,KAAK,OAAO,QAAQ,QAAQ,OAAO,GAAG;AAC1D,YAAI,OAAO,UAAU,UAAU;AAC7B,iBAAO,GAAG,IAAI;AAAA,QAChB;AAAA,MACF;AAAA,IACF;AACA,WAAO;AAAA,EACT;AAEA,QAAM,kBAAkB,CACtB,SACA,WAAiC,CAAC,MAC/B;AACH,WAAO,IAAI,0BAA0B,SAAS,UAAU;AAAA,MACtD;AAAA,MACA,SAAS;AAAA,IACX,CAAC;AAAA,EACH;AAEA,QAAM,WAAW;AAAA,IACf,IAAI;AAAA,IACJ,MAAM;AAAA,IACN,MAAM,aAAa;AAEjB,UAAI;AACF,cAAM,WAAW;AAAA,UACf,QAAQ,QAAQ;AAAA,UAChB,yBAAyB;AAAA,UACzB,aAAa;AAAA,QACf,CAAC;AAAA,MACH,SAAS,OAAO;AACd,cAAM,IAAI,MAAM,4CAA4C,KAAK,EAAE;AAAA,MACrE;AAAA,IACF;AAAA,IACA,MAAM;AAAA,EACR;AAEA,SAAO;AACT;AAGO,IAAM,WAAW,eAAe;","names":[]}
|
|
1
|
+
{"version":3,"sources":["../src/deepseek-provider.ts","../src/deepseek-chat-settings.ts","../src/utils/chinese-params-preprocessor.ts","../src/utils/request-transformer.ts","../src/utils/response-transformer.ts","../src/utils/stream-parser.ts","../src/deepseek-chat-language-model.ts"],"sourcesContent":["// DeepSeek Provider 工厂 - 基于官方 @ai-sdk/deepseek 扩展\n\nimport { createDeepSeek as createOfficialProvider } from '@ai-sdk/deepseek';\nimport { loadApiKey } from '@ai-sdk/provider-utils';\nimport type { DeepSeekProviderSettings } from './types';\n\ninterface DeepSeekProviderInstance {\n id: string;\n name: string;\n initialize(): Promise<void>;\n chat(modelId: string, settings?: any): any;\n}\n\nexport async function createProvider(options: DeepSeekProviderSettings = {}): Promise<DeepSeekProviderInstance> {\n const officialProvider = createOfficialProvider({\n apiKey: options.apiKey,\n baseURL: options.baseURL,\n headers: options.headers,\n });\n\n const provider: DeepSeekProviderInstance = {\n id: 'deepseek',\n name: 'DeepSeek AI',\n async initialize() {\n await loadApiKey({\n apiKey: options.apiKey,\n environmentVariableName: 'DEEPSEEK_API_KEY',\n description: 'DeepSeek API key',\n });\n },\n chat: officialProvider.chat,\n };\n\n return provider;\n}\n\nexport const deepseek = createProvider();","// DeepSeek Chat 模型配置\n\nimport type { DeepSeekChatSettings } from './types';\n\nexport function validateChatSettings(settings: DeepSeekChatSettings = {}): Required<DeepSeekChatSettings> {\n return {\n temperature: settings.temperature ?? 0.7,\n topP: settings.topP ?? 1.0,\n maxTokens: settings.maxTokens ?? 4096,\n presencePenalty: settings.presencePenalty ?? 0,\n frequencyPenalty: settings.frequencyPenalty ?? 0,\n };\n}","// 中文参数预处理器\n\nimport type { DeepSeekRequest } from '../types';\n\nexport class ChineseParamsPreprocessor {\n private cityMap: Record<string, string> = {\n // 中国城市\n '北京': 'Beijing',\n '上海': 'Shanghai',\n '广州': 'Guangzhou',\n '深圳': 'Shenzhen',\n '杭州': 'Hangzhou',\n '南京': 'Nanjing',\n '成都': 'Chengdu',\n '武汉': 'Wuhan',\n '西安': 'Xi\\'an',\n '重庆': 'Chongqing',\n '天津': 'Tianjin',\n '苏州': 'Suzhou',\n '长沙': 'Changsha',\n '青岛': 'Qingdao',\n '大连': 'Dalian',\n '厦门': 'Xiamen',\n '宁波': 'Ningbo',\n '无锡': 'Wuxi',\n '佛山': 'Foshan',\n '东莞': 'Dongguan',\n '合肥': 'Hefei',\n '郑州': 'Zhengzhou',\n '福州': 'Fuzhou',\n '济南': 'Jinan',\n '昆明': 'Kunming',\n '沈阳': 'Shenyang',\n '长春': 'Changchun',\n '哈尔滨': 'Harbin',\n '石家庄': 'Shijiazhuang',\n '太原': 'Taiyuan',\n '南昌': 'Nanchang',\n '南宁': 'Nanning',\n '贵阳': 'Guiyang',\n '兰州': 'Lanzhou',\n '海口': 'Haikou',\n '呼和浩特': 'Hohhot',\n '银川': 'Yinchuan',\n '西宁': 'Xining',\n '乌鲁木齐': 'Urumqi',\n '拉萨': 'Lhasa',\n // 国际城市\n '伦敦': 'London',\n '纽约': 'New York',\n '东京': 'Tokyo',\n '巴黎': 'Paris',\n '柏林': 'Berlin',\n '悉尼': 'Sydney',\n '洛杉矶': 'Los Angeles',\n '多伦多': 'Toronto',\n '新加坡': 'Singapore',\n '香港': 'Hong Kong',\n '台北': 'Taipei',\n '首尔': 'Seoul',\n '曼谷': 'Bangkok',\n '迪拜': 'Dubai',\n '莫斯科': 'Moscow',\n '旧金山': 'San Francisco',\n '波士顿': 'Boston',\n '芝加哥': 'Chicago',\n '华盛顿': 'Washington',\n '休斯顿': 'Houston',\n '费城': 'Philadelphia',\n '凤凰城': 'Phoenix',\n '圣迭戈': 'San Diego',\n '达拉斯': 'Dallas',\n '圣何塞': 'San Jose',\n '奥斯汀': 'Austin',\n '温哥华': 'Vancouver',\n '蒙特利尔': 'Montreal',\n '卡尔加里': 'Calgary',\n '渥太华': 'Ottawa',\n '魁北克': 'Quebec',\n '埃德蒙顿': 'Edmonton',\n '温尼伯': 'Winnipeg',\n '哈利法克斯': 'Halifax',\n '维多利亚': 'Victoria',\n };\n\n process(request: DeepSeekRequest): DeepSeekRequest {\n const processed = JSON.parse(JSON.stringify(request));\n\n // 处理消息内容\n if (processed.messages) {\n processed.messages = this.processMessages(processed.messages);\n }\n\n // 处理工具参数\n if (processed.tools) {\n processed.tools = this.processTools(processed.tools);\n }\n\n return processed;\n }\n\n private processMessages(messages: any[]): any[] {\n return messages.map(msg => {\n if (msg.content && typeof msg.content === 'string') {\n msg.content = this.replaceCityNames(msg.content);\n } else if (Array.isArray(msg.content)) {\n msg.content = msg.content.map((part: any) => {\n if (part.type === 'text' && part.text) {\n part.text = this.replaceCityNames(part.text);\n }\n return part;\n });\n }\n return msg;\n });\n }\n\n private processTools(tools: any[]): any[] {\n return tools.map(tool => {\n if (tool.function?.parameters?.properties) {\n this.processProperties(tool.function.parameters.properties);\n }\n if (tool.function?.description) {\n tool.function.description = this.replaceCityNames(tool.function.description);\n }\n return tool;\n });\n }\n\n private processProperties(properties: any): void {\n for (const key in properties) {\n if (properties[key].description) {\n properties[key].description = this.replaceCityNames(properties[key].description);\n }\n if (properties[key].example) {\n const exampleStr = JSON.stringify(properties[key].example);\n properties[key].example = JSON.parse(this.replaceCityNames(exampleStr));\n }\n }\n }\n\n private replaceCityNames(text: string): string {\n let result = text;\n for (const [chinese, english] of Object.entries(this.cityMap)) {\n // 使用正则表达式替换,避免替换部分匹配的词\n const regex = new RegExp(`\\\\b${chinese}\\\\b`, 'g');\n result = result.replace(regex, english);\n }\n return result;\n }\n\n // 添加自定义城市映射\n addCityMapping(chinese: string, english: string): void {\n this.cityMap[chinese] = english;\n }\n\n // 批量添加城市映射\n addCityMappings(mappings: Record<string, string>): void {\n Object.assign(this.cityMap, mappings);\n }\n}","// 请求转换器\n\nimport type {\n LanguageModelV1CallOptions,\n LanguageModelV1Prompt,\n} from '@ai-sdk/provider';\nimport type { DeepSeekRequest } from '../types';\nimport { ChineseParamsPreprocessor } from './chinese-params-preprocessor';\n\nexport class RequestTransformer {\n private preprocessor: ChineseParamsPreprocessor;\n\n constructor() {\n this.preprocessor = new ChineseParamsPreprocessor();\n }\n\n transform(modelId: string, options: LanguageModelV1CallOptions): DeepSeekRequest {\n const transformed: DeepSeekRequest = {\n model: modelId,\n messages: this.transformMessages(options.prompt),\n stream: false,\n };\n\n // 添加温度参数\n if (options.temperature !== undefined) {\n transformed.temperature = options.temperature;\n }\n\n // 添加 top_p 参数\n if (options.topP !== undefined) {\n transformed.top_p = options.topP;\n }\n\n // 添加 max_tokens 参数\n if (options.maxTokens !== undefined) {\n transformed.max_tokens = options.maxTokens;\n }\n\n // 添加 presence_penalty 参数\n if (options.presencePenalty !== undefined) {\n transformed.presence_penalty = options.presencePenalty;\n }\n\n // 添加 frequency_penalty 参数\n if (options.frequencyPenalty !== undefined) {\n transformed.frequency_penalty = options.frequencyPenalty;\n }\n\n // 添加工具配置\n const mode = options.mode;\n let toolsArray: Array<any> = [];\n let toolChoice: any;\n\n // 只有 'regular' 模式支持工具\n if (mode && mode.type === 'regular') {\n toolsArray = mode.tools ?? [];\n toolChoice = mode.toolChoice;\n }\n\n if (toolsArray.length > 0) {\n transformed.tools = this.transformTools(toolsArray);\n transformed.tool_choice = this.transformToolChoice(toolChoice);\n }\n\n // 预处理中文参数\n return this.preprocessor.process(transformed);\n }\n\n private transformMessages(messages: LanguageModelV1Prompt): Array<any> {\n return messages.map((msg) => {\n const transformed: any = {\n role: msg.role,\n };\n\n // 处理内容\n if (msg.content && Array.isArray(msg.content)) {\n const contentParts: Array<any> = [];\n let hasToolCalls = false;\n let toolCallsArray: Array<any> = [];\n\n for (const part of msg.content) {\n // 处理文本内容\n if (part.type === 'text') {\n contentParts.push({ type: 'text', text: part.text });\n }\n // 处理图像内容\n else if (part.type === 'image') {\n // V1 image part has 'image' property, convert to data for DeepSeek API\n const imageData = part.image;\n if (typeof imageData === 'string') {\n contentParts.push({ type: 'image_url', image_url: { url: imageData } });\n } else if (imageData instanceof URL) {\n contentParts.push({ type: 'image_url', image_url: { url: imageData.toString() } });\n }\n }\n // 处理工具调用 (V1 格式)\n else if (part.type === 'tool-call') {\n hasToolCalls = true;\n toolCallsArray.push({\n id: part.toolCallId,\n type: 'function',\n function: {\n name: part.toolName,\n arguments: typeof part.args === 'string' ? part.args : JSON.stringify(part.args),\n },\n });\n }\n // 处理工具结果 (V1 格式)\n else if (part.type === 'tool-result') {\n // Tool results in V1 don't get added to content directly\n // They are already part of the prompt structure\n }\n // 处理推理内容\n else if (part.type === 'reasoning' || part.type === 'redacted-reasoning') {\n // DeepSeek API doesn't support reasoning content directly\n // We'll add it as text if there's a text field\n if ('text' in part && part.text) {\n contentParts.push({ type: 'text', text: part.text });\n }\n }\n // 处理文件内容\n else if (part.type === 'file') {\n contentParts.push({\n type: 'text',\n text: `[File: ${part.filename || 'unnamed'}]`,\n });\n }\n // 其他未知类型\n else {\n contentParts.push(part);\n }\n }\n\n // 设置内容\n if (contentParts.length > 0) {\n transformed.content = contentParts;\n }\n\n // 添加工具调用\n if (hasToolCalls && toolCallsArray.length > 0) {\n transformed.tool_calls = toolCallsArray;\n }\n } else if (typeof msg.content === 'string') {\n // 系统消息等内容为字符串的情况\n transformed.content = msg.content;\n }\n\n return transformed;\n });\n }\n\n private transformTools(tools: Array<any>): Array<any> {\n return tools.map(tool => ({\n type: 'function',\n function: {\n name: tool.name,\n description: tool.description,\n parameters: tool.parameters,\n },\n }));\n }\n\n private transformToolChoice(toolChoice: any): 'auto' | 'none' | 'required' | string {\n if (toolChoice === undefined || toolChoice === 'auto') {\n return 'auto';\n }\n if (toolChoice === 'none') {\n return 'none';\n }\n if (toolChoice === 'required') {\n return 'required';\n }\n if (typeof toolChoice === 'object' && toolChoice.type === 'tool') {\n return JSON.stringify({\n type: 'function',\n function: {\n name: toolChoice.toolName,\n },\n });\n }\n return 'auto';\n }\n}","// 响应转换器 - V1 API 兼容版本\n\nimport type {\n LanguageModelV1,\n LanguageModelV1FinishReason,\n LanguageModelV1CallWarning,\n LanguageModelV1FunctionToolCall,\n} from '@ai-sdk/provider';\nimport type { DeepSeekResponse } from '../types';\n\n// V1 API 返回结果类型\nexport interface V1GenerateResult {\n text?: string;\n reasoning?: string;\n usage: {\n promptTokens: number;\n completionTokens: number;\n };\n finishReason: LanguageModelV1FinishReason;\n toolCalls?: LanguageModelV1FunctionToolCall[];\n rawCall: {\n rawPrompt: unknown;\n rawSettings: Record<string, unknown>;\n };\n warnings?: LanguageModelV1CallWarning[];\n}\n\nexport class ResponseTransformer {\n transform(apiResponse: DeepSeekResponse): V1GenerateResult {\n const choice = apiResponse.choices?.[0];\n const message = choice?.message || {};\n\n // 修复 finish_reason\n const finishReason = this.fixFinishReason(\n choice?.finish_reason,\n message.tool_calls\n );\n\n return {\n text: message.content || '',\n finishReason: finishReason as LanguageModelV1FinishReason,\n usage: {\n promptTokens: apiResponse.usage?.prompt_tokens || 0,\n completionTokens: apiResponse.usage?.completion_tokens || 0,\n },\n rawCall: {\n rawPrompt: '',\n rawSettings: {},\n },\n toolCalls: this.transformToolCalls(message.tool_calls),\n warnings: [],\n };\n }\n\n private fixFinishReason(\n originalFinishReason: string | undefined,\n toolCalls: any[] | undefined\n ): string {\n // 如果有 tool_calls 但 finish_reason 是 \"stop\",修复为 \"tool-calls\"\n if (toolCalls && toolCalls.length > 0 && originalFinishReason === 'stop') {\n return 'tool-calls';\n }\n\n // 映射其他 finish_reason 值\n const finishReasonMap: Record<string, string> = {\n 'stop': 'stop',\n 'length': 'length',\n 'content_filter': 'content-filter',\n 'tool_calls': 'tool-calls',\n 'function_call': 'tool-calls',\n };\n\n return finishReasonMap[originalFinishReason || ''] || 'stop';\n }\n\n private transformToolCalls(toolCalls: any[] | undefined): LanguageModelV1FunctionToolCall[] {\n if (!toolCalls || toolCalls.length === 0) {\n return [];\n }\n\n return toolCalls.map((tc: any) => ({\n toolCallType: 'function' as const,\n toolCallId: tc.id,\n toolName: tc.function.name,\n args: typeof tc.function.arguments === 'string'\n ? tc.function.arguments\n : JSON.stringify(tc.function.arguments),\n }));\n }\n}","// 流解析器\n\nimport type { LanguageModelV1StreamPart } from '@ai-sdk/provider';\nimport type { DeepSeekStreamResponse, DeepSeekUsage } from '../types';\n\ninterface DeepSeekStreamChunk {\n id: string;\n object: string;\n created: number;\n model: string;\n choices: Array<{\n index: number;\n delta: {\n role?: string;\n content?: string;\n tool_calls?: Array<{\n id: string;\n type: 'function';\n function: {\n name: string;\n arguments: string;\n };\n }>;\n };\n finish_reason?: string;\n }>;\n usage?: DeepSeekUsage;\n}\n\nexport class StreamParser {\n async *parse(body: ReadableStream<Uint8Array>): AsyncIterable<LanguageModelV1StreamPart> {\n const reader = body.getReader();\n const decoder = new TextDecoder('utf-8');\n let buffer = '';\n let partialJson = '';\n\n try {\n while (true) {\n const { done, value } = await reader.read();\n if (done) break;\n\n buffer += decoder.decode(value, { stream: true });\n const lines = buffer.split('\\n');\n buffer = lines.pop() || '';\n\n for (const line of lines) {\n if (line.trim() === '') continue;\n if (line.startsWith(':')) continue; // SSE 注释\n\n if (line.startsWith('data: ')) {\n const data = line.slice(6).trim();\n if (data === '[DONE]') {\n yield* this.processPartialJson(partialJson);\n partialJson = '';\n return;\n }\n\n try {\n // 尝试解析 JSON\n const parsed = JSON.parse(data);\n yield* this.processPartialJson(partialJson);\n partialJson = '';\n yield* this.processChunk(parsed);\n } catch (e) {\n // JSON 解析失败,可能是部分数据\n partialJson += data;\n }\n }\n }\n }\n\n // 处理剩余的部分 JSON\n yield* this.processPartialJson(partialJson);\n } finally {\n reader.releaseLock();\n }\n }\n\n private async *processPartialJson(partialJson: string): AsyncIterable<LanguageModelV1StreamPart> {\n if (!partialJson.trim()) return;\n\n try {\n const parsed = JSON.parse(partialJson);\n yield* this.processChunk(parsed);\n } catch (e) {\n // 仍然无法解析,跳过\n console.warn('Failed to parse partial JSON:', partialJson);\n }\n }\n\n private async *processChunk(chunk: DeepSeekStreamChunk): AsyncIterable<LanguageModelV1StreamPart> {\n const delta = chunk.choices?.[0]?.delta;\n\n // 工具调用增量 - JSON in content 格式\n if (delta?.content) {\n const toolCallData = this.tryParseToolCall(delta.content);\n if (toolCallData) {\n yield {\n type: 'tool-call-delta',\n toolCallType: 'function',\n toolCallId: toolCallData.id || `tool-${Date.now()}-${Math.random().toString(36).substr(2, 9)}`,\n toolName: toolCallData.name,\n argsTextDelta: JSON.stringify(toolCallData.arguments || {}),\n } as LanguageModelV1StreamPart;\n } else {\n // 普通文本增量\n yield {\n type: 'text-delta',\n textDelta: delta.content,\n } as LanguageModelV1StreamPart;\n }\n }\n\n // 工具调用增量 - 标准 tool_calls 格式\n if (delta?.tool_calls && delta.tool_calls.length > 0) {\n for (const toolCall of delta.tool_calls) {\n yield {\n type: 'tool-call-delta',\n toolCallType: 'function',\n toolCallId: toolCall.id,\n toolName: toolCall.function?.name || '',\n argsTextDelta: toolCall.function?.arguments || '',\n } as LanguageModelV1StreamPart;\n }\n }\n\n // 完成信号\n const finishReason = chunk.choices?.[0]?.finish_reason;\n if (finishReason) {\n // V1 API: usage 在 finish 类型的 part 中\n const usage = chunk.usage\n ? {\n promptTokens: chunk.usage.prompt_tokens,\n completionTokens: chunk.usage.completion_tokens,\n }\n : { promptTokens: 0, completionTokens: 0 };\n\n yield {\n type: 'finish',\n finishReason: this.mapFinishReason(finishReason),\n usage,\n } as LanguageModelV1StreamPart;\n }\n }\n\n private mapFinishReason(reason: string): 'stop' | 'length' | 'content-filter' | 'tool-calls' | 'error' | 'other' | 'unknown' {\n const reasonMap: Record<string, 'stop' | 'length' | 'content-filter' | 'tool-calls' | 'error' | 'other' | 'unknown'> = {\n 'stop': 'stop',\n 'length': 'length',\n 'content_filter': 'content-filter',\n 'tool_calls': 'tool-calls',\n 'function_call': 'tool-calls',\n 'error': 'error',\n };\n return reasonMap[reason] || 'unknown';\n }\n\n private tryParseToolCall(content: string): { id?: string; name: string; arguments: Record<string, unknown> } | null {\n try {\n const parsed = JSON.parse(content);\n // 检查是否是工具调用格式: { name: \"...\", arguments: {...} }\n if (parsed && typeof parsed.name === 'string' && (parsed.arguments === undefined || typeof parsed.arguments === 'object')) {\n return {\n id: parsed.id,\n name: parsed.name,\n arguments: parsed.arguments || {},\n };\n }\n } catch {\n // JSON 解析失败,不是工具调用\n }\n return null;\n }\n}\n\n\n","// DeepSeek Chat 模型实现\n\nimport type {\n LanguageModelV1,\n LanguageModelV1CallOptions,\n} from '@ai-sdk/provider';\nimport type { DeepSeekChatSettings, DeepSeekResponse } from './types';\nimport { validateChatSettings } from './deepseek-chat-settings';\nimport { RequestTransformer } from './utils/request-transformer';\nimport { ResponseTransformer } from './utils/response-transformer';\nimport { StreamParser } from './utils/stream-parser';\n\nexport class DeepSeekChatLanguageModel implements LanguageModelV1 {\n readonly specificationVersion = 'v1' as const;\n readonly provider = 'deepseek' as const;\n readonly modelId: string;\n readonly settings: Required<DeepSeekChatSettings>;\n readonly defaultObjectGenerationMode: 'json' | 'tool' | undefined = undefined;\n\n private baseURL: string;\n private headers: () => Record<string, string>;\n private requestTransformer: RequestTransformer;\n private responseTransformer: ResponseTransformer;\n private streamParser: StreamParser;\n\n constructor(\n modelId: string,\n settings: DeepSeekChatSettings = {},\n config: {\n baseURL: string;\n headers: () => Record<string, string>;\n }\n ) {\n this.modelId = modelId;\n this.settings = validateChatSettings(settings);\n this.baseURL = config.baseURL;\n this.headers = config.headers;\n this.requestTransformer = new RequestTransformer();\n this.responseTransformer = new ResponseTransformer();\n this.streamParser = new StreamParser();\n }\n\n async doGenerate(options: LanguageModelV1CallOptions) {\n const requestBody = this.requestTransformer.transform(this.modelId, options);\n\n const response = await fetch(`${this.baseURL}/chat/completions`, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n ...this.headers(),\n },\n body: JSON.stringify(requestBody),\n });\n\n if (!response.ok) {\n const error = await response.text();\n throw new Error(`DeepSeek API error: ${response.status} - ${error}`);\n }\n\n const apiResponse = (await response.json()) as DeepSeekResponse;\n return this.responseTransformer.transform(apiResponse);\n }\n\n async doStream(options: LanguageModelV1CallOptions) {\n const requestBody = {\n ...this.requestTransformer.transform(this.modelId, options),\n stream: true,\n };\n\n const response = await fetch(`${this.baseURL}/chat/completions`, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n 'Accept': 'text/event-stream',\n ...this.headers(),\n },\n body: JSON.stringify(requestBody),\n });\n\n if (!response.ok) {\n const error = await response.text();\n throw new Error(`DeepSeek API error: ${response.status} - ${error}`);\n }\n\n if (!response.body) {\n throw new Error('Response body is null');\n }\n\n return {\n stream: this.streamParser.parse(response.body) as any,\n rawCall: {\n rawPrompt: requestBody as unknown,\n rawSettings: {},\n },\n warnings: [],\n };\n }\n}"],"mappings":"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","names":[]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@hopemyl619/deepseek",
|
|
3
|
-
"version": "0.
|
|
3
|
+
"version": "0.2.0",
|
|
4
4
|
"description": "DeepSeek provider for the Vercel AI SDK with Chinese parameter preprocessing and automatic finish_reason fix",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"exports": {
|
|
@@ -15,7 +15,7 @@
|
|
|
15
15
|
"src"
|
|
16
16
|
],
|
|
17
17
|
"scripts": {
|
|
18
|
-
"build": "tsup",
|
|
18
|
+
"build": "tsup --no-dts",
|
|
19
19
|
"dev": "tsup --watch",
|
|
20
20
|
"test": "vitest",
|
|
21
21
|
"typecheck": "tsc --noEmit",
|
|
@@ -23,6 +23,7 @@
|
|
|
23
23
|
"prepublishOnly": "npm run build"
|
|
24
24
|
},
|
|
25
25
|
"dependencies": {
|
|
26
|
+
"@ai-sdk/deepseek": "^2.0.11",
|
|
26
27
|
"@ai-sdk/provider": "^1.0.2",
|
|
27
28
|
"@ai-sdk/provider-utils": "^2.0.4"
|
|
28
29
|
},
|
package/src/deepseek-provider.ts
CHANGED
|
@@ -1,66 +1,37 @@
|
|
|
1
|
-
// DeepSeek Provider 工厂
|
|
1
|
+
// DeepSeek Provider 工厂 - 基于官方 @ai-sdk/deepseek 扩展
|
|
2
2
|
|
|
3
|
-
import {
|
|
4
|
-
import {
|
|
5
|
-
import type { DeepSeekProviderSettings
|
|
3
|
+
import { createDeepSeek as createOfficialProvider } from '@ai-sdk/deepseek';
|
|
4
|
+
import { loadApiKey } from '@ai-sdk/provider-utils';
|
|
5
|
+
import type { DeepSeekProviderSettings } from './types';
|
|
6
6
|
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
)
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
const apiKeyResult = loadApiKey({
|
|
14
|
-
apiKey: options.apiKey,
|
|
15
|
-
environmentVariableName: 'DEEPSEEK_API_KEY',
|
|
16
|
-
description: 'DeepSeek API key',
|
|
17
|
-
});
|
|
18
|
-
// loadApiKey 可能返回 undefined,使用空字符串作为默认值并确保类型为 string
|
|
19
|
-
const apiKey = (apiKeyResult ?? '') as string;
|
|
20
|
-
const authHeader: string = 'Bearer ' + apiKey;
|
|
21
|
-
const result: Record<string, string> = {
|
|
22
|
-
Authorization: authHeader,
|
|
23
|
-
};
|
|
24
|
-
if (options.headers) {
|
|
25
|
-
for (const [key, value] of Object.entries(options.headers)) {
|
|
26
|
-
if (typeof value === 'string') {
|
|
27
|
-
result[key] = value;
|
|
28
|
-
}
|
|
29
|
-
}
|
|
30
|
-
}
|
|
31
|
-
return result;
|
|
32
|
-
};
|
|
7
|
+
interface DeepSeekProviderInstance {
|
|
8
|
+
id: string;
|
|
9
|
+
name: string;
|
|
10
|
+
initialize(): Promise<void>;
|
|
11
|
+
chat(modelId: string, settings?: any): any;
|
|
12
|
+
}
|
|
33
13
|
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
headers: getHeaders,
|
|
41
|
-
});
|
|
42
|
-
};
|
|
14
|
+
export async function createProvider(options: DeepSeekProviderSettings = {}): Promise<DeepSeekProviderInstance> {
|
|
15
|
+
const officialProvider = createOfficialProvider({
|
|
16
|
+
apiKey: options.apiKey,
|
|
17
|
+
baseURL: options.baseURL,
|
|
18
|
+
headers: options.headers,
|
|
19
|
+
});
|
|
43
20
|
|
|
44
|
-
const provider = {
|
|
21
|
+
const provider: DeepSeekProviderInstance = {
|
|
45
22
|
id: 'deepseek',
|
|
46
23
|
name: 'DeepSeek AI',
|
|
47
24
|
async initialize() {
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
description: 'DeepSeek API key',
|
|
54
|
-
});
|
|
55
|
-
} catch (error) {
|
|
56
|
-
throw new Error(`DeepSeek provider initialization failed: ${error}`);
|
|
57
|
-
}
|
|
25
|
+
await loadApiKey({
|
|
26
|
+
apiKey: options.apiKey,
|
|
27
|
+
environmentVariableName: 'DEEPSEEK_API_KEY',
|
|
28
|
+
description: 'DeepSeek API key',
|
|
29
|
+
});
|
|
58
30
|
},
|
|
59
|
-
chat:
|
|
31
|
+
chat: officialProvider.chat,
|
|
60
32
|
};
|
|
61
33
|
|
|
62
34
|
return provider;
|
|
63
35
|
}
|
|
64
36
|
|
|
65
|
-
// 默认导出
|
|
66
37
|
export const deepseek = createProvider();
|
package/dist/index.d.ts
DELETED
|
@@ -1,184 +0,0 @@
|
|
|
1
|
-
import { LanguageModelV1FinishReason, LanguageModelV1FunctionToolCall, LanguageModelV1CallWarning, LanguageModelV1, LanguageModelV1CallOptions, LanguageModelV1StreamPart } from '@ai-sdk/provider';
|
|
2
|
-
|
|
3
|
-
interface DeepSeekProviderSettings {
|
|
4
|
-
apiKey?: string;
|
|
5
|
-
baseURL?: string;
|
|
6
|
-
headers?: Record<string, string | undefined>;
|
|
7
|
-
}
|
|
8
|
-
interface DeepSeekChatSettings {
|
|
9
|
-
temperature?: number;
|
|
10
|
-
topP?: number;
|
|
11
|
-
maxTokens?: number;
|
|
12
|
-
presencePenalty?: number;
|
|
13
|
-
frequencyPenalty?: number;
|
|
14
|
-
}
|
|
15
|
-
interface DeepSeekMessage {
|
|
16
|
-
role: 'system' | 'user' | 'assistant' | 'tool';
|
|
17
|
-
content: string | Array<{
|
|
18
|
-
type: 'text' | 'image_url';
|
|
19
|
-
text?: string;
|
|
20
|
-
image_url?: {
|
|
21
|
-
url: string;
|
|
22
|
-
};
|
|
23
|
-
}>;
|
|
24
|
-
tool_calls?: DeepSeekToolCall[];
|
|
25
|
-
tool_call_id?: string;
|
|
26
|
-
}
|
|
27
|
-
interface DeepSeekToolCall {
|
|
28
|
-
id: string;
|
|
29
|
-
type: 'function';
|
|
30
|
-
function: {
|
|
31
|
-
name: string;
|
|
32
|
-
arguments: string;
|
|
33
|
-
};
|
|
34
|
-
}
|
|
35
|
-
interface DeepSeekTool {
|
|
36
|
-
type: 'function';
|
|
37
|
-
function: {
|
|
38
|
-
name: string;
|
|
39
|
-
description?: string;
|
|
40
|
-
parameters?: Record<string, any>;
|
|
41
|
-
};
|
|
42
|
-
}
|
|
43
|
-
interface DeepSeekRequest {
|
|
44
|
-
model: string;
|
|
45
|
-
messages: DeepSeekMessage[];
|
|
46
|
-
temperature?: number;
|
|
47
|
-
top_p?: number;
|
|
48
|
-
max_tokens?: number;
|
|
49
|
-
presence_penalty?: number;
|
|
50
|
-
frequency_penalty?: number;
|
|
51
|
-
stream?: boolean;
|
|
52
|
-
tools?: DeepSeekTool[];
|
|
53
|
-
tool_choice?: 'auto' | 'none' | 'required' | string;
|
|
54
|
-
}
|
|
55
|
-
interface DeepSeekChoice {
|
|
56
|
-
index: number;
|
|
57
|
-
message: {
|
|
58
|
-
role: string;
|
|
59
|
-
content: string | null;
|
|
60
|
-
tool_calls?: DeepSeekToolCall[];
|
|
61
|
-
};
|
|
62
|
-
finish_reason: string;
|
|
63
|
-
}
|
|
64
|
-
interface DeepSeekUsage {
|
|
65
|
-
prompt_tokens: number;
|
|
66
|
-
completion_tokens: number;
|
|
67
|
-
total_tokens: number;
|
|
68
|
-
}
|
|
69
|
-
interface DeepSeekResponse {
|
|
70
|
-
id: string;
|
|
71
|
-
object: string;
|
|
72
|
-
created: number;
|
|
73
|
-
model: string;
|
|
74
|
-
choices: DeepSeekChoice[];
|
|
75
|
-
usage: DeepSeekUsage;
|
|
76
|
-
}
|
|
77
|
-
interface DeepSeekStreamChoice {
|
|
78
|
-
index: number;
|
|
79
|
-
delta: {
|
|
80
|
-
role?: string;
|
|
81
|
-
content?: string;
|
|
82
|
-
tool_calls?: DeepSeekToolCall[];
|
|
83
|
-
};
|
|
84
|
-
finish_reason?: string;
|
|
85
|
-
}
|
|
86
|
-
interface DeepSeekStreamResponse {
|
|
87
|
-
id: string;
|
|
88
|
-
object: string;
|
|
89
|
-
created: number;
|
|
90
|
-
model: string;
|
|
91
|
-
choices: DeepSeekStreamChoice[];
|
|
92
|
-
}
|
|
93
|
-
|
|
94
|
-
interface V1GenerateResult {
|
|
95
|
-
text?: string;
|
|
96
|
-
reasoning?: string;
|
|
97
|
-
usage: {
|
|
98
|
-
promptTokens: number;
|
|
99
|
-
completionTokens: number;
|
|
100
|
-
};
|
|
101
|
-
finishReason: LanguageModelV1FinishReason;
|
|
102
|
-
toolCalls?: LanguageModelV1FunctionToolCall[];
|
|
103
|
-
rawCall: {
|
|
104
|
-
rawPrompt: unknown;
|
|
105
|
-
rawSettings: Record<string, unknown>;
|
|
106
|
-
};
|
|
107
|
-
warnings?: LanguageModelV1CallWarning[];
|
|
108
|
-
}
|
|
109
|
-
declare class ResponseTransformer {
|
|
110
|
-
transform(apiResponse: DeepSeekResponse): V1GenerateResult;
|
|
111
|
-
private fixFinishReason;
|
|
112
|
-
private transformToolCalls;
|
|
113
|
-
}
|
|
114
|
-
|
|
115
|
-
declare class DeepSeekChatLanguageModel implements LanguageModelV1 {
|
|
116
|
-
readonly specificationVersion: "v1";
|
|
117
|
-
readonly provider: "deepseek";
|
|
118
|
-
readonly modelId: string;
|
|
119
|
-
readonly settings: Required<DeepSeekChatSettings>;
|
|
120
|
-
readonly defaultObjectGenerationMode: 'json' | 'tool' | undefined;
|
|
121
|
-
private baseURL;
|
|
122
|
-
private headers;
|
|
123
|
-
private requestTransformer;
|
|
124
|
-
private responseTransformer;
|
|
125
|
-
private streamParser;
|
|
126
|
-
constructor(modelId: string, settings: DeepSeekChatSettings | undefined, config: {
|
|
127
|
-
baseURL: string;
|
|
128
|
-
headers: () => Record<string, string>;
|
|
129
|
-
});
|
|
130
|
-
doGenerate(options: LanguageModelV1CallOptions): Promise<V1GenerateResult>;
|
|
131
|
-
doStream(options: LanguageModelV1CallOptions): Promise<{
|
|
132
|
-
stream: any;
|
|
133
|
-
rawCall: {
|
|
134
|
-
rawPrompt: unknown;
|
|
135
|
-
rawSettings: {};
|
|
136
|
-
};
|
|
137
|
-
warnings: never[];
|
|
138
|
-
}>;
|
|
139
|
-
}
|
|
140
|
-
|
|
141
|
-
declare function createProvider(options?: DeepSeekProviderSettings): Promise<{
|
|
142
|
-
id: string;
|
|
143
|
-
name: string;
|
|
144
|
-
initialize(): Promise<void>;
|
|
145
|
-
chat: (modelId: string, settings?: DeepSeekChatSettings) => DeepSeekChatLanguageModel;
|
|
146
|
-
}>;
|
|
147
|
-
declare const deepseek: Promise<{
|
|
148
|
-
id: string;
|
|
149
|
-
name: string;
|
|
150
|
-
initialize(): Promise<void>;
|
|
151
|
-
chat: (modelId: string, settings?: DeepSeekChatSettings) => DeepSeekChatLanguageModel;
|
|
152
|
-
}>;
|
|
153
|
-
|
|
154
|
-
declare function validateChatSettings(settings?: DeepSeekChatSettings): Required<DeepSeekChatSettings>;
|
|
155
|
-
|
|
156
|
-
declare class ChineseParamsPreprocessor {
|
|
157
|
-
private cityMap;
|
|
158
|
-
process(request: DeepSeekRequest): DeepSeekRequest;
|
|
159
|
-
private processMessages;
|
|
160
|
-
private processTools;
|
|
161
|
-
private processProperties;
|
|
162
|
-
private replaceCityNames;
|
|
163
|
-
addCityMapping(chinese: string, english: string): void;
|
|
164
|
-
addCityMappings(mappings: Record<string, string>): void;
|
|
165
|
-
}
|
|
166
|
-
|
|
167
|
-
declare class RequestTransformer {
|
|
168
|
-
private preprocessor;
|
|
169
|
-
constructor();
|
|
170
|
-
transform(modelId: string, options: LanguageModelV1CallOptions): DeepSeekRequest;
|
|
171
|
-
private transformMessages;
|
|
172
|
-
private transformTools;
|
|
173
|
-
private transformToolChoice;
|
|
174
|
-
}
|
|
175
|
-
|
|
176
|
-
declare class StreamParser {
|
|
177
|
-
parse(body: ReadableStream<Uint8Array>): AsyncIterable<LanguageModelV1StreamPart>;
|
|
178
|
-
private processPartialJson;
|
|
179
|
-
private processChunk;
|
|
180
|
-
private mapFinishReason;
|
|
181
|
-
private tryParseToolCall;
|
|
182
|
-
}
|
|
183
|
-
|
|
184
|
-
export { ChineseParamsPreprocessor, DeepSeekChatLanguageModel, type DeepSeekChatSettings, type DeepSeekChoice, type DeepSeekMessage, type DeepSeekProviderSettings, type DeepSeekRequest, type DeepSeekResponse, type DeepSeekStreamChoice, type DeepSeekStreamResponse, type DeepSeekTool, type DeepSeekToolCall, type DeepSeekUsage, RequestTransformer, ResponseTransformer, StreamParser, createProvider, deepseek, validateChatSettings };
|