openclaw-vchat-plugin 0.0.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 (70) hide show
  1. package/bin/openclaw-vchat.js +110 -0
  2. package/dist/commands.d.ts +18 -0
  3. package/dist/commands.d.ts.map +1 -0
  4. package/dist/commands.js +509 -0
  5. package/dist/commands.js.map +1 -0
  6. package/dist/constants.d.ts +14 -0
  7. package/dist/constants.d.ts.map +1 -0
  8. package/dist/constants.js +51 -0
  9. package/dist/constants.js.map +1 -0
  10. package/dist/gateway-client.d.ts +43 -0
  11. package/dist/gateway-client.d.ts.map +1 -0
  12. package/dist/gateway-client.js +623 -0
  13. package/dist/gateway-client.js.map +1 -0
  14. package/dist/group-manager.d.ts +30 -0
  15. package/dist/group-manager.d.ts.map +1 -0
  16. package/dist/group-manager.js +107 -0
  17. package/dist/group-manager.js.map +1 -0
  18. package/dist/index.d.ts +2 -0
  19. package/dist/index.d.ts.map +1 -0
  20. package/dist/index.js +382 -0
  21. package/dist/index.js.map +1 -0
  22. package/dist/media-handler.d.ts +31 -0
  23. package/dist/media-handler.d.ts.map +1 -0
  24. package/dist/media-handler.js +67 -0
  25. package/dist/media-handler.js.map +1 -0
  26. package/dist/message-handler.d.ts +52 -0
  27. package/dist/message-handler.d.ts.map +1 -0
  28. package/dist/message-handler.js +291 -0
  29. package/dist/message-handler.js.map +1 -0
  30. package/dist/relay-server.d.ts +16 -0
  31. package/dist/relay-server.d.ts.map +1 -0
  32. package/dist/relay-server.js +877 -0
  33. package/dist/relay-server.js.map +1 -0
  34. package/dist/routes/config.routes.d.ts +12 -0
  35. package/dist/routes/config.routes.d.ts.map +1 -0
  36. package/dist/routes/config.routes.js +175 -0
  37. package/dist/routes/config.routes.js.map +1 -0
  38. package/dist/services/config.service.d.ts +57 -0
  39. package/dist/services/config.service.d.ts.map +1 -0
  40. package/dist/services/config.service.js +361 -0
  41. package/dist/services/config.service.js.map +1 -0
  42. package/dist/session-key.d.ts +8 -0
  43. package/dist/session-key.d.ts.map +1 -0
  44. package/dist/session-key.js +28 -0
  45. package/dist/session-key.js.map +1 -0
  46. package/dist/session-manager.d.ts +32 -0
  47. package/dist/session-manager.d.ts.map +1 -0
  48. package/dist/session-manager.js +303 -0
  49. package/dist/session-manager.js.map +1 -0
  50. package/dist/types.d.ts +81 -0
  51. package/dist/types.d.ts.map +1 -0
  52. package/dist/types.js +3 -0
  53. package/dist/types.js.map +1 -0
  54. package/nginx-proxy.conf +24 -0
  55. package/package.json +51 -0
  56. package/src/commands.ts +499 -0
  57. package/src/constants.ts +49 -0
  58. package/src/gateway-client.ts +648 -0
  59. package/src/group-manager.ts +119 -0
  60. package/src/index.ts +443 -0
  61. package/src/media-handler.ts +70 -0
  62. package/src/message-handler.ts +419 -0
  63. package/src/relay-server.ts +979 -0
  64. package/src/routes/config.routes.ts +144 -0
  65. package/src/services/config.service.ts +398 -0
  66. package/src/session-key.ts +30 -0
  67. package/src/session-manager.ts +374 -0
  68. package/src/types.ts +96 -0
  69. package/start.sh +5 -0
  70. package/tsconfig.json +26 -0
