@1claw/cli 0.34.7 → 0.35.1

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 (77) hide show
  1. package/README.md +66 -5
  2. package/dist/bin/1claw.js +0 -0
  3. package/dist/src/commands/containers.d.ts +3 -0
  4. package/dist/src/commands/containers.d.ts.map +1 -0
  5. package/dist/src/commands/containers.js +130 -0
  6. package/dist/src/commands/containers.js.map +1 -0
  7. package/dist/src/commands/daemon.js +18 -9
  8. package/dist/src/commands/daemon.js.map +1 -1
  9. package/dist/src/commands/deploy.d.ts +3 -0
  10. package/dist/src/commands/deploy.d.ts.map +1 -0
  11. package/dist/src/commands/deploy.js +158 -0
  12. package/dist/src/commands/deploy.js.map +1 -0
  13. package/dist/src/commands/eject.d.ts +3 -0
  14. package/dist/src/commands/eject.d.ts.map +1 -0
  15. package/dist/src/commands/eject.js +84 -0
  16. package/dist/src/commands/eject.js.map +1 -0
  17. package/dist/src/commands/init.d.ts +3 -0
  18. package/dist/src/commands/init.d.ts.map +1 -0
  19. package/dist/src/commands/init.js +490 -0
  20. package/dist/src/commands/init.js.map +1 -0
  21. package/dist/src/commands/local.d.ts.map +1 -1
  22. package/dist/src/commands/local.js +61 -14
  23. package/dist/src/commands/local.js.map +1 -1
  24. package/dist/src/commands/publish.d.ts +3 -0
  25. package/dist/src/commands/publish.d.ts.map +1 -0
  26. package/dist/src/commands/publish.js +153 -0
  27. package/dist/src/commands/publish.js.map +1 -0
  28. package/dist/src/deploy/google-cloud/main.tf +74 -0
  29. package/dist/src/deploy/google-cloud/outputs.tf +9 -0
  30. package/dist/src/deploy/google-cloud/variables.tf +44 -0
  31. package/dist/src/docker/base/Dockerfile +34 -0
  32. package/dist/src/docker/base/chat-ui/index.html +110 -0
  33. package/dist/src/docker/base/chat-ui/server.js +233 -0
  34. package/dist/src/docker/base/entrypoint.sh +61 -0
  35. package/dist/src/docker/base/healthcheck.sh +2 -0
  36. package/dist/src/docker/compose.yaml +20 -0
  37. package/dist/src/docker/templates/Dockerfile.tmpl +7 -0
  38. package/dist/src/index.d.ts.map +1 -1
  39. package/dist/src/index.js +11 -0
  40. package/dist/src/index.js.map +1 -1
  41. package/dist/src/lib/container-config.d.ts +43 -0
  42. package/dist/src/lib/container-config.d.ts.map +1 -0
  43. package/dist/src/lib/container-config.js +98 -0
  44. package/dist/src/lib/container-config.js.map +1 -0
  45. package/dist/src/lib/daemon-control.d.ts +10 -0
  46. package/dist/src/lib/daemon-control.d.ts.map +1 -0
  47. package/dist/src/lib/daemon-control.js +58 -0
  48. package/dist/src/lib/daemon-control.js.map +1 -0
  49. package/dist/src/lib/docker-client.d.ts +58 -0
  50. package/dist/src/lib/docker-client.d.ts.map +1 -0
  51. package/dist/src/lib/docker-client.js +193 -0
  52. package/dist/src/lib/docker-client.js.map +1 -0
  53. package/dist/src/lib/image-build.d.ts +30 -0
  54. package/dist/src/lib/image-build.d.ts.map +1 -0
  55. package/dist/src/lib/image-build.js +74 -0
  56. package/dist/src/lib/image-build.js.map +1 -0
  57. package/dist/src/lib/paths.d.ts +14 -0
  58. package/dist/src/lib/paths.d.ts.map +1 -0
  59. package/dist/src/lib/paths.js +38 -0
  60. package/dist/src/lib/paths.js.map +1 -0
  61. package/dist/src/modules/ampersend/config.json +8 -0
  62. package/dist/src/modules/ampersend/module.yaml +32 -0
  63. package/dist/src/modules/ampersend/startup.sh +4 -0
  64. package/dist/src/modules/elizaos/config.json +7 -0
  65. package/dist/src/modules/elizaos/module.yaml +30 -0
  66. package/dist/src/modules/elizaos/startup.sh +3 -0
  67. package/dist/src/modules/langchain/module.yaml +30 -0
  68. package/dist/src/modules/langchain/startup.sh +3 -0
  69. package/dist/src/modules/onchain/module.yaml +28 -0
  70. package/dist/src/modules/onchain/startup.sh +3 -0
  71. package/dist/src/modules/registry.d.ts +48 -0
  72. package/dist/src/modules/registry.d.ts.map +1 -0
  73. package/dist/src/modules/registry.js +201 -0
  74. package/dist/src/modules/registry.js.map +1 -0
  75. package/dist/src/modules/scaffold-agent/module.yaml +30 -0
  76. package/dist/src/modules/scaffold-agent/startup.sh +3 -0
  77. package/package.json +4 -3
