@bobfrankston/mailx 1.0.450 → 1.0.452

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 (198) hide show
  1. package/bin/mailx.js.map +1 -0
  2. package/bin/mailx.ts +1498 -0
  3. package/bin/postinstall.js.map +1 -0
  4. package/bin/postinstall.ts +41 -0
  5. package/bin/tsconfig.json +10 -0
  6. package/client/.gitattributes +10 -0
  7. package/client/app.js +51 -2
  8. package/client/app.js.map +1 -0
  9. package/client/app.ts +3112 -0
  10. package/client/components/address-book.js.map +1 -0
  11. package/client/components/address-book.ts +204 -0
  12. package/client/components/alarms.js.map +1 -0
  13. package/client/components/alarms.ts +276 -0
  14. package/client/components/calendar-sidebar.js.map +1 -0
  15. package/client/components/calendar-sidebar.ts +474 -0
  16. package/client/components/calendar.js.map +1 -0
  17. package/client/components/calendar.ts +211 -0
  18. package/client/components/context-menu.js.map +1 -0
  19. package/client/components/context-menu.ts +95 -0
  20. package/client/components/folder-picker.js.map +1 -0
  21. package/client/components/folder-picker.ts +127 -0
  22. package/client/components/folder-tree.js.map +1 -0
  23. package/client/components/folder-tree.ts +1069 -0
  24. package/client/components/message-list.js.map +1 -0
  25. package/client/components/message-list.ts +1129 -0
  26. package/client/components/message-viewer.js.map +1 -0
  27. package/client/components/message-viewer.ts +1257 -0
  28. package/client/components/outbox-view.js.map +1 -0
  29. package/client/components/outbox-view.ts +102 -0
  30. package/client/components/tasks.js.map +1 -0
  31. package/client/components/tasks.ts +234 -0
  32. package/client/compose/compose.js.map +1 -0
  33. package/client/compose/compose.ts +1231 -0
  34. package/client/compose/editor.js.map +1 -0
  35. package/client/compose/editor.ts +599 -0
  36. package/client/compose/ghost-text.js.map +1 -0
  37. package/client/compose/ghost-text.ts +140 -0
  38. package/client/index.html +1 -0
  39. package/client/lib/android-bootstrap.js.map +1 -0
  40. package/client/lib/android-bootstrap.ts +9 -0
  41. package/client/lib/api-client.js.map +1 -0
  42. package/client/lib/api-client.ts +439 -0
  43. package/client/lib/local-service.js.map +1 -0
  44. package/client/lib/local-service.ts +646 -0
  45. package/client/lib/local-store.js.map +1 -0
  46. package/client/lib/local-store.ts +283 -0
  47. package/client/lib/message-state.js.map +1 -0
  48. package/client/lib/message-state.ts +140 -0
  49. package/client/tsconfig.json +19 -0
  50. package/package.json +15 -15
  51. package/packages/mailx-api/.gitattributes +10 -0
  52. package/packages/mailx-api/index.d.ts.map +1 -0
  53. package/packages/mailx-api/index.js.map +1 -0
  54. package/packages/mailx-api/index.ts +283 -0
  55. package/packages/mailx-api/tsconfig.json +9 -0
  56. package/packages/mailx-compose/.gitattributes +10 -0
  57. package/packages/mailx-compose/index.d.ts.map +1 -0
  58. package/packages/mailx-compose/index.js.map +1 -0
  59. package/packages/mailx-compose/index.ts +85 -0
  60. package/packages/mailx-compose/tsconfig.json +9 -0
  61. package/packages/mailx-core/index.d.ts.map +1 -0
  62. package/packages/mailx-core/index.js.map +1 -0
  63. package/packages/mailx-core/index.ts +424 -0
  64. package/packages/mailx-core/ipc.d.ts.map +1 -0
  65. package/packages/mailx-core/ipc.js.map +1 -0
  66. package/packages/mailx-core/ipc.ts +62 -0
  67. package/packages/mailx-core/tsconfig.json +9 -0
  68. package/packages/mailx-host/.gitattributes +10 -0
  69. package/packages/mailx-host/index.d.ts.map +1 -0
  70. package/packages/mailx-host/index.js.map +1 -0
  71. package/packages/mailx-host/index.ts +38 -0
  72. package/packages/mailx-host/package.json +10 -2
  73. package/packages/mailx-host/tsconfig.json +9 -0
  74. package/packages/mailx-send/.gitattributes +10 -0
  75. package/packages/mailx-send/cli-queue.d.ts.map +1 -0
  76. package/packages/mailx-send/cli-queue.js.map +1 -0
  77. package/packages/mailx-send/cli-queue.ts +62 -0
  78. package/packages/mailx-send/cli-send.d.ts.map +1 -0
  79. package/packages/mailx-send/cli-send.js.map +1 -0
  80. package/packages/mailx-send/cli-send.ts +83 -0
  81. package/packages/mailx-send/cli.d.ts.map +1 -0
  82. package/packages/mailx-send/cli.js.map +1 -0
  83. package/packages/mailx-send/cli.ts +126 -0
  84. package/packages/mailx-send/index.d.ts.map +1 -0
  85. package/packages/mailx-send/index.js.map +1 -0
  86. package/packages/mailx-send/index.ts +333 -0
  87. package/packages/mailx-send/mailsend/cli.d.ts.map +1 -0
  88. package/packages/mailx-send/mailsend/cli.js.map +1 -0
  89. package/packages/mailx-send/mailsend/cli.ts +81 -0
  90. package/packages/mailx-send/mailsend/index.d.ts.map +1 -0
  91. package/packages/mailx-send/mailsend/index.js.map +1 -0
  92. package/packages/mailx-send/mailsend/index.ts +333 -0
  93. package/packages/mailx-send/mailsend/package-lock.json +65 -0
  94. package/packages/mailx-send/mailsend/tsconfig.json +21 -0
  95. package/packages/mailx-send/package-lock.json +65 -0
  96. package/packages/mailx-send/package.json +1 -1
  97. package/packages/mailx-send/tsconfig.json +21 -0
  98. package/packages/mailx-server/.gitattributes +10 -0
  99. package/packages/mailx-server/index.d.ts.map +1 -0
  100. package/packages/mailx-server/index.js.map +1 -0
  101. package/packages/mailx-server/index.ts +429 -0
  102. package/packages/mailx-server/tsconfig.json +9 -0
  103. package/packages/mailx-service/google-sync.d.ts.map +1 -0
  104. package/packages/mailx-service/google-sync.js.map +1 -0
  105. package/packages/mailx-service/google-sync.ts +238 -0
  106. package/packages/mailx-service/index.d.ts.map +1 -0
  107. package/packages/mailx-service/index.js.map +1 -0
  108. package/packages/mailx-service/index.ts +2461 -0
  109. package/packages/mailx-service/jsonrpc.d.ts.map +1 -0
  110. package/packages/mailx-service/jsonrpc.js.map +1 -0
  111. package/packages/mailx-service/jsonrpc.ts +268 -0
  112. package/packages/mailx-service/tsconfig.json +9 -0
  113. package/packages/mailx-settings/.gitattributes +10 -0
  114. package/packages/mailx-settings/cloud.d.ts.map +1 -0
  115. package/packages/mailx-settings/cloud.js.map +1 -0
  116. package/packages/mailx-settings/cloud.ts +388 -0
  117. package/packages/mailx-settings/index.d.ts.map +1 -0
  118. package/packages/mailx-settings/index.js.map +1 -0
  119. package/packages/mailx-settings/index.ts +892 -0
  120. package/packages/mailx-settings/tsconfig.json +9 -0
  121. package/packages/mailx-store/.gitattributes +10 -0
  122. package/packages/mailx-store/db.d.ts.map +1 -0
  123. package/packages/mailx-store/db.js.map +1 -0
  124. package/packages/mailx-store/db.ts +2007 -0
  125. package/packages/mailx-store/file-store.d.ts.map +1 -0
  126. package/packages/mailx-store/file-store.js.map +1 -0
  127. package/packages/mailx-store/file-store.ts +82 -0
  128. package/packages/mailx-store/index.d.ts.map +1 -0
  129. package/packages/mailx-store/index.js.map +1 -0
  130. package/packages/mailx-store/index.ts +7 -0
  131. package/packages/mailx-store/tsconfig.json +9 -0
  132. package/packages/mailx-store-web/android-bootstrap.d.ts.map +1 -0
  133. package/packages/mailx-store-web/android-bootstrap.js.map +1 -0
  134. package/packages/mailx-store-web/android-bootstrap.ts +1262 -0
  135. package/packages/mailx-store-web/db.d.ts.map +1 -0
  136. package/packages/mailx-store-web/db.js.map +1 -0
  137. package/packages/mailx-store-web/db.ts +756 -0
  138. package/packages/mailx-store-web/gmail-api-web.d.ts.map +1 -0
  139. package/packages/mailx-store-web/gmail-api-web.js.map +1 -0
  140. package/packages/mailx-store-web/gmail-api-web.ts +11 -0
  141. package/packages/mailx-store-web/imap-web-provider.d.ts.map +1 -0
  142. package/packages/mailx-store-web/imap-web-provider.js.map +1 -0
  143. package/packages/mailx-store-web/imap-web-provider.ts +156 -0
  144. package/packages/mailx-store-web/index.d.ts.map +1 -0
  145. package/packages/mailx-store-web/index.js.map +1 -0
  146. package/packages/mailx-store-web/index.ts +10 -0
  147. package/packages/mailx-store-web/main-thread-host.d.ts.map +1 -0
  148. package/packages/mailx-store-web/main-thread-host.js.map +1 -0
  149. package/packages/mailx-store-web/main-thread-host.ts +322 -0
  150. package/packages/mailx-store-web/package.json +4 -4
  151. package/packages/mailx-store-web/provider-types.d.ts.map +1 -0
  152. package/packages/mailx-store-web/provider-types.js.map +1 -0
  153. package/packages/mailx-store-web/provider-types.ts +7 -0
  154. package/packages/mailx-store-web/sync-manager.d.ts.map +1 -0
  155. package/packages/mailx-store-web/sync-manager.js.map +1 -0
  156. package/packages/mailx-store-web/sync-manager.ts +508 -0
  157. package/packages/mailx-store-web/tsconfig.json +10 -0
  158. package/packages/mailx-store-web/web-jsonrpc.d.ts.map +1 -0
  159. package/packages/mailx-store-web/web-jsonrpc.js.map +1 -0
  160. package/packages/mailx-store-web/web-jsonrpc.ts +116 -0
  161. package/packages/mailx-store-web/web-message-store.d.ts.map +1 -0
  162. package/packages/mailx-store-web/web-message-store.js.map +1 -0
  163. package/packages/mailx-store-web/web-message-store.ts +97 -0
  164. package/packages/mailx-store-web/web-service.d.ts.map +1 -0
  165. package/packages/mailx-store-web/web-service.js.map +1 -0
  166. package/packages/mailx-store-web/web-service.ts +616 -0
  167. package/packages/mailx-store-web/web-settings.d.ts.map +1 -0
  168. package/packages/mailx-store-web/web-settings.js.map +1 -0
  169. package/packages/mailx-store-web/web-settings.ts +522 -0
  170. package/packages/mailx-store-web/worker-entry.d.ts.map +1 -0
  171. package/packages/mailx-store-web/worker-entry.js.map +1 -0
  172. package/packages/mailx-store-web/worker-entry.ts +215 -0
  173. package/packages/mailx-store-web/worker-tcp-transport.d.ts.map +1 -0
  174. package/packages/mailx-store-web/worker-tcp-transport.js.map +1 -0
  175. package/packages/mailx-store-web/worker-tcp-transport.ts +101 -0
  176. package/packages/mailx-types/.gitattributes +10 -0
  177. package/packages/mailx-types/index.d.ts.map +1 -0
  178. package/packages/mailx-types/index.js.map +1 -0
  179. package/packages/mailx-types/index.ts +498 -0
  180. package/packages/mailx-types/tsconfig.json +9 -0
  181. package/tsconfig.base.json +2 -1
  182. package/tsconfig.json +9 -0
  183. package/build-apk.cmd +0 -3
  184. package/npmg.bat +0 -6
  185. package/packages/mailx-imap/index.d.ts +0 -442
  186. package/packages/mailx-imap/index.js +0 -3669
  187. package/packages/mailx-imap/package.json +0 -25
  188. package/packages/mailx-imap/providers/gmail-api.d.ts +0 -8
  189. package/packages/mailx-imap/providers/gmail-api.js +0 -8
  190. package/packages/mailx-imap/providers/types.d.ts +0 -9
  191. package/packages/mailx-imap/providers/types.js +0 -9
  192. package/packages/mailx-imap/tsconfig.tsbuildinfo +0 -1
  193. package/rebuild.cmd +0 -23
  194. package/tdview.cmd +0 -2
  195. package/temp.ps1 +0 -10
  196. package/test-smtp-direct.mjs +0 -4
  197. package/unbash.cmd +0 -55
  198. package/unwedge.cmd +0 -1
