@smartledger.technology/openai-claw 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 (107) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +134 -0
  3. package/dist/agent.js +262 -0
  4. package/dist/agent.js.map +1 -0
  5. package/dist/autopr/index.js +127 -0
  6. package/dist/autopr/index.js.map +1 -0
  7. package/dist/client.js +199 -0
  8. package/dist/client.js.map +1 -0
  9. package/dist/commands/index.js +624 -0
  10. package/dist/commands/index.js.map +1 -0
  11. package/dist/config.js +71 -0
  12. package/dist/config.js.map +1 -0
  13. package/dist/cost.js +97 -0
  14. package/dist/cost.js.map +1 -0
  15. package/dist/eval/cli.js +32 -0
  16. package/dist/eval/cli.js.map +1 -0
  17. package/dist/eval/index.js +134 -0
  18. package/dist/eval/index.js.map +1 -0
  19. package/dist/hooks/index.js +69 -0
  20. package/dist/hooks/index.js.map +1 -0
  21. package/dist/index.js +246 -0
  22. package/dist/index.js.map +1 -0
  23. package/dist/input.js +96 -0
  24. package/dist/input.js.map +1 -0
  25. package/dist/mcp/index.js +135 -0
  26. package/dist/mcp/index.js.map +1 -0
  27. package/dist/memory/compaction.js +70 -0
  28. package/dist/memory/compaction.js.map +1 -0
  29. package/dist/memory/index.js +56 -0
  30. package/dist/memory/index.js.map +1 -0
  31. package/dist/notifications/index.js +80 -0
  32. package/dist/notifications/index.js.map +1 -0
  33. package/dist/permissions/index.js +104 -0
  34. package/dist/permissions/index.js.map +1 -0
  35. package/dist/planmode.js +12 -0
  36. package/dist/planmode.js.map +1 -0
  37. package/dist/plugins/index.js +239 -0
  38. package/dist/plugins/index.js.map +1 -0
  39. package/dist/prompts/system.js +148 -0
  40. package/dist/prompts/system.js.map +1 -0
  41. package/dist/rag/index.js +158 -0
  42. package/dist/rag/index.js.map +1 -0
  43. package/dist/self-review/index.js +157 -0
  44. package/dist/self-review/index.js.map +1 -0
  45. package/dist/session.js +113 -0
  46. package/dist/session.js.map +1 -0
  47. package/dist/skills/index.js +39 -0
  48. package/dist/skills/index.js.map +1 -0
  49. package/dist/subagent.js +128 -0
  50. package/dist/subagent.js.map +1 -0
  51. package/dist/subagents/index.js +61 -0
  52. package/dist/subagents/index.js.map +1 -0
  53. package/dist/tools/bash.js +94 -0
  54. package/dist/tools/bash.js.map +1 -0
  55. package/dist/tools/edit.js +64 -0
  56. package/dist/tools/edit.js.map +1 -0
  57. package/dist/tools/glob.js +34 -0
  58. package/dist/tools/glob.js.map +1 -0
  59. package/dist/tools/grep.js +87 -0
  60. package/dist/tools/grep.js.map +1 -0
  61. package/dist/tools/index.js +44 -0
  62. package/dist/tools/index.js.map +1 -0
  63. package/dist/tools/ls.js +39 -0
  64. package/dist/tools/ls.js.map +1 -0
  65. package/dist/tools/read.js +126 -0
  66. package/dist/tools/read.js.map +1 -0
  67. package/dist/tools/semantic.js +40 -0
  68. package/dist/tools/semantic.js.map +1 -0
  69. package/dist/tools/shell.js +131 -0
  70. package/dist/tools/shell.js.map +1 -0
  71. package/dist/tools/task.js +91 -0
  72. package/dist/tools/task.js.map +1 -0
  73. package/dist/tools/todo.js +64 -0
  74. package/dist/tools/todo.js.map +1 -0
  75. package/dist/tools/types.js +7 -0
  76. package/dist/tools/types.js.map +1 -0
  77. package/dist/tools/webfetch.js +62 -0
  78. package/dist/tools/webfetch.js.map +1 -0
  79. package/dist/tools/websearch.js +82 -0
  80. package/dist/tools/websearch.js.map +1 -0
  81. package/dist/tools/write.js +39 -0
  82. package/dist/tools/write.js.map +1 -0
  83. package/dist/ui/repl.js +203 -0
  84. package/dist/ui/repl.js.map +1 -0
  85. package/dist/ui/tui/App.js +234 -0
  86. package/dist/ui/tui/App.js.map +1 -0
  87. package/dist/ui/tui/Diff.js +38 -0
  88. package/dist/ui/tui/Diff.js.map +1 -0
  89. package/dist/ui/tui/MessageView.js +50 -0
  90. package/dist/ui/tui/MessageView.js.map +1 -0
  91. package/dist/ui/tui/PermissionPrompt.js +47 -0
  92. package/dist/ui/tui/PermissionPrompt.js.map +1 -0
  93. package/dist/ui/tui/SlashSuggest.js +36 -0
  94. package/dist/ui/tui/SlashSuggest.js.map +1 -0
  95. package/dist/ui/tui/StatusBar.js +31 -0
  96. package/dist/ui/tui/StatusBar.js.map +1 -0
  97. package/dist/ui/tui/highlight.js +66 -0
  98. package/dist/ui/tui/highlight.js.map +1 -0
  99. package/dist/ui/tui/index.js +12 -0
  100. package/dist/ui/tui/index.js.map +1 -0
  101. package/dist/ui/tui/markdown.js +46 -0
  102. package/dist/ui/tui/markdown.js.map +1 -0
  103. package/dist/ui/tui/types.js +2 -0
  104. package/dist/ui/tui/types.js.map +1 -0
  105. package/dist/web/index.js +195 -0
  106. package/dist/web/index.js.map +1 -0
  107. package/package.json +79 -0
