@bobfrankston/mailx 1.0.451 → 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 -3684
  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,81 @@
1
+ #!/usr/bin/env node
2
+ /**
3
+ * mailsend CLI — queue and send email from the command line.
4
+ *
5
+ * Usage:
6
+ * mailsend --to user@example.com --subject "Hello" --body "Message text"
7
+ * mailsend -q gmail --to user@example.com --subject "Test"
8
+ * mailsend --status
9
+ * mailsend --process
10
+ */
11
+
12
+ import * as fs from "node:fs";
13
+ import { initMailSend, queueMessage, getMailSender, type MailSendConfig } from "./index.js";
14
+
15
+ const args = process.argv.slice(2);
16
+
17
+ function getArg(name: string): string | undefined {
18
+ const short: Record<string, string> = { q: "queue" };
19
+ const i = Math.max(args.indexOf(`--${name}`), args.indexOf(`-${Object.entries(short).find(([, v]) => v === name)?.[0] || ""}`));
20
+ if (i === -1) return undefined;
21
+ return args[i + 1];
22
+ }
23
+
24
+ function hasFlag(name: string): boolean {
25
+ return args.includes(`--${name}`);
26
+ }
27
+
28
+ // Load config
29
+ const configPath = getArg("config") || "mailsend.json";
30
+ if (!fs.existsSync(configPath)) {
31
+ console.error(`Config file not found: ${configPath}`);
32
+ console.error(`Create a mailsend.json with baseDir, accounts, etc.`);
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("status")) {
40
+ const status = sender.getStatus();
41
+ for (const [id, s] of Object.entries(status)) {
42
+ console.log(` ${id}: ${s.pending} pending, ${s.failed} failed`);
43
+ }
44
+ process.exit(0);
45
+ }
46
+
47
+ if (hasFlag("process")) {
48
+ await sender.processAllQueues();
49
+ process.exit(0);
50
+ }
51
+
52
+ const to = getArg("to");
53
+ const subject = getArg("subject") || "(no subject)";
54
+ const body = getArg("body") || "";
55
+ const from = getArg("from") || "";
56
+ const accountId = getArg("queue") || config.defaultAccount || "";
57
+
58
+ if (!to) {
59
+ console.error("Usage: mailsend --to addr [--subject text] [--body text] [--from addr] [-q account]");
60
+ console.error(" mailsend --status");
61
+ console.error(" mailsend --process");
62
+ process.exit(1);
63
+ }
64
+
65
+ const acct = config.accounts[accountId];
66
+ if (!acct) {
67
+ console.error(`Unknown account: ${accountId}`);
68
+ console.error(`Available: ${Object.keys(config.accounts).join(", ")}`);
69
+ process.exit(1);
70
+ }
71
+
72
+ const filename = queueMessage({
73
+ from: from || acct.from || acct.user,
74
+ to: to.split(",").map(s => s.trim()),
75
+ subject,
76
+ text: body,
77
+ accountId,
78
+ });
79
+
80
+ console.log(`Queued: ${filename}`);
81
+ setTimeout(() => process.exit(0), 2000);
@@ -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,65 @@
1
+ {
2
+ "name": "@bobfrankston/mailsend",
3
+ "version": "0.1.0",
4
+ "lockfileVersion": 3,
5
+ "requires": true,
6
+ "packages": {
7
+ "": {
8
+ "name": "@bobfrankston/mailsend",
9
+ "version": "0.1.0",
10
+ "license": "MIT",
11
+ "dependencies": {
12
+ "@bobfrankston/oauthsupport": "file:../../projects/oauth/oauthsupport",
13
+ "nodemailer": "^7.0.0"
14
+ },
15
+ "bin": {
16
+ "mailsend": "cli.js"
17
+ },
18
+ "devDependencies": {
19
+ "@types/node": "^22.0.0",
20
+ "@types/nodemailer": "^6.4.0"
21
+ }
22
+ },
23
+ "../../projects/oauth/oauthsupport": {},
24
+ "node_modules/@bobfrankston/oauthsupport": {
25
+ "resolved": "../../projects/oauth/oauthsupport",
26
+ "link": true
27
+ },
28
+ "node_modules/@types/node": {
29
+ "version": "22.19.15",
30
+ "resolved": "https://registry.npmjs.org/@types/node/-/node-22.19.15.tgz",
31
+ "integrity": "sha512-F0R/h2+dsy5wJAUe3tAU6oqa2qbWY5TpNfL/RGmo1y38hiyO1w3x2jPtt76wmuaJI4DQnOBu21cNXQ2STIUUWg==",
32
+ "dev": true,
33
+ "license": "MIT",
34
+ "dependencies": {
35
+ "undici-types": "~6.21.0"
36
+ }
37
+ },
38
+ "node_modules/@types/nodemailer": {
39
+ "version": "6.4.23",
40
+ "resolved": "https://registry.npmjs.org/@types/nodemailer/-/nodemailer-6.4.23.tgz",
41
+ "integrity": "sha512-aFV3/NsYFLSx9mbb5gtirBSXJnAlrusoKNuPbxsASWc7vrKLmIrTQRpdcxNcSFL3VW2A2XpeLEavwb2qMi6nlQ==",
42
+ "dev": true,
43
+ "license": "MIT",
44
+ "dependencies": {
45
+ "@types/node": "*"
46
+ }
47
+ },
48
+ "node_modules/nodemailer": {
49
+ "version": "7.0.13",
50
+ "resolved": "https://registry.npmjs.org/nodemailer/-/nodemailer-7.0.13.tgz",
51
+ "integrity": "sha512-PNDFSJdP+KFgdsG3ZzMXCgquO7I6McjY2vlqILjtJd0hy8wEvtugS9xKRF2NWlPNGxvLCXlTNIae4serI7dinw==",
52
+ "license": "MIT-0",
53
+ "engines": {
54
+ "node": ">=6.0.0"
55
+ }
56
+ },
57
+ "node_modules/undici-types": {
58
+ "version": "6.21.0",
59
+ "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-6.21.0.tgz",
60
+ "integrity": "sha512-iwDZqg0QAGrg9Rav5H4n0M64c3mkR59cJ6wQp+7C4nI0gsmExaedaYLNO44eT4AtBBwjbTiGPMlt2Md0T9H9JQ==",
61
+ "dev": true,
62
+ "license": "MIT"
63
+ }
64
+ }
65
+ }
@@ -0,0 +1,21 @@
1
+ {
2
+ "compilerOptions": {
3
+ "target": "esnext",
4
+ "module": "NodeNext",
5
+ "moduleResolution": "NodeNext",
6
+ "allowSyntheticDefaultImports": true,
7
+ "esModuleInterop": true,
8
+ "strict": true,
9
+ "strictNullChecks": false,
10
+ "noImplicitAny": true,
11
+ "skipLibCheck": true,
12
+ "declaration": true,
13
+ "declarationMap": true,
14
+ "sourceMap": true,
15
+ "outDir": ".",
16
+ "rootDir": ".",
17
+ "newLine": "lf"
18
+ },
19
+ "include": ["*.ts"],
20
+ "exclude": ["node_modules", "prev", "cruft"]
21
+ }
@@ -0,0 +1,65 @@
1
+ {
2
+ "name": "@bobfrankston/mailsend",
3
+ "version": "0.1.1",
4
+ "lockfileVersion": 3,
5
+ "requires": true,
6
+ "packages": {
7
+ "": {
8
+ "name": "@bobfrankston/mailsend",
9
+ "version": "0.1.1",
10
+ "license": "MIT",
11
+ "dependencies": {
12
+ "@bobfrankston/oauthsupport": "file:../../projects/oauth/oauthsupport",
13
+ "nodemailer": "^7.0.0"
14
+ },
15
+ "bin": {
16
+ "mailsend": "cli.js"
17
+ },
18
+ "devDependencies": {
19
+ "@types/node": "^22.0.0",
20
+ "@types/nodemailer": "^6.4.0"
21
+ }
22
+ },
23
+ "../../projects/oauth/oauthsupport": {},
24
+ "node_modules/@bobfrankston/oauthsupport": {
25
+ "resolved": "../../projects/oauth/oauthsupport",
26
+ "link": true
27
+ },
28
+ "node_modules/@types/node": {
29
+ "version": "22.19.15",
30
+ "resolved": "https://registry.npmjs.org/@types/node/-/node-22.19.15.tgz",
31
+ "integrity": "sha512-F0R/h2+dsy5wJAUe3tAU6oqa2qbWY5TpNfL/RGmo1y38hiyO1w3x2jPtt76wmuaJI4DQnOBu21cNXQ2STIUUWg==",
32
+ "dev": true,
33
+ "license": "MIT",
34
+ "dependencies": {
35
+ "undici-types": "~6.21.0"
36
+ }
37
+ },
38
+ "node_modules/@types/nodemailer": {
39
+ "version": "6.4.23",
40
+ "resolved": "https://registry.npmjs.org/@types/nodemailer/-/nodemailer-6.4.23.tgz",
41
+ "integrity": "sha512-aFV3/NsYFLSx9mbb5gtirBSXJnAlrusoKNuPbxsASWc7vrKLmIrTQRpdcxNcSFL3VW2A2XpeLEavwb2qMi6nlQ==",
42
+ "dev": true,
43
+ "license": "MIT",
44
+ "dependencies": {
45
+ "@types/node": "*"
46
+ }
47
+ },
48
+ "node_modules/nodemailer": {
49
+ "version": "7.0.13",
50
+ "resolved": "https://registry.npmjs.org/nodemailer/-/nodemailer-7.0.13.tgz",
51
+ "integrity": "sha512-PNDFSJdP+KFgdsG3ZzMXCgquO7I6McjY2vlqILjtJd0hy8wEvtugS9xKRF2NWlPNGxvLCXlTNIae4serI7dinw==",
52
+ "license": "MIT-0",
53
+ "engines": {
54
+ "node": ">=6.0.0"
55
+ }
56
+ },
57
+ "node_modules/undici-types": {
58
+ "version": "6.21.0",
59
+ "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-6.21.0.tgz",
60
+ "integrity": "sha512-iwDZqg0QAGrg9Rav5H4n0M64c3mkR59cJ6wQp+7C4nI0gsmExaedaYLNO44eT4AtBBwjbTiGPMlt2Md0T9H9JQ==",
61
+ "dev": true,
62
+ "license": "MIT"
63
+ }
64
+ }
65
+ }
@@ -24,7 +24,7 @@
24
24
  "license": "MIT",
