foliko 1.0.75 → 1.0.76
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/.claude/settings.local.json +159 -157
- package/cli/bin/foliko.js +12 -12
- package/cli/src/commands/chat.js +143 -143
- package/cli/src/commands/list.js +93 -93
- package/cli/src/index.js +75 -75
- package/cli/src/ui/chat-ui.js +201 -201
- package/cli/src/utils/ansi.js +40 -40
- package/cli/src/utils/markdown.js +292 -292
- package/examples/ambient-example.js +194 -194
- package/examples/basic.js +115 -115
- package/examples/bootstrap.js +121 -121
- package/examples/mcp-example.js +56 -56
- package/examples/skill-example.js +49 -49
- package/examples/test-chat.js +137 -137
- package/examples/test-mcp.js +85 -85
- package/examples/test-reload.js +59 -59
- package/examples/test-telegram.js +50 -50
- package/examples/test-tg-bot.js +45 -45
- package/examples/test-tg-simple.js +47 -47
- package/examples/test-tg.js +62 -62
- package/examples/test-think.js +43 -43
- package/examples/test-web-plugin.js +103 -103
- package/examples/test-weixin-feishu.js +103 -103
- package/examples/workflow.js +158 -158
- package/package.json +1 -1
- package/plugins/ai-plugin.js +102 -102
- package/plugins/ambient-agent/EventWatcher.js +113 -113
- package/plugins/ambient-agent/ExplorerLoop.js +640 -640
- package/plugins/ambient-agent/GoalManager.js +197 -197
- package/plugins/ambient-agent/Reflector.js +95 -95
- package/plugins/ambient-agent/StateStore.js +90 -90
- package/plugins/ambient-agent/constants.js +101 -101
- package/plugins/ambient-agent/index.js +579 -579
- package/plugins/audit-plugin.js +187 -187
- package/plugins/default-plugins.js +662 -662
- package/plugins/email/constants.js +64 -64
- package/plugins/email/handlers.js +461 -461
- package/plugins/email/index.js +278 -278
- package/plugins/email/monitor.js +269 -269
- package/plugins/email/parser.js +138 -138
- package/plugins/email/reply.js +151 -151
- package/plugins/email/utils.js +124 -124
- package/plugins/feishu-plugin.js +481 -481
- package/plugins/file-system-plugin.js +826 -826
- package/plugins/install-plugin.js +199 -199
- package/plugins/python-executor-plugin.js +367 -367
- package/plugins/python-plugin-loader.js +481 -481
- package/plugins/rules-plugin.js +294 -294
- package/plugins/scheduler-plugin.js +691 -691
- package/plugins/session-plugin.js +369 -369
- package/plugins/shell-executor-plugin.js +197 -197
- package/plugins/storage-plugin.js +240 -240
- package/plugins/subagent-plugin.js +845 -845
- package/plugins/telegram-plugin.js +482 -482
- package/plugins/think-plugin.js +345 -345
- package/plugins/tools-plugin.js +196 -196
- package/plugins/web-plugin.js +606 -606
- package/plugins/weixin-plugin.js +545 -545
- package/src/capabilities/index.js +11 -11
- package/src/capabilities/skill-manager.js +609 -609
- package/src/capabilities/workflow-engine.js +1109 -1109
- package/src/core/agent-chat.js +882 -882
- package/src/core/agent.js +892 -892
- package/src/core/framework.js +465 -465
- package/src/core/index.js +19 -19
- package/src/core/plugin-base.js +219 -219
- package/src/core/plugin-manager.js +863 -863
- package/src/core/provider.js +114 -114
- package/src/core/sub-agent-config.js +264 -264
- package/src/core/system-prompt-builder.js +120 -120
- package/src/core/tool-registry.js +517 -517
- package/src/core/tool-router.js +297 -297
- package/src/executors/executor-base.js +58 -58
- package/src/executors/mcp-executor.js +741 -741
- package/src/index.js +25 -25
- package/src/utils/circuit-breaker.js +301 -301
- package/src/utils/error-boundary.js +363 -363
- package/src/utils/error.js +374 -374
- package/src/utils/event-emitter.js +97 -97
- package/src/utils/id.js +133 -133
- package/src/utils/index.js +217 -217
- package/src/utils/logger.js +181 -181
- package/src/utils/plugin-helpers.js +90 -90
- package/src/utils/retry.js +122 -122
- package/src/utils/sandbox.js +292 -292
- package/test/tool-registry-validation.test.js +218 -218
- package/website/script.js +136 -136
- package/foliko-1.0.75.tgz +0 -0
package/plugins/weixin-plugin.js
CHANGED
|
@@ -1,546 +1,546 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* WeChat 插件
|
|
3
|
-
* 使用 @chnak/weixin-bot 实现微信对话
|
|
4
|
-
*
|
|
5
|
-
* 配置:
|
|
6
|
-
* - forceLogin: 是否强制重新扫码登录
|
|
7
|
-
* - qrcodeTerminal: 是否在终端渲染二维码 (默认 true)
|
|
8
|
-
*/
|
|
9
|
-
|
|
10
|
-
const { Plugin } = require('../src/core/plugin-base')
|
|
11
|
-
const { logger } = require('../src/utils/logger')
|
|
12
|
-
const log = logger.child('WeChat')
|
|
13
|
-
const {cleanResponse} =require('../src/utils')
|
|
14
|
-
const removeMarkdown = require('remove-markdown');
|
|
15
|
-
const { z } = require('zod')
|
|
16
|
-
const { WeixinBot } = require('@chnak/weixin-bot')
|
|
17
|
-
|
|
18
|
-
class WeixinPlugin extends Plugin {
|
|
19
|
-
constructor(config = {}) {
|
|
20
|
-
super()
|
|
21
|
-
this.name = 'weixin'
|
|
22
|
-
this.version = '1.0.0'
|
|
23
|
-
this.description = '微信对话插件,使用微信网页账号进行对话'
|
|
24
|
-
this.priority = 80
|
|
25
|
-
// 默认不启用,需要在 plugins.json 中设置 enabled: true
|
|
26
|
-
this.enabled = false
|
|
27
|
-
this.path=`.agent/data`
|
|
28
|
-
this.systemPrompt=`你是一个微信助手。
|
|
29
|
-
|
|
30
|
-
**重要:** 子Agent 匹配规则必须遵守:
|
|
31
|
-
- 根据【子 Agent 匹配表】,将任务委托给最匹配的子Agent处理
|
|
32
|
-
- 使用 subagent_call 工具并指定 agentName 来委托任务
|
|
33
|
-
- 只有当没有匹配的子Agent时,才直接调用工具
|
|
34
|
-
|
|
35
|
-
**命令执行规范:**
|
|
36
|
-
- 执行 npx skills add / npx skills remove / npx skills list 命令时,必须自动添加参数:-a openclaw -y
|
|
37
|
-
- 禁止添加 -g 参数
|
|
38
|
-
- 例如:npx skills add xxx 应该执行为 npx skills add xxx -a openclaw -y`
|
|
39
|
-
|
|
40
|
-
this.config = {
|
|
41
|
-
forceLogin: config.forceLogin || process.env.WEIXIN_FORCE_LOGIN === 'true',
|
|
42
|
-
qrcodeTerminal: config.qrcodeTerminal !== false && process.env.WEIXIN_QRCODE_TERMINAL !== 'false',
|
|
43
|
-
allowedUsers: config.allowedUsers || []
|
|
44
|
-
}
|
|
45
|
-
|
|
46
|
-
this._framework = null
|
|
47
|
-
this._bot = null
|
|
48
|
-
this._sessionPlugin = null
|
|
49
|
-
this._sessionDeleteHandler = null
|
|
50
|
-
this._sessionAgents = new Map() // userId -> Agent
|
|
51
|
-
this._qrcodeTerminal = null
|
|
52
|
-
this._origStderrWrite = null
|
|
53
|
-
this._initialized = false
|
|
54
|
-
}
|
|
55
|
-
|
|
56
|
-
install(framework) {
|
|
57
|
-
this._framework = framework
|
|
58
|
-
return this
|
|
59
|
-
}
|
|
60
|
-
|
|
61
|
-
async start(framework) {
|
|
62
|
-
// 防止重复初始化
|
|
63
|
-
if (this._initialized) return this
|
|
64
|
-
this._initialized = true
|
|
65
|
-
|
|
66
|
-
// 获取 SessionPlugin 引用
|
|
67
|
-
this._sessionPlugin = framework.pluginManager.get('session')
|
|
68
|
-
|
|
69
|
-
// 监听 SessionPlugin 的会话删除事件
|
|
70
|
-
if (this._sessionPlugin) {
|
|
71
|
-
this._sessionDeleteHandler = (sessionId) => {
|
|
72
|
-
// 只处理 weixin 会话的删除
|
|
73
|
-
if (sessionId.startsWith('weixin_')) {
|
|
74
|
-
log.info(` Session deleted: ${sessionId}`)
|
|
75
|
-
}
|
|
76
|
-
}
|
|
77
|
-
this._sessionPlugin.on('session:deleted', this._sessionDeleteHandler)
|
|
78
|
-
}
|
|
79
|
-
|
|
80
|
-
// 监听定时提醒事件
|
|
81
|
-
if (this._framework) {
|
|
82
|
-
this._framework.on('scheduler:reminder', async (data) => {
|
|
83
|
-
log.info(' Received scheduler reminder:', data)
|
|
84
|
-
await this._handleScheduledReminder(data)
|
|
85
|
-
})
|
|
86
|
-
|
|
87
|
-
// 监听 webhook 事件
|
|
88
|
-
this._framework.on('webhook:received', async (data) => {
|
|
89
|
-
log.info(' Received webhook event:', data)
|
|
90
|
-
await this._handleWebhookNotification(data)
|
|
91
|
-
})
|
|
92
|
-
|
|
93
|
-
// 监听统一通知事件
|
|
94
|
-
this._framework.on('notification', async (data) => {
|
|
95
|
-
log.info(' Received notification:', data)
|
|
96
|
-
await this._handleNotification(data)
|
|
97
|
-
})
|
|
98
|
-
}
|
|
99
|
-
|
|
100
|
-
// 异步初始化 Bot
|
|
101
|
-
this._initBotAsync().catch(err => {
|
|
102
|
-
log.error(' Failed to initialize bot:', err.message)
|
|
103
|
-
})
|
|
104
|
-
return this
|
|
105
|
-
}
|
|
106
|
-
|
|
107
|
-
async _initBotAsync() {
|
|
108
|
-
|
|
109
|
-
// 如果启用终端二维码渲染
|
|
110
|
-
if (this.config.qrcodeTerminal) {
|
|
111
|
-
try {
|
|
112
|
-
this._qrcodeTerminal = require('qrcode-terminal')
|
|
113
|
-
this._interceptQRCode()
|
|
114
|
-
} catch (err) {
|
|
115
|
-
log.warn(' qrcode-terminal not installed:', err.message)
|
|
116
|
-
}
|
|
117
|
-
}
|
|
118
|
-
|
|
119
|
-
this._bot = new WeixinBot({
|
|
120
|
-
tokenPath:`${this.path}/${this.name}.json`,
|
|
121
|
-
onError: (err) => {
|
|
122
|
-
log.error(' Error:', err instanceof Error ? err.stack ?? err.message : String(err))
|
|
123
|
-
},
|
|
124
|
-
})
|
|
125
|
-
|
|
126
|
-
const loginOptions = { force: this.config.forceLogin }
|
|
127
|
-
log.info('', this.config.forceLogin ? '强制重新扫码登录...' : '正在登录(已有凭证则自动跳过扫码)...')
|
|
128
|
-
|
|
129
|
-
const creds = await this._bot.login(loginOptions)
|
|
130
|
-
log.info(' 登录成功 — Bot ID:', creds.accountId)
|
|
131
|
-
log.info(' 关联用户:', creds.userId)
|
|
132
|
-
log.info(' API 地址:', creds.baseUrl)
|
|
133
|
-
|
|
134
|
-
// 注册消息处理
|
|
135
|
-
this._bot.onMessage(async (msg) => {
|
|
136
|
-
await this._handleMessage(msg)
|
|
137
|
-
})
|
|
138
|
-
|
|
139
|
-
// 启动Bot
|
|
140
|
-
await this._bot.run()
|
|
141
|
-
log.info(' 开始接收微信消息')
|
|
142
|
-
}
|
|
143
|
-
|
|
144
|
-
/**
|
|
145
|
-
* 拦截 SDK 的 stderr 输出,渲染二维码到终端
|
|
146
|
-
*/
|
|
147
|
-
_interceptQRCode() {
|
|
148
|
-
if (!this._qrcodeTerminal) return
|
|
149
|
-
|
|
150
|
-
this._origStderrWrite = process.stderr.write.bind(process.stderr)
|
|
151
|
-
process.stderr.write = ((chunk, ...args) => {
|
|
152
|
-
const str = typeof chunk === 'string' ? chunk : chunk.toString()
|
|
153
|
-
// 检测到登录 URL,渲染二维码
|
|
154
|
-
if (str.startsWith('https://') && str.includes('qrcode=')) {
|
|
155
|
-
const url = str.trim()
|
|
156
|
-
this._qrcodeTerminal.generate(url, { small: true }, (qr) => {
|
|
157
|
-
this._origStderrWrite(qr + '\n')
|
|
158
|
-
})
|
|
159
|
-
}
|
|
160
|
-
return this._origStderrWrite(chunk, ...args)
|
|
161
|
-
})
|
|
162
|
-
}
|
|
163
|
-
|
|
164
|
-
/**
|
|
165
|
-
* 获取主Agent
|
|
166
|
-
*/
|
|
167
|
-
_getMainAgent() {
|
|
168
|
-
if (this._framework._mainAgent) {
|
|
169
|
-
return this._framework._mainAgent
|
|
170
|
-
}
|
|
171
|
-
const agents = this._framework._agents || []
|
|
172
|
-
return agents.length > 0 ? agents[agents.length - 1] : null
|
|
173
|
-
}
|
|
174
|
-
|
|
175
|
-
/**
|
|
176
|
-
* 获取或创建会话Agent
|
|
177
|
-
* 使用 SessionPlugin 统一管理会话历史
|
|
178
|
-
*/
|
|
179
|
-
_getSessionAgent(userId) {
|
|
180
|
-
// 检查缓存
|
|
181
|
-
if (this._sessionAgents.has(userId)) {
|
|
182
|
-
const agent = this._sessionAgents.get(userId)
|
|
183
|
-
log.info(' Reusing cached session agent for userId:', userId)
|
|
184
|
-
return { agent, sessionId: `weixin_${userId}` }
|
|
185
|
-
}
|
|
186
|
-
|
|
187
|
-
// 创建新 agent
|
|
188
|
-
const agent = this._framework.createSessionAgent(`weixin_${userId}`, {
|
|
189
|
-
systemPrompt: this.systemPrompt,
|
|
190
|
-
sharedPrompt: `工作目录: {{WORK_DIR}}`,
|
|
191
|
-
metadata: { WORK_DIR: process.cwd() }
|
|
192
|
-
})
|
|
193
|
-
this._sessionAgents.set(userId, agent)
|
|
194
|
-
log.info(' Created new session agent for userId:', userId)
|
|
195
|
-
|
|
196
|
-
// 使用 SessionPlugin 管理会话历史
|
|
197
|
-
if (this._sessionPlugin) {
|
|
198
|
-
const sessionId = `weixin_${userId}`
|
|
199
|
-
this._sessionPlugin.getOrCreateSession(sessionId, {
|
|
200
|
-
metadata: { platform: 'weixin', userId }
|
|
201
|
-
})
|
|
202
|
-
}
|
|
203
|
-
|
|
204
|
-
log.info(` Session ready for user: ${userId}`)
|
|
205
|
-
return { agent, sessionId: `weixin_${userId}` }
|
|
206
|
-
}
|
|
207
|
-
|
|
208
|
-
/**
|
|
209
|
-
* 处理消息
|
|
210
|
-
*/
|
|
211
|
-
async _handleMessage(msg) {
|
|
212
|
-
if (!msg || !msg.userId) return
|
|
213
|
-
|
|
214
|
-
const userId = msg.userId
|
|
215
|
-
// 从 SessionPlugin 获取历史消息数量
|
|
216
|
-
let messageCount = 0
|
|
217
|
-
if (this._sessionPlugin) {
|
|
218
|
-
const session = this._sessionPlugin.getSession(`weixin_${userId}`)
|
|
219
|
-
messageCount = session?.messages?.length || 0
|
|
220
|
-
}
|
|
221
|
-
|
|
222
|
-
log.info(` #${messageCount + 1} | 类型: ${msg.type} | 用户: ${userId}`)
|
|
223
|
-
log.info(` 内容: ${msg.text}`)
|
|
224
|
-
|
|
225
|
-
// 非文本消息暂不处理
|
|
226
|
-
if (msg.type !== 'text' || !msg.text) {
|
|
227
|
-
log.info(' Unsupported message type or no text')
|
|
228
|
-
return
|
|
229
|
-
}
|
|
230
|
-
|
|
231
|
-
const text = msg.text.trim()
|
|
232
|
-
await this._processChat(userId, text, msg)
|
|
233
|
-
}
|
|
234
|
-
|
|
235
|
-
/**
|
|
236
|
-
* 处理对话
|
|
237
|
-
*/
|
|
238
|
-
async _processChat(userId, text, originalMsg) {
|
|
239
|
-
const sessionInfo = this._getSessionAgent(userId)
|
|
240
|
-
if (!sessionInfo) {
|
|
241
|
-
log.error(' No session agent available')
|
|
242
|
-
return
|
|
243
|
-
}
|
|
244
|
-
|
|
245
|
-
const { agent, sessionId } = sessionInfo
|
|
246
|
-
|
|
247
|
-
// 使用 SessionPlugin 添加用户消息到历史
|
|
248
|
-
if (this._sessionPlugin) {
|
|
249
|
-
this._sessionPlugin.addMessage(sessionId, { role: 'user', content: text })
|
|
250
|
-
}
|
|
251
|
-
|
|
252
|
-
// 发送正在输入状态
|
|
253
|
-
try {
|
|
254
|
-
await this._bot.sendTyping(userId)
|
|
255
|
-
} catch { /* typing 失败不影响回复 */ }
|
|
256
|
-
|
|
257
|
-
await new Promise((resolve) => setTimeout(resolve, 1000))
|
|
258
|
-
|
|
259
|
-
try {
|
|
260
|
-
// 使用非流式响应
|
|
261
|
-
const result = await agent.chat(text, {
|
|
262
|
-
sessionId: sessionId
|
|
263
|
-
})
|
|
264
|
-
const fullResponse = cleanResponse(result.message || '')
|
|
265
|
-
|
|
266
|
-
// 保存助手回复到历史(使用 SessionPlugin)
|
|
267
|
-
if (this._sessionPlugin) {
|
|
268
|
-
this._sessionPlugin.addMessage(sessionId, { role: 'assistant', content: fullResponse })
|
|
269
|
-
}
|
|
270
|
-
|
|
271
|
-
// 发送回复(超过500字自动分批)
|
|
272
|
-
if (fullResponse) {
|
|
273
|
-
await this._sendMessageBatch(originalMsg, userId, fullResponse, true)
|
|
274
|
-
log.info(` 回复成功 (${fullResponse.length} 字符)`)
|
|
275
|
-
} else {
|
|
276
|
-
await this._sendMessageBatch(originalMsg, userId, '抱歉,我没有收到有效的回复。', true)
|
|
277
|
-
}
|
|
278
|
-
|
|
279
|
-
} catch (err) {
|
|
280
|
-
log.error(' Chat error:', err)
|
|
281
|
-
await this._sendMessageBatch(originalMsg, userId, `发生错误:${err.message}`, true)
|
|
282
|
-
}
|
|
283
|
-
}
|
|
284
|
-
|
|
285
|
-
/**
|
|
286
|
-
* 权限检查
|
|
287
|
-
*/
|
|
288
|
-
_checkPermission(userId) {
|
|
289
|
-
if (!this.config.allowedUsers || this.config.allowedUsers.length === 0) {
|
|
290
|
-
return true
|
|
291
|
-
}
|
|
292
|
-
return this.config.allowedUsers.includes(userId)
|
|
293
|
-
}
|
|
294
|
-
|
|
295
|
-
/**
|
|
296
|
-
* 发送消息(超过500字自动分批)
|
|
297
|
-
* @param {Object} originalMsg - 原始消息对象(用于reply)
|
|
298
|
-
* @param {string} userId - 用户ID(用于sendText)
|
|
299
|
-
* @param {string} text - 要发送的文本
|
|
300
|
-
* @param {boolean} useReply - 是否使用reply方式
|
|
301
|
-
*/
|
|
302
|
-
async _sendMessageBatch(originalMsg, userId, text, useReply = true) {
|
|
303
|
-
const MAX_LEN = 500
|
|
304
|
-
text=removeMarkdown(text)
|
|
305
|
-
if (!text || text.length <= MAX_LEN) {
|
|
306
|
-
if (useReply && originalMsg) {
|
|
307
|
-
await this._bot.reply(originalMsg, text)
|
|
308
|
-
} else {
|
|
309
|
-
await this._bot.sendText(userId, text)
|
|
310
|
-
}
|
|
311
|
-
return
|
|
312
|
-
}
|
|
313
|
-
|
|
314
|
-
// 分批发送
|
|
315
|
-
const chunks = []
|
|
316
|
-
for (let i = 0; i < text.length; i += MAX_LEN) {
|
|
317
|
-
chunks.push(text.slice(i, i + MAX_LEN))
|
|
318
|
-
}
|
|
319
|
-
|
|
320
|
-
log.info(` Message too long (${text.length}), splitting into ${chunks.length} parts`)
|
|
321
|
-
|
|
322
|
-
for (let i = 0; i < chunks.length; i++) {
|
|
323
|
-
const chunk = `[${i + 1}/${chunks.length}]\n${chunks[i]}`
|
|
324
|
-
if (useReply && originalMsg) {
|
|
325
|
-
await this._bot.reply(originalMsg, chunk)
|
|
326
|
-
} else {
|
|
327
|
-
await this._bot.sendText(userId, chunk)
|
|
328
|
-
}
|
|
329
|
-
// 批次间隔,避免发送太快
|
|
330
|
-
if (i < chunks.length - 1) {
|
|
331
|
-
await new Promise(r => setTimeout(r, 300))
|
|
332
|
-
}
|
|
333
|
-
}
|
|
334
|
-
}
|
|
335
|
-
|
|
336
|
-
/**
|
|
337
|
-
* 清除会话
|
|
338
|
-
*/
|
|
339
|
-
_clearSession(userId) {
|
|
340
|
-
// 清除 SessionPlugin 中的会话(会触发 session:deleted 事件)
|
|
341
|
-
if (this._sessionPlugin) {
|
|
342
|
-
const sessionId = `weixin_${userId}`
|
|
343
|
-
this._sessionPlugin.deleteSession(sessionId)
|
|
344
|
-
}
|
|
345
|
-
}
|
|
346
|
-
|
|
347
|
-
/**
|
|
348
|
-
* 处理定时提醒
|
|
349
|
-
*/
|
|
350
|
-
async _handleScheduledReminder(data) {
|
|
351
|
-
const { taskName, message, sessionId } = data
|
|
352
|
-
|
|
353
|
-
if (!this._bot) {
|
|
354
|
-
log.warn(' Bot not ready, cannot send reminder')
|
|
355
|
-
return
|
|
356
|
-
}
|
|
357
|
-
|
|
358
|
-
// 构建提醒消息
|
|
359
|
-
const reminderText = `🔔 [${taskName}]\n\n${message}`
|
|
360
|
-
|
|
361
|
-
// 如果有 sessionId 是 weixin 类型的,发送到对应用户
|
|
362
|
-
if (sessionId && sessionId.startsWith('weixin_')) {
|
|
363
|
-
const userId = sessionId.replace('weixin_', '')
|
|
364
|
-
try {
|
|
365
|
-
await this._sendMessageBatch(null, userId, reminderText, false)
|
|
366
|
-
log.info(` Reminder sent to user ${userId}`)
|
|
367
|
-
} catch (err) {
|
|
368
|
-
log.error(` Failed to send reminder:`, err.message)
|
|
369
|
-
}
|
|
370
|
-
return
|
|
371
|
-
}
|
|
372
|
-
|
|
373
|
-
// 其他情况(包括 null 或其他 sessionId),发送到最近的 WeChat 会话
|
|
374
|
-
if (this._sessionPlugin) {
|
|
375
|
-
const allSessions = this._sessionPlugin.listSessions()
|
|
376
|
-
const weixinSessions = allSessions
|
|
377
|
-
.filter(s => s.id.startsWith('weixin_'))
|
|
378
|
-
.sort((a, b) => new Date(b.lastActive).getTime() - new Date(a.lastActive).getTime())
|
|
379
|
-
|
|
380
|
-
if (weixinSessions.length > 0) {
|
|
381
|
-
const userId = weixinSessions[0].id.replace('weixin_', '')
|
|
382
|
-
try {
|
|
383
|
-
await this._sendMessageBatch(null, userId, reminderText, false)
|
|
384
|
-
log.info(` Reminder sent to recent user ${userId}`)
|
|
385
|
-
} catch (err) {
|
|
386
|
-
log.error(` Failed to send reminder:`, err.message)
|
|
387
|
-
}
|
|
388
|
-
} else {
|
|
389
|
-
log.warn(' No WeChat session found to send reminder')
|
|
390
|
-
}
|
|
391
|
-
}
|
|
392
|
-
}
|
|
393
|
-
|
|
394
|
-
/**
|
|
395
|
-
* 处理 webhook 通知
|
|
396
|
-
*/
|
|
397
|
-
async _handleWebhookNotification(data) {
|
|
398
|
-
const { data: webhookData, response, sessionId } = data
|
|
399
|
-
|
|
400
|
-
if (!this._bot) {
|
|
401
|
-
log.warn(' Bot not ready, cannot send webhook notification')
|
|
402
|
-
return
|
|
403
|
-
}
|
|
404
|
-
|
|
405
|
-
// 只处理 weixin 相关的 session
|
|
406
|
-
if (!sessionId || !sessionId.startsWith('weixin_')) {
|
|
407
|
-
return
|
|
408
|
-
}
|
|
409
|
-
|
|
410
|
-
const userId = sessionId.replace('weixin_', '')
|
|
411
|
-
const notificationText = `📥 [Webhook 接收]\n\n路径: ${webhookData.path}\n方法: ${webhookData.method}\n\n处理结果: ${response || '处理中...'}`
|
|
412
|
-
|
|
413
|
-
try {
|
|
414
|
-
await this._sendMessageBatch(null, userId, notificationText, false)
|
|
415
|
-
log.info(` Webhook notification sent to user ${userId}`)
|
|
416
|
-
} catch (err) {
|
|
417
|
-
log.error(` Failed to send webhook notification:`, err.message)
|
|
418
|
-
}
|
|
419
|
-
}
|
|
420
|
-
|
|
421
|
-
/**
|
|
422
|
-
* 处理统一通知
|
|
423
|
-
*/
|
|
424
|
-
async _handleNotification(data) {
|
|
425
|
-
const { title, message, source, level, sessionId } = data
|
|
426
|
-
|
|
427
|
-
if (!this._bot) {
|
|
428
|
-
log.warn(' Bot not ready, cannot send notification')
|
|
429
|
-
return
|
|
430
|
-
}
|
|
431
|
-
|
|
432
|
-
// 确定用户ID
|
|
433
|
-
let userId = null
|
|
434
|
-
let effectiveSessionId = sessionId
|
|
435
|
-
|
|
436
|
-
// 如果没有 sessionId,尝试从执行上下文获取
|
|
437
|
-
if (!effectiveSessionId) {
|
|
438
|
-
const ctx = this._framework.getExecutionContext()
|
|
439
|
-
if (ctx?.sessionId) {
|
|
440
|
-
effectiveSessionId = ctx.sessionId
|
|
441
|
-
}
|
|
442
|
-
}
|
|
443
|
-
|
|
444
|
-
if (effectiveSessionId && effectiveSessionId.startsWith('weixin_')) {
|
|
445
|
-
userId = effectiveSessionId.replace('weixin_', '')
|
|
446
|
-
} else if (this._sessionPlugin) {
|
|
447
|
-
// 获取最近的 weixin 会话
|
|
448
|
-
const sessions = this._sessionPlugin.listSessions()
|
|
449
|
-
.filter(s => s.id.startsWith('weixin_'))
|
|
450
|
-
.sort((a, b) => new Date(b.lastActive).getTime() - new Date(a.lastActive).getTime())
|
|
451
|
-
if (sessions.length > 0) {
|
|
452
|
-
userId = sessions[0].id.replace('weixin_', '')
|
|
453
|
-
}
|
|
454
|
-
}
|
|
455
|
-
|
|
456
|
-
if (!userId) {
|
|
457
|
-
log.warn(' No weixin session found for notification')
|
|
458
|
-
return
|
|
459
|
-
}
|
|
460
|
-
|
|
461
|
-
// 格式化通知消息
|
|
462
|
-
const levelEmoji = {
|
|
463
|
-
info: 'ℹ️',
|
|
464
|
-
warning: '⚠️',
|
|
465
|
-
success: '✅',
|
|
466
|
-
error: '❌'
|
|
467
|
-
}
|
|
468
|
-
const emoji = levelEmoji[level] || 'ℹ️'
|
|
469
|
-
const notificationText = `${emoji} [${source}] ${title}\n\n${message}`
|
|
470
|
-
|
|
471
|
-
try {
|
|
472
|
-
await this._sendMessageBatch(null, userId, notificationText, false)
|
|
473
|
-
log.info(` Notification sent to user ${userId}`)
|
|
474
|
-
} catch (err) {
|
|
475
|
-
log.error(` Failed to send notification:`, err.message)
|
|
476
|
-
}
|
|
477
|
-
}
|
|
478
|
-
|
|
479
|
-
/**
|
|
480
|
-
* 获取插件状态
|
|
481
|
-
*/
|
|
482
|
-
getStatus() {
|
|
483
|
-
// 从 SessionPlugin 获取 WeChat 相关会话
|
|
484
|
-
let sessions = []
|
|
485
|
-
if (this._sessionPlugin) {
|
|
486
|
-
const allSessions = this._sessionPlugin.listSessions()
|
|
487
|
-
sessions = allSessions
|
|
488
|
-
.filter(s => s.id.startsWith('weixin_'))
|
|
489
|
-
.map(s => ({
|
|
490
|
-
userId: s.id.replace('weixin_', ''),
|
|
491
|
-
historyLength: s.messageCount,
|
|
492
|
-
lastActive: s.lastActive
|
|
493
|
-
}))
|
|
494
|
-
}
|
|
495
|
-
|
|
496
|
-
return {
|
|
497
|
-
connected: !!this._bot,
|
|
498
|
-
sessionCount: sessions.length,
|
|
499
|
-
sessions,
|
|
500
|
-
config: {
|
|
501
|
-
forceLogin: this.config.forceLogin,
|
|
502
|
-
qrcodeTerminal: this.config.qrcodeTerminal
|
|
503
|
-
}
|
|
504
|
-
}
|
|
505
|
-
}
|
|
506
|
-
|
|
507
|
-
/**
|
|
508
|
-
* 停止 Bot
|
|
509
|
-
*/
|
|
510
|
-
stopBot() {
|
|
511
|
-
if (this._bot) {
|
|
512
|
-
this._bot.stop()
|
|
513
|
-
this._bot = null
|
|
514
|
-
log.info(' Bot stopped')
|
|
515
|
-
}
|
|
516
|
-
}
|
|
517
|
-
|
|
518
|
-
reload(framework) {
|
|
519
|
-
this._framework = framework
|
|
520
|
-
this._initialized = false
|
|
521
|
-
this.stopBot()
|
|
522
|
-
this.start(framework)
|
|
523
|
-
}
|
|
524
|
-
|
|
525
|
-
uninstall(framework) {
|
|
526
|
-
// 销毁所有 session agents
|
|
527
|
-
for (const agent of this._sessionAgents.values()) {
|
|
528
|
-
agent.destroy()
|
|
529
|
-
}
|
|
530
|
-
this._sessionAgents.clear()
|
|
531
|
-
|
|
532
|
-
this.stopBot()
|
|
533
|
-
if (this._sessionPlugin && this._sessionDeleteHandler) {
|
|
534
|
-
this._sessionPlugin.off('session:deleted', this._sessionDeleteHandler)
|
|
535
|
-
this._sessionDeleteHandler = null
|
|
536
|
-
}
|
|
537
|
-
this._sessionPlugin = null
|
|
538
|
-
this._framework = null
|
|
539
|
-
// 恢复 stderr
|
|
540
|
-
if (this._origStderrWrite) {
|
|
541
|
-
process.stderr.write = this._origStderrWrite
|
|
542
|
-
}
|
|
543
|
-
}
|
|
544
|
-
}
|
|
545
|
-
|
|
1
|
+
/**
|
|
2
|
+
* WeChat 插件
|
|
3
|
+
* 使用 @chnak/weixin-bot 实现微信对话
|
|
4
|
+
*
|
|
5
|
+
* 配置:
|
|
6
|
+
* - forceLogin: 是否强制重新扫码登录
|
|
7
|
+
* - qrcodeTerminal: 是否在终端渲染二维码 (默认 true)
|
|
8
|
+
*/
|
|
9
|
+
|
|
10
|
+
const { Plugin } = require('../src/core/plugin-base')
|
|
11
|
+
const { logger } = require('../src/utils/logger')
|
|
12
|
+
const log = logger.child('WeChat')
|
|
13
|
+
const {cleanResponse} =require('../src/utils')
|
|
14
|
+
const removeMarkdown = require('remove-markdown');
|
|
15
|
+
const { z } = require('zod')
|
|
16
|
+
const { WeixinBot } = require('@chnak/weixin-bot')
|
|
17
|
+
|
|
18
|
+
class WeixinPlugin extends Plugin {
|
|
19
|
+
constructor(config = {}) {
|
|
20
|
+
super()
|
|
21
|
+
this.name = 'weixin'
|
|
22
|
+
this.version = '1.0.0'
|
|
23
|
+
this.description = '微信对话插件,使用微信网页账号进行对话'
|
|
24
|
+
this.priority = 80
|
|
25
|
+
// 默认不启用,需要在 plugins.json 中设置 enabled: true
|
|
26
|
+
this.enabled = false
|
|
27
|
+
this.path=`.agent/data`
|
|
28
|
+
this.systemPrompt=`你是一个微信助手。
|
|
29
|
+
|
|
30
|
+
**重要:** 子Agent 匹配规则必须遵守:
|
|
31
|
+
- 根据【子 Agent 匹配表】,将任务委托给最匹配的子Agent处理
|
|
32
|
+
- 使用 subagent_call 工具并指定 agentName 来委托任务
|
|
33
|
+
- 只有当没有匹配的子Agent时,才直接调用工具
|
|
34
|
+
|
|
35
|
+
**命令执行规范:**
|
|
36
|
+
- 执行 npx skills add / npx skills remove / npx skills list 命令时,必须自动添加参数:-a openclaw -y
|
|
37
|
+
- 禁止添加 -g 参数
|
|
38
|
+
- 例如:npx skills add xxx 应该执行为 npx skills add xxx -a openclaw -y`
|
|
39
|
+
|
|
40
|
+
this.config = {
|
|
41
|
+
forceLogin: config.forceLogin || process.env.WEIXIN_FORCE_LOGIN === 'true',
|
|
42
|
+
qrcodeTerminal: config.qrcodeTerminal !== false && process.env.WEIXIN_QRCODE_TERMINAL !== 'false',
|
|
43
|
+
allowedUsers: config.allowedUsers || []
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
this._framework = null
|
|
47
|
+
this._bot = null
|
|
48
|
+
this._sessionPlugin = null
|
|
49
|
+
this._sessionDeleteHandler = null
|
|
50
|
+
this._sessionAgents = new Map() // userId -> Agent
|
|
51
|
+
this._qrcodeTerminal = null
|
|
52
|
+
this._origStderrWrite = null
|
|
53
|
+
this._initialized = false
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
install(framework) {
|
|
57
|
+
this._framework = framework
|
|
58
|
+
return this
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
async start(framework) {
|
|
62
|
+
// 防止重复初始化
|
|
63
|
+
if (this._initialized) return this
|
|
64
|
+
this._initialized = true
|
|
65
|
+
|
|
66
|
+
// 获取 SessionPlugin 引用
|
|
67
|
+
this._sessionPlugin = framework.pluginManager.get('session')
|
|
68
|
+
|
|
69
|
+
// 监听 SessionPlugin 的会话删除事件
|
|
70
|
+
if (this._sessionPlugin) {
|
|
71
|
+
this._sessionDeleteHandler = (sessionId) => {
|
|
72
|
+
// 只处理 weixin 会话的删除
|
|
73
|
+
if (sessionId.startsWith('weixin_')) {
|
|
74
|
+
log.info(` Session deleted: ${sessionId}`)
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
this._sessionPlugin.on('session:deleted', this._sessionDeleteHandler)
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
// 监听定时提醒事件
|
|
81
|
+
if (this._framework) {
|
|
82
|
+
this._framework.on('scheduler:reminder', async (data) => {
|
|
83
|
+
log.info(' Received scheduler reminder:', data)
|
|
84
|
+
await this._handleScheduledReminder(data)
|
|
85
|
+
})
|
|
86
|
+
|
|
87
|
+
// 监听 webhook 事件
|
|
88
|
+
this._framework.on('webhook:received', async (data) => {
|
|
89
|
+
log.info(' Received webhook event:', data)
|
|
90
|
+
await this._handleWebhookNotification(data)
|
|
91
|
+
})
|
|
92
|
+
|
|
93
|
+
// 监听统一通知事件
|
|
94
|
+
this._framework.on('notification', async (data) => {
|
|
95
|
+
log.info(' Received notification:', data)
|
|
96
|
+
await this._handleNotification(data)
|
|
97
|
+
})
|
|
98
|
+
}
|
|
99
|
+
|
|
100
|
+
// 异步初始化 Bot
|
|
101
|
+
this._initBotAsync().catch(err => {
|
|
102
|
+
log.error(' Failed to initialize bot:', err.message)
|
|
103
|
+
})
|
|
104
|
+
return this
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
async _initBotAsync() {
|
|
108
|
+
|
|
109
|
+
// 如果启用终端二维码渲染
|
|
110
|
+
if (this.config.qrcodeTerminal) {
|
|
111
|
+
try {
|
|
112
|
+
this._qrcodeTerminal = require('qrcode-terminal')
|
|
113
|
+
this._interceptQRCode()
|
|
114
|
+
} catch (err) {
|
|
115
|
+
log.warn(' qrcode-terminal not installed:', err.message)
|
|
116
|
+
}
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
this._bot = new WeixinBot({
|
|
120
|
+
tokenPath:`${this.path}/${this.name}.json`,
|
|
121
|
+
onError: (err) => {
|
|
122
|
+
log.error(' Error:', err instanceof Error ? err.stack ?? err.message : String(err))
|
|
123
|
+
},
|
|
124
|
+
})
|
|
125
|
+
|
|
126
|
+
const loginOptions = { force: this.config.forceLogin }
|
|
127
|
+
log.info('', this.config.forceLogin ? '强制重新扫码登录...' : '正在登录(已有凭证则自动跳过扫码)...')
|
|
128
|
+
|
|
129
|
+
const creds = await this._bot.login(loginOptions)
|
|
130
|
+
log.info(' 登录成功 — Bot ID:', creds.accountId)
|
|
131
|
+
log.info(' 关联用户:', creds.userId)
|
|
132
|
+
log.info(' API 地址:', creds.baseUrl)
|
|
133
|
+
|
|
134
|
+
// 注册消息处理
|
|
135
|
+
this._bot.onMessage(async (msg) => {
|
|
136
|
+
await this._handleMessage(msg)
|
|
137
|
+
})
|
|
138
|
+
|
|
139
|
+
// 启动Bot
|
|
140
|
+
await this._bot.run()
|
|
141
|
+
log.info(' 开始接收微信消息')
|
|
142
|
+
}
|
|
143
|
+
|
|
144
|
+
/**
|
|
145
|
+
* 拦截 SDK 的 stderr 输出,渲染二维码到终端
|
|
146
|
+
*/
|
|
147
|
+
_interceptQRCode() {
|
|
148
|
+
if (!this._qrcodeTerminal) return
|
|
149
|
+
|
|
150
|
+
this._origStderrWrite = process.stderr.write.bind(process.stderr)
|
|
151
|
+
process.stderr.write = ((chunk, ...args) => {
|
|
152
|
+
const str = typeof chunk === 'string' ? chunk : chunk.toString()
|
|
153
|
+
// 检测到登录 URL,渲染二维码
|
|
154
|
+
if (str.startsWith('https://') && str.includes('qrcode=')) {
|
|
155
|
+
const url = str.trim()
|
|
156
|
+
this._qrcodeTerminal.generate(url, { small: true }, (qr) => {
|
|
157
|
+
this._origStderrWrite(qr + '\n')
|
|
158
|
+
})
|
|
159
|
+
}
|
|
160
|
+
return this._origStderrWrite(chunk, ...args)
|
|
161
|
+
})
|
|
162
|
+
}
|
|
163
|
+
|
|
164
|
+
/**
|
|
165
|
+
* 获取主Agent
|
|
166
|
+
*/
|
|
167
|
+
_getMainAgent() {
|
|
168
|
+
if (this._framework._mainAgent) {
|
|
169
|
+
return this._framework._mainAgent
|
|
170
|
+
}
|
|
171
|
+
const agents = this._framework._agents || []
|
|
172
|
+
return agents.length > 0 ? agents[agents.length - 1] : null
|
|
173
|
+
}
|
|
174
|
+
|
|
175
|
+
/**
|
|
176
|
+
* 获取或创建会话Agent
|
|
177
|
+
* 使用 SessionPlugin 统一管理会话历史
|
|
178
|
+
*/
|
|
179
|
+
_getSessionAgent(userId) {
|
|
180
|
+
// 检查缓存
|
|
181
|
+
if (this._sessionAgents.has(userId)) {
|
|
182
|
+
const agent = this._sessionAgents.get(userId)
|
|
183
|
+
log.info(' Reusing cached session agent for userId:', userId)
|
|
184
|
+
return { agent, sessionId: `weixin_${userId}` }
|
|
185
|
+
}
|
|
186
|
+
|
|
187
|
+
// 创建新 agent
|
|
188
|
+
const agent = this._framework.createSessionAgent(`weixin_${userId}`, {
|
|
189
|
+
systemPrompt: this.systemPrompt,
|
|
190
|
+
sharedPrompt: `工作目录: {{WORK_DIR}}`,
|
|
191
|
+
metadata: { WORK_DIR: process.cwd() }
|
|
192
|
+
})
|
|
193
|
+
this._sessionAgents.set(userId, agent)
|
|
194
|
+
log.info(' Created new session agent for userId:', userId)
|
|
195
|
+
|
|
196
|
+
// 使用 SessionPlugin 管理会话历史
|
|
197
|
+
if (this._sessionPlugin) {
|
|
198
|
+
const sessionId = `weixin_${userId}`
|
|
199
|
+
this._sessionPlugin.getOrCreateSession(sessionId, {
|
|
200
|
+
metadata: { platform: 'weixin', userId }
|
|
201
|
+
})
|
|
202
|
+
}
|
|
203
|
+
|
|
204
|
+
log.info(` Session ready for user: ${userId}`)
|
|
205
|
+
return { agent, sessionId: `weixin_${userId}` }
|
|
206
|
+
}
|
|
207
|
+
|
|
208
|
+
/**
|
|
209
|
+
* 处理消息
|
|
210
|
+
*/
|
|
211
|
+
async _handleMessage(msg) {
|
|
212
|
+
if (!msg || !msg.userId) return
|
|
213
|
+
|
|
214
|
+
const userId = msg.userId
|
|
215
|
+
// 从 SessionPlugin 获取历史消息数量
|
|
216
|
+
let messageCount = 0
|
|
217
|
+
if (this._sessionPlugin) {
|
|
218
|
+
const session = this._sessionPlugin.getSession(`weixin_${userId}`)
|
|
219
|
+
messageCount = session?.messages?.length || 0
|
|
220
|
+
}
|
|
221
|
+
|
|
222
|
+
log.info(` #${messageCount + 1} | 类型: ${msg.type} | 用户: ${userId}`)
|
|
223
|
+
log.info(` 内容: ${msg.text}`)
|
|
224
|
+
|
|
225
|
+
// 非文本消息暂不处理
|
|
226
|
+
if (msg.type !== 'text' || !msg.text) {
|
|
227
|
+
log.info(' Unsupported message type or no text')
|
|
228
|
+
return
|
|
229
|
+
}
|
|
230
|
+
|
|
231
|
+
const text = msg.text.trim()
|
|
232
|
+
await this._processChat(userId, text, msg)
|
|
233
|
+
}
|
|
234
|
+
|
|
235
|
+
/**
|
|
236
|
+
* 处理对话
|
|
237
|
+
*/
|
|
238
|
+
async _processChat(userId, text, originalMsg) {
|
|
239
|
+
const sessionInfo = this._getSessionAgent(userId)
|
|
240
|
+
if (!sessionInfo) {
|
|
241
|
+
log.error(' No session agent available')
|
|
242
|
+
return
|
|
243
|
+
}
|
|
244
|
+
|
|
245
|
+
const { agent, sessionId } = sessionInfo
|
|
246
|
+
|
|
247
|
+
// 使用 SessionPlugin 添加用户消息到历史
|
|
248
|
+
if (this._sessionPlugin) {
|
|
249
|
+
this._sessionPlugin.addMessage(sessionId, { role: 'user', content: text })
|
|
250
|
+
}
|
|
251
|
+
|
|
252
|
+
// 发送正在输入状态
|
|
253
|
+
try {
|
|
254
|
+
await this._bot.sendTyping(userId)
|
|
255
|
+
} catch { /* typing 失败不影响回复 */ }
|
|
256
|
+
|
|
257
|
+
await new Promise((resolve) => setTimeout(resolve, 1000))
|
|
258
|
+
|
|
259
|
+
try {
|
|
260
|
+
// 使用非流式响应
|
|
261
|
+
const result = await agent.chat(text, {
|
|
262
|
+
sessionId: sessionId
|
|
263
|
+
})
|
|
264
|
+
const fullResponse = cleanResponse(result.message || '')
|
|
265
|
+
|
|
266
|
+
// 保存助手回复到历史(使用 SessionPlugin)
|
|
267
|
+
if (this._sessionPlugin) {
|
|
268
|
+
this._sessionPlugin.addMessage(sessionId, { role: 'assistant', content: fullResponse })
|
|
269
|
+
}
|
|
270
|
+
|
|
271
|
+
// 发送回复(超过500字自动分批)
|
|
272
|
+
if (fullResponse) {
|
|
273
|
+
await this._sendMessageBatch(originalMsg, userId, fullResponse, true)
|
|
274
|
+
log.info(` 回复成功 (${fullResponse.length} 字符)`)
|
|
275
|
+
} else {
|
|
276
|
+
await this._sendMessageBatch(originalMsg, userId, '抱歉,我没有收到有效的回复。', true)
|
|
277
|
+
}
|
|
278
|
+
|
|
279
|
+
} catch (err) {
|
|
280
|
+
log.error(' Chat error:', err)
|
|
281
|
+
await this._sendMessageBatch(originalMsg, userId, `发生错误:${err.message}`, true)
|
|
282
|
+
}
|
|
283
|
+
}
|
|
284
|
+
|
|
285
|
+
/**
|
|
286
|
+
* 权限检查
|
|
287
|
+
*/
|
|
288
|
+
_checkPermission(userId) {
|
|
289
|
+
if (!this.config.allowedUsers || this.config.allowedUsers.length === 0) {
|
|
290
|
+
return true
|
|
291
|
+
}
|
|
292
|
+
return this.config.allowedUsers.includes(userId)
|
|
293
|
+
}
|
|
294
|
+
|
|
295
|
+
/**
|
|
296
|
+
* 发送消息(超过500字自动分批)
|
|
297
|
+
* @param {Object} originalMsg - 原始消息对象(用于reply)
|
|
298
|
+
* @param {string} userId - 用户ID(用于sendText)
|
|
299
|
+
* @param {string} text - 要发送的文本
|
|
300
|
+
* @param {boolean} useReply - 是否使用reply方式
|
|
301
|
+
*/
|
|
302
|
+
async _sendMessageBatch(originalMsg, userId, text, useReply = true) {
|
|
303
|
+
const MAX_LEN = 500
|
|
304
|
+
text=removeMarkdown(text)
|
|
305
|
+
if (!text || text.length <= MAX_LEN) {
|
|
306
|
+
if (useReply && originalMsg) {
|
|
307
|
+
await this._bot.reply(originalMsg, text)
|
|
308
|
+
} else {
|
|
309
|
+
await this._bot.sendText(userId, text)
|
|
310
|
+
}
|
|
311
|
+
return
|
|
312
|
+
}
|
|
313
|
+
|
|
314
|
+
// 分批发送
|
|
315
|
+
const chunks = []
|
|
316
|
+
for (let i = 0; i < text.length; i += MAX_LEN) {
|
|
317
|
+
chunks.push(text.slice(i, i + MAX_LEN))
|
|
318
|
+
}
|
|
319
|
+
|
|
320
|
+
log.info(` Message too long (${text.length}), splitting into ${chunks.length} parts`)
|
|
321
|
+
|
|
322
|
+
for (let i = 0; i < chunks.length; i++) {
|
|
323
|
+
const chunk = `[${i + 1}/${chunks.length}]\n${chunks[i]}`
|
|
324
|
+
if (useReply && originalMsg) {
|
|
325
|
+
await this._bot.reply(originalMsg, chunk)
|
|
326
|
+
} else {
|
|
327
|
+
await this._bot.sendText(userId, chunk)
|
|
328
|
+
}
|
|
329
|
+
// 批次间隔,避免发送太快
|
|
330
|
+
if (i < chunks.length - 1) {
|
|
331
|
+
await new Promise(r => setTimeout(r, 300))
|
|
332
|
+
}
|
|
333
|
+
}
|
|
334
|
+
}
|
|
335
|
+
|
|
336
|
+
/**
|
|
337
|
+
* 清除会话
|
|
338
|
+
*/
|
|
339
|
+
_clearSession(userId) {
|
|
340
|
+
// 清除 SessionPlugin 中的会话(会触发 session:deleted 事件)
|
|
341
|
+
if (this._sessionPlugin) {
|
|
342
|
+
const sessionId = `weixin_${userId}`
|
|
343
|
+
this._sessionPlugin.deleteSession(sessionId)
|
|
344
|
+
}
|
|
345
|
+
}
|
|
346
|
+
|
|
347
|
+
/**
|
|
348
|
+
* 处理定时提醒
|
|
349
|
+
*/
|
|
350
|
+
async _handleScheduledReminder(data) {
|
|
351
|
+
const { taskName, message, sessionId } = data
|
|
352
|
+
|
|
353
|
+
if (!this._bot) {
|
|
354
|
+
log.warn(' Bot not ready, cannot send reminder')
|
|
355
|
+
return
|
|
356
|
+
}
|
|
357
|
+
|
|
358
|
+
// 构建提醒消息
|
|
359
|
+
const reminderText = `🔔 [${taskName}]\n\n${message}`
|
|
360
|
+
|
|
361
|
+
// 如果有 sessionId 是 weixin 类型的,发送到对应用户
|
|
362
|
+
if (sessionId && sessionId.startsWith('weixin_')) {
|
|
363
|
+
const userId = sessionId.replace('weixin_', '')
|
|
364
|
+
try {
|
|
365
|
+
await this._sendMessageBatch(null, userId, reminderText, false)
|
|
366
|
+
log.info(` Reminder sent to user ${userId}`)
|
|
367
|
+
} catch (err) {
|
|
368
|
+
log.error(` Failed to send reminder:`, err.message)
|
|
369
|
+
}
|
|
370
|
+
return
|
|
371
|
+
}
|
|
372
|
+
|
|
373
|
+
// 其他情况(包括 null 或其他 sessionId),发送到最近的 WeChat 会话
|
|
374
|
+
if (this._sessionPlugin) {
|
|
375
|
+
const allSessions = this._sessionPlugin.listSessions()
|
|
376
|
+
const weixinSessions = allSessions
|
|
377
|
+
.filter(s => s.id.startsWith('weixin_'))
|
|
378
|
+
.sort((a, b) => new Date(b.lastActive).getTime() - new Date(a.lastActive).getTime())
|
|
379
|
+
|
|
380
|
+
if (weixinSessions.length > 0) {
|
|
381
|
+
const userId = weixinSessions[0].id.replace('weixin_', '')
|
|
382
|
+
try {
|
|
383
|
+
await this._sendMessageBatch(null, userId, reminderText, false)
|
|
384
|
+
log.info(` Reminder sent to recent user ${userId}`)
|
|
385
|
+
} catch (err) {
|
|
386
|
+
log.error(` Failed to send reminder:`, err.message)
|
|
387
|
+
}
|
|
388
|
+
} else {
|
|
389
|
+
log.warn(' No WeChat session found to send reminder')
|
|
390
|
+
}
|
|
391
|
+
}
|
|
392
|
+
}
|
|
393
|
+
|
|
394
|
+
/**
|
|
395
|
+
* 处理 webhook 通知
|
|
396
|
+
*/
|
|
397
|
+
async _handleWebhookNotification(data) {
|
|
398
|
+
const { data: webhookData, response, sessionId } = data
|
|
399
|
+
|
|
400
|
+
if (!this._bot) {
|
|
401
|
+
log.warn(' Bot not ready, cannot send webhook notification')
|
|
402
|
+
return
|
|
403
|
+
}
|
|
404
|
+
|
|
405
|
+
// 只处理 weixin 相关的 session
|
|
406
|
+
if (!sessionId || !sessionId.startsWith('weixin_')) {
|
|
407
|
+
return
|
|
408
|
+
}
|
|
409
|
+
|
|
410
|
+
const userId = sessionId.replace('weixin_', '')
|
|
411
|
+
const notificationText = `📥 [Webhook 接收]\n\n路径: ${webhookData.path}\n方法: ${webhookData.method}\n\n处理结果: ${response || '处理中...'}`
|
|
412
|
+
|
|
413
|
+
try {
|
|
414
|
+
await this._sendMessageBatch(null, userId, notificationText, false)
|
|
415
|
+
log.info(` Webhook notification sent to user ${userId}`)
|
|
416
|
+
} catch (err) {
|
|
417
|
+
log.error(` Failed to send webhook notification:`, err.message)
|
|
418
|
+
}
|
|
419
|
+
}
|
|
420
|
+
|
|
421
|
+
/**
|
|
422
|
+
* 处理统一通知
|
|
423
|
+
*/
|
|
424
|
+
async _handleNotification(data) {
|
|
425
|
+
const { title, message, source, level, sessionId } = data
|
|
426
|
+
|
|
427
|
+
if (!this._bot) {
|
|
428
|
+
log.warn(' Bot not ready, cannot send notification')
|
|
429
|
+
return
|
|
430
|
+
}
|
|
431
|
+
|
|
432
|
+
// 确定用户ID
|
|
433
|
+
let userId = null
|
|
434
|
+
let effectiveSessionId = sessionId
|
|
435
|
+
|
|
436
|
+
// 如果没有 sessionId,尝试从执行上下文获取
|
|
437
|
+
if (!effectiveSessionId) {
|
|
438
|
+
const ctx = this._framework.getExecutionContext()
|
|
439
|
+
if (ctx?.sessionId) {
|
|
440
|
+
effectiveSessionId = ctx.sessionId
|
|
441
|
+
}
|
|
442
|
+
}
|
|
443
|
+
|
|
444
|
+
if (effectiveSessionId && effectiveSessionId.startsWith('weixin_')) {
|
|
445
|
+
userId = effectiveSessionId.replace('weixin_', '')
|
|
446
|
+
} else if (this._sessionPlugin) {
|
|
447
|
+
// 获取最近的 weixin 会话
|
|
448
|
+
const sessions = this._sessionPlugin.listSessions()
|
|
449
|
+
.filter(s => s.id.startsWith('weixin_'))
|
|
450
|
+
.sort((a, b) => new Date(b.lastActive).getTime() - new Date(a.lastActive).getTime())
|
|
451
|
+
if (sessions.length > 0) {
|
|
452
|
+
userId = sessions[0].id.replace('weixin_', '')
|
|
453
|
+
}
|
|
454
|
+
}
|
|
455
|
+
|
|
456
|
+
if (!userId) {
|
|
457
|
+
log.warn(' No weixin session found for notification')
|
|
458
|
+
return
|
|
459
|
+
}
|
|
460
|
+
|
|
461
|
+
// 格式化通知消息
|
|
462
|
+
const levelEmoji = {
|
|
463
|
+
info: 'ℹ️',
|
|
464
|
+
warning: '⚠️',
|
|
465
|
+
success: '✅',
|
|
466
|
+
error: '❌'
|
|
467
|
+
}
|
|
468
|
+
const emoji = levelEmoji[level] || 'ℹ️'
|
|
469
|
+
const notificationText = `${emoji} [${source}] ${title}\n\n${message}`
|
|
470
|
+
|
|
471
|
+
try {
|
|
472
|
+
await this._sendMessageBatch(null, userId, notificationText, false)
|
|
473
|
+
log.info(` Notification sent to user ${userId}`)
|
|
474
|
+
} catch (err) {
|
|
475
|
+
log.error(` Failed to send notification:`, err.message)
|
|
476
|
+
}
|
|
477
|
+
}
|
|
478
|
+
|
|
479
|
+
/**
|
|
480
|
+
* 获取插件状态
|
|
481
|
+
*/
|
|
482
|
+
getStatus() {
|
|
483
|
+
// 从 SessionPlugin 获取 WeChat 相关会话
|
|
484
|
+
let sessions = []
|
|
485
|
+
if (this._sessionPlugin) {
|
|
486
|
+
const allSessions = this._sessionPlugin.listSessions()
|
|
487
|
+
sessions = allSessions
|
|
488
|
+
.filter(s => s.id.startsWith('weixin_'))
|
|
489
|
+
.map(s => ({
|
|
490
|
+
userId: s.id.replace('weixin_', ''),
|
|
491
|
+
historyLength: s.messageCount,
|
|
492
|
+
lastActive: s.lastActive
|
|
493
|
+
}))
|
|
494
|
+
}
|
|
495
|
+
|
|
496
|
+
return {
|
|
497
|
+
connected: !!this._bot,
|
|
498
|
+
sessionCount: sessions.length,
|
|
499
|
+
sessions,
|
|
500
|
+
config: {
|
|
501
|
+
forceLogin: this.config.forceLogin,
|
|
502
|
+
qrcodeTerminal: this.config.qrcodeTerminal
|
|
503
|
+
}
|
|
504
|
+
}
|
|
505
|
+
}
|
|
506
|
+
|
|
507
|
+
/**
|
|
508
|
+
* 停止 Bot
|
|
509
|
+
*/
|
|
510
|
+
stopBot() {
|
|
511
|
+
if (this._bot) {
|
|
512
|
+
this._bot.stop()
|
|
513
|
+
this._bot = null
|
|
514
|
+
log.info(' Bot stopped')
|
|
515
|
+
}
|
|
516
|
+
}
|
|
517
|
+
|
|
518
|
+
reload(framework) {
|
|
519
|
+
this._framework = framework
|
|
520
|
+
this._initialized = false
|
|
521
|
+
this.stopBot()
|
|
522
|
+
this.start(framework)
|
|
523
|
+
}
|
|
524
|
+
|
|
525
|
+
uninstall(framework) {
|
|
526
|
+
// 销毁所有 session agents
|
|
527
|
+
for (const agent of this._sessionAgents.values()) {
|
|
528
|
+
agent.destroy()
|
|
529
|
+
}
|
|
530
|
+
this._sessionAgents.clear()
|
|
531
|
+
|
|
532
|
+
this.stopBot()
|
|
533
|
+
if (this._sessionPlugin && this._sessionDeleteHandler) {
|
|
534
|
+
this._sessionPlugin.off('session:deleted', this._sessionDeleteHandler)
|
|
535
|
+
this._sessionDeleteHandler = null
|
|
536
|
+
}
|
|
537
|
+
this._sessionPlugin = null
|
|
538
|
+
this._framework = null
|
|
539
|
+
// 恢复 stderr
|
|
540
|
+
if (this._origStderrWrite) {
|
|
541
|
+
process.stderr.write = this._origStderrWrite
|
|
542
|
+
}
|
|
543
|
+
}
|
|
544
|
+
}
|
|
545
|
+
|
|
546
546
|
module.exports = { WeixinPlugin }
|