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