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.
- package/bin/openclaw-vchat.js +110 -0
- package/dist/commands.d.ts +18 -0
- package/dist/commands.d.ts.map +1 -0
- package/dist/commands.js +509 -0
- package/dist/commands.js.map +1 -0
- package/dist/constants.d.ts +14 -0
- package/dist/constants.d.ts.map +1 -0
- package/dist/constants.js +51 -0
- package/dist/constants.js.map +1 -0
- package/dist/gateway-client.d.ts +43 -0
- package/dist/gateway-client.d.ts.map +1 -0
- package/dist/gateway-client.js +623 -0
- package/dist/gateway-client.js.map +1 -0
- package/dist/group-manager.d.ts +30 -0
- package/dist/group-manager.d.ts.map +1 -0
- package/dist/group-manager.js +107 -0
- package/dist/group-manager.js.map +1 -0
- package/dist/index.d.ts +2 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +382 -0
- package/dist/index.js.map +1 -0
- package/dist/media-handler.d.ts +31 -0
- package/dist/media-handler.d.ts.map +1 -0
- package/dist/media-handler.js +67 -0
- package/dist/media-handler.js.map +1 -0
- package/dist/message-handler.d.ts +52 -0
- package/dist/message-handler.d.ts.map +1 -0
- package/dist/message-handler.js +291 -0
- package/dist/message-handler.js.map +1 -0
- package/dist/relay-server.d.ts +16 -0
- package/dist/relay-server.d.ts.map +1 -0
- package/dist/relay-server.js +877 -0
- package/dist/relay-server.js.map +1 -0
- package/dist/routes/config.routes.d.ts +12 -0
- package/dist/routes/config.routes.d.ts.map +1 -0
- package/dist/routes/config.routes.js +175 -0
- package/dist/routes/config.routes.js.map +1 -0
- package/dist/services/config.service.d.ts +57 -0
- package/dist/services/config.service.d.ts.map +1 -0
- package/dist/services/config.service.js +361 -0
- package/dist/services/config.service.js.map +1 -0
- package/dist/session-key.d.ts +8 -0
- package/dist/session-key.d.ts.map +1 -0
- package/dist/session-key.js +28 -0
- package/dist/session-key.js.map +1 -0
- package/dist/session-manager.d.ts +32 -0
- package/dist/session-manager.d.ts.map +1 -0
- package/dist/session-manager.js +303 -0
- package/dist/session-manager.js.map +1 -0
- package/dist/types.d.ts +81 -0
- package/dist/types.d.ts.map +1 -0
- package/dist/types.js +3 -0
- package/dist/types.js.map +1 -0
- package/nginx-proxy.conf +24 -0
- package/package.json +51 -0
- package/src/commands.ts +499 -0
- package/src/constants.ts +49 -0
- package/src/gateway-client.ts +648 -0
- package/src/group-manager.ts +119 -0
- package/src/index.ts +443 -0
- package/src/media-handler.ts +70 -0
- package/src/message-handler.ts +419 -0
- package/src/relay-server.ts +979 -0
- package/src/routes/config.routes.ts +144 -0
- package/src/services/config.service.ts +398 -0
- package/src/session-key.ts +30 -0
- package/src/session-manager.ts +374 -0
- package/src/types.ts +96 -0
- package/start.sh +5 -0
- 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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"}
|
package/dist/index.d.ts
ADDED
|
@@ -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
|