@sulala/agent 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 (98) hide show
  1. package/README.md +120 -0
  2. package/bin/sulala.mjs +7 -0
  3. package/dist/agent/loop.d.ts +38 -0
  4. package/dist/agent/loop.d.ts.map +1 -0
  5. package/dist/agent/loop.js +384 -0
  6. package/dist/agent/loop.js.map +1 -0
  7. package/dist/agent/session-queue.d.ts +2 -0
  8. package/dist/agent/session-queue.d.ts.map +1 -0
  9. package/dist/agent/session-queue.js +13 -0
  10. package/dist/agent/session-queue.js.map +1 -0
  11. package/dist/agent/skill-install.d.ts +41 -0
  12. package/dist/agent/skill-install.d.ts.map +1 -0
  13. package/dist/agent/skill-install.js +211 -0
  14. package/dist/agent/skill-install.js.map +1 -0
  15. package/dist/agent/skills-config.d.ts +24 -0
  16. package/dist/agent/skills-config.d.ts.map +1 -0
  17. package/dist/agent/skills-config.js +126 -0
  18. package/dist/agent/skills-config.js.map +1 -0
  19. package/dist/agent/skills-watcher.d.ts +7 -0
  20. package/dist/agent/skills-watcher.d.ts.map +1 -0
  21. package/dist/agent/skills-watcher.js +32 -0
  22. package/dist/agent/skills-watcher.js.map +1 -0
  23. package/dist/agent/skills.d.ts +32 -0
  24. package/dist/agent/skills.d.ts.map +1 -0
  25. package/dist/agent/skills.js +208 -0
  26. package/dist/agent/skills.js.map +1 -0
  27. package/dist/agent/skills.test.d.ts +2 -0
  28. package/dist/agent/skills.test.d.ts.map +1 -0
  29. package/dist/agent/skills.test.js +59 -0
  30. package/dist/agent/skills.test.js.map +1 -0
  31. package/dist/agent/tools.d.ts +8 -0
  32. package/dist/agent/tools.d.ts.map +1 -0
  33. package/dist/agent/tools.js +147 -0
  34. package/dist/agent/tools.js.map +1 -0
  35. package/dist/ai/orchestrator.d.ts +38 -0
  36. package/dist/ai/orchestrator.d.ts.map +1 -0
  37. package/dist/ai/orchestrator.js +360 -0
  38. package/dist/ai/orchestrator.js.map +1 -0
  39. package/dist/ai/orchestrator.test.d.ts +2 -0
  40. package/dist/ai/orchestrator.test.d.ts.map +1 -0
  41. package/dist/ai/orchestrator.test.js +29 -0
  42. package/dist/ai/orchestrator.test.js.map +1 -0
  43. package/dist/cli.d.ts +3 -0
  44. package/dist/cli.d.ts.map +1 -0
  45. package/dist/cli.js +278 -0
  46. package/dist/cli.js.map +1 -0
  47. package/dist/config.d.ts +4 -0
  48. package/dist/config.d.ts.map +1 -0
  49. package/dist/config.js +77 -0
  50. package/dist/config.js.map +1 -0
  51. package/dist/config.test.d.ts +2 -0
  52. package/dist/config.test.d.ts.map +1 -0
  53. package/dist/config.test.js +16 -0
  54. package/dist/config.test.js.map +1 -0
  55. package/dist/db/index.d.ts +42 -0
  56. package/dist/db/index.d.ts.map +1 -0
  57. package/dist/db/index.js +121 -0
  58. package/dist/db/index.js.map +1 -0
  59. package/dist/db/schema.sql +74 -0
  60. package/dist/gateway/server.d.ts +13 -0
  61. package/dist/gateway/server.d.ts.map +1 -0
  62. package/dist/gateway/server.js +566 -0
  63. package/dist/gateway/server.js.map +1 -0
  64. package/dist/index.d.ts +2 -0
  65. package/dist/index.d.ts.map +1 -0
  66. package/dist/index.js +78 -0
  67. package/dist/index.js.map +1 -0
  68. package/dist/plugins/index.d.ts +52 -0
  69. package/dist/plugins/index.d.ts.map +1 -0
  70. package/dist/plugins/index.js +176 -0
  71. package/dist/plugins/index.js.map +1 -0
  72. package/dist/scheduler/cron.d.ts +8 -0
  73. package/dist/scheduler/cron.d.ts.map +1 -0
  74. package/dist/scheduler/cron.js +31 -0
  75. package/dist/scheduler/cron.js.map +1 -0
  76. package/dist/scheduler/queue.d.ts +13 -0
  77. package/dist/scheduler/queue.d.ts.map +1 -0
  78. package/dist/scheduler/queue.js +75 -0
  79. package/dist/scheduler/queue.js.map +1 -0
  80. package/dist/scheduler/queue.test.d.ts +2 -0
  81. package/dist/scheduler/queue.test.d.ts.map +1 -0
  82. package/dist/scheduler/queue.test.js +41 -0
  83. package/dist/scheduler/queue.test.js.map +1 -0
  84. package/dist/types.d.ts +149 -0
  85. package/dist/types.d.ts.map +1 -0
  86. package/dist/types.js +2 -0
  87. package/dist/types.js.map +1 -0
  88. package/dist/watcher/index.d.ts +15 -0
  89. package/dist/watcher/index.d.ts.map +1 -0
  90. package/dist/watcher/index.js +87 -0
  91. package/dist/watcher/index.js.map +1 -0
  92. package/dist/webhooks.d.ts +2 -0
  93. package/dist/webhooks.d.ts.map +1 -0
  94. package/dist/webhooks.js +38 -0
  95. package/dist/webhooks.js.map +1 -0
  96. package/package.json +62 -0
  97. package/src/db/schema.sql +74 -0
  98. package/src/index.ts +83 -0
