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.
Files changed (88) hide show
  1. package/.claude/settings.local.json +159 -157
  2. package/cli/bin/foliko.js +12 -12
  3. package/cli/src/commands/chat.js +143 -143
  4. package/cli/src/commands/list.js +93 -93
  5. package/cli/src/index.js +75 -75
  6. package/cli/src/ui/chat-ui.js +201 -201
  7. package/cli/src/utils/ansi.js +40 -40
  8. package/cli/src/utils/markdown.js +292 -292
  9. package/examples/ambient-example.js +194 -194
  10. package/examples/basic.js +115 -115
  11. package/examples/bootstrap.js +121 -121
  12. package/examples/mcp-example.js +56 -56
  13. package/examples/skill-example.js +49 -49
  14. package/examples/test-chat.js +137 -137
  15. package/examples/test-mcp.js +85 -85
  16. package/examples/test-reload.js +59 -59
  17. package/examples/test-telegram.js +50 -50
  18. package/examples/test-tg-bot.js +45 -45
  19. package/examples/test-tg-simple.js +47 -47
  20. package/examples/test-tg.js +62 -62
  21. package/examples/test-think.js +43 -43
  22. package/examples/test-web-plugin.js +103 -103
  23. package/examples/test-weixin-feishu.js +103 -103
  24. package/examples/workflow.js +158 -158
  25. package/package.json +1 -1
  26. package/plugins/ai-plugin.js +102 -102
  27. package/plugins/ambient-agent/EventWatcher.js +113 -113
  28. package/plugins/ambient-agent/ExplorerLoop.js +640 -640
  29. package/plugins/ambient-agent/GoalManager.js +197 -197
  30. package/plugins/ambient-agent/Reflector.js +95 -95
  31. package/plugins/ambient-agent/StateStore.js +90 -90
  32. package/plugins/ambient-agent/constants.js +101 -101
  33. package/plugins/ambient-agent/index.js +579 -579
  34. package/plugins/audit-plugin.js +187 -187
  35. package/plugins/default-plugins.js +662 -662
  36. package/plugins/email/constants.js +64 -64
  37. package/plugins/email/handlers.js +461 -461
  38. package/plugins/email/index.js +278 -278
  39. package/plugins/email/monitor.js +269 -269
  40. package/plugins/email/parser.js +138 -138
  41. package/plugins/email/reply.js +151 -151
  42. package/plugins/email/utils.js +124 -124
  43. package/plugins/feishu-plugin.js +481 -481
  44. package/plugins/file-system-plugin.js +826 -826
  45. package/plugins/install-plugin.js +199 -199
  46. package/plugins/python-executor-plugin.js +367 -367
  47. package/plugins/python-plugin-loader.js +481 -481
  48. package/plugins/rules-plugin.js +294 -294
  49. package/plugins/scheduler-plugin.js +691 -691
  50. package/plugins/session-plugin.js +369 -369
  51. package/plugins/shell-executor-plugin.js +197 -197
  52. package/plugins/storage-plugin.js +240 -240
  53. package/plugins/subagent-plugin.js +845 -845
  54. package/plugins/telegram-plugin.js +482 -482
  55. package/plugins/think-plugin.js +345 -345
  56. package/plugins/tools-plugin.js +196 -196
  57. package/plugins/web-plugin.js +606 -606
  58. package/plugins/weixin-plugin.js +545 -545
  59. package/src/capabilities/index.js +11 -11
  60. package/src/capabilities/skill-manager.js +609 -609
  61. package/src/capabilities/workflow-engine.js +1109 -1109
  62. package/src/core/agent-chat.js +882 -882
  63. package/src/core/agent.js +892 -892
  64. package/src/core/framework.js +465 -465
  65. package/src/core/index.js +19 -19
  66. package/src/core/plugin-base.js +219 -219
  67. package/src/core/plugin-manager.js +863 -863
  68. package/src/core/provider.js +114 -114
  69. package/src/core/sub-agent-config.js +264 -264
  70. package/src/core/system-prompt-builder.js +120 -120
  71. package/src/core/tool-registry.js +517 -517
  72. package/src/core/tool-router.js +297 -297
  73. package/src/executors/executor-base.js +58 -58
  74. package/src/executors/mcp-executor.js +741 -741
  75. package/src/index.js +25 -25
  76. package/src/utils/circuit-breaker.js +301 -301
  77. package/src/utils/error-boundary.js +363 -363
  78. package/src/utils/error.js +374 -374
  79. package/src/utils/event-emitter.js +97 -97
  80. package/src/utils/id.js +133 -133
  81. package/src/utils/index.js +217 -217
  82. package/src/utils/logger.js +181 -181
  83. package/src/utils/plugin-helpers.js +90 -90
  84. package/src/utils/retry.js +122 -122
  85. package/src/utils/sandbox.js +292 -292
  86. package/test/tool-registry-validation.test.js +218 -218
  87. package/website/script.js +136 -136
  88. package/foliko-1.0.75.tgz +0 -0
@@ -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 }