@oflow-ai/core 0.1.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/LICENSE +21 -0
- package/README.md +81 -0
- package/dist/agents/index.d.ts +35 -0
- package/dist/agents/index.js +233 -0
- package/dist/ai/chinese-provider.d.ts +146 -0
- package/dist/ai/chinese-provider.js +193 -0
- package/dist/ai/custom-provider.d.ts +11 -0
- package/dist/ai/custom-provider.js +113 -0
- package/dist/ai/index.d.ts +7 -0
- package/dist/ai/index.js +42 -0
- package/dist/ai/openai-provider.d.ts +18 -0
- package/dist/ai/openai-provider.js +161 -0
- package/dist/config/index.d.ts +20 -0
- package/dist/config/index.js +83 -0
- package/dist/conversation.d.ts +26 -0
- package/dist/conversation.js +126 -0
- package/dist/index.d.ts +10 -0
- package/dist/index.js +72 -0
- package/dist/mcp/index.d.ts +48 -0
- package/dist/mcp/index.js +175 -0
- package/dist/sandbox/index.d.ts +31 -0
- package/dist/sandbox/index.js +197 -0
- package/dist/skills/index.d.ts +16 -0
- package/dist/skills/index.js +169 -0
- package/dist/tools/ask-user-question.d.ts +62 -0
- package/dist/tools/ask-user-question.js +71 -0
- package/dist/tools/base.d.ts +16 -0
- package/dist/tools/base.js +39 -0
- package/dist/tools/glob.d.ts +27 -0
- package/dist/tools/glob.js +125 -0
- package/dist/tools/image-read.d.ts +42 -0
- package/dist/tools/image-read.js +125 -0
- package/dist/tools/index.d.ts +27 -0
- package/dist/tools/index.js +127 -0
- package/dist/tools/list-directory.d.ts +28 -0
- package/dist/tools/list-directory.js +94 -0
- package/dist/tools/pdf-extract.d.ts +32 -0
- package/dist/tools/pdf-extract.js +130 -0
- package/dist/tools/read-file.d.ts +31 -0
- package/dist/tools/read-file.js +116 -0
- package/dist/tools/replace.d.ts +35 -0
- package/dist/tools/replace.js +93 -0
- package/dist/tools/run-shell-command.d.ts +35 -0
- package/dist/tools/run-shell-command.js +81 -0
- package/dist/tools/save-memory.d.ts +22 -0
- package/dist/tools/save-memory.js +91 -0
- package/dist/tools/search-file-content.d.ts +42 -0
- package/dist/tools/search-file-content.js +153 -0
- package/dist/tools/task.d.ts +46 -0
- package/dist/tools/task.js +54 -0
- package/dist/tools/web-fetch.d.ts +26 -0
- package/dist/tools/web-fetch.js +81 -0
- package/dist/tools/web-search.d.ts +35 -0
- package/dist/tools/web-search.js +86 -0
- package/dist/tools/write-file.d.ts +25 -0
- package/dist/tools/write-file.js +76 -0
- package/dist/types/index.d.ts +166 -0
- package/dist/types/index.js +43 -0
- package/package.json +54 -0
|
@@ -0,0 +1,193 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
3
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
4
|
+
};
|
|
5
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
+
exports.ChineseProvider = exports.DEFAULT_CHINESE_MODELS = exports.CHINESE_MODELS = void 0;
|
|
7
|
+
const axios_1 = __importDefault(require("axios"));
|
|
8
|
+
// 国内模型配置
|
|
9
|
+
exports.CHINESE_MODELS = {
|
|
10
|
+
deepseek: {
|
|
11
|
+
name: 'DeepSeek (深度求索)',
|
|
12
|
+
baseUrl: 'https://api.deepseek.com/v1',
|
|
13
|
+
models: [
|
|
14
|
+
{ id: 'deepseek-chat', name: 'DeepSeek Chat', desc: '通用对话模型' },
|
|
15
|
+
{ id: 'deepseek-reasoner', name: 'DeepSeek Reasoner', desc: '推理增强模型 (显示思考过程)' }
|
|
16
|
+
]
|
|
17
|
+
},
|
|
18
|
+
qwen: {
|
|
19
|
+
name: '通义千问 (Qwen)',
|
|
20
|
+
baseUrl: 'https://dashscope.aliyuncs.com/compatible-mode/v1',
|
|
21
|
+
models: [
|
|
22
|
+
{ id: 'qwen-turbo', name: 'Qwen Turbo', desc: '快速响应' },
|
|
23
|
+
{ id: 'qwen-plus', name: 'Qwen Plus', desc: '均衡性能' },
|
|
24
|
+
{ id: 'qwen-max', name: 'Qwen Max', desc: '最强能力' },
|
|
25
|
+
{ id: 'qwen-long', name: 'Qwen Long', desc: '长文本处理' }
|
|
26
|
+
]
|
|
27
|
+
},
|
|
28
|
+
zhipu: {
|
|
29
|
+
name: '智谱 GLM',
|
|
30
|
+
baseUrl: 'https://open.bigmodel.cn/api/paas/v4',
|
|
31
|
+
models: [
|
|
32
|
+
{ id: 'glm-4-plus', name: 'GLM-4 Plus', desc: '最新旗舰模型' },
|
|
33
|
+
{ id: 'glm-4', name: 'GLM-4', desc: '主力模型' },
|
|
34
|
+
{ id: 'glm-4-flash', name: 'GLM-4 Flash', desc: '快速响应' },
|
|
35
|
+
{ id: 'glm-4-long', name: 'GLM-4 Long', desc: '长文本处理' }
|
|
36
|
+
]
|
|
37
|
+
},
|
|
38
|
+
baichuan: {
|
|
39
|
+
name: '百川 (Baichuan)',
|
|
40
|
+
baseUrl: 'https://api.baichuan-ai.com/v1',
|
|
41
|
+
models: [
|
|
42
|
+
{ id: 'Baichuan4', name: 'Baichuan4', desc: '旗舰模型' },
|
|
43
|
+
{ id: 'Baichuan3-Turbo', name: 'Baichuan3 Turbo', desc: '快速响应' }
|
|
44
|
+
]
|
|
45
|
+
},
|
|
46
|
+
yi: {
|
|
47
|
+
name: '零一万物 (Yi)',
|
|
48
|
+
baseUrl: 'https://api.lingyiwanwu.com/v1',
|
|
49
|
+
models: [
|
|
50
|
+
{ id: 'yi-lightning', name: 'Yi Lightning', desc: '快速响应' },
|
|
51
|
+
{ id: 'yi-large', name: 'Yi Large', desc: '主力模型' },
|
|
52
|
+
{ id: 'yi-medium', name: 'Yi Medium', desc: '均衡性价比' }
|
|
53
|
+
]
|
|
54
|
+
},
|
|
55
|
+
moonshot: {
|
|
56
|
+
name: '月之暗面 (Moonshot)',
|
|
57
|
+
baseUrl: 'https://api.moonshot.cn/v1',
|
|
58
|
+
models: [
|
|
59
|
+
{ id: 'moonshot-v1-8k', name: 'Moonshot V1 8K', desc: '标准版' },
|
|
60
|
+
{ id: 'moonshot-v1-32k', name: 'Moonshot V1 32K', desc: '长文本版' },
|
|
61
|
+
{ id: 'moonshot-v1-128k', name: 'Moonshot V1 128K', desc: '超长文本版' }
|
|
62
|
+
]
|
|
63
|
+
},
|
|
64
|
+
siliconflow: {
|
|
65
|
+
name: 'SiliconFlow (一站式API)',
|
|
66
|
+
baseUrl: 'https://api.siliconflow.cn/v1',
|
|
67
|
+
models: [
|
|
68
|
+
{ id: 'Qwen/Qwen2.5-72B-Instruct', name: 'Qwen2.5 72B', desc: '通义千问开源版' },
|
|
69
|
+
{ id: 'deepseek-ai/DeepSeek-V2.5', name: 'DeepSeek V2.5', desc: 'DeepSeek开源版' },
|
|
70
|
+
{ id: 'THUDM/glm-4-9b-chat', name: 'GLM-4 9B', desc: '智谱开源版' }
|
|
71
|
+
]
|
|
72
|
+
}
|
|
73
|
+
};
|
|
74
|
+
// 默认推荐的国内模型
|
|
75
|
+
exports.DEFAULT_CHINESE_MODELS = [
|
|
76
|
+
{ provider: 'deepseek', model: 'deepseek-chat', reason: '高性价比, 支持工具调用' },
|
|
77
|
+
{ provider: 'qwen', model: 'qwen-plus', reason: '综合能力强, 中文优化' },
|
|
78
|
+
{ provider: 'zhipu', model: 'glm-4-flash', reason: '免费额度, 快速响应' }
|
|
79
|
+
];
|
|
80
|
+
class ChineseProvider {
|
|
81
|
+
name = 'chinese';
|
|
82
|
+
client;
|
|
83
|
+
apiKey;
|
|
84
|
+
defaultModel;
|
|
85
|
+
providerName;
|
|
86
|
+
constructor(apiKey, baseUrl, defaultModel, providerName) {
|
|
87
|
+
this.apiKey = apiKey;
|
|
88
|
+
this.defaultModel = defaultModel || 'deepseek-chat';
|
|
89
|
+
this.providerName = providerName || 'chinese';
|
|
90
|
+
this.client = axios_1.default.create({
|
|
91
|
+
baseURL: baseUrl,
|
|
92
|
+
headers: {
|
|
93
|
+
'Content-Type': 'application/json',
|
|
94
|
+
'Authorization': `Bearer ${apiKey}`
|
|
95
|
+
}
|
|
96
|
+
});
|
|
97
|
+
}
|
|
98
|
+
async chat(options) {
|
|
99
|
+
const response = await this.client.post('/chat/completions', {
|
|
100
|
+
model: this.defaultModel,
|
|
101
|
+
messages: options.messages,
|
|
102
|
+
tools: options.tools,
|
|
103
|
+
tool_choice: options.toolChoice,
|
|
104
|
+
max_tokens: options.maxTokens,
|
|
105
|
+
temperature: options.temperature,
|
|
106
|
+
stream: false
|
|
107
|
+
});
|
|
108
|
+
const choice = response.data.choices[0];
|
|
109
|
+
return {
|
|
110
|
+
id: response.data.id,
|
|
111
|
+
content: choice.message.content,
|
|
112
|
+
toolCalls: choice.message.tool_calls?.map((tc) => ({
|
|
113
|
+
id: tc.id,
|
|
114
|
+
type: 'function',
|
|
115
|
+
function: {
|
|
116
|
+
name: tc.function.name,
|
|
117
|
+
arguments: tc.function.arguments
|
|
118
|
+
}
|
|
119
|
+
})),
|
|
120
|
+
finishReason: choice.finish_reason,
|
|
121
|
+
usage: response.data.usage ? {
|
|
122
|
+
promptTokens: response.data.usage.prompt_tokens,
|
|
123
|
+
completionTokens: response.data.usage.completion_tokens,
|
|
124
|
+
totalTokens: response.data.usage.total_tokens
|
|
125
|
+
} : undefined,
|
|
126
|
+
// 支持思考过程 (如DeepSeek Reasoner)
|
|
127
|
+
reasoning: choice.message.reasoning_content || choice.reasoning_content
|
|
128
|
+
};
|
|
129
|
+
}
|
|
130
|
+
async chatStream(options, onChunk) {
|
|
131
|
+
const response = await this.client.post('/chat/completions', {
|
|
132
|
+
model: this.defaultModel,
|
|
133
|
+
messages: options.messages,
|
|
134
|
+
tools: options.tools,
|
|
135
|
+
tool_choice: options.toolChoice,
|
|
136
|
+
max_tokens: options.maxTokens,
|
|
137
|
+
temperature: options.temperature,
|
|
138
|
+
stream: true
|
|
139
|
+
}, {
|
|
140
|
+
responseType: 'stream'
|
|
141
|
+
});
|
|
142
|
+
const stream = response.data;
|
|
143
|
+
let buffer = '';
|
|
144
|
+
stream.on('data', (chunk) => {
|
|
145
|
+
buffer += chunk.toString();
|
|
146
|
+
const lines = buffer.split('\n');
|
|
147
|
+
buffer = lines.pop() || '';
|
|
148
|
+
for (const line of lines) {
|
|
149
|
+
const trimmed = line.trim();
|
|
150
|
+
if (trimmed.startsWith('data: ')) {
|
|
151
|
+
const data = trimmed.slice(6);
|
|
152
|
+
if (data === '[DONE]')
|
|
153
|
+
continue;
|
|
154
|
+
try {
|
|
155
|
+
const parsed = JSON.parse(data);
|
|
156
|
+
const delta = parsed.choices[0]?.delta;
|
|
157
|
+
const finishReason = parsed.choices[0]?.finish_reason;
|
|
158
|
+
// 处理思考过程 (DeepSeek等支持)
|
|
159
|
+
const reasoning = delta?.reasoning_content || delta?.reasoning;
|
|
160
|
+
onChunk({
|
|
161
|
+
id: parsed.id,
|
|
162
|
+
delta: {
|
|
163
|
+
content: delta?.content ?? undefined,
|
|
164
|
+
toolCalls: delta?.tool_calls,
|
|
165
|
+
role: delta?.role,
|
|
166
|
+
reasoning: reasoning
|
|
167
|
+
},
|
|
168
|
+
finishReason: finishReason || null
|
|
169
|
+
});
|
|
170
|
+
}
|
|
171
|
+
catch {
|
|
172
|
+
// Skip invalid JSON
|
|
173
|
+
}
|
|
174
|
+
}
|
|
175
|
+
}
|
|
176
|
+
});
|
|
177
|
+
return new Promise((resolve, reject) => {
|
|
178
|
+
stream.on('end', resolve);
|
|
179
|
+
stream.on('error', reject);
|
|
180
|
+
});
|
|
181
|
+
}
|
|
182
|
+
async getModels() {
|
|
183
|
+
try {
|
|
184
|
+
const response = await this.client.get('/models');
|
|
185
|
+
return response.data.data.map((m) => m.id).sort();
|
|
186
|
+
}
|
|
187
|
+
catch {
|
|
188
|
+
return [this.defaultModel];
|
|
189
|
+
}
|
|
190
|
+
}
|
|
191
|
+
}
|
|
192
|
+
exports.ChineseProvider = ChineseProvider;
|
|
193
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"chinese-provider.js","sourceRoot":"","sources":["../../src/ai/chinese-provider.ts"],"names":[],"mappings":"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","sourcesContent":["import axios, { AxiosInstance } from 'axios';\nimport {\n  AIProvider,\n  ChatCompletionOptions,\n  ChatCompletionResponse,\n  StreamChunk\n} from '../types';\n\n// 国内模型配置\nexport const CHINESE_MODELS = {\n  deepseek: {\n    name: 'DeepSeek (深度求索)',\n    baseUrl: 'https://api.deepseek.com/v1',\n    models: [\n      { id: 'deepseek-chat', name: 'DeepSeek Chat', desc: '通用对话模型' },\n      { id: 'deepseek-reasoner', name: 'DeepSeek Reasoner', desc: '推理增强模型 (显示思考过程)' }\n    ]\n  },\n  qwen: {\n    name: '通义千问 (Qwen)',\n    baseUrl: 'https://dashscope.aliyuncs.com/compatible-mode/v1',\n    models: [\n      { id: 'qwen-turbo', name: 'Qwen Turbo', desc: '快速响应' },\n      { id: 'qwen-plus', name: 'Qwen Plus', desc: '均衡性能' },\n      { id: 'qwen-max', name: 'Qwen Max', desc: '最强能力' },\n      { id: 'qwen-long', name: 'Qwen Long', desc: '长文本处理' }\n    ]\n  },\n  zhipu: {\n    name: '智谱 GLM',\n    baseUrl: 'https://open.bigmodel.cn/api/paas/v4',\n    models: [\n      { id: 'glm-4-plus', name: 'GLM-4 Plus', desc: '最新旗舰模型' },\n      { id: 'glm-4', name: 'GLM-4', desc: '主力模型' },\n      { id: 'glm-4-flash', name: 'GLM-4 Flash', desc: '快速响应' },\n      { id: 'glm-4-long', name: 'GLM-4 Long', desc: '长文本处理' }\n    ]\n  },\n  baichuan: {\n    name: '百川 (Baichuan)',\n    baseUrl: 'https://api.baichuan-ai.com/v1',\n    models: [\n      { id: 'Baichuan4', name: 'Baichuan4', desc: '旗舰模型' },\n      { id: 'Baichuan3-Turbo', name: 'Baichuan3 Turbo', desc: '快速响应' }\n    ]\n  },\n  yi: {\n    name: '零一万物 (Yi)',\n    baseUrl: 'https://api.lingyiwanwu.com/v1',\n    models: [\n      { id: 'yi-lightning', name: 'Yi Lightning', desc: '快速响应' },\n      { id: 'yi-large', name: 'Yi Large', desc: '主力模型' },\n      { id: 'yi-medium', name: 'Yi Medium', desc: '均衡性价比' }\n    ]\n  },\n  moonshot: {\n    name: '月之暗面 (Moonshot)',\n    baseUrl: 'https://api.moonshot.cn/v1',\n    models: [\n      { id: 'moonshot-v1-8k', name: 'Moonshot V1 8K', desc: '标准版' },\n      { id: 'moonshot-v1-32k', name: 'Moonshot V1 32K', desc: '长文本版' },\n      { id: 'moonshot-v1-128k', name: 'Moonshot V1 128K', desc: '超长文本版' }\n    ]\n  },\n  siliconflow: {\n    name: 'SiliconFlow (一站式API)',\n    baseUrl: 'https://api.siliconflow.cn/v1',\n    models: [\n      { id: 'Qwen/Qwen2.5-72B-Instruct', name: 'Qwen2.5 72B', desc: '通义千问开源版' },\n      { id: 'deepseek-ai/DeepSeek-V2.5', name: 'DeepSeek V2.5', desc: 'DeepSeek开源版' },\n      { id: 'THUDM/glm-4-9b-chat', name: 'GLM-4 9B', desc: '智谱开源版' }\n    ]\n  }\n} as const;\n\n// 默认推荐的国内模型\nexport const DEFAULT_CHINESE_MODELS = [\n  { provider: 'deepseek', model: 'deepseek-chat', reason: '高性价比, 支持工具调用' },\n  { provider: 'qwen', model: 'qwen-plus', reason: '综合能力强, 中文优化' },\n  { provider: 'zhipu', model: 'glm-4-flash', reason: '免费额度, 快速响应' }\n] as const;\n\nexport class ChineseProvider implements AIProvider {\n  name = 'chinese';\n  private client: AxiosInstance;\n  private apiKey: string;\n  private defaultModel: string;\n  private providerName: string;\n\n  constructor(\n    apiKey: string, \n    baseUrl: string, \n    defaultModel?: string,\n    providerName?: string\n  ) {\n    this.apiKey = apiKey;\n    this.defaultModel = defaultModel || 'deepseek-chat';\n    this.providerName = providerName || 'chinese';\n    this.client = axios.create({\n      baseURL: baseUrl,\n      headers: {\n        'Content-Type': 'application/json',\n        'Authorization': `Bearer ${apiKey}`\n      }\n    });\n  }\n\n  async chat(options: ChatCompletionOptions): Promise<ChatCompletionResponse> {\n    const response = await this.client.post('/chat/completions', {\n      model: this.defaultModel,\n      messages: options.messages,\n      tools: options.tools,\n      tool_choice: options.toolChoice,\n      max_tokens: options.maxTokens,\n      temperature: options.temperature,\n      stream: false\n    });\n\n    const choice = response.data.choices[0];\n    return {\n      id: response.data.id,\n      content: choice.message.content,\n      toolCalls: choice.message.tool_calls?.map((tc: any) => ({\n        id: tc.id,\n        type: 'function' as const,\n        function: {\n          name: tc.function.name,\n          arguments: tc.function.arguments\n        }\n      })),\n      finishReason: choice.finish_reason,\n      usage: response.data.usage ? {\n        promptTokens: response.data.usage.prompt_tokens,\n        completionTokens: response.data.usage.completion_tokens,\n        totalTokens: response.data.usage.total_tokens\n      } : undefined,\n      // 支持思考过程 (如DeepSeek Reasoner)\n      reasoning: choice.message.reasoning_content || choice.reasoning_content\n    };\n  }\n\n  async chatStream(\n    options: ChatCompletionOptions,\n    onChunk: (chunk: StreamChunk) => void\n  ): Promise<void> {\n    const response = await this.client.post('/chat/completions', {\n      model: this.defaultModel,\n      messages: options.messages,\n      tools: options.tools,\n      tool_choice: options.toolChoice,\n      max_tokens: options.maxTokens,\n      temperature: options.temperature,\n      stream: true\n    }, {\n      responseType: 'stream'\n    });\n\n    const stream = response.data;\n    let buffer = '';\n\n    stream.on('data', (chunk: Buffer) => {\n      buffer += chunk.toString();\n      const lines = buffer.split('\\n');\n      buffer = lines.pop() || '';\n\n      for (const line of lines) {\n        const trimmed = line.trim();\n        if (trimmed.startsWith('data: ')) {\n          const data = trimmed.slice(6);\n          if (data === '[DONE]') continue;\n\n          try {\n            const parsed = JSON.parse(data);\n            const delta = parsed.choices[0]?.delta;\n            const finishReason = parsed.choices[0]?.finish_reason;\n\n            // 处理思考过程 (DeepSeek等支持)\n            const reasoning = delta?.reasoning_content || delta?.reasoning;\n\n            onChunk({\n              id: parsed.id,\n              delta: {\n                content: delta?.content ?? undefined,\n                toolCalls: delta?.tool_calls,\n                role: delta?.role,\n                reasoning: reasoning\n              },\n              finishReason: finishReason || null\n            });\n          } catch {\n            // Skip invalid JSON\n          }\n        }\n      }\n    });\n\n    return new Promise((resolve, reject) => {\n      stream.on('end', resolve);\n      stream.on('error', reject);\n    });\n  }\n\n  async getModels(): Promise<string[]> {\n    try {\n      const response = await this.client.get('/models');\n      return response.data.data.map((m: any) => m.id).sort();\n    } catch {\n      return [this.defaultModel];\n    }\n  }\n}"]}
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
import { AIProvider, ChatCompletionOptions, ChatCompletionResponse, StreamChunk } from '../types';
|
|
2
|
+
export declare class CustomProvider implements AIProvider {
|
|
3
|
+
name: string;
|
|
4
|
+
private client;
|
|
5
|
+
private apiKey;
|
|
6
|
+
private defaultModel;
|
|
7
|
+
constructor(apiKey: string, baseUrl: string, defaultModel?: string);
|
|
8
|
+
chat(options: ChatCompletionOptions): Promise<ChatCompletionResponse>;
|
|
9
|
+
chatStream(options: ChatCompletionOptions, onChunk: (chunk: StreamChunk) => void): Promise<void>;
|
|
10
|
+
getModels(): Promise<string[]>;
|
|
11
|
+
}
|
|
@@ -0,0 +1,113 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
3
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
4
|
+
};
|
|
5
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
+
exports.CustomProvider = void 0;
|
|
7
|
+
const axios_1 = __importDefault(require("axios"));
|
|
8
|
+
class CustomProvider {
|
|
9
|
+
name = 'custom';
|
|
10
|
+
client;
|
|
11
|
+
apiKey;
|
|
12
|
+
defaultModel;
|
|
13
|
+
constructor(apiKey, baseUrl, defaultModel) {
|
|
14
|
+
this.apiKey = apiKey;
|
|
15
|
+
this.defaultModel = defaultModel || 'gpt-4o';
|
|
16
|
+
this.client = axios_1.default.create({
|
|
17
|
+
baseURL: baseUrl.endsWith('/v1') ? baseUrl : `${baseUrl}/v1`,
|
|
18
|
+
headers: {
|
|
19
|
+
'Content-Type': 'application/json',
|
|
20
|
+
'Authorization': `Bearer ${apiKey}`
|
|
21
|
+
}
|
|
22
|
+
});
|
|
23
|
+
}
|
|
24
|
+
async chat(options) {
|
|
25
|
+
const response = await this.client.post('/chat/completions', {
|
|
26
|
+
model: this.defaultModel,
|
|
27
|
+
messages: options.messages,
|
|
28
|
+
tools: options.tools,
|
|
29
|
+
tool_choice: options.toolChoice,
|
|
30
|
+
max_tokens: options.maxTokens,
|
|
31
|
+
temperature: options.temperature
|
|
32
|
+
});
|
|
33
|
+
const choice = response.data.choices[0];
|
|
34
|
+
return {
|
|
35
|
+
id: response.data.id,
|
|
36
|
+
content: choice.message.content,
|
|
37
|
+
toolCalls: choice.message.tool_calls?.map((tc) => ({
|
|
38
|
+
id: tc.id,
|
|
39
|
+
type: 'function',
|
|
40
|
+
function: {
|
|
41
|
+
name: tc.function.name,
|
|
42
|
+
arguments: tc.function.arguments
|
|
43
|
+
}
|
|
44
|
+
})),
|
|
45
|
+
finishReason: choice.finish_reason,
|
|
46
|
+
usage: response.data.usage ? {
|
|
47
|
+
promptTokens: response.data.usage.prompt_tokens,
|
|
48
|
+
completionTokens: response.data.usage.completion_tokens,
|
|
49
|
+
totalTokens: response.data.usage.total_tokens
|
|
50
|
+
} : undefined
|
|
51
|
+
};
|
|
52
|
+
}
|
|
53
|
+
async chatStream(options, onChunk) {
|
|
54
|
+
const response = await this.client.post('/chat/completions', {
|
|
55
|
+
model: this.defaultModel,
|
|
56
|
+
messages: options.messages,
|
|
57
|
+
tools: options.tools,
|
|
58
|
+
tool_choice: options.toolChoice,
|
|
59
|
+
max_tokens: options.maxTokens,
|
|
60
|
+
temperature: options.temperature,
|
|
61
|
+
stream: true
|
|
62
|
+
}, {
|
|
63
|
+
responseType: 'stream'
|
|
64
|
+
});
|
|
65
|
+
const stream = response.data;
|
|
66
|
+
let buffer = '';
|
|
67
|
+
stream.on('data', (chunk) => {
|
|
68
|
+
buffer += chunk.toString();
|
|
69
|
+
const lines = buffer.split('\n');
|
|
70
|
+
buffer = lines.pop() || '';
|
|
71
|
+
for (const line of lines) {
|
|
72
|
+
const trimmed = line.trim();
|
|
73
|
+
if (trimmed.startsWith('data: ')) {
|
|
74
|
+
const data = trimmed.slice(6);
|
|
75
|
+
if (data === '[DONE]')
|
|
76
|
+
continue;
|
|
77
|
+
try {
|
|
78
|
+
const parsed = JSON.parse(data);
|
|
79
|
+
const delta = parsed.choices[0]?.delta;
|
|
80
|
+
const finishReason = parsed.choices[0]?.finish_reason;
|
|
81
|
+
onChunk({
|
|
82
|
+
id: parsed.id,
|
|
83
|
+
delta: {
|
|
84
|
+
content: delta?.content,
|
|
85
|
+
toolCalls: delta?.tool_calls,
|
|
86
|
+
role: delta?.role
|
|
87
|
+
},
|
|
88
|
+
finishReason: finishReason || null
|
|
89
|
+
});
|
|
90
|
+
}
|
|
91
|
+
catch {
|
|
92
|
+
// Skip invalid JSON
|
|
93
|
+
}
|
|
94
|
+
}
|
|
95
|
+
}
|
|
96
|
+
});
|
|
97
|
+
return new Promise((resolve, reject) => {
|
|
98
|
+
stream.on('end', resolve);
|
|
99
|
+
stream.on('error', reject);
|
|
100
|
+
});
|
|
101
|
+
}
|
|
102
|
+
async getModels() {
|
|
103
|
+
try {
|
|
104
|
+
const response = await this.client.get('/models');
|
|
105
|
+
return response.data.data.map((m) => m.id).sort();
|
|
106
|
+
}
|
|
107
|
+
catch {
|
|
108
|
+
return [this.defaultModel];
|
|
109
|
+
}
|
|
110
|
+
}
|
|
111
|
+
}
|
|
112
|
+
exports.CustomProvider = CustomProvider;
|
|
113
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"custom-provider.js","sourceRoot":"","sources":["../../src/ai/custom-provider.ts"],"names":[],"mappings":"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","sourcesContent":["import axios, { AxiosInstance } from 'axios';\nimport {\n  AIProvider,\n  ChatCompletionOptions,\n  ChatCompletionResponse,\n  StreamChunk\n} from '../types';\n\nexport class CustomProvider implements AIProvider {\n  name = 'custom';\n  private client: AxiosInstance;\n  private apiKey: string;\n  private defaultModel: string;\n\n  constructor(apiKey: string, baseUrl: string, defaultModel?: string) {\n    this.apiKey = apiKey;\n    this.defaultModel = defaultModel || 'gpt-4o';\n    this.client = axios.create({\n      baseURL: baseUrl.endsWith('/v1') ? baseUrl : `${baseUrl}/v1`,\n      headers: {\n        'Content-Type': 'application/json',\n        'Authorization': `Bearer ${apiKey}`\n      }\n    });\n  }\n\n  async chat(options: ChatCompletionOptions): Promise<ChatCompletionResponse> {\n    const response = await this.client.post('/chat/completions', {\n      model: this.defaultModel,\n      messages: options.messages,\n      tools: options.tools,\n      tool_choice: options.toolChoice,\n      max_tokens: options.maxTokens,\n      temperature: options.temperature\n    });\n\n    const choice = response.data.choices[0];\n    return {\n      id: response.data.id,\n      content: choice.message.content,\n      toolCalls: choice.message.tool_calls?.map((tc: any) => ({\n        id: tc.id,\n        type: 'function' as const,\n        function: {\n          name: tc.function.name,\n          arguments: tc.function.arguments\n        }\n      })),\n      finishReason: choice.finish_reason,\n      usage: response.data.usage ? {\n        promptTokens: response.data.usage.prompt_tokens,\n        completionTokens: response.data.usage.completion_tokens,\n        totalTokens: response.data.usage.total_tokens\n      } : undefined\n    };\n  }\n\n  async chatStream(\n    options: ChatCompletionOptions,\n    onChunk: (chunk: StreamChunk) => void\n  ): Promise<void> {\n    const response = await this.client.post('/chat/completions', {\n      model: this.defaultModel,\n      messages: options.messages,\n      tools: options.tools,\n      tool_choice: options.toolChoice,\n      max_tokens: options.maxTokens,\n      temperature: options.temperature,\n      stream: true\n    }, {\n      responseType: 'stream'\n    });\n\n    const stream = response.data;\n    let buffer = '';\n\n    stream.on('data', (chunk: Buffer) => {\n      buffer += chunk.toString();\n      const lines = buffer.split('\\n');\n      buffer = lines.pop() || '';\n\n      for (const line of lines) {\n        const trimmed = line.trim();\n        if (trimmed.startsWith('data: ')) {\n          const data = trimmed.slice(6);\n          if (data === '[DONE]') continue;\n\n          try {\n            const parsed = JSON.parse(data);\n            const delta = parsed.choices[0]?.delta;\n            const finishReason = parsed.choices[0]?.finish_reason;\n\n            onChunk({\n              id: parsed.id,\n              delta: {\n                content: delta?.content,\n                toolCalls: delta?.tool_calls,\n                role: delta?.role\n              },\n              finishReason: finishReason || null\n            });\n          } catch {\n            // Skip invalid JSON\n          }\n        }\n      }\n    });\n\n    return new Promise((resolve, reject) => {\n      stream.on('end', resolve);\n      stream.on('error', reject);\n    });\n  }\n\n  async getModels(): Promise<string[]> {\n    try {\n      const response = await this.client.get('/models');\n      return response.data.data.map((m: any) => m.id).sort();\n    } catch {\n      return [this.defaultModel];\n    }\n  }\n}\n"]}
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
import { AuthConfig } from '../types';
|
|
2
|
+
import { AIProvider } from './openai-provider';
|
|
3
|
+
export type { AIProvider } from './openai-provider';
|
|
4
|
+
export declare function createAIProvider(auth: AuthConfig, model?: string): AIProvider;
|
|
5
|
+
export { OpenAIProvider } from './openai-provider';
|
|
6
|
+
export { CustomProvider } from './custom-provider';
|
|
7
|
+
export { ChineseProvider } from './chinese-provider';
|
package/dist/ai/index.js
ADDED
|
@@ -0,0 +1,42 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.ChineseProvider = exports.CustomProvider = exports.OpenAIProvider = void 0;
|
|
4
|
+
exports.createAIProvider = createAIProvider;
|
|
5
|
+
const types_1 = require("../types");
|
|
6
|
+
const openai_provider_1 = require("./openai-provider");
|
|
7
|
+
const custom_provider_1 = require("./custom-provider");
|
|
8
|
+
const chinese_provider_1 = require("./chinese-provider");
|
|
9
|
+
function createAIProvider(auth, model) {
|
|
10
|
+
const defaultModel = model || auth.model || 'gpt-4o';
|
|
11
|
+
switch (auth.provider) {
|
|
12
|
+
case 'openai':
|
|
13
|
+
return new openai_provider_1.OpenAIProvider(auth.apiKey, auth.baseUrl, defaultModel);
|
|
14
|
+
case 'custom':
|
|
15
|
+
return new custom_provider_1.CustomProvider(auth.apiKey, auth.baseUrl || 'https://api.openai.com/v1', defaultModel);
|
|
16
|
+
// 国内模型提供商
|
|
17
|
+
case 'deepseek':
|
|
18
|
+
return new chinese_provider_1.ChineseProvider(auth.apiKey, auth.baseUrl || types_1.CHINESE_MODELS.deepseek.baseUrl, defaultModel, 'deepseek');
|
|
19
|
+
case 'qwen':
|
|
20
|
+
return new chinese_provider_1.ChineseProvider(auth.apiKey, auth.baseUrl || types_1.CHINESE_MODELS.qwen.baseUrl, defaultModel, 'qwen');
|
|
21
|
+
case 'zhipu':
|
|
22
|
+
return new chinese_provider_1.ChineseProvider(auth.apiKey, auth.baseUrl || types_1.CHINESE_MODELS.zhipu.baseUrl, defaultModel, 'zhipu');
|
|
23
|
+
case 'baichuan':
|
|
24
|
+
return new chinese_provider_1.ChineseProvider(auth.apiKey, auth.baseUrl || types_1.CHINESE_MODELS.baichuan.baseUrl, defaultModel, 'baichuan');
|
|
25
|
+
case 'yi':
|
|
26
|
+
return new chinese_provider_1.ChineseProvider(auth.apiKey, auth.baseUrl || types_1.CHINESE_MODELS.yi.baseUrl, defaultModel, 'yi');
|
|
27
|
+
case 'moonshot':
|
|
28
|
+
return new chinese_provider_1.ChineseProvider(auth.apiKey, auth.baseUrl || types_1.CHINESE_MODELS.moonshot.baseUrl, defaultModel, 'moonshot');
|
|
29
|
+
case 'siliconflow':
|
|
30
|
+
return new chinese_provider_1.ChineseProvider(auth.apiKey, auth.baseUrl || types_1.CHINESE_MODELS.siliconflow.baseUrl, defaultModel, 'siliconflow');
|
|
31
|
+
case 'oflow':
|
|
32
|
+
default:
|
|
33
|
+
return new custom_provider_1.CustomProvider(auth.apiKey, auth.baseUrl || 'https://api.openai.com/v1', defaultModel);
|
|
34
|
+
}
|
|
35
|
+
}
|
|
36
|
+
var openai_provider_2 = require("./openai-provider");
|
|
37
|
+
Object.defineProperty(exports, "OpenAIProvider", { enumerable: true, get: function () { return openai_provider_2.OpenAIProvider; } });
|
|
38
|
+
var custom_provider_2 = require("./custom-provider");
|
|
39
|
+
Object.defineProperty(exports, "CustomProvider", { enumerable: true, get: function () { return custom_provider_2.CustomProvider; } });
|
|
40
|
+
var chinese_provider_2 = require("./chinese-provider");
|
|
41
|
+
Object.defineProperty(exports, "ChineseProvider", { enumerable: true, get: function () { return chinese_provider_2.ChineseProvider; } });
|
|
42
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvYWkvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBT0EsNENBdUVDO0FBOUVELG9DQUE4RjtBQUM5Rix1REFBK0Q7QUFDL0QsdURBQW1EO0FBQ25ELHlEQUFxRDtBQUlyRCxTQUFnQixnQkFBZ0IsQ0FBQyxJQUFnQixFQUFFLEtBQWM7SUFDL0QsTUFBTSxZQUFZLEdBQUcsS0FBSyxJQUFJLElBQUksQ0FBQyxLQUFLLElBQUksUUFBUSxDQUFDO0lBRXJELFFBQVEsSUFBSSxDQUFDLFFBQVEsRUFBRSxDQUFDO1FBQ3RCLEtBQUssUUFBUTtZQUNYLE9BQU8sSUFBSSxnQ0FBYyxDQUFDLElBQUksQ0FBQyxNQUFPLEVBQUUsSUFBSSxDQUFDLE9BQU8sRUFBRSxZQUFZLENBQUMsQ0FBQztRQUV0RSxLQUFLLFFBQVE7WUFDWCxPQUFPLElBQUksZ0NBQWMsQ0FBQyxJQUFJLENBQUMsTUFBTyxFQUFFLElBQUksQ0FBQyxPQUFPLElBQUksMkJBQTJCLEVBQUUsWUFBWSxDQUFDLENBQUM7UUFFckcsVUFBVTtRQUNWLEtBQUssVUFBVTtZQUNiLE9BQU8sSUFBSSxrQ0FBZSxDQUN4QixJQUFJLENBQUMsTUFBTyxFQUNaLElBQUksQ0FBQyxPQUFPLElBQUksc0JBQWMsQ0FBQyxRQUFRLENBQUMsT0FBTyxFQUMvQyxZQUFZLEVBQ1osVUFBVSxDQUNYLENBQUM7UUFFSixLQUFLLE1BQU07WUFDVCxPQUFPLElBQUksa0NBQWUsQ0FDeEIsSUFBSSxDQUFDLE1BQU8sRUFDWixJQUFJLENBQUMsT0FBTyxJQUFJLHNCQUFjLENBQUMsSUFBSSxDQUFDLE9BQU8sRUFDM0MsWUFBWSxFQUNaLE1BQU0sQ0FDUCxDQUFDO1FBRUosS0FBSyxPQUFPO1lBQ1YsT0FBTyxJQUFJLGtDQUFlLENBQ3hCLElBQUksQ0FBQyxNQUFPLEVBQ1osSUFBSSxDQUFDLE9BQU8sSUFBSSxzQkFBYyxDQUFDLEtBQUssQ0FBQyxPQUFPLEVBQzVDLFlBQVksRUFDWixPQUFPLENBQ1IsQ0FBQztRQUVKLEtBQUssVUFBVTtZQUNiLE9BQU8sSUFBSSxrQ0FBZSxDQUN4QixJQUFJLENBQUMsTUFBTyxFQUNaLElBQUksQ0FBQyxPQUFPLElBQUksc0JBQWMsQ0FBQyxRQUFRLENBQUMsT0FBTyxFQUMvQyxZQUFZLEVBQ1osVUFBVSxDQUNYLENBQUM7UUFFSixLQUFLLElBQUk7WUFDUCxPQUFPLElBQUksa0NBQWUsQ0FDeEIsSUFBSSxDQUFDLE1BQU8sRUFDWixJQUFJLENBQUMsT0FBTyxJQUFJLHNCQUFjLENBQUMsRUFBRSxDQUFDLE9BQU8sRUFDekMsWUFBWSxFQUNaLElBQUksQ0FDTCxDQUFDO1FBRUosS0FBSyxVQUFVO1lBQ2IsT0FBTyxJQUFJLGtDQUFlLENBQ3hCLElBQUksQ0FBQyxNQUFPLEVBQ1osSUFBSSxDQUFDLE9BQU8sSUFBSSxzQkFBYyxDQUFDLFFBQVEsQ0FBQyxPQUFPLEVBQy9DLFlBQVksRUFDWixVQUFVLENBQ1gsQ0FBQztRQUVKLEtBQUssYUFBYTtZQUNoQixPQUFPLElBQUksa0NBQWUsQ0FDeEIsSUFBSSxDQUFDLE1BQU8sRUFDWixJQUFJLENBQUMsT0FBTyxJQUFJLHNCQUFjLENBQUMsV0FBVyxDQUFDLE9BQU8sRUFDbEQsWUFBWSxFQUNaLGFBQWEsQ0FDZCxDQUFDO1FBRUosS0FBSyxPQUFPLENBQUM7UUFDYjtZQUNFLE9BQU8sSUFBSSxnQ0FBYyxDQUFDLElBQUksQ0FBQyxNQUFPLEVBQUUsSUFBSSxDQUFDLE9BQU8sSUFBSSwyQkFBMkIsRUFBRSxZQUFZLENBQUMsQ0FBQztJQUN2RyxDQUFDO0FBQ0gsQ0FBQztBQUVELHFEQUFtRDtBQUExQyxpSEFBQSxjQUFjLE9BQUE7QUFDdkIscURBQW1EO0FBQTFDLGlIQUFBLGNBQWMsT0FBQTtBQUN2Qix1REFBcUQ7QUFBNUMsbUhBQUEsZUFBZSxPQUFBIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQXV0aENvbmZpZywgQ0hJTkVTRV9NT0RFTFMsIENoaW5lc2VQcm92aWRlciBhcyBDaGluZXNlUHJvdmlkZXJUeXBlIH0gZnJvbSAnLi4vdHlwZXMnO1xuaW1wb3J0IHsgQUlQcm92aWRlciwgT3BlbkFJUHJvdmlkZXIgfSBmcm9tICcuL29wZW5haS1wcm92aWRlcic7XG5pbXBvcnQgeyBDdXN0b21Qcm92aWRlciB9IGZyb20gJy4vY3VzdG9tLXByb3ZpZGVyJztcbmltcG9ydCB7IENoaW5lc2VQcm92aWRlciB9IGZyb20gJy4vY2hpbmVzZS1wcm92aWRlcic7XG5cbmV4cG9ydCB0eXBlIHsgQUlQcm92aWRlciB9IGZyb20gJy4vb3BlbmFpLXByb3ZpZGVyJztcblxuZXhwb3J0IGZ1bmN0aW9uIGNyZWF0ZUFJUHJvdmlkZXIoYXV0aDogQXV0aENvbmZpZywgbW9kZWw/OiBzdHJpbmcpOiBBSVByb3ZpZGVyIHtcbiAgY29uc3QgZGVmYXVsdE1vZGVsID0gbW9kZWwgfHwgYXV0aC5tb2RlbCB8fCAnZ3B0LTRvJztcblxuICBzd2l0Y2ggKGF1dGgucHJvdmlkZXIpIHtcbiAgICBjYXNlICdvcGVuYWknOlxuICAgICAgcmV0dXJuIG5ldyBPcGVuQUlQcm92aWRlcihhdXRoLmFwaUtleSEsIGF1dGguYmFzZVVybCwgZGVmYXVsdE1vZGVsKTtcbiAgICBcbiAgICBjYXNlICdjdXN0b20nOlxuICAgICAgcmV0dXJuIG5ldyBDdXN0b21Qcm92aWRlcihhdXRoLmFwaUtleSEsIGF1dGguYmFzZVVybCB8fCAnaHR0cHM6Ly9hcGkub3BlbmFpLmNvbS92MScsIGRlZmF1bHRNb2RlbCk7XG4gICAgXG4gICAgLy8g5Zu95YaF5qih5Z6L5o+Q5L6b5ZWGXG4gICAgY2FzZSAnZGVlcHNlZWsnOlxuICAgICAgcmV0dXJuIG5ldyBDaGluZXNlUHJvdmlkZXIoXG4gICAgICAgIGF1dGguYXBpS2V5ISwgXG4gICAgICAgIGF1dGguYmFzZVVybCB8fCBDSElORVNFX01PREVMUy5kZWVwc2Vlay5iYXNlVXJsLFxuICAgICAgICBkZWZhdWx0TW9kZWwsXG4gICAgICAgICdkZWVwc2VlaydcbiAgICAgICk7XG4gICAgXG4gICAgY2FzZSAncXdlbic6XG4gICAgICByZXR1cm4gbmV3IENoaW5lc2VQcm92aWRlcihcbiAgICAgICAgYXV0aC5hcGlLZXkhLCBcbiAgICAgICAgYXV0aC5iYXNlVXJsIHx8IENISU5FU0VfTU9ERUxTLnF3ZW4uYmFzZVVybCxcbiAgICAgICAgZGVmYXVsdE1vZGVsLFxuICAgICAgICAncXdlbidcbiAgICAgICk7XG4gICAgXG4gICAgY2FzZSAnemhpcHUnOlxuICAgICAgcmV0dXJuIG5ldyBDaGluZXNlUHJvdmlkZXIoXG4gICAgICAgIGF1dGguYXBpS2V5ISwgXG4gICAgICAgIGF1dGguYmFzZVVybCB8fCBDSElORVNFX01PREVMUy56aGlwdS5iYXNlVXJsLFxuICAgICAgICBkZWZhdWx0TW9kZWwsXG4gICAgICAgICd6aGlwdSdcbiAgICAgICk7XG4gICAgXG4gICAgY2FzZSAnYmFpY2h1YW4nOlxuICAgICAgcmV0dXJuIG5ldyBDaGluZXNlUHJvdmlkZXIoXG4gICAgICAgIGF1dGguYXBpS2V5ISwgXG4gICAgICAgIGF1dGguYmFzZVVybCB8fCBDSElORVNFX01PREVMUy5iYWljaHVhbi5iYXNlVXJsLFxuICAgICAgICBkZWZhdWx0TW9kZWwsXG4gICAgICAgICdiYWljaHVhbidcbiAgICAgICk7XG4gICAgXG4gICAgY2FzZSAneWknOlxuICAgICAgcmV0dXJuIG5ldyBDaGluZXNlUHJvdmlkZXIoXG4gICAgICAgIGF1dGguYXBpS2V5ISwgXG4gICAgICAgIGF1dGguYmFzZVVybCB8fCBDSElORVNFX01PREVMUy55aS5iYXNlVXJsLFxuICAgICAgICBkZWZhdWx0TW9kZWwsXG4gICAgICAgICd5aSdcbiAgICAgICk7XG4gICAgXG4gICAgY2FzZSAnbW9vbnNob3QnOlxuICAgICAgcmV0dXJuIG5ldyBDaGluZXNlUHJvdmlkZXIoXG4gICAgICAgIGF1dGguYXBpS2V5ISwgXG4gICAgICAgIGF1dGguYmFzZVVybCB8fCBDSElORVNFX01PREVMUy5tb29uc2hvdC5iYXNlVXJsLFxuICAgICAgICBkZWZhdWx0TW9kZWwsXG4gICAgICAgICdtb29uc2hvdCdcbiAgICAgICk7XG4gICAgXG4gICAgY2FzZSAnc2lsaWNvbmZsb3cnOlxuICAgICAgcmV0dXJuIG5ldyBDaGluZXNlUHJvdmlkZXIoXG4gICAgICAgIGF1dGguYXBpS2V5ISwgXG4gICAgICAgIGF1dGguYmFzZVVybCB8fCBDSElORVNFX01PREVMUy5zaWxpY29uZmxvdy5iYXNlVXJsLFxuICAgICAgICBkZWZhdWx0TW9kZWwsXG4gICAgICAgICdzaWxpY29uZmxvdydcbiAgICAgICk7XG4gICAgXG4gICAgY2FzZSAnb2Zsb3cnOlxuICAgIGRlZmF1bHQ6XG4gICAgICByZXR1cm4gbmV3IEN1c3RvbVByb3ZpZGVyKGF1dGguYXBpS2V5ISwgYXV0aC5iYXNlVXJsIHx8ICdodHRwczovL2FwaS5vcGVuYWkuY29tL3YxJywgZGVmYXVsdE1vZGVsKTtcbiAgfVxufVxuXG5leHBvcnQgeyBPcGVuQUlQcm92aWRlciB9IGZyb20gJy4vb3BlbmFpLXByb3ZpZGVyJztcbmV4cG9ydCB7IEN1c3RvbVByb3ZpZGVyIH0gZnJvbSAnLi9jdXN0b20tcHJvdmlkZXInO1xuZXhwb3J0IHsgQ2hpbmVzZVByb3ZpZGVyIH0gZnJvbSAnLi9jaGluZXNlLXByb3ZpZGVyJzsiXX0=
|
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
import { ChatCompletionOptions, ChatCompletionResponse, StreamChunk } from '../types';
|
|
2
|
+
export interface AIProvider {
|
|
3
|
+
name: string;
|
|
4
|
+
chat(options: ChatCompletionOptions): Promise<ChatCompletionResponse>;
|
|
5
|
+
chatStream(options: ChatCompletionOptions, onChunk: (chunk: StreamChunk) => void): Promise<void>;
|
|
6
|
+
getModels(): Promise<string[]>;
|
|
7
|
+
}
|
|
8
|
+
export declare class OpenAIProvider implements AIProvider {
|
|
9
|
+
name: string;
|
|
10
|
+
private client;
|
|
11
|
+
private defaultModel;
|
|
12
|
+
private models;
|
|
13
|
+
constructor(apiKey: string, baseUrl?: string, defaultModel?: string);
|
|
14
|
+
chat(options: ChatCompletionOptions): Promise<ChatCompletionResponse>;
|
|
15
|
+
chatStream(options: ChatCompletionOptions, onChunk: (chunk: StreamChunk) => void): Promise<void>;
|
|
16
|
+
getModels(): Promise<string[]>;
|
|
17
|
+
private convertMessages;
|
|
18
|
+
}
|
|
@@ -0,0 +1,161 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
3
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
4
|
+
};
|
|
5
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
+
exports.OpenAIProvider = void 0;
|
|
7
|
+
const openai_1 = __importDefault(require("openai"));
|
|
8
|
+
class OpenAIProvider {
|
|
9
|
+
name = 'openai';
|
|
10
|
+
client;
|
|
11
|
+
defaultModel;
|
|
12
|
+
models = [
|
|
13
|
+
'gpt-4o',
|
|
14
|
+
'gpt-4o-mini',
|
|
15
|
+
'gpt-4-turbo',
|
|
16
|
+
'gpt-4',
|
|
17
|
+
'gpt-3.5-turbo',
|
|
18
|
+
'o1-preview',
|
|
19
|
+
'o1-mini'
|
|
20
|
+
];
|
|
21
|
+
constructor(apiKey, baseUrl, defaultModel) {
|
|
22
|
+
this.client = new openai_1.default({
|
|
23
|
+
apiKey,
|
|
24
|
+
baseURL: baseUrl || 'https://api.openai.com/v1'
|
|
25
|
+
});
|
|
26
|
+
this.defaultModel = defaultModel || 'gpt-4o';
|
|
27
|
+
}
|
|
28
|
+
async chat(options) {
|
|
29
|
+
const response = await this.client.chat.completions.create({
|
|
30
|
+
model: this.defaultModel,
|
|
31
|
+
messages: this.convertMessages(options.messages),
|
|
32
|
+
tools: options.tools,
|
|
33
|
+
tool_choice: options.toolChoice,
|
|
34
|
+
max_tokens: options.maxTokens,
|
|
35
|
+
temperature: options.temperature
|
|
36
|
+
});
|
|
37
|
+
const choice = response.choices[0];
|
|
38
|
+
return {
|
|
39
|
+
id: response.id,
|
|
40
|
+
content: choice.message.content,
|
|
41
|
+
toolCalls: choice.message.tool_calls?.map(tc => ({
|
|
42
|
+
id: tc.id,
|
|
43
|
+
type: 'function',
|
|
44
|
+
function: {
|
|
45
|
+
name: tc.function.name,
|
|
46
|
+
arguments: tc.function.arguments
|
|
47
|
+
}
|
|
48
|
+
})),
|
|
49
|
+
finishReason: choice.finish_reason,
|
|
50
|
+
usage: response.usage ? {
|
|
51
|
+
promptTokens: response.usage.prompt_tokens,
|
|
52
|
+
completionTokens: response.usage.completion_tokens,
|
|
53
|
+
totalTokens: response.usage.total_tokens
|
|
54
|
+
} : undefined
|
|
55
|
+
};
|
|
56
|
+
}
|
|
57
|
+
async chatStream(options, onChunk) {
|
|
58
|
+
const stream = await this.client.chat.completions.create({
|
|
59
|
+
model: this.defaultModel,
|
|
60
|
+
messages: this.convertMessages(options.messages),
|
|
61
|
+
tools: options.tools,
|
|
62
|
+
tool_choice: options.toolChoice,
|
|
63
|
+
max_tokens: options.maxTokens,
|
|
64
|
+
temperature: options.temperature,
|
|
65
|
+
stream: true
|
|
66
|
+
});
|
|
67
|
+
let currentToolCalls = new Map();
|
|
68
|
+
for await (const chunk of stream) {
|
|
69
|
+
const delta = chunk.choices[0]?.delta;
|
|
70
|
+
const finishReason = chunk.choices[0]?.finish_reason;
|
|
71
|
+
// Handle tool calls streaming
|
|
72
|
+
if (delta?.tool_calls) {
|
|
73
|
+
for (const tc of delta.tool_calls) {
|
|
74
|
+
const index = tc.index;
|
|
75
|
+
if (!currentToolCalls.has(index)) {
|
|
76
|
+
currentToolCalls.set(index, {
|
|
77
|
+
id: tc.id || '',
|
|
78
|
+
type: 'function',
|
|
79
|
+
function: { name: '', arguments: '' }
|
|
80
|
+
});
|
|
81
|
+
}
|
|
82
|
+
const current = currentToolCalls.get(index);
|
|
83
|
+
if (tc.id)
|
|
84
|
+
current.id = tc.id;
|
|
85
|
+
if (tc.function?.name)
|
|
86
|
+
current.function.name = tc.function.name;
|
|
87
|
+
if (tc.function?.arguments)
|
|
88
|
+
current.function.arguments += tc.function.arguments;
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
onChunk({
|
|
92
|
+
id: chunk.id,
|
|
93
|
+
delta: {
|
|
94
|
+
content: delta?.content ?? undefined,
|
|
95
|
+
toolCalls: Array.from(currentToolCalls.values()),
|
|
96
|
+
role: delta?.role
|
|
97
|
+
},
|
|
98
|
+
finishReason: finishReason || null
|
|
99
|
+
});
|
|
100
|
+
if (finishReason) {
|
|
101
|
+
currentToolCalls.clear();
|
|
102
|
+
}
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
async getModels() {
|
|
106
|
+
try {
|
|
107
|
+
const response = await this.client.models.list();
|
|
108
|
+
return response.data
|
|
109
|
+
.filter(m => m.id.includes('gpt') || m.id.includes('o1'))
|
|
110
|
+
.map(m => m.id)
|
|
111
|
+
.sort();
|
|
112
|
+
}
|
|
113
|
+
catch {
|
|
114
|
+
return this.models;
|
|
115
|
+
}
|
|
116
|
+
}
|
|
117
|
+
convertMessages(messages) {
|
|
118
|
+
return messages.map(msg => {
|
|
119
|
+
if (msg.role === 'system') {
|
|
120
|
+
return { role: 'system', content: msg.content };
|
|
121
|
+
}
|
|
122
|
+
if (msg.role === 'user') {
|
|
123
|
+
if (Array.isArray(msg.content)) {
|
|
124
|
+
return {
|
|
125
|
+
role: 'user',
|
|
126
|
+
content: msg.content.map(c => {
|
|
127
|
+
if (c.type === 'text')
|
|
128
|
+
return { type: 'text', text: c.text };
|
|
129
|
+
return { type: 'image_url', image_url: { url: c.imageUrl.url } };
|
|
130
|
+
})
|
|
131
|
+
};
|
|
132
|
+
}
|
|
133
|
+
return { role: 'user', content: msg.content };
|
|
134
|
+
}
|
|
135
|
+
if (msg.role === 'assistant') {
|
|
136
|
+
return {
|
|
137
|
+
role: 'assistant',
|
|
138
|
+
content: msg.content,
|
|
139
|
+
tool_calls: msg.toolCalls?.map(tc => ({
|
|
140
|
+
id: tc.id,
|
|
141
|
+
type: 'function',
|
|
142
|
+
function: {
|
|
143
|
+
name: tc.function.name,
|
|
144
|
+
arguments: tc.function.arguments
|
|
145
|
+
}
|
|
146
|
+
}))
|
|
147
|
+
};
|
|
148
|
+
}
|
|
149
|
+
if (msg.role === 'tool') {
|
|
150
|
+
return {
|
|
151
|
+
role: 'tool',
|
|
152
|
+
tool_call_id: msg.toolCallId,
|
|
153
|
+
content: msg.content
|
|
154
|
+
};
|
|
155
|
+
}
|
|
156
|
+
return { role: msg.role, content: msg.content };
|
|
157
|
+
});
|
|
158
|
+
}
|
|
159
|
+
}
|
|
160
|
+
exports.OpenAIProvider = OpenAIProvider;
|
|
161
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"openai-provider.js","sourceRoot":"","sources":["../../src/ai/openai-provider.ts"],"names":[],"mappings":"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","sourcesContent":["import OpenAI from 'openai';\nimport {\n  ChatCompletionOptions,\n  ChatCompletionResponse,\n  StreamChunk,\n  ToolCall\n} from '../types';\n\nexport interface AIProvider {\n  name: string;\n  chat(options: ChatCompletionOptions): Promise<ChatCompletionResponse>;\n  chatStream(options: ChatCompletionOptions, onChunk: (chunk: StreamChunk) => void): Promise<void>;\n  getModels(): Promise<string[]>;\n}\n\nexport class OpenAIProvider implements AIProvider {\n  name = 'openai';\n  private client: OpenAI;\n  private defaultModel: string;\n  private models: string[] = [\n    'gpt-4o',\n    'gpt-4o-mini',\n    'gpt-4-turbo',\n    'gpt-4',\n    'gpt-3.5-turbo',\n    'o1-preview',\n    'o1-mini'\n  ];\n\n  constructor(apiKey: string, baseUrl?: string, defaultModel?: string) {\n    this.client = new OpenAI({\n      apiKey,\n      baseURL: baseUrl || 'https://api.openai.com/v1'\n    });\n    this.defaultModel = defaultModel || 'gpt-4o';\n  }\n\n  async chat(options: ChatCompletionOptions): Promise<ChatCompletionResponse> {\n    const response = await this.client.chat.completions.create({\n      model: this.defaultModel,\n      messages: this.convertMessages(options.messages),\n      tools: options.tools,\n      tool_choice: options.toolChoice,\n      max_tokens: options.maxTokens,\n      temperature: options.temperature\n    });\n\n    const choice = response.choices[0];\n    return {\n      id: response.id,\n      content: choice.message.content,\n      toolCalls: choice.message.tool_calls?.map(tc => ({\n        id: tc.id,\n        type: 'function' as const,\n        function: {\n          name: tc.function.name,\n          arguments: tc.function.arguments\n        }\n      })),\n      finishReason: choice.finish_reason,\n      usage: response.usage ? {\n        promptTokens: response.usage.prompt_tokens,\n        completionTokens: response.usage.completion_tokens,\n        totalTokens: response.usage.total_tokens\n      } : undefined\n    };\n  }\n\n  async chatStream(\n    options: ChatCompletionOptions,\n    onChunk: (chunk: StreamChunk) => void\n  ): Promise<void> {\n    const stream = await this.client.chat.completions.create({\n      model: this.defaultModel,\n      messages: this.convertMessages(options.messages),\n      tools: options.tools,\n      tool_choice: options.toolChoice,\n      max_tokens: options.maxTokens,\n      temperature: options.temperature,\n      stream: true\n    });\n\n    let currentToolCalls: Map<number, Partial<ToolCall>> = new Map();\n\n    for await (const chunk of stream) {\n      const delta = chunk.choices[0]?.delta;\n      const finishReason = chunk.choices[0]?.finish_reason;\n\n      // Handle tool calls streaming\n      if (delta?.tool_calls) {\n        for (const tc of delta.tool_calls) {\n          const index = tc.index;\n          if (!currentToolCalls.has(index)) {\n            currentToolCalls.set(index, {\n              id: tc.id || '',\n              type: 'function',\n              function: { name: '', arguments: '' }\n            });\n          }\n          const current = currentToolCalls.get(index)!;\n          if (tc.id) current.id = tc.id;\n          if (tc.function?.name) current.function!.name = tc.function.name;\n          if (tc.function?.arguments) current.function!.arguments += tc.function.arguments;\n        }\n      }\n\n      onChunk({\n        id: chunk.id,\n        delta: {\n          content: delta?.content ?? undefined,\n          toolCalls: Array.from(currentToolCalls.values()),\n          role: delta?.role\n        },\n        finishReason: finishReason || null\n      });\n\n      if (finishReason) {\n        currentToolCalls.clear();\n      }\n    }\n  }\n\n  async getModels(): Promise<string[]> {\n    try {\n      const response = await this.client.models.list();\n      return response.data\n        .filter(m => m.id.includes('gpt') || m.id.includes('o1'))\n        .map(m => m.id)\n        .sort();\n    } catch {\n      return this.models;\n    }\n  }\n\n  private convertMessages(messages: ChatCompletionOptions['messages']): OpenAI.ChatCompletionMessageParam[] {\n    return messages.map(msg => {\n      if (msg.role === 'system') {\n        return { role: 'system', content: msg.content as string };\n      }\n      if (msg.role === 'user') {\n        if (Array.isArray(msg.content)) {\n          return {\n            role: 'user',\n            content: msg.content.map(c => {\n              if (c.type === 'text') return { type: 'text', text: c.text };\n              return { type: 'image_url', image_url: { url: c.imageUrl!.url } };\n            })\n          };\n        }\n        return { role: 'user', content: msg.content as string };\n      }\n      if (msg.role === 'assistant') {\n        return {\n          role: 'assistant',\n          content: msg.content as string,\n          tool_calls: msg.toolCalls?.map(tc => ({\n            id: tc.id,\n            type: 'function' as const,\n            function: {\n              name: tc.function.name,\n              arguments: tc.function.arguments\n            }\n          }))\n        };\n      }\n      if (msg.role === 'tool') {\n        return {\n          role: 'tool',\n          tool_call_id: msg.toolCallId!,\n          content: msg.content as string\n        };\n      }\n      return { role: msg.role, content: msg.content as string };\n    }) as OpenAI.ChatCompletionMessageParam[];\n  }\n}\n"]}
|