@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.
Files changed (59) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +81 -0
  3. package/dist/agents/index.d.ts +35 -0
  4. package/dist/agents/index.js +233 -0
  5. package/dist/ai/chinese-provider.d.ts +146 -0
  6. package/dist/ai/chinese-provider.js +193 -0
  7. package/dist/ai/custom-provider.d.ts +11 -0
  8. package/dist/ai/custom-provider.js +113 -0
  9. package/dist/ai/index.d.ts +7 -0
  10. package/dist/ai/index.js +42 -0
  11. package/dist/ai/openai-provider.d.ts +18 -0
  12. package/dist/ai/openai-provider.js +161 -0
  13. package/dist/config/index.d.ts +20 -0
  14. package/dist/config/index.js +83 -0
  15. package/dist/conversation.d.ts +26 -0
  16. package/dist/conversation.js +126 -0
  17. package/dist/index.d.ts +10 -0
  18. package/dist/index.js +72 -0
  19. package/dist/mcp/index.d.ts +48 -0
  20. package/dist/mcp/index.js +175 -0
  21. package/dist/sandbox/index.d.ts +31 -0
  22. package/dist/sandbox/index.js +197 -0
  23. package/dist/skills/index.d.ts +16 -0
  24. package/dist/skills/index.js +169 -0
  25. package/dist/tools/ask-user-question.d.ts +62 -0
  26. package/dist/tools/ask-user-question.js +71 -0
  27. package/dist/tools/base.d.ts +16 -0
  28. package/dist/tools/base.js +39 -0
  29. package/dist/tools/glob.d.ts +27 -0
  30. package/dist/tools/glob.js +125 -0
  31. package/dist/tools/image-read.d.ts +42 -0
  32. package/dist/tools/image-read.js +125 -0
  33. package/dist/tools/index.d.ts +27 -0
  34. package/dist/tools/index.js +127 -0
  35. package/dist/tools/list-directory.d.ts +28 -0
  36. package/dist/tools/list-directory.js +94 -0
  37. package/dist/tools/pdf-extract.d.ts +32 -0
  38. package/dist/tools/pdf-extract.js +130 -0
  39. package/dist/tools/read-file.d.ts +31 -0
  40. package/dist/tools/read-file.js +116 -0
  41. package/dist/tools/replace.d.ts +35 -0
  42. package/dist/tools/replace.js +93 -0
  43. package/dist/tools/run-shell-command.d.ts +35 -0
  44. package/dist/tools/run-shell-command.js +81 -0
  45. package/dist/tools/save-memory.d.ts +22 -0
  46. package/dist/tools/save-memory.js +91 -0
  47. package/dist/tools/search-file-content.d.ts +42 -0
  48. package/dist/tools/search-file-content.js +153 -0
  49. package/dist/tools/task.d.ts +46 -0
  50. package/dist/tools/task.js +54 -0
  51. package/dist/tools/web-fetch.d.ts +26 -0
  52. package/dist/tools/web-fetch.js +81 -0
  53. package/dist/tools/web-search.d.ts +35 -0
  54. package/dist/tools/web-search.js +86 -0
  55. package/dist/tools/write-file.d.ts +25 -0
  56. package/dist/tools/write-file.js +76 -0
  57. package/dist/types/index.d.ts +166 -0
  58. package/dist/types/index.js +43 -0
  59. 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';
@@ -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"]}