25
25
  "dependencies": {
26
26
  "nodemailer": "^7.0.0",
27
- "@bobfrankston/oauthsupport": "file:../../../../projects/oauth/oauthsupport"
27
+ "@bobfrankston/oauthsupport": "file:../../../../../projects/oauth/oauthsupport"
28
28
  },
29
29
  "devDependencies": {
30
30
  "@types/nodemailer": "^6.4.0",
@@ -0,0 +1,21 @@
1
+ {
2
+ "compilerOptions": {
3
+ "target": "esnext",
4
+ "module": "NodeNext",
5
+ "moduleResolution": "NodeNext",
6
+ "allowSyntheticDefaultImports": true,
7
+ "esModuleInterop": true,
8
+ "strict": true,
9
+ "strictNullChecks": false,
10
+ "noImplicitAny": true,
11
+ "skipLibCheck": true,
12
+ "declaration": true,
13
+ "declarationMap": true,
14
+ "sourceMap": true,
15
+ "outDir": ".",
16
+ "rootDir": ".",
17
+ "newLine": "lf"
18
+ },
19
+ "include": ["*.ts"],
20
+ "exclude": ["node_modules", "prev", "cruft"]
21
+ }
@@ -0,0 +1,10 @@
1
+ # Force LF line endings for all text files
2
+ * text=auto eol=lf
3
+
4
+ # Ensure these are always LF
5
+ *.ts text eol=lf
6
+ *.js text eol=lf
7
+ *.json text eol=lf
8
+ *.md text eol=lf
9
+ *.yml text eol=lf
10
+ *.yaml text eol=lf
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["index.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAoEH,QAAA,MAAM,GAAG,6CAAY,CAAC;AAmOtB,iBAAe,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC,CAyFpC;AAyCD,OAAO,EAAE,GAAG,EAAE,KAAK,EAAE,CAAC"}