@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,20 @@
1
+ import { Config, AuthConfig } from '../types';
2
+ export declare class ConfigManager {
3
+ private config;
4
+ constructor();
5
+ get<K extends keyof Config>(key: K): Config[K];
6
+ set<K extends keyof Config>(key: K, value: Config[K]): void;
7
+ getAll(): Config;
8
+ setAll(config: Partial<Config>): void;
9
+ clear(): void;
10
+ getAuth(): AuthConfig | undefined;
11
+ setAuth(auth: AuthConfig): void;
12
+ isAuthenticated(): boolean;
13
+ getDefaultModel(): string;
14
+ setDefaultModel(model: string): void;
15
+ getShowThinking(): boolean;
16
+ setShowThinking(show: boolean): void;
17
+ toggleShowThinking(): boolean;
18
+ getConfigPath(): string;
19
+ }
20
+ export declare function getConfigManager(): ConfigManager;
@@ -0,0 +1,83 @@
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.ConfigManager = void 0;
7
+ exports.getConfigManager = getConfigManager;
8
+ const conf_1 = __importDefault(require("conf"));
9
+ const DEFAULT_CONFIG = {
10
+ defaultModel: 'gpt-4o',
11
+ maxTokens: 4096,
12
+ temperature: 0.7,
13
+ sandbox: false,
14
+ showThinking: false, // 默认不显示思考过程
15
+ mcpServers: {}
16
+ };
17
+ class ConfigManager {
18
+ config;
19
+ constructor() {
20
+ this.config = new conf_1.default({
21
+ projectName: 'oflow',
22
+ defaults: DEFAULT_CONFIG
23
+ });
24
+ }
25
+ get(key) {
26
+ return this.config.get(key);
27
+ }
28
+ set(key, value) {
29
+ this.config.set(key, value);
30
+ }
31
+ getAll() {
32
+ return this.config.store;
33
+ }
34
+ setAll(config) {
35
+ for (const [key, value] of Object.entries(config)) {
36
+ this.config.set(key, value);
37
+ }
38
+ }
39
+ clear() {
40
+ this.config.clear();
41
+ }
42
+ getAuth() {
43
+ return this.config.get('auth');
44
+ }
45
+ setAuth(auth) {
46
+ this.config.set('auth', auth);
47
+ }
48
+ isAuthenticated() {
49
+ const auth = this.getAuth();
50
+ return !!(auth?.apiKey);
51
+ }
52
+ getDefaultModel() {
53
+ return this.config.get('defaultModel') || 'gpt-4o';
54
+ }
55
+ setDefaultModel(model) {
56
+ this.config.set('defaultModel', model);
57
+ }
58
+ // 思考过程显示控制
59
+ getShowThinking() {
60
+ return this.config.get('showThinking') ?? false;
61
+ }
62
+ setShowThinking(show) {
63
+ this.config.set('showThinking', show);
64
+ }
65
+ toggleShowThinking() {
66
+ const current = this.getShowThinking();
67
+ this.setShowThinking(!current);
68
+ return !current;
69
+ }
70
+ getConfigPath() {
71
+ return this.config.path;
72
+ }
73
+ }
74
+ exports.ConfigManager = ConfigManager;
75
+ // Singleton instance
76
+ let configManager = null;
77
+ function getConfigManager() {
78
+ if (!configManager) {
79
+ configManager = new ConfigManager();
80
+ }
81
+ return configManager;
82
+ }
83
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvY29uZmlnL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7OztBQXdGQSw0Q0FLQztBQTdGRCxnREFBd0I7QUFHeEIsTUFBTSxjQUFjLEdBQW9CO0lBQ3RDLFlBQVksRUFBRSxRQUFRO0lBQ3RCLFNBQVMsRUFBRSxJQUFJO0lBQ2YsV0FBVyxFQUFFLEdBQUc7SUFDaEIsT0FBTyxFQUFFLEtBQUs7SUFDZCxZQUFZLEVBQUUsS0FBSyxFQUFFLFlBQVk7SUFDakMsVUFBVSxFQUFFLEVBQUU7Q0FDZixDQUFDO0FBRUYsTUFBYSxhQUFhO0lBQ2hCLE1BQU0sQ0FBZTtJQUU3QjtRQUNFLElBQUksQ0FBQyxNQUFNLEdBQUcsSUFBSSxjQUFJLENBQVM7WUFDN0IsV0FBVyxFQUFFLE9BQU87WUFDcEIsUUFBUSxFQUFFLGNBQXdCO1NBQ25DLENBQUMsQ0FBQztJQUNMLENBQUM7SUFFRCxHQUFHLENBQXlCLEdBQU07UUFDaEMsT0FBTyxJQUFJLENBQUMsTUFBTSxDQUFDLEdBQUcsQ0FBQyxHQUFHLENBQUMsQ0FBQztJQUM5QixDQUFDO0lBRUQsR0FBRyxDQUF5QixHQUFNLEVBQUUsS0FBZ0I7UUFDbEQsSUFBSSxDQUFDLE1BQU0sQ0FBQyxHQUFHLENBQUMsR0FBRyxFQUFFLEtBQUssQ0FBQyxDQUFDO0lBQzlCLENBQUM7SUFFRCxNQUFNO1FBQ0osT0FBTyxJQUFJLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQztJQUMzQixDQUFDO0lBRUQsTUFBTSxDQUFDLE1BQXVCO1FBQzVCLEtBQUssTUFBTSxDQUFDLEdBQUcsRUFBRSxLQUFLLENBQUMsSUFBSSxNQUFNLENBQUMsT0FBTyxDQUFDLE1BQU0sQ0FBQyxFQUFFLENBQUM7WUFDbEQsSUFBSSxDQUFDLE1BQU0sQ0FBQyxHQUFHLENBQUMsR0FBbUIsRUFBRSxLQUE2QixDQUFDLENBQUM7UUFDdEUsQ0FBQztJQUNILENBQUM7SUFFRCxLQUFLO1FBQ0gsSUFBSSxDQUFDLE1BQU0sQ0FBQyxLQUFLLEVBQUUsQ0FBQztJQUN0QixDQUFDO0lBRUQsT0FBTztRQUNMLE9BQU8sSUFBSSxDQUFDLE1BQU0sQ0FBQyxHQUFHLENBQUMsTUFBTSxDQUFDLENBQUM7SUFDakMsQ0FBQztJQUVELE9BQU8sQ0FBQyxJQUFnQjtRQUN0QixJQUFJLENBQUMsTUFBTSxDQUFDLEdBQUcsQ0FBQyxNQUFNLEVBQUUsSUFBSSxDQUFDLENBQUM7SUFDaEMsQ0FBQztJQUVELGVBQWU7UUFDYixNQUFNLElBQUksR0FBRyxJQUFJLENBQUMsT0FBTyxFQUFFLENBQUM7UUFDNUIsT0FBTyxDQUFDLENBQUMsQ0FBQyxJQUFJLEVBQUUsTUFBTSxDQUFDLENBQUM7SUFDMUIsQ0FBQztJQUVELGVBQWU7UUFDYixPQUFPLElBQUksQ0FBQyxNQUFNLENBQUMsR0FBRyxDQUFDLGNBQWMsQ0FBQyxJQUFJLFFBQVEsQ0FBQztJQUNyRCxDQUFDO0lBRUQsZUFBZSxDQUFDLEtBQWE7UUFDM0IsSUFBSSxDQUFDLE1BQU0sQ0FBQyxHQUFHLENBQUMsY0FBYyxFQUFFLEtBQUssQ0FBQyxDQUFDO0lBQ3pDLENBQUM7SUFFRCxXQUFXO0lBQ1gsZUFBZTtRQUNiLE9BQU8sSUFBSSxDQUFDLE1BQU0sQ0FBQyxHQUFHLENBQUMsY0FBYyxDQUFDLElBQUksS0FBSyxDQUFDO0lBQ2xELENBQUM7SUFFRCxlQUFlLENBQUMsSUFBYTtRQUMzQixJQUFJLENBQUMsTUFBTSxDQUFDLEdBQUcsQ0FBQyxjQUFjLEVBQUUsSUFBSSxDQUFDLENBQUM7SUFDeEMsQ0FBQztJQUVELGtCQUFrQjtRQUNoQixNQUFNLE9BQU8sR0FBRyxJQUFJLENBQUMsZUFBZSxFQUFFLENBQUM7UUFDdkMsSUFBSSxDQUFDLGVBQWUsQ0FBQyxDQUFDLE9BQU8sQ0FBQyxDQUFDO1FBQy9CLE9BQU8sQ0FBQyxPQUFPLENBQUM7SUFDbEIsQ0FBQztJQUVELGFBQWE7UUFDWCxPQUFPLElBQUksQ0FBQyxNQUFNLENBQUMsSUFBSSxDQUFDO0lBQzFCLENBQUM7Q0FDRjtBQXZFRCxzQ0F1RUM7QUFFRCxxQkFBcUI7QUFDckIsSUFBSSxhQUFhLEdBQXlCLElBQUksQ0FBQztBQUUvQyxTQUFnQixnQkFBZ0I7SUFDOUIsSUFBSSxDQUFDLGFBQWEsRUFBRSxDQUFDO1FBQ25CLGFBQWEsR0FBRyxJQUFJLGFBQWEsRUFBRSxDQUFDO0lBQ3RDLENBQUM7SUFDRCxPQUFPLGFBQWEsQ0FBQztBQUN2QixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IENvbmYgZnJvbSAnY29uZic7XG5pbXBvcnQgeyBDb25maWcsIEF1dGhDb25maWcgfSBmcm9tICcuLi90eXBlcyc7XG5cbmNvbnN0IERFRkFVTFRfQ09ORklHOiBQYXJ0aWFsPENvbmZpZz4gPSB7XG4gIGRlZmF1bHRNb2RlbDogJ2dwdC00bycsXG4gIG1heFRva2VuczogNDA5NixcbiAgdGVtcGVyYXR1cmU6IDAuNyxcbiAgc2FuZGJveDogZmFsc2UsXG4gIHNob3dUaGlua2luZzogZmFsc2UsIC8vIOm7mOiupOS4jeaYvuekuuaAneiAg+i/h+eoi1xuICBtY3BTZXJ2ZXJzOiB7fVxufTtcblxuZXhwb3J0IGNsYXNzIENvbmZpZ01hbmFnZXIge1xuICBwcml2YXRlIGNvbmZpZzogQ29uZjxDb25maWc+O1xuICBcbiAgY29uc3RydWN0b3IoKSB7XG4gICAgdGhpcy5jb25maWcgPSBuZXcgQ29uZjxDb25maWc+KHtcbiAgICAgIHByb2plY3ROYW1lOiAnb2Zsb3cnLFxuICAgICAgZGVmYXVsdHM6IERFRkFVTFRfQ09ORklHIGFzIENvbmZpZ1xuICAgIH0pO1xuICB9XG5cbiAgZ2V0PEsgZXh0ZW5kcyBrZXlvZiBDb25maWc+KGtleTogSyk6IENvbmZpZ1tLXSB7XG4gICAgcmV0dXJuIHRoaXMuY29uZmlnLmdldChrZXkpO1xuICB9XG5cbiAgc2V0PEsgZXh0ZW5kcyBrZXlvZiBDb25maWc+KGtleTogSywgdmFsdWU6IENvbmZpZ1tLXSk6IHZvaWQge1xuICAgIHRoaXMuY29uZmlnLnNldChrZXksIHZhbHVlKTtcbiAgfVxuXG4gIGdldEFsbCgpOiBDb25maWcge1xuICAgIHJldHVybiB0aGlzLmNvbmZpZy5zdG9yZTtcbiAgfVxuXG4gIHNldEFsbChjb25maWc6IFBhcnRpYWw8Q29uZmlnPik6IHZvaWQge1xuICAgIGZvciAoY29uc3QgW2tleSwgdmFsdWVdIG9mIE9iamVjdC5lbnRyaWVzKGNvbmZpZykpIHtcbiAgICAgIHRoaXMuY29uZmlnLnNldChrZXkgYXMga2V5b2YgQ29uZmlnLCB2YWx1ZSBhcyBDb25maWdba2V5b2YgQ29uZmlnXSk7XG4gICAgfVxuICB9XG5cbiAgY2xlYXIoKTogdm9pZCB7XG4gICAgdGhpcy5jb25maWcuY2xlYXIoKTtcbiAgfVxuXG4gIGdldEF1dGgoKTogQXV0aENvbmZpZyB8IHVuZGVmaW5lZCB7XG4gICAgcmV0dXJuIHRoaXMuY29uZmlnLmdldCgnYXV0aCcpO1xuICB9XG5cbiAgc2V0QXV0aChhdXRoOiBBdXRoQ29uZmlnKTogdm9pZCB7XG4gICAgdGhpcy5jb25maWcuc2V0KCdhdXRoJywgYXV0aCk7XG4gIH1cblxuICBpc0F1dGhlbnRpY2F0ZWQoKTogYm9vbGVhbiB7XG4gICAgY29uc3QgYXV0aCA9IHRoaXMuZ2V0QXV0aCgpO1xuICAgIHJldHVybiAhIShhdXRoPy5hcGlLZXkpO1xuICB9XG5cbiAgZ2V0RGVmYXVsdE1vZGVsKCk6IHN0cmluZyB7XG4gICAgcmV0dXJuIHRoaXMuY29uZmlnLmdldCgnZGVmYXVsdE1vZGVsJykgfHwgJ2dwdC00byc7XG4gIH1cblxuICBzZXREZWZhdWx0TW9kZWwobW9kZWw6IHN0cmluZyk6IHZvaWQge1xuICAgIHRoaXMuY29uZmlnLnNldCgnZGVmYXVsdE1vZGVsJywgbW9kZWwpO1xuICB9XG5cbiAgLy8g5oCd6ICD6L+H56iL5pi+56S65o6n5Yi2XG4gIGdldFNob3dUaGlua2luZygpOiBib29sZWFuIHtcbiAgICByZXR1cm4gdGhpcy5jb25maWcuZ2V0KCdzaG93VGhpbmtpbmcnKSA/PyBmYWxzZTtcbiAgfVxuXG4gIHNldFNob3dUaGlua2luZyhzaG93OiBib29sZWFuKTogdm9pZCB7XG4gICAgdGhpcy5jb25maWcuc2V0KCdzaG93VGhpbmtpbmcnLCBzaG93KTtcbiAgfVxuXG4gIHRvZ2dsZVNob3dUaGlua2luZygpOiBib29sZWFuIHtcbiAgICBjb25zdCBjdXJyZW50ID0gdGhpcy5nZXRTaG93VGhpbmtpbmcoKTtcbiAgICB0aGlzLnNldFNob3dUaGlua2luZyghY3VycmVudCk7XG4gICAgcmV0dXJuICFjdXJyZW50O1xuICB9XG5cbiAgZ2V0Q29uZmlnUGF0aCgpOiBzdHJpbmcge1xuICAgIHJldHVybiB0aGlzLmNvbmZpZy5wYXRoO1xuICB9XG59XG5cbi8vIFNpbmdsZXRvbiBpbnN0YW5jZVxubGV0IGNvbmZpZ01hbmFnZXI6IENvbmZpZ01hbmFnZXIgfCBudWxsID0gbnVsbDtcblxuZXhwb3J0IGZ1bmN0aW9uIGdldENvbmZpZ01hbmFnZXIoKTogQ29uZmlnTWFuYWdlciB7XG4gIGlmICghY29uZmlnTWFuYWdlcikge1xuICAgIGNvbmZpZ01hbmFnZXIgPSBuZXcgQ29uZmlnTWFuYWdlcigpO1xuICB9XG4gIHJldHVybiBjb25maWdNYW5hZ2VyO1xufVxuIl19
@@ -0,0 +1,26 @@
1
+ import { Message, ToolCall } from './types';
2
+ import { AIProvider } from './ai/openai-provider';
3
+ export interface ConversationOptions {
4
+ provider: AIProvider;
5
+ workingDirectory?: string;
6
+ maxTokens?: number;
7
+ temperature?: number;
8
+ showThinking?: boolean;
9
+ onContent?: (content: string) => void;
10
+ onThinking?: (thinking: string) => void;
11
+ onToolCall?: (toolCall: ToolCall) => void;
12
+ onToolResult?: (result: string) => void;
13
+ }
14
+ export declare class Conversation {
15
+ private messages;
16
+ private provider;
17
+ private toolRegistry;
18
+ private options;
19
+ constructor(options: ConversationOptions);
20
+ addMessage(message: Message): void;
21
+ setSystemPrompt(prompt: string): void;
22
+ getMessages(): Message[];
23
+ clear(): void;
24
+ sendMessage(userMessage: string): Promise<string>;
25
+ private processConversation;
26
+ }
@@ -0,0 +1,126 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.Conversation = void 0;
4
+ const tools_1 = require("./tools");
5
+ class Conversation {
6
+ messages = [];
7
+ provider;
8
+ toolRegistry;
9
+ options;
10
+ constructor(options) {
11
+ this.provider = options.provider;
12
+ this.toolRegistry = (0, tools_1.getToolRegistry)();
13
+ this.options = options;
14
+ }
15
+ addMessage(message) {
16
+ this.messages.push(message);
17
+ }
18
+ setSystemPrompt(prompt) {
19
+ // Remove existing system prompt if any
20
+ this.messages = this.messages.filter(m => m.role !== 'system');
21
+ // Add new system prompt at the beginning
22
+ this.messages.unshift({ role: 'system', content: prompt });
23
+ }
24
+ getMessages() {
25
+ return this.messages;
26
+ }
27
+ clear() {
28
+ this.messages = [];
29
+ }
30
+ async sendMessage(userMessage) {
31
+ this.addMessage({ role: 'user', content: userMessage });
32
+ return this.processConversation();
33
+ }
34
+ async processConversation() {
35
+ let finalResponse = '';
36
+ let iterations = 0;
37
+ const maxIterations = 20; // Prevent infinite loops
38
+ while (iterations < maxIterations) {
39
+ iterations++;
40
+ const chatOptions = {
41
+ messages: this.messages,
42
+ tools: this.toolRegistry.getDefinitions(),
43
+ toolChoice: 'auto',
44
+ maxTokens: this.options.maxTokens || 4096,
45
+ temperature: this.options.temperature || 0.7,
46
+ stream: true
47
+ };
48
+ let currentContent = '';
49
+ let currentThinking = '';
50
+ let toolCalls = [];
51
+ let currentId = '';
52
+ try {
53
+ await this.provider.chatStream(chatOptions, (chunk) => {
54
+ currentId = chunk.id;
55
+ // 处理思考过程
56
+ if (chunk.delta.reasoning && this.options.showThinking) {
57
+ currentThinking += chunk.delta.reasoning;
58
+ this.options.onThinking?.(chunk.delta.reasoning);
59
+ }
60
+ if (chunk.delta.content) {
61
+ currentContent += chunk.delta.content;
62
+ this.options.onContent?.(chunk.delta.content);
63
+ }
64
+ if (chunk.delta.toolCalls) {
65
+ for (const tc of chunk.delta.toolCalls) {
66
+ if (tc.id && tc.function?.name) {
67
+ toolCalls.push({
68
+ id: tc.id,
69
+ type: 'function',
70
+ function: {
71
+ name: tc.function.name,
72
+ arguments: tc.function.arguments || ''
73
+ }
74
+ });
75
+ }
76
+ else {
77
+ // Update existing tool call
78
+ const existing = toolCalls.find(t => t.id === tc.id);
79
+ if (existing && tc.function?.arguments) {
80
+ existing.function.arguments += tc.function.arguments;
81
+ }
82
+ }
83
+ }
84
+ }
85
+ });
86
+ // Add assistant message
87
+ const assistantMessage = {
88
+ role: 'assistant',
89
+ content: currentContent || ''
90
+ };
91
+ if (toolCalls.length > 0) {
92
+ assistantMessage.toolCalls = toolCalls;
93
+ }
94
+ this.messages.push(assistantMessage);
95
+ // Process tool calls
96
+ if (toolCalls.length > 0) {
97
+ for (const toolCall of toolCalls) {
98
+ this.options.onToolCall?.(toolCall);
99
+ const toolOptions = {
100
+ workingDirectory: this.options.workingDirectory
101
+ };
102
+ const result = await this.toolRegistry.execute(toolCall, toolOptions);
103
+ this.options.onToolResult?.(result.content);
104
+ // Add tool result message
105
+ this.messages.push({
106
+ role: 'tool',
107
+ toolCallId: toolCall.id,
108
+ content: result.content
109
+ });
110
+ }
111
+ // Continue conversation to get final response
112
+ continue;
113
+ }
114
+ // No tool calls, we have the final response
115
+ finalResponse = currentContent;
116
+ break;
117
+ }
118
+ catch (error) {
119
+ throw new Error(`Conversation error: ${error instanceof Error ? error.message : String(error)}`);
120
+ }
121
+ }
122
+ return finalResponse;
123
+ }
124
+ }
125
+ exports.Conversation = Conversation;
126
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"conversation.js","sourceRoot":"","sources":["../src/conversation.ts"],"names":[],"mappings":";;;AAEA,mCAA4E;AAc5E,MAAa,YAAY;IACf,QAAQ,GAAc,EAAE,CAAC;IACzB,QAAQ,CAAa;IACrB,YAAY,CAAe;IAC3B,OAAO,CAAsB;IAErC,YAAY,OAA4B;QACtC,IAAI,CAAC,QAAQ,GAAG,OAAO,CAAC,QAAQ,CAAC;QACjC,IAAI,CAAC,YAAY,GAAG,IAAA,uBAAe,GAAE,CAAC;QACtC,IAAI,CAAC,OAAO,GAAG,OAAO,CAAC;IACzB,CAAC;IAED,UAAU,CAAC,OAAgB;QACzB,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;IAC9B,CAAC;IAED,eAAe,CAAC,MAAc;QAC5B,uCAAuC;QACvC,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,KAAK,QAAQ,CAAC,CAAC;QAC/D,yCAAyC;QACzC,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC,EAAE,IAAI,EAAE,QAAQ,EAAE,OAAO,EAAE,MAAM,EAAE,CAAC,CAAC;IAC7D,CAAC;IAED,WAAW;QACT,OAAO,IAAI,CAAC,QAAQ,CAAC;IACvB,CAAC;IAED,KAAK;QACH,IAAI,CAAC,QAAQ,GAAG,EAAE,CAAC;IACrB,CAAC;IAED,KAAK,CAAC,WAAW,CAAC,WAAmB;QACnC,IAAI,CAAC,UAAU,CAAC,EAAE,IAAI,EAAE,MAAM,EAAE,OAAO,EAAE,WAAW,EAAE,CAAC,CAAC;QACxD,OAAO,IAAI,CAAC,mBAAmB,EAAE,CAAC;IACpC,CAAC;IAEO,KAAK,CAAC,mBAAmB;QAC/B,IAAI,aAAa,GAAG,EAAE,CAAC;QACvB,IAAI,UAAU,GAAG,CAAC,CAAC;QACnB,MAAM,aAAa,GAAG,EAAE,CAAC,CAAC,yBAAyB;QAEnD,OAAO,UAAU,GAAG,aAAa,EAAE,CAAC;YAClC,UAAU,EAAE,CAAC;YAEb,MAAM,WAAW,GAA0B;gBACzC,QAAQ,EAAE,IAAI,CAAC,QAAQ;gBACvB,KAAK,EAAE,IAAI,CAAC,YAAY,CAAC,cAAc,EAAE;gBACzC,UAAU,EAAE,MAAM;gBAClB,SAAS,EAAE,IAAI,CAAC,OAAO,CAAC,SAAS,IAAI,IAAI;gBACzC,WAAW,EAAE,IAAI,CAAC,OAAO,CAAC,WAAW,IAAI,GAAG;gBAC5C,MAAM,EAAE,IAAI;aACb,CAAC;YAEF,IAAI,cAAc,GAAG,EAAE,CAAC;YACxB,IAAI,eAAe,GAAG,EAAE,CAAC;YACzB,IAAI,SAAS,GAAe,EAAE,CAAC;YAC/B,IAAI,SAAS,GAAG,EAAE,CAAC;YAEnB,IAAI,CAAC;gBACH,MAAM,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,WAAW,EAAE,CAAC,KAAkB,EAAE,EAAE;oBACjE,SAAS,GAAG,KAAK,CAAC,EAAE,CAAC;oBAErB,SAAS;oBACT,IAAI,KAAK,CAAC,KAAK,CAAC,SAAS,IAAI,IAAI,CAAC,OAAO,CAAC,YAAY,EAAE,CAAC;wBACvD,eAAe,IAAI,KAAK,CAAC,KAAK,CAAC,SAAS,CAAC;wBACzC,IAAI,CAAC,OAAO,CAAC,UAAU,EAAE,CAAC,KAAK,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;oBACnD,CAAC;oBAED,IAAI,KAAK,CAAC,KAAK,CAAC,OAAO,EAAE,CAAC;wBACxB,cAAc,IAAI,KAAK,CAAC,KAAK,CAAC,OAAO,CAAC;wBACtC,IAAI,CAAC,OAAO,CAAC,SAAS,EAAE,CAAC,KAAK,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC;oBAChD,CAAC;oBAED,IAAI,KAAK,CAAC,KAAK,CAAC,SAAS,EAAE,CAAC;wBAC1B,KAAK,MAAM,EAAE,IAAI,KAAK,CAAC,KAAK,CAAC,SAAS,EAAE,CAAC;4BACvC,IAAI,EAAE,CAAC,EAAE,IAAI,EAAE,CAAC,QAAQ,EAAE,IAAI,EAAE,CAAC;gCAC/B,SAAS,CAAC,IAAI,CAAC;oCACb,EAAE,EAAE,EAAE,CAAC,EAAE;oCACT,IAAI,EAAE,UAAU;oCAChB,QAAQ,EAAE;wCACR,IAAI,EAAE,EAAE,CAAC,QAAQ,CAAC,IAAI;wCACtB,SAAS,EAAE,EAAE,CAAC,QAAQ,CAAC,SAAS,IAAI,EAAE;qCACvC;iCACF,CAAC,CAAC;4BACL,CAAC;iCAAM,CAAC;gCACN,4BAA4B;gCAC5B,MAAM,QAAQ,GAAG,SAAS,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,EAAE,KAAK,EAAE,CAAC,EAAE,CAAC,CAAC;gCACrD,IAAI,QAAQ,IAAI,EAAE,CAAC,QAAQ,EAAE,SAAS,EAAE,CAAC;oCACvC,QAAQ,CAAC,QAAQ,CAAC,SAAS,IAAI,EAAE,CAAC,QAAQ,CAAC,SAAS,CAAC;gCACvD,CAAC;4BACH,CAAC;wBACH,CAAC;oBACH,CAAC;gBACH,CAAC,CAAC,CAAC;gBAEH,wBAAwB;gBACxB,MAAM,gBAAgB,GAAY;oBAChC,IAAI,EAAE,WAAW;oBACjB,OAAO,EAAE,cAAc,IAAI,EAAE;iBAC9B,CAAC;gBAEF,IAAI,SAAS,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;oBACzB,gBAAgB,CAAC,SAAS,GAAG,SAAS,CAAC;gBACzC,CAAC;gBAED,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,gBAAgB,CAAC,CAAC;gBAErC,qBAAqB;gBACrB,IAAI,SAAS,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;oBACzB,KAAK,MAAM,QAAQ,IAAI,SAAS,EAAE,CAAC;wBACjC,IAAI,CAAC,OAAO,CAAC,UAAU,EAAE,CAAC,QAAQ,CAAC,CAAC;wBAEpC,MAAM,WAAW,GAAuB;4BACtC,gBAAgB,EAAE,IAAI,CAAC,OAAO,CAAC,gBAAgB;yBAChD,CAAC;wBAEF,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,YAAY,CAAC,OAAO,CAAC,QAAQ,EAAE,WAAW,CAAC,CAAC;wBAEtE,IAAI,CAAC,OAAO,CAAC,YAAY,EAAE,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;wBAE5C,0BAA0B;wBAC1B,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC;4BACjB,IAAI,EAAE,MAAM;4BACZ,UAAU,EAAE,QAAQ,CAAC,EAAE;4BACvB,OAAO,EAAE,MAAM,CAAC,OAAO;yBACxB,CAAC,CAAC;oBACL,CAAC;oBAED,8CAA8C;oBAC9C,SAAS;gBACX,CAAC;gBAED,4CAA4C;gBAC5C,aAAa,GAAG,cAAc,CAAC;gBAC/B,MAAM;YAER,CAAC;YAAC,OAAO,KAAK,EAAE,CAAC;gBACf,MAAM,IAAI,KAAK,CAAC,uBAAuB,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;YACnG,CAAC;QACH,CAAC;QAED,OAAO,aAAa,CAAC;IACvB,CAAC;CACF;AA/ID,oCA+IC","sourcesContent":["import { Message, ChatCompletionOptions, ToolCall, StreamChunk } from './types';\nimport { AIProvider } from './ai/openai-provider';\nimport { ToolRegistry, getToolRegistry, ToolExecuteOptions } from './tools';\n\nexport interface ConversationOptions {\n  provider: AIProvider;\n  workingDirectory?: string;\n  maxTokens?: number;\n  temperature?: number;\n  showThinking?: boolean; // 是否显示思考过程\n  onContent?: (content: string) => void;\n  onThinking?: (thinking: string) => void; // 思考过程回调\n  onToolCall?: (toolCall: ToolCall) => void;\n  onToolResult?: (result: string) => void;\n}\n\nexport class Conversation {\n  private messages: Message[] = [];\n  private provider: AIProvider;\n  private toolRegistry: ToolRegistry;\n  private options: ConversationOptions;\n\n  constructor(options: ConversationOptions) {\n    this.provider = options.provider;\n    this.toolRegistry = getToolRegistry();\n    this.options = options;\n  }\n\n  addMessage(message: Message): void {\n    this.messages.push(message);\n  }\n\n  setSystemPrompt(prompt: string): void {\n    // Remove existing system prompt if any\n    this.messages = this.messages.filter(m => m.role !== 'system');\n    // Add new system prompt at the beginning\n    this.messages.unshift({ role: 'system', content: prompt });\n  }\n\n  getMessages(): Message[] {\n    return this.messages;\n  }\n\n  clear(): void {\n    this.messages = [];\n  }\n\n  async sendMessage(userMessage: string): Promise<string> {\n    this.addMessage({ role: 'user', content: userMessage });\n    return this.processConversation();\n  }\n\n  private async processConversation(): Promise<string> {\n    let finalResponse = '';\n    let iterations = 0;\n    const maxIterations = 20; // Prevent infinite loops\n\n    while (iterations < maxIterations) {\n      iterations++;\n\n      const chatOptions: ChatCompletionOptions = {\n        messages: this.messages,\n        tools: this.toolRegistry.getDefinitions(),\n        toolChoice: 'auto',\n        maxTokens: this.options.maxTokens || 4096,\n        temperature: this.options.temperature || 0.7,\n        stream: true\n      };\n\n      let currentContent = '';\n      let currentThinking = '';\n      let toolCalls: ToolCall[] = [];\n      let currentId = '';\n\n      try {\n        await this.provider.chatStream(chatOptions, (chunk: StreamChunk) => {\n          currentId = chunk.id;\n          \n          // 处理思考过程\n          if (chunk.delta.reasoning && this.options.showThinking) {\n            currentThinking += chunk.delta.reasoning;\n            this.options.onThinking?.(chunk.delta.reasoning);\n          }\n\n          if (chunk.delta.content) {\n            currentContent += chunk.delta.content;\n            this.options.onContent?.(chunk.delta.content);\n          }\n\n          if (chunk.delta.toolCalls) {\n            for (const tc of chunk.delta.toolCalls) {\n              if (tc.id && tc.function?.name) {\n                toolCalls.push({\n                  id: tc.id,\n                  type: 'function',\n                  function: {\n                    name: tc.function.name,\n                    arguments: tc.function.arguments || ''\n                  }\n                });\n              } else {\n                // Update existing tool call\n                const existing = toolCalls.find(t => t.id === tc.id);\n                if (existing && tc.function?.arguments) {\n                  existing.function.arguments += tc.function.arguments;\n                }\n              }\n            }\n          }\n        });\n\n        // Add assistant message\n        const assistantMessage: Message = {\n          role: 'assistant',\n          content: currentContent || ''\n        };\n\n        if (toolCalls.length > 0) {\n          assistantMessage.toolCalls = toolCalls;\n        }\n\n        this.messages.push(assistantMessage);\n\n        // Process tool calls\n        if (toolCalls.length > 0) {\n          for (const toolCall of toolCalls) {\n            this.options.onToolCall?.(toolCall);\n\n            const toolOptions: ToolExecuteOptions = {\n              workingDirectory: this.options.workingDirectory\n            };\n\n            const result = await this.toolRegistry.execute(toolCall, toolOptions);\n            \n            this.options.onToolResult?.(result.content);\n\n            // Add tool result message\n            this.messages.push({\n              role: 'tool',\n              toolCallId: toolCall.id,\n              content: result.content\n            });\n          }\n\n          // Continue conversation to get final response\n          continue;\n        }\n\n        // No tool calls, we have the final response\n        finalResponse = currentContent;\n        break;\n\n      } catch (error) {\n        throw new Error(`Conversation error: ${error instanceof Error ? error.message : String(error)}`);\n      }\n    }\n\n    return finalResponse;\n  }\n}\n"]}
@@ -0,0 +1,10 @@
1
+ export * from './types';
2
+ export { ConfigManager, getConfigManager } from './config';
3
+ export { createAIProvider, OpenAIProvider, CustomProvider, ChineseProvider } from './ai';
4
+ export { ToolRegistry, getToolRegistry, BaseTool, ReadFileTool, WriteFileTool, ListDirectoryTool, GlobTool, SearchFileContentTool, RunShellCommandTool, WebSearchTool, WebFetchTool, ReplaceTool, ImageReadTool, PDFExtractTool, AskUserQuestionTool, TaskTool, SaveMemoryTool } from './tools';
5
+ export { Conversation } from './conversation';
6
+ export { MCPClient, MCPManager } from './mcp';
7
+ export { SkillManager } from './skills';
8
+ export { Agent, AgentFactory, AgentDefinition, AgentResult, AgentType, AGENT_DEFINITIONS } from './agents';
9
+ export { SandboxExecutor, SandboxBackend, SandboxConfig, SandboxResult, createSandboxExecutor } from './sandbox';
10
+ export { CHINESE_MODELS, DEFAULT_CHINESE_MODELS } from './ai/chinese-provider';
package/dist/index.js ADDED
@@ -0,0 +1,72 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __exportStar = (this && this.__exportStar) || function(m, exports) {
14
+ for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
15
+ };
16
+ Object.defineProperty(exports, "__esModule", { value: true });
17
+ exports.DEFAULT_CHINESE_MODELS = exports.CHINESE_MODELS = exports.createSandboxExecutor = exports.SandboxExecutor = exports.AGENT_DEFINITIONS = exports.AgentFactory = exports.Agent = exports.SkillManager = exports.MCPManager = exports.MCPClient = exports.Conversation = exports.SaveMemoryTool = exports.TaskTool = exports.AskUserQuestionTool = exports.PDFExtractTool = exports.ImageReadTool = exports.ReplaceTool = exports.WebFetchTool = exports.WebSearchTool = exports.RunShellCommandTool = exports.SearchFileContentTool = exports.GlobTool = exports.ListDirectoryTool = exports.WriteFileTool = exports.ReadFileTool = exports.BaseTool = exports.getToolRegistry = exports.ToolRegistry = exports.ChineseProvider = exports.CustomProvider = exports.OpenAIProvider = exports.createAIProvider = exports.getConfigManager = exports.ConfigManager = void 0;
18
+ // Types
19
+ __exportStar(require("./types"), exports);
20
+ // Configuration
21
+ var config_1 = require("./config");
22
+ Object.defineProperty(exports, "ConfigManager", { enumerable: true, get: function () { return config_1.ConfigManager; } });
23
+ Object.defineProperty(exports, "getConfigManager", { enumerable: true, get: function () { return config_1.getConfigManager; } });
24
+ // AI Providers
25
+ var ai_1 = require("./ai");
26
+ Object.defineProperty(exports, "createAIProvider", { enumerable: true, get: function () { return ai_1.createAIProvider; } });
27
+ Object.defineProperty(exports, "OpenAIProvider", { enumerable: true, get: function () { return ai_1.OpenAIProvider; } });
28
+ Object.defineProperty(exports, "CustomProvider", { enumerable: true, get: function () { return ai_1.CustomProvider; } });
29
+ Object.defineProperty(exports, "ChineseProvider", { enumerable: true, get: function () { return ai_1.ChineseProvider; } });
30
+ // Tools
31
+ var tools_1 = require("./tools");
32
+ Object.defineProperty(exports, "ToolRegistry", { enumerable: true, get: function () { return tools_1.ToolRegistry; } });
33
+ Object.defineProperty(exports, "getToolRegistry", { enumerable: true, get: function () { return tools_1.getToolRegistry; } });
34
+ Object.defineProperty(exports, "BaseTool", { enumerable: true, get: function () { return tools_1.BaseTool; } });
35
+ Object.defineProperty(exports, "ReadFileTool", { enumerable: true, get: function () { return tools_1.ReadFileTool; } });
36
+ Object.defineProperty(exports, "WriteFileTool", { enumerable: true, get: function () { return tools_1.WriteFileTool; } });
37
+ Object.defineProperty(exports, "ListDirectoryTool", { enumerable: true, get: function () { return tools_1.ListDirectoryTool; } });
38
+ Object.defineProperty(exports, "GlobTool", { enumerable: true, get: function () { return tools_1.GlobTool; } });
39
+ Object.defineProperty(exports, "SearchFileContentTool", { enumerable: true, get: function () { return tools_1.SearchFileContentTool; } });
40
+ Object.defineProperty(exports, "RunShellCommandTool", { enumerable: true, get: function () { return tools_1.RunShellCommandTool; } });
41
+ Object.defineProperty(exports, "WebSearchTool", { enumerable: true, get: function () { return tools_1.WebSearchTool; } });
42
+ Object.defineProperty(exports, "WebFetchTool", { enumerable: true, get: function () { return tools_1.WebFetchTool; } });
43
+ Object.defineProperty(exports, "ReplaceTool", { enumerable: true, get: function () { return tools_1.ReplaceTool; } });
44
+ Object.defineProperty(exports, "ImageReadTool", { enumerable: true, get: function () { return tools_1.ImageReadTool; } });
45
+ Object.defineProperty(exports, "PDFExtractTool", { enumerable: true, get: function () { return tools_1.PDFExtractTool; } });
46
+ Object.defineProperty(exports, "AskUserQuestionTool", { enumerable: true, get: function () { return tools_1.AskUserQuestionTool; } });
47
+ Object.defineProperty(exports, "TaskTool", { enumerable: true, get: function () { return tools_1.TaskTool; } });
48
+ Object.defineProperty(exports, "SaveMemoryTool", { enumerable: true, get: function () { return tools_1.SaveMemoryTool; } });
49
+ // Conversation
50
+ var conversation_1 = require("./conversation");
51
+ Object.defineProperty(exports, "Conversation", { enumerable: true, get: function () { return conversation_1.Conversation; } });
52
+ // MCP
53
+ var mcp_1 = require("./mcp");
54
+ Object.defineProperty(exports, "MCPClient", { enumerable: true, get: function () { return mcp_1.MCPClient; } });
55
+ Object.defineProperty(exports, "MCPManager", { enumerable: true, get: function () { return mcp_1.MCPManager; } });
56
+ // Skills
57
+ var skills_1 = require("./skills");
58
+ Object.defineProperty(exports, "SkillManager", { enumerable: true, get: function () { return skills_1.SkillManager; } });
59
+ // Agents
60
+ var agents_1 = require("./agents");
61
+ Object.defineProperty(exports, "Agent", { enumerable: true, get: function () { return agents_1.Agent; } });
62
+ Object.defineProperty(exports, "AgentFactory", { enumerable: true, get: function () { return agents_1.AgentFactory; } });
63
+ Object.defineProperty(exports, "AGENT_DEFINITIONS", { enumerable: true, get: function () { return agents_1.AGENT_DEFINITIONS; } });
64
+ // Sandbox
65
+ var sandbox_1 = require("./sandbox");
66
+ Object.defineProperty(exports, "SandboxExecutor", { enumerable: true, get: function () { return sandbox_1.SandboxExecutor; } });
67
+ Object.defineProperty(exports, "createSandboxExecutor", { enumerable: true, get: function () { return sandbox_1.createSandboxExecutor; } });
68
+ // 国内模型常量
69
+ var chinese_provider_1 = require("./ai/chinese-provider");
70
+ Object.defineProperty(exports, "CHINESE_MODELS", { enumerable: true, get: function () { return chinese_provider_1.CHINESE_MODELS; } });
71
+ Object.defineProperty(exports, "DEFAULT_CHINESE_MODELS", { enumerable: true, get: function () { return chinese_provider_1.DEFAULT_CHINESE_MODELS; } });
72
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFBQSxRQUFRO0FBQ1IsMENBQXdCO0FBRXhCLGdCQUFnQjtBQUNoQixtQ0FBMkQ7QUFBbEQsdUdBQUEsYUFBYSxPQUFBO0FBQUUsMEdBQUEsZ0JBQWdCLE9BQUE7QUFFeEMsZUFBZTtBQUNmLDJCQUF5RjtBQUFoRixzR0FBQSxnQkFBZ0IsT0FBQTtBQUFFLG9HQUFBLGNBQWMsT0FBQTtBQUFFLG9HQUFBLGNBQWMsT0FBQTtBQUFFLHFHQUFBLGVBQWUsT0FBQTtBQUUxRSxRQUFRO0FBQ1IsaUNBa0JpQjtBQWpCZixxR0FBQSxZQUFZLE9BQUE7QUFDWix3R0FBQSxlQUFlLE9BQUE7QUFDZixpR0FBQSxRQUFRLE9BQUE7QUFDUixxR0FBQSxZQUFZLE9BQUE7QUFDWixzR0FBQSxhQUFhLE9BQUE7QUFDYiwwR0FBQSxpQkFBaUIsT0FBQTtBQUNqQixpR0FBQSxRQUFRLE9BQUE7QUFDUiw4R0FBQSxxQkFBcUIsT0FBQTtBQUNyQiw0R0FBQSxtQkFBbUIsT0FBQTtBQUNuQixzR0FBQSxhQUFhLE9BQUE7QUFDYixxR0FBQSxZQUFZLE9BQUE7QUFDWixvR0FBQSxXQUFXLE9BQUE7QUFDWCxzR0FBQSxhQUFhLE9BQUE7QUFDYix1R0FBQSxjQUFjLE9BQUE7QUFDZCw0R0FBQSxtQkFBbUIsT0FBQTtBQUNuQixpR0FBQSxRQUFRLE9BQUE7QUFDUix1R0FBQSxjQUFjLE9BQUE7QUFHaEIsZUFBZTtBQUNmLCtDQUE4QztBQUFyQyw0R0FBQSxZQUFZLE9BQUE7QUFFckIsTUFBTTtBQUNOLDZCQUE4QztBQUFyQyxnR0FBQSxTQUFTLE9BQUE7QUFBRSxpR0FBQSxVQUFVLE9BQUE7QUFFOUIsU0FBUztBQUNULG1DQUF3QztBQUEvQixzR0FBQSxZQUFZLE9BQUE7QUFFckIsU0FBUztBQUNULG1DQU9rQjtBQU5oQiwrRkFBQSxLQUFLLE9BQUE7QUFDTCxzR0FBQSxZQUFZLE9BQUE7QUFJWiwyR0FBQSxpQkFBaUIsT0FBQTtBQUduQixVQUFVO0FBQ1YscUNBTW1CO0FBTGpCLDBHQUFBLGVBQWUsT0FBQTtBQUlmLGdIQUFBLHFCQUFxQixPQUFBO0FBR3ZCLFNBQVM7QUFDVCwwREFBK0U7QUFBdEUsa0hBQUEsY0FBYyxPQUFBO0FBQUUsMEhBQUEsc0JBQXNCLE9BQUEiLCJzb3VyY2VzQ29udGVudCI6WyIvLyBUeXBlc1xuZXhwb3J0ICogZnJvbSAnLi90eXBlcyc7XG5cbi8vIENvbmZpZ3VyYXRpb25cbmV4cG9ydCB7IENvbmZpZ01hbmFnZXIsIGdldENvbmZpZ01hbmFnZXIgfSBmcm9tICcuL2NvbmZpZyc7XG5cbi8vIEFJIFByb3ZpZGVyc1xuZXhwb3J0IHsgY3JlYXRlQUlQcm92aWRlciwgT3BlbkFJUHJvdmlkZXIsIEN1c3RvbVByb3ZpZGVyLCBDaGluZXNlUHJvdmlkZXIgfSBmcm9tICcuL2FpJztcblxuLy8gVG9vbHNcbmV4cG9ydCB7IFxuICBUb29sUmVnaXN0cnksIFxuICBnZXRUb29sUmVnaXN0cnksXG4gIEJhc2VUb29sLFxuICBSZWFkRmlsZVRvb2wsXG4gIFdyaXRlRmlsZVRvb2wsXG4gIExpc3REaXJlY3RvcnlUb29sLFxuICBHbG9iVG9vbCxcbiAgU2VhcmNoRmlsZUNvbnRlbnRUb29sLFxuICBSdW5TaGVsbENvbW1hbmRUb29sLFxuICBXZWJTZWFyY2hUb29sLFxuICBXZWJGZXRjaFRvb2wsXG4gIFJlcGxhY2VUb29sLFxuICBJbWFnZVJlYWRUb29sLFxuICBQREZFeHRyYWN0VG9vbCxcbiAgQXNrVXNlclF1ZXN0aW9uVG9vbCxcbiAgVGFza1Rvb2wsXG4gIFNhdmVNZW1vcnlUb29sXG59IGZyb20gJy4vdG9vbHMnO1xuXG4vLyBDb252ZXJzYXRpb25cbmV4cG9ydCB7IENvbnZlcnNhdGlvbiB9IGZyb20gJy4vY29udmVyc2F0aW9uJztcblxuLy8gTUNQXG5leHBvcnQgeyBNQ1BDbGllbnQsIE1DUE1hbmFnZXIgfSBmcm9tICcuL21jcCc7XG5cbi8vIFNraWxsc1xuZXhwb3J0IHsgU2tpbGxNYW5hZ2VyIH0gZnJvbSAnLi9za2lsbHMnO1xuXG4vLyBBZ2VudHNcbmV4cG9ydCB7IFxuICBBZ2VudCwgXG4gIEFnZW50RmFjdG9yeSwgXG4gIEFnZW50RGVmaW5pdGlvbixcbiAgQWdlbnRSZXN1bHQsXG4gIEFnZW50VHlwZSxcbiAgQUdFTlRfREVGSU5JVElPTlMgXG59IGZyb20gJy4vYWdlbnRzJztcblxuLy8gU2FuZGJveFxuZXhwb3J0IHsgXG4gIFNhbmRib3hFeGVjdXRvciwgXG4gIFNhbmRib3hCYWNrZW5kLFxuICBTYW5kYm94Q29uZmlnLFxuICBTYW5kYm94UmVzdWx0LFxuICBjcmVhdGVTYW5kYm94RXhlY3V0b3IgXG59IGZyb20gJy4vc2FuZGJveCc7XG5cbi8vIOWbveWGheaooeWei+W4uOmHj1xuZXhwb3J0IHsgQ0hJTkVTRV9NT0RFTFMsIERFRkFVTFRfQ0hJTkVTRV9NT0RFTFMgfSBmcm9tICcuL2FpL2NoaW5lc2UtcHJvdmlkZXInOyJdfQ==
@@ -0,0 +1,48 @@
1
+ import { EventEmitter } from 'events';
2
+ import { ToolDefinition } from '../types';
3
+ export interface MCPServerConfig {
4
+ command: string;
5
+ args?: string[];
6
+ env?: Record<string, string>;
7
+ }
8
+ export interface MCPTool {
9
+ name: string;
10
+ description: string;
11
+ inputSchema: Record<string, unknown>;
12
+ }
13
+ export interface MCPClientEvents {
14
+ 'tool-list': (tools: MCPTool[]) => void;
15
+ 'tool-result': (result: unknown) => void;
16
+ 'error': (error: Error) => void;
17
+ 'close': () => void;
18
+ }
19
+ export declare class MCPClient extends EventEmitter {
20
+ private process;
21
+ private serverName;
22
+ private config;
23
+ private requestId;
24
+ private pendingRequests;
25
+ private buffer;
26
+ private tools;
27
+ constructor(serverName: string, config: MCPServerConfig);
28
+ connect(): Promise<void>;
29
+ disconnect(): Promise<void>;
30
+ private handleData;
31
+ private handleResponse;
32
+ private sendRequest;
33
+ private loadTools;
34
+ getTools(): MCPTool[];
35
+ getToolDefinitions(): ToolDefinition[];
36
+ callTool(name: string, args: Record<string, unknown>): Promise<unknown>;
37
+ isConnected(): boolean;
38
+ }
39
+ export declare class MCPManager {
40
+ private clients;
41
+ addServer(name: string, config: MCPServerConfig): Promise<MCPClient>;
42
+ removeServer(name: string): Promise<void>;
43
+ getClient(name: string): MCPClient | undefined;
44
+ getAllClients(): MCPClient[];
45
+ getAllTools(): MCPTool[];
46
+ getAllToolDefinitions(): ToolDefinition[];
47
+ disconnectAll(): Promise<void>;
48
+ }
@@ -0,0 +1,175 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.MCPManager = exports.MCPClient = void 0;
4
+ const child_process_1 = require("child_process");
5
+ const events_1 = require("events");
6
+ class MCPClient extends events_1.EventEmitter {
7
+ process = null;
8
+ serverName;
9
+ config;
10
+ requestId = 0;
11
+ pendingRequests = new Map();
12
+ buffer = '';
13
+ tools = [];
14
+ constructor(serverName, config) {
15
+ super();
16
+ this.serverName = serverName;
17
+ this.config = config;
18
+ }
19
+ async connect() {
20
+ return new Promise((resolve, reject) => {
21
+ const env = { ...process.env, ...this.config.env };
22
+ this.process = (0, child_process_1.spawn)(this.config.command, this.config.args || [], {
23
+ env,
24
+ stdio: ['pipe', 'pipe', 'pipe']
25
+ });
26
+ this.process.stdout?.on('data', (data) => {
27
+ this.handleData(data.toString());
28
+ });
29
+ this.process.stderr?.on('data', (data) => {
30
+ console.error(`[MCP ${this.serverName}] ${data.toString()}`);
31
+ });
32
+ this.process.on('error', (error) => {
33
+ this.emit('error', error);
34
+ reject(error);
35
+ });
36
+ this.process.on('close', () => {
37
+ this.emit('close');
38
+ });
39
+ // Initialize connection
40
+ this.sendRequest('initialize', {
41
+ protocolVersion: '2024-11-05',
42
+ capabilities: {},
43
+ clientInfo: {
44
+ name: 'oflow-cli',
45
+ version: '0.1.0'
46
+ }
47
+ }).then(() => {
48
+ return this.loadTools();
49
+ }).then(() => {
50
+ resolve();
51
+ }).catch(reject);
52
+ });
53
+ }
54
+ async disconnect() {
55
+ if (this.process) {
56
+ this.process.kill();
57
+ this.process = null;
58
+ }
59
+ }
60
+ handleData(data) {
61
+ this.buffer += data;
62
+ const lines = this.buffer.split('\n');
63
+ this.buffer = lines.pop() || '';
64
+ for (const line of lines) {
65
+ if (line.trim()) {
66
+ try {
67
+ const response = JSON.parse(line);
68
+ this.handleResponse(response);
69
+ }
70
+ catch {
71
+ // Ignore parse errors
72
+ }
73
+ }
74
+ }
75
+ }
76
+ handleResponse(response) {
77
+ if (response.id !== undefined && this.pendingRequests.has(response.id)) {
78
+ const pending = this.pendingRequests.get(response.id);
79
+ this.pendingRequests.delete(response.id);
80
+ if (response.error) {
81
+ pending.reject(new Error(response.error.message));
82
+ }
83
+ else {
84
+ pending.resolve(response.result);
85
+ }
86
+ }
87
+ }
88
+ sendRequest(method, params = {}) {
89
+ return new Promise((resolve, reject) => {
90
+ const id = this.requestId++;
91
+ this.pendingRequests.set(id, { resolve, reject });
92
+ const request = JSON.stringify({
93
+ jsonrpc: '2.0',
94
+ id,
95
+ method,
96
+ params
97
+ }) + '\n';
98
+ this.process?.stdin?.write(request);
99
+ });
100
+ }
101
+ async loadTools() {
102
+ try {
103
+ const result = await this.sendRequest('tools/list');
104
+ this.tools = result.tools || [];
105
+ this.emit('tool-list', this.tools);
106
+ }
107
+ catch {
108
+ // Server might not support tools
109
+ this.tools = [];
110
+ }
111
+ }
112
+ getTools() {
113
+ return this.tools;
114
+ }
115
+ getToolDefinitions() {
116
+ return this.tools.map(tool => ({
117
+ type: 'function',
118
+ function: {
119
+ name: `${this.serverName}__${tool.name}`,
120
+ description: tool.description,
121
+ parameters: tool.inputSchema
122
+ }
123
+ }));
124
+ }
125
+ async callTool(name, args) {
126
+ const result = await this.sendRequest('tools/call', {
127
+ name,
128
+ arguments: args
129
+ });
130
+ return result;
131
+ }
132
+ isConnected() {
133
+ return this.process !== null && !this.process.killed;
134
+ }
135
+ }
136
+ exports.MCPClient = MCPClient;
137
+ class MCPManager {
138
+ clients = new Map();
139
+ async addServer(name, config) {
140
+ if (this.clients.has(name)) {
141
+ await this.removeServer(name);
142
+ }
143
+ const client = new MCPClient(name, config);
144
+ await client.connect();
145
+ this.clients.set(name, client);
146
+ return client;
147
+ }
148
+ async removeServer(name) {
149
+ const client = this.clients.get(name);
150
+ if (client) {
151
+ await client.disconnect();
152
+ this.clients.delete(name);
153
+ }
154
+ }
155
+ getClient(name) {
156
+ return this.clients.get(name);
157
+ }
158
+ getAllClients() {
159
+ return Array.from(this.clients.values());
160
+ }
161
+ getAllTools() {
162
+ return this.getAllClients().flatMap(client => client.getTools());
163
+ }
164
+ getAllToolDefinitions() {
165
+ return this.getAllClients().flatMap(client => client.getToolDefinitions());
166
+ }
167
+ async disconnectAll() {
168
+ for (const client of this.clients.values()) {
169
+ await client.disconnect();
170
+ }
171
+ this.clients.clear();
172
+ }
173
+ }
174
+ exports.MCPManager = MCPManager;
175
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/mcp/index.ts"],"names":[],"mappings":"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","sourcesContent":["import { spawn, ChildProcess } from 'child_process';\nimport { EventEmitter } from 'events';\nimport { ToolDefinition } from '../types';\n\nexport interface MCPServerConfig {\n  command: string;\n  args?: string[];\n  env?: Record<string, string>;\n}\n\nexport interface MCPTool {\n  name: string;\n  description: string;\n  inputSchema: Record<string, unknown>;\n}\n\nexport interface MCPClientEvents {\n  'tool-list': (tools: MCPTool[]) => void;\n  'tool-result': (result: unknown) => void;\n  'error': (error: Error) => void;\n  'close': () => void;\n}\n\nexport class MCPClient extends EventEmitter {\n  private process: ChildProcess | null = null;\n  private serverName: string;\n  private config: MCPServerConfig;\n  private requestId = 0;\n  private pendingRequests: Map<number, {\n    resolve: (value: unknown) => void;\n    reject: (error: Error) => void;\n  }> = new Map();\n  private buffer = '';\n  private tools: MCPTool[] = [];\n\n  constructor(serverName: string, config: MCPServerConfig) {\n    super();\n    this.serverName = serverName;\n    this.config = config;\n  }\n\n  async connect(): Promise<void> {\n    return new Promise((resolve, reject) => {\n      const env = { ...process.env, ...this.config.env };\n      \n      this.process = spawn(this.config.command, this.config.args || [], {\n        env,\n        stdio: ['pipe', 'pipe', 'pipe']\n      });\n\n      this.process.stdout?.on('data', (data: Buffer) => {\n        this.handleData(data.toString());\n      });\n\n      this.process.stderr?.on('data', (data: Buffer) => {\n        console.error(`[MCP ${this.serverName}] ${data.toString()}`);\n      });\n\n      this.process.on('error', (error: Error) => {\n        this.emit('error', error);\n        reject(error);\n      });\n\n      this.process.on('close', () => {\n        this.emit('close');\n      });\n\n      // Initialize connection\n      this.sendRequest('initialize', {\n        protocolVersion: '2024-11-05',\n        capabilities: {},\n        clientInfo: {\n          name: 'oflow-cli',\n          version: '0.1.0'\n        }\n      }).then(() => {\n        return this.loadTools();\n      }).then(() => {\n        resolve();\n      }).catch(reject);\n    });\n  }\n\n  async disconnect(): Promise<void> {\n    if (this.process) {\n      this.process.kill();\n      this.process = null;\n    }\n  }\n\n  private handleData(data: string): void {\n    this.buffer += data;\n    \n    const lines = this.buffer.split('\\n');\n    this.buffer = lines.pop() || '';\n\n    for (const line of lines) {\n      if (line.trim()) {\n        try {\n          const response = JSON.parse(line);\n          this.handleResponse(response);\n        } catch {\n          // Ignore parse errors\n        }\n      }\n    }\n  }\n\n  private handleResponse(response: any): void {\n    if (response.id !== undefined && this.pendingRequests.has(response.id)) {\n      const pending = this.pendingRequests.get(response.id)!;\n      this.pendingRequests.delete(response.id);\n\n      if (response.error) {\n        pending.reject(new Error(response.error.message));\n      } else {\n        pending.resolve(response.result);\n      }\n    }\n  }\n\n  private sendRequest(method: string, params: unknown = {}): Promise<unknown> {\n    return new Promise((resolve, reject) => {\n      const id = this.requestId++;\n      \n      this.pendingRequests.set(id, { resolve, reject });\n\n      const request = JSON.stringify({\n        jsonrpc: '2.0',\n        id,\n        method,\n        params\n      }) + '\\n';\n\n      this.process?.stdin?.write(request);\n    });\n  }\n\n  private async loadTools(): Promise<void> {\n    try {\n      const result = await this.sendRequest('tools/list') as { tools: MCPTool[] };\n      this.tools = result.tools || [];\n      this.emit('tool-list', this.tools);\n    } catch {\n      // Server might not support tools\n      this.tools = [];\n    }\n  }\n\n  getTools(): MCPTool[] {\n    return this.tools;\n  }\n\n  getToolDefinitions(): ToolDefinition[] {\n    return this.tools.map(tool => ({\n      type: 'function' as const,\n      function: {\n        name: `${this.serverName}__${tool.name}`,\n        description: tool.description,\n        parameters: tool.inputSchema\n      }\n    }));\n  }\n\n  async callTool(name: string, args: Record<string, unknown>): Promise<unknown> {\n    const result = await this.sendRequest('tools/call', {\n      name,\n      arguments: args\n    });\n    return result;\n  }\n\n  isConnected(): boolean {\n    return this.process !== null && !this.process.killed;\n  }\n}\n\nexport class MCPManager {\n  private clients: Map<string, MCPClient> = new Map();\n\n  async addServer(name: string, config: MCPServerConfig): Promise<MCPClient> {\n    if (this.clients.has(name)) {\n      await this.removeServer(name);\n    }\n\n    const client = new MCPClient(name, config);\n    await client.connect();\n    this.clients.set(name, client);\n    \n    return client;\n  }\n\n  async removeServer(name: string): Promise<void> {\n    const client = this.clients.get(name);\n    if (client) {\n      await client.disconnect();\n      this.clients.delete(name);\n    }\n  }\n\n  getClient(name: string): MCPClient | undefined {\n    return this.clients.get(name);\n  }\n\n  getAllClients(): MCPClient[] {\n    return Array.from(this.clients.values());\n  }\n\n  getAllTools(): MCPTool[] {\n    return this.getAllClients().flatMap(client => client.getTools());\n  }\n\n  getAllToolDefinitions(): ToolDefinition[] {\n    return this.getAllClients().flatMap(client => client.getToolDefinitions());\n  }\n\n  async disconnectAll(): Promise<void> {\n    for (const client of this.clients.values()) {\n      await client.disconnect();\n    }\n    this.clients.clear();\n  }\n}\n"]}