package/dist/index.js ADDED
@@ -0,0 +1,246 @@
1
+ #!/usr/bin/env node
2
+ import dotenv from "dotenv";
3
+ import path from "node:path";
4
+ import fs from "node:fs";
5
+ import yargs from "yargs";
6
+ import { hideBin } from "yargs/helpers";
7
+ import chalk from "chalk";
8
+ import { loadConfig } from "./config.js";
9
+ // Load .env files before anything else reads process.env.
10
+ // Order (later wins): bundled project .env, ~/.openai-claw/.env, cwd .env, cwd .env.local.
11
+ const here = path.dirname(new URL(import.meta.url).pathname);
12
+ const projectRoot = path.resolve(here, "..");
13
+ const cwd = process.cwd();
14
+ const userHomeEnv = path.join(process.env.HOME ?? "", ".openai-claw", ".env");
15
+ const envCandidates = [
16
+ path.join(projectRoot, ".env"),
17
+ path.join(projectRoot, ".env.local"),
18
+ userHomeEnv,
19
+ ...(cwd !== projectRoot ? [path.join(cwd, ".env"), path.join(cwd, ".env.local")] : []),
20
+ ];
21
+ for (const file of envCandidates) {
22
+ if (fs.existsSync(file))
23
+ dotenv.config({ path: file, override: false });
24
+ }
25
+ import { Agent } from "./agent.js";
26
+ import { getAllTools } from "./tools/index.js";
27
+ import { PermissionManager } from "./permissions/index.js";
28
+ import { runSubagent } from "./subagent.js";
29
+ import { startRepl } from "./ui/repl.js";
30
+ import { startTui } from "./ui/tui/index.js";
31
+ import { saveSession } from "./session.js";
32
+ import { loadMcpServerSpecs, startMcpServers, disconnectAll } from "./mcp/index.js";
33
+ import { prepareUserMessage } from "./input.js";
34
+ import { HookRunner } from "./hooks/index.js";
35
+ import { loadTodos } from "./tools/todo.js";
36
+ const PKG_VERSION = "0.1.0";
37
+ async function main() {
38
+ // Top-level subcommands handled before the interactive yargs parser.
39
+ const rawArgs = process.argv.slice(2);
40
+ if (rawArgs[0] === "install" || rawArgs[0] === "uninstall" || rawArgs[0] === "plugins") {
41
+ return runPluginCli(rawArgs);
42
+ }
43
+ if (rawArgs[0] === "pr") {
44
+ return runPrCli(rawArgs.slice(1));
45
+ }
46
+ if (rawArgs[0] === "dashboard") {
47
+ return runDashboardCli(rawArgs.slice(1));
48
+ }
49
+ const argv = await yargs(hideBin(process.argv))
50
+ .scriptName("claw")
51
+ .version(PKG_VERSION)
52
+ .option("model", { type: "string", describe: "Model to use (overrides config)" })
53
+ .option("workdir", { type: "string", describe: "Working directory (defaults to cwd)" })
54
+ .option("mode", {
55
+ type: "string",
56
+ choices: ["ask", "acceptEdits", "bypassPermissions", "plan"],
57
+ describe: "Permission mode",
58
+ })
59
+ .option("prompt", { alias: "p", type: "string", describe: "Run a single prompt non-interactively and exit" })
60
+ .option("continue", { type: "boolean", default: false, describe: "Resume the last saved session in this project" })
61
+ .option("append-system-prompt", { type: "string", describe: "Extra text to append to the system prompt" })
62
+ .option("dangerously-skip-permissions", {
63
+ type: "boolean",
64
+ default: false,
65
+ describe: "Run with permissionMode=bypassPermissions for this session",
66
+ })
67
+ .option("tui", { type: "boolean", default: true, describe: "Use the ink TUI (default). --no-tui falls back to readline." })
68
+ .help()
69
+ .parseAsync();
70
+ let config;
71
+ try {
72
+ config = loadConfig({
73
+ ...(argv.model ? { model: argv.model } : {}),
74
+ ...(argv.workdir ? { workdir: argv.workdir } : {}),
75
+ ...(argv.mode ? { permissionMode: argv.mode } : {}),
76
+ ...(argv["dangerously-skip-permissions"] ? { permissionMode: "bypassPermissions" } : {}),
77
+ });
78
+ }
79
+ catch (e) {
80
+ console.error(chalk.red(e.message));
81
+ process.exit(1);
82
+ }
83
+ const mcpSpecs = loadMcpServerSpecs(config);
84
+ const mcpTools = mcpSpecs.length > 0 ? await startMcpServers(mcpSpecs) : [];
85
+ if (mcpTools.length > 0) {
86
+ console.error(chalk.dim(`[mcp] loaded ${mcpTools.length} tool(s) from ${mcpSpecs.length} server(s)`));
87
+ }
88
+ const tools = [...getAllTools(config), ...mcpTools];
89
+ const permissions = new PermissionManager(config);
90
+ const hookRunner = new HookRunner(config);
91
+ loadTodos(config.memoryDir);
92
+ const systemPromptExtras = [];
93
+ if (typeof argv["append-system-prompt"] === "string" && argv["append-system-prompt"].length > 0) {
94
+ systemPromptExtras.push(argv["append-system-prompt"]);
95
+ }
96
+ const agent = new Agent({
97
+ config,
98
+ tools,
99
+ permissionCheck: (tool, input) => permissions.check(tool, input),
100
+ spawnSubagent: (req) => runSubagent(config, (t, i) => permissions.check(t, i), req),
101
+ runHook: (event, payload) => hookRunner.run(event, payload),
102
+ systemPromptExtras,
103
+ });
104
+ let printSessionId;
105
+ if (argv.continue) {
106
+ try {
107
+ const { loadSession } = await import("./session.js");
108
+ const data = loadSession(config);
109
+ if (data) {
110
+ agent.replaceConversation(data.messages);
111
+ printSessionId = data.id;
112
+ console.error(chalk.dim(`[resumed ${data.messages.length} message(s) from session ${data.id} (${data.savedAt})]`));
113
+ }
114
+ else {
115
+ console.error(chalk.dim("[no saved session to resume]"));
116
+ }
117
+ }
118
+ catch (e) {
119
+ console.error(chalk.red(`failed to resume: ${e?.message ?? e}`));
120
+ }
121
+ }
122
+ // If stdin is piped and no --prompt given, read it as the single prompt.
123
+ let promptArg = argv.prompt;
124
+ if (!promptArg && !process.stdin.isTTY) {
125
+ promptArg = await readStdinToEnd();
126
+ }
127
+ if (promptArg && promptArg.length > 0) {
128
+ let sawError = false;
129
+ agent.pushUser(prepareUserMessage(promptArg, config).content);
130
+ await agent.run((evt) => {
131
+ if (evt.type === "text_delta")
132
+ process.stdout.write(evt.data);
133
+ if (evt.type === "tool_call") {
134
+ const d = evt.data;
135
+ process.stderr.write("\n" + chalk.blue(`▸ ${d.preview ?? d.name}`) + "\n");
136
+ }
137
+ if (evt.type === "tool_result") {
138
+ const d = evt.data;
139
+ if (d.isError)
140
+ process.stderr.write(chalk.red(d.content.slice(0, 4000)) + "\n");
141
+ }
142
+ if (evt.type === "error") {
143
+ process.stderr.write(chalk.red(String(evt.data)) + "\n");
144
+ sawError = true;
145
+ }
146
+ });
147
+ process.stdout.write("\n");
148
+ try {
149
+ saveSession(config, agent.conversation, printSessionId);
150
+ }
151
+ catch { }
152
+ await disconnectAll();
153
+ if (sawError)
154
+ process.exit(1);
155
+ return;
156
+ }
157
+ if (argv.tui) {
158
+ await startTui({ agent, config, permissions });
159
+ }
160
+ else {
161
+ await startRepl({ agent, config, permissions });
162
+ }
163
+ await disconnectAll();
164
+ }
165
+ function readStdinToEnd() {
166
+ return new Promise((resolve) => {
167
+ let buf = "";
168
+ process.stdin.setEncoding("utf8");
169
+ process.stdin.on("data", (c) => (buf += c));
170
+ process.stdin.on("end", () => resolve(buf.trim()));
171
+ });
172
+ }
173
+ async function runPluginCli(args) {
174
+ const config = loadConfig();
175
+ const { installPlugin, removePlugin, listInstalled, searchRegistry } = await import("./plugins/index.js");
176
+ const [cmd, ...rest] = args;
177
+ if (cmd === "install") {
178
+ const src = rest[0];
179
+ if (!src) {
180
+ console.error(chalk.red("usage: claw install <name-or-git-url>"));
181
+ process.exit(2);
182
+ }
183
+ console.error(chalk.dim(`installing ${src}…`));
184
+ const r = installPlugin(config, src);
185
+ if (!r.ok) {
186
+ console.error(chalk.red(r.error));
187
+ process.exit(1);
188
+ }
189
+ const p = r.entry;
190
+ console.error(chalk.dim(`installed ${p.name} (${p.ref?.slice(0, 7) ?? "?"}) skills=${p.provides.skills.length} agents=${p.provides.agents.length} mcp=${p.provides.mcp.length}`));
191
+ return;
192
+ }
193
+ if (cmd === "uninstall") {
194
+ const name = rest[0];
195
+ if (!name) {
196
+ console.error(chalk.red("usage: claw uninstall <name>"));
197
+ process.exit(2);
198
+ }
199
+ const r = removePlugin(config, name);
200
+ if (!r.ok) {
201
+ console.error(chalk.red(r.error));
202
+ process.exit(1);
203
+ }
204
+ console.error(chalk.dim(`removed ${name}`));
205
+ return;
206
+ }
207
+ if (cmd === "plugins") {
208
+ const sub = rest[0] ?? "list";
209
+ if (sub === "list") {
210
+ for (const p of listInstalled(config))
211
+ console.log(`${p.name}\t${p.source}\t${p.ref ?? ""}`);
212
+ return;
213
+ }
214
+ if (sub === "search") {
215
+ const q = rest.slice(1).join(" ");
216
+ for (const h of searchRegistry(q))
217
+ console.log(`${h.name}\t${h.url}\t${h.description}`);
218
+ return;
219
+ }
220
+ console.error(chalk.red("usage: claw plugins [list|search <q>]"));
221
+ process.exit(2);
222
+ }
223
+ }
224
+ async function runPrCli(args) {
225
+ const task = args.join(" ").trim();
226
+ if (!task) {
227
+ console.error(chalk.red('usage: claw pr "<task description>"'));
228
+ process.exit(2);
229
+ }
230
+ const { runAutoPr } = await import("./autopr/index.js");
231
+ const config = loadConfig();
232
+ const ok = await runAutoPr(config, task);
233
+ process.exit(ok ? 0 : 1);
234
+ }
235
+ async function runDashboardCli(args) {
236
+ const portArg = args.find((a) => a.startsWith("--port="))?.split("=")[1];
237
+ const port = parseInt(portArg ?? "3737", 10);
238
+ const { startDashboard } = await import("./web/index.js");
239
+ const config = loadConfig();
240
+ await startDashboard(config, port);
241
+ }
242
+ main().catch((e) => {
243
+ console.error(chalk.red(e?.stack ?? e?.message ?? String(e)));
244
+ process.exit(1);
245
+ });
246
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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"}
package/dist/input.js ADDED
@@ -0,0 +1,96 @@
1
+ import fs from "node:fs";
2
+ import path from "node:path";
3
+ const IMAGE_EXTS = new Set([".png", ".jpg", ".jpeg", ".gif", ".webp"]);
4
+ const MIME = {
5
+ ".png": "image/png",
6
+ ".jpg": "image/jpeg",
7
+ ".jpeg": "image/jpeg",
8
+ ".gif": "image/gif",
9
+ ".webp": "image/webp",
10
+ };
11
+ const MAX_IMAGE_BYTES = 5 * 1024 * 1024;
12
+ /**
13
+ * Resolve `@path` references in user input.
14
+ * - Text files: inlined as <file> blocks.
15
+ * - Images (.png/.jpg/.jpeg/.gif/.webp): attached as image_url content parts.
16
+ * - Directories: listed as a directory snippet.
17
+ */
18
+ export function prepareUserMessage(input, config) {
19
+ const re = /(?:^|\s)@([^\s]+)/g;
20
+ const refs = [];
21
+ let m;
22
+ while ((m = re.exec(input)))
23
+ refs.push(m[1]);
24
+ const seen = new Set();
25
+ const textAppends = [];
26
+ const images = [];
27
+ const attachments = [];
28
+ for (const rel of refs) {
29
+ if (seen.has(rel))
30
+ continue;
31
+ seen.add(rel);
32
+ const abs = path.isAbsolute(rel) ? rel : path.join(config.workdir, rel);
33
+ const ext = path.extname(abs).toLowerCase();
34
+ if (IMAGE_EXTS.has(ext)) {
35
+ try {
36
+ const stat = fs.statSync(abs);
37
+ if (stat.size > MAX_IMAGE_BYTES) {
38
+ textAppends.push(`<image path="${abs}" error="too large (${stat.size} bytes, max ${MAX_IMAGE_BYTES})"/>`);
39
+ continue;
40
+ }
41
+ const data = fs.readFileSync(abs);
42
+ const b64 = data.toString("base64");
43
+ images.push({
44
+ type: "image_url",
45
+ image_url: { url: `data:${MIME[ext]};base64,${b64}`, detail: "auto" },
46
+ });
47
+ attachments.push(`📷 ${path.basename(abs)} (${formatSize(stat.size)})`);
48
+ }
49
+ catch (e) {
50
+ textAppends.push(`<image path="${abs}" error="${e?.message ?? String(e)}"/>`);
51
+ }
52
+ continue;
53
+ }
54
+ // Non-image: text file or directory.
55
+ try {
56
+ const stat = fs.statSync(abs);
57
+ if (stat.isDirectory()) {
58
+ const entries = fs.readdirSync(abs).slice(0, 80).join("\n");
59
+ textAppends.push(`<directory path="${abs}">\n${entries}\n</directory>`);
60
+ attachments.push(`📁 ${path.basename(abs)}/`);
61
+ continue;
62
+ }
63
+ if (stat.size > 200_000) {
64
+ textAppends.push(`<file path="${abs}" note="too large (${stat.size} bytes), use Read tool"/>`);
65
+ continue;
66
+ }
67
+ const body = fs.readFileSync(abs, "utf8");
68
+ textAppends.push(`<file path="${abs}">\n${body}\n</file>`);
69
+ attachments.push(`📄 ${path.basename(abs)}`);
70
+ }
71
+ catch {
72
+ textAppends.push(`<file path="${abs}" error="not found"/>`);
73
+ }
74
+ }
75
+ const textContent = textAppends.length ? `${input}\n\n${textAppends.join("\n\n")}` : input;
76
+ if (images.length === 0) {
77
+ return { content: textContent, attachments };
78
+ }
79
+ return {
80
+ content: [{ type: "text", text: textContent }, ...images],
81
+ attachments,
82
+ };
83
+ }
84
+ /** Legacy entry — preserved for callers that only need text expansion. */
85
+ export function expandAtRefs(input, config) {
86
+ const prep = prepareUserMessage(input, config);
87
+ return typeof prep.content === "string" ? prep.content : prep.content.find((p) => p.type === "text")?.text ?? input;
88
+ }
89
+ function formatSize(n) {
90
+ if (n < 1024)
91
+ return `${n}B`;
92
+ if (n < 1024 * 1024)
93
+ return `${(n / 1024).toFixed(1)}KB`;
94
+ return `${(n / 1024 / 1024).toFixed(1)}MB`;
95
+ }
96
+ //# sourceMappingURL=input.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"input.js","sourceRoot":"","sources":["../src/input.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,135 @@
1
+ import fs from "node:fs";
2
+ import path from "node:path";
3
+ import { Client } from "@modelcontextprotocol/sdk/client/index.js";
4
+ import { StdioClientTransport } from "@modelcontextprotocol/sdk/client/stdio.js";
5
+ import { StreamableHTTPClientTransport } from "@modelcontextprotocol/sdk/client/streamableHttp.js";
6
+ import { ok, err } from "../tools/types.js";
7
+ let connected = [];
8
+ export function loadMcpServerSpecs(config) {
9
+ const user = readJson(path.join(config.homeDir, "settings.json"));
10
+ const proj = readJson(path.join(config.workdir, ".claw", "settings.json"));
11
+ const merged = {
12
+ ...(user.mcpServers ?? {}),
13
+ ...(proj.mcpServers ?? {}),
14
+ };
15
+ return Object.entries(merged).map(([name, cfg]) => ({ name, config: cfg }));
16
+ }
17
+ export async function startMcpServers(specs) {
18
+ await disconnectAll();
19
+ for (const spec of specs) {
20
+ try {
21
+ const server = await connectOne(spec);
22
+ connected.push(server);
23
+ }
24
+ catch (e) {
25
+ console.error(`MCP server '${spec.name}' failed to start: ${e?.message ?? e}`);
26
+ }
27
+ }
28
+ return connected.flatMap((c) => c.tools);
29
+ }
30
+ export async function disconnectAll() {
31
+ for (const s of connected) {
32
+ try {
33
+ await s.client.close();
34
+ }
35
+ catch { }
36
+ }
37
+ connected = [];
38
+ }
39
+ async function connectOne(spec) {
40
+ const transport = spec.config.type === "http"
41
+ ? new StreamableHTTPClientTransport(new URL(spec.config.url), {
42
+ requestInit: { headers: spec.config.headers },
43
+ })
44
+ : new StdioClientTransport({
45
+ command: spec.config.command,
46
+ args: spec.config.args ?? [],
47
+ env: { ...process.env, ...(spec.config.env ?? {}) },
48
+ cwd: spec.config.cwd,
49
+ });
50
+ const client = new Client({ name: "openai-claw", version: "0.1.0" }, { capabilities: {} });
51
+ await client.connect(transport);
52
+ const list = await client.listTools();
53
+ const tools = list.tools.map((t) => wrapTool(spec.name, client, t));
54
+ // Resources and prompts are optional; servers may not implement them.
55
+ let resources = [];
56
+ let prompts = [];
57
+ try {
58
+ const r = await client.listResources();
59
+ resources = (r.resources ?? []).map((x) => ({
60
+ uri: x.uri,
61
+ name: x.name,
62
+ description: x.description,
63
+ }));
64
+ }
65
+ catch { }
66
+ try {
67
+ const p = await client.listPrompts();
68
+ prompts = (p.prompts ?? []).map((x) => ({
69
+ name: x.name,
70
+ description: x.description,
71
+ }));
72
+ }
73
+ catch { }
74
+ return { name: spec.name, client, tools, resources, prompts };
75
+ }
76
+ export function getMcpDirectory() {
77
+ return {
78
+ resources: connected.flatMap((c) => c.resources.map((r) => ({ server: c.name, ...r }))),
79
+ prompts: connected.flatMap((c) => c.prompts.map((p) => ({ server: c.name, ...p }))),
80
+ };
81
+ }
82
+ function wrapTool(serverName, client, remote) {
83
+ const localName = `mcp__${serverName}__${remote.name}`;
84
+ return {
85
+ name: localName,
86
+ description: `[MCP:${serverName}] ${remote.description ?? remote.name}`,
87
+ needsPermission: true,
88
+ mutates: true,
89
+ parameters: normalizeSchema(remote.inputSchema),
90
+ async run(input) {
91
+ try {
92
+ const res = await client.callTool({ name: remote.name, arguments: input });
93
+ const parts = res.content ?? [];
94
+ const text = parts
95
+ .map((p) => {
96
+ if (p.type === "text")
97
+ return p.text;
98
+ if (p.type === "resource")
99
+ return `[resource ${p.resource?.uri ?? ""}]`;
100
+ return JSON.stringify(p);
101
+ })
102
+ .join("\n");
103
+ if (res.isError)
104
+ return err(text || "(MCP tool returned isError)");
105
+ return ok(text || "(no output)");
106
+ }
107
+ catch (e) {
108
+ return err(`MCP call failed: ${e?.message ?? String(e)}`);
109
+ }
110
+ },
111
+ preview: (input) => `${localName} ${JSON.stringify(input).slice(0, 80)}`,
112
+ };
113
+ }
114
+ function normalizeSchema(schema) {
115
+ if (!schema || typeof schema !== "object") {
116
+ return { type: "object", properties: {}, additionalProperties: true };
117
+ }
118
+ return {
119
+ type: "object",
120
+ properties: schema.properties ?? {},
121
+ required: schema.required,
122
+ additionalProperties: schema.additionalProperties ?? true,
123
+ };
124
+ }
125
+ function readJson(p) {
126
+ try {
127
+ if (!fs.existsSync(p))
128
+ return {};
129
+ return JSON.parse(fs.readFileSync(p, "utf8"));
130
+ }
131
+ catch {
132
+ return {};
133
+ }
134
+ }
135
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/mcp/index.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,70 @@
1
+ const TOK_PER_CHAR = 0.25; // rough heuristic — 4 chars ≈ 1 token
2
+ export function estimateTokens(messages) {
3
+ let n = 0;
4
+ for (const m of messages) {
5
+ if (typeof m.content === "string") {
6
+ n += m.content.length;
7
+ }
8
+ else if (Array.isArray(m.content)) {
9
+ for (const part of m.content) {
10
+ if (part.type === "text")
11
+ n += part.text.length;
12
+ else if (part.type === "image_url") {
13
+ // Rough constant for an image (varies with size; ~1.5k tokens at "auto" detail).
14
+ n += 6_000;
15
+ }
16
+ }
17
+ }
18
+ if (m.tool_calls)
19
+ for (const c of m.tool_calls)
20
+ n += (c.function.arguments?.length ?? 0) + (c.function.name?.length ?? 0);
21
+ }
22
+ return Math.ceil(n * TOK_PER_CHAR);
23
+ }
24
+ /**
25
+ * If the conversation is approaching the context window, summarize older turns
26
+ * into one synthetic assistant message and keep recent turns intact.
27
+ */
28
+ export async function compactIfNeeded(messages, config, client, force = false) {
29
+ const tokens = estimateTokens(messages);
30
+ const limit = Math.floor(config.contextWindow * config.compactThreshold);
31
+ if (!force && tokens < limit)
32
+ return null;
33
+ // Keep system + last 8 messages; summarize the middle.
34
+ const sys = messages[0];
35
+ const tail = messages.slice(-8);
36
+ const middle = messages.slice(1, -8);
37
+ if (middle.length === 0)
38
+ return null;
39
+ const transcript = middle
40
+ .map((m) => {
41
+ const tag = m.role.toUpperCase();
42
+ let body = "";
43
+ if (typeof m.content === "string")
44
+ body = m.content;
45
+ else if (Array.isArray(m.content)) {
46
+ body = m.content
47
+ .map((p) => (p.type === "text" ? p.text : "[image]"))
48
+ .join(" ");
49
+ }
50
+ else
51
+ body = "(tool call)";
52
+ return `[${tag}] ${body.slice(0, 4000)}`;
53
+ })
54
+ .join("\n\n");
55
+ const summaryReq = [
56
+ {
57
+ role: "system",
58
+ content: "You compress conversation transcripts into a dense recap so the next turn can continue without losing context. Capture: user intent, decisions made, files touched, blockers, and what's next. No conversational filler.",
59
+ },
60
+ { role: "user", content: `Summarize:\n\n${transcript}` },
61
+ ];
62
+ const res = await client.complete(summaryReq, []);
63
+ const summary = res.content ?? "(compaction failed)";
64
+ return [
65
+ sys,
66
+ { role: "user", content: `<conversation-summary>\n${summary}\n</conversation-summary>` },
67
+ ...tail,
68
+ ];
69
+ }
70
+ //# sourceMappingURL=compaction.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"compaction.js","sourceRoot":"","sources":["../../src/memory/compaction.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,56 @@
1
+ import fs from "node:fs";
2
+ import path from "node:path";
3
+ import matter from "gray-matter";
4
+ export function listMemories(config) {
5
+ const dir = config.memoryDir;
6
+ if (!fs.existsSync(dir))
7
+ return [];
8
+ const files = fs.readdirSync(dir).filter((f) => f.endsWith(".md") && f !== "MEMORY.md");
9
+ const entries = [];
10
+ for (const f of files) {
11
+ const full = path.join(dir, f);
12
+ try {
13
+ const parsed = matter(fs.readFileSync(full, "utf8"));
14
+ const fm = parsed.data ?? {};
15
+ entries.push({
16
+ name: fm.name ?? f.replace(/\.md$/, ""),
17
+ description: fm.description ?? "",
18
+ type: (fm.metadata?.type ?? fm.type ?? "project"),
19
+ body: parsed.content.trim(),
20
+ file: full,
21
+ });
22
+ }
23
+ catch {
24
+ // skip malformed
25
+ }
26
+ }
27
+ return entries;
28
+ }
29
+ export function writeMemory(config, entry) {
30
+ const file = path.join(config.memoryDir, `${entry.name}.md`);
31
+ const frontmatter = matter.stringify(entry.body, {
32
+ name: entry.name,
33
+ description: entry.description,
34
+ metadata: { type: entry.type },
35
+ });
36
+ fs.writeFileSync(file, frontmatter);
37
+ updateIndex(config);
38
+ return file;
39
+ }
40
+ export function deleteMemory(config, name) {
41
+ const file = path.join(config.memoryDir, `${name}.md`);
42
+ if (!fs.existsSync(file))
43
+ return false;
44
+ fs.unlinkSync(file);
45
+ updateIndex(config);
46
+ return true;
47
+ }
48
+ function updateIndex(config) {
49
+ const entries = listMemories(config);
50
+ const lines = entries
51
+ .sort((a, b) => a.name.localeCompare(b.name))
52
+ .map((e) => `- [${e.name}](${path.basename(e.file)}) — ${e.description}`);
53
+ const idx = path.join(config.memoryDir, "MEMORY.md");
54
+ fs.writeFileSync(idx, lines.join("\n") + "\n");
55
+ }
56
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/memory/index.ts"],"names":[],"mappings":"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"}