@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,31 @@
1
+ export type SandboxBackend = 'none' | 'docker' | 'podman';
2
+ export interface SandboxConfig {
3
+ backend: SandboxBackend;
4
+ image?: string;
5
+ timeout?: number;
6
+ memoryLimit?: string;
7
+ cpuLimit?: string;
8
+ networkEnabled?: boolean;
9
+ mountPoints?: {
10
+ host: string;
11
+ container: string;
12
+ }[];
13
+ }
14
+ export interface SandboxResult {
15
+ stdout: string;
16
+ stderr: string;
17
+ exitCode: number;
18
+ timedOut: boolean;
19
+ }
20
+ export declare class SandboxExecutor {
21
+ private config;
22
+ private containerId;
23
+ constructor(config?: Partial<SandboxConfig>);
24
+ execute(command: string, cwd?: string): Promise<SandboxResult>;
25
+ private executeDirectly;
26
+ private executeInDocker;
27
+ private executeInPodman;
28
+ private runContainerCommand;
29
+ static detectBackend(): Promise<SandboxBackend>;
30
+ }
31
+ export declare function createSandboxExecutor(config?: Partial<SandboxConfig>): Promise<SandboxExecutor>;
@@ -0,0 +1,197 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.SandboxExecutor = void 0;
4
+ exports.createSandboxExecutor = createSandboxExecutor;
5
+ const child_process_1 = require("child_process");
6
+ const DEFAULT_IMAGE = 'node:20-slim';
7
+ const DEFAULT_TIMEOUT = 120000; // 2 minutes
8
+ class SandboxExecutor {
9
+ config;
10
+ containerId = null;
11
+ constructor(config) {
12
+ this.config = {
13
+ backend: config?.backend || 'none',
14
+ image: config?.image || DEFAULT_IMAGE,
15
+ timeout: config?.timeout || DEFAULT_TIMEOUT,
16
+ memoryLimit: config?.memoryLimit,
17
+ cpuLimit: config?.cpuLimit,
18
+ networkEnabled: config?.networkEnabled ?? false,
19
+ mountPoints: config?.mountPoints || []
20
+ };
21
+ }
22
+ async execute(command, cwd) {
23
+ switch (this.config.backend) {
24
+ case 'docker':
25
+ return this.executeInDocker(command, cwd);
26
+ case 'podman':
27
+ return this.executeInPodman(command, cwd);
28
+ case 'none':
29
+ default:
30
+ return this.executeDirectly(command, cwd);
31
+ }
32
+ }
33
+ async executeDirectly(command, cwd) {
34
+ return new Promise((resolve) => {
35
+ const isWindows = process.platform === 'win32';
36
+ const shell = isWindows ? 'powershell.exe' : '/bin/bash';
37
+ const shellArgs = isWindows
38
+ ? ['-NoProfile', '-Command', command]
39
+ : ['-c', command];
40
+ const proc = (0, child_process_1.spawn)(shell, shellArgs, {
41
+ cwd: cwd || process.cwd(),
42
+ env: process.env
43
+ });
44
+ let stdout = '';
45
+ let stderr = '';
46
+ let timedOut = false;
47
+ const timeoutId = setTimeout(() => {
48
+ timedOut = true;
49
+ proc.kill();
50
+ }, this.config.timeout);
51
+ proc.stdout.on('data', (data) => {
52
+ stdout += data.toString();
53
+ });
54
+ proc.stderr.on('data', (data) => {
55
+ stderr += data.toString();
56
+ });
57
+ proc.on('close', (code) => {
58
+ clearTimeout(timeoutId);
59
+ resolve({
60
+ stdout,
61
+ stderr,
62
+ exitCode: code || 0,
63
+ timedOut
64
+ });
65
+ });
66
+ proc.on('error', (error) => {
67
+ clearTimeout(timeoutId);
68
+ resolve({
69
+ stdout,
70
+ stderr: error.message,
71
+ exitCode: 1,
72
+ timedOut: false
73
+ });
74
+ });
75
+ });
76
+ }
77
+ async executeInDocker(command, cwd) {
78
+ const workingDir = cwd || process.cwd();
79
+ // 构建Docker运行命令
80
+ const dockerArgs = [
81
+ 'run',
82
+ '--rm',
83
+ '-v', `${workingDir}:/workspace`,
84
+ '-w', '/workspace'
85
+ ];
86
+ // 网络设置
87
+ if (!this.config.networkEnabled) {
88
+ dockerArgs.push('--network', 'none');
89
+ }
90
+ // 内存限制
91
+ if (this.config.memoryLimit) {
92
+ dockerArgs.push('--memory', this.config.memoryLimit);
93
+ }
94
+ // CPU限制
95
+ if (this.config.cpuLimit) {
96
+ dockerArgs.push('--cpus', this.config.cpuLimit);
97
+ }
98
+ // 超时设置
99
+ dockerArgs.push('--timeout', String(Math.floor(this.config.timeout / 1000)));
100
+ // 添加镜像和命令
101
+ dockerArgs.push(this.config.image);
102
+ dockerArgs.push('sh', '-c', command);
103
+ return this.runContainerCommand('docker', dockerArgs);
104
+ }
105
+ async executeInPodman(command, cwd) {
106
+ const workingDir = cwd || process.cwd();
107
+ // Podman参数与Docker类似
108
+ const podmanArgs = [
109
+ 'run',
110
+ '--rm',
111
+ '-v', `${workingDir}:/workspace`,
112
+ '-w', '/workspace'
113
+ ];
114
+ if (!this.config.networkEnabled) {
115
+ podmanArgs.push('--network', 'none');
116
+ }
117
+ if (this.config.memoryLimit) {
118
+ podmanArgs.push('--memory', this.config.memoryLimit);
119
+ }
120
+ if (this.config.cpuLimit) {
121
+ podmanArgs.push('--cpus', this.config.cpuLimit);
122
+ }
123
+ podmanArgs.push(this.config.image);
124
+ podmanArgs.push('sh', '-c', command);
125
+ return this.runContainerCommand('podman', podmanArgs);
126
+ }
127
+ runContainerCommand(runtime, args) {
128
+ return new Promise((resolve) => {
129
+ const proc = (0, child_process_1.spawn)(runtime, args, {
130
+ env: process.env
131
+ });
132
+ let stdout = '';
133
+ let stderr = '';
134
+ let timedOut = false;
135
+ const timeoutId = setTimeout(() => {
136
+ timedOut = true;
137
+ proc.kill();
138
+ }, this.config.timeout);
139
+ proc.stdout.on('data', (data) => {
140
+ stdout += data.toString();
141
+ });
142
+ proc.stderr.on('data', (data) => {
143
+ stderr += data.toString();
144
+ });
145
+ proc.on('close', (code) => {
146
+ clearTimeout(timeoutId);
147
+ resolve({
148
+ stdout,
149
+ stderr,
150
+ exitCode: code || 0,
151
+ timedOut
152
+ });
153
+ });
154
+ proc.on('error', (error) => {
155
+ clearTimeout(timeoutId);
156
+ resolve({
157
+ stdout,
158
+ stderr: `Failed to run ${runtime}: ${error.message}`,
159
+ exitCode: 1,
160
+ timedOut: false
161
+ });
162
+ });
163
+ });
164
+ }
165
+ static async detectBackend() {
166
+ // 检查Docker是否可用
167
+ try {
168
+ const dockerCheck = await new Promise((resolve) => {
169
+ const proc = (0, child_process_1.spawn)('docker', ['--version']);
170
+ proc.on('close', (code) => resolve(code === 0));
171
+ proc.on('error', () => resolve(false));
172
+ });
173
+ if (dockerCheck)
174
+ return 'docker';
175
+ }
176
+ catch { }
177
+ // 检查Podman是否可用
178
+ try {
179
+ const podmanCheck = await new Promise((resolve) => {
180
+ const proc = (0, child_process_1.spawn)('podman', ['--version']);
181
+ proc.on('close', (code) => resolve(code === 0));
182
+ proc.on('error', () => resolve(false));
183
+ });
184
+ if (podmanCheck)
185
+ return 'podman';
186
+ }
187
+ catch { }
188
+ return 'none';
189
+ }
190
+ }
191
+ exports.SandboxExecutor = SandboxExecutor;
192
+ // 工具函数:创建沙箱执行器
193
+ async function createSandboxExecutor(config) {
194
+ const backend = config?.backend || await SandboxExecutor.detectBackend();
195
+ return new SandboxExecutor({ ...config, backend });
196
+ }
197
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/sandbox/index.ts"],"names":[],"mappings":"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","sourcesContent":["import { spawn, ChildProcess } from 'child_process';\nimport * as path from 'path';\nimport * as fs from 'fs/promises';\n\nexport type SandboxBackend = 'none' | 'docker' | 'podman';\n\nexport interface SandboxConfig {\n  backend: SandboxBackend;\n  image?: string;\n  timeout?: number;\n  memoryLimit?: string;\n  cpuLimit?: string;\n  networkEnabled?: boolean;\n  mountPoints?: { host: string; container: string }[];\n}\n\nexport interface SandboxResult {\n  stdout: string;\n  stderr: string;\n  exitCode: number;\n  timedOut: boolean;\n}\n\nconst DEFAULT_IMAGE = 'node:20-slim';\nconst DEFAULT_TIMEOUT = 120000; // 2 minutes\n\nexport class SandboxExecutor {\n  private config: SandboxConfig;\n  private containerId: string | null = null;\n\n  constructor(config?: Partial<SandboxConfig>) {\n    this.config = {\n      backend: config?.backend || 'none',\n      image: config?.image || DEFAULT_IMAGE,\n      timeout: config?.timeout || DEFAULT_TIMEOUT,\n      memoryLimit: config?.memoryLimit,\n      cpuLimit: config?.cpuLimit,\n      networkEnabled: config?.networkEnabled ?? false,\n      mountPoints: config?.mountPoints || []\n    };\n  }\n\n  async execute(command: string, cwd?: string): Promise<SandboxResult> {\n    switch (this.config.backend) {\n      case 'docker':\n        return this.executeInDocker(command, cwd);\n      case 'podman':\n        return this.executeInPodman(command, cwd);\n      case 'none':\n      default:\n        return this.executeDirectly(command, cwd);\n    }\n  }\n\n  private async executeDirectly(command: string, cwd?: string): Promise<SandboxResult> {\n    return new Promise((resolve) => {\n      const isWindows = process.platform === 'win32';\n      const shell = isWindows ? 'powershell.exe' : '/bin/bash';\n      const shellArgs = isWindows \n        ? ['-NoProfile', '-Command', command]\n        : ['-c', command];\n\n      const proc = spawn(shell, shellArgs, {\n        cwd: cwd || process.cwd(),\n        env: process.env\n      });\n\n      let stdout = '';\n      let stderr = '';\n      let timedOut = false;\n\n      const timeoutId = setTimeout(() => {\n        timedOut = true;\n        proc.kill();\n      }, this.config.timeout);\n\n      proc.stdout.on('data', (data) => {\n        stdout += data.toString();\n      });\n\n      proc.stderr.on('data', (data) => {\n        stderr += data.toString();\n      });\n\n      proc.on('close', (code) => {\n        clearTimeout(timeoutId);\n        resolve({\n          stdout,\n          stderr,\n          exitCode: code || 0,\n          timedOut\n        });\n      });\n\n      proc.on('error', (error) => {\n        clearTimeout(timeoutId);\n        resolve({\n          stdout,\n          stderr: error.message,\n          exitCode: 1,\n          timedOut: false\n        });\n      });\n    });\n  }\n\n  private async executeInDocker(command: string, cwd?: string): Promise<SandboxResult> {\n    const workingDir = cwd || process.cwd();\n    \n    // 构建Docker运行命令\n    const dockerArgs = [\n      'run',\n      '--rm',\n      '-v', `${workingDir}:/workspace`,\n      '-w', '/workspace'\n    ];\n\n    // 网络设置\n    if (!this.config.networkEnabled) {\n      dockerArgs.push('--network', 'none');\n    }\n\n    // 内存限制\n    if (this.config.memoryLimit) {\n      dockerArgs.push('--memory', this.config.memoryLimit);\n    }\n\n    // CPU限制\n    if (this.config.cpuLimit) {\n      dockerArgs.push('--cpus', this.config.cpuLimit);\n    }\n\n    // 超时设置\n    dockerArgs.push('--timeout', String(Math.floor(this.config.timeout! / 1000)));\n\n    // 添加镜像和命令\n    dockerArgs.push(this.config.image!);\n    dockerArgs.push('sh', '-c', command);\n\n    return this.runContainerCommand('docker', dockerArgs);\n  }\n\n  private async executeInPodman(command: string, cwd?: string): Promise<SandboxResult> {\n    const workingDir = cwd || process.cwd();\n    \n    // Podman参数与Docker类似\n    const podmanArgs = [\n      'run',\n      '--rm',\n      '-v', `${workingDir}:/workspace`,\n      '-w', '/workspace'\n    ];\n\n    if (!this.config.networkEnabled) {\n      podmanArgs.push('--network', 'none');\n    }\n\n    if (this.config.memoryLimit) {\n      podmanArgs.push('--memory', this.config.memoryLimit);\n    }\n\n    if (this.config.cpuLimit) {\n      podmanArgs.push('--cpus', this.config.cpuLimit);\n    }\n\n    podmanArgs.push(this.config.image!);\n    podmanArgs.push('sh', '-c', command);\n\n    return this.runContainerCommand('podman', podmanArgs);\n  }\n\n  private runContainerCommand(runtime: string, args: string[]): Promise<SandboxResult> {\n    return new Promise((resolve) => {\n      const proc = spawn(runtime, args, {\n        env: process.env\n      });\n\n      let stdout = '';\n      let stderr = '';\n      let timedOut = false;\n\n      const timeoutId = setTimeout(() => {\n        timedOut = true;\n        proc.kill();\n      }, this.config.timeout);\n\n      proc.stdout.on('data', (data) => {\n        stdout += data.toString();\n      });\n\n      proc.stderr.on('data', (data) => {\n        stderr += data.toString();\n      });\n\n      proc.on('close', (code) => {\n        clearTimeout(timeoutId);\n        resolve({\n          stdout,\n          stderr,\n          exitCode: code || 0,\n          timedOut\n        });\n      });\n\n      proc.on('error', (error) => {\n        clearTimeout(timeoutId);\n        resolve({\n          stdout,\n          stderr: `Failed to run ${runtime}: ${error.message}`,\n          exitCode: 1,\n          timedOut: false\n        });\n      });\n    });\n  }\n\n  static async detectBackend(): Promise<SandboxBackend> {\n    // 检查Docker是否可用\n    try {\n      const dockerCheck = await new Promise<boolean>((resolve) => {\n        const proc = spawn('docker', ['--version']);\n        proc.on('close', (code) => resolve(code === 0));\n        proc.on('error', () => resolve(false));\n      });\n      if (dockerCheck) return 'docker';\n    } catch {}\n\n    // 检查Podman是否可用\n    try {\n      const podmanCheck = await new Promise<boolean>((resolve) => {\n        const proc = spawn('podman', ['--version']);\n        proc.on('close', (code) => resolve(code === 0));\n        proc.on('error', () => resolve(false));\n      });\n      if (podmanCheck) return 'podman';\n    } catch {}\n\n    return 'none';\n  }\n}\n\n// 工具函数：创建沙箱执行器\nexport async function createSandboxExecutor(config?: Partial<SandboxConfig>): Promise<SandboxExecutor> {\n  const backend = config?.backend || await SandboxExecutor.detectBackend();\n  return new SandboxExecutor({ ...config, backend });\n}\n"]}
@@ -0,0 +1,16 @@
1
+ import { Skill } from '../types';
2
+ export declare class SkillManager {
3
+ private skillsDir;
4
+ private projectSkillsDir;
5
+ private skills;
6
+ constructor(skillsDir?: string, projectSkillsDir?: string);
7
+ loadSkills(): Promise<Skill[]>;
8
+ private loadSkillsFromDir;
9
+ private parseSkillFile;
10
+ getSkill(id: string): Skill | undefined;
11
+ getAllSkills(): Skill[];
12
+ getSkillContent(id: string): Promise<string | null>;
13
+ installSkill(source: string, scope?: 'global' | 'project'): Promise<Skill>;
14
+ private copyDir;
15
+ uninstallSkill(id: string): Promise<boolean>;
16
+ }
@@ -0,0 +1,169 @@
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 __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || (function () {
19
+ var ownKeys = function(o) {
20
+ ownKeys = Object.getOwnPropertyNames || function (o) {
21
+ var ar = [];
22
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
23
+ return ar;
24
+ };
25
+ return ownKeys(o);
26
+ };
27
+ return function (mod) {
28
+ if (mod && mod.__esModule) return mod;
29
+ var result = {};
30
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
31
+ __setModuleDefault(result, mod);
32
+ return result;
33
+ };
34
+ })();
35
+ Object.defineProperty(exports, "__esModule", { value: true });
36
+ exports.SkillManager = void 0;
37
+ const fs = __importStar(require("fs/promises"));
38
+ const path = __importStar(require("path"));
39
+ const SKILL_FILE = 'SKILL.md';
40
+ class SkillManager {
41
+ skillsDir;
42
+ projectSkillsDir;
43
+ skills = new Map();
44
+ constructor(skillsDir, projectSkillsDir) {
45
+ this.skillsDir = skillsDir || path.join(process.cwd(), '.oflow', 'skills');
46
+ this.projectSkillsDir = projectSkillsDir || path.join(process.cwd(), '.oflow', 'skills');
47
+ }
48
+ async loadSkills() {
49
+ this.skills.clear();
50
+ // Load global skills
51
+ await this.loadSkillsFromDir(this.skillsDir, 'global');
52
+ // Load project skills (override global with same name)
53
+ await this.loadSkillsFromDir(this.projectSkillsDir, 'project');
54
+ return this.getAllSkills();
55
+ }
56
+ async loadSkillsFromDir(dir, scope) {
57
+ try {
58
+ const entries = await fs.readdir(dir, { withFileTypes: true });
59
+ for (const entry of entries) {
60
+ if (!entry.isDirectory())
61
+ continue;
62
+ const skillPath = path.join(dir, entry.name);
63
+ const skillFile = path.join(skillPath, SKILL_FILE);
64
+ try {
65
+ const content = await fs.readFile(skillFile, 'utf-8');
66
+ const skill = this.parseSkillFile(skillPath, content);
67
+ if (skill) {
68
+ this.skills.set(skill.id, { ...skill, path: skillPath });
69
+ }
70
+ }
71
+ catch {
72
+ // Skip invalid skills
73
+ }
74
+ }
75
+ }
76
+ catch {
77
+ // Directory doesn't exist
78
+ }
79
+ }
80
+ parseSkillFile(skillPath, content) {
81
+ try {
82
+ // Parse YAML frontmatter
83
+ const frontmatterMatch = content.match(/^---\n([\s\S]*?)\n---\n([\s\S]*)$/);
84
+ if (!frontmatterMatch) {
85
+ return null;
86
+ }
87
+ const frontmatter = frontmatterMatch[1];
88
+ const mainContent = frontmatterMatch[2];
89
+ // Parse frontmatter (simple key: value parsing)
90
+ const meta = {};
91
+ frontmatter.split('\n').forEach(line => {
92
+ const match = line.match(/^(\w+):\s*(.*)$/);
93
+ if (match) {
94
+ meta[match[1]] = match[2];
95
+ }
96
+ });
97
+ return {
98
+ id: meta.id || path.basename(skillPath),
99
+ name: meta.name || path.basename(skillPath),
100
+ description: meta.description || '',
101
+ version: meta.version || '1.0.0',
102
+ path: skillPath,
103
+ mainFile: SKILL_FILE
104
+ };
105
+ }
106
+ catch {
107
+ return null;
108
+ }
109
+ }
110
+ getSkill(id) {
111
+ return this.skills.get(id);
112
+ }
113
+ getAllSkills() {
114
+ return Array.from(this.skills.values());
115
+ }
116
+ async getSkillContent(id) {
117
+ const skill = this.skills.get(id);
118
+ if (!skill)
119
+ return null;
120
+ try {
121
+ const skillFile = path.join(skill.path, SKILL_FILE);
122
+ return await fs.readFile(skillFile, 'utf-8');
123
+ }
124
+ catch {
125
+ return null;
126
+ }
127
+ }
128
+ async installSkill(source, scope = 'project') {
129
+ const targetDir = scope === 'global' ? this.skillsDir : this.projectSkillsDir;
130
+ const skillId = path.basename(source).replace(/\.git$/, '');
131
+ const targetPath = path.join(targetDir, skillId);
132
+ // For now, we'll just copy from a local path
133
+ // In a real implementation, this would support git repos, archives, etc.
134
+ await fs.mkdir(targetDir, { recursive: true });
135
+ // Copy directory
136
+ await this.copyDir(source, targetPath);
137
+ // Reload skills
138
+ await this.loadSkills();
139
+ const skill = this.skills.get(skillId);
140
+ if (!skill) {
141
+ throw new Error('Failed to load installed skill');
142
+ }
143
+ return skill;
144
+ }
145
+ async copyDir(src, dest) {
146
+ await fs.mkdir(dest, { recursive: true });
147
+ const entries = await fs.readdir(src, { withFileTypes: true });
148
+ for (const entry of entries) {
149
+ const srcPath = path.join(src, entry.name);
150
+ const destPath = path.join(dest, entry.name);
151
+ if (entry.isDirectory()) {
152
+ await this.copyDir(srcPath, destPath);
153
+ }
154
+ else {
155
+ await fs.copyFile(srcPath, destPath);
156
+ }
157
+ }
158
+ }
159
+ async uninstallSkill(id) {
160
+ const skill = this.skills.get(id);
161
+ if (!skill)
162
+ return false;
163
+ await fs.rm(skill.path, { recursive: true, force: true });
164
+ this.skills.delete(id);
165
+ return true;
166
+ }
167
+ }
168
+ exports.SkillManager = SkillManager;
169
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/skills/index.ts"],"names":[],"mappings":"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","sourcesContent":["import * as fs from 'fs/promises';\nimport * as path from 'path';\nimport { Skill } from '../types';\n\nconst SKILL_FILE = 'SKILL.md';\n\nexport class SkillManager {\n  private skillsDir: string;\n  private projectSkillsDir: string;\n  private skills: Map<string, Skill> = new Map();\n\n  constructor(skillsDir?: string, projectSkillsDir?: string) {\n    this.skillsDir = skillsDir || path.join(process.cwd(), '.oflow', 'skills');\n    this.projectSkillsDir = projectSkillsDir || path.join(process.cwd(), '.oflow', 'skills');\n  }\n\n  async loadSkills(): Promise<Skill[]> {\n    this.skills.clear();\n    \n    // Load global skills\n    await this.loadSkillsFromDir(this.skillsDir, 'global');\n    \n    // Load project skills (override global with same name)\n    await this.loadSkillsFromDir(this.projectSkillsDir, 'project');\n\n    return this.getAllSkills();\n  }\n\n  private async loadSkillsFromDir(dir: string, scope: 'global' | 'project'): Promise<void> {\n    try {\n      const entries = await fs.readdir(dir, { withFileTypes: true });\n      \n      for (const entry of entries) {\n        if (!entry.isDirectory()) continue;\n        \n        const skillPath = path.join(dir, entry.name);\n        const skillFile = path.join(skillPath, SKILL_FILE);\n        \n        try {\n          const content = await fs.readFile(skillFile, 'utf-8');\n          const skill = this.parseSkillFile(skillPath, content);\n          \n          if (skill) {\n            this.skills.set(skill.id, { ...skill, path: skillPath });\n          }\n        } catch {\n          // Skip invalid skills\n        }\n      }\n    } catch {\n      // Directory doesn't exist\n    }\n  }\n\n  private parseSkillFile(skillPath: string, content: string): Skill | null {\n    try {\n      // Parse YAML frontmatter\n      const frontmatterMatch = content.match(/^---\\n([\\s\\S]*?)\\n---\\n([\\s\\S]*)$/);\n      \n      if (!frontmatterMatch) {\n        return null;\n      }\n\n      const frontmatter = frontmatterMatch[1];\n      const mainContent = frontmatterMatch[2];\n\n      // Parse frontmatter (simple key: value parsing)\n      const meta: Record<string, string> = {};\n      frontmatter.split('\\n').forEach(line => {\n        const match = line.match(/^(\\w+):\\s*(.*)$/);\n        if (match) {\n          meta[match[1]] = match[2];\n        }\n      });\n\n      return {\n        id: meta.id || path.basename(skillPath),\n        name: meta.name || path.basename(skillPath),\n        description: meta.description || '',\n        version: meta.version || '1.0.0',\n        path: skillPath,\n        mainFile: SKILL_FILE\n      };\n    } catch {\n      return null;\n    }\n  }\n\n  getSkill(id: string): Skill | undefined {\n    return this.skills.get(id);\n  }\n\n  getAllSkills(): Skill[] {\n    return Array.from(this.skills.values());\n  }\n\n  async getSkillContent(id: string): Promise<string | null> {\n    const skill = this.skills.get(id);\n    if (!skill) return null;\n\n    try {\n      const skillFile = path.join(skill.path, SKILL_FILE);\n      return await fs.readFile(skillFile, 'utf-8');\n    } catch {\n      return null;\n    }\n  }\n\n  async installSkill(source: string, scope: 'global' | 'project' = 'project'): Promise<Skill> {\n    const targetDir = scope === 'global' ? this.skillsDir : this.projectSkillsDir;\n    const skillId = path.basename(source).replace(/\\.git$/, '');\n    const targetPath = path.join(targetDir, skillId);\n\n    // For now, we'll just copy from a local path\n    // In a real implementation, this would support git repos, archives, etc.\n    await fs.mkdir(targetDir, { recursive: true });\n    \n    // Copy directory\n    await this.copyDir(source, targetPath);\n    \n    // Reload skills\n    await this.loadSkills();\n    \n    const skill = this.skills.get(skillId);\n    if (!skill) {\n      throw new Error('Failed to load installed skill');\n    }\n    \n    return skill;\n  }\n\n  private async copyDir(src: string, dest: string): Promise<void> {\n    await fs.mkdir(dest, { recursive: true });\n    const entries = await fs.readdir(src, { withFileTypes: true });\n\n    for (const entry of entries) {\n      const srcPath = path.join(src, entry.name);\n      const destPath = path.join(dest, entry.name);\n\n      if (entry.isDirectory()) {\n        await this.copyDir(srcPath, destPath);\n      } else {\n        await fs.copyFile(srcPath, destPath);\n      }\n    }\n  }\n\n  async uninstallSkill(id: string): Promise<boolean> {\n    const skill = this.skills.get(id);\n    if (!skill) return false;\n\n    await fs.rm(skill.path, { recursive: true, force: true });\n    this.skills.delete(id);\n    \n    return true;\n  }\n}\n"]}
@@ -0,0 +1,62 @@
1
+ import { BaseTool, ToolExecuteOptions } from './base';
2
+ interface AskUserQuestionParams {
3
+ questions: Array<{
4
+ question: string;
5
+ header: string;
6
+ options: Array<{
7
+ label: string;
8
+ description: string;
9
+ }>;
10
+ multiSelect?: boolean;
11
+ }>;
12
+ }
13
+ export declare class AskUserQuestionTool extends BaseTool<AskUserQuestionParams> {
14
+ name: string;
15
+ description: string;
16
+ parameters: {
17
+ type: string;
18
+ properties: {
19
+ questions: {
20
+ type: string;
21
+ description: string;
22
+ items: {
23
+ type: string;
24
+ properties: {
25
+ question: {
26
+ type: string;
27
+ description: string;
28
+ };
29
+ header: {
30
+ type: string;
31
+ description: string;
32
+ };
33
+ options: {
34
+ type: string;
35
+ description: string;
36
+ items: {
37
+ type: string;
38
+ properties: {
39
+ label: {
40
+ type: string;
41
+ description: string;
42
+ };
43
+ description: {
44
+ type: string;
45
+ description: string;
46
+ };
47
+ };
48
+ };
49
+ };
50
+ multiSelect: {
51
+ type: string;
52
+ description: string;
53
+ };
54
+ };
55
+ };
56
+ };
57
+ };
58
+ required: string[];
59
+ };
60
+ execute(params: AskUserQuestionParams, options?: ToolExecuteOptions): Promise<string>;
61
+ }
62
+ export {};
@@ -0,0 +1,71 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.AskUserQuestionTool = void 0;
4
+ const base_1 = require("./base");
5
+ class AskUserQuestionTool extends base_1.BaseTool {
6
+ name = 'ask_user_question';
7
+ description = 'Use this tool when you need to ask the user questions during execution to gather preferences, clarify instructions, or make decisions.';
8
+ parameters = {
9
+ type: 'object',
10
+ properties: {
11
+ questions: {
12
+ type: 'array',
13
+ description: 'Questions to ask the user (1-4 questions)',
14
+ items: {
15
+ type: 'object',
16
+ properties: {
17
+ question: {
18
+ type: 'string',
19
+ description: 'The complete question to ask'
20
+ },
21
+ header: {
22
+ type: 'string',
23
+ description: 'Very short label (max 12 chars)'
24
+ },
25
+ options: {
26
+ type: 'array',
27
+ description: 'Available choices (2-4 options)',
28
+ items: {
29
+ type: 'object',
30
+ properties: {
31
+ label: {
32
+ type: 'string',
33
+ description: 'Display text for this option'
34
+ },
35
+ description: {
36
+ type: 'string',
37
+ description: 'Explanation of this option'
38
+ }
39
+ }
40
+ }
41
+ },
42
+ multiSelect: {
43
+ type: 'boolean',
44
+ description: 'Allow multiple selections'
45
+ }
46
+ }
47
+ }
48
+ }
49
+ },
50
+ required: ['questions']
51
+ };
52
+ async execute(params, options) {
53
+ this.validateRequired(params, ['questions']);
54
+ if (params.questions.length === 0 || params.questions.length > 4) {
55
+ return 'Error: Must provide 1-4 questions';
56
+ }
57
+ // 返回需要用户回答的问题
58
+ // 在交互模式下,CLI会捕获这个并显示给用户
59
+ return JSON.stringify({
60
+ type: 'user_question',
61
+ questions: params.questions.map(q => ({
62
+ question: q.question,
63
+ header: q.header,
64
+ options: q.options,
65
+ multiSelect: q.multiSelect || false
66
+ }))
67
+ });
68
+ }
69
+ }
70
+ exports.AskUserQuestionTool = AskUserQuestionTool;
71
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXNrLXVzZXItcXVlc3Rpb24uanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvdG9vbHMvYXNrLXVzZXItcXVlc3Rpb24udHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEsaUNBQXVFO0FBZ0J2RSxNQUFhLG1CQUFvQixTQUFRLGVBQStCO0lBQ3RFLElBQUksR0FBRyxtQkFBbUIsQ0FBQztJQUMzQixXQUFXLEdBQUcsd0lBQXdJLENBQUM7SUFDdkosVUFBVSxHQUFHO1FBQ1gsSUFBSSxFQUFFLFFBQVE7UUFDZCxVQUFVLEVBQUU7WUFDVixTQUFTLEVBQUU7Z0JBQ1QsSUFBSSxFQUFFLE9BQU87Z0JBQ2IsV0FBVyxFQUFFLDJDQUEyQztnQkFDeEQsS0FBSyxFQUFFO29CQUNMLElBQUksRUFBRSxRQUFRO29CQUNkLFVBQVUsRUFBRTt3QkFDVixRQUFRLEVBQUU7NEJBQ1IsSUFBSSxFQUFFLFFBQVE7NEJBQ2QsV0FBVyxFQUFFLDhCQUE4Qjt5QkFDNUM7d0JBQ0QsTUFBTSxFQUFFOzRCQUNOLElBQUksRUFBRSxRQUFROzRCQUNkLFdBQVcsRUFBRSxpQ0FBaUM7eUJBQy9DO3dCQUNELE9BQU8sRUFBRTs0QkFDUCxJQUFJLEVBQUUsT0FBTzs0QkFDYixXQUFXLEVBQUUsaUNBQWlDOzRCQUM5QyxLQUFLLEVBQUU7Z0NBQ0wsSUFBSSxFQUFFLFFBQVE7Z0NBQ2QsVUFBVSxFQUFFO29DQUNWLEtBQUssRUFBRTt3Q0FDTCxJQUFJLEVBQUUsUUFBUTt3Q0FDZCxXQUFXLEVBQUUsOEJBQThCO3FDQUM1QztvQ0FDRCxXQUFXLEVBQUU7d0NBQ1gsSUFBSSxFQUFFLFFBQVE7d0NBQ2QsV0FBVyxFQUFFLDRCQUE0QjtxQ0FDMUM7aUNBQ0Y7NkJBQ0Y7eUJBQ0Y7d0JBQ0QsV0FBVyxFQUFFOzRCQUNYLElBQUksRUFBRSxTQUFTOzRCQUNmLFdBQVcsRUFBRSwyQkFBMkI7eUJBQ3pDO3FCQUNGO2lCQUNGO2FBQ0Y7U0FDRjtRQUNELFFBQVEsRUFBRSxDQUFDLFdBQVcsQ0FBQztLQUN4QixDQUFDO0lBRUYsS0FBSyxDQUFDLE9BQU8sQ0FBQyxNQUE2QixFQUFFLE9BQTRCO1FBQ3ZFLElBQUksQ0FBQyxnQkFBZ0IsQ0FBQyxNQUFNLEVBQUUsQ0FBQyxXQUFXLENBQUMsQ0FBQyxDQUFDO1FBRTdDLElBQUksTUFBTSxDQUFDLFNBQVMsQ0FBQyxNQUFNLEtBQUssQ0FBQyxJQUFJLE1BQU0sQ0FBQyxTQUFTLENBQUMsTUFBTSxHQUFHLENBQUMsRUFBRSxDQUFDO1lBQ2pFLE9BQU8sbUNBQW1DLENBQUM7UUFDN0MsQ0FBQztRQUVELGNBQWM7UUFDZCx3QkFBd0I7UUFDeEIsT0FBTyxJQUFJLENBQUMsU0FBUyxDQUFDO1lBQ3BCLElBQUksRUFBRSxlQUFlO1lBQ3JCLFNBQVMsRUFBRSxNQUFNLENBQUMsU0FBUyxDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQUMsRUFBRSxDQUFDLENBQUM7Z0JBQ3BDLFFBQVEsRUFBRSxDQUFDLENBQUMsUUFBUTtnQkFDcEIsTUFBTSxFQUFFLENBQUMsQ0FBQyxNQUFNO2dCQUNoQixPQUFPLEVBQUUsQ0FBQyxDQUFDLE9BQU87Z0JBQ2xCLFdBQVcsRUFBRSxDQUFDLENBQUMsV0FBVyxJQUFJLEtBQUs7YUFDcEMsQ0FBQyxDQUFDO1NBQ0osQ0FBQyxDQUFDO0lBQ0wsQ0FBQztDQUNGO0FBbkVELGtEQW1FQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEJhc2VUb29sLCBUb29sRXhlY3V0ZU9wdGlvbnMsIGZvcm1hdFRvb2xFcnJvciB9IGZyb20gJy4vYmFzZSc7XG5pbXBvcnQgKiBhcyBmcyBmcm9tICdmcy9wcm9taXNlcyc7XG5pbXBvcnQgKiBhcyBwYXRoIGZyb20gJ3BhdGgnO1xuXG5pbnRlcmZhY2UgQXNrVXNlclF1ZXN0aW9uUGFyYW1zIHtcbiAgcXVlc3Rpb25zOiBBcnJheTx7XG4gICAgcXVlc3Rpb246IHN0cmluZztcbiAgICBoZWFkZXI6IHN0cmluZztcbiAgICBvcHRpb25zOiBBcnJheTx7XG4gICAgICBsYWJlbDogc3RyaW5nO1xuICAgICAgZGVzY3JpcHRpb246IHN0cmluZztcbiAgICB9PjtcbiAgICBtdWx0aVNlbGVjdD86IGJvb2xlYW47XG4gIH0+O1xufVxuXG5leHBvcnQgY2xhc3MgQXNrVXNlclF1ZXN0aW9uVG9vbCBleHRlbmRzIEJhc2VUb29sPEFza1VzZXJRdWVzdGlvblBhcmFtcz4ge1xuICBuYW1lID0gJ2Fza191c2VyX3F1ZXN0aW9uJztcbiAgZGVzY3JpcHRpb24gPSAnVXNlIHRoaXMgdG9vbCB3aGVuIHlvdSBuZWVkIHRvIGFzayB0aGUgdXNlciBxdWVzdGlvbnMgZHVyaW5nIGV4ZWN1dGlvbiB0byBnYXRoZXIgcHJlZmVyZW5jZXMsIGNsYXJpZnkgaW5zdHJ1Y3Rpb25zLCBvciBtYWtlIGRlY2lzaW9ucy4nO1xuICBwYXJhbWV0ZXJzID0ge1xuICAgIHR5cGU6ICdvYmplY3QnLFxuICAgIHByb3BlcnRpZXM6IHtcbiAgICAgIHF1ZXN0aW9uczoge1xuICAgICAgICB0eXBlOiAnYXJyYXknLFxuICAgICAgICBkZXNjcmlwdGlvbjogJ1F1ZXN0aW9ucyB0byBhc2sgdGhlIHVzZXIgKDEtNCBxdWVzdGlvbnMpJyxcbiAgICAgICAgaXRlbXM6IHtcbiAgICAgICAgICB0eXBlOiAnb2JqZWN0JyxcbiAgICAgICAgICBwcm9wZXJ0aWVzOiB7XG4gICAgICAgICAgICBxdWVzdGlvbjoge1xuICAgICAgICAgICAgICB0eXBlOiAnc3RyaW5nJyxcbiAgICAgICAgICAgICAgZGVzY3JpcHRpb246ICdUaGUgY29tcGxldGUgcXVlc3Rpb24gdG8gYXNrJ1xuICAgICAgICAgICAgfSxcbiAgICAgICAgICAgIGhlYWRlcjoge1xuICAgICAgICAgICAgICB0eXBlOiAnc3RyaW5nJyxcbiAgICAgICAgICAgICAgZGVzY3JpcHRpb246ICdWZXJ5IHNob3J0IGxhYmVsIChtYXggMTIgY2hhcnMpJ1xuICAgICAgICAgICAgfSxcbiAgICAgICAgICAgIG9wdGlvbnM6IHtcbiAgICAgICAgICAgICAgdHlwZTogJ2FycmF5JyxcbiAgICAgICAgICAgICAgZGVzY3JpcHRpb246ICdBdmFpbGFibGUgY2hvaWNlcyAoMi00IG9wdGlvbnMpJyxcbiAgICAgICAgICAgICAgaXRlbXM6IHtcbiAgICAgICAgICAgICAgICB0eXBlOiAnb2JqZWN0JyxcbiAgICAgICAgICAgICAgICBwcm9wZXJ0aWVzOiB7XG4gICAgICAgICAgICAgICAgICBsYWJlbDoge1xuICAgICAgICAgICAgICAgICAgICB0eXBlOiAnc3RyaW5nJyxcbiAgICAgICAgICAgICAgICAgICAgZGVzY3JpcHRpb246ICdEaXNwbGF5IHRleHQgZm9yIHRoaXMgb3B0aW9uJ1xuICAgICAgICAgICAgICAgICAgfSxcbiAgICAgICAgICAgICAgICAgIGRlc2NyaXB0aW9uOiB7XG4gICAgICAgICAgICAgICAgICAgIHR5cGU6ICdzdHJpbmcnLFxuICAgICAgICAgICAgICAgICAgICBkZXNjcmlwdGlvbjogJ0V4cGxhbmF0aW9uIG9mIHRoaXMgb3B0aW9uJ1xuICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfSxcbiAgICAgICAgICAgIG11bHRpU2VsZWN0OiB7XG4gICAgICAgICAgICAgIHR5cGU6ICdib29sZWFuJyxcbiAgICAgICAgICAgICAgZGVzY3JpcHRpb246ICdBbGxvdyBtdWx0aXBsZSBzZWxlY3Rpb25zJ1xuICAgICAgICAgICAgfVxuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgfVxuICAgIH0sXG4gICAgcmVxdWlyZWQ6IFsncXVlc3Rpb25zJ11cbiAgfTtcblxuICBhc3luYyBleGVjdXRlKHBhcmFtczogQXNrVXNlclF1ZXN0aW9uUGFyYW1zLCBvcHRpb25zPzogVG9vbEV4ZWN1dGVPcHRpb25zKTogUHJvbWlzZTxzdHJpbmc+IHtcbiAgICB0aGlzLnZhbGlkYXRlUmVxdWlyZWQocGFyYW1zLCBbJ3F1ZXN0aW9ucyddKTtcblxuICAgIGlmIChwYXJhbXMucXVlc3Rpb25zLmxlbmd0aCA9PT0gMCB8fCBwYXJhbXMucXVlc3Rpb25zLmxlbmd0aCA+IDQpIHtcbiAgICAgIHJldHVybiAnRXJyb3I6IE11c3QgcHJvdmlkZSAxLTQgcXVlc3Rpb25zJztcbiAgICB9XG5cbiAgICAvLyDov5Tlm57pnIDopoHnlKjmiLflm57nrZTnmoTpl67pophcbiAgICAvLyDlnKjkuqTkupLmqKHlvI/kuIvvvIxDTEnkvJrmjZXojrfov5nkuKrlubbmmL7npLrnu5nnlKjmiLdcbiAgICByZXR1cm4gSlNPTi5zdHJpbmdpZnkoe1xuICAgICAgdHlwZTogJ3VzZXJfcXVlc3Rpb24nLFxuICAgICAgcXVlc3Rpb25zOiBwYXJhbXMucXVlc3Rpb25zLm1hcChxID0+ICh7XG4gICAgICAgIHF1ZXN0aW9uOiBxLnF1ZXN0aW9uLFxuICAgICAgICBoZWFkZXI6IHEuaGVhZGVyLFxuICAgICAgICBvcHRpb25zOiBxLm9wdGlvbnMsXG4gICAgICAgIG11bHRpU2VsZWN0OiBxLm11bHRpU2VsZWN0IHx8IGZhbHNlXG4gICAgICB9KSlcbiAgICB9KTtcbiAgfVxufVxuIl19
@@ -0,0 +1,16 @@
1
+ import { ToolDefinition, ToolResult } from '../types';
2
+ export interface ToolExecuteOptions {
3
+ workingDirectory?: string;
4
+ timeout?: number;
5
+ sandbox?: boolean;
6
+ }
7
+ export declare abstract class BaseTool<TParams = Record<string, unknown>> {
8
+ abstract readonly name: string;
9
+ abstract readonly description: string;
10
+ abstract readonly parameters: Record<string, unknown>;
11
+ getDefinition(): ToolDefinition;
12
+ abstract execute(params: TParams, options?: ToolExecuteOptions): Promise<string>;
13
+ protected validateRequired(params: TParams, required: string[]): void;
14
+ }
15
+ export declare function createToolResult(toolCallId: string, content: string, isError?: boolean): ToolResult;
16
+ export declare function formatToolError(error: unknown): string;