@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.
- package/README.md +120 -0
- package/bin/sulala.mjs +7 -0
- package/dist/agent/loop.d.ts +38 -0
- package/dist/agent/loop.d.ts.map +1 -0
- package/dist/agent/loop.js +384 -0
- package/dist/agent/loop.js.map +1 -0
- package/dist/agent/session-queue.d.ts +2 -0
- package/dist/agent/session-queue.d.ts.map +1 -0
- package/dist/agent/session-queue.js +13 -0
- package/dist/agent/session-queue.js.map +1 -0
- package/dist/agent/skill-install.d.ts +41 -0
- package/dist/agent/skill-install.d.ts.map +1 -0
- package/dist/agent/skill-install.js +211 -0
- package/dist/agent/skill-install.js.map +1 -0
- package/dist/agent/skills-config.d.ts +24 -0
- package/dist/agent/skills-config.d.ts.map +1 -0
- package/dist/agent/skills-config.js +126 -0
- package/dist/agent/skills-config.js.map +1 -0
- package/dist/agent/skills-watcher.d.ts +7 -0
- package/dist/agent/skills-watcher.d.ts.map +1 -0
- package/dist/agent/skills-watcher.js +32 -0
- package/dist/agent/skills-watcher.js.map +1 -0
- package/dist/agent/skills.d.ts +32 -0
- package/dist/agent/skills.d.ts.map +1 -0
- package/dist/agent/skills.js +208 -0
- package/dist/agent/skills.js.map +1 -0
- package/dist/agent/skills.test.d.ts +2 -0
- package/dist/agent/skills.test.d.ts.map +1 -0
- package/dist/agent/skills.test.js +59 -0
- package/dist/agent/skills.test.js.map +1 -0
- package/dist/agent/tools.d.ts +8 -0
- package/dist/agent/tools.d.ts.map +1 -0
- package/dist/agent/tools.js +147 -0
- package/dist/agent/tools.js.map +1 -0
- package/dist/ai/orchestrator.d.ts +38 -0
- package/dist/ai/orchestrator.d.ts.map +1 -0
- package/dist/ai/orchestrator.js +360 -0
- package/dist/ai/orchestrator.js.map +1 -0
- package/dist/ai/orchestrator.test.d.ts +2 -0
- package/dist/ai/orchestrator.test.d.ts.map +1 -0
- package/dist/ai/orchestrator.test.js +29 -0
- package/dist/ai/orchestrator.test.js.map +1 -0
- package/dist/cli.d.ts +3 -0
- package/dist/cli.d.ts.map +1 -0
- package/dist/cli.js +278 -0
- package/dist/cli.js.map +1 -0
- package/dist/config.d.ts +4 -0
- package/dist/config.d.ts.map +1 -0
- package/dist/config.js +77 -0
- package/dist/config.js.map +1 -0
- package/dist/config.test.d.ts +2 -0
- package/dist/config.test.d.ts.map +1 -0
- package/dist/config.test.js +16 -0
- package/dist/config.test.js.map +1 -0
- package/dist/db/index.d.ts +42 -0
- package/dist/db/index.d.ts.map +1 -0
- package/dist/db/index.js +121 -0
- package/dist/db/index.js.map +1 -0
- package/dist/db/schema.sql +74 -0
- package/dist/gateway/server.d.ts +13 -0
- package/dist/gateway/server.d.ts.map +1 -0
- package/dist/gateway/server.js +566 -0
- package/dist/gateway/server.js.map +1 -0
- package/dist/index.d.ts +2 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +78 -0
- package/dist/index.js.map +1 -0
- package/dist/plugins/index.d.ts +52 -0
- package/dist/plugins/index.d.ts.map +1 -0
- package/dist/plugins/index.js +176 -0
- package/dist/plugins/index.js.map +1 -0
- package/dist/scheduler/cron.d.ts +8 -0
- package/dist/scheduler/cron.d.ts.map +1 -0
- package/dist/scheduler/cron.js +31 -0
- package/dist/scheduler/cron.js.map +1 -0
- package/dist/scheduler/queue.d.ts +13 -0
- package/dist/scheduler/queue.d.ts.map +1 -0
- package/dist/scheduler/queue.js +75 -0
- package/dist/scheduler/queue.js.map +1 -0
- package/dist/scheduler/queue.test.d.ts +2 -0
- package/dist/scheduler/queue.test.d.ts.map +1 -0
- package/dist/scheduler/queue.test.js +41 -0
- package/dist/scheduler/queue.test.js.map +1 -0
- package/dist/types.d.ts +149 -0
- package/dist/types.d.ts.map +1 -0
- package/dist/types.js +2 -0
- package/dist/types.js.map +1 -0
- package/dist/watcher/index.d.ts +15 -0
- package/dist/watcher/index.d.ts.map +1 -0
- package/dist/watcher/index.js +87 -0
- package/dist/watcher/index.js.map +1 -0
- package/dist/webhooks.d.ts +2 -0
- package/dist/webhooks.d.ts.map +1 -0
- package/dist/webhooks.js +38 -0
- package/dist/webhooks.js.map +1 -0
- package/package.json +62 -0
- package/src/db/schema.sql +74 -0
- 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 @@
|
|
|
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"}
|