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
@@ -0,0 +1,64 @@
1
+ /**
2
+ * Email 插件 - 常量定义
3
+ */
4
+
5
+ /**
6
+ * IMAP 文件夹列表
7
+ */
8
+ const IMAP_FOLDERS = [
9
+ 'INBOX', // 收件箱
10
+ 'Drafts', // 草稿箱
11
+ 'Sent', // 已发送
12
+ 'Trash', // 垃圾箱
13
+ 'Junk', // 广告邮件
14
+ 'Archive' // 归档
15
+ ]
16
+
17
+ /**
18
+ * 默认 SMTP 配置
19
+ */
20
+ const DEFAULT_SMTP = {
21
+ host: process.env.SMTP_HOST || 'smtp.gmail.com',
22
+ port: parseInt(process.env.SMTP_PORT) || 587,
23
+ secure: process.env.SMTP_SECURE === 'true'
24
+ }
25
+
26
+ /**
27
+ * 默认 IMAP 配置
28
+ */
29
+ const DEFAULT_IMAP = {
30
+ host: process.env.IMAP_HOST || 'imap.gmail.com',
31
+ port: parseInt(process.env.IMAP_PORT) || 993,
32
+ tls: true,
33
+ tlsOptions: { rejectUnauthorized: false }
34
+ }
35
+
36
+ /**
37
+ * IMAP 客户端标识
38
+ */
39
+ const IMAP_CLIENT_INFO = {
40
+ name: process.env.IMAP_CLIENT_NAME || 'FolikoAgent',
41
+ version: process.env.IMAP_CLIENT_VERSION || '1.0.0',
42
+ vendor: process.env.IMAP_CLIENT_VENDOR || 'Foliko',
43
+ supportEmail: process.env.IMAP_CLIENT_SUPPORT_EMAIL || 'unknown@example.com'
44
+ }
45
+
46
+ /**
47
+ * 邮件默认值
48
+ */
49
+ const EMAIL_DEFAULTS = {
50
+ box: 'INBOX',
51
+ limit: 10,
52
+ interval: 60, // 秒
53
+ timeout: 60000, // 毫秒
54
+ recentlyEmailTTL: 5 * 60 * 1000, // 5分钟
55
+ processedEmailTTL: 24 * 60 * 60 * 1000 // 24小时
56
+ }
57
+
58
+ module.exports = {
59
+ IMAP_FOLDERS,
60
+ DEFAULT_SMTP,
61
+ DEFAULT_IMAP,
62
+ IMAP_CLIENT_INFO,
63
+ EMAIL_DEFAULTS
64
+ }
@@ -0,0 +1,461 @@
1
+ /**
2
+ * Email 插件 - 邮件处理器
3
+ */
4
+
5
+ const Imap = require('imap-mkl')
6
+ const { simpleParser } = require('mailparser')
7
+ const { getConfig, processAttachment } = require('./utils')
8
+ const { DEFAULT_IMAP, IMAP_CLIENT_INFO } = require('./constants')
9
+
10
+ /**
11
+ * 创建 IMAP 配置
12
+ * @param {Object} config - 用户配置
13
+ * @returns {Object} IMAP 配置
14
+ */
15
+ function createImapConfig(config = {}) {
16
+ const cfg = getConfig(config)
17
+ return {
18
+ ...DEFAULT_IMAP,
19
+ user: cfg.user,
20
+ password: cfg.password,
21
+ host: cfg.host,
22
+ port: cfg.port,
23
+ id: IMAP_CLIENT_INFO
24
+ }
25
+ }
26
+
27
+ /**
28
+ * 发送邮件
29
+ * @param {Object} emailPlugin - EmailPlugin 实例
30
+ * @param {Object} args - 发送参数
31
+ * @returns {Promise<Object>} 发送结果
32
+ */
33
+ async function sendEmail(emailPlugin, args) {
34
+ try {
35
+ const nodemailer = require('nodemailer')
36
+ const { DEFAULT_SMTP } = require('./constants')
37
+
38
+ const smtpConfig = {
39
+ ...DEFAULT_SMTP,
40
+ auth: {
41
+ user: process.env.SMTP_USER,
42
+ pass: process.env.SMTP_PASS
43
+ }
44
+ }
45
+
46
+ const transporter = nodemailer.createTransport(smtpConfig)
47
+
48
+ const mailOptions = {
49
+ from: process.env.FROM_EMAIL || smtpConfig.auth.user,
50
+ to: args.to,
51
+ subject: args.subject,
52
+ text: args.isHtml ? undefined : args.body,
53
+ html: args.isHtml ? args.body : undefined,
54
+ cc: args.cc,
55
+ bcc: args.bcc,
56
+ inReplyTo: args.inReplyTo,
57
+ references: args.references
58
+ }
59
+
60
+ // 处理附件
61
+ if (args.attachments && args.attachments.length > 0) {
62
+ mailOptions.attachments = []
63
+ for (const att of args.attachments) {
64
+ const processed = await processAttachment(att)
65
+ mailOptions.attachments.push(processed)
66
+ }
67
+ }
68
+
69
+ const info = await transporter.sendMail(mailOptions)
70
+ return {
71
+ success: true,
72
+ message: '邮件发送成功',
73
+ messageId: info.messageId,
74
+ accepted: info.accepted,
75
+ rejected: info.rejected
76
+ }
77
+ } catch (error) {
78
+ return {
79
+ success: false,
80
+ error: error.message,
81
+ details: '请检查 SMTP 配置是否正确'
82
+ }
83
+ }
84
+ }
85
+
86
+ /**
87
+ * 读取邮件
88
+ * @param {Object} emailPlugin - EmailPlugin 实例
89
+ * @param {Object} args - 读取参数
90
+ * @returns {Promise<Object>} 读取结果
91
+ */
92
+ async function readEmails(emailPlugin, args) {
93
+ try {
94
+ const box = args.box || 'INBOX'
95
+ const limit = args.limit || 10
96
+ const unreadOnly = args.unreadOnly || false
97
+
98
+ const imapConfig = createImapConfig(args)
99
+ const emails = await fetchEmails(imapConfig, box, limit, unreadOnly, args.searchCriteria)
100
+
101
+ return {
102
+ success: true,
103
+ count: emails.length,
104
+ emails
105
+ }
106
+ } catch (error) {
107
+ return {
108
+ success: false,
109
+ error: error.message,
110
+ details: '请检查 IMAP 配置是否正确'
111
+ }
112
+ }
113
+ }
114
+
115
+ /**
116
+ * 获取未读邮件数量
117
+ * @param {Object} emailPlugin - EmailPlugin 实例
118
+ * @param {Object} args - 参数
119
+ * @returns {Promise<Object>} 未读数量
120
+ */
121
+ async function getUnreadCount(emailPlugin, args) {
122
+ try {
123
+ const imapConfig = createImapConfig(args)
124
+ const box = args.box || 'INBOX'
125
+ const count = await fetchUnreadCount(imapConfig, box)
126
+
127
+ return {
128
+ success: true,
129
+ unreadCount: count,
130
+ box
131
+ }
132
+ } catch (error) {
133
+ return {
134
+ success: false,
135
+ error: error.message
136
+ }
137
+ }
138
+ }
139
+
140
+ /**
141
+ * 标记邮件为已读
142
+ * @param {Object} emailPlugin - EmailPlugin 实例
143
+ * @param {Object} args - 参数 { uid: 邮件UID }
144
+ * @returns {Promise<Object>} 操作结果
145
+ */
146
+ async function markAsRead(emailPlugin, args) {
147
+ try {
148
+ const imapConfig = createImapConfig(args)
149
+ await markEmailAsRead(imapConfig, args.uid)
150
+
151
+ return {
152
+ success: true,
153
+ message: '邮件已标记为已读'
154
+ }
155
+ } catch (error) {
156
+ return {
157
+ success: false,
158
+ error: error.message
159
+ }
160
+ }
161
+ }
162
+
163
+ /**
164
+ * 删除邮件
165
+ * @param {Object} emailPlugin - EmailPlugin 实例
166
+ * @param {Object} args - 参数 { uid: 邮件UID, box?: 文件夹 }
167
+ * @returns {Promise<Object>} 操作结果
168
+ */
169
+ async function deleteEmail(emailPlugin, args) {
170
+ try {
171
+ const messageId = args.messageId || args.uid
172
+ if (!messageId) {
173
+ return {
174
+ success: false,
175
+ error: 'messageId 是必填参数'
176
+ }
177
+ }
178
+
179
+ const imapConfig = createImapConfig(args)
180
+ const box = args.box || 'INBOX'
181
+ await moveToTrash(imapConfig, box, messageId)
182
+
183
+ return {
184
+ success: true,
185
+ message: '邮件已删除'
186
+ }
187
+ } catch (error) {
188
+ return {
189
+ success: false,
190
+ error: error.message
191
+ }
192
+ }
193
+ }
194
+
195
+ /**
196
+ * 获取邮箱配置信息
197
+ * @returns {Object} 配置信息
198
+ */
199
+ function getEmailConfig() {
200
+ return {
201
+ success: true,
202
+ message: '邮箱配置通过环境变量设置',
203
+ config: {
204
+ smtp_host: process.env.SMTP_HOST || 'smtp.gmail.com',
205
+ smtp_port: process.env.SMTP_PORT || 587,
206
+ smtp_secure: process.env.SMTP_SECURE || 'false',
207
+ imap_host: process.env.IMAP_HOST || 'imap.gmail.com',
208
+ imap_port: process.env.IMAP_PORT || 993,
209
+ from_email: process.env.FROM_EMAIL || '(未设置)',
210
+ client_name: process.env.IMAP_CLIENT_NAME || 'FolikoAgent',
211
+ client_version: process.env.IMAP_CLIENT_VERSION || '1.0.0'
212
+ }
213
+ }
214
+ }
215
+
216
+ // ============ 内部方法 ============
217
+
218
+ /**
219
+ * 获取邮件列表
220
+ */
221
+ function fetchEmails(imapConfig, box, limit, unreadOnly, searchCriteria) {
222
+ return new Promise((resolve, reject) => {
223
+ const imap = new Imap(imapConfig)
224
+ const emails = []
225
+
226
+ const cleanup = () => {
227
+ try { imap.end() } catch (e) {}
228
+ }
229
+
230
+ imap.on('ready', () => {
231
+ imap.openBox(box, true, (err) => {
232
+ if (err) {
233
+ cleanup()
234
+ return reject(err)
235
+ }
236
+
237
+ let searchFilter = searchCriteria
238
+ ? searchCriteria.split(' ').filter(Boolean)
239
+ : ['ALL']
240
+ if (unreadOnly) {
241
+ searchFilter = ['UNSEEN']
242
+ }
243
+
244
+ imap.search(searchFilter, (err, results) => {
245
+ if (err) {
246
+ cleanup()
247
+ return reject(err)
248
+ }
249
+
250
+ if (!results || results.length === 0) {
251
+ cleanup()
252
+ return resolve([])
253
+ }
254
+
255
+ const fetchIds = results.slice(-limit)
256
+ const f = imap.fetch(fetchIds, { bodies: '' })
257
+
258
+ f.on('message', (msg) => {
259
+ let email = {}
260
+ let bodyParsed = false
261
+ let uid = null // 在 attributes 事件中获取 UID
262
+
263
+ // 先获取 UID(比 body 更可靠)
264
+ msg.on('attributes', (attrs) => {
265
+ uid = attrs.uid
266
+ email.uid = uid
267
+ email.id = uid
268
+ email.flags = attrs.flags
269
+ })
270
+
271
+ msg.on('body', (stream) => {
272
+ simpleParser(stream).then(mail => {
273
+ // 确保 uid 已设置(如果 attributes 先触发)
274
+ if (!email.uid && uid) {
275
+ email.uid = uid
276
+ email.id = uid
277
+ }
278
+ email.subject = mail.subject
279
+ email.from = mail.from?.text || ''
280
+ email.to = mail.to?.text || ''
281
+ email.date = mail.date?.toISOString() || ''
282
+ email.text = mail.text || mail.textAsHtml || ''
283
+ email.html = mail.html
284
+ email.attachments = mail.attachments?.map(a => ({
285
+ filename: a.filename,
286
+ contentType: a.contentType
287
+ })) || []
288
+ bodyParsed = true
289
+ }).catch(err => {
290
+ email.error = err.message
291
+ bodyParsed = true
292
+ })
293
+ })
294
+
295
+ msg.on('end', () => {
296
+ const checkDone = () => {
297
+ if (bodyParsed) {
298
+ // 再次确保 uid 已设置
299
+ if (!email.uid && uid) {
300
+ email.uid = uid
301
+ email.id = uid
302
+ }
303
+ emails.push(email)
304
+ } else {
305
+ setTimeout(checkDone, 10)
306
+ }
307
+ }
308
+ checkDone()
309
+ })
310
+ })
311
+
312
+ f.on('error', (err) => {
313
+ cleanup()
314
+ reject(err)
315
+ })
316
+
317
+ f.on('end', () => {
318
+ // 添加超时保护,避免永久等待
319
+ const timeoutMs = 30000
320
+ const startTime = Date.now()
321
+
322
+ const waitForAll = () => {
323
+ if (emails.length === fetchIds.length) {
324
+ cleanup()
325
+ resolve(emails)
326
+ } else if (Date.now() - startTime > timeoutMs) {
327
+ cleanup()
328
+ resolve(emails) // 返回已获取的邮件
329
+ } else {
330
+ setTimeout(waitForAll, 50)
331
+ }
332
+ }
333
+ waitForAll()
334
+ })
335
+ })
336
+ })
337
+ })
338
+
339
+ imap.on('error', (err) => reject(err))
340
+ imap.on('end', () => {})
341
+
342
+ imap.connect()
343
+ })
344
+ }
345
+
346
+ /**
347
+ * 获取未读邮件数量
348
+ */
349
+ function fetchUnreadCount(imapConfig, box) {
350
+ return new Promise((resolve, reject) => {
351
+ const imap = new Imap(imapConfig)
352
+
353
+ const cleanup = () => {
354
+ try { imap.end() } catch (e) {}
355
+ }
356
+
357
+ imap.on('ready', () => {
358
+ imap.openBox(box, true, (err, boxObj) => {
359
+ if (err) {
360
+ cleanup()
361
+ return reject(err)
362
+ }
363
+ const unreadCount = boxObj.messages.unread
364
+ cleanup()
365
+ resolve(unreadCount)
366
+ })
367
+ })
368
+
369
+ imap.on('error', (err) => reject(err))
370
+ imap.on('end', () => {})
371
+
372
+ imap.connect()
373
+ })
374
+ }
375
+
376
+ /**
377
+ * 标记邮件为已读
378
+ * @param {Object} imapConfig - IMAP 配置
379
+ * @param {number|string} uid - 邮件 UID
380
+ */
381
+ function markEmailAsRead(imapConfig, uid) {
382
+ return new Promise((resolve, reject) => {
383
+ const imap = new Imap(imapConfig)
384
+
385
+ const cleanup = () => {
386
+ try { imap.end() } catch (e) {}
387
+ }
388
+
389
+ imap.on('ready', () => {
390
+ imap.openBox('INBOX', true, (err) => {
391
+ if (err) {
392
+ cleanup()
393
+ return reject(err)
394
+ }
395
+ imap.addFlags(uid, '\\Seen', (err) => {
396
+ cleanup()
397
+ if (err) reject(err)
398
+ else resolve()
399
+ })
400
+ })
401
+ })
402
+
403
+ imap.on('error', (err) => reject(err))
404
+ imap.on('end', () => {})
405
+
406
+ imap.connect()
407
+ })
408
+ }
409
+
410
+ /**
411
+ * 移动邮件到垃圾箱
412
+ * @param {Object} imapConfig - IMAP 配置
413
+ * @param {string} box - 邮箱文件夹
414
+ * @param {number|string} uid - 邮件 UID
415
+ */
416
+ function moveToTrash(imapConfig, box, uid) {
417
+ return new Promise((resolve, reject) => {
418
+ const imap = new Imap(imapConfig)
419
+
420
+ const cleanup = () => {
421
+ try { imap.end() } catch (e) {}
422
+ }
423
+
424
+ imap.on('ready', () => {
425
+ imap.openBox(box, true, (err) => {
426
+ if (err) {
427
+ cleanup()
428
+ return reject(err)
429
+ }
430
+ // 标记邮件为已删除
431
+ imap.addFlags(uid, '\\Deleted', (err) => {
432
+ if (err) {
433
+ cleanup()
434
+ return reject(err)
435
+ }
436
+ // 执行 expunge 永久删除
437
+ imap.expunge((err) => {
438
+ cleanup()
439
+ if (err) reject(err)
440
+ else resolve()
441
+ })
442
+ })
443
+ })
444
+ })
445
+
446
+ imap.on('error', (err) => reject(err))
447
+ imap.on('end', () => {})
448
+
449
+ imap.connect()
450
+ })
451
+ }
452
+
453
+ module.exports = {
454
+ sendEmail,
455
+ readEmails,
456
+ getUnreadCount,
457
+ markAsRead,
458
+ deleteEmail,
459
+ getEmailConfig,
460
+ createImapConfig
461
+ }