@cecwxf/wtt 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 (115) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +147 -0
  3. package/bin/openclaw-wtt-bootstrap.mjs +181 -0
  4. package/dist/channel.d.ts +275 -0
  5. package/dist/channel.d.ts.map +1 -0
  6. package/dist/channel.js +2088 -0
  7. package/dist/channel.js.map +1 -0
  8. package/dist/commands/account.d.ts +16 -0
  9. package/dist/commands/account.d.ts.map +1 -0
  10. package/dist/commands/account.js +37 -0
  11. package/dist/commands/account.js.map +1 -0
  12. package/dist/commands/bind.d.ts +3 -0
  13. package/dist/commands/bind.d.ts.map +1 -0
  14. package/dist/commands/bind.js +102 -0
  15. package/dist/commands/bind.js.map +1 -0
  16. package/dist/commands/config.d.ts +3 -0
  17. package/dist/commands/config.d.ts.map +1 -0
  18. package/dist/commands/config.js +38 -0
  19. package/dist/commands/config.js.map +1 -0
  20. package/dist/commands/delegate.d.ts +7 -0
  21. package/dist/commands/delegate.d.ts.map +1 -0
  22. package/dist/commands/delegate.js +99 -0
  23. package/dist/commands/delegate.js.map +1 -0
  24. package/dist/commands/formatter.d.ts +8 -0
  25. package/dist/commands/formatter.d.ts.map +1 -0
  26. package/dist/commands/formatter.js +198 -0
  27. package/dist/commands/formatter.js.map +1 -0
  28. package/dist/commands/handlers.d.ts +3 -0
  29. package/dist/commands/handlers.d.ts.map +1 -0
  30. package/dist/commands/handlers.js +79 -0
  31. package/dist/commands/handlers.js.map +1 -0
  32. package/dist/commands/http.d.ts +26 -0
  33. package/dist/commands/http.d.ts.map +1 -0
  34. package/dist/commands/http.js +190 -0
  35. package/dist/commands/http.js.map +1 -0
  36. package/dist/commands/index.d.ts +3 -0
  37. package/dist/commands/index.d.ts.map +1 -0
  38. package/dist/commands/index.js +2 -0
  39. package/dist/commands/index.js.map +1 -0
  40. package/dist/commands/parser.d.ts +5 -0
  41. package/dist/commands/parser.d.ts.map +1 -0
  42. package/dist/commands/parser.js +325 -0
  43. package/dist/commands/parser.js.map +1 -0
  44. package/dist/commands/pipeline.d.ts +7 -0
  45. package/dist/commands/pipeline.d.ts.map +1 -0
  46. package/dist/commands/pipeline.js +99 -0
  47. package/dist/commands/pipeline.js.map +1 -0
  48. package/dist/commands/router.d.ts +18 -0
  49. package/dist/commands/router.d.ts.map +1 -0
  50. package/dist/commands/router.js +74 -0
  51. package/dist/commands/router.js.map +1 -0
  52. package/dist/commands/setup.d.ts +7 -0
  53. package/dist/commands/setup.d.ts.map +1 -0
  54. package/dist/commands/setup.js +89 -0
  55. package/dist/commands/setup.js.map +1 -0
  56. package/dist/commands/task.d.ts +26 -0
  57. package/dist/commands/task.d.ts.map +1 -0
  58. package/dist/commands/task.js +438 -0
  59. package/dist/commands/task.js.map +1 -0
  60. package/dist/commands/types.d.ts +173 -0
  61. package/dist/commands/types.d.ts.map +1 -0
  62. package/dist/commands/types.js +2 -0
  63. package/dist/commands/types.js.map +1 -0
  64. package/dist/e2e-crypto.d.ts +36 -0
  65. package/dist/e2e-crypto.d.ts.map +1 -0
  66. package/dist/e2e-crypto.js +166 -0
  67. package/dist/e2e-crypto.js.map +1 -0
  68. package/dist/index.d.ts +20 -0
  69. package/dist/index.d.ts.map +1 -0
  70. package/dist/index.js +21 -0
  71. package/dist/index.js.map +1 -0
  72. package/dist/plugin-config.d.ts +32 -0
  73. package/dist/plugin-config.d.ts.map +1 -0
  74. package/dist/plugin-config.js +268 -0
  75. package/dist/plugin-config.js.map +1 -0
  76. package/dist/runtime/index.d.ts +13 -0
  77. package/dist/runtime/index.d.ts.map +1 -0
  78. package/dist/runtime/index.js +7 -0
  79. package/dist/runtime/index.js.map +1 -0
  80. package/dist/runtime/progress-ticker.d.ts +65 -0
  81. package/dist/runtime/progress-ticker.d.ts.map +1 -0
  82. package/dist/runtime/progress-ticker.js +116 -0
  83. package/dist/runtime/progress-ticker.js.map +1 -0
  84. package/dist/runtime/session-binding.d.ts +16 -0
  85. package/dist/runtime/session-binding.d.ts.map +1 -0
  86. package/dist/runtime/session-binding.js +20 -0
  87. package/dist/runtime/session-binding.js.map +1 -0
  88. package/dist/runtime/status-transition.d.ts +19 -0
  89. package/dist/runtime/status-transition.d.ts.map +1 -0
  90. package/dist/runtime/status-transition.js +95 -0
  91. package/dist/runtime/status-transition.js.map +1 -0
  92. package/dist/runtime/task-executor-persistence.d.ts +63 -0
  93. package/dist/runtime/task-executor-persistence.d.ts.map +1 -0
  94. package/dist/runtime/task-executor-persistence.js +201 -0
  95. package/dist/runtime/task-executor-persistence.js.map +1 -0
  96. package/dist/runtime/task-executor.d.ts +169 -0
  97. package/dist/runtime/task-executor.d.ts.map +1 -0
  98. package/dist/runtime/task-executor.js +1230 -0
  99. package/dist/runtime/task-executor.js.map +1 -0
  100. package/dist/runtime/task-status-handler.d.ts +28 -0
  101. package/dist/runtime/task-status-handler.d.ts.map +1 -0
  102. package/dist/runtime/task-status-handler.js +102 -0
  103. package/dist/runtime/task-status-handler.js.map +1 -0
  104. package/dist/types.d.ts +159 -0
  105. package/dist/types.d.ts.map +1 -0
  106. package/dist/types.js +6 -0
  107. package/dist/types.js.map +1 -0
  108. package/dist/ws-client.d.ts +90 -0
  109. package/dist/ws-client.d.ts.map +1 -0
  110. package/dist/ws-client.js +385 -0
  111. package/dist/ws-client.js.map +1 -0
  112. package/index.ts +19 -0
  113. package/openclaw.plugin.json +49 -0
  114. package/package.json +62 -0
  115. package/scripts/install-bootstrap-cli.sh +54 -0
