@n24q02m/better-email-mcp 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (97) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +162 -0
  3. package/bin/cli.mjs +13 -0
  4. package/build/scripts/start-server.d.ts +6 -0
  5. package/build/scripts/start-server.d.ts.map +1 -0
  6. package/build/scripts/start-server.js +21 -0
  7. package/build/scripts/start-server.js.map +1 -0
  8. package/build/src/docs/attachments.md +48 -0
  9. package/build/src/docs/folders.md +45 -0
  10. package/build/src/docs/messages.md +92 -0
  11. package/build/src/docs/send.md +52 -0
  12. package/build/src/init-server.d.ts +40 -0
  13. package/build/src/init-server.d.ts.map +1 -0
  14. package/build/src/init-server.js +57 -0
  15. package/build/src/init-server.js.map +1 -0
  16. package/build/src/tools/composite/attachments.d.ts +17 -0
  17. package/build/src/tools/composite/attachments.d.ts.map +1 -0
  18. package/build/src/tools/composite/attachments.js +77 -0
  19. package/build/src/tools/composite/attachments.js.map +1 -0
  20. package/build/src/tools/composite/attachments.test.d.ts +2 -0
  21. package/build/src/tools/composite/attachments.test.d.ts.map +1 -0
  22. package/build/src/tools/composite/attachments.test.js +138 -0
  23. package/build/src/tools/composite/attachments.test.js.map +1 -0
  24. package/build/src/tools/composite/folders.d.ts +14 -0
  25. package/build/src/tools/composite/folders.d.ts.map +1 -0
  26. package/build/src/tools/composite/folders.js +57 -0
  27. package/build/src/tools/composite/folders.js.map +1 -0
  28. package/build/src/tools/composite/folders.test.d.ts +2 -0
  29. package/build/src/tools/composite/folders.test.d.ts.map +1 -0
  30. package/build/src/tools/composite/folders.test.js +61 -0
  31. package/build/src/tools/composite/folders.test.js.map +1 -0
  32. package/build/src/tools/composite/messages.d.ts +20 -0
  33. package/build/src/tools/composite/messages.d.ts.map +1 -0
  34. package/build/src/tools/composite/messages.js +244 -0
  35. package/build/src/tools/composite/messages.js.map +1 -0
  36. package/build/src/tools/composite/messages.test.d.ts +2 -0
  37. package/build/src/tools/composite/messages.test.d.ts.map +1 -0
  38. package/build/src/tools/composite/messages.test.js +217 -0
  39. package/build/src/tools/composite/messages.test.js.map +1 -0
  40. package/build/src/tools/composite/send.d.ts +21 -0
  41. package/build/src/tools/composite/send.d.ts.map +1 -0
  42. package/build/src/tools/composite/send.js +127 -0
  43. package/build/src/tools/composite/send.js.map +1 -0
  44. package/build/src/tools/composite/send.test.d.ts +2 -0
  45. package/build/src/tools/composite/send.test.d.ts.map +1 -0
  46. package/build/src/tools/composite/send.test.js +232 -0
  47. package/build/src/tools/composite/send.test.js.map +1 -0
  48. package/build/src/tools/helpers/config.d.ts +33 -0
  49. package/build/src/tools/helpers/config.d.ts.map +1 -0
  50. package/build/src/tools/helpers/config.js +167 -0
  51. package/build/src/tools/helpers/config.js.map +1 -0
  52. package/build/src/tools/helpers/config.test.d.ts +2 -0
  53. package/build/src/tools/helpers/config.test.d.ts.map +1 -0
  54. package/build/src/tools/helpers/config.test.js +141 -0
  55. package/build/src/tools/helpers/config.test.js.map +1 -0
  56. package/build/src/tools/helpers/errors.d.ts +34 -0
  57. package/build/src/tools/helpers/errors.d.ts.map +1 -0
  58. package/build/src/tools/helpers/errors.js +152 -0
  59. package/build/src/tools/helpers/errors.js.map +1 -0
  60. package/build/src/tools/helpers/errors.test.d.ts +2 -0
  61. package/build/src/tools/helpers/errors.test.d.ts.map +1 -0
  62. package/build/src/tools/helpers/errors.test.js +203 -0
  63. package/build/src/tools/helpers/errors.test.js.map +1 -0
  64. package/build/src/tools/helpers/html-utils.d.ts +10 -0
  65. package/build/src/tools/helpers/html-utils.d.ts.map +1 -0
  66. package/build/src/tools/helpers/html-utils.js +29 -0
  67. package/build/src/tools/helpers/html-utils.js.map +1 -0
  68. package/build/src/tools/helpers/html-utils.test.d.ts +2 -0
  69. package/build/src/tools/helpers/html-utils.test.d.ts.map +1 -0
  70. package/build/src/tools/helpers/html-utils.test.js +103 -0
  71. package/build/src/tools/helpers/html-utils.test.js.map +1 -0
  72. package/build/src/tools/helpers/imap-client.d.ts +90 -0
  73. package/build/src/tools/helpers/imap-client.d.ts.map +1 -0
  74. package/build/src/tools/helpers/imap-client.js +321 -0
  75. package/build/src/tools/helpers/imap-client.js.map +1 -0
  76. package/build/src/tools/helpers/imap-client.test.d.ts +2 -0
  77. package/build/src/tools/helpers/imap-client.test.d.ts.map +1 -0
  78. package/build/src/tools/helpers/imap-client.test.js +401 -0
  79. package/build/src/tools/helpers/imap-client.test.js.map +1 -0
  80. package/build/src/tools/helpers/smtp-client.d.ts +38 -0
  81. package/build/src/tools/helpers/smtp-client.d.ts.map +1 -0
  82. package/build/src/tools/helpers/smtp-client.js +124 -0
  83. package/build/src/tools/helpers/smtp-client.js.map +1 -0
  84. package/build/src/tools/helpers/smtp-client.test.d.ts +2 -0
  85. package/build/src/tools/helpers/smtp-client.test.d.ts.map +1 -0
  86. package/build/src/tools/helpers/smtp-client.test.js +210 -0
  87. package/build/src/tools/helpers/smtp-client.test.js.map +1 -0
  88. package/build/src/tools/registry.d.ts +11 -0
  89. package/build/src/tools/registry.d.ts.map +1 -0
  90. package/build/src/tools/registry.js +262 -0
  91. package/build/src/tools/registry.js.map +1 -0
  92. package/build/src/tools/registry.test.d.ts +2 -0
  93. package/build/src/tools/registry.test.d.ts.map +1 -0
  94. package/build/src/tools/registry.test.js +135 -0
  95. package/build/src/tools/registry.test.js.map +1 -0
  96. package/build/tsconfig.tsbuildinfo +1 -0
  97. package/package.json +81 -0
