@clappstore/connect 0.1.0 → 0.2.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.
- package/dist/credentials.d.ts +5 -0
- package/dist/credentials.d.ts.map +1 -0
- package/dist/credentials.js +32 -0
- package/dist/credentials.js.map +1 -0
- package/dist/defaults.d.ts +2 -0
- package/dist/defaults.d.ts.map +1 -0
- package/dist/defaults.js +64 -0
- package/dist/defaults.js.map +1 -0
- package/dist/index.js +41 -6
- package/dist/index.js.map +1 -1
- package/package.json +1 -1
|
@@ -0,0 +1,5 @@
|
|
|
1
|
+
declare const CREDENTIALS_PATH: string;
|
|
2
|
+
export declare function loadToken(relay: string, agentId: string): string | null;
|
|
3
|
+
export declare function saveToken(relay: string, agentId: string, token: string): void;
|
|
4
|
+
export { CREDENTIALS_PATH };
|
|
5
|
+
//# sourceMappingURL=credentials.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"credentials.d.ts","sourceRoot":"","sources":["../src/credentials.ts"],"names":[],"mappings":"AAIA,QAAA,MAAM,gBAAgB,QAIrB,CAAC;AAsBF,wBAAgB,SAAS,CAAC,KAAK,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,GAAG,MAAM,GAAG,IAAI,CAIvE;AAED,wBAAgB,SAAS,CAAC,KAAK,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,GAAG,IAAI,CAQ7E;AAED,OAAO,EAAE,gBAAgB,EAAE,CAAC"}
|
|
@@ -0,0 +1,32 @@
|
|
|
1
|
+
import { readFileSync, writeFileSync, mkdirSync } from "node:fs";
|
|
2
|
+
import { resolve, dirname } from "node:path";
|
|
3
|
+
import { homedir } from "node:os";
|
|
4
|
+
const CREDENTIALS_PATH = resolve(homedir(), ".openclaw", "clapps-credentials.json");
|
|
5
|
+
function credentialKey(relay, agentId) {
|
|
6
|
+
return `${relay}:${agentId}`;
|
|
7
|
+
}
|
|
8
|
+
function readCredentials() {
|
|
9
|
+
try {
|
|
10
|
+
const raw = readFileSync(CREDENTIALS_PATH, "utf-8");
|
|
11
|
+
return JSON.parse(raw);
|
|
12
|
+
}
|
|
13
|
+
catch {
|
|
14
|
+
return {};
|
|
15
|
+
}
|
|
16
|
+
}
|
|
17
|
+
export function loadToken(relay, agentId) {
|
|
18
|
+
const creds = readCredentials();
|
|
19
|
+
const entry = creds[credentialKey(relay, agentId)];
|
|
20
|
+
return entry?.token ?? null;
|
|
21
|
+
}
|
|
22
|
+
export function saveToken(relay, agentId, token) {
|
|
23
|
+
const creds = readCredentials();
|
|
24
|
+
creds[credentialKey(relay, agentId)] = {
|
|
25
|
+
token,
|
|
26
|
+
createdAt: new Date().toISOString(),
|
|
27
|
+
};
|
|
28
|
+
mkdirSync(dirname(CREDENTIALS_PATH), { recursive: true });
|
|
29
|
+
writeFileSync(CREDENTIALS_PATH, JSON.stringify(creds, null, 2) + "\n");
|
|
30
|
+
}
|
|
31
|
+
export { CREDENTIALS_PATH };
|
|
32
|
+
//# sourceMappingURL=credentials.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"credentials.js","sourceRoot":"","sources":["../src/credentials.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,aAAa,EAAE,SAAS,EAAE,MAAM,SAAS,CAAC;AACjE,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAC7C,OAAO,EAAE,OAAO,EAAE,MAAM,SAAS,CAAC;AAElC,MAAM,gBAAgB,GAAG,OAAO,CAC9B,OAAO,EAAE,EACT,WAAW,EACX,yBAAyB,CAC1B,CAAC;AASF,SAAS,aAAa,CAAC,KAAa,EAAE,OAAe;IACnD,OAAO,GAAG,KAAK,IAAI,OAAO,EAAE,CAAC;AAC/B,CAAC;AAED,SAAS,eAAe;IACtB,IAAI,CAAC;QACH,MAAM,GAAG,GAAG,YAAY,CAAC,gBAAgB,EAAE,OAAO,CAAC,CAAC;QACpD,OAAO,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;IACzB,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,EAAE,CAAC;IACZ,CAAC;AACH,CAAC;AAED,MAAM,UAAU,SAAS,CAAC,KAAa,EAAE,OAAe;IACtD,MAAM,KAAK,GAAG,eAAe,EAAE,CAAC;IAChC,MAAM,KAAK,GAAG,KAAK,CAAC,aAAa,CAAC,KAAK,EAAE,OAAO,CAAC,CAAC,CAAC;IACnD,OAAO,KAAK,EAAE,KAAK,IAAI,IAAI,CAAC;AAC9B,CAAC;AAED,MAAM,UAAU,SAAS,CAAC,KAAa,EAAE,OAAe,EAAE,KAAa;IACrE,MAAM,KAAK,GAAG,eAAe,EAAE,CAAC;IAChC,KAAK,CAAC,aAAa,CAAC,KAAK,EAAE,OAAO,CAAC,CAAC,GAAG;QACrC,KAAK;QACL,SAAS,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;KACpC,CAAC;IACF,SAAS,CAAC,OAAO,CAAC,gBAAgB,CAAC,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;IAC1D,aAAa,CAAC,gBAAgB,EAAE,IAAI,CAAC,SAAS,CAAC,KAAK,EAAE,IAAI,EAAE,CAAC,CAAC,GAAG,IAAI,CAAC,CAAC;AACzE,CAAC;AAED,OAAO,EAAE,gBAAgB,EAAE,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"defaults.d.ts","sourceRoot":"","sources":["../src/defaults.ts"],"names":[],"mappings":"AA6CA,wBAAgB,YAAY,CAAC,QAAQ,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,GAAG,IAAI,CA2BrE"}
|
package/dist/defaults.js
ADDED
|
@@ -0,0 +1,64 @@
|
|
|
1
|
+
import { existsSync, readFileSync, writeFileSync } from "node:fs";
|
|
2
|
+
import { resolve } from "node:path";
|
|
3
|
+
const DEFAULT_CHAT_VIEW = `---
|
|
4
|
+
name: Chat
|
|
5
|
+
domain: default
|
|
6
|
+
version: 0.1.0
|
|
7
|
+
---
|
|
8
|
+
|
|
9
|
+
## State Bindings
|
|
10
|
+
- \`messages\` -> array
|
|
11
|
+
- \`loading\` -> boolean
|
|
12
|
+
|
|
13
|
+
## Layout
|
|
14
|
+
\`\`\`clapp-layout
|
|
15
|
+
Column(gap=4):
|
|
16
|
+
List(data=messages):
|
|
17
|
+
Conditional(when=loading):
|
|
18
|
+
Skeleton:
|
|
19
|
+
IntentForm(intent=chat.send, submitLabel=Send):
|
|
20
|
+
TextInput(name=text, placeholder=Type a message...):
|
|
21
|
+
\`\`\`
|
|
22
|
+
|
|
23
|
+
## Intents
|
|
24
|
+
| Name | Payload | Description |
|
|
25
|
+
|------|---------|-------------|
|
|
26
|
+
| chat.send | \`{ text: string }\` | Send a chat message |
|
|
27
|
+
`;
|
|
28
|
+
const DEFAULT_CHAT_APP = {
|
|
29
|
+
id: "chat",
|
|
30
|
+
name: "Chat",
|
|
31
|
+
emoji: "\u{1F4AC}",
|
|
32
|
+
description: "Chat with the agent",
|
|
33
|
+
pinned: true,
|
|
34
|
+
};
|
|
35
|
+
export function seedDefaults(viewsDir, stateDir) {
|
|
36
|
+
// Seed chat.view.md if missing
|
|
37
|
+
const viewPath = resolve(viewsDir, "chat.view.md");
|
|
38
|
+
if (!existsSync(viewPath)) {
|
|
39
|
+
writeFileSync(viewPath, DEFAULT_CHAT_VIEW, "utf-8");
|
|
40
|
+
console.log(`📝 Created default chat view: ${viewPath}`);
|
|
41
|
+
}
|
|
42
|
+
// Seed or merge _apps.json
|
|
43
|
+
const appsPath = resolve(stateDir, "_apps.json");
|
|
44
|
+
if (!existsSync(appsPath)) {
|
|
45
|
+
writeFileSync(appsPath, JSON.stringify([DEFAULT_CHAT_APP], null, 2), "utf-8");
|
|
46
|
+
console.log(`📝 Created default apps registry: ${appsPath}`);
|
|
47
|
+
}
|
|
48
|
+
else {
|
|
49
|
+
try {
|
|
50
|
+
const existing = JSON.parse(readFileSync(appsPath, "utf-8"));
|
|
51
|
+
const hasChatEntry = existing.some((app) => app.id === "chat");
|
|
52
|
+
if (!hasChatEntry) {
|
|
53
|
+
existing.push(DEFAULT_CHAT_APP);
|
|
54
|
+
writeFileSync(appsPath, JSON.stringify(existing, null, 2), "utf-8");
|
|
55
|
+
console.log(`📝 Added chat app to existing apps registry`);
|
|
56
|
+
}
|
|
57
|
+
}
|
|
58
|
+
catch {
|
|
59
|
+
// If _apps.json is malformed, don't overwrite — let the user fix it
|
|
60
|
+
console.warn(`⚠️ Could not parse ${appsPath}, skipping chat app seeding`);
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
//# sourceMappingURL=defaults.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"defaults.js","sourceRoot":"","sources":["../src/defaults.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,YAAY,EAAE,aAAa,EAAE,MAAM,SAAS,CAAC;AAClE,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAUpC,MAAM,iBAAiB,GAAG;;;;;;;;;;;;;;;;;;;;;;;;CAwBzB,CAAC;AAEF,MAAM,gBAAgB,GAAa;IACjC,EAAE,EAAE,MAAM;IACV,IAAI,EAAE,MAAM;IACZ,KAAK,EAAE,WAAW;IAClB,WAAW,EAAE,qBAAqB;IAClC,MAAM,EAAE,IAAI;CACb,CAAC;AAEF,MAAM,UAAU,YAAY,CAAC,QAAgB,EAAE,QAAgB;IAC7D,+BAA+B;IAC/B,MAAM,QAAQ,GAAG,OAAO,CAAC,QAAQ,EAAE,cAAc,CAAC,CAAC;IACnD,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC,EAAE,CAAC;QAC1B,aAAa,CAAC,QAAQ,EAAE,iBAAiB,EAAE,OAAO,CAAC,CAAC;QACpD,OAAO,CAAC,GAAG,CAAC,iCAAiC,QAAQ,EAAE,CAAC,CAAC;IAC3D,CAAC;IAED,2BAA2B;IAC3B,MAAM,QAAQ,GAAG,OAAO,CAAC,QAAQ,EAAE,YAAY,CAAC,CAAC;IACjD,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC,EAAE,CAAC;QAC1B,aAAa,CAAC,QAAQ,EAAE,IAAI,CAAC,SAAS,CAAC,CAAC,gBAAgB,CAAC,EAAE,IAAI,EAAE,CAAC,CAAC,EAAE,OAAO,CAAC,CAAC;QAC9E,OAAO,CAAC,GAAG,CAAC,qCAAqC,QAAQ,EAAE,CAAC,CAAC;IAC/D,CAAC;SAAM,CAAC;QACN,IAAI,CAAC;YACH,MAAM,QAAQ,GAAe,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC,CAAC;YACzE,MAAM,YAAY,GAAG,QAAQ,CAAC,IAAI,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,GAAG,CAAC,EAAE,KAAK,MAAM,CAAC,CAAC;YAC/D,IAAI,CAAC,YAAY,EAAE,CAAC;gBAClB,QAAQ,CAAC,IAAI,CAAC,gBAAgB,CAAC,CAAC;gBAChC,aAAa,CAAC,QAAQ,EAAE,IAAI,CAAC,SAAS,CAAC,QAAQ,EAAE,IAAI,EAAE,CAAC,CAAC,EAAE,OAAO,CAAC,CAAC;gBACpE,OAAO,CAAC,GAAG,CAAC,6CAA6C,CAAC,CAAC;YAC7D,CAAC;QACH,CAAC;QAAC,MAAM,CAAC;YACP,oEAAoE;YACpE,OAAO,CAAC,IAAI,CAAC,uBAAuB,QAAQ,6BAA6B,CAAC,CAAC;QAC7E,CAAC;IACH,CAAC;AACH,CAAC"}
|
package/dist/index.js
CHANGED
|
@@ -5,6 +5,8 @@ import { mkdirSync } from "node:fs";
|
|
|
5
5
|
import { AgentClient } from "./agent-client.js";
|
|
6
6
|
import { IntentPoller } from "./intent-poller.js";
|
|
7
7
|
import { StateWatcher } from "./state-watcher.js";
|
|
8
|
+
import { loadToken, saveToken, CREDENTIALS_PATH } from "./credentials.js";
|
|
9
|
+
import { seedDefaults } from "./defaults.js";
|
|
8
10
|
function parseArgs(args) {
|
|
9
11
|
const opts = {};
|
|
10
12
|
for (let i = 0; i < args.length; i++) {
|
|
@@ -15,10 +17,41 @@ function parseArgs(args) {
|
|
|
15
17
|
}
|
|
16
18
|
return opts;
|
|
17
19
|
}
|
|
20
|
+
async function selfRegister(relayUrl, agentId) {
|
|
21
|
+
const res = await fetch(`${relayUrl}/api/agent/register`, {
|
|
22
|
+
method: "POST",
|
|
23
|
+
headers: { "Content-Type": "application/json" },
|
|
24
|
+
body: JSON.stringify({ agentId }),
|
|
25
|
+
});
|
|
26
|
+
if (res.status === 409) {
|
|
27
|
+
console.error(`❌ Agent ID "${agentId}" is already registered. If this is your agent, use --token YOUR_TOKEN`);
|
|
28
|
+
process.exit(1);
|
|
29
|
+
}
|
|
30
|
+
if (!res.ok) {
|
|
31
|
+
const body = await res.text();
|
|
32
|
+
throw new Error(`Registration failed (${res.status}): ${body}`);
|
|
33
|
+
}
|
|
34
|
+
const { token } = (await res.json());
|
|
35
|
+
return token;
|
|
36
|
+
}
|
|
37
|
+
async function resolveToken(relayUrl, agentId, explicitToken) {
|
|
38
|
+
// 1. Explicit --token flag wins
|
|
39
|
+
if (explicitToken)
|
|
40
|
+
return explicitToken;
|
|
41
|
+
// 2. Try saved credentials
|
|
42
|
+
const saved = loadToken(relayUrl, agentId);
|
|
43
|
+
if (saved)
|
|
44
|
+
return saved;
|
|
45
|
+
// 3. Self-register
|
|
46
|
+
const token = await selfRegister(relayUrl, agentId);
|
|
47
|
+
saveToken(relayUrl, agentId, token);
|
|
48
|
+
console.log(`✅ Registered agent "${agentId}" (token saved to ${CREDENTIALS_PATH})`);
|
|
49
|
+
return token;
|
|
50
|
+
}
|
|
18
51
|
async function main() {
|
|
19
52
|
const opts = parseArgs(process.argv.slice(2));
|
|
20
53
|
const relayUrl = opts.relay ?? "https://clapps.clawlab.app";
|
|
21
|
-
const
|
|
54
|
+
const explicitToken = opts.token;
|
|
22
55
|
const agentId = opts["agent-id"];
|
|
23
56
|
const agentUrl = opts.agent ?? "http://localhost:18789";
|
|
24
57
|
const agentToken = opts["agent-token"];
|
|
@@ -26,13 +59,17 @@ async function main() {
|
|
|
26
59
|
resolve(homedir(), ".openclaw", "workspace", "ui", "state");
|
|
27
60
|
const viewsDir = opts["views-dir"] ??
|
|
28
61
|
resolve(homedir(), ".openclaw", "workspace", "ui", "views");
|
|
29
|
-
if (!
|
|
30
|
-
console.error("Usage: clapps-connect --
|
|
62
|
+
if (!agentId) {
|
|
63
|
+
console.error("Usage: clapps-connect --agent-id AGENT_ID [--token TOKEN] [--relay URL] [--agent URL] [--agent-token TOKEN] [--views-dir PATH]");
|
|
31
64
|
process.exit(1);
|
|
32
65
|
}
|
|
66
|
+
console.log(`🔗 Connecting to relay: ${relayUrl}`);
|
|
67
|
+
const token = await resolveToken(relayUrl, agentId, explicitToken);
|
|
33
68
|
// Ensure directories exist
|
|
34
69
|
mkdirSync(stateDir, { recursive: true });
|
|
35
70
|
mkdirSync(viewsDir, { recursive: true });
|
|
71
|
+
// Seed default clapp files (chat UI, apps registry)
|
|
72
|
+
seedDefaults(viewsDir, stateDir);
|
|
36
73
|
const agentClient = new AgentClient({ agentUrl, agentToken });
|
|
37
74
|
const intentPoller = new IntentPoller({
|
|
38
75
|
relayUrl,
|
|
@@ -49,9 +86,7 @@ async function main() {
|
|
|
49
86
|
agentId,
|
|
50
87
|
onError: (err) => console.error("[state-watcher]", err.message),
|
|
51
88
|
});
|
|
52
|
-
console.log(
|
|
53
|
-
console.log(`🤖 Agent at: ${agentUrl}`);
|
|
54
|
-
console.log(`👤 Agent ID: ${agentId}`);
|
|
89
|
+
console.log(`🤖 Agent ID: ${agentId}`);
|
|
55
90
|
console.log(`📁 Watching state: ${stateDir}`);
|
|
56
91
|
console.log(`📄 Watching views: ${viewsDir}`);
|
|
57
92
|
// Create/reuse a stable link token for browser access
|
package/dist/index.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":";AAEA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AACpC,OAAO,EAAE,OAAO,EAAE,MAAM,SAAS,CAAC;AAClC,OAAO,EAAE,SAAS,EAAE,MAAM,SAAS,CAAC;AACpC,OAAO,EAAE,WAAW,EAAE,MAAM,mBAAmB,CAAC;AAChD,OAAO,EAAE,YAAY,EAAE,MAAM,oBAAoB,CAAC;AAClD,OAAO,EAAE,YAAY,EAAE,MAAM,oBAAoB,CAAC;
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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"}
|