@@ -0,0 +1,30 @@
1
+ /**
2
+ * 虚拟群数据模型
3
+ */
4
+ export interface VirtualGroup {
5
+ id: string;
6
+ name: string;
7
+ emoji: string;
8
+ mode: 'broadcast' | 'mention';
9
+ agentIds: string[];
10
+ createdAt: number;
11
+ updatedAt: number;
12
+ }
13
+ /**
14
+ * 虚拟群管理器 — 本地 JSON 持久化
15
+ */
16
+ export declare class GroupManager {
17
+ private groups;
18
+ private dataFile;
19
+ constructor(dataDir: string);
20
+ list(): VirtualGroup[];
21
+ get(groupId: string): VirtualGroup | undefined;
22
+ create(name: string, emoji?: string, agentIds?: string[], mode?: 'broadcast' | 'mention'): VirtualGroup;
23
+ update(groupId: string, data: Partial<Pick<VirtualGroup, 'name' | 'emoji' | 'mode'>>): VirtualGroup | null;
24
+ delete(groupId: string): boolean;
25
+ addAgent(groupId: string, agentId: string): VirtualGroup | null;
26
+ removeAgent(groupId: string, agentId: string): VirtualGroup | null;
27
+ private load;
28
+ private save;
29
+ }
30
+ //# sourceMappingURL=group-manager.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"group-manager.d.ts","sourceRoot":"","sources":["../src/group-manager.ts"],"names":[],"mappings":"AAIA;;GAEG;AACH,MAAM,WAAW,YAAY;IACzB,EAAE,EAAE,MAAM,CAAC;IACX,IAAI,EAAE,MAAM,CAAC;IACb,KAAK,EAAE,MAAM,CAAC;IACd,IAAI,EAAE,WAAW,GAAG,SAAS,CAAC;IAC9B,QAAQ,EAAE,MAAM,EAAE,CAAC;IACnB,SAAS,EAAE,MAAM,CAAC;IAClB,SAAS,EAAE,MAAM,CAAC;CACrB;AAED;;GAEG;AACH,qBAAa,YAAY;IACrB,OAAO,CAAC,MAAM,CAAsB;IACpC,OAAO,CAAC,QAAQ,CAAS;gBAEb,OAAO,EAAE,MAAM;IAO3B,IAAI,IAAI,YAAY,EAAE;IAItB,GAAG,CAAC,OAAO,EAAE,MAAM,GAAG,YAAY,GAAG,SAAS;IAI9C,MAAM,CAAC,IAAI,EAAE,MAAM,EAAE,KAAK,GAAE,MAAa,EAAE,QAAQ,GAAE,MAAM,EAAO,EAAE,IAAI,GAAE,WAAW,GAAG,SAAqB,GAAG,YAAY;IAe5H,MAAM,CAAC,OAAO,EAAE,MAAM,EAAE,IAAI,EAAE,OAAO,CAAC,IAAI,CAAC,YAAY,EAAE,MAAM,GAAG,OAAO,GAAG,MAAM,CAAC,CAAC,GAAG,YAAY,GAAG,IAAI;IAW1G,MAAM,CAAC,OAAO,EAAE,MAAM,GAAG,OAAO;IAYhC,QAAQ,CAAC,OAAO,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,GAAG,YAAY,GAAG,IAAI;IAW/D,WAAW,CAAC,OAAO,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,GAAG,YAAY,GAAG,IAAI;IAWlE,OAAO,CAAC,IAAI;IAYZ,OAAO,CAAC,IAAI;CAOf"}
@@ -0,0 +1,107 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.GroupManager = void 0;
7
+ const fs_1 = __importDefault(require("fs"));
8
+ const path_1 = __importDefault(require("path"));
9
+ const crypto_1 = require("crypto");
10
+ /**
11
+ * 虚拟群管理器 — 本地 JSON 持久化
12
+ */
13
+ class GroupManager {
14
+ constructor(dataDir) {
15
+ this.groups = [];
16
+ this.dataFile = path_1.default.join(dataDir, 'groups.json');
17
+ this.load();
18
+ }
19
+ // ===== CRUD =====
20
+ list() {
21
+ return this.groups;
22
+ }
23
+ get(groupId) {
24
+ return this.groups.find(g => g.id === groupId);
25
+ }
26
+ create(name, emoji = '👥', agentIds = [], mode = 'mention') {
27
+ const group = {
28
+ id: (0, crypto_1.randomUUID)().substring(0, 8),
29
+ name,
30
+ emoji,
31
+ mode,
32
+ agentIds,
33
+ createdAt: Date.now(),
34
+ updatedAt: Date.now(),
35
+ };
36
+ this.groups.push(group);
37
+ this.save();
38
+ return group;
39
+ }
40
+ update(groupId, data) {
41
+ const group = this.get(groupId);
42
+ if (!group)
43
+ return null;
44
+ if (data.name !== undefined)
45
+ group.name = data.name;
46
+ if (data.emoji !== undefined)
47
+ group.emoji = data.emoji;
48
+ if (data.mode !== undefined)
49
+ group.mode = data.mode;
50
+ group.updatedAt = Date.now();
51
+ this.save();
52
+ return group;
53
+ }
54
+ delete(groupId) {
55
+ const before = this.groups.length;
56
+ this.groups = this.groups.filter(g => g.id !== groupId);
57
+ if (this.groups.length < before) {
58
+ this.save();
59
+ return true;
60
+ }
61
+ return false;
62
+ }
63
+ // ===== Agent 绑定 =====
64
+ addAgent(groupId, agentId) {
65
+ const group = this.get(groupId);
66
+ if (!group)
67
+ return null;
68
+ if (!group.agentIds.includes(agentId)) {
69
+ group.agentIds.push(agentId);
70
+ group.updatedAt = Date.now();
71
+ this.save();
72
+ }
73
+ return group;
74
+ }
75
+ removeAgent(groupId, agentId) {
76
+ const group = this.get(groupId);
77
+ if (!group)
78
+ return null;
79
+ group.agentIds = group.agentIds.filter(id => id !== agentId);
80
+ group.updatedAt = Date.now();
81
+ this.save();
82
+ return group;
83
+ }
84
+ // ===== 持久化 =====
85
+ load() {
86
+ try {
87
+ if (fs_1.default.existsSync(this.dataFile)) {
88
+ const raw = fs_1.default.readFileSync(this.dataFile, 'utf-8');
89
+ this.groups = JSON.parse(raw);
90
+ }
91
+ }
92
+ catch (err) {
93
+ console.error('[GroupManager] 加载失败:', err);
94
+ this.groups = [];
95
+ }
96
+ }
97
+ save() {
98
+ try {
99
+ fs_1.default.writeFileSync(this.dataFile, JSON.stringify(this.groups, null, 2), 'utf-8');
100
+ }
101
+ catch (err) {
102
+ console.error('[GroupManager] 保存失败:', err);
103
+ }
104
+ }
105
+ }
106
+ exports.GroupManager = GroupManager;
107
+ //# sourceMappingURL=group-manager.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"group-manager.js","sourceRoot":"","sources":["../src/group-manager.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":""}
package/dist/index.js ADDED
@@ -0,0 +1,382 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ const path_1 = __importDefault(require("path"));
7
+ const fs_1 = __importDefault(require("fs"));
8
+ const node_fetch_1 = __importDefault(require("node-fetch"));
9
+ const ws_1 = __importDefault(require("ws"));
10
+ const relay_server_1 = require("./relay-server");
11
+ const gateway_client_1 = require("./gateway-client");
12
+ const commands_1 = require("./commands");
13
+ const message_handler_1 = require("./message-handler");
14
+ const session_key_1 = require("./session-key");
15
+ // @ts-ignore
16
+ const qrcode_terminal_1 = __importDefault(require("qrcode-terminal"));
17
+ /**
18
+ * OpenClaw VChat Plugin 入口
19
+ *
20
+ * 启动流程:
21
+ * 1. 检查 ~/.openclaw/wechat-device.json 是否有 deviceKey
22
+ * 2. 有 → 用 deviceKey 注册 → 建立反向 WS 隧道
23
+ * 3. 没有 → 调 /api/devices/pair/init → 终端打印 QR 码 → 轮询配对 → 保存密钥
24
+ */
25
+ const DATA_DIR = path_1.default.join(__dirname, '..', 'data');
26
+ const UPLOAD_DIR = path_1.default.join(DATA_DIR, 'uploads');
27
+ const OPENCLAW_HOME = process.env.OPENCLAW_HOME || path_1.default.join(process.env.HOME || '/root', '.openclaw');
28
+ const DEVICE_CONFIG_PATH = path_1.default.join(OPENCLAW_HOME, 'wechat-device.json');
29
+ const BACKEND_URL = process.env.WECHAT_BACKEND_URL || 'https://api.deck0.dev';
30
+ fs_1.default.mkdirSync(UPLOAD_DIR, { recursive: true });
31
+ fs_1.default.mkdirSync(OPENCLAW_HOME, { recursive: true });
32
+ const config = {
33
+ port: 39217,
34
+ openclawGatewayUrl: process.env.OPENCLAW_GATEWAY_URL || 'http://localhost:18789',
35
+ openclawModel: process.env.OPENCLAW_MODEL || 'nvidia/z-ai/glm5',
36
+ openclawAuthToken: process.env.OPENCLAW_AUTH_TOKEN || '',
37
+ uploadDir: UPLOAD_DIR,
38
+ maxFileSize: 10 * 1024 * 1024,
39
+ maxVoiceDuration: 60,
40
+ internalSecret: process.env.PLUGIN_SECRET || '',
41
+ allowLocalFallback: process.env.WECHAT_PLUGIN_ENABLE_LOCAL_FALLBACK === '1',
42
+ };
43
+ function readDeviceConfig() {
44
+ try {
45
+ return JSON.parse(fs_1.default.readFileSync(DEVICE_CONFIG_PATH, 'utf-8'));
46
+ }
47
+ catch {
48
+ return null;
49
+ }
50
+ }
51
+ function saveDeviceConfig(cfg) {
52
+ fs_1.default.writeFileSync(DEVICE_CONFIG_PATH, JSON.stringify(cfg, null, 2) + '\n', 'utf-8');
53
+ }
54
+ // ===== 反向 WS 隧道 =====
55
+ let tunnelWs = null;
56
+ let gateway;
57
+ let relayServer = null;
58
+ let tunnelMessageHandler = null;
59
+ const tunnelAbortByUser = new Map();
60
+ function sendTunnelFrame(payload) {
61
+ if (tunnelWs && tunnelWs.readyState === ws_1.default.OPEN) {
62
+ tunnelWs.send(JSON.stringify(payload));
63
+ }
64
+ }
65
+ function normalizeSenderId(value) {
66
+ const normalized = String(value || '').trim();
67
+ return normalized || 'unknown';
68
+ }
69
+ function ensureTunnelSession(userId, sessionId, agentId, groupId) {
70
+ if (!relayServer)
71
+ throw new Error('插件尚未完成初始化');
72
+ const normalizedAgentId = (0, session_key_1.sanitizeWeChatId)(agentId, 'main');
73
+ const existingSessionId = String(sessionId || '').trim();
74
+ if (existingSessionId) {
75
+ return relayServer.sessionManager.ensureSessionBinding(userId, existingSessionId, normalizedAgentId, (0, session_key_1.buildWeChatGatewaySessionKey)(userId, existingSessionId, normalizedAgentId, groupId)).id;
76
+ }
77
+ const created = relayServer.sessionManager.createSession(userId, undefined, normalizedAgentId);
78
+ return relayServer.sessionManager.ensureSessionBinding(userId, created.id, normalizedAgentId, (0, session_key_1.buildWeChatGatewaySessionKey)(userId, created.id, normalizedAgentId, groupId)).id;
79
+ }
80
+ function connectTunnel(deviceToken) {
81
+ const wsUrl = BACKEND_URL.replace(/^https:/, 'wss:').replace(/^http:/, 'ws:')
82
+ + `/ws/plugin?deviceToken=${encodeURIComponent(deviceToken)}`;
83
+ console.log('[隧道] 正在连接后端...');
84
+ tunnelWs = new ws_1.default(wsUrl);
85
+ tunnelWs.on('open', () => {
86
+ console.log('[隧道] ✅ 已连接到后端,等待小程序消息');
87
+ });
88
+ tunnelWs.on('message', (data) => {
89
+ // 后端转发的小程序消息 → 处理并回复
90
+ handleClientMessage(data.toString());
91
+ });
92
+ tunnelWs.on('close', (code, reason) => {
93
+ console.log(`[隧道] 断开 (${code}): ${reason}`);
94
+ for (const abort of tunnelAbortByUser.values()) {
95
+ try {
96
+ abort();
97
+ }
98
+ catch { /* ignore */ }
99
+ }
100
+ tunnelAbortByUser.clear();
101
+ tunnelWs = null;
102
+ // 自动重连
103
+ setTimeout(() => {
104
+ console.log('[隧道] 尝试重连...');
105
+ connectTunnel(deviceToken);
106
+ }, 5000);
107
+ });
108
+ tunnelWs.on('error', (err) => {
109
+ console.error('[隧道] 连接错误:', err.message);
110
+ });
111
+ }
112
+ /**
113
+ * 处理小程序通过隧道发来的消息
114
+ */
115
+ function handleClientMessage(raw) {
116
+ try {
117
+ const msg = JSON.parse(raw);
118
+ const userId = normalizeSenderId(msg.userId);
119
+ const requestId = String(msg.requestId || '').trim();
120
+ if (msg.type === 'ping') {
121
+ sendTunnelFrame({ type: 'pong' });
122
+ return;
123
+ }
124
+ if (msg.type === 'chat') {
125
+ if (!relayServer || !tunnelMessageHandler) {
126
+ sendTunnelFrame({ type: 'error', message: '插件尚未完成初始化', error: '插件尚未完成初始化' });
127
+ return;
128
+ }
129
+ const content = String(msg.content || '').trim();
130
+ if (!content) {
131
+ sendTunnelFrame({ type: 'error', message: '消息内容不能为空', error: '消息内容不能为空' });
132
+ return;
133
+ }
134
+ const agentId = String(msg.agentId || 'main');
135
+ const sessionId = String(msg.sessionId || '').trim();
136
+ const groupId = msg.groupId ? String(msg.groupId) : undefined;
137
+ const sid = ensureTunnelSession(userId, sessionId, agentId, groupId);
138
+ console.log(`[BridgeTunnel] requestId=${requestId || '-'} type=chat agent=${agentId} session=${sid} user=${userId}`);
139
+ const mediaUrl = msg.mediaUrl ? String(msg.mediaUrl) : undefined;
140
+ const rawMessageType = String(msg.messageType || 'text');
141
+ const messageType = rawMessageType === 'voice' || rawMessageType === 'image' || rawMessageType === 'command' || rawMessageType === 'system'
142
+ ? rawMessageType
143
+ : 'text';
144
+ const routingHint = {
145
+ commandRoute: typeof msg.commandRoute === 'string' ? msg.commandRoute : undefined,
146
+ routeSource: typeof msg.routeSource === 'string' ? msg.routeSource : undefined,
147
+ requestId: typeof msg.requestId === 'string' ? msg.requestId : undefined,
148
+ };
149
+ const previousAbort = tunnelAbortByUser.get(userId);
150
+ if (previousAbort) {
151
+ previousAbort();
152
+ tunnelAbortByUser.delete(userId);
153
+ }
154
+ sendTunnelFrame({ type: 'start', sessionId: sid, agentId, requestId });
155
+ let currentAbort = null;
156
+ let streamFinished = false;
157
+ const abort = tunnelMessageHandler.handleMessageStream(userId, sid, content, messageType, {
158
+ onThinking: (text) => {
159
+ sendTunnelFrame({ type: 'thinking', text, sessionId: sid, agentId, requestId });
160
+ },
161
+ onChunk: (text) => {
162
+ sendTunnelFrame({
163
+ type: 'chunk',
164
+ text,
165
+ delta: text,
166
+ sessionId: sid,
167
+ agentId,
168
+ requestId,
169
+ });
170
+ },
171
+ onDone: (userMessage, assistantMessage) => {
172
+ streamFinished = true;
173
+ if (currentAbort && tunnelAbortByUser.get(userId) === currentAbort)
174
+ tunnelAbortByUser.delete(userId);
175
+ sendTunnelFrame({
176
+ type: 'done',
177
+ sessionId: sid,
178
+ agentId,
179
+ userMessage,
180
+ assistantMessage,
181
+ text: assistantMessage?.content || '',
182
+ requestId,
183
+ });
184
+ },
185
+ onError: (error) => {
186
+ streamFinished = true;
187
+ if (currentAbort && tunnelAbortByUser.get(userId) === currentAbort)
188
+ tunnelAbortByUser.delete(userId);
189
+ sendTunnelFrame({
190
+ type: 'error',
191
+ sessionId: sid,
192
+ agentId,
193
+ message: error,
194
+ error,
195
+ requestId,
196
+ });
197
+ },
198
+ }, agentId, mediaUrl, undefined, groupId, routingHint);
199
+ currentAbort = abort;
200
+ if (streamFinished) {
201
+ return;
202
+ }
203
+ tunnelAbortByUser.set(userId, abort);
204
+ return;
205
+ }
206
+ if (msg.type === 'stop') {
207
+ const abort = tunnelAbortByUser.get(userId);
208
+ if (abort) {
209
+ abort();
210
+ tunnelAbortByUser.delete(userId);
211
+ }
212
+ sendTunnelFrame({ type: 'stopped', requestId });
213
+ return;
214
+ }
215
+ sendTunnelFrame({
216
+ type: 'error',
217
+ message: `未知消息类型: ${msg.type}`,
218
+ error: `未知消息类型: ${msg.type}`,
219
+ });
220
+ }
221
+ catch (err) {
222
+ console.error('[隧道] 处理消息错误:', err);
223
+ sendTunnelFrame({
224
+ type: 'error',
225
+ message: '隧道消息解析失败',
226
+ error: '隧道消息解析失败',
227
+ });
228
+ }
229
+ }
230
+ // ===== 配对流程 =====
231
+ async function startPairing() {
232
+ console.log('\n📱 首次使用,需要与小程序配对...\n');
233
+ // 调后端 init
234
+ const initRes = await (0, node_fetch_1.default)(`${BACKEND_URL}/api/devices/pair/init`, { method: 'POST' });
235
+ if (!initRes.ok) {
236
+ throw new Error(`配对初始化失败: ${initRes.status}`);
237
+ }
238
+ const { pairingToken, pairingCode } = await initRes.json();
239
+ // 终端显示二维码
240
+ const qrContent = JSON.stringify({ type: 'openclaw-pair', token: pairingToken });
241
+ console.log('╔══════════════════════════════════════════╗');
242
+ console.log('║ 请用 OpenClaw 小程序扫描下方二维码配对 ║');
243
+ console.log('╚══════════════════════════════════════════╝');
244
+ console.log('');
245
+ qrcode_terminal_1.default.generate(qrContent, { small: true }, (qr) => {
246
+ console.log(qr);
247
+ });
248
+ console.log(` 配对码: ${pairingCode} (5分钟内有效)`);
249
+ console.log('');
250
+ console.log(' 等待小程序扫码...');
251
+ console.log('');
252
+ // 轮询
253
+ return new Promise((resolve, reject) => {
254
+ const pollInterval = setInterval(async () => {
255
+ try {
256
+ const statusRes = await (0, node_fetch_1.default)(`${BACKEND_URL}/api/devices/pair/status?token=${encodeURIComponent(pairingToken)}`);
257
+ const status = await statusRes.json();
258
+ if (status.status === 'paired') {
259
+ clearInterval(pollInterval);
260
+ console.log('✅ 配对成功!设备已绑定\n');
261
+ const cfg = {
262
+ deviceKey: status.deviceKey,
263
+ deviceToken: status.deviceToken,
264
+ backendUrl: BACKEND_URL,
265
+ pairedAt: new Date().toISOString(),
266
+ };
267
+ saveDeviceConfig(cfg);
268
+ resolve(cfg);
269
+ }
270
+ else if (status.status === 'expired' || status.status === 'not_found') {
271
+ clearInterval(pollInterval);
272
+ reject(new Error('配对已过期,请重新启动插件'));
273
+ }
274
+ // pending → 继续轮询
275
+ }
276
+ catch (err) {
277
+ console.error('[配对] 轮询错误:', err.message);
278
+ }
279
+ }, 2000);
280
+ // 5分钟超时
281
+ setTimeout(() => {
282
+ clearInterval(pollInterval);
283
+ reject(new Error('配对超时(5分钟),请重新启动插件'));
284
+ }, 5 * 60 * 1000);
285
+ });
286
+ }
287
+ // ===== 注册设备 =====
288
+ async function registerDevice(deviceKey) {
289
+ const res = await (0, node_fetch_1.default)(`${BACKEND_URL}/api/devices/register`, {
290
+ method: 'POST',
291
+ headers: { 'Content-Type': 'application/json' },
292
+ body: JSON.stringify({ deviceKey }),
293
+ });
294
+ if (!res.ok) {
295
+ const errBody = await res.text();
296
+ throw new Error(`设备注册失败: ${res.status} ${errBody}`);
297
+ }
298
+ const result = await res.json();
299
+ return result.deviceToken;
300
+ }
301
+ // ===== 主启动 =====
302
+ async function start() {
303
+ gateway = new gateway_client_1.GatewayClient(config.openclawGatewayUrl, config.openclawAuthToken);
304
+ (0, commands_1.setGatewayClient)(gateway);
305
+ // 启动本地中继服务器(用于 HTTP API 调用,如 agent 列表等)
306
+ const relay = (0, relay_server_1.createRelayServer)(config, gateway);
307
+ relayServer = relay;
308
+ tunnelMessageHandler = new message_handler_1.MessageHandler(config, relay.sessionManager, gateway);
309
+ relay.server.listen(config.port, '127.0.0.1', () => {
310
+ // 不打印,下面统一打印
311
+ });
312
+ let deviceCfg = readDeviceConfig();
313
+ if (!deviceCfg || !deviceCfg.deviceKey) {
314
+ // 首次使用 → 扫码配对
315
+ try {
316
+ deviceCfg = await startPairing();
317
+ }
318
+ catch (err) {
319
+ console.error('❌', err.message);
320
+ process.exit(1);
321
+ }
322
+ }
323
+ // 注册设备获取 token
324
+ let deviceToken = deviceCfg.deviceToken;
325
+ try {
326
+ deviceToken = await registerDevice(deviceCfg.deviceKey);
327
+ // 更新本地 token
328
+ deviceCfg.deviceToken = deviceToken;
329
+ saveDeviceConfig(deviceCfg);
330
+ }
331
+ catch (err) {
332
+ console.error('⚠️ 设备注册失败:', err.message);
333
+ if (!deviceToken) {
334
+ console.error('❌ 无法连接后端,请检查网络或重新配对');
335
+ process.exit(1);
336
+ }
337
+ }
338
+ // 建立反向 WS 隧道
339
+ connectTunnel(deviceToken);
340
+ console.log('');
341
+ console.log('🔌 =============================================');
342
+ console.log(' OpenClaw VChat Plugin');
343
+ console.log(' =============================================');
344
+ console.log(` 📡 本地端口: http://127.0.0.1:${config.port}`);
345
+ console.log(` 🔗 OpenClaw: ${config.openclawGatewayUrl}`);
346
+ console.log(` 🤖 AI 模型: ${config.openclawModel}`);
347
+ console.log(` 🌐 后端: ${BACKEND_URL}`);
348
+ console.log(` 🔑 设备: 已配对`);
349
+ console.log(' =============================================');
350
+ console.log('');
351
+ }
352
+ start().catch((err) => {
353
+ console.error('❌ 启动失败:', err);
354
+ process.exit(1);
355
+ });
356
+ // 优雅关闭
357
+ const shutdown = () => {
358
+ console.log('\n🛑 正在关闭插件...');
359
+ for (const abort of tunnelAbortByUser.values()) {
360
+ try {
361
+ abort();
362
+ }
363
+ catch { /* ignore */ }
364
+ }
365
+ tunnelAbortByUser.clear();
366
+ if (tunnelWs)
367
+ tunnelWs.close();
368
+ if (relayServer) {
369
+ relayServer.server.close(() => {
370
+ relayServer.sessionManager.close();
371
+ console.log('✅ 插件已关闭');
372
+ process.exit(0);
373
+ });
374
+ }
375
+ else {
376
+ process.exit(0);
377
+ }
378
+ setTimeout(() => { process.exit(1); }, 5000);
379
+ };
380
+ process.on('SIGINT', shutdown);
381
+ process.on('SIGTERM', shutdown);
382
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,31 @@
1
+ import { PluginConfig } from './types';
2
+ /**
3
+ * 媒体文件处理:语音和图片的存储与访问
4
+ */
5
+ export declare class MediaHandler {
6
+ private uploadDir;
7
+ constructor(config: PluginConfig);
8
+ /**
9
+ * 保存语音文件,返回文件路径和访问 URL
10
+ */
11
+ saveVoice(buffer: Buffer, originalName: string): {
12
+ filePath: string;
13
+ url: string;
14
+ };
15
+ /**
16
+ * 保存图片文件,返回文件路径和访问 URL
17
+ */
18
+ saveImage(buffer: Buffer, originalName: string): {
19
+ filePath: string;
20
+ url: string;
21
+ };
22
+ /**
23
+ * 删除媒体文件
24
+ */
25
+ deleteFile(filePath: string): void;
26
+ /**
27
+ * 获取上传目录的绝对路径(用于 Express 静态文件服务)
28
+ */
29
+ getUploadDir(): string;
30
+ }
31
+ //# sourceMappingURL=media-handler.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"media-handler.d.ts","sourceRoot":"","sources":["../src/media-handler.ts"],"names":[],"mappings":"AAGA,OAAO,EAAE,YAAY,EAAE,MAAM,SAAS,CAAC;AAEvC;;GAEG;AACH,qBAAa,YAAY;IACrB,OAAO,CAAC,SAAS,CAAS;gBAEd,MAAM,EAAE,YAAY;IAOhC;;OAEG;IACH,SAAS,CAAC,MAAM,EAAE,MAAM,EAAE,YAAY,EAAE,MAAM,GAAG;QAAE,QAAQ,EAAE,MAAM,CAAC;QAAC,GAAG,EAAE,MAAM,CAAA;KAAE;IAalF;;OAEG;IACH,SAAS,CAAC,MAAM,EAAE,MAAM,EAAE,YAAY,EAAE,MAAM,GAAG;QAAE,QAAQ,EAAE,MAAM,CAAC;QAAC,GAAG,EAAE,MAAM,CAAA;KAAE;IAalF;;OAEG;IACH,UAAU,CAAC,QAAQ,EAAE,MAAM,GAAG,IAAI;IAUlC;;OAEG;IACH,YAAY,IAAI,MAAM;CAGzB"}
@@ -0,0 +1,67 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.MediaHandler = void 0;
7
+ const path_1 = __importDefault(require("path"));
8
+ const fs_1 = __importDefault(require("fs"));
9
+ const uuid_1 = require("uuid");
10
+ /**
11
+ * 媒体文件处理:语音和图片的存储与访问
12
+ */
13
+ class MediaHandler {
14
+ constructor(config) {
15
+ this.uploadDir = config.uploadDir;
16
+ // 确保上传目录存在
17
+ fs_1.default.mkdirSync(path_1.default.join(this.uploadDir, 'voice'), { recursive: true });
18
+ fs_1.default.mkdirSync(path_1.default.join(this.uploadDir, 'image'), { recursive: true });
19
+ }
20
+ /**
21
+ * 保存语音文件,返回文件路径和访问 URL
22
+ */
23
+ saveVoice(buffer, originalName) {
24
+ const ext = path_1.default.extname(originalName) || '.mp3';
25
+ const filename = `${(0, uuid_1.v4)()}${ext}`;
26
+ const filePath = path_1.default.join(this.uploadDir, 'voice', filename);
27
+ fs_1.default.writeFileSync(filePath, buffer);
28
+ return {
29
+ filePath,
30
+ url: `/uploads/voice/${filename}`,
31
+ };
32
+ }
33
+ /**
34
+ * 保存图片文件,返回文件路径和访问 URL
35
+ */
36
+ saveImage(buffer, originalName) {
37
+ const ext = path_1.default.extname(originalName) || '.jpg';
38
+ const filename = `${(0, uuid_1.v4)()}${ext}`;
39
+ const filePath = path_1.default.join(this.uploadDir, 'image', filename);
40
+ fs_1.default.writeFileSync(filePath, buffer);
41
+ return {
42
+ filePath,
43
+ url: `/uploads/image/${filename}`,
44
+ };
45
+ }
46
+ /**
47
+ * 删除媒体文件
48
+ */
49
+ deleteFile(filePath) {
50
+ try {
51
+ if (fs_1.default.existsSync(filePath)) {
52
+ fs_1.default.unlinkSync(filePath);
53
+ }
54
+ }
55
+ catch (err) {
56
+ console.error('[MediaHandler] 删除文件失败:', filePath, err);
57
+ }
58
+ }
59
+ /**
60
+ * 获取上传目录的绝对路径(用于 Express 静态文件服务)
61
+ */
62
+ getUploadDir() {
63
+ return this.uploadDir;
64
+ }
65
+ }
66
+ exports.MediaHandler = MediaHandler;
67
+ //# sourceMappingURL=media-handler.js.map