@@ -0,0 +1 @@
1
+ {"version":3,"file":"send.js","sourceRoot":"","sources":["../../../../src/tools/composite/send.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=send.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"send.test.d.ts","sourceRoot":"","sources":["../../../../src/tools/composite/send.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,232 @@
1
+ import { beforeEach, describe, expect, it, vi } from 'vitest';
2
+ // --- Mocks ---
3
+ vi.mock('../helpers/imap-client.js', () => ({
4
+ readEmail: vi.fn()
5
+ }));
6
+ vi.mock('../helpers/smtp-client.js', () => ({
7
+ sendNewEmail: vi.fn(),
8
+ replyToEmail: vi.fn(),
9
+ forwardEmail: vi.fn()
10
+ }));
11
+ import { readEmail } from '../helpers/imap-client.js';
12
+ import { forwardEmail, replyToEmail, sendNewEmail } from '../helpers/smtp-client.js';
13
+ import { send } from './send.js';
14
+ const mockReadEmail = vi.mocked(readEmail);
15
+ const mockSendNewEmail = vi.mocked(sendNewEmail);
16
+ const mockReplyToEmail = vi.mocked(replyToEmail);
17
+ const mockForwardEmail = vi.mocked(forwardEmail);
18
+ const accounts = [
19
+ {
20
+ id: 'user1_gmail_com',
21
+ email: 'user1@gmail.com',
22
+ password: 'pass1',
23
+ imap: { host: 'imap.gmail.com', port: 993, secure: true },
24
+ smtp: { host: 'smtp.gmail.com', port: 465, secure: true }
25
+ }
26
+ ];
27
+ beforeEach(() => {
28
+ vi.clearAllMocks();
29
+ });
30
+ // ============================================================================
31
+ // new
32
+ // ============================================================================
33
+ describe('send - new', () => {
34
+ it('sends a new email', async () => {
35
+ mockSendNewEmail.mockResolvedValue({ success: true, message_id: '<new123@gmail.com>' });
36
+ const result = await send(accounts, {
37
+ action: 'new',
38
+ account: 'user1@gmail.com',
39
+ to: 'recipient@test.com',
40
+ subject: 'Hello',
41
+ body: 'World'
42
+ });
43
+ expect(result.action).toBe('new');
44
+ expect(result.from).toBe('user1@gmail.com');
45
+ expect(result.to).toBe('recipient@test.com');
46
+ expect(result.success).toBe(true);
47
+ expect(mockSendNewEmail).toHaveBeenCalledWith(accounts[0], expect.objectContaining({ to: 'recipient@test.com', subject: 'Hello', body: 'World' }));
48
+ });
49
+ it('throws when subject is missing for new email', async () => {
50
+ await expect(send(accounts, {
51
+ action: 'new',
52
+ account: 'user1@gmail.com',
53
+ to: 'r@test.com',
54
+ subject: '',
55
+ body: 'text'
56
+ })).rejects.toThrow('subject is required');
57
+ });
58
+ it('passes cc and bcc', async () => {
59
+ mockSendNewEmail.mockResolvedValue({ success: true, message_id: '<id>' });
60
+ await send(accounts, {
61
+ action: 'new',
62
+ account: 'user1@gmail.com',
63
+ to: 'r@test.com',
64
+ subject: 'T',
65
+ body: 'B',
66
+ cc: 'cc@test.com',
67
+ bcc: 'bcc@test.com'
68
+ });
69
+ expect(mockSendNewEmail).toHaveBeenCalledWith(accounts[0], expect.objectContaining({ cc: 'cc@test.com', bcc: 'bcc@test.com' }));
70
+ });
71
+ });
72
+ // ============================================================================
73
+ // reply
74
+ // ============================================================================
75
+ describe('send - reply', () => {
76
+ it('replies to an email with thread headers', async () => {
77
+ mockReadEmail.mockResolvedValue({
78
+ account_id: 'user1_gmail_com',
79
+ account_email: 'user1@gmail.com',
80
+ uid: 42,
81
+ message_id: '<original@test>',
82
+ references: '<ref1@test>',
83
+ subject: 'Original Subject',
84
+ from: 'sender@test.com',
85
+ to: 'user1@gmail.com',
86
+ date: '2025-01-01',
87
+ flags: [],
88
+ body_text: 'original body',
89
+ attachments: []
90
+ });
91
+ mockReplyToEmail.mockResolvedValue({ success: true, message_id: '<reply123@gmail.com>' });
92
+ const result = await send(accounts, {
93
+ action: 'reply',
94
+ account: 'user1@gmail.com',
95
+ to: 'sender@test.com',
96
+ subject: '',
97
+ body: 'My reply',
98
+ uid: 42
99
+ });
100
+ expect(result.action).toBe('reply');
101
+ expect(result.in_reply_to).toBe('<original@test>');
102
+ expect(mockReplyToEmail).toHaveBeenCalledWith(accounts[0], expect.objectContaining({
103
+ in_reply_to: '<original@test>',
104
+ references: '<ref1@test>'
105
+ }));
106
+ });
107
+ it('throws when uid is missing for reply', async () => {
108
+ await expect(send(accounts, {
109
+ action: 'reply',
110
+ account: 'user1@gmail.com',
111
+ to: 'x@test.com',
112
+ subject: 'T',
113
+ body: 'B'
114
+ // no uid
115
+ })).rejects.toThrow('uid is required');
116
+ });
117
+ it('uses original subject when no subject provided', async () => {
118
+ mockReadEmail.mockResolvedValue({
119
+ account_id: 'user1_gmail_com',
120
+ account_email: 'user1@gmail.com',
121
+ uid: 1,
122
+ message_id: '<msg@test>',
123
+ subject: 'Original',
124
+ from: 'x@test.com',
125
+ to: 'y@test.com',
126
+ date: '2025-01-01',
127
+ flags: [],
128
+ body_text: 'body',
129
+ attachments: []
130
+ });
131
+ mockReplyToEmail.mockResolvedValue({ success: true, message_id: '<r@test>' });
132
+ const result = await send(accounts, {
133
+ action: 'reply',
134
+ account: 'user1@gmail.com',
135
+ to: 'x@test.com',
136
+ subject: '',
137
+ body: 'reply',
138
+ uid: 1
139
+ });
140
+ expect(result.subject).toBe('Re: Original');
141
+ });
142
+ });
143
+ // ============================================================================
144
+ // forward
145
+ // ============================================================================
146
+ describe('send - forward', () => {
147
+ it('forwards email with original body', async () => {
148
+ mockReadEmail.mockResolvedValue({
149
+ account_id: 'user1_gmail_com',
150
+ account_email: 'user1@gmail.com',
151
+ uid: 50,
152
+ subject: 'FW Subject',
153
+ from: 'sender@test.com',
154
+ to: 'user1@gmail.com',
155
+ date: '2025-01-01',
156
+ flags: [],
157
+ body_text: 'original content',
158
+ attachments: []
159
+ });
160
+ mockForwardEmail.mockResolvedValue({ success: true, message_id: '<fwd123@gmail.com>' });
161
+ const result = await send(accounts, {
162
+ action: 'forward',
163
+ account: 'user1@gmail.com',
164
+ to: 'third@test.com',
165
+ subject: '',
166
+ body: 'Check this',
167
+ uid: 50
168
+ });
169
+ expect(result.action).toBe('forward');
170
+ expect(mockForwardEmail).toHaveBeenCalledWith(accounts[0], expect.objectContaining({ original_body: 'original content' }));
171
+ });
172
+ it('throws when uid is missing for forward', async () => {
173
+ await expect(send(accounts, {
174
+ action: 'forward',
175
+ account: 'user1@gmail.com',
176
+ to: 'x@test.com',
177
+ subject: 'T',
178
+ body: 'B'
179
+ })).rejects.toThrow('uid is required');
180
+ });
181
+ });
182
+ // ============================================================================
183
+ // validation
184
+ // ============================================================================
185
+ describe('send - validation', () => {
186
+ it('throws when account is missing', async () => {
187
+ await expect(send(accounts, {
188
+ action: 'new',
189
+ account: '',
190
+ to: 'x@test.com',
191
+ subject: 'T',
192
+ body: 'B'
193
+ })).rejects.toThrow();
194
+ });
195
+ it('throws when to is missing', async () => {
196
+ await expect(send(accounts, {
197
+ action: 'new',
198
+ account: 'user1@gmail.com',
199
+ to: '',
200
+ subject: 'T',
201
+ body: 'B'
202
+ })).rejects.toThrow();
203
+ });
204
+ it('throws when body is missing', async () => {
205
+ await expect(send(accounts, {
206
+ action: 'new',
207
+ account: 'user1@gmail.com',
208
+ to: 'x@test.com',
209
+ subject: 'T',
210
+ body: ''
211
+ })).rejects.toThrow();
212
+ });
213
+ it('throws when account not found', async () => {
214
+ await expect(send(accounts, {
215
+ action: 'new',
216
+ account: 'unknown@test.com',
217
+ to: 'x@test.com',
218
+ subject: 'T',
219
+ body: 'B'
220
+ })).rejects.toThrow('Account not found');
221
+ });
222
+ it('throws for unknown action', async () => {
223
+ await expect(send(accounts, {
224
+ action: 'unknown',
225
+ account: 'user1@gmail.com',
226
+ to: 'x@test.com',
227
+ subject: 'T',
228
+ body: 'B'
229
+ })).rejects.toThrow();
230
+ });
231
+ });
232
+ //# sourceMappingURL=send.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"send.test.js","sourceRoot":"","sources":["../../../../src/tools/composite/send.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,33 @@
1
+ /**
2
+ * Configuration Parser
3
+ * Parses EMAIL_CREDENTIALS env var and auto-discovers IMAP/SMTP settings
4
+ */
5
+ export interface ServerConfig {
6
+ host: string;
7
+ port: number;
8
+ secure: boolean;
9
+ }
10
+ export interface AccountConfig {
11
+ id: string;
12
+ email: string;
13
+ password: string;
14
+ imap: ServerConfig;
15
+ smtp: ServerConfig;
16
+ }
17
+ /**
18
+ * Parse EMAIL_CREDENTIALS environment variable
19
+ *
20
+ * Supported formats:
21
+ * - Simple: email1:password1,email2:password2
22
+ * - With custom IMAP host: email1:password1:imap.custom.com,email2:password2
23
+ * - Passwords with commas are NOT supported (use env var per account instead)
24
+ *
25
+ * For passwords containing colons, use the 3-field format to disambiguate:
26
+ * email:password_with:colon:imap_host
27
+ */
28
+ export declare function parseCredentials(envValue: string): AccountConfig[];
29
+ /**
30
+ * Load and validate configuration from environment
31
+ */
32
+ export declare function loadConfig(): AccountConfig[];
33
+ //# sourceMappingURL=config.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"config.d.ts","sourceRoot":"","sources":["../../../../src/tools/helpers/config.ts"],"names":[],"mappings":"AAAA;;;GAGG;AA8CH,MAAM,WAAW,YAAY;IAC3B,IAAI,EAAE,MAAM,CAAA;IACZ,IAAI,EAAE,MAAM,CAAA;IACZ,MAAM,EAAE,OAAO,CAAA;CAChB;AAED,MAAM,WAAW,aAAa;IAC5B,EAAE,EAAE,MAAM,CAAA;IACV,KAAK,EAAE,MAAM,CAAA;IACb,QAAQ,EAAE,MAAM,CAAA;IAChB,IAAI,EAAE,YAAY,CAAA;IAClB,IAAI,EAAE,YAAY,CAAA;CACnB;AA+BD;;;;;;;;;;GAUG;AACH,wBAAgB,gBAAgB,CAAC,QAAQ,EAAE,MAAM,GAAG,aAAa,EAAE,CA2ElE;AAED;;GAEG;AACH,wBAAgB,UAAU,IAAI,aAAa,EAAE,CAM5C"}
@@ -0,0 +1,167 @@
1
+ /**
2
+ * Configuration Parser
3
+ * Parses EMAIL_CREDENTIALS env var and auto-discovers IMAP/SMTP settings
4
+ */
5
+ /** Well-known email provider settings */
6
+ const PROVIDER_MAP = {
7
+ 'gmail.com': {
8
+ imap: { host: 'imap.gmail.com', port: 993, secure: true },
9
+ smtp: { host: 'smtp.gmail.com', port: 465, secure: true }
10
+ },
11
+ 'googlemail.com': {
12
+ imap: { host: 'imap.gmail.com', port: 993, secure: true },
13
+ smtp: { host: 'smtp.gmail.com', port: 465, secure: true }
14
+ },
15
+ 'outlook.com': {
16
+ imap: { host: 'outlook.office365.com', port: 993, secure: true },
17
+ smtp: { host: 'smtp.office365.com', port: 587, secure: false }
18
+ },
19
+ 'hotmail.com': {
20
+ imap: { host: 'outlook.office365.com', port: 993, secure: true },
21
+ smtp: { host: 'smtp.office365.com', port: 587, secure: false }
22
+ },
23
+ 'live.com': {
24
+ imap: { host: 'outlook.office365.com', port: 993, secure: true },
25
+ smtp: { host: 'smtp.office365.com', port: 587, secure: false }
26
+ },
27
+ 'yahoo.com': {
28
+ imap: { host: 'imap.mail.yahoo.com', port: 993, secure: true },
29
+ smtp: { host: 'smtp.mail.yahoo.com', port: 465, secure: true }
30
+ },
31
+ 'icloud.com': {
32
+ imap: { host: 'imap.mail.me.com', port: 993, secure: true },
33
+ smtp: { host: 'smtp.mail.me.com', port: 587, secure: false }
34
+ },
35
+ 'me.com': {
36
+ imap: { host: 'imap.mail.me.com', port: 993, secure: true },
37
+ smtp: { host: 'smtp.mail.me.com', port: 587, secure: false }
38
+ },
39
+ 'zoho.com': {
40
+ imap: { host: 'imap.zoho.com', port: 993, secure: true },
41
+ smtp: { host: 'smtp.zoho.com', port: 465, secure: true }
42
+ },
43
+ 'protonmail.com': {
44
+ imap: { host: 'imap.protonmail.ch', port: 993, secure: true },
45
+ smtp: { host: 'smtp.protonmail.ch', port: 465, secure: true }
46
+ }
47
+ };
48
+ /**
49
+ * Auto-discover IMAP/SMTP settings from email domain
50
+ */
51
+ function discoverSettings(email) {
52
+ const domain = email.split('@')[1]?.toLowerCase();
53
+ if (!domain)
54
+ return null;
55
+ // Check exact domain match
56
+ if (PROVIDER_MAP[domain]) {
57
+ return PROVIDER_MAP[domain];
58
+ }
59
+ // Check if subdomain matches (e.g. user@work.gmail.com)
60
+ for (const [providerDomain, settings] of Object.entries(PROVIDER_MAP)) {
61
+ if (domain.endsWith(`.${providerDomain}`)) {
62
+ return settings;
63
+ }
64
+ }
65
+ return null;
66
+ }
67
+ /**
68
+ * Generate a safe ID from email address
69
+ */
70
+ function emailToId(email) {
71
+ return email.replace(/[@.]/g, '_').toLowerCase();
72
+ }
73
+ /**
74
+ * Parse EMAIL_CREDENTIALS environment variable
75
+ *
76
+ * Supported formats:
77
+ * - Simple: email1:password1,email2:password2
78
+ * - With custom IMAP host: email1:password1:imap.custom.com,email2:password2
79
+ * - Passwords with commas are NOT supported (use env var per account instead)
80
+ *
81
+ * For passwords containing colons, use the 3-field format to disambiguate:
82
+ * email:password_with:colon:imap_host
83
+ */
84
+ export function parseCredentials(envValue) {
85
+ if (!envValue || envValue.trim() === '') {
86
+ return [];
87
+ }
88
+ const accounts = [];
89
+ const entries = envValue.split(',');
90
+ for (const entry of entries) {
91
+ const trimmed = entry.trim();
92
+ if (!trimmed)
93
+ continue;
94
+ const parts = trimmed.split(':');
95
+ if (parts.length < 2) {
96
+ console.error(`Skipping invalid credential entry (expected email:password): ${trimmed.substring(0, 20)}...`);
97
+ continue;
98
+ }
99
+ const email = parts[0].trim();
100
+ let password;
101
+ let customImapHost;
102
+ if (parts.length === 2) {
103
+ // email:password
104
+ password = parts[1];
105
+ }
106
+ else if (parts.length === 3) {
107
+ // Could be email:password:imap_host OR email:password_with_colon
108
+ // Heuristic: if last part looks like a hostname (contains a dot), treat as imap host
109
+ const lastPart = parts[2];
110
+ if (lastPart.includes('.')) {
111
+ password = parts[1];
112
+ customImapHost = lastPart;
113
+ }
114
+ else {
115
+ // password contains a colon
116
+ password = `${parts[1]}:${parts[2]}`;
117
+ }
118
+ }
119
+ else {
120
+ // 4+ parts: everything between email and last part (if hostname) is password
121
+ const lastPart = parts[parts.length - 1];
122
+ if (lastPart.includes('.')) {
123
+ password = parts.slice(1, -1).join(':');
124
+ customImapHost = lastPart;
125
+ }
126
+ else {
127
+ password = parts.slice(1).join(':');
128
+ }
129
+ }
130
+ // Auto-discover or use custom host
131
+ let imap;
132
+ let smtp;
133
+ if (customImapHost) {
134
+ imap = { host: customImapHost, port: 993, secure: true };
135
+ // Guess SMTP from IMAP host
136
+ smtp = { host: customImapHost.replace('imap.', 'smtp.'), port: 587, secure: false };
137
+ }
138
+ else {
139
+ const discovered = discoverSettings(email);
140
+ if (!discovered) {
141
+ console.error(`Cannot auto-discover settings for ${email}. Use format: email:password:imap.server.com`);
142
+ continue;
143
+ }
144
+ imap = discovered.imap;
145
+ smtp = discovered.smtp;
146
+ }
147
+ accounts.push({
148
+ id: emailToId(email),
149
+ email,
150
+ password,
151
+ imap,
152
+ smtp
153
+ });
154
+ }
155
+ return accounts;
156
+ }
157
+ /**
158
+ * Load and validate configuration from environment
159
+ */
160
+ export function loadConfig() {
161
+ const credentials = process.env.EMAIL_CREDENTIALS;
162
+ if (!credentials) {
163
+ return [];
164
+ }
165
+ return parseCredentials(credentials);
166
+ }
167
+ //# sourceMappingURL=config.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"config.js","sourceRoot":"","sources":["../../../../src/tools/helpers/config.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=config.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"config.test.d.ts","sourceRoot":"","sources":["../../../../src/tools/helpers/config.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,141 @@
1
+ import { describe, expect, it, vi } from 'vitest';
2
+ import { loadConfig, parseCredentials } from './config.js';
3
+ describe('parseCredentials', () => {
4
+ it('returns empty array for empty string', () => {
5
+ expect(parseCredentials('')).toEqual([]);
6
+ });
7
+ it('returns empty array for whitespace-only string', () => {
8
+ expect(parseCredentials(' ')).toEqual([]);
9
+ });
10
+ it('parses single Gmail account', () => {
11
+ const result = parseCredentials('user@gmail.com:mypassword');
12
+ expect(result).toHaveLength(1);
13
+ expect(result[0]).toMatchObject({
14
+ id: 'user_gmail_com',
15
+ email: 'user@gmail.com',
16
+ password: 'mypassword',
17
+ imap: { host: 'imap.gmail.com', port: 993, secure: true },
18
+ smtp: { host: 'smtp.gmail.com', port: 465, secure: true }
19
+ });
20
+ });
21
+ it('parses multiple accounts separated by comma', () => {
22
+ const result = parseCredentials('user1@gmail.com:pass1,user2@outlook.com:pass2');
23
+ expect(result).toHaveLength(2);
24
+ expect(result[0].email).toBe('user1@gmail.com');
25
+ expect(result[1].email).toBe('user2@outlook.com');
26
+ });
27
+ it('auto-discovers Outlook settings', () => {
28
+ const result = parseCredentials('user@outlook.com:pass');
29
+ expect(result[0].imap.host).toBe('outlook.office365.com');
30
+ expect(result[0].smtp.host).toBe('smtp.office365.com');
31
+ expect(result[0].smtp.port).toBe(587);
32
+ expect(result[0].smtp.secure).toBe(false);
33
+ });
34
+ it('auto-discovers Hotmail as Outlook', () => {
35
+ const result = parseCredentials('user@hotmail.com:pass');
36
+ expect(result[0].imap.host).toBe('outlook.office365.com');
37
+ });
38
+ it('auto-discovers Live as Outlook', () => {
39
+ const result = parseCredentials('user@live.com:pass');
40
+ expect(result[0].imap.host).toBe('outlook.office365.com');
41
+ });
42
+ it('auto-discovers Yahoo settings', () => {
43
+ const result = parseCredentials('user@yahoo.com:pass');
44
+ expect(result[0].imap.host).toBe('imap.mail.yahoo.com');
45
+ expect(result[0].smtp.host).toBe('smtp.mail.yahoo.com');
46
+ });
47
+ it('auto-discovers iCloud settings', () => {
48
+ const result = parseCredentials('user@icloud.com:pass');
49
+ expect(result[0].imap.host).toBe('imap.mail.me.com');
50
+ expect(result[0].smtp.host).toBe('smtp.mail.me.com');
51
+ });
52
+ it('auto-discovers me.com as iCloud', () => {
53
+ const result = parseCredentials('user@me.com:pass');
54
+ expect(result[0].imap.host).toBe('imap.mail.me.com');
55
+ });
56
+ it('auto-discovers Zoho settings', () => {
57
+ const result = parseCredentials('user@zoho.com:pass');
58
+ expect(result[0].imap.host).toBe('imap.zoho.com');
59
+ });
60
+ it('auto-discovers ProtonMail settings', () => {
61
+ const result = parseCredentials('user@protonmail.com:pass');
62
+ expect(result[0].imap.host).toBe('imap.protonmail.ch');
63
+ });
64
+ it('auto-discovers googlemail.com as Gmail', () => {
65
+ const result = parseCredentials('user@googlemail.com:pass');
66
+ expect(result[0].imap.host).toBe('imap.gmail.com');
67
+ });
68
+ it('handles custom IMAP host (3-part format with hostname)', () => {
69
+ const result = parseCredentials('user@custom.com:mypass:imap.custom.com');
70
+ expect(result).toHaveLength(1);
71
+ expect(result[0].imap.host).toBe('imap.custom.com');
72
+ expect(result[0].smtp.host).toBe('smtp.custom.com');
73
+ expect(result[0].password).toBe('mypass');
74
+ });
75
+ it('handles password with colon (3-part format without hostname)', () => {
76
+ const result = parseCredentials('user@gmail.com:pass:word');
77
+ expect(result).toHaveLength(1);
78
+ expect(result[0].password).toBe('pass:word');
79
+ expect(result[0].imap.host).toBe('imap.gmail.com');
80
+ });
81
+ it('handles password with multiple colons and custom host', () => {
82
+ const result = parseCredentials('user@custom.com:pass:with:colons:imap.server.com');
83
+ expect(result).toHaveLength(1);
84
+ expect(result[0].password).toBe('pass:with:colons');
85
+ expect(result[0].imap.host).toBe('imap.server.com');
86
+ });
87
+ it('handles password with multiple colons and no custom host', () => {
88
+ const result = parseCredentials('user@gmail.com:pass:with:colons:nohostname');
89
+ expect(result).toHaveLength(1);
90
+ expect(result[0].password).toBe('pass:with:colons:nohostname');
91
+ });
92
+ it('skips invalid entries with only one part', () => {
93
+ const spy = vi.spyOn(console, 'error').mockImplementation(() => { });
94
+ const result = parseCredentials('justanemail');
95
+ expect(result).toHaveLength(0);
96
+ spy.mockRestore();
97
+ });
98
+ it('skips unknown domains without custom host', () => {
99
+ const spy = vi.spyOn(console, 'error').mockImplementation(() => { });
100
+ const result = parseCredentials('user@unknowndomain.xyz:password');
101
+ expect(result).toHaveLength(0);
102
+ spy.mockRestore();
103
+ });
104
+ it('trims whitespace around entries', () => {
105
+ const result = parseCredentials(' user@gmail.com:pass , user2@yahoo.com:pass2 ');
106
+ expect(result).toHaveLength(2);
107
+ expect(result[0].email).toBe('user@gmail.com');
108
+ expect(result[1].email).toBe('user2@yahoo.com');
109
+ });
110
+ it('skips empty entries between commas', () => {
111
+ const result = parseCredentials('user@gmail.com:pass,,user2@yahoo.com:pass2');
112
+ expect(result).toHaveLength(2);
113
+ });
114
+ it('generates correct ID from email', () => {
115
+ const result = parseCredentials('My.User@gmail.com:pass');
116
+ expect(result[0].id).toBe('my_user_gmail_com');
117
+ });
118
+ });
119
+ describe('loadConfig', () => {
120
+ it('returns empty array when EMAIL_CREDENTIALS is not set', () => {
121
+ const original = process.env.EMAIL_CREDENTIALS;
122
+ delete process.env.EMAIL_CREDENTIALS;
123
+ expect(loadConfig()).toEqual([]);
124
+ if (original)
125
+ process.env.EMAIL_CREDENTIALS = original;
126
+ });
127
+ it('parses EMAIL_CREDENTIALS from environment', () => {
128
+ const original = process.env.EMAIL_CREDENTIALS;
129
+ process.env.EMAIL_CREDENTIALS = 'test@gmail.com:testpass';
130
+ const result = loadConfig();
131
+ expect(result).toHaveLength(1);
132
+ expect(result[0].email).toBe('test@gmail.com');
133
+ if (original) {
134
+ process.env.EMAIL_CREDENTIALS = original;
135
+ }
136
+ else {
137
+ delete process.env.EMAIL_CREDENTIALS;
138
+ }
139
+ });
140
+ });
141
+ //# sourceMappingURL=config.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"config.test.js","sourceRoot":"","sources":["../../../../src/tools/helpers/config.test.ts"],"names":[],"mappings":"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"}