@@ -0,0 +1,208 @@
1
+ import { readdirSync, readFileSync, statSync, existsSync } from 'fs';
2
+ import { join, resolve } from 'path';
3
+ import { spawnSync } from 'child_process';
4
+ import { isSkillEnabled } from './skills-config.js';
5
+ function parseFrontmatter(content) {
6
+ const match = content.match(/^---\r?\n([\s\S]*?)\r?\n---\r?\n?([\s\S]*)$/);
7
+ if (!match)
8
+ return {};
9
+ const block = match[1];
10
+ let name;
11
+ let description;
12
+ let metadataRaw;
13
+ let inMetadata = false;
14
+ const metaLines = [];
15
+ for (const line of block.split(/\r?\n/)) {
16
+ const nameMatch = line.match(/^name:\s*(.+)$/);
17
+ if (nameMatch) {
18
+ name = nameMatch[1].trim().replace(/^['"]|['"]$/g, '');
19
+ inMetadata = false;
20
+ continue;
21
+ }
22
+ const descMatch = line.match(/^description:\s*(.+)$/);
23
+ if (descMatch) {
24
+ description = descMatch[1].trim().replace(/^['"]|['"]$/g, '');
25
+ inMetadata = false;
26
+ continue;
27
+ }
28
+ const metaStart = line.match(/^metadata:\s*$/);
29
+ if (metaStart) {
30
+ inMetadata = true;
31
+ continue;
32
+ }
33
+ if (inMetadata) {
34
+ metaLines.push(line);
35
+ }
36
+ }
37
+ if (metaLines.length) {
38
+ metadataRaw = metaLines.join('\n').trim();
39
+ }
40
+ return { name, description, metadata: metadataRaw };
41
+ }
42
+ export function validateSkillContent(content) {
43
+ const errors = [];
44
+ const { name, description, metadata } = parseFrontmatter(content);
45
+ if (!name?.trim())
46
+ errors.push('Missing or empty name');
47
+ if (!description?.trim())
48
+ errors.push('Missing or empty description');
49
+ const bins = extractRequiredBins(metadata);
50
+ return {
51
+ valid: errors.length === 0,
52
+ name: name?.trim(),
53
+ description: description?.trim(),
54
+ bins: bins.length ? bins : undefined,
55
+ errors,
56
+ };
57
+ }
58
+ function extractRequiredBins(metadataRaw) {
59
+ if (!metadataRaw)
60
+ return [];
61
+ try {
62
+ const jsonStr = metadataRaw.replace(/^\s+/gm, ' ').replace(/\s+/g, ' ');
63
+ const parsed = JSON.parse(jsonStr);
64
+ return (parsed?.sulala?.requires?.bins || []);
65
+ }
66
+ catch {
67
+ const m = metadataRaw.match(/"bins"\s*:\s*\[([^\]]*)\]/);
68
+ if (!m)
69
+ return [];
70
+ const inner = m[1];
71
+ return inner
72
+ .split(',')
73
+ .map((s) => s.trim().replace(/^["']|["']$/g, ''))
74
+ .filter(Boolean);
75
+ }
76
+ }
77
+ function extractRequiredEnv(metadataRaw) {
78
+ if (!metadataRaw)
79
+ return [];
80
+ try {
81
+ const jsonStr = metadataRaw.replace(/^\s+/gm, ' ').replace(/\s+/g, ' ');
82
+ const parsed = JSON.parse(jsonStr);
83
+ const env = parsed?.sulala?.requires?.env;
84
+ return Array.isArray(env) ? env.filter((e) => typeof e === 'string' && e.trim().length > 0) : [];
85
+ }
86
+ catch {
87
+ const m = metadataRaw.match(/"env"\s*:\s*\[([^\]]*)\]/);
88
+ if (!m)
89
+ return [];
90
+ const inner = m[1];
91
+ return inner
92
+ .split(',')
93
+ .map((s) => s.trim().replace(/^["']|["']$/g, ''))
94
+ .filter(Boolean);
95
+ }
96
+ }
97
+ function binAvailable(bin) {
98
+ const allowed = (process.env.ALLOWED_BINARIES || '')
99
+ .split(',')
100
+ .map((s) => s.trim().toLowerCase())
101
+ .filter(Boolean);
102
+ if (allowed.includes(bin.toLowerCase()))
103
+ return true;
104
+ const r = spawnSync('which', [bin], { encoding: 'utf8' });
105
+ return r.status === 0 && !!r.stdout?.trim();
106
+ }
107
+ /** Collect all required bins from enabled skills (for auto-merge with ALLOWED_BINARIES). */
108
+ export function getAllRequiredBins(config) {
109
+ const skills = listSkills(config);
110
+ const bins = new Set();
111
+ for (const s of skills) {
112
+ for (const b of s.bins || []) {
113
+ if (b?.trim())
114
+ bins.add(b.trim().toLowerCase());
115
+ }
116
+ }
117
+ return [...bins];
118
+ }
119
+ /** Paths in precedence order: workspace > managed > bundled > extra. */
120
+ export function getSkillPaths(config) {
121
+ const out = [];
122
+ const cwd = process.cwd();
123
+ if (config.agentContextPath?.trim()) {
124
+ out.push({
125
+ path: resolve(cwd, config.agentContextPath.trim()),
126
+ source: 'workspace',
127
+ });
128
+ }
129
+ out.push({ path: config.skillsManagedDir, source: 'managed' });
130
+ out.push({ path: config.skillsBundledDir, source: 'bundled' });
131
+ for (const p of config.skillsPluginDirs) {
132
+ out.push({ path: p, source: 'plugin' });
133
+ }
134
+ for (const p of config.skillsExtraDirs) {
135
+ out.push({ path: resolve(cwd, p), source: 'extra' });
136
+ }
137
+ return out;
138
+ }
139
+ function scanDirForSkills(base, source) {
140
+ if (!existsSync(base))
141
+ return [];
142
+ const skills = [];
143
+ try {
144
+ const stat = statSync(base);
145
+ const files = [];
146
+ if (stat.isFile() && (base.endsWith('.md') || base.endsWith('.txt'))) {
147
+ files.push({ path: base, name: base.split(/[/\\]/).pop() || '' });
148
+ }
149
+ else if (stat.isDirectory()) {
150
+ const names = readdirSync(base).sort();
151
+ for (const n of names) {
152
+ if (!n.endsWith('.md') && !n.endsWith('.txt'))
153
+ continue;
154
+ files.push({ path: join(base, n), name: n });
155
+ }
156
+ }
157
+ for (const { path: fullPath, name: fileName } of files) {
158
+ if (!fileName.endsWith('.md'))
159
+ continue;
160
+ try {
161
+ const raw = readFileSync(fullPath, 'utf8');
162
+ const { name, description, metadata } = parseFrontmatter(raw);
163
+ if (!name || !description)
164
+ continue;
165
+ const bins = extractRequiredBins(metadata);
166
+ const env = extractRequiredEnv(metadata);
167
+ const missing = [];
168
+ for (const b of bins) {
169
+ if (!binAvailable(b))
170
+ missing.push(b);
171
+ }
172
+ const status = bins.length === 0 ? 'unknown' : missing.length === 0 ? 'eligible' : 'blocked';
173
+ skills.push({
174
+ name,
175
+ description,
176
+ filePath: fullPath,
177
+ status,
178
+ source,
179
+ bins: bins.length ? bins : undefined,
180
+ env: env.length ? env : undefined,
181
+ ...(missing.length ? { missing } : {}),
182
+ });
183
+ }
184
+ catch {
185
+ // skip
186
+ }
187
+ }
188
+ }
189
+ catch {
190
+ // skip
191
+ }
192
+ return skills;
193
+ }
194
+ /** List skills from all paths with precedence: workspace > managed > bundled > extra. By default filtered by skills.entries.<name>.enabled; set includeDisabled true to return all (e.g. for dashboard). */
195
+ export function listSkills(config, options) {
196
+ const paths = getSkillPaths(config);
197
+ const byName = new Map();
198
+ for (const { path: p, source } of paths) {
199
+ for (const s of scanDirForSkills(p, source)) {
200
+ if (!byName.has(s.name)) {
201
+ if (options?.includeDisabled || isSkillEnabled(s.name))
202
+ byName.set(s.name, s);
203
+ }
204
+ }
205
+ }
206
+ return Array.from(byName.values()).sort((a, b) => a.name.localeCompare(b.name));
207
+ }
208
+ //# sourceMappingURL=skills.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"skills.js","sourceRoot":"","sources":["../../src/agent/skills.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=skills.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"skills.test.d.ts","sourceRoot":"","sources":["../../src/agent/skills.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,59 @@
1
+ import { describe, it, expect } from "vitest";
2
+ import { validateSkillContent } from "./skills.js";
3
+ describe("skills", () => {
4
+ describe("validateSkillContent", () => {
5
+ it("accepts valid skill with name and description", () => {
6
+ const content = `---
7
+ name: test-skill
8
+ description: A test skill
9
+ ---
10
+ # Body`;
11
+ const r = validateSkillContent(content);
12
+ expect(r.valid).toBe(true);
13
+ expect(r.name).toBe("test-skill");
14
+ expect(r.description).toBe("A test skill");
15
+ expect(r.errors).toHaveLength(0);
16
+ });
17
+ it("rejects missing name", () => {
18
+ const content = `---
19
+ description: Only description
20
+ ---
21
+ # Body`;
22
+ const r = validateSkillContent(content);
23
+ expect(r.valid).toBe(false);
24
+ expect(r.errors).toContain("Missing or empty name");
25
+ });
26
+ it("rejects missing description", () => {
27
+ const content = `---
28
+ name: no-desc
29
+ ---
30
+ # Body`;
31
+ const r = validateSkillContent(content);
32
+ expect(r.valid).toBe(false);
33
+ expect(r.errors).toContain("Missing or empty description");
34
+ });
35
+ it("extracts required bins from metadata", () => {
36
+ const content = `---
37
+ name: git
38
+ description: Git ops
39
+ metadata:
40
+ { "sulala": { "requires": { "bins": ["git"] } } }
41
+ ---
42
+ # Body`;
43
+ const r = validateSkillContent(content);
44
+ expect(r.valid).toBe(true);
45
+ expect(r.bins).toEqual(["git"]);
46
+ });
47
+ it("accepts skill without metadata", () => {
48
+ const content = `---
49
+ name: simple
50
+ description: No bins
51
+ ---
52
+ # Body`;
53
+ const r = validateSkillContent(content);
54
+ expect(r.valid).toBe(true);
55
+ expect(r.bins).toBeUndefined();
56
+ });
57
+ });
58
+ });
59
+ //# sourceMappingURL=skills.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"skills.test.js","sourceRoot":"","sources":["../../src/agent/skills.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,EAAE,EAAE,MAAM,EAAE,MAAM,QAAQ,CAAC;AAC9C,OAAO,EAAE,oBAAoB,EAAE,MAAM,aAAa,CAAC;AAEnD,QAAQ,CAAC,QAAQ,EAAE,GAAG,EAAE;IACtB,QAAQ,CAAC,sBAAsB,EAAE,GAAG,EAAE;QACpC,EAAE,CAAC,+CAA+C,EAAE,GAAG,EAAE;YACvD,MAAM,OAAO,GAAG;;;;OAIf,CAAC;YACF,MAAM,CAAC,GAAG,oBAAoB,CAAC,OAAO,CAAC,CAAC;YACxC,MAAM,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YAC3B,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;YAClC,MAAM,CAAC,CAAC,CAAC,WAAW,CAAC,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC;YAC3C,MAAM,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC;QACnC,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,sBAAsB,EAAE,GAAG,EAAE;YAC9B,MAAM,OAAO,GAAG;;;OAGf,CAAC;YACF,MAAM,CAAC,GAAG,oBAAoB,CAAC,OAAO,CAAC,CAAC;YACxC,MAAM,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;YAC5B,MAAM,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,SAAS,CAAC,uBAAuB,CAAC,CAAC;QACtD,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,6BAA6B,EAAE,GAAG,EAAE;YACrC,MAAM,OAAO,GAAG;;;OAGf,CAAC;YACF,MAAM,CAAC,GAAG,oBAAoB,CAAC,OAAO,CAAC,CAAC;YACxC,MAAM,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;YAC5B,MAAM,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,SAAS,CAAC,8BAA8B,CAAC,CAAC;QAC7D,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,sCAAsC,EAAE,GAAG,EAAE;YAC9C,MAAM,OAAO,GAAG;;;;;;OAMf,CAAC;YACF,MAAM,CAAC,GAAG,oBAAoB,CAAC,OAAO,CAAC,CAAC;YACxC,MAAM,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YAC3B,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,OAAO,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC;QAClC,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,gCAAgC,EAAE,GAAG,EAAE;YACxC,MAAM,OAAO,GAAG;;;;OAIf,CAAC;YACF,MAAM,CAAC,GAAG,oBAAoB,CAAC,OAAO,CAAC,CAAC;YACxC,MAAM,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YAC3B,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,aAAa,EAAE,CAAC;QACjC,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}
@@ -0,0 +1,8 @@
1
+ import type { ToolDef } from '../types.js';
2
+ export declare function registerTool(tool: ToolDef): void;
3
+ export declare function getTool(name: string): ToolDef | undefined;
4
+ export declare function listTools(): ToolDef[];
5
+ export declare function executeTool(name: string, args: Record<string, unknown>): Promise<unknown>;
6
+ /** Built-in: enqueue a task (type + optional payload) */
7
+ export declare function registerBuiltInTools(enqueueTask: (taskId: string) => void): void;
8
+ //# sourceMappingURL=tools.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"tools.d.ts","sourceRoot":"","sources":["../../src/agent/tools.ts"],"names":[],"mappings":"AAOA,OAAO,KAAK,EAAE,OAAO,EAAE,MAAM,aAAa,CAAC;AAI3C,wBAAgB,YAAY,CAAC,IAAI,EAAE,OAAO,GAAG,IAAI,CAEhD;AAED,wBAAgB,OAAO,CAAC,IAAI,EAAE,MAAM,GAAG,OAAO,GAAG,SAAS,CAEzD;AAED,wBAAgB,SAAS,IAAI,OAAO,EAAE,CAErC;AAED,wBAAgB,WAAW,CAAC,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,GAAG,OAAO,CAAC,OAAO,CAAC,CAKzF;AAED,yDAAyD;AACzD,wBAAgB,oBAAoB,CAAC,WAAW,EAAE,CAAC,MAAM,EAAE,MAAM,KAAK,IAAI,GAAG,IAAI,CAkHhF"}
@@ -0,0 +1,147 @@
1
+ import { readFileSync, existsSync } from 'fs';
2
+ import { resolve, relative } from 'path';
3
+ import { spawnSync } from 'child_process';
4
+ import { insertTask } from '../db/index.js';
5
+ import { config } from '../config.js';
6
+ import { getAllRequiredBins } from './skills.js';
7
+ import { getSkillConfigEnv } from './skills-config.js';
8
+ const registry = new Map();
9
+ export function registerTool(tool) {
10
+ registry.set(tool.name, tool);
11
+ }
12
+ export function getTool(name) {
13
+ return registry.get(name);
14
+ }
15
+ export function listTools() {
16
+ return [...registry.values()];
17
+ }
18
+ export function executeTool(name, args) {
19
+ const tool = registry.get(name);
20
+ if (!tool)
21
+ throw new Error(`Unknown tool: ${name}`);
22
+ const result = tool.execute(args);
23
+ return Promise.resolve(result);
24
+ }
25
+ /** Built-in: enqueue a task (type + optional payload) */
26
+ export function registerBuiltInTools(enqueueTask) {
27
+ registerTool({
28
+ name: 'run_task',
29
+ description: 'Enqueue a background task by type and optional payload. Use for scheduling work (e.g. heartbeat, file_event, or custom types).',
30
+ parameters: {
31
+ type: 'object',
32
+ properties: {
33
+ type: { type: 'string', description: 'Task type (e.g. heartbeat, file_event)' },
34
+ payload: { type: 'object', description: 'Optional JSON payload' },
35
+ },
36
+ required: ['type'],
37
+ },
38
+ execute: (args) => {
39
+ const type = args.type;
40
+ const payload = args.payload ?? null;
41
+ const taskId = `task_${Date.now()}_${Math.random().toString(36).slice(2, 9)}`;
42
+ insertTask({ id: taskId, type, payload: payload ?? undefined });
43
+ enqueueTask(taskId);
44
+ return { taskId, type, status: 'enqueued' };
45
+ },
46
+ });
47
+ if (process.env.ALLOW_SHELL_TOOL === '1') {
48
+ const envBins = (process.env.ALLOWED_BINARIES || '')
49
+ .split(',')
50
+ .map((b) => b.trim().toLowerCase())
51
+ .filter(Boolean);
52
+ const skillBins = getAllRequiredBins(config);
53
+ const allowed = [...new Set([...envBins, ...skillBins])];
54
+ registerTool({
55
+ name: 'run_command',
56
+ description: 'Run a single command (binary + args). Only binaries in the ALLOWED_BINARIES list are permitted. Use for skills that document CLI usage (e.g. memo for Apple Notes): read the skill doc, then run the commands it describes.',
57
+ parameters: {
58
+ type: 'object',
59
+ properties: {
60
+ binary: { type: 'string', description: 'Executable name (e.g. memo, git). Must be in ALLOWED_BINARIES.' },
61
+ args: {
62
+ type: 'array',
63
+ items: { type: 'string' },
64
+ description: 'Arguments (e.g. ["notes", "-a", "buy saiko"] for memo).',
65
+ },
66
+ },
67
+ required: ['binary', 'args'],
68
+ },
69
+ execute: (args) => {
70
+ const binary = String(args.binary || '').trim().toLowerCase();
71
+ if (!binary)
72
+ return { error: 'binary is required' };
73
+ if (allowed.length > 0 && !allowed.includes(binary)) {
74
+ return { error: `binary "${binary}" is not in ALLOWED_BINARIES (allowed: ${allowed.join(', ')})` };
75
+ }
76
+ const rawArgs = Array.isArray(args.args) ? args.args : [];
77
+ const argsList = rawArgs.map((a) => String(a).replace(/\0/g, '').trim()).filter((_, i) => i < 50);
78
+ if (binary === 'curl') {
79
+ const allowedHosts = (process.env.ALLOWED_CURL_HOSTS || '').split(',').map((h) => h.trim().toLowerCase()).filter(Boolean);
80
+ if (allowedHosts.length > 0) {
81
+ const urls = argsList.join(' ').match(/https?:\/\/[^/\s"'<>]+/g) || [];
82
+ for (const u of urls) {
83
+ try {
84
+ const host = new URL(u).hostname.toLowerCase();
85
+ const allowed = allowedHosts.some((h) => host === h || host.endsWith('.' + h));
86
+ if (!allowed)
87
+ return { error: `curl URL host not allowed: ${host} (ALLOWED_CURL_HOSTS)` };
88
+ }
89
+ catch {
90
+ // skip malformed URL
91
+ }
92
+ }
93
+ }
94
+ }
95
+ try {
96
+ const skillEnv = getSkillConfigEnv();
97
+ const env = { ...process.env, ...skillEnv };
98
+ const result = spawnSync(binary, argsList, { encoding: 'utf8', timeout: 30000, env });
99
+ if (result.error)
100
+ return { error: result.error.message };
101
+ return {
102
+ status: result.status ?? null,
103
+ stdout: result.stdout?.trim() ?? '',
104
+ stderr: result.stderr?.trim() ?? '',
105
+ };
106
+ }
107
+ catch (e) {
108
+ return { error: e.message };
109
+ }
110
+ },
111
+ });
112
+ }
113
+ if (config.agentWorkspaceRoot) {
114
+ const workspaceRoot = resolve(process.cwd(), config.agentWorkspaceRoot);
115
+ registerTool({
116
+ name: 'read_file',
117
+ description: 'Read the contents of a file from the workspace. Path is relative to the workspace root.',
118
+ parameters: {
119
+ type: 'object',
120
+ properties: {
121
+ path: { type: 'string', description: 'Relative path to the file (e.g. README.md, src/index.ts)' },
122
+ },
123
+ required: ['path'],
124
+ },
125
+ execute: (args) => {
126
+ const rawPath = args.path || '';
127
+ if (!rawPath.trim())
128
+ return { error: 'path is required' };
129
+ const requested = resolve(workspaceRoot, rawPath.trim());
130
+ const relativePath = relative(workspaceRoot, requested);
131
+ if (relativePath.startsWith('..')) {
132
+ return { error: 'path must be inside the workspace' };
133
+ }
134
+ if (!existsSync(requested))
135
+ return { error: `file not found: ${rawPath}` };
136
+ try {
137
+ const content = readFileSync(requested, 'utf8');
138
+ return { path: rawPath.trim(), content };
139
+ }
140
+ catch (e) {
141
+ return { error: e.message };
142
+ }
143
+ },
144
+ });
145
+ }
146
+ }
147
+ //# sourceMappingURL=tools.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"tools.js","sourceRoot":"","sources":["../../src/agent/tools.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,38 @@
1
+ import type { AIAdapter, CompleteOptions } from '../types.js';
2
+ export declare function registerProvider(name: string, adapter: AIAdapter): void;
3
+ export declare function getProvider(name?: string | null): AIAdapter;
4
+ export type StreamChunkEvent = {
5
+ type: 'delta';
6
+ content: string;
7
+ } | {
8
+ type: 'finish';
9
+ content: string;
10
+ tool_calls?: Array<{
11
+ id: string;
12
+ name: string;
13
+ arguments: string;
14
+ }>;
15
+ usage?: Record<string, number>;
16
+ };
17
+ /** Stream completion (OpenAI and OpenRouter). Falls back to non-streaming for other providers. */
18
+ export declare function completeStream(options: CompleteOptions, onChunk: (ev: StreamChunkEvent) => void): Promise<{
19
+ content: string;
20
+ tool_calls?: Array<{
21
+ id: string;
22
+ name: string;
23
+ arguments: string;
24
+ }>;
25
+ usage?: Record<string, number>;
26
+ }>;
27
+ export declare function complete(options?: CompleteOptions): Promise<{
28
+ id: string;
29
+ content: string;
30
+ usage?: Record<string, number>;
31
+ meta?: Record<string, unknown>;
32
+ tool_calls?: Array<{
33
+ id: string;
34
+ name: string;
35
+ arguments: string;
36
+ }>;
37
+ }>;
38
+ //# sourceMappingURL=orchestrator.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"orchestrator.d.ts","sourceRoot":"","sources":["../../src/ai/orchestrator.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,SAAS,EAAE,eAAe,EAAE,MAAM,aAAa,CAAC;AAK9D,wBAAgB,gBAAgB,CAAC,IAAI,EAAE,MAAM,EAAE,OAAO,EAAE,SAAS,GAAG,IAAI,CAEvE;AAED,wBAAgB,WAAW,CAAC,IAAI,CAAC,EAAE,MAAM,GAAG,IAAI,GAAG,SAAS,CAK3D;AAED,MAAM,MAAM,gBAAgB,GACxB;IAAE,IAAI,EAAE,OAAO,CAAC;IAAC,OAAO,EAAE,MAAM,CAAA;CAAE,GAClC;IAAE,IAAI,EAAE,QAAQ,CAAC;IAAC,OAAO,EAAE,MAAM,CAAC;IAAC,UAAU,CAAC,EAAE,KAAK,CAAC;QAAE,EAAE,EAAE,MAAM,CAAC;QAAC,IAAI,EAAE,MAAM,CAAC;QAAC,SAAS,EAAE,MAAM,CAAA;KAAE,CAAC,CAAC;IAAC,KAAK,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAAA;CAAE,CAAC;AAE7I,kGAAkG;AAClG,wBAAsB,cAAc,CAClC,OAAO,EAAE,eAAe,EACxB,OAAO,EAAE,CAAC,EAAE,EAAE,gBAAgB,KAAK,IAAI,GACtC,OAAO,CAAC;IAAE,OAAO,EAAE,MAAM,CAAC;IAAC,UAAU,CAAC,EAAE,KAAK,CAAC;QAAE,EAAE,EAAE,MAAM,CAAC;QAAC,IAAI,EAAE,MAAM,CAAC;QAAC,SAAS,EAAE,MAAM,CAAA;KAAE,CAAC,CAAC;IAAC,KAAK,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAAA;CAAE,CAAC,CAwEnI;AAMD,wBAAsB,QAAQ,CAAC,OAAO,GAAE,eAAoB,GAAG,OAAO,CAAC;IACrE,EAAE,EAAE,MAAM,CAAC;IACX,OAAO,EAAE,MAAM,CAAC;IAChB,KAAK,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;IAC/B,IAAI,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;IAC/B,UAAU,CAAC,EAAE,KAAK,CAAC;QAAE,EAAE,EAAE,MAAM,CAAC;QAAC,IAAI,EAAE,MAAM,CAAC;QAAC,SAAS,EAAE,MAAM,CAAA;KAAE,CAAC,CAAC;CACrE,CAAC,CAkCD"}