@@ -0,0 +1,385 @@
1
+ /**
2
+ * WTT Cloud WebSocket Client — persistent connection to the WTT cloud service.
3
+ *
4
+ * Features:
5
+ * - Auto-reconnect with exponential backoff
6
+ * - In-band JWT auth
7
+ * - Heartbeat ping/pong (text frames, not WS ping)
8
+ * - Request/response correlation via request_id
9
+ * - E2E encryption (optional, per-account)
10
+ * - Server push handling (new_message, task_status)
11
+ */
12
+ import { deriveKey, encryptText, decryptText, toBase64, fromBase64, } from "./e2e-crypto.js";
13
+ export class WTTCloudClient {
14
+ ws = null;
15
+ account;
16
+ e2eKey = null;
17
+ pending = new Map();
18
+ heartbeatTimer = null;
19
+ reconnectTimer = null;
20
+ reconnectAttempts = 0;
21
+ maxReconnectAttempts;
22
+ heartbeatInterval;
23
+ closed = false;
24
+ reqCounter = 0;
25
+ wsUrlCursor = 0;
26
+ // Handlers
27
+ onMessage;
28
+ onTaskStatus;
29
+ onConnect;
30
+ onDisconnect;
31
+ onError;
32
+ log;
33
+ constructor(opts) {
34
+ this.account = opts.account;
35
+ this.onMessage = opts.onMessage;
36
+ this.onTaskStatus = opts.onTaskStatus;
37
+ this.onConnect = opts.onConnect;
38
+ this.onDisconnect = opts.onDisconnect;
39
+ this.onError = opts.onError;
40
+ this.heartbeatInterval = opts.heartbeatInterval ?? 30_000;
41
+ this.maxReconnectAttempts = opts.maxReconnectAttempts ?? Infinity;
42
+ this.log = opts.log ?? ((level, msg, data) => {
43
+ const fn = level === "error" ? console.error : level === "warn" ? console.warn : console.log;
44
+ fn(`[WTT-WS] ${msg}`, data ?? "");
45
+ });
46
+ }
47
+ // ---------------------------------------------------------------------------
48
+ // Lifecycle
49
+ // ---------------------------------------------------------------------------
50
+ async connect() {
51
+ this.closed = false;
52
+ this.reconnectAttempts = 0;
53
+ // Derive E2E key if password is configured
54
+ if (this.account.config.e2ePassword) {
55
+ this.log("info", "Deriving E2E key...");
56
+ this.e2eKey = await deriveKey(this.account.config.e2ePassword, this.account.agentId);
57
+ this.log("info", "E2E key derived");
58
+ }
59
+ this.doConnect();
60
+ }
61
+ disconnect() {
62
+ this.closed = true;
63
+ this.clearTimers();
64
+ if (this.ws) {
65
+ this.ws.close(1000, "client disconnect");
66
+ this.ws = null;
67
+ }
68
+ // Reject all pending requests
69
+ for (const [id, p] of this.pending) {
70
+ clearTimeout(p.timer);
71
+ p.reject(new Error("Client disconnected"));
72
+ }
73
+ this.pending.clear();
74
+ }
75
+ get connected() {
76
+ if (!this.ws)
77
+ return false;
78
+ const state = this.ws.readyState ?? this.ws.readyState;
79
+ return state === 1; // OPEN
80
+ }
81
+ getAccount() {
82
+ return { ...this.account, config: { ...this.account.config } };
83
+ }
84
+ hasE2EKey() {
85
+ return Boolean(this.e2eKey && this.e2eKey.length > 0);
86
+ }
87
+ // ---------------------------------------------------------------------------
88
+ // Actions (public API)
89
+ // ---------------------------------------------------------------------------
90
+ async sendAction(action, payload, timeoutMs = 15_000) {
91
+ if (!this.connected)
92
+ throw new Error("Not connected to WTT cloud");
93
+ const requestId = `${action}-${++this.reqCounter}-${Date.now().toString(36)}`;
94
+ const msg = { action, request_id: requestId, ...payload };
95
+ return new Promise((resolve, reject) => {
96
+ const timer = setTimeout(() => {
97
+ this.pending.delete(requestId);
98
+ reject(new Error(`Action ${action} timed out after ${timeoutMs}ms`));
99
+ }, timeoutMs);
100
+ this.pending.set(requestId, { resolve, reject, timer });
101
+ this.send(JSON.stringify(msg));
102
+ });
103
+ }
104
+ // Convenience methods
105
+ async list(limit) {
106
+ return this.sendAction("list", { limit });
107
+ }
108
+ async find(query) {
109
+ return this.sendAction("find", { query });
110
+ }
111
+ async join(topicId) {
112
+ return this.sendAction("join", { topic_id: topicId });
113
+ }
114
+ async leave(topicId) {
115
+ return this.sendAction("leave", { topic_id: topicId });
116
+ }
117
+ async subscribed() {
118
+ return this.sendAction("subscribed", {});
119
+ }
120
+ async publish(topicId, content, opts) {
121
+ let finalContent = content;
122
+ let encrypted = false;
123
+ if (opts?.encrypt && this.e2eKey) {
124
+ const contextId = crypto.randomUUID();
125
+ const ciphertext = await encryptText(this.e2eKey, content, contextId);
126
+ finalContent = JSON.stringify({ c: toBase64(ciphertext), ctx: contextId });
127
+ encrypted = true;
128
+ }
129
+ return this.sendAction("publish", {
130
+ topic_id: topicId,
131
+ content: finalContent,
132
+ content_type: opts?.contentType,
133
+ semantic_type: opts?.semanticType,
134
+ encrypted,
135
+ });
136
+ }
137
+ async poll(limit) {
138
+ return this.sendAction("poll", { limit });
139
+ }
140
+ async p2p(targetAgentId, content, encrypt = false) {
141
+ let finalContent = content;
142
+ let encrypted = false;
143
+ if (encrypt && this.e2eKey) {
144
+ const contextId = crypto.randomUUID();
145
+ const ciphertext = await encryptText(this.e2eKey, content, contextId);
146
+ finalContent = JSON.stringify({ c: toBase64(ciphertext), ctx: contextId });
147
+ encrypted = true;
148
+ }
149
+ return this.sendAction("p2p", {
150
+ target_agent_id: targetAgentId,
151
+ content: finalContent,
152
+ encrypted,
153
+ });
154
+ }
155
+ async history(topicId, limit, beforeId) {
156
+ return this.sendAction("history", {
157
+ topic_id: topicId,
158
+ limit,
159
+ before_id: beforeId,
160
+ });
161
+ }
162
+ async detail(topicId) {
163
+ return this.sendAction("detail", { topic_id: topicId });
164
+ }
165
+ async typing(topicId, state, ttlMs = 6000) {
166
+ return this.sendAction("typing", {
167
+ topic_id: topicId,
168
+ state,
169
+ ttl_ms: ttlMs,
170
+ });
171
+ }
172
+ // ---------------------------------------------------------------------------
173
+ // E2E decryption helper (for consumers)
174
+ // ---------------------------------------------------------------------------
175
+ async decryptMessage(content) {
176
+ if (!this.e2eKey)
177
+ return content;
178
+ try {
179
+ const { c, ctx } = JSON.parse(content);
180
+ if (!c || !ctx)
181
+ return content;
182
+ const ciphertext = fromBase64(c);
183
+ return await decryptText(this.e2eKey, ciphertext, ctx);
184
+ }
185
+ catch {
186
+ return content; // not encrypted or failed to decrypt
187
+ }
188
+ }
189
+ // ---------------------------------------------------------------------------
190
+ // Connection internals
191
+ // ---------------------------------------------------------------------------
192
+ resolveWsUrls() {
193
+ const raw = this.account.cloudUrl.replace(/\/$/, "");
194
+ const direct = `${raw.replace(/^http/, "ws")}/ws/${this.account.agentId}`;
195
+ const strippedRaw = raw.replace(/\/api\/v1\/?$/i, "");
196
+ const stripped = `${strippedRaw.replace(/^http/, "ws")}/ws/${this.account.agentId}`;
197
+ const urls = [stripped, direct].filter(Boolean);
198
+ return Array.from(new Set(urls));
199
+ }
200
+ async doConnect() {
201
+ const wsUrls = this.resolveWsUrls();
202
+ const wsUrl = wsUrls[this.wsUrlCursor % wsUrls.length];
203
+ this.log("info", `Connecting to ${wsUrl}`);
204
+ try {
205
+ if (typeof globalThis.WebSocket !== "undefined") {
206
+ // Browser / Deno / CF Workers
207
+ this.ws = new WebSocket(wsUrl);
208
+ }
209
+ else {
210
+ // Node.js — use ws package
211
+ const { default: WS } = await import("ws");
212
+ this.ws = new WS(wsUrl);
213
+ }
214
+ }
215
+ catch (err) {
216
+ this.log("error", "Failed to create WebSocket", err);
217
+ this.scheduleReconnect();
218
+ return;
219
+ }
220
+ const ws = this.ws;
221
+ const onOpen = () => {
222
+ this.log("info", "WebSocket connected, authenticating...");
223
+ this.reconnectAttempts = 0;
224
+ // Send auth as first message
225
+ this.send(JSON.stringify({ action: "auth", token: this.account.token }));
226
+ this.startHeartbeat();
227
+ this.onConnect?.();
228
+ };
229
+ const onMessage = (event) => {
230
+ const raw = typeof event === "object" && "data" in event ? String(event.data) : String(event);
231
+ if (raw === "pong")
232
+ return; // heartbeat response
233
+ try {
234
+ const msg = JSON.parse(raw);
235
+ if (msg.type === "action_result") {
236
+ this.log("debug", `WS action_result received: ok=${msg.ok} keys=${Object.keys(msg.data ?? {}).join(",")}`);
237
+ }
238
+ this.handleServerMessage(msg);
239
+ }
240
+ catch {
241
+ this.log("warn", "Non-JSON message received", raw);
242
+ }
243
+ };
244
+ const onClose = (event) => {
245
+ const code = "code" in event ? event.code : undefined;
246
+ const reason = "reason" in event ? event.reason : "";
247
+ this.log("info", `WebSocket closed: ${code} ${reason}`);
248
+ this.clearTimers();
249
+ this.onDisconnect?.();
250
+ if (!this.closed) {
251
+ const wsUrls = this.resolveWsUrls();
252
+ if (wsUrls.length > 1) {
253
+ this.wsUrlCursor = (this.wsUrlCursor + 1) % wsUrls.length;
254
+ this.log("info", `Switching WS endpoint candidate -> ${wsUrls[this.wsUrlCursor]}`);
255
+ }
256
+ this.scheduleReconnect();
257
+ }
258
+ };
259
+ const onError = (err) => {
260
+ const error = err instanceof Error ? err : new Error("WebSocket error");
261
+ this.log("error", "WebSocket error", error);
262
+ this.onError?.(error);
263
+ };
264
+ // Attach listeners (works for both browser WebSocket and ws package)
265
+ if ("addEventListener" in ws) {
266
+ ws.addEventListener("open", onOpen);
267
+ ws.addEventListener("message", onMessage);
268
+ ws.addEventListener("close", onClose);
269
+ ws.addEventListener("error", onError);
270
+ }
271
+ else {
272
+ const nodeWs = ws;
273
+ nodeWs.on("open", onOpen);
274
+ nodeWs.on("message", (data) => onMessage({ data: data.toString() }));
275
+ nodeWs.on("close", (code, reason) => onClose({ code, reason: reason?.toString() }));
276
+ nodeWs.on("error", onError);
277
+ }
278
+ }
279
+ handleServerMessage(msg) {
280
+ if (msg.type === "action_result") {
281
+ const result = msg;
282
+ // Capture agent display name from auth response
283
+ if (result.ok && result.data && typeof result.data === "object") {
284
+ const data = result.data;
285
+ if (data.authenticated && data.agent_display_name) {
286
+ this.account = { ...this.account, name: String(data.agent_display_name) };
287
+ this.log("info", `Agent display name resolved: ${this.account.name}`);
288
+ }
289
+ }
290
+ const p = this.pending.get(result.request_id);
291
+ if (p) {
292
+ clearTimeout(p.timer);
293
+ this.pending.delete(result.request_id);
294
+ if (result.ok) {
295
+ p.resolve(result.data);
296
+ }
297
+ else {
298
+ p.reject(new Error(result.error ?? "Action failed"));
299
+ }
300
+ }
301
+ return;
302
+ }
303
+ if (msg.type === "new_message") {
304
+ this.onMessage?.(msg);
305
+ return;
306
+ }
307
+ if (msg.type === "task_status") {
308
+ this.onTaskStatus?.(msg);
309
+ return;
310
+ }
311
+ if (msg.type === "e2e_key_request") {
312
+ const requestId = String(msg.request_id ?? "").trim();
313
+ if (!requestId)
314
+ return;
315
+ this.log("info", `e2e_key_request received rid=${requestId}`);
316
+ if (!this.e2eKey) {
317
+ this.log("warn", `e2e_key_request rid=${requestId} failed: e2e_not_configured`);
318
+ this.send(JSON.stringify({
319
+ type: "e2e_key_response",
320
+ request_id: requestId,
321
+ ok: false,
322
+ error: "e2e_not_configured",
323
+ }));
324
+ return;
325
+ }
326
+ this.send(JSON.stringify({
327
+ type: "e2e_key_response",
328
+ request_id: requestId,
329
+ ok: true,
330
+ key_b64: toBase64(this.e2eKey),
331
+ }));
332
+ this.log("info", `e2e_key_response sent rid=${requestId}`);
333
+ return;
334
+ }
335
+ this.log("debug", "Unknown message type", msg);
336
+ }
337
+ // ---------------------------------------------------------------------------
338
+ // Heartbeat & reconnect
339
+ // ---------------------------------------------------------------------------
340
+ startHeartbeat() {
341
+ this.stopHeartbeat();
342
+ this.heartbeatTimer = setInterval(() => {
343
+ if (this.connected) {
344
+ this.send("ping"); // WTT uses text "ping", not WS ping frames
345
+ }
346
+ }, this.heartbeatInterval);
347
+ }
348
+ stopHeartbeat() {
349
+ if (this.heartbeatTimer) {
350
+ clearInterval(this.heartbeatTimer);
351
+ this.heartbeatTimer = null;
352
+ }
353
+ }
354
+ scheduleReconnect() {
355
+ if (this.closed || this.reconnectAttempts >= this.maxReconnectAttempts) {
356
+ this.log("warn", "Max reconnect attempts reached, giving up");
357
+ return;
358
+ }
359
+ const delay = Math.min(1000 * 2 ** this.reconnectAttempts, 30_000);
360
+ this.reconnectAttempts++;
361
+ this.log("info", `Reconnecting in ${delay}ms (attempt ${this.reconnectAttempts})`);
362
+ this.reconnectTimer = setTimeout(() => {
363
+ this.reconnectTimer = null;
364
+ this.doConnect();
365
+ }, delay);
366
+ }
367
+ clearTimers() {
368
+ this.stopHeartbeat();
369
+ if (this.reconnectTimer) {
370
+ clearTimeout(this.reconnectTimer);
371
+ this.reconnectTimer = null;
372
+ }
373
+ }
374
+ send(data) {
375
+ try {
376
+ if (this.ws && this.connected) {
377
+ this.ws.send(data);
378
+ }
379
+ }
380
+ catch (err) {
381
+ this.log("error", "Send failed", err);
382
+ }
383
+ }
384
+ }
385
+ //# sourceMappingURL=ws-client.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ws-client.js","sourceRoot":"","sources":["../src/ws-client.ts"],"names":[],"mappings":"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"}
package/index.ts ADDED
@@ -0,0 +1,19 @@
1
+ import { wttPlugin } from "./src/channel.js";
2
+
3
+ export { processWTTCommandText } from "./src/channel.js";
4
+
5
+ const plugin = {
6
+ id: "wtt",
7
+ name: "WTT",
8
+ description: "WTT channel plugin",
9
+ configSchema: {
10
+ type: "object",
11
+ additionalProperties: false,
12
+ properties: {},
13
+ },
14
+ register(api: any) {
15
+ api.registerChannel({ plugin: wttPlugin });
16
+ },
17
+ };
18
+
19
+ export default plugin;
@@ -0,0 +1,49 @@
1
+ {
2
+ "id": "wtt",
3
+ "channels": ["wtt"],
4
+ "name": "WTT",
5
+ "description": "WTT channel plugin for OpenClaw",
6
+ "configSchema": {
7
+ "type": "object",
8
+ "additionalProperties": false,
9
+ "properties": {
10
+ "accounts": {
11
+ "type": "object",
12
+ "additionalProperties": {
13
+ "type": "object",
14
+ "additionalProperties": false,
15
+ "properties": {
16
+ "enabled": { "type": "boolean" },
17
+ "name": { "type": "string" },
18
+ "cloudUrl": { "type": "string" },
19
+ "agentId": { "type": "string" },
20
+ "token": { "type": "string" },
21
+ "e2ePassword": { "type": "string" },
22
+ "inboundPollIntervalMs": { "type": "number" },
23
+ "inboundPollLimit": { "type": "number" },
24
+ "inboundDedupWindowMs": { "type": "number" },
25
+ "inboundDedupMaxEntries": { "type": "number" },
26
+ "slashCompat": { "type": "boolean" },
27
+ "slashCompatWttPrefixOnly": { "type": "boolean" },
28
+ "slashBypassMentionGate": { "type": "boolean" },
29
+ "taskExecutorScope": { "type": "string", "enum": ["all", "pipeline_only"] }
30
+ }
31
+ }
32
+ },
33
+ "enabled": { "type": "boolean" },
34
+ "name": { "type": "string" },
35
+ "cloudUrl": { "type": "string" },
36
+ "agentId": { "type": "string" },
37
+ "token": { "type": "string" },
38
+ "e2ePassword": { "type": "string" },
39
+ "inboundPollIntervalMs": { "type": "number" },
40
+ "inboundPollLimit": { "type": "number" },
41
+ "inboundDedupWindowMs": { "type": "number" },
42
+ "inboundDedupMaxEntries": { "type": "number" },
43
+ "slashCompat": { "type": "boolean" },
44
+ "slashCompatWttPrefixOnly": { "type": "boolean" },
45
+ "slashBypassMentionGate": { "type": "boolean" },
46
+ "taskExecutorScope": { "type": "string", "enum": ["all", "pipeline_only"] }
47
+ }
48
+ }
49
+ }
package/package.json ADDED
@@ -0,0 +1,62 @@
1
+ {
2
+ "name": "@cecwxf/wtt",
3
+ "version": "0.1.0",
4
+ "description": "WTT channel plugin for OpenClaw — real-time Agent communication via Topics",
5
+ "license": "MIT",
6
+ "type": "module",
7
+ "main": "dist/index.js",
8
+ "types": "dist/index.d.ts",
9
+ "exports": {
10
+ ".": {
11
+ "import": "./dist/index.js",
12
+ "types": "./dist/index.d.ts"
13
+ }
14
+ },
15
+ "bin": {
16
+ "openclaw-wtt-bootstrap": "bin/openclaw-wtt-bootstrap.mjs"
17
+ },
18
+ "files": ["dist", "README.md", "index.ts", "openclaw.plugin.json", "bin/openclaw-wtt-bootstrap.mjs", "scripts/install-bootstrap-cli.sh"],
19
+ "scripts": {
20
+ "build": "tsc",
21
+ "dev": "tsc --watch",
22
+ "clean": "rm -rf dist",
23
+ "test:commands": "node test_command_router.mjs",
24
+ "test:runtime": "node test_runtime.mjs",
25
+ "test:inbound": "node test_inbound.mjs",
26
+ "install:bootstrap-cli": "bash ./scripts/install-bootstrap-cli.sh"
27
+ },
28
+ "dependencies": {
29
+ "ws": "^8.18.0"
30
+ },
31
+ "devDependencies": {
32
+ "@types/ws": "^8.5.13",
33
+ "@types/node": "^22.0.0",
34
+ "typescript": "^5.7.0"
35
+ },
36
+ "openclaw": {
37
+ "extensions": [
38
+ "./index.ts"
39
+ ],
40
+ "channel": {
41
+ "id": "wtt",
42
+ "label": "WTT",
43
+ "selectionLabel": "WTT (WebSocket)",
44
+ "docsPath": "/channels/wtt",
45
+ "docsLabel": "wtt",
46
+ "blurb": "WTT real-time topic + p2p communication channel.",
47
+ "aliases": [
48
+ "want-to-talk"
49
+ ],
50
+ "order": 95
51
+ }
52
+ },
53
+ "keywords": ["wtt", "openclaw", "plugin", "agent", "chat", "e2e-encryption"],
54
+ "repository": {
55
+ "type": "git",
56
+ "url": "https://github.com/cecwxf/wtt.git",
57
+ "directory": "wtt_plugin"
58
+ },
59
+ "publishConfig": {
60
+ "access": "public"
61
+ }
62
+ }
@@ -0,0 +1,54 @@
1
+ #!/usr/bin/env bash
2
+ set -euo pipefail
3
+
4
+ # Install openclaw-wtt-bootstrap as a system-level command.
5
+ #
6
+ # Usage:
7
+ # bash scripts/install-bootstrap-cli.sh
8
+ # bash scripts/install-bootstrap-cli.sh --target /usr/local/bin/openclaw-wtt-bootstrap
9
+ # bash scripts/install-bootstrap-cli.sh --uninstall
10
+
11
+ TARGET="/usr/local/bin/openclaw-wtt-bootstrap"
12
+ UNINSTALL="0"
13
+
14
+ while [[ $# -gt 0 ]]; do
15
+ case "$1" in
16
+ --target)
17
+ TARGET="$2"
18
+ shift 2
19
+ ;;
20
+ --uninstall)
21
+ UNINSTALL="1"
22
+ shift
23
+ ;;
24
+ *)
25
+ echo "Unknown arg: $1" >&2
26
+ exit 2
27
+ ;;
28
+ esac
29
+ done
30
+
31
+ SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
32
+ SRC="$(cd "$SCRIPT_DIR/.." && pwd)/bin/openclaw-wtt-bootstrap.mjs"
33
+
34
+ if [[ "$UNINSTALL" == "1" ]]; then
35
+ if [[ -L "$TARGET" || -f "$TARGET" ]]; then
36
+ rm -f "$TARGET"
37
+ echo "Removed $TARGET"
38
+ else
39
+ echo "Nothing to remove: $TARGET"
40
+ fi
41
+ exit 0
42
+ fi
43
+
44
+ if [[ ! -f "$SRC" ]]; then
45
+ echo "Source not found: $SRC" >&2
46
+ exit 1
47
+ fi
48
+
49
+ mkdir -p "$(dirname "$TARGET")"
50
+ ln -sf "$SRC" "$TARGET"
51
+ chmod +x "$SRC" "$TARGET"
52
+
53
+ echo "Installed: $TARGET -> $SRC"
54
+ echo "Try: openclaw-wtt-bootstrap --help"