@@ -0,0 +1,233 @@
1
+ #!/usr/bin/env node
2
+ // 1Claw agent chat UI — a tiny, zero-dependency HTTP server.
3
+ //
4
+ // It serves a single-page chat interface and bridges browser requests to the
5
+ // host daemon over the mounted Unix socket. Secret VALUES never transit this
6
+ // process: the daemon injects them into outbound requests and returns only the
7
+ // upstream response. This server only ever sees secret NAMES and metadata.
8
+ "use strict";
9
+
10
+ const http = require("node:http");
11
+ const { readFileSync } = require("node:fs");
12
+ const { join } = require("node:path");
13
+
14
+ const PORT = parseInt(process.env.CHAT_UI_PORT || "3000", 10);
15
+ const SOCKET = process.env.ONECLAW_DAEMON_SOCKET || "/run/1claw/daemon.sock";
16
+ const AGENT_ID = process.env.ONECLAW_AGENT_ID || "";
17
+ const MODULES = (process.env.ONECLAW_CONTAINER_MODULES || "")
18
+ .split(",")
19
+ .map((m) => m.trim())
20
+ .filter(Boolean);
21
+ const MODE = process.env.ONECLAW_LOCAL_VAULT === "true" ? "local" : "cloud";
22
+
23
+ let INDEX_HTML = "<h1>1Claw Agent Running</h1>";
24
+ try {
25
+ INDEX_HTML = readFileSync(join(__dirname, "index.html"), "utf-8");
26
+ } catch {
27
+ /* fall back to placeholder */
28
+ }
29
+
30
+ /** Call the host daemon over the Unix socket. */
31
+ function daemonRequest(method, path, body) {
32
+ return new Promise((resolve, reject) => {
33
+ const payload = body ? JSON.stringify(body) : undefined;
34
+ const req = http.request(
35
+ {
36
+ socketPath: SOCKET,
37
+ path,
38
+ method,
39
+ headers: {
40
+ "Content-Type": "application/json",
41
+ ...(payload
42
+ ? { "Content-Length": Buffer.byteLength(payload) }
43
+ : {}),
44
+ },
45
+ timeout: 10000,
46
+ },
47
+ (res) => {
48
+ const chunks = [];
49
+ res.on("data", (c) => chunks.push(c));
50
+ res.on("end", () => {
51
+ const text = Buffer.concat(chunks).toString("utf-8");
52
+ let parsed = null;
53
+ try {
54
+ parsed = text ? JSON.parse(text) : null;
55
+ } catch {
56
+ parsed = { raw: text };
57
+ }
58
+ resolve({ status: res.statusCode || 0, body: parsed });
59
+ });
60
+ },
61
+ );
62
+ req.on("error", reject);
63
+ req.on("timeout", () => req.destroy(new Error("daemon timeout")));
64
+ if (payload) req.write(payload);
65
+ req.end();
66
+ });
67
+ }
68
+
69
+ function json(res, status, obj) {
70
+ const body = JSON.stringify(obj);
71
+ res.writeHead(status, {
72
+ "Content-Type": "application/json",
73
+ "Content-Length": Buffer.byteLength(body),
74
+ });
75
+ res.end(body);
76
+ }
77
+
78
+ function readBody(req) {
79
+ return new Promise((resolve, reject) => {
80
+ const chunks = [];
81
+ let size = 0;
82
+ req.on("data", (c) => {
83
+ size += c.length;
84
+ if (size > 1024 * 1024) {
85
+ reject(new Error("body too large"));
86
+ req.destroy();
87
+ return;
88
+ }
89
+ chunks.push(c);
90
+ });
91
+ req.on("end", () => resolve(Buffer.concat(chunks).toString("utf-8")));
92
+ req.on("error", reject);
93
+ });
94
+ }
95
+
96
+ async function daemonReachable() {
97
+ if (MODE !== "local") return false;
98
+ try {
99
+ const r = await daemonRequest("GET", "/health");
100
+ return r.status === 200;
101
+ } catch {
102
+ return false;
103
+ }
104
+ }
105
+
106
+ // A minimal, LLM-optional assistant. Demonstrates the daemon trust boundary
107
+ // with slash commands; if an LLM provider is wired in later it replaces this.
108
+ async function handleChat(message) {
109
+ const text = (message || "").trim();
110
+
111
+ if (text === "/help" || text === "") {
112
+ return {
113
+ reply:
114
+ "1Claw agent ready. Commands:\n" +
115
+ " /secrets — list secret names available via the daemon\n" +
116
+ " /info — show this agent's runtime info\n" +
117
+ " /proxy <name> <url> — make a request with the secret injected by the daemon\n" +
118
+ "Secret values never enter this container.",
119
+ tool: null,
120
+ };
121
+ }
122
+
123
+ if (text === "/info") {
124
+ return {
125
+ reply: `Agent ${AGENT_ID || "(local)"} · mode=${MODE} · modules=${MODULES.join(", ") || "none"}`,
126
+ tool: "info",
127
+ };
128
+ }
129
+
130
+ if (text === "/secrets") {
131
+ try {
132
+ const r = await daemonRequest("GET", "/secrets");
133
+ if (r.status !== 200) {
134
+ return { reply: `Daemon returned ${r.status}`, tool: "list_secrets" };
135
+ }
136
+ const names = (r.body.secrets || []).map((s) => s.name);
137
+ return {
138
+ reply: names.length
139
+ ? "Available secrets:\n" + names.map((n) => " • " + n).join("\n")
140
+ : "No secrets available to this agent.",
141
+ tool: "list_secrets",
142
+ };
143
+ } catch (err) {
144
+ return { reply: `Could not reach daemon: ${err.message}`, tool: "list_secrets" };
145
+ }
146
+ }
147
+
148
+ if (text.startsWith("/proxy ")) {
149
+ const parts = text.split(/\s+/);
150
+ const secretName = parts[1];
151
+ const url = parts[2];
152
+ if (!secretName || !url) {
153
+ return { reply: "Usage: /proxy <secret-name> <url>", tool: null };
154
+ }
155
+ try {
156
+ const r = await daemonRequest("POST", "/proxy", {
157
+ secretName,
158
+ url,
159
+ method: "GET",
160
+ });
161
+ return {
162
+ reply:
163
+ r.status === 200
164
+ ? `Proxied request (secret injected by daemon). Upstream status: ${r.body.status}`
165
+ : `Daemon denied request (${r.status}): ${r.body && r.body.error}`,
166
+ tool: "proxy_request",
167
+ };
168
+ } catch (err) {
169
+ return { reply: `Proxy error: ${err.message}`, tool: "proxy_request" };
170
+ }
171
+ }
172
+
173
+ return {
174
+ reply:
175
+ "No LLM provider is configured in this container yet. " +
176
+ "Try /help, /secrets, /info, or /proxy. " +
177
+ "Wire a model via a module to enable conversational replies.",
178
+ tool: null,
179
+ };
180
+ }
181
+
182
+ const server = http.createServer(async (req, res) => {
183
+ const url = req.url || "/";
184
+ const method = req.method || "GET";
185
+
186
+ if (url === "/health") {
187
+ return json(res, 200, { status: "ok", agent: AGENT_ID || null, mode: MODE });
188
+ }
189
+
190
+ if (url === "/api/info" && method === "GET") {
191
+ return json(res, 200, {
192
+ agentId: AGENT_ID || null,
193
+ modules: MODULES,
194
+ mode: MODE,
195
+ daemonReachable: await daemonReachable(),
196
+ });
197
+ }
198
+
199
+ if (url === "/api/secrets" && method === "GET") {
200
+ try {
201
+ const r = await daemonRequest("GET", "/secrets");
202
+ return json(res, r.status, r.body || { secrets: [] });
203
+ } catch (err) {
204
+ return json(res, 502, { error: `daemon unreachable: ${err.message}` });
205
+ }
206
+ }
207
+
208
+ if (url === "/api/chat" && method === "POST") {
209
+ let body;
210
+ try {
211
+ body = JSON.parse((await readBody(req)) || "{}");
212
+ } catch {
213
+ return json(res, 400, { error: "invalid JSON" });
214
+ }
215
+ const result = await handleChat(body.message);
216
+ return json(res, 200, result);
217
+ }
218
+
219
+ if (url === "/" || url === "/index.html") {
220
+ res.writeHead(200, { "Content-Type": "text/html; charset=utf-8" });
221
+ return res.end(INDEX_HTML);
222
+ }
223
+
224
+ json(res, 404, { error: "not found" });
225
+ });
226
+
227
+ server.listen(PORT, "0.0.0.0", () => {
228
+ console.log(`[chat-ui] listening on :${PORT} (mode=${MODE})`);
229
+ });
230
+
231
+ for (const sig of ["SIGINT", "SIGTERM"]) {
232
+ process.on(sig, () => server.close(() => process.exit(0)));
233
+ }
@@ -0,0 +1,61 @@
1
+ #!/bin/sh
2
+ # 1Claw agent container entrypoint.
3
+ # Supports two modes:
4
+ # local (ONECLAW_LOCAL_VAULT=true) → credentials injected by the host daemon
5
+ # over the mounted Unix socket. The key
6
+ # NEVER enters the container.
7
+ # cloud (ONECLAW_LOCAL_VAULT=false) → the agent API key is provided directly
8
+ # (e.g. via a Secret Manager mount).
9
+ set -e
10
+
11
+ CHAT_UI_PORT="${CHAT_UI_PORT:-3000}"
12
+
13
+ echo "─────────────────────────────────────────────"
14
+ echo " 1Claw agent container starting"
15
+ echo " Agent ID: ${ONECLAW_AGENT_ID:-not set}"
16
+ echo " Modules: ${ONECLAW_CONTAINER_MODULES:-none}"
17
+
18
+ if [ "$ONECLAW_LOCAL_VAULT" = "true" ]; then
19
+ echo " Mode: local (daemon socket)"
20
+ echo " Socket: $ONECLAW_DAEMON_SOCKET"
21
+ if [ ! -S "$ONECLAW_DAEMON_SOCKET" ]; then
22
+ echo ""
23
+ echo "ERROR: Daemon socket not found at $ONECLAW_DAEMON_SOCKET"
24
+ echo ""
25
+ echo "The 1Claw daemon must be running on the host with the socket mounted:"
26
+ echo " 1claw daemon start"
27
+ echo " docker run -v ~/.config/1claw/daemon.sock:/run/1claw/daemon.sock:ro ..."
28
+ exit 1
29
+ fi
30
+ echo "─────────────────────────────────────────────"
31
+ # Start the MCP server in local daemon mode (best-effort).
32
+ if command -v 1claw-mcp >/dev/null 2>&1; then
33
+ ONECLAW_LOCAL_VAULT=true 1claw-mcp --local 2>/tmp/mcp.log &
34
+ else
35
+ echo "NOTE: @1claw/mcp not installed in image; chat UI uses the daemon directly."
36
+ fi
37
+ else
38
+ echo " Mode: cloud (agent API key)"
39
+ if [ -z "$ONECLAW_AGENT_API_KEY" ]; then
40
+ echo ""
41
+ echo "ERROR: ONECLAW_AGENT_API_KEY is not set (cloud mode)."
42
+ echo "Provide it via a Secret Manager mount or -e ONECLAW_AGENT_API_KEY=..."
43
+ exit 1
44
+ fi
45
+ echo "─────────────────────────────────────────────"
46
+ if command -v 1claw-mcp >/dev/null 2>&1; then
47
+ 1claw-mcp 2>/tmp/mcp.log &
48
+ fi
49
+ fi
50
+
51
+ # Run module startup hooks (each module may drop an executable startup.sh).
52
+ for hook in /app/modules/*/startup.sh; do
53
+ if [ -x "$hook" ]; then
54
+ echo "Running module hook: $hook"
55
+ "$hook" || echo "WARN: module hook $hook exited non-zero"
56
+ fi
57
+ done
58
+
59
+ # Start the chat UI (the container's health anchor) in the foreground.
60
+ echo "Ready: http://0.0.0.0:${CHAT_UI_PORT}"
61
+ exec node /app/chat-ui/server.js
@@ -0,0 +1,2 @@
1
+ #!/bin/sh
2
+ curl -sf "http://localhost:${CHAT_UI_PORT:-3000}/health" >/dev/null || exit 1
@@ -0,0 +1,20 @@
1
+ # docker-compose template for a 1Claw agent container (local daemon mode).
2
+ # Generated by `1claw eject`. The daemon must be running on the host:
3
+ # 1claw daemon start
4
+ services:
5
+ agent:
6
+ image: {{IMAGE}}
7
+ container_name: {{NAME}}
8
+ restart: unless-stopped
9
+ ports:
10
+ - "{{PORT}}:3000"
11
+ volumes:
12
+ # Host daemon socket — credentials are injected here; the container
13
+ # never receives the raw agent key.
14
+ - {{DAEMON_SOCKET}}:/run/1claw/daemon.sock:ro
15
+ environment:
16
+ ONECLAW_LOCAL_VAULT: "true"
17
+ ONECLAW_DAEMON_SOCKET: /run/1claw/daemon.sock
18
+ ONECLAW_AGENT_ID: "{{AGENT_ID}}"
19
+ ONECLAW_CONTAINER_MODULES: "{{MODULES}}"
20
+ CHAT_UI_PORT: "3000"
@@ -0,0 +1,7 @@
1
+ # Generated by `1claw init --docker` / `1claw eject`.
2
+ # Base image provides @1claw/mcp + chat UI + entrypoint. Module layers are
3
+ # appended below. Edit freely, then build with:
4
+ # docker build -t my-agent .
5
+ FROM {{BASE_IMAGE}}
6
+
7
+ {{MODULE_LAYERS}}
@@ -1 +1 @@
1
- {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/index.ts"],"names":[],"mappings":"AAGA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AA+CpC,wBAAgB,aAAa,IAAI,OAAO,CA0EvC"}
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/index.ts"],"names":[],"mappings":"AAGA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAoDpC,wBAAgB,aAAa,IAAI,OAAO,CAiFvC"}
package/dist/src/index.js CHANGED
@@ -26,6 +26,11 @@ import { webhookCommand } from "./commands/webhook.js";
26
26
  import { platformCommand } from "./commands/platform.js";
27
27
  import { deviceCommand } from "./commands/device.js";
28
28
  import { approvalCommand } from "./commands/approval.js";
29
+ import { initCommand } from "./commands/init.js";
30
+ import { publishCommand } from "./commands/publish.js";
31
+ import { ejectCommand } from "./commands/eject.js";
32
+ import { containersCommand } from "./commands/containers.js";
33
+ import { deployCommand } from "./commands/deploy.js";
29
34
  import { setOutputFormat, setApiUrl } from "./config.js";
30
35
  const __dirname = dirname(fileURLToPath(import.meta.url));
31
36
  // Compiled to dist/src/index.js → ../../package.json; ts-node from src/ → ../package.json
@@ -65,6 +70,12 @@ export function createProgram() {
65
70
  program.addCommand(importCommand);
66
71
  program.addCommand(localCommand);
67
72
  program.addCommand(daemonCommand);
73
+ // Containerized agent runtime
74
+ program.addCommand(initCommand);
75
+ program.addCommand(publishCommand);
76
+ program.addCommand(ejectCommand);
77
+ program.addCommand(containersCommand);
78
+ program.addCommand(deployCommand);
68
79
  // Config
69
80
  program.addCommand(configCommand);
70
81
  // Treasury
@@ -1 +1 @@
1
- {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/index.ts"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,YAAY,EAAE,MAAM,SAAS,CAAC;AACnD,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AAC1C,OAAO,EAAE,aAAa,EAAE,MAAM,UAAU,CAAC;AACzC,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AACpC,OAAO,EACH,YAAY,EACZ,aAAa,EACb,aAAa,GAChB,MAAM,qBAAqB,CAAC;AAC7B,OAAO,EACH,qBAAqB,EACrB,oBAAoB,GACvB,MAAM,8BAA8B,CAAC;AACtC,OAAO,EACH,kBAAkB,EAClB,kBAAkB,GACrB,MAAM,uBAAuB,CAAC;AAC/B,OAAO,EAAE,WAAW,EAAE,MAAM,oBAAoB,CAAC;AACjD,OAAO,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAC;AACnD,OAAO,EAAE,aAAa,EAAE,MAAM,sBAAsB,CAAC;AACrD,OAAO,EAAE,UAAU,EAAE,MAAM,mBAAmB,CAAC;AAC/C,OAAO,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAC;AACnD,OAAO,EAAE,aAAa,EAAE,MAAM,sBAAsB,CAAC;AACrD,OAAO,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAC;AACnD,OAAO,EAAE,cAAc,EAAE,MAAM,uBAAuB,CAAC;AACvD,OAAO,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAC;AACnD,OAAO,EAAE,UAAU,EAAE,MAAM,mBAAmB,CAAC;AAC/C,OAAO,EAAE,aAAa,EAAE,MAAM,sBAAsB,CAAC;AACrD,OAAO,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAC;AACnD,OAAO,EAAE,aAAa,EAAE,MAAM,sBAAsB,CAAC;AACrD,OAAO,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAC;AACnD,OAAO,EAAE,aAAa,EAAE,MAAM,sBAAsB,CAAC;AACrD,OAAO,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAC;AACnD,OAAO,EAAE,eAAe,EAAE,MAAM,wBAAwB,CAAC;AACzD,OAAO,EAAE,cAAc,EAAE,MAAM,uBAAuB,CAAC;AACvD,OAAO,EAAE,eAAe,EAAE,MAAM,wBAAwB,CAAC;AACzD,OAAO,EAAE,aAAa,EAAE,MAAM,sBAAsB,CAAC;AACrD,OAAO,EAAE,eAAe,EAAE,MAAM,wBAAwB,CAAC;AACzD,OAAO,EAAE,WAAW,EAAE,MAAM,oBAAoB,CAAC;AACjD,OAAO,EAAE,cAAc,EAAE,MAAM,uBAAuB,CAAC;AACvD,OAAO,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAC;AACnD,OAAO,EAAE,iBAAiB,EAAE,MAAM,0BAA0B,CAAC;AAC7D,OAAO,EAAE,aAAa,EAAE,MAAM,sBAAsB,CAAC;AACrD,OAAO,EAAE,eAAe,EAAE,SAAS,EAAE,MAAM,aAAa,CAAC;AAEzD,MAAM,SAAS,GAAG,OAAO,CAAC,aAAa,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC;AAC1D,0FAA0F;AAC1F,MAAM,WAAW,GAAG,CAAC,IAAI,CAAC,SAAS,EAAE,oBAAoB,CAAC,EAAE,IAAI,CAAC,SAAS,EAAE,iBAAiB,CAAC,CAAC,CAAC,IAAI,CAChG,CAAC,CAAC,EAAE,EAAE,CAAC,UAAU,CAAC,CAAC,CAAC,CACvB,CAAC;AACF,IAAI,CAAC,WAAW,EAAE,CAAC;IACf,MAAM,IAAI,KAAK,CAAC,iDAAiD,CAAC,CAAC;AACvE,CAAC;AACD,MAAM,iBAAiB,GAAG,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC,WAAW,EAAE,MAAM,CAAC,CAAC,CAAC,OAAiB,CAAC;AAE1F,MAAM,UAAU,aAAa;IACzB,MAAM,OAAO,GAAG,IAAI,OAAO,CAAC,OAAO,CAAC;SAC/B,OAAO,CAAC,iBAAiB,CAAC;SAC1B,WAAW,CACR,mEAAmE,CACtE,CAAC;IAEN,OAAO;IACP,OAAO,CAAC,UAAU,CAAC,YAAY,CAAC,CAAC;IACjC,OAAO,CAAC,UAAU,CAAC,aAAa,CAAC,CAAC;IAClC,OAAO,CAAC,UAAU,CAAC,aAAa,CAAC,CAAC;IAClC,OAAO,CAAC,UAAU,CAAC,qBAAqB,CAAC,CAAC;IAC1C,OAAO,CAAC,UAAU,CAAC,oBAAoB,CAAC,CAAC;IACzC,OAAO,CAAC,UAAU,CAAC,kBAAkB,CAAC,CAAC;IACvC,OAAO,CAAC,UAAU,CAAC,kBAAkB,CAAC,CAAC;IACvC,OAAO,CAAC,UAAU,CAAC,WAAW,CAAC,CAAC;IAEhC,iBAAiB;IACjB,OAAO,CAAC,UAAU,CAAC,YAAY,CAAC,CAAC;IACjC,OAAO,CAAC,UAAU,CAAC,aAAa,CAAC,CAAC;IAClC,OAAO,CAAC,UAAU,CAAC,UAAU,CAAC,CAAC;IAC/B,OAAO,CAAC,UAAU,CAAC,YAAY,CAAC,CAAC;IACjC,OAAO,CAAC,UAAU,CAAC,aAAa,CAAC,CAAC;IAElC,UAAU;IACV,OAAO,CAAC,UAAU,CAAC,YAAY,CAAC,CAAC;IAEjC,UAAU;IACV,OAAO,CAAC,UAAU,CAAC,cAAc,CAAC,CAAC;IAEnC,WAAW;IACX,OAAO,CAAC,UAAU,CAAC,YAAY,CAAC,CAAC;IACjC,OAAO,CAAC,UAAU,CAAC,UAAU,CAAC,CAAC;IAE/B,8BAA8B;IAC9B,OAAO,CAAC,UAAU,CAAC,YAAY,CAAC,CAAC;IACjC,OAAO,CAAC,UAAU,CAAC,aAAa,CAAC,CAAC;IAClC,OAAO,CAAC,UAAU,CAAC,YAAY,CAAC,CAAC;IACjC,OAAO,CAAC,UAAU,CAAC,aAAa,CAAC,CAAC;IAElC,8BAA8B;IAC9B,OAAO,CAAC,UAAU,CAAC,WAAW,CAAC,CAAC;IAChC,OAAO,CAAC,UAAU,CAAC,cAAc,CAAC,CAAC;IACnC,OAAO,CAAC,UAAU,CAAC,YAAY,CAAC,CAAC;IACjC,OAAO,CAAC,UAAU,CAAC,iBAAiB,CAAC,CAAC;IACtC,OAAO,CAAC,UAAU,CAAC,aAAa,CAAC,CAAC;IAElC,SAAS;IACT,OAAO,CAAC,UAAU,CAAC,aAAa,CAAC,CAAC;IAElC,WAAW;IACX,OAAO,CAAC,UAAU,CAAC,eAAe,CAAC,CAAC;IAEpC,WAAW;IACX,OAAO,CAAC,UAAU,CAAC,cAAc,CAAC,CAAC;IAEnC,QAAQ;IACR,OAAO,CAAC,UAAU,CAAC,YAAY,CAAC,CAAC;IAEjC,WAAW;IACX,OAAO,CAAC,UAAU,CAAC,eAAe,CAAC,CAAC;IAEpC,mBAAmB;IACnB,OAAO,CAAC,UAAU,CAAC,aAAa,CAAC,CAAC;IAClC,OAAO,CAAC,UAAU,CAAC,eAAe,CAAC,CAAC;IAEpC,iBAAiB;IACjB,OAAO,CAAC,MAAM,CAAC,QAAQ,EAAE,oCAAoC,CAAC,CAAC;IAC/D,OAAO,CAAC,MAAM,CAAC,iBAAiB,EAAE,sCAAsC,CAAC,CAAC;IAE1E,OAAO,CAAC,IAAI,CAAC,WAAW,EAAE,GAAG,EAAE;QAC3B,MAAM,UAAU,GAAG,OAAO,CAAC,IAAI,EAAE,CAAC;QAClC,IAAI,UAAU,CAAC,IAAI,EAAE,CAAC;YAClB,eAAe,CAAC,MAAM,CAAC,CAAC;QAC5B,CAAC;QACD,IAAI,UAAU,CAAC,MAAM,EAAE,CAAC;YACpB,SAAS,CAAC,UAAU,CAAC,MAAM,CAAC,CAAC;QACjC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,OAAO,OAAO,CAAC;AACnB,CAAC"}
@@ -0,0 +1,43 @@
1
+ export declare const CONTAINERS_DIR: string;
2
+ export interface ContainerState {
3
+ containerName: string;
4
+ containerId: string | null;
5
+ agentId: string | null;
6
+ vaultId: string | null;
7
+ image: string;
8
+ modules: string[];
9
+ port: number;
10
+ createdAt: string;
11
+ /** Local-vault path where the daemon stores the agent key (cloud mode only). */
12
+ localVaultPath: string | null;
13
+ /** Set by `1claw publish` once an image is pushed to a registry. */
14
+ customImage: string | null;
15
+ publishedAt?: string;
16
+ /** "local" (daemon socket) or "cloud" (agent API key direct). */
17
+ mode: "local" | "cloud";
18
+ /** Deployment info, populated by `1claw deploy`. */
19
+ deployment?: {
20
+ provider: string;
21
+ serviceUrl?: string;
22
+ region?: string;
23
+ deployedAt: string;
24
+ };
25
+ }
26
+ /** A short, URL/identifier-safe random id (8 hex chars). */
27
+ export declare function shortId(): string;
28
+ /** Generate a default container name: docker-agent-<shortId>. */
29
+ export declare function generateContainerName(): string;
30
+ /** Restrict names to docker-safe characters. */
31
+ export declare function sanitizeName(name: string): string;
32
+ export declare function isValidContainerName(name: string): boolean;
33
+ export declare function saveContainerState(state: ContainerState): void;
34
+ export declare function loadContainerState(name: string): ContainerState | null;
35
+ export declare function deleteContainerState(name: string): boolean;
36
+ export declare function listContainerStates(): ContainerState[];
37
+ /** Check whether a TCP port is free to bind on localhost. */
38
+ export declare function isPortAvailable(port: number): Promise<boolean>;
39
+ /** Find the first available port at or after `start` (bounded search). */
40
+ export declare function findAvailablePort(start: number): Promise<number>;
41
+ /** Label namespace applied to every CLI-managed container. */
42
+ export declare const MANAGED_LABEL = "1claw.managed";
43
+ //# sourceMappingURL=container-config.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"container-config.d.ts","sourceRoot":"","sources":["../../../src/lib/container-config.ts"],"names":[],"mappings":"AAiBA,eAAO,MAAM,cAAc,QAAiC,CAAC;AAE7D,MAAM,WAAW,cAAc;IAC3B,aAAa,EAAE,MAAM,CAAC;IACtB,WAAW,EAAE,MAAM,GAAG,IAAI,CAAC;IAC3B,OAAO,EAAE,MAAM,GAAG,IAAI,CAAC;IACvB,OAAO,EAAE,MAAM,GAAG,IAAI,CAAC;IACvB,KAAK,EAAE,MAAM,CAAC;IACd,OAAO,EAAE,MAAM,EAAE,CAAC;IAClB,IAAI,EAAE,MAAM,CAAC;IACb,SAAS,EAAE,MAAM,CAAC;IAClB,gFAAgF;IAChF,cAAc,EAAE,MAAM,GAAG,IAAI,CAAC;IAC9B,oEAAoE;IACpE,WAAW,EAAE,MAAM,GAAG,IAAI,CAAC;IAC3B,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,iEAAiE;IACjE,IAAI,EAAE,OAAO,GAAG,OAAO,CAAC;IACxB,oDAAoD;IACpD,UAAU,CAAC,EAAE;QACT,QAAQ,EAAE,MAAM,CAAC;QACjB,UAAU,CAAC,EAAE,MAAM,CAAC;QACpB,MAAM,CAAC,EAAE,MAAM,CAAC;QAChB,UAAU,EAAE,MAAM,CAAC;KACtB,CAAC;CACL;AAYD,4DAA4D;AAC5D,wBAAgB,OAAO,IAAI,MAAM,CAEhC;AAED,iEAAiE;AACjE,wBAAgB,qBAAqB,IAAI,MAAM,CAE9C;AAED,gDAAgD;AAChD,wBAAgB,YAAY,CAAC,IAAI,EAAE,MAAM,GAAG,MAAM,CAEjD;AAED,wBAAgB,oBAAoB,CAAC,IAAI,EAAE,MAAM,GAAG,OAAO,CAG1D;AAED,wBAAgB,kBAAkB,CAAC,KAAK,EAAE,cAAc,GAAG,IAAI,CAS9D;AAED,wBAAgB,kBAAkB,CAAC,IAAI,EAAE,MAAM,GAAG,cAAc,GAAG,IAAI,CAQtE;AAED,wBAAgB,oBAAoB,CAAC,IAAI,EAAE,MAAM,GAAG,OAAO,CAK1D;AAED,wBAAgB,mBAAmB,IAAI,cAAc,EAAE,CAgBtD;AAED,6DAA6D;AAC7D,wBAAgB,eAAe,CAAC,IAAI,EAAE,MAAM,GAAG,OAAO,CAAC,OAAO,CAAC,CAS9D;AAED,0EAA0E;AAC1E,wBAAsB,iBAAiB,CAAC,KAAK,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,CAAC,CAOtE;AAED,8DAA8D;AAC9D,eAAO,MAAM,aAAa,kBAAkB,CAAC"}
@@ -0,0 +1,98 @@
1
+ import { existsSync, mkdirSync, readFileSync, writeFileSync, readdirSync, unlinkSync, chmodSync, } from "node:fs";
2
+ import { homedir } from "node:os";
3
+ import { join } from "node:path";
4
+ import { randomBytes } from "node:crypto";
5
+ import { createServer } from "node:net";
6
+ const CONFIG_DIR = process.env.ONECLAW_CONFIG_DIR || join(homedir(), ".config", "1claw");
7
+ export const CONTAINERS_DIR = join(CONFIG_DIR, "containers");
8
+ function ensureContainersDir() {
9
+ if (!existsSync(CONTAINERS_DIR)) {
10
+ mkdirSync(CONTAINERS_DIR, { recursive: true });
11
+ }
12
+ }
13
+ function statePath(name) {
14
+ return join(CONTAINERS_DIR, `${sanitizeName(name)}.json`);
15
+ }
16
+ /** A short, URL/identifier-safe random id (8 hex chars). */
17
+ export function shortId() {
18
+ return randomBytes(4).toString("hex");
19
+ }
20
+ /** Generate a default container name: docker-agent-<shortId>. */
21
+ export function generateContainerName() {
22
+ return `docker-agent-${shortId()}`;
23
+ }
24
+ /** Restrict names to docker-safe characters. */
25
+ export function sanitizeName(name) {
26
+ return name.replace(/[^a-zA-Z0-9_.-]/g, "-");
27
+ }
28
+ export function isValidContainerName(name) {
29
+ // Docker: [a-zA-Z0-9][a-zA-Z0-9_.-]+
30
+ return /^[a-zA-Z0-9][a-zA-Z0-9_.-]*$/.test(name);
31
+ }
32
+ export function saveContainerState(state) {
33
+ ensureContainersDir();
34
+ const path = statePath(state.containerName);
35
+ writeFileSync(path, JSON.stringify(state, null, 2) + "\n", "utf-8");
36
+ try {
37
+ chmodSync(path, 0o600);
38
+ }
39
+ catch {
40
+ // best-effort
41
+ }
42
+ }
43
+ export function loadContainerState(name) {
44
+ const path = statePath(name);
45
+ if (!existsSync(path))
46
+ return null;
47
+ try {
48
+ return JSON.parse(readFileSync(path, "utf-8"));
49
+ }
50
+ catch {
51
+ return null;
52
+ }
53
+ }
54
+ export function deleteContainerState(name) {
55
+ const path = statePath(name);
56
+ if (!existsSync(path))
57
+ return false;
58
+ unlinkSync(path);
59
+ return true;
60
+ }
61
+ export function listContainerStates() {
62
+ if (!existsSync(CONTAINERS_DIR))
63
+ return [];
64
+ const states = [];
65
+ for (const file of readdirSync(CONTAINERS_DIR)) {
66
+ if (!file.endsWith(".json"))
67
+ continue;
68
+ try {
69
+ states.push(JSON.parse(readFileSync(join(CONTAINERS_DIR, file), "utf-8")));
70
+ }
71
+ catch {
72
+ // skip corrupt entries
73
+ }
74
+ }
75
+ return states.sort((a, b) => b.createdAt.localeCompare(a.createdAt));
76
+ }
77
+ /** Check whether a TCP port is free to bind on localhost. */
78
+ export function isPortAvailable(port) {
79
+ return new Promise((resolve) => {
80
+ const server = createServer();
81
+ server.once("error", () => resolve(false));
82
+ server.once("listening", () => {
83
+ server.close(() => resolve(true));
84
+ });
85
+ server.listen(port, "127.0.0.1");
86
+ });
87
+ }
88
+ /** Find the first available port at or after `start` (bounded search). */
89
+ export async function findAvailablePort(start) {
90
+ for (let p = start; p < start + 100; p++) {
91
+ if (await isPortAvailable(p))
92
+ return p;
93
+ }
94
+ throw new Error(`No available port found in range ${start}-${start + 100}.`);
95
+ }
96
+ /** Label namespace applied to every CLI-managed container. */
97
+ export const MANAGED_LABEL = "1claw.managed";
98
+ //# sourceMappingURL=container-config.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"container-config.js","sourceRoot":"","sources":["../../../src/lib/container-config.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,10 @@
1
+ export declare function daemonSocketPath(): string;
2
+ /** Ping the daemon's /health over its Unix socket. */
3
+ export declare function daemonHealthy(socketPath?: string): Promise<boolean>;
4
+ /**
5
+ * Start the local daemon as a detached background process. The passphrase is
6
+ * passed via the child's environment (ONECLAW_VAULT_PASSPHRASE) so no
7
+ * interactive prompt is needed. Resolves once the socket answers /health.
8
+ */
9
+ export declare function startDaemonDetached(passphrase: string, socketPath?: string, timeoutMs?: number): Promise<boolean>;
10
+ //# sourceMappingURL=daemon-control.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"daemon-control.d.ts","sourceRoot":"","sources":["../../../src/lib/daemon-control.ts"],"names":[],"mappings":"AASA,wBAAgB,gBAAgB,IAAI,MAAM,CAEzC;AAED,sDAAsD;AACtD,wBAAgB,aAAa,CAAC,UAAU,SAAqB,GAAG,OAAO,CAAC,OAAO,CAAC,CAoB/E;AAED;;;;GAIG;AACH,wBAAsB,mBAAmB,CACrC,UAAU,EAAE,MAAM,EAClB,UAAU,SAAqB,EAC/B,SAAS,SAAQ,GAClB,OAAO,CAAC,OAAO,CAAC,CAwBlB"}
@@ -0,0 +1,58 @@
1
+ import { request as httpRequest } from "node:http";
2
+ import { spawn } from "node:child_process";
3
+ import { existsSync } from "node:fs";
4
+ import { homedir } from "node:os";
5
+ import { join } from "node:path";
6
+ const CONFIG_DIR = process.env.ONECLAW_CONFIG_DIR || join(homedir(), ".config", "1claw");
7
+ export function daemonSocketPath() {
8
+ return process.env.ONECLAW_DAEMON_SOCKET || join(CONFIG_DIR, "daemon.sock");
9
+ }
10
+ /** Ping the daemon's /health over its Unix socket. */
11
+ export function daemonHealthy(socketPath = daemonSocketPath()) {
12
+ return new Promise((resolve) => {
13
+ if (!existsSync(socketPath)) {
14
+ resolve(false);
15
+ return;
16
+ }
17
+ const req = httpRequest({ socketPath, path: "/health", method: "GET", timeout: 2000 }, (res) => {
18
+ res.resume();
19
+ resolve(res.statusCode === 200);
20
+ });
21
+ req.on("error", () => resolve(false));
22
+ req.on("timeout", () => {
23
+ req.destroy();
24
+ resolve(false);
25
+ });
26
+ req.end();
27
+ });
28
+ }
29
+ /**
30
+ * Start the local daemon as a detached background process. The passphrase is
31
+ * passed via the child's environment (ONECLAW_VAULT_PASSPHRASE) so no
32
+ * interactive prompt is needed. Resolves once the socket answers /health.
33
+ */
34
+ export async function startDaemonDetached(passphrase, socketPath = daemonSocketPath(), timeoutMs = 10000) {
35
+ // process.argv[1] is the CLI entry (dist/bin/1claw.js). Re-invoke it so the
36
+ // daemon runs the same code path in both dev and installed environments.
37
+ const cliEntry = process.argv[1];
38
+ const child = spawn(process.execPath, [cliEntry, "daemon", "start", "--socket", socketPath], {
39
+ detached: true,
40
+ stdio: "ignore",
41
+ env: {
42
+ ...process.env,
43
+ ONECLAW_VAULT_PASSPHRASE: passphrase,
44
+ },
45
+ });
46
+ child.unref();
47
+ const deadline = Date.now() + timeoutMs;
48
+ while (Date.now() < deadline) {
49
+ if (await daemonHealthy(socketPath))
50
+ return true;
51
+ await sleep(250);
52
+ }
53
+ return false;
54
+ }
55
+ function sleep(ms) {
56
+ return new Promise((r) => setTimeout(r, ms));
57
+ }
58
+ //# sourceMappingURL=daemon-control.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"daemon-control.js","sourceRoot":"","sources":["../../../src/lib/daemon-control.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,IAAI,WAAW,EAAE,MAAM,WAAW,CAAC;AACnD,OAAO,EAAE,KAAK,EAAE,MAAM,oBAAoB,CAAC;AAC3C,OAAO,EAAE,UAAU,EAAE,MAAM,SAAS,CAAC;AACrC,OAAO,EAAE,OAAO,EAAE,MAAM,SAAS,CAAC;AAClC,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AAEjC,MAAM,UAAU,GACZ,OAAO,CAAC,GAAG,CAAC,kBAAkB,IAAI,IAAI,CAAC,OAAO,EAAE,EAAE,SAAS,EAAE,OAAO,CAAC,CAAC;AAE1E,MAAM,UAAU,gBAAgB;IAC5B,OAAO,OAAO,CAAC,GAAG,CAAC,qBAAqB,IAAI,IAAI,CAAC,UAAU,EAAE,aAAa,CAAC,CAAC;AAChF,CAAC;AAED,sDAAsD;AACtD,MAAM,UAAU,aAAa,CAAC,UAAU,GAAG,gBAAgB,EAAE;IACzD,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,EAAE;QAC3B,IAAI,CAAC,UAAU,CAAC,UAAU,CAAC,EAAE,CAAC;YAC1B,OAAO,CAAC,KAAK,CAAC,CAAC;YACf,OAAO;QACX,CAAC;QACD,MAAM,GAAG,GAAG,WAAW,CACnB,EAAE,UAAU,EAAE,IAAI,EAAE,SAAS,EAAE,MAAM,EAAE,KAAK,EAAE,OAAO,EAAE,IAAI,EAAE,EAC7D,CAAC,GAAG,EAAE,EAAE;YACJ,GAAG,CAAC,MAAM,EAAE,CAAC;YACb,OAAO,CAAC,GAAG,CAAC,UAAU,KAAK,GAAG,CAAC,CAAC;QACpC,CAAC,CACJ,CAAC;QACF,GAAG,CAAC,EAAE,CAAC,OAAO,EAAE,GAAG,EAAE,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC;QACtC,GAAG,CAAC,EAAE,CAAC,SAAS,EAAE,GAAG,EAAE;YACnB,GAAG,CAAC,OAAO,EAAE,CAAC;YACd,OAAO,CAAC,KAAK,CAAC,CAAC;QACnB,CAAC,CAAC,CAAC;QACH,GAAG,CAAC,GAAG,EAAE,CAAC;IACd,CAAC,CAAC,CAAC;AACP,CAAC;AAED;;;;GAIG;AACH,MAAM,CAAC,KAAK,UAAU,mBAAmB,CACrC,UAAkB,EAClB,UAAU,GAAG,gBAAgB,EAAE,EAC/B,SAAS,GAAG,KAAK;IAEjB,4EAA4E;IAC5E,yEAAyE;IACzE,MAAM,QAAQ,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IACjC,MAAM,KAAK,GAAG,KAAK,CACf,OAAO,CAAC,QAAQ,EAChB,CAAC,QAAQ,EAAE,QAAQ,EAAE,OAAO,EAAE,UAAU,EAAE,UAAU,CAAC,EACrD;QACI,QAAQ,EAAE,IAAI;QACd,KAAK,EAAE,QAAQ;QACf,GAAG,EAAE;YACD,GAAG,OAAO,CAAC,GAAG;YACd,wBAAwB,EAAE,UAAU;SACvC;KACJ,CACJ,CAAC;IACF,KAAK,CAAC,KAAK,EAAE,CAAC;IAEd,MAAM,QAAQ,GAAG,IAAI,CAAC,GAAG,EAAE,GAAG,SAAS,CAAC;IACxC,OAAO,IAAI,CAAC,GAAG,EAAE,GAAG,QAAQ,EAAE,CAAC;QAC3B,IAAI,MAAM,aAAa,CAAC,UAAU,CAAC;YAAE,OAAO,IAAI,CAAC;QACjD,MAAM,KAAK,CAAC,GAAG,CAAC,CAAC;IACrB,CAAC;IACD,OAAO,KAAK,CAAC;AACjB,CAAC;AAED,SAAS,KAAK,CAAC,EAAU;IACrB,OAAO,IAAI,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,UAAU,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC;AACjD,CAAC"}