@sliverp/qqbot 1.3.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.
- package/README.md +231 -0
- package/clawdbot.plugin.json +16 -0
- package/dist/index.d.ts +17 -0
- package/dist/index.js +22 -0
- package/dist/src/api.d.ts +194 -0
- package/dist/src/api.js +555 -0
- package/dist/src/channel.d.ts +3 -0
- package/dist/src/channel.js +146 -0
- package/dist/src/config.d.ts +25 -0
- package/dist/src/config.js +148 -0
- package/dist/src/gateway.d.ts +17 -0
- package/dist/src/gateway.js +722 -0
- package/dist/src/image-server.d.ts +62 -0
- package/dist/src/image-server.js +401 -0
- package/dist/src/known-users.d.ts +100 -0
- package/dist/src/known-users.js +264 -0
- package/dist/src/onboarding.d.ts +10 -0
- package/dist/src/onboarding.js +190 -0
- package/dist/src/outbound.d.ts +149 -0
- package/dist/src/outbound.js +476 -0
- package/dist/src/proactive.d.ts +170 -0
- package/dist/src/proactive.js +398 -0
- package/dist/src/runtime.d.ts +3 -0
- package/dist/src/runtime.js +10 -0
- package/dist/src/session-store.d.ts +49 -0
- package/dist/src/session-store.js +242 -0
- package/dist/src/types.d.ts +116 -0
- package/dist/src/types.js +1 -0
- package/dist/src/utils/image-size.d.ts +51 -0
- package/dist/src/utils/image-size.js +234 -0
- package/dist/src/utils/payload.d.ts +112 -0
- package/dist/src/utils/payload.js +186 -0
- package/index.ts +27 -0
- package/moltbot.plugin.json +16 -0
- package/node_modules/ws/LICENSE +20 -0
- package/node_modules/ws/README.md +548 -0
- package/node_modules/ws/browser.js +8 -0
- package/node_modules/ws/index.js +13 -0
- package/node_modules/ws/lib/buffer-util.js +131 -0
- package/node_modules/ws/lib/constants.js +19 -0
- package/node_modules/ws/lib/event-target.js +292 -0
- package/node_modules/ws/lib/extension.js +203 -0
- package/node_modules/ws/lib/limiter.js +55 -0
- package/node_modules/ws/lib/permessage-deflate.js +528 -0
- package/node_modules/ws/lib/receiver.js +706 -0
- package/node_modules/ws/lib/sender.js +602 -0
- package/node_modules/ws/lib/stream.js +161 -0
- package/node_modules/ws/lib/subprotocol.js +62 -0
- package/node_modules/ws/lib/validation.js +152 -0
- package/node_modules/ws/lib/websocket-server.js +554 -0
- package/node_modules/ws/lib/websocket.js +1393 -0
- package/node_modules/ws/package.json +69 -0
- package/node_modules/ws/wrapper.mjs +8 -0
- package/openclaw.plugin.json +16 -0
- package/package.json +38 -0
- package/qqbot-1.3.0.tgz +0 -0
- package/scripts/proactive-api-server.ts +346 -0
- package/scripts/send-proactive.ts +273 -0
- package/scripts/upgrade.sh +106 -0
- package/skills/qqbot-cron/SKILL.md +490 -0
- package/skills/qqbot-media/SKILL.md +138 -0
- package/src/api.ts +752 -0
- package/src/channel.ts +303 -0
- package/src/config.ts +172 -0
- package/src/gateway.ts +1588 -0
- package/src/image-server.ts +474 -0
- package/src/known-users.ts +358 -0
- package/src/onboarding.ts +254 -0
- package/src/openclaw-plugin-sdk.d.ts +483 -0
- package/src/outbound.ts +571 -0
- package/src/proactive.ts +528 -0
- package/src/runtime.ts +14 -0
- package/src/session-store.ts +292 -0
- package/src/types.ts +123 -0
- package/src/utils/image-size.ts +266 -0
- package/src/utils/payload.ts +265 -0
- package/tsconfig.json +16 -0
- package/upgrade-and-run.sh +89 -0
|
@@ -0,0 +1,116 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* QQ Bot 配置类型
|
|
3
|
+
*/
|
|
4
|
+
export interface QQBotConfig {
|
|
5
|
+
appId: string;
|
|
6
|
+
clientSecret?: string;
|
|
7
|
+
clientSecretFile?: string;
|
|
8
|
+
}
|
|
9
|
+
/**
|
|
10
|
+
* 解析后的 QQ Bot 账户
|
|
11
|
+
*/
|
|
12
|
+
export interface ResolvedQQBotAccount {
|
|
13
|
+
accountId: string;
|
|
14
|
+
name?: string;
|
|
15
|
+
enabled: boolean;
|
|
16
|
+
appId: string;
|
|
17
|
+
clientSecret: string;
|
|
18
|
+
secretSource: "config" | "file" | "env" | "none";
|
|
19
|
+
/** 系统提示词 */
|
|
20
|
+
systemPrompt?: string;
|
|
21
|
+
/** 图床服务器公网地址 */
|
|
22
|
+
imageServerBaseUrl?: string;
|
|
23
|
+
/** 是否支持 markdown 消息(默认 false,需要机器人具备该权限才能启用) */
|
|
24
|
+
markdownSupport?: boolean;
|
|
25
|
+
config: QQBotAccountConfig;
|
|
26
|
+
}
|
|
27
|
+
/**
|
|
28
|
+
* QQ Bot 账户配置
|
|
29
|
+
*/
|
|
30
|
+
export interface QQBotAccountConfig {
|
|
31
|
+
enabled?: boolean;
|
|
32
|
+
name?: string;
|
|
33
|
+
appId?: string;
|
|
34
|
+
clientSecret?: string;
|
|
35
|
+
clientSecretFile?: string;
|
|
36
|
+
dmPolicy?: "open" | "pairing" | "allowlist";
|
|
37
|
+
allowFrom?: string[];
|
|
38
|
+
/** 系统提示词,会添加在用户消息前面 */
|
|
39
|
+
systemPrompt?: string;
|
|
40
|
+
/** 图床服务器公网地址,用于发送图片,例如 http://your-ip:18765 */
|
|
41
|
+
imageServerBaseUrl?: string;
|
|
42
|
+
/** 是否支持 markdown 消息(默认 false,需要机器人具备该权限才能启用) */
|
|
43
|
+
markdownSupport?: boolean;
|
|
44
|
+
}
|
|
45
|
+
/**
|
|
46
|
+
* 富媒体附件
|
|
47
|
+
*/
|
|
48
|
+
export interface MessageAttachment {
|
|
49
|
+
content_type: string;
|
|
50
|
+
filename?: string;
|
|
51
|
+
height?: number;
|
|
52
|
+
width?: number;
|
|
53
|
+
size?: number;
|
|
54
|
+
url: string;
|
|
55
|
+
}
|
|
56
|
+
/**
|
|
57
|
+
* C2C 消息事件
|
|
58
|
+
*/
|
|
59
|
+
export interface C2CMessageEvent {
|
|
60
|
+
author: {
|
|
61
|
+
id: string;
|
|
62
|
+
union_openid: string;
|
|
63
|
+
user_openid: string;
|
|
64
|
+
};
|
|
65
|
+
content: string;
|
|
66
|
+
id: string;
|
|
67
|
+
timestamp: string;
|
|
68
|
+
message_scene?: {
|
|
69
|
+
source: string;
|
|
70
|
+
};
|
|
71
|
+
attachments?: MessageAttachment[];
|
|
72
|
+
}
|
|
73
|
+
/**
|
|
74
|
+
* 频道 AT 消息事件
|
|
75
|
+
*/
|
|
76
|
+
export interface GuildMessageEvent {
|
|
77
|
+
id: string;
|
|
78
|
+
channel_id: string;
|
|
79
|
+
guild_id: string;
|
|
80
|
+
content: string;
|
|
81
|
+
timestamp: string;
|
|
82
|
+
author: {
|
|
83
|
+
id: string;
|
|
84
|
+
username?: string;
|
|
85
|
+
bot?: boolean;
|
|
86
|
+
};
|
|
87
|
+
member?: {
|
|
88
|
+
nick?: string;
|
|
89
|
+
joined_at?: string;
|
|
90
|
+
};
|
|
91
|
+
attachments?: MessageAttachment[];
|
|
92
|
+
}
|
|
93
|
+
/**
|
|
94
|
+
* 群聊 AT 消息事件
|
|
95
|
+
*/
|
|
96
|
+
export interface GroupMessageEvent {
|
|
97
|
+
author: {
|
|
98
|
+
id: string;
|
|
99
|
+
member_openid: string;
|
|
100
|
+
};
|
|
101
|
+
content: string;
|
|
102
|
+
id: string;
|
|
103
|
+
timestamp: string;
|
|
104
|
+
group_id: string;
|
|
105
|
+
group_openid: string;
|
|
106
|
+
attachments?: MessageAttachment[];
|
|
107
|
+
}
|
|
108
|
+
/**
|
|
109
|
+
* WebSocket 事件负载
|
|
110
|
+
*/
|
|
111
|
+
export interface WSPayload {
|
|
112
|
+
op: number;
|
|
113
|
+
d?: unknown;
|
|
114
|
+
s?: number;
|
|
115
|
+
t?: string;
|
|
116
|
+
}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,51 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* 图片尺寸工具
|
|
3
|
+
* 用于获取图片尺寸,生成 QQBot 的 markdown 图片格式
|
|
4
|
+
*
|
|
5
|
+
* QQBot markdown 图片格式: 
|
|
6
|
+
*/
|
|
7
|
+
import { Buffer } from "buffer";
|
|
8
|
+
export interface ImageSize {
|
|
9
|
+
width: number;
|
|
10
|
+
height: number;
|
|
11
|
+
}
|
|
12
|
+
/** 默认图片尺寸(当无法获取时使用) */
|
|
13
|
+
export declare const DEFAULT_IMAGE_SIZE: ImageSize;
|
|
14
|
+
/**
|
|
15
|
+
* 从图片数据 Buffer 解析尺寸
|
|
16
|
+
*/
|
|
17
|
+
export declare function parseImageSize(buffer: Buffer): ImageSize | null;
|
|
18
|
+
/**
|
|
19
|
+
* 从公网 URL 获取图片尺寸
|
|
20
|
+
* 只下载前 64KB 数据,足够解析大部分图片格式的头部
|
|
21
|
+
*/
|
|
22
|
+
export declare function getImageSizeFromUrl(url: string, timeoutMs?: number): Promise<ImageSize | null>;
|
|
23
|
+
/**
|
|
24
|
+
* 从 Base64 Data URL 获取图片尺寸
|
|
25
|
+
*/
|
|
26
|
+
export declare function getImageSizeFromDataUrl(dataUrl: string): ImageSize | null;
|
|
27
|
+
/**
|
|
28
|
+
* 获取图片尺寸(自动判断来源)
|
|
29
|
+
* @param source - 图片 URL 或 Base64 Data URL
|
|
30
|
+
* @returns 图片尺寸,失败返回 null
|
|
31
|
+
*/
|
|
32
|
+
export declare function getImageSize(source: string): Promise<ImageSize | null>;
|
|
33
|
+
/**
|
|
34
|
+
* 生成 QQBot markdown 图片格式
|
|
35
|
+
* 格式: 
|
|
36
|
+
*
|
|
37
|
+
* @param url - 图片 URL
|
|
38
|
+
* @param size - 图片尺寸,如果为 null 则使用默认尺寸
|
|
39
|
+
* @returns QQBot markdown 图片字符串
|
|
40
|
+
*/
|
|
41
|
+
export declare function formatQQBotMarkdownImage(url: string, size: ImageSize | null): string;
|
|
42
|
+
/**
|
|
43
|
+
* 检查 markdown 图片是否已经包含 QQBot 格式的尺寸信息
|
|
44
|
+
* 格式: 
|
|
45
|
+
*/
|
|
46
|
+
export declare function hasQQBotImageSize(markdownImage: string): boolean;
|
|
47
|
+
/**
|
|
48
|
+
* 从已有的 QQBot 格式 markdown 图片中提取尺寸
|
|
49
|
+
* 格式: 
|
|
50
|
+
*/
|
|
51
|
+
export declare function extractQQBotImageSize(markdownImage: string): ImageSize | null;
|
|
@@ -0,0 +1,234 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* 图片尺寸工具
|
|
3
|
+
* 用于获取图片尺寸,生成 QQBot 的 markdown 图片格式
|
|
4
|
+
*
|
|
5
|
+
* QQBot markdown 图片格式: 
|
|
6
|
+
*/
|
|
7
|
+
import { Buffer } from "buffer";
|
|
8
|
+
/** 默认图片尺寸(当无法获取时使用) */
|
|
9
|
+
export const DEFAULT_IMAGE_SIZE = { width: 512, height: 512 };
|
|
10
|
+
/**
|
|
11
|
+
* 从 PNG 文件头解析图片尺寸
|
|
12
|
+
* PNG 文件头结构: 前 8 字节是签名,IHDR 块从第 8 字节开始
|
|
13
|
+
* IHDR 块: 长度(4) + 类型(4, "IHDR") + 宽度(4) + 高度(4) + ...
|
|
14
|
+
*/
|
|
15
|
+
function parsePngSize(buffer) {
|
|
16
|
+
// PNG 签名: 89 50 4E 47 0D 0A 1A 0A
|
|
17
|
+
if (buffer.length < 24)
|
|
18
|
+
return null;
|
|
19
|
+
if (buffer[0] !== 0x89 || buffer[1] !== 0x50 || buffer[2] !== 0x4E || buffer[3] !== 0x47) {
|
|
20
|
+
return null;
|
|
21
|
+
}
|
|
22
|
+
// IHDR 块从第 8 字节开始,宽度在第 16-19 字节,高度在第 20-23 字节
|
|
23
|
+
const width = buffer.readUInt32BE(16);
|
|
24
|
+
const height = buffer.readUInt32BE(20);
|
|
25
|
+
return { width, height };
|
|
26
|
+
}
|
|
27
|
+
/**
|
|
28
|
+
* 从 JPEG 文件解析图片尺寸
|
|
29
|
+
* JPEG 尺寸在 SOF0/SOF2 块中
|
|
30
|
+
*/
|
|
31
|
+
function parseJpegSize(buffer) {
|
|
32
|
+
// JPEG 签名: FF D8 FF
|
|
33
|
+
if (buffer.length < 4)
|
|
34
|
+
return null;
|
|
35
|
+
if (buffer[0] !== 0xFF || buffer[1] !== 0xD8) {
|
|
36
|
+
return null;
|
|
37
|
+
}
|
|
38
|
+
let offset = 2;
|
|
39
|
+
while (offset < buffer.length - 9) {
|
|
40
|
+
if (buffer[offset] !== 0xFF) {
|
|
41
|
+
offset++;
|
|
42
|
+
continue;
|
|
43
|
+
}
|
|
44
|
+
const marker = buffer[offset + 1];
|
|
45
|
+
// SOF0 (0xC0) 或 SOF2 (0xC2) 包含图片尺寸
|
|
46
|
+
if (marker === 0xC0 || marker === 0xC2) {
|
|
47
|
+
// 格式: FF C0 长度(2) 精度(1) 高度(2) 宽度(2)
|
|
48
|
+
if (offset + 9 <= buffer.length) {
|
|
49
|
+
const height = buffer.readUInt16BE(offset + 5);
|
|
50
|
+
const width = buffer.readUInt16BE(offset + 7);
|
|
51
|
+
return { width, height };
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
// 跳过当前块
|
|
55
|
+
if (offset + 3 < buffer.length) {
|
|
56
|
+
const blockLength = buffer.readUInt16BE(offset + 2);
|
|
57
|
+
offset += 2 + blockLength;
|
|
58
|
+
}
|
|
59
|
+
else {
|
|
60
|
+
break;
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
return null;
|
|
64
|
+
}
|
|
65
|
+
/**
|
|
66
|
+
* 从 GIF 文件头解析图片尺寸
|
|
67
|
+
* GIF 文件头: GIF87a 或 GIF89a (6字节) + 宽度(2) + 高度(2)
|
|
68
|
+
*/
|
|
69
|
+
function parseGifSize(buffer) {
|
|
70
|
+
if (buffer.length < 10)
|
|
71
|
+
return null;
|
|
72
|
+
const signature = buffer.toString("ascii", 0, 6);
|
|
73
|
+
if (signature !== "GIF87a" && signature !== "GIF89a") {
|
|
74
|
+
return null;
|
|
75
|
+
}
|
|
76
|
+
const width = buffer.readUInt16LE(6);
|
|
77
|
+
const height = buffer.readUInt16LE(8);
|
|
78
|
+
return { width, height };
|
|
79
|
+
}
|
|
80
|
+
/**
|
|
81
|
+
* 从 WebP 文件解析图片尺寸
|
|
82
|
+
* WebP 文件头: RIFF(4) + 文件大小(4) + WEBP(4) + VP8/VP8L/VP8X(4) + ...
|
|
83
|
+
*/
|
|
84
|
+
function parseWebpSize(buffer) {
|
|
85
|
+
if (buffer.length < 30)
|
|
86
|
+
return null;
|
|
87
|
+
// 检查 RIFF 和 WEBP 签名
|
|
88
|
+
const riff = buffer.toString("ascii", 0, 4);
|
|
89
|
+
const webp = buffer.toString("ascii", 8, 12);
|
|
90
|
+
if (riff !== "RIFF" || webp !== "WEBP") {
|
|
91
|
+
return null;
|
|
92
|
+
}
|
|
93
|
+
const chunkType = buffer.toString("ascii", 12, 16);
|
|
94
|
+
// VP8 (有损压缩)
|
|
95
|
+
if (chunkType === "VP8 ") {
|
|
96
|
+
// VP8 帧头从第 23 字节开始,检查签名 9D 01 2A
|
|
97
|
+
if (buffer.length >= 30 && buffer[23] === 0x9D && buffer[24] === 0x01 && buffer[25] === 0x2A) {
|
|
98
|
+
const width = buffer.readUInt16LE(26) & 0x3FFF;
|
|
99
|
+
const height = buffer.readUInt16LE(28) & 0x3FFF;
|
|
100
|
+
return { width, height };
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
// VP8L (无损压缩)
|
|
104
|
+
if (chunkType === "VP8L") {
|
|
105
|
+
// VP8L 签名: 0x2F
|
|
106
|
+
if (buffer.length >= 25 && buffer[20] === 0x2F) {
|
|
107
|
+
const bits = buffer.readUInt32LE(21);
|
|
108
|
+
const width = (bits & 0x3FFF) + 1;
|
|
109
|
+
const height = ((bits >> 14) & 0x3FFF) + 1;
|
|
110
|
+
return { width, height };
|
|
111
|
+
}
|
|
112
|
+
}
|
|
113
|
+
// VP8X (扩展格式)
|
|
114
|
+
if (chunkType === "VP8X") {
|
|
115
|
+
if (buffer.length >= 30) {
|
|
116
|
+
// 宽度和高度在第 24-26 和 27-29 字节(24位小端)
|
|
117
|
+
const width = (buffer[24] | (buffer[25] << 8) | (buffer[26] << 16)) + 1;
|
|
118
|
+
const height = (buffer[27] | (buffer[28] << 8) | (buffer[29] << 16)) + 1;
|
|
119
|
+
return { width, height };
|
|
120
|
+
}
|
|
121
|
+
}
|
|
122
|
+
return null;
|
|
123
|
+
}
|
|
124
|
+
/**
|
|
125
|
+
* 从图片数据 Buffer 解析尺寸
|
|
126
|
+
*/
|
|
127
|
+
export function parseImageSize(buffer) {
|
|
128
|
+
// 尝试各种格式
|
|
129
|
+
return parsePngSize(buffer)
|
|
130
|
+
?? parseJpegSize(buffer)
|
|
131
|
+
?? parseGifSize(buffer)
|
|
132
|
+
?? parseWebpSize(buffer);
|
|
133
|
+
}
|
|
134
|
+
/**
|
|
135
|
+
* 从公网 URL 获取图片尺寸
|
|
136
|
+
* 只下载前 64KB 数据,足够解析大部分图片格式的头部
|
|
137
|
+
*/
|
|
138
|
+
export async function getImageSizeFromUrl(url, timeoutMs = 5000) {
|
|
139
|
+
try {
|
|
140
|
+
const controller = new AbortController();
|
|
141
|
+
const timeoutId = setTimeout(() => controller.abort(), timeoutMs);
|
|
142
|
+
// 使用 Range 请求只获取前 64KB
|
|
143
|
+
const response = await fetch(url, {
|
|
144
|
+
signal: controller.signal,
|
|
145
|
+
headers: {
|
|
146
|
+
"Range": "bytes=0-65535",
|
|
147
|
+
"User-Agent": "QQBot-Image-Size-Detector/1.0",
|
|
148
|
+
},
|
|
149
|
+
});
|
|
150
|
+
clearTimeout(timeoutId);
|
|
151
|
+
if (!response.ok && response.status !== 206) {
|
|
152
|
+
console.log(`[image-size] Failed to fetch ${url}: ${response.status}`);
|
|
153
|
+
return null;
|
|
154
|
+
}
|
|
155
|
+
const arrayBuffer = await response.arrayBuffer();
|
|
156
|
+
const buffer = Buffer.from(arrayBuffer);
|
|
157
|
+
const size = parseImageSize(buffer);
|
|
158
|
+
if (size) {
|
|
159
|
+
console.log(`[image-size] Got size from URL: ${size.width}x${size.height} - ${url.slice(0, 60)}...`);
|
|
160
|
+
}
|
|
161
|
+
return size;
|
|
162
|
+
}
|
|
163
|
+
catch (err) {
|
|
164
|
+
console.log(`[image-size] Error fetching ${url.slice(0, 60)}...: ${err}`);
|
|
165
|
+
return null;
|
|
166
|
+
}
|
|
167
|
+
}
|
|
168
|
+
/**
|
|
169
|
+
* 从 Base64 Data URL 获取图片尺寸
|
|
170
|
+
*/
|
|
171
|
+
export function getImageSizeFromDataUrl(dataUrl) {
|
|
172
|
+
try {
|
|
173
|
+
// 格式: 
|
|
174
|
+
const matches = dataUrl.match(/^data:image\/[^;]+;base64,(.+)$/);
|
|
175
|
+
if (!matches) {
|
|
176
|
+
return null;
|
|
177
|
+
}
|
|
178
|
+
const base64Data = matches[1];
|
|
179
|
+
const buffer = Buffer.from(base64Data, "base64");
|
|
180
|
+
const size = parseImageSize(buffer);
|
|
181
|
+
if (size) {
|
|
182
|
+
console.log(`[image-size] Got size from Base64: ${size.width}x${size.height}`);
|
|
183
|
+
}
|
|
184
|
+
return size;
|
|
185
|
+
}
|
|
186
|
+
catch (err) {
|
|
187
|
+
console.log(`[image-size] Error parsing Base64: ${err}`);
|
|
188
|
+
return null;
|
|
189
|
+
}
|
|
190
|
+
}
|
|
191
|
+
/**
|
|
192
|
+
* 获取图片尺寸(自动判断来源)
|
|
193
|
+
* @param source - 图片 URL 或 Base64 Data URL
|
|
194
|
+
* @returns 图片尺寸,失败返回 null
|
|
195
|
+
*/
|
|
196
|
+
export async function getImageSize(source) {
|
|
197
|
+
if (source.startsWith("data:")) {
|
|
198
|
+
return getImageSizeFromDataUrl(source);
|
|
199
|
+
}
|
|
200
|
+
if (source.startsWith("http://") || source.startsWith("https://")) {
|
|
201
|
+
return getImageSizeFromUrl(source);
|
|
202
|
+
}
|
|
203
|
+
return null;
|
|
204
|
+
}
|
|
205
|
+
/**
|
|
206
|
+
* 生成 QQBot markdown 图片格式
|
|
207
|
+
* 格式: 
|
|
208
|
+
*
|
|
209
|
+
* @param url - 图片 URL
|
|
210
|
+
* @param size - 图片尺寸,如果为 null 则使用默认尺寸
|
|
211
|
+
* @returns QQBot markdown 图片字符串
|
|
212
|
+
*/
|
|
213
|
+
export function formatQQBotMarkdownImage(url, size) {
|
|
214
|
+
const { width, height } = size ?? DEFAULT_IMAGE_SIZE;
|
|
215
|
+
return ``;
|
|
216
|
+
}
|
|
217
|
+
/**
|
|
218
|
+
* 检查 markdown 图片是否已经包含 QQBot 格式的尺寸信息
|
|
219
|
+
* 格式: 
|
|
220
|
+
*/
|
|
221
|
+
export function hasQQBotImageSize(markdownImage) {
|
|
222
|
+
return /!\[#\d+px\s+#\d+px\]/.test(markdownImage);
|
|
223
|
+
}
|
|
224
|
+
/**
|
|
225
|
+
* 从已有的 QQBot 格式 markdown 图片中提取尺寸
|
|
226
|
+
* 格式: 
|
|
227
|
+
*/
|
|
228
|
+
export function extractQQBotImageSize(markdownImage) {
|
|
229
|
+
const match = markdownImage.match(/!\[#(\d+)px\s+#(\d+)px\]/);
|
|
230
|
+
if (match) {
|
|
231
|
+
return { width: parseInt(match[1], 10), height: parseInt(match[2], 10) };
|
|
232
|
+
}
|
|
233
|
+
return null;
|
|
234
|
+
}
|
|
@@ -0,0 +1,112 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* QQBot 结构化消息载荷工具
|
|
3
|
+
*
|
|
4
|
+
* 用于处理 AI 输出的结构化消息载荷,包括:
|
|
5
|
+
* - 定时提醒载荷 (cron_reminder)
|
|
6
|
+
* - 媒体消息载荷 (media)
|
|
7
|
+
*/
|
|
8
|
+
/**
|
|
9
|
+
* 定时提醒载荷
|
|
10
|
+
*/
|
|
11
|
+
export interface CronReminderPayload {
|
|
12
|
+
type: 'cron_reminder';
|
|
13
|
+
/** 提醒内容 */
|
|
14
|
+
content: string;
|
|
15
|
+
/** 目标类型:c2c (私聊) 或 group (群聊) */
|
|
16
|
+
targetType: 'c2c' | 'group';
|
|
17
|
+
/** 目标地址:user_openid 或 group_openid */
|
|
18
|
+
targetAddress: string;
|
|
19
|
+
/** 原始消息 ID(可选) */
|
|
20
|
+
originalMessageId?: string;
|
|
21
|
+
}
|
|
22
|
+
/**
|
|
23
|
+
* 媒体消息载荷
|
|
24
|
+
*/
|
|
25
|
+
export interface MediaPayload {
|
|
26
|
+
type: 'media';
|
|
27
|
+
/** 媒体类型:image, audio, video */
|
|
28
|
+
mediaType: 'image' | 'audio' | 'video';
|
|
29
|
+
/** 来源类型:url 或 file */
|
|
30
|
+
source: 'url' | 'file';
|
|
31
|
+
/** 媒体路径或 URL */
|
|
32
|
+
path: string;
|
|
33
|
+
/** 媒体描述(可选) */
|
|
34
|
+
caption?: string;
|
|
35
|
+
}
|
|
36
|
+
/**
|
|
37
|
+
* QQBot 载荷联合类型
|
|
38
|
+
*/
|
|
39
|
+
export type QQBotPayload = CronReminderPayload | MediaPayload;
|
|
40
|
+
/**
|
|
41
|
+
* 解析结果
|
|
42
|
+
*/
|
|
43
|
+
export interface ParseResult {
|
|
44
|
+
/** 是否为结构化载荷 */
|
|
45
|
+
isPayload: boolean;
|
|
46
|
+
/** 解析后的载荷对象(如果是结构化载荷) */
|
|
47
|
+
payload?: QQBotPayload;
|
|
48
|
+
/** 原始文本(如果不是结构化载荷) */
|
|
49
|
+
text?: string;
|
|
50
|
+
/** 解析错误信息(如果解析失败) */
|
|
51
|
+
error?: string;
|
|
52
|
+
}
|
|
53
|
+
/**
|
|
54
|
+
* 解析 AI 输出的结构化载荷
|
|
55
|
+
*
|
|
56
|
+
* 检测消息是否以 QQBOT_PAYLOAD: 前缀开头,如果是则提取并解析 JSON
|
|
57
|
+
*
|
|
58
|
+
* @param text AI 输出的原始文本
|
|
59
|
+
* @returns 解析结果
|
|
60
|
+
*
|
|
61
|
+
* @example
|
|
62
|
+
* const result = parseQQBotPayload('QQBOT_PAYLOAD:\n{"type": "media", "mediaType": "image", ...}');
|
|
63
|
+
* if (result.isPayload && result.payload) {
|
|
64
|
+
* // 处理结构化载荷
|
|
65
|
+
* }
|
|
66
|
+
*/
|
|
67
|
+
export declare function parseQQBotPayload(text: string): ParseResult;
|
|
68
|
+
/**
|
|
69
|
+
* 将定时提醒载荷编码为 Cron 消息格式
|
|
70
|
+
*
|
|
71
|
+
* 将 JSON 编码为 Base64,并添加 QQBOT_CRON: 前缀
|
|
72
|
+
*
|
|
73
|
+
* @param payload 定时提醒载荷
|
|
74
|
+
* @returns 编码后的消息字符串,格式为 QQBOT_CRON:{base64}
|
|
75
|
+
*
|
|
76
|
+
* @example
|
|
77
|
+
* const message = encodePayloadForCron({
|
|
78
|
+
* type: 'cron_reminder',
|
|
79
|
+
* content: '喝水时间到!',
|
|
80
|
+
* targetType: 'c2c',
|
|
81
|
+
* targetAddress: 'user_openid_xxx'
|
|
82
|
+
* });
|
|
83
|
+
* // 返回: QQBOT_CRON:eyJ0eXBlIjoiY3Jvbl9yZW1pbmRlciIs...
|
|
84
|
+
*/
|
|
85
|
+
export declare function encodePayloadForCron(payload: CronReminderPayload): string;
|
|
86
|
+
/**
|
|
87
|
+
* 解码 Cron 消息中的载荷
|
|
88
|
+
*
|
|
89
|
+
* 检测 QQBOT_CRON: 前缀,解码 Base64 并解析 JSON
|
|
90
|
+
*
|
|
91
|
+
* @param message Cron 触发时收到的消息
|
|
92
|
+
* @returns 解码结果,包含是否为 Cron 载荷、解析后的载荷对象或错误信息
|
|
93
|
+
*
|
|
94
|
+
* @example
|
|
95
|
+
* const result = decodeCronPayload('QQBOT_CRON:eyJ0eXBlIjoiY3Jvbl9yZW1pbmRlciIs...');
|
|
96
|
+
* if (result.isCronPayload && result.payload) {
|
|
97
|
+
* // 处理定时提醒
|
|
98
|
+
* }
|
|
99
|
+
*/
|
|
100
|
+
export declare function decodeCronPayload(message: string): {
|
|
101
|
+
isCronPayload: boolean;
|
|
102
|
+
payload?: CronReminderPayload;
|
|
103
|
+
error?: string;
|
|
104
|
+
};
|
|
105
|
+
/**
|
|
106
|
+
* 判断载荷是否为定时提醒类型
|
|
107
|
+
*/
|
|
108
|
+
export declare function isCronReminderPayload(payload: QQBotPayload): payload is CronReminderPayload;
|
|
109
|
+
/**
|
|
110
|
+
* 判断载荷是否为媒体消息类型
|
|
111
|
+
*/
|
|
112
|
+
export declare function isMediaPayload(payload: QQBotPayload): payload is MediaPayload;
|
|
@@ -0,0 +1,186 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* QQBot 结构化消息载荷工具
|
|
3
|
+
*
|
|
4
|
+
* 用于处理 AI 输出的结构化消息载荷,包括:
|
|
5
|
+
* - 定时提醒载荷 (cron_reminder)
|
|
6
|
+
* - 媒体消息载荷 (media)
|
|
7
|
+
*/
|
|
8
|
+
// ============================================
|
|
9
|
+
// 常量定义
|
|
10
|
+
// ============================================
|
|
11
|
+
/** AI 输出的结构化载荷前缀 */
|
|
12
|
+
const PAYLOAD_PREFIX = 'QQBOT_PAYLOAD:';
|
|
13
|
+
/** Cron 消息存储的前缀 */
|
|
14
|
+
const CRON_PREFIX = 'QQBOT_CRON:';
|
|
15
|
+
// ============================================
|
|
16
|
+
// 解析函数
|
|
17
|
+
// ============================================
|
|
18
|
+
/**
|
|
19
|
+
* 解析 AI 输出的结构化载荷
|
|
20
|
+
*
|
|
21
|
+
* 检测消息是否以 QQBOT_PAYLOAD: 前缀开头,如果是则提取并解析 JSON
|
|
22
|
+
*
|
|
23
|
+
* @param text AI 输出的原始文本
|
|
24
|
+
* @returns 解析结果
|
|
25
|
+
*
|
|
26
|
+
* @example
|
|
27
|
+
* const result = parseQQBotPayload('QQBOT_PAYLOAD:\n{"type": "media", "mediaType": "image", ...}');
|
|
28
|
+
* if (result.isPayload && result.payload) {
|
|
29
|
+
* // 处理结构化载荷
|
|
30
|
+
* }
|
|
31
|
+
*/
|
|
32
|
+
export function parseQQBotPayload(text) {
|
|
33
|
+
const trimmedText = text.trim();
|
|
34
|
+
// 检查是否以 QQBOT_PAYLOAD: 开头
|
|
35
|
+
if (!trimmedText.startsWith(PAYLOAD_PREFIX)) {
|
|
36
|
+
return {
|
|
37
|
+
isPayload: false,
|
|
38
|
+
text: text
|
|
39
|
+
};
|
|
40
|
+
}
|
|
41
|
+
// 提取 JSON 内容(去掉前缀)
|
|
42
|
+
const jsonContent = trimmedText.slice(PAYLOAD_PREFIX.length).trim();
|
|
43
|
+
if (!jsonContent) {
|
|
44
|
+
return {
|
|
45
|
+
isPayload: true,
|
|
46
|
+
error: '载荷内容为空'
|
|
47
|
+
};
|
|
48
|
+
}
|
|
49
|
+
try {
|
|
50
|
+
const payload = JSON.parse(jsonContent);
|
|
51
|
+
// 验证必要字段
|
|
52
|
+
if (!payload.type) {
|
|
53
|
+
return {
|
|
54
|
+
isPayload: true,
|
|
55
|
+
error: '载荷缺少 type 字段'
|
|
56
|
+
};
|
|
57
|
+
}
|
|
58
|
+
// 根据 type 进行额外验证
|
|
59
|
+
if (payload.type === 'cron_reminder') {
|
|
60
|
+
if (!payload.content || !payload.targetType || !payload.targetAddress) {
|
|
61
|
+
return {
|
|
62
|
+
isPayload: true,
|
|
63
|
+
error: 'cron_reminder 载荷缺少必要字段 (content, targetType, targetAddress)'
|
|
64
|
+
};
|
|
65
|
+
}
|
|
66
|
+
}
|
|
67
|
+
else if (payload.type === 'media') {
|
|
68
|
+
if (!payload.mediaType || !payload.source || !payload.path) {
|
|
69
|
+
return {
|
|
70
|
+
isPayload: true,
|
|
71
|
+
error: 'media 载荷缺少必要字段 (mediaType, source, path)'
|
|
72
|
+
};
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
return {
|
|
76
|
+
isPayload: true,
|
|
77
|
+
payload
|
|
78
|
+
};
|
|
79
|
+
}
|
|
80
|
+
catch (e) {
|
|
81
|
+
return {
|
|
82
|
+
isPayload: true,
|
|
83
|
+
error: `JSON 解析失败: ${e instanceof Error ? e.message : String(e)}`
|
|
84
|
+
};
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
// ============================================
|
|
88
|
+
// Cron 编码/解码函数
|
|
89
|
+
// ============================================
|
|
90
|
+
/**
|
|
91
|
+
* 将定时提醒载荷编码为 Cron 消息格式
|
|
92
|
+
*
|
|
93
|
+
* 将 JSON 编码为 Base64,并添加 QQBOT_CRON: 前缀
|
|
94
|
+
*
|
|
95
|
+
* @param payload 定时提醒载荷
|
|
96
|
+
* @returns 编码后的消息字符串,格式为 QQBOT_CRON:{base64}
|
|
97
|
+
*
|
|
98
|
+
* @example
|
|
99
|
+
* const message = encodePayloadForCron({
|
|
100
|
+
* type: 'cron_reminder',
|
|
101
|
+
* content: '喝水时间到!',
|
|
102
|
+
* targetType: 'c2c',
|
|
103
|
+
* targetAddress: 'user_openid_xxx'
|
|
104
|
+
* });
|
|
105
|
+
* // 返回: QQBOT_CRON:eyJ0eXBlIjoiY3Jvbl9yZW1pbmRlciIs...
|
|
106
|
+
*/
|
|
107
|
+
export function encodePayloadForCron(payload) {
|
|
108
|
+
const jsonString = JSON.stringify(payload);
|
|
109
|
+
const base64 = Buffer.from(jsonString, 'utf-8').toString('base64');
|
|
110
|
+
return `${CRON_PREFIX}${base64}`;
|
|
111
|
+
}
|
|
112
|
+
/**
|
|
113
|
+
* 解码 Cron 消息中的载荷
|
|
114
|
+
*
|
|
115
|
+
* 检测 QQBOT_CRON: 前缀,解码 Base64 并解析 JSON
|
|
116
|
+
*
|
|
117
|
+
* @param message Cron 触发时收到的消息
|
|
118
|
+
* @returns 解码结果,包含是否为 Cron 载荷、解析后的载荷对象或错误信息
|
|
119
|
+
*
|
|
120
|
+
* @example
|
|
121
|
+
* const result = decodeCronPayload('QQBOT_CRON:eyJ0eXBlIjoiY3Jvbl9yZW1pbmRlciIs...');
|
|
122
|
+
* if (result.isCronPayload && result.payload) {
|
|
123
|
+
* // 处理定时提醒
|
|
124
|
+
* }
|
|
125
|
+
*/
|
|
126
|
+
export function decodeCronPayload(message) {
|
|
127
|
+
const trimmedMessage = message.trim();
|
|
128
|
+
// 检查是否以 QQBOT_CRON: 开头
|
|
129
|
+
if (!trimmedMessage.startsWith(CRON_PREFIX)) {
|
|
130
|
+
return {
|
|
131
|
+
isCronPayload: false
|
|
132
|
+
};
|
|
133
|
+
}
|
|
134
|
+
// 提取 Base64 内容
|
|
135
|
+
const base64Content = trimmedMessage.slice(CRON_PREFIX.length);
|
|
136
|
+
if (!base64Content) {
|
|
137
|
+
return {
|
|
138
|
+
isCronPayload: true,
|
|
139
|
+
error: 'Cron 载荷内容为空'
|
|
140
|
+
};
|
|
141
|
+
}
|
|
142
|
+
try {
|
|
143
|
+
// Base64 解码
|
|
144
|
+
const jsonString = Buffer.from(base64Content, 'base64').toString('utf-8');
|
|
145
|
+
const payload = JSON.parse(jsonString);
|
|
146
|
+
// 验证类型
|
|
147
|
+
if (payload.type !== 'cron_reminder') {
|
|
148
|
+
return {
|
|
149
|
+
isCronPayload: true,
|
|
150
|
+
error: `期望 type 为 cron_reminder,实际为 ${payload.type}`
|
|
151
|
+
};
|
|
152
|
+
}
|
|
153
|
+
// 验证必要字段
|
|
154
|
+
if (!payload.content || !payload.targetType || !payload.targetAddress) {
|
|
155
|
+
return {
|
|
156
|
+
isCronPayload: true,
|
|
157
|
+
error: 'Cron 载荷缺少必要字段'
|
|
158
|
+
};
|
|
159
|
+
}
|
|
160
|
+
return {
|
|
161
|
+
isCronPayload: true,
|
|
162
|
+
payload
|
|
163
|
+
};
|
|
164
|
+
}
|
|
165
|
+
catch (e) {
|
|
166
|
+
return {
|
|
167
|
+
isCronPayload: true,
|
|
168
|
+
error: `Cron 载荷解码失败: ${e instanceof Error ? e.message : String(e)}`
|
|
169
|
+
};
|
|
170
|
+
}
|
|
171
|
+
}
|
|
172
|
+
// ============================================
|
|
173
|
+
// 辅助函数
|
|
174
|
+
// ============================================
|
|
175
|
+
/**
|
|
176
|
+
* 判断载荷是否为定时提醒类型
|
|
177
|
+
*/
|
|
178
|
+
export function isCronReminderPayload(payload) {
|
|
179
|
+
return payload.type === 'cron_reminder';
|
|
180
|
+
}
|
|
181
|
+
/**
|
|
182
|
+
* 判断载荷是否为媒体消息类型
|
|
183
|
+
*/
|
|
184
|
+
export function isMediaPayload(payload) {
|
|
185
|
+
return payload.type === 'media';
|
|
186
|
+
}
|