@@ -0,0 +1,62 @@
1
+ #!/usr/bin/env node
2
+ /**
3
+ * mailq — manage the outgoing mail queue.
4
+ *
5
+ * Usage:
6
+ * mailq Show queue status (all accounts)
7
+ * mailq --process Process all queues now (send pending, retry failed)
8
+ * mailq --process -q gmail Process only the gmail queue
9
+ */
10
+
11
+ import * as fs from "node:fs";
12
+ import { initMailSend, type MailSendConfig } from "./index.js";
13
+
14
+ const args = process.argv.slice(2);
15
+
16
+ function getArg(name: string): string | undefined {
17
+ const shorts: Record<string, string> = { q: "queue" };
18
+ const longIdx = args.indexOf(`--${name}`);
19
+ const shortKey = Object.entries(shorts).find(([, v]) => v === name)?.[0];
20
+ const shortIdx = shortKey ? args.indexOf(`-${shortKey}`) : -1;
21
+ const i = Math.max(longIdx, shortIdx);
22
+ if (i === -1 || i + 1 >= args.length) return undefined;
23
+ return args[i + 1];
24
+ }
25
+
26
+ function hasFlag(name: string): boolean {
27
+ return args.includes(`--${name}`);
28
+ }
29
+
30
+ const configPath = getArg("config") || "mailsend.json";
31
+ if (!fs.existsSync(configPath)) {
32
+ console.error(`Config file not found: ${configPath}`);
33
+ process.exit(1);
34
+ }
35
+
36
+ const config: MailSendConfig = JSON.parse(fs.readFileSync(configPath, "utf-8"));
37
+ const sender = initMailSend(config);
38
+
39
+ if (hasFlag("process")) {
40
+ const accountId = getArg("queue");
41
+ if (accountId) {
42
+ await sender.processQueue(accountId);
43
+ } else {
44
+ await sender.processAllQueues();
45
+ }
46
+ // Show status after processing
47
+ const status = sender.getStatus();
48
+ for (const [id, s] of Object.entries(status)) {
49
+ console.log(` ${id}: ${s.pending} pending, ${s.failed} failed`);
50
+ }
51
+ } else {
52
+ // Default: show status
53
+ const status = sender.getStatus();
54
+ let total = 0;
55
+ for (const [id, s] of Object.entries(status)) {
56
+ console.log(` ${id}: ${s.pending} pending, ${s.failed} failed`);
57
+ total += s.pending + s.failed;
58
+ }
59
+ if (total === 0) console.log(" All queues empty");
60
+ }
61
+
62
+ process.exit(0);
@@ -0,0 +1 @@
1
+ {"version":3,"file":"cli-send.d.ts","sourceRoot":"","sources":["cli-send.ts"],"names":[],"mappings":";AACA;;;;;;;;;;;;;GAaG"}
@@ -0,0 +1 @@
1
+ {"version":3,"file":"cli-send.js","sourceRoot":"","sources":["cli-send.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,83 @@
1
+ #!/usr/bin/env node
2
+ /**
3
+ * mailsend — create and send an email message.
4
+ *
5
+ * Usage:
6
+ * mailsend --to addr --subject "text" --body "text"
7
+ * mailsend -q gmail --from "Bob <bob@gmail.com>" --to addr --subject "text"
8
+ * mailsend --json message.json Read message from JSON file
9
+ * echo '{"to":["a@b.com"],...}' | mailsend --json - Read from stdin
10
+ *
11
+ * JSON format:
12
+ * { "to": ["addr"], "cc": ["addr"], "from": "Name <addr>",
13
+ * "subject": "text", "text": "body", "html": "<p>body</p>",
14
+ * "accountId": "iecc" }
15
+ */
16
+
17
+ import * as fs from "node:fs";
18
+ import { initMailSend, queueMessage, type MailSendConfig, type MailMessage } from "./index.js";
19
+
20
+ const args = process.argv.slice(2);
21
+
22
+ function getArg(name: string): string | undefined {
23
+ const shorts: Record<string, string> = { q: "queue" };
24
+ const longIdx = args.indexOf(`--${name}`);
25
+ const shortKey = Object.entries(shorts).find(([, v]) => v === name)?.[0];
26
+ const shortIdx = shortKey ? args.indexOf(`-${shortKey}`) : -1;
27
+ const i = Math.max(longIdx, shortIdx);
28
+ if (i === -1 || i + 1 >= args.length) return undefined;
29
+ return args[i + 1];
30
+ }
31
+
32
+ const configPath = getArg("config") || "mailsend.json";
33
+ if (!fs.existsSync(configPath)) {
34
+ console.error(`Config file not found: ${configPath}`);
35
+ process.exit(1);
36
+ }
37
+
38
+ const config: MailSendConfig = JSON.parse(fs.readFileSync(configPath, "utf-8"));
39
+ const sender = initMailSend(config);
40
+ const accountId = getArg("queue") || config.defaultAccount || "";
41
+
42
+ let msg: Partial<MailMessage> = {};
43
+
44
+ const jsonArg = getArg("json");
45
+ if (jsonArg) {
46
+ const raw = jsonArg === "-"
47
+ ? fs.readFileSync(0, "utf-8")
48
+ : fs.readFileSync(jsonArg, "utf-8");
49
+ msg = JSON.parse(raw);
50
+ } else {
51
+ const to = getArg("to");
52
+ if (!to) {
53
+ console.error("Usage: mailsend --to addr --subject text [--body text] [--from addr] [-q account]");
54
+ console.error(" mailsend --json message.json [-q account]");
55
+ process.exit(1);
56
+ }
57
+ msg = {
58
+ to: to.split(",").map(s => s.trim()),
59
+ subject: getArg("subject") || "(no subject)",
60
+ text: getArg("body") || "",
61
+ html: getArg("html"),
62
+ };
63
+ }
64
+
65
+ const acctId = msg.accountId || accountId;
66
+ const acct = config.accounts[acctId];
67
+ if (!acct) {
68
+ console.error(`Unknown account: ${acctId}`);
69
+ console.error(`Available: ${Object.keys(config.accounts).join(", ")}`);
70
+ process.exit(1);
71
+ }
72
+
73
+ msg.from = msg.from || getArg("from") || acct.from || acct.user;
74
+ msg.accountId = acctId;
75
+
76
+ if (!msg.to?.length) {
77
+ console.error("Message must have at least one recipient (to)");
78
+ process.exit(1);
79
+ }
80
+
81
+ const filename = queueMessage(msg as MailMessage);
82
+ console.log(`Queued: ${filename}`);
83
+ setTimeout(() => process.exit(0), 2000);
@@ -0,0 +1 @@
1
+ {"version":3,"file":"cli.d.ts","sourceRoot":"","sources":["cli.ts"],"names":[],"mappings":";AACA;;;;;;;;;;;;;;;;;;GAkBG"}
@@ -0,0 +1 @@
1
+ {"version":3,"file":"cli.js","sourceRoot":"","sources":["cli.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,126 @@
1
+ #!/usr/bin/env node
2
+ /**
3
+ * mailx-send CLI — queue and send email.
4
+ *
5
+ * Queue management (process queued .ltr files):
6
+ * mailx-send --process Process all account queues now
7
+ * mailx-send --process -q gmail Process only the gmail queue
8
+ * mailx-send --status Show queue status
9
+ *
10
+ * Create and queue a new message:
11
+ * mailx-send --new --to addr --subject "text" --body "text"
12
+ * mailx-send --new -q gmail --from "Bob <bob@gmail.com>" --to addr --subject "text"
13
+ * mailx-send --new --json message.json Read message from JSON file
14
+ * echo '{"to":["a@b.com"],...}' | mailx-send --new --json - Read from stdin
15
+ *
16
+ * JSON message format:
17
+ * { "to": ["addr"], "cc": ["addr"], "from": "Name <addr>",
18
+ * "subject": "text", "text": "body", "html": "<p>body</p>",
19
+ * "accountId": "iecc" }
20
+ */
21
+
22
+ import * as fs from "node:fs";
23
+ import { initMailSend, queueMessage, type MailSendConfig, type MailMessage } from "./index.js";
24
+
25
+ const args = process.argv.slice(2);
26
+
27
+ function getArg(name: string): string | undefined {
28
+ const shorts: Record<string, string> = { q: "queue" };
29
+ const longIdx = args.indexOf(`--${name}`);
30
+ const shortKey = Object.entries(shorts).find(([, v]) => v === name)?.[0];
31
+ const shortIdx = shortKey ? args.indexOf(`-${shortKey}`) : -1;
32
+ const i = Math.max(longIdx, shortIdx);
33
+ if (i === -1 || i + 1 >= args.length) return undefined;
34
+ return args[i + 1];
35
+ }
36
+
37
+ function hasFlag(name: string): boolean {
38
+ return args.includes(`--${name}`);
39
+ }
40
+
41
+ // Load config
42
+ const configPath = getArg("config") || "mailsend.json";
43
+ if (!fs.existsSync(configPath)) {
44
+ console.error(`Config file not found: ${configPath}`);
45
+ console.error(`Create a mailsend.json with baseDir, accounts, etc.`);
46
+ process.exit(1);
47
+ }
48
+
49
+ const config: MailSendConfig = JSON.parse(fs.readFileSync(configPath, "utf-8"));
50
+ const sender = initMailSend(config);
51
+ const accountId = getArg("queue") || config.defaultAccount || "";
52
+
53
+ // ── Queue management ──
54
+
55
+ if (hasFlag("status")) {
56
+ const status = sender.getStatus();
57
+ for (const [id, s] of Object.entries(status)) {
58
+ console.log(` ${id}: ${s.pending} pending, ${s.failed} failed`);
59
+ }
60
+ process.exit(0);
61
+ }
62
+
63
+ if (hasFlag("process") && !hasFlag("new")) {
64
+ if (accountId && accountId !== config.defaultAccount) {
65
+ await sender.processQueue(accountId);
66
+ } else {
67
+ await sender.processAllQueues();
68
+ }
69
+ process.exit(0);
70
+ }
71
+
72
+ // ── Create new message ──
73
+
74
+ if (hasFlag("new")) {
75
+ let msg: Partial<MailMessage> = {};
76
+
77
+ const jsonArg = getArg("json");
78
+ if (jsonArg) {
79
+ // Load from JSON file or stdin
80
+ const raw = jsonArg === "-"
81
+ ? fs.readFileSync(0, "utf-8") // stdin
82
+ : fs.readFileSync(jsonArg, "utf-8");
83
+ msg = JSON.parse(raw);
84
+ } else {
85
+ // Build from CLI args
86
+ const to = getArg("to");
87
+ if (!to) {
88
+ console.error("--new requires --to or --json");
89
+ process.exit(1);
90
+ }
91
+ msg = {
92
+ to: to.split(",").map(s => s.trim()),
93
+ subject: getArg("subject") || "(no subject)",
94
+ text: getArg("body") || "",
95
+ html: getArg("html"),
96
+ };
97
+ }
98
+
99
+ // Set from/account
100
+ const acctId = msg.accountId || accountId;
101
+ const acct = config.accounts[acctId];
102
+ if (!acct) {
103
+ console.error(`Unknown account: ${acctId}`);
104
+ console.error(`Available: ${Object.keys(config.accounts).join(", ")}`);
105
+ process.exit(1);
106
+ }
107
+
108
+ msg.from = msg.from || getArg("from") || acct.from || acct.user;
109
+ msg.accountId = acctId;
110
+
111
+ if (!msg.to?.length) {
112
+ console.error("Message must have at least one recipient (to)");
113
+ process.exit(1);
114
+ }
115
+
116
+ const filename = queueMessage(msg as MailMessage);
117
+ console.log(`Queued: ${filename}`);
118
+ setTimeout(() => process.exit(0), 2000);
119
+ } else if (!hasFlag("status") && !hasFlag("process")) {
120
+ console.error("Usage:");
121
+ console.error(" mailx-send --new --to addr --subject text [--body text] [-q account]");
122
+ console.error(" mailx-send --new --json message.json [-q account]");
123
+ console.error(" mailx-send --process [-q account]");
124
+ console.error(" mailx-send --status");
125
+ process.exit(1);
126
+ }
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["index.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;GAUG;AAQH,MAAM,WAAW,UAAU;IACvB,IAAI,EAAE,MAAM,CAAC;IACb,IAAI,EAAE,MAAM,CAAC;IACb,MAAM,CAAC,EAAE,OAAO,CAAC;IACjB,IAAI,EAAE,UAAU,GAAG,QAAQ,CAAC;IAC5B,IAAI,EAAE,MAAM,CAAC;IACb,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,gEAAgE;IAChE,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,aAAa,CAAC,EAAE,MAAM,OAAO,CAAC,MAAM,CAAC,CAAC;CACzC;AAED,MAAM,WAAW,WAAW;IACxB,IAAI,EAAE,MAAM,CAAC;IACb,EAAE,EAAE,MAAM,EAAE,CAAC;IACb,EAAE,CAAC,EAAE,MAAM,EAAE,CAAC;IACd,GAAG,CAAC,EAAE,MAAM,EAAE,CAAC;IACf,OAAO,EAAE,MAAM,CAAC;IAChB,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,UAAU,CAAC,EAAE,MAAM,CAAC;IACpB,SAAS,CAAC,EAAE,MAAM,CAAC;CACtB;AAED,MAAM,WAAW,cAAc;IAC3B,oCAAoC;IACpC,OAAO,EAAE,MAAM,CAAC;IAChB,uCAAuC;IACvC,QAAQ,EAAE,MAAM,CAAC,MAAM,EAAE,UAAU,CAAC,CAAC;IACrC,8CAA8C;IAC9C,OAAO,CAAC,EAAE,OAAO,CAAC;IAClB,gDAAgD;IAChD,eAAe,CAAC,EAAE,MAAM,CAAC;IACzB,yBAAyB;IACzB,cAAc,CAAC,EAAE,MAAM,CAAC;CAC3B;AAoDD,wBAAgB,WAAW,CAAC,GAAG,EAAE,WAAW,GAAG,MAAM,CA0BpD;AAiCD,qBAAa,UAAU;IACnB,OAAO,CAAC,MAAM,CAAiB;IAC/B,OAAO,CAAC,UAAU,CAAS;IAC3B,OAAO,CAAC,UAAU,CAA+C;gBAErD,MAAM,EAAE,cAAc;IAMlC,OAAO,CAAC,UAAU;IAIlB,OAAO,CAAC,QAAQ;IAMhB,OAAO,CAAC,OAAO;IAIf,OAAO,CAAC,OAAO;IAMf,6CAA6C;IAC7C,YAAY,CAAC,GAAG,EAAE,WAAW,GAAG,MAAM;IAiBtC,+CAA+C;IACzC,YAAY,CAAC,SAAS,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IAoBpD,iCAAiC;IAC3B,gBAAgB,IAAI,OAAO,CAAC,IAAI,CAAC;YAMzB,QAAQ;IAkCtB,iEAAiE;IACjE,WAAW,IAAI,IAAI;IAYnB,OAAO,CAAC,WAAW;IAQnB,UAAU,IAAI,IAAI;IAOlB,mCAAmC;IACnC,SAAS,IAAI,MAAM,CAAC,MAAM,EAAE;QAAE,OAAO,EAAE,MAAM,CAAC;QAAC,MAAM,EAAE,MAAM,CAAA;KAAE,CAAC;CAUnE;AAID,wBAAgB,YAAY,CAAC,MAAM,EAAE,cAAc,GAAG,UAAU,CAI/D;AAED,wBAAgB,aAAa,IAAI,UAAU,GAAG,IAAI,CAEjD;AAED,wBAAgB,YAAY,CAAC,GAAG,EAAE,WAAW,GAAG,MAAM,CAGrD"}
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["index.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,333 @@
1
+ /**
2
+ * @bobfrankston/mailsend
3
+ * Queue-based mail sender. Each account has its own queue directory.
4
+ * Messages queued as .ltr (RFC 822) files, sent via SMTP with retry.
5
+ * Supports password and OAuth2 (Gmail) authentication.
6
+ *
7
+ * Directory structure:
8
+ * {baseDir}/{accountId}/queue/ — pending .ltr files
9
+ * {baseDir}/{accountId}/sent/ — sent messages (optional, YYYY/MM/DD)
10
+ * {baseDir}/{accountId}/fail/ — failed messages for retry
11
+ */
12
+
13
+ import * as fs from "node:fs";
14
+ import * as path from "node:path";
15
+ import { createTransport, type Transporter } from "nodemailer";
16
+
17
+ // ── Types ──
18
+
19
+ export interface SmtpConfig {
20
+ host: string;
21
+ port: number;
22
+ secure?: boolean;
23
+ auth: "password" | "oauth2";
24
+ user: string;
25
+ password?: string;
26
+ /** Default From address, e.g. "Bob Frankston <bob@iecc.com>" */
27
+ from?: string;
28
+ tokenProvider?: () => Promise<string>;
29
+ }
30
+
31
+ export interface MailMessage {
32
+ from: string;
33
+ to: string[];
34
+ cc?: string[];
35
+ bcc?: string[];
36
+ subject: string;
37
+ html?: string;
38
+ text?: string;
39
+ inReplyTo?: string;
40
+ references?: string;
41
+ accountId?: string;
42
+ }
43
+
44
+ export interface MailSendConfig {
45
+ /** Base directory for all queues */
46
+ baseDir: string;
47
+ /** SMTP configs keyed by account ID */
48
+ accounts: Record<string, SmtpConfig>;
49
+ /** Log sent messages to sent/ subdirectory */
50
+ logSent?: boolean;
51
+ /** Retry interval in ms (default: 5 minutes) */
52
+ retryIntervalMs?: number;
53
+ /** Default account ID */
54
+ defaultAccount?: string;
55
+ }
56
+
57
+ // ── File helpers ──
58
+
59
+ function generateFilename(): string {
60
+ const now = new Date();
61
+ const pad2 = (n: number) => String(n).padStart(2, "0");
62
+ const ts = `${now.getFullYear()}${pad2(now.getMonth() + 1)}${pad2(now.getDate())}_${pad2(now.getHours())}${pad2(now.getMinutes())}${pad2(now.getSeconds())}`;
63
+ const seq = String(Math.floor(Math.random() * 10000)).padStart(4, "0");
64
+ return `${ts}-${seq}.ltr`;
65
+ }
66
+
67
+ function datePath(): string {
68
+ const dt = new Date();
69
+ const pad2 = (n: number) => String(n).padStart(2, "0");
70
+ return `${dt.getFullYear()}/${pad2(dt.getMonth() + 1)}/${pad2(dt.getDate())}`;
71
+ }
72
+
73
+ function moveToDateDir(filePath: string, destBase: string): void {
74
+ const dest = path.join(destBase, datePath());
75
+ fs.mkdirSync(dest, { recursive: true });
76
+ fs.renameSync(filePath, path.join(dest, path.basename(filePath)));
77
+ }
78
+
79
+ function listLtr(dir: string): string[] {
80
+ if (!fs.existsSync(dir)) return [];
81
+ return fs.readdirSync(dir).filter(f => f.endsWith(".ltr")).sort();
82
+ }
83
+
84
+ function walkLtrCount(dir: string): number {
85
+ if (!fs.existsSync(dir)) return 0;
86
+ let count = 0;
87
+ for (const entry of fs.readdirSync(dir, { withFileTypes: true })) {
88
+ if (entry.isDirectory()) count += walkLtrCount(path.join(dir, entry.name));
89
+ else if (entry.name.endsWith(".ltr")) count++;
90
+ }
91
+ return count;
92
+ }
93
+
94
+ function walkLtrFiles(dir: string): string[] {
95
+ if (!fs.existsSync(dir)) return [];
96
+ const files: string[] = [];
97
+ for (const entry of fs.readdirSync(dir, { withFileTypes: true })) {
98
+ const full = path.join(dir, entry.name);
99
+ if (entry.isDirectory()) files.push(...walkLtrFiles(full));
100
+ else if (entry.name.endsWith(".ltr")) files.push(full);
101
+ }
102
+ return files;
103
+ }
104
+
105
+ // ── RFC 822 builder ──
106
+
107
+ export function buildRfc822(msg: MailMessage): string {
108
+ const lines: string[] = [];
109
+ lines.push(`From: ${msg.from}`);
110
+ lines.push(`To: ${msg.to.join(", ")}`);
111
+ if (msg.cc?.length) lines.push(`Cc: ${msg.cc.join(", ")}`);
112
+ if (msg.bcc?.length) lines.push(`Bcc: ${msg.bcc.join(", ")}`);
113
+ lines.push(`Subject: ${msg.subject}`);
114
+ lines.push(`Date: ${new Date().toUTCString()}`);
115
+ if (msg.inReplyTo) lines.push(`In-Reply-To: ${msg.inReplyTo}`);
116
+ if (msg.references) lines.push(`References: ${msg.references}`);
117
+ if (msg.accountId) lines.push(`X-MailSend-Account: ${msg.accountId}`);
118
+ lines.push(`MIME-Version: 1.0`);
119
+
120
+ if (msg.html) {
121
+ lines.push(`Content-Type: text/html; charset=UTF-8`);
122
+ lines.push(`Content-Transfer-Encoding: 8bit`);
123
+ lines.push("");
124
+ lines.push(msg.html);
125
+ } else {
126
+ lines.push(`Content-Type: text/plain; charset=UTF-8`);
127
+ lines.push(`Content-Transfer-Encoding: 8bit`);
128
+ lines.push("");
129
+ lines.push(msg.text || "");
130
+ }
131
+
132
+ return lines.join("\r\n");
133
+ }
134
+
135
+ // ── Transport cache ──
136
+
137
+ const transports: Map<string, Transporter> = new Map();
138
+
139
+ function getTransport(config: SmtpConfig): Transporter {
140
+ const key = `${config.host}:${config.port}:${config.user}`;
141
+ let transport = transports.get(key);
142
+ if (!transport) {
143
+ const opts: any = {
144
+ host: config.host,
145
+ port: config.port,
146
+ secure: config.secure ?? (config.port === 465),
147
+ tls: { rejectUnauthorized: false },
148
+ };
149
+
150
+ if (config.auth === "password") {
151
+ opts.auth = { user: config.user, pass: config.password };
152
+ } else if (config.auth === "oauth2") {
153
+ opts.auth = { type: "OAuth2", user: config.user };
154
+ }
155
+
156
+ transport = createTransport(opts);
157
+ transports.set(key, transport);
158
+ }
159
+ return transport;
160
+ }
161
+
162
+ // ── MailSender ──
163
+
164
+ let singleton: MailSender | null = null;
165
+
166
+ export class MailSender {
167
+ private config: MailSendConfig;
168
+ private processing = false;
169
+ private retryTimer: ReturnType<typeof setInterval> | null = null;
170
+
171
+ constructor(config: MailSendConfig) {
172
+ this.config = config;
173
+ // Ensure base directory exists
174
+ fs.mkdirSync(config.baseDir, { recursive: true });
175
+ }
176
+
177
+ private accountDir(accountId: string): string {
178
+ return path.join(this.config.baseDir, accountId);
179
+ }
180
+
181
+ private queueDir(accountId: string): string {
182
+ const dir = path.join(this.accountDir(accountId), "queue");
183
+ fs.mkdirSync(dir, { recursive: true });
184
+ return dir;
185
+ }
186
+
187
+ private sentDir(accountId: string): string {
188
+ return path.join(this.accountDir(accountId), "sent");
189
+ }
190
+
191
+ private failDir(accountId: string): string {
192
+ const dir = path.join(this.accountDir(accountId), "fail");
193
+ fs.mkdirSync(dir, { recursive: true });
194
+ return dir;
195
+ }
196
+
197
+ /** Queue a message. Returns the filename. */
198
+ queueMessage(msg: MailMessage): string {
199
+ const accountId = msg.accountId || this.config.defaultAccount;
200
+ if (!accountId) throw new Error("No accountId and no defaultAccount configured");
201
+ msg.accountId = accountId;
202
+
203
+ const content = buildRfc822(msg);
204
+ const filename = generateFilename();
205
+ fs.writeFileSync(path.join(this.queueDir(accountId), filename), content);
206
+
207
+ // Process immediately (non-blocking)
208
+ this.processQueue(accountId).catch(e =>
209
+ console.error(`[mailsend] Process error: ${e.message}`)
210
+ );
211
+
212
+ return filename;
213
+ }
214
+
215
+ /** Process pending messages for one account */
216
+ async processQueue(accountId: string): Promise<void> {
217
+ if (this.processing) return;
218
+ this.processing = true;
219
+
220
+ try {
221
+ const smtpConfig = this.config.accounts[accountId];
222
+ if (!smtpConfig) {
223
+ console.error(`[mailsend] No SMTP config for: ${accountId}`);
224
+ return;
225
+ }
226
+
227
+ const files = listLtr(this.queueDir(accountId));
228
+ for (const file of files) {
229
+ await this.sendFile(path.join(this.queueDir(accountId), file), accountId, smtpConfig);
230
+ }
231
+ } finally {
232
+ this.processing = false;
233
+ }
234
+ }
235
+
236
+ /** Process all account queues */
237
+ async processAllQueues(): Promise<void> {
238
+ for (const accountId of Object.keys(this.config.accounts)) {
239
+ await this.processQueue(accountId);
240
+ }
241
+ }
242
+
243
+ private async sendFile(filePath: string, accountId: string, smtpConfig: SmtpConfig): Promise<boolean> {
244
+ const content = fs.readFileSync(filePath, "utf-8");
245
+
246
+ try {
247
+ const transport = getTransport(smtpConfig);
248
+
249
+ if (smtpConfig.auth === "oauth2" && smtpConfig.tokenProvider) {
250
+ const token = await smtpConfig.tokenProvider();
251
+ (transport as any).options.auth.accessToken = token;
252
+ }
253
+
254
+ const result = await transport.sendMail({ raw: content });
255
+
256
+ if (result.rejected?.length > 0) {
257
+ console.error(`[mailsend] Rejected: ${result.rejected.join(", ")} — ${path.basename(filePath)}`);
258
+ moveToDateDir(filePath, this.failDir(accountId));
259
+ return false;
260
+ }
261
+
262
+ console.log(`[mailsend] Sent: ${path.basename(filePath)} → ${result.accepted?.join(", ")} (${result.messageId})`);
263
+
264
+ if (this.config.logSent) {
265
+ moveToDateDir(filePath, this.sentDir(accountId));
266
+ } else {
267
+ fs.unlinkSync(filePath);
268
+ }
269
+ return true;
270
+ } catch (e: any) {
271
+ console.error(`[mailsend] Error: ${path.basename(filePath)}: ${e.message}`);
272
+ moveToDateDir(filePath, this.failDir(accountId));
273
+ return false;
274
+ }
275
+ }
276
+
277
+ /** Start background worker — retries failed, processes queues */
278
+ startWorker(): void {
279
+ if (this.retryTimer) return;
280
+ const interval = this.config.retryIntervalMs || 300000;
281
+
282
+ this.retryTimer = setInterval(async () => {
283
+ for (const accountId of Object.keys(this.config.accounts)) {
284
+ this.retryFailed(accountId);
285
+ }
286
+ await this.processAllQueues();
287
+ }, interval);
288
+ }
289
+
290
+ private retryFailed(accountId: string): void {
291
+ const failFiles = walkLtrFiles(this.failDir(accountId));
292
+ const queueDir = this.queueDir(accountId);
293
+ for (const file of failFiles) {
294
+ fs.renameSync(file, path.join(queueDir, path.basename(file)));
295
+ }
296
+ }
297
+
298
+ stopWorker(): void {
299
+ if (this.retryTimer) {
300
+ clearInterval(this.retryTimer);
301
+ this.retryTimer = null;
302
+ }
303
+ }
304
+
305
+ /** Get queue status per account */
306
+ getStatus(): Record<string, { pending: number; failed: number }> {
307
+ const result: Record<string, { pending: number; failed: number }> = {};
308
+ for (const accountId of Object.keys(this.config.accounts)) {
309
+ result[accountId] = {
310
+ pending: listLtr(this.queueDir(accountId)).length,
311
+ failed: walkLtrCount(this.failDir(accountId)),
312
+ };
313
+ }
314
+ return result;
315
+ }
316
+ }
317
+
318
+ // ── Singleton API ──
319
+
320
+ export function initMailSend(config: MailSendConfig): MailSender {
321
+ singleton = new MailSender(config);
322
+ singleton.startWorker();
323
+ return singleton;
324
+ }
325
+
326
+ export function getMailSender(): MailSender | null {
327
+ return singleton;
328
+ }
329
+
330
+ export function queueMessage(msg: MailMessage): string {
331
+ if (!singleton) throw new Error("mailsend not initialized — call initMailSend() first");
332
+ return singleton.queueMessage(msg);
333
+ }
@@ -0,0 +1 @@
1
+ {"version":3,"file":"cli.d.ts","sourceRoot":"","sources":["cli.ts"],"names":[],"mappings":";AACA;;;;;;;;GAQG"}
@@ -0,0 +1 @@
1
+ {"version":3,"file":"cli.js","sourceRoot":"","sources":["cli.ts"],"names":[],"mappings":"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"}