agent-inbox 0.2.4 → 0.2.5

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 (138) hide show
  1. package/CLAUDE.md +1 -92
  2. package/README.md +6 -73
  3. package/dist/federation/connection-manager.d.ts +0 -8
  4. package/dist/federation/connection-manager.d.ts.map +1 -1
  5. package/dist/federation/connection-manager.js +0 -12
  6. package/dist/federation/connection-manager.js.map +1 -1
  7. package/dist/federation/delivery-queue.d.ts +3 -11
  8. package/dist/federation/delivery-queue.d.ts.map +1 -1
  9. package/dist/federation/delivery-queue.js +8 -38
  10. package/dist/federation/delivery-queue.js.map +1 -1
  11. package/dist/index.d.ts +0 -17
  12. package/dist/index.d.ts.map +1 -1
  13. package/dist/index.js +0 -98
  14. package/dist/index.js.map +1 -1
  15. package/dist/jsonrpc/mail-push-types.d.ts +22 -2
  16. package/dist/jsonrpc/mail-push-types.d.ts.map +1 -1
  17. package/dist/jsonrpc/mail-push-types.js +18 -1
  18. package/dist/jsonrpc/mail-push-types.js.map +1 -1
  19. package/dist/jsonrpc/mail-push.d.ts +12 -1
  20. package/dist/jsonrpc/mail-push.d.ts.map +1 -1
  21. package/dist/jsonrpc/mail-push.js +13 -2
  22. package/dist/jsonrpc/mail-push.js.map +1 -1
  23. package/dist/jsonrpc/mail-server.d.ts.map +1 -1
  24. package/dist/jsonrpc/mail-server.js +42 -18
  25. package/dist/jsonrpc/mail-server.js.map +1 -1
  26. package/dist/router/message-router.d.ts +0 -15
  27. package/dist/router/message-router.d.ts.map +1 -1
  28. package/dist/router/message-router.js +3 -25
  29. package/dist/router/message-router.js.map +1 -1
  30. package/dist/storage/interface.d.ts +2 -9
  31. package/dist/storage/interface.d.ts.map +1 -1
  32. package/dist/storage/memory.d.ts +1 -4
  33. package/dist/storage/memory.d.ts.map +1 -1
  34. package/dist/storage/memory.js +6 -12
  35. package/dist/storage/memory.js.map +1 -1
  36. package/dist/storage/sqlite.d.ts +1 -6
  37. package/dist/storage/sqlite.d.ts.map +1 -1
  38. package/dist/storage/sqlite.js +6 -28
  39. package/dist/storage/sqlite.js.map +1 -1
  40. package/dist/types.d.ts +0 -79
  41. package/dist/types.d.ts.map +1 -1
  42. package/docs/DESIGN.md +0 -15
  43. package/package.json +3 -28
  44. package/rules/agent-inbox.md +0 -1
  45. package/src/federation/connection-manager.ts +0 -12
  46. package/src/federation/delivery-queue.ts +8 -38
  47. package/src/index.ts +0 -148
  48. package/src/jsonrpc/mail-push-types.ts +43 -2
  49. package/src/jsonrpc/mail-push.ts +34 -3
  50. package/src/jsonrpc/mail-server.ts +45 -26
  51. package/src/router/message-router.ts +4 -41
  52. package/src/storage/interface.ts +2 -11
  53. package/src/storage/memory.ts +8 -15
  54. package/src/storage/sqlite.ts +9 -36
  55. package/src/types.ts +0 -73
  56. package/test/load.test.ts +1 -1
  57. package/test/mail-push.test.ts +101 -1
  58. package/test/mail-server.test.ts +108 -0
  59. package/AGENTS.md +0 -18
  60. package/dist/federation/queue-store.d.ts +0 -42
  61. package/dist/federation/queue-store.d.ts.map +0 -1
  62. package/dist/federation/queue-store.js +0 -87
  63. package/dist/federation/queue-store.js.map +0 -1
  64. package/dist/index.d.mts +0 -2
  65. package/dist/index.mjs +0 -1
  66. package/dist/index.mjs.map +0 -1
  67. package/dist/mail/address-book.d.ts +0 -43
  68. package/dist/mail/address-book.d.ts.map +0 -1
  69. package/dist/mail/address-book.js +0 -95
  70. package/dist/mail/address-book.js.map +0 -1
  71. package/dist/mail/attachment-store.d.ts +0 -31
  72. package/dist/mail/attachment-store.d.ts.map +0 -1
  73. package/dist/mail/attachment-store.js +0 -74
  74. package/dist/mail/attachment-store.js.map +0 -1
  75. package/dist/mail/email-mapper.d.ts +0 -41
  76. package/dist/mail/email-mapper.d.ts.map +0 -1
  77. package/dist/mail/email-mapper.js +0 -216
  78. package/dist/mail/email-mapper.js.map +0 -1
  79. package/dist/mail/fs-attachment-store.d.ts +0 -38
  80. package/dist/mail/fs-attachment-store.d.ts.map +0 -1
  81. package/dist/mail/fs-attachment-store.js +0 -165
  82. package/dist/mail/fs-attachment-store.js.map +0 -1
  83. package/dist/mail/mail-gateway.d.ts +0 -114
  84. package/dist/mail/mail-gateway.d.ts.map +0 -1
  85. package/dist/mail/mail-gateway.js +0 -402
  86. package/dist/mail/mail-gateway.js.map +0 -1
  87. package/dist/mail/provider-transport.d.ts +0 -138
  88. package/dist/mail/provider-transport.d.ts.map +0 -1
  89. package/dist/mail/provider-transport.js +0 -434
  90. package/dist/mail/provider-transport.js.map +0 -1
  91. package/dist/mail/rate-limiter.d.ts +0 -20
  92. package/dist/mail/rate-limiter.d.ts.map +0 -1
  93. package/dist/mail/rate-limiter.js +0 -56
  94. package/dist/mail/rate-limiter.js.map +0 -1
  95. package/dist/mail/smtp-transport.d.ts +0 -141
  96. package/dist/mail/smtp-transport.d.ts.map +0 -1
  97. package/dist/mail/smtp-transport.js +0 -415
  98. package/dist/mail/smtp-transport.js.map +0 -1
  99. package/dist/mail/types.d.ts +0 -177
  100. package/dist/mail/types.d.ts.map +0 -1
  101. package/dist/mail/types.js +0 -11
  102. package/dist/mail/types.js.map +0 -1
  103. package/dist/router/destination.d.ts +0 -69
  104. package/dist/router/destination.d.ts.map +0 -1
  105. package/dist/router/destination.js +0 -106
  106. package/dist/router/destination.js.map +0 -1
  107. package/docs/MAIL-INTEROP-PLAN.md +0 -660
  108. package/renovate.json5 +0 -6
  109. package/src/federation/queue-store.ts +0 -124
  110. package/src/mail/address-book.ts +0 -111
  111. package/src/mail/attachment-store.ts +0 -90
  112. package/src/mail/email-mapper.ts +0 -288
  113. package/src/mail/fs-attachment-store.ts +0 -163
  114. package/src/mail/mail-gateway.ts +0 -505
  115. package/src/mail/provider-transport.ts +0 -577
  116. package/src/mail/rate-limiter.ts +0 -51
  117. package/src/mail/smtp-transport.ts +0 -589
  118. package/src/mail/types.ts +0 -221
  119. package/src/router/destination.ts +0 -140
  120. package/test/federation/delivery-queue-sqlite.test.ts +0 -158
  121. package/test/mail/address-book.test.ts +0 -111
  122. package/test/mail/attachment-store-contract.test.ts +0 -92
  123. package/test/mail/attachment-store.test.ts +0 -69
  124. package/test/mail/destination.test.ts +0 -115
  125. package/test/mail/dsn-parse.test.ts +0 -239
  126. package/test/mail/email-mapper.test.ts +0 -341
  127. package/test/mail/external-id.test.ts +0 -43
  128. package/test/mail/fs-attachment-store.test.ts +0 -134
  129. package/test/mail/full-flow-e2e.test.ts +0 -200
  130. package/test/mail/mail-gateway.test.ts +0 -419
  131. package/test/mail/mail-transport-contract.test.ts +0 -134
  132. package/test/mail/mock-mail.ts +0 -161
  133. package/test/mail/mock-postmark.ts +0 -66
  134. package/test/mail/provider-transport.test.ts +0 -381
  135. package/test/mail/rate-limiter.test.ts +0 -48
  136. package/test/mail/router-mail-integration.test.ts +0 -138
  137. package/test/mail/smtp-e2e.test.ts +0 -98
  138. package/test/mail/smtp-transport.test.ts +0 -138
@@ -1,43 +0,0 @@
1
- /**
2
- * Mail identity / address book.
3
- *
4
- * Maps between agent ids and external email addresses, and declares which
5
- * domains we own (for inbound RCPT acceptance). Plus-addressing aware:
6
- * `agent+scope@domain` resolves to agent `agent` in inbox scope `scope`.
7
- *
8
- * See docs/MAIL-INTEROP-PLAN.md §6.
9
- */
10
- import type { MailIdentityConfig } from "../types.js";
11
- import type { MailAddress } from "./types.js";
12
- export interface ResolvedInbound {
13
- agentId: string;
14
- /** Inbox scope from plus-addressing, if present. */
15
- scope?: string;
16
- }
17
- export declare class AddressBook {
18
- private localDomains;
19
- /** lower-cased address → agentId */
20
- private byAddress;
21
- /** agentId → canonical address */
22
- private byAgent;
23
- private catchAllAgentId?;
24
- /** Address mappings that collided (same address → multiple agents). */
25
- readonly conflicts: string[];
26
- constructor(config: MailIdentityConfig);
27
- /** Primary local domain — used to mint Message-IDs and default from-addresses. */
28
- primaryDomain(): string;
29
- /** True if `domain` (or a parent of it) is one we receive mail for. */
30
- ownsDomain(domain: string): boolean;
31
- /**
32
- * Resolve an inbound recipient email to a local agent (+ scope).
33
- * Returns undefined if the domain is not ours, or the mailbox is unknown and
34
- * no catch-all is configured.
35
- */
36
- resolveInbound(address: string): ResolvedInbound | undefined;
37
- /**
38
- * Resolve the from-address for an outbound sender agent. Falls back to
39
- * `<agentId>@<primaryDomain>` when the agent has no explicit mapping.
40
- */
41
- resolveFrom(agentId: string): MailAddress;
42
- }
43
- //# sourceMappingURL=address-book.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"address-book.d.ts","sourceRoot":"","sources":["../../src/mail/address-book.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AAEH,OAAO,KAAK,EAAE,kBAAkB,EAAE,MAAM,aAAa,CAAC;AACtD,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,YAAY,CAAC;AAE9C,MAAM,WAAW,eAAe;IAC9B,OAAO,EAAE,MAAM,CAAC;IAChB,oDAAoD;IACpD,KAAK,CAAC,EAAE,MAAM,CAAC;CAChB;AAED,qBAAa,WAAW;IACtB,OAAO,CAAC,YAAY,CAAW;IAC/B,oCAAoC;IACpC,OAAO,CAAC,SAAS,CAA6B;IAC9C,kCAAkC;IAClC,OAAO,CAAC,OAAO,CAA6B;IAC5C,OAAO,CAAC,eAAe,CAAC,CAAS;IACjC,uEAAuE;IACvE,QAAQ,CAAC,SAAS,EAAE,MAAM,EAAE,CAAM;gBAEtB,MAAM,EAAE,kBAAkB;IAkBtC,kFAAkF;IAClF,aAAa,IAAI,MAAM;IAIvB,uEAAuE;IACvE,UAAU,CAAC,MAAM,EAAE,MAAM,GAAG,OAAO;IAOnC;;;;OAIG;IACH,cAAc,CAAC,OAAO,EAAE,MAAM,GAAG,eAAe,GAAG,SAAS;IAoB5D;;;OAGG;IACH,WAAW,CAAC,OAAO,EAAE,MAAM,GAAG,WAAW;CAK1C"}
@@ -1,95 +0,0 @@
1
- /**
2
- * Mail identity / address book.
3
- *
4
- * Maps between agent ids and external email addresses, and declares which
5
- * domains we own (for inbound RCPT acceptance). Plus-addressing aware:
6
- * `agent+scope@domain` resolves to agent `agent` in inbox scope `scope`.
7
- *
8
- * See docs/MAIL-INTEROP-PLAN.md §6.
9
- */
10
- export class AddressBook {
11
- localDomains;
12
- /** lower-cased address → agentId */
13
- byAddress = new Map();
14
- /** agentId → canonical address */
15
- byAgent = new Map();
16
- catchAllAgentId;
17
- /** Address mappings that collided (same address → multiple agents). */
18
- conflicts = [];
19
- constructor(config) {
20
- this.localDomains = config.localDomains.map((d) => d.toLowerCase());
21
- this.catchAllAgentId = config.catchAllAgentId;
22
- for (const { agentId, address } of config.mappings) {
23
- const lower = address.toLowerCase();
24
- const existing = this.byAddress.get(lower);
25
- if (existing !== undefined && existing !== agentId) {
26
- // Two agents mapped to the same address — inbound routing would be
27
- // ambiguous. Keep the first and record the collision for surfacing.
28
- this.conflicts.push(`${address} → ${existing} & ${agentId}`);
29
- continue;
30
- }
31
- this.byAddress.set(lower, agentId);
32
- // First mapping for an agent is its canonical from-address.
33
- if (!this.byAgent.has(agentId))
34
- this.byAgent.set(agentId, address);
35
- }
36
- }
37
- /** Primary local domain — used to mint Message-IDs and default from-addresses. */
38
- primaryDomain() {
39
- return this.localDomains[0] ?? "localhost";
40
- }
41
- /** True if `domain` (or a parent of it) is one we receive mail for. */
42
- ownsDomain(domain) {
43
- const d = domain.toLowerCase();
44
- return this.localDomains.some((own) => d === own || d.endsWith(`.${own}`));
45
- }
46
- /**
47
- * Resolve an inbound recipient email to a local agent (+ scope).
48
- * Returns undefined if the domain is not ours, or the mailbox is unknown and
49
- * no catch-all is configured.
50
- */
51
- resolveInbound(address) {
52
- const parsed = splitAddress(address);
53
- if (!parsed)
54
- return undefined;
55
- const { local, domain } = parsed;
56
- if (!this.ownsDomain(domain))
57
- return undefined;
58
- // Plus-addressing: agent+scope → base agent, scope.
59
- const plusIdx = local.indexOf("+");
60
- const base = plusIdx === -1 ? local : local.slice(0, plusIdx);
61
- const scope = plusIdx === -1 ? undefined : local.slice(plusIdx + 1) || undefined;
62
- const canonical = `${base}@${domain}`.toLowerCase();
63
- const agentId = this.byAddress.get(canonical);
64
- if (agentId)
65
- return { agentId, scope };
66
- if (this.catchAllAgentId)
67
- return { agentId: this.catchAllAgentId, scope };
68
- return undefined;
69
- }
70
- /**
71
- * Resolve the from-address for an outbound sender agent. Falls back to
72
- * `<agentId>@<primaryDomain>` when the agent has no explicit mapping.
73
- */
74
- resolveFrom(agentId) {
75
- const mapped = this.byAgent.get(agentId);
76
- if (mapped)
77
- return { address: mapped };
78
- return { address: `${agentId}@${this.primaryDomain()}` };
79
- }
80
- }
81
- /** Split an addr-spec into local-part and domain. Returns undefined if malformed. */
82
- function splitAddress(address) {
83
- const at = address.indexOf("@");
84
- if (at <= 0 || at !== address.lastIndexOf("@") || at === address.length - 1) {
85
- // Reject empty local/domain and unquoted multiple "@" (e.g. "a@b@c"), so
86
- // this layer agrees with parseAddress (which splits on the first "@").
87
- return undefined;
88
- }
89
- const domain = address.slice(at + 1);
90
- // A domain must not contain "/" (a federation scope separator) — defensive.
91
- if (domain.includes("/"))
92
- return undefined;
93
- return { local: address.slice(0, at), domain };
94
- }
95
- //# sourceMappingURL=address-book.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"address-book.js","sourceRoot":"","sources":["../../src/mail/address-book.ts"],"names":[],"mappings":"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"}
@@ -1,31 +0,0 @@
1
- /**
2
- * Attachment byte stores.
3
- *
4
- * `MailAttachment.contentRef` is opaque; bytes live behind an AttachmentStore.
5
- * Default impl is SQLite-backed (atomic with the message, single-file backup,
6
- * works with the in-memory DB used in tests). A filesystem impl is opt-in for
7
- * large/high-volume deployments (added later). Refs are content-addressed
8
- * (sha256) so identical bytes dedup. See docs/MAIL-INTEROP-PLAN.md §4c.
9
- */
10
- import type Database from "better-sqlite3";
11
- import type { AttachmentStore } from "./types.js";
12
- export declare class SqliteAttachmentStore implements AttachmentStore {
13
- private db;
14
- private table;
15
- private putStmt;
16
- private getStmt;
17
- private delStmt;
18
- /**
19
- * @param db A better-sqlite3 handle (lifecycle owned by the caller).
20
- * @param prefix Table name prefix to match the Storage convention. Default "".
21
- */
22
- constructor(db: Database.Database, prefix?: string);
23
- private migrate;
24
- put(bytes: Buffer, meta: {
25
- contentType: string;
26
- filename?: string;
27
- }): Promise<string>;
28
- get(ref: string): Promise<Buffer>;
29
- delete(ref: string): Promise<void>;
30
- }
31
- //# sourceMappingURL=attachment-store.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"attachment-store.d.ts","sourceRoot":"","sources":["../../src/mail/attachment-store.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AAEH,OAAO,KAAK,QAAQ,MAAM,gBAAgB,CAAC;AAE3C,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,YAAY,CAAC;AAElD,qBAAa,qBAAsB,YAAW,eAAe;IAWzD,OAAO,CAAC,EAAE;IAVZ,OAAO,CAAC,KAAK,CAAS;IACtB,OAAO,CAAC,OAAO,CAAqB;IACpC,OAAO,CAAC,OAAO,CAAqB;IACpC,OAAO,CAAC,OAAO,CAAqB;IAEpC;;;OAGG;gBAEO,EAAE,EAAE,QAAQ,CAAC,QAAQ,EAC7B,MAAM,GAAE,MAAW;IAerB,OAAO,CAAC,OAAO;IAaT,GAAG,CACP,KAAK,EAAE,MAAM,EACb,IAAI,EAAE;QAAE,WAAW,EAAE,MAAM,CAAC;QAAC,QAAQ,CAAC,EAAE,MAAM,CAAA;KAAE,GAC/C,OAAO,CAAC,MAAM,CAAC;IAcZ,GAAG,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,CAAC;IAOjC,MAAM,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;CAIzC"}
@@ -1,74 +0,0 @@
1
- /**
2
- * Attachment byte stores.
3
- *
4
- * `MailAttachment.contentRef` is opaque; bytes live behind an AttachmentStore.
5
- * Default impl is SQLite-backed (atomic with the message, single-file backup,
6
- * works with the in-memory DB used in tests). A filesystem impl is opt-in for
7
- * large/high-volume deployments (added later). Refs are content-addressed
8
- * (sha256) so identical bytes dedup. See docs/MAIL-INTEROP-PLAN.md §4c.
9
- */
10
- import { createHash } from "node:crypto";
11
- export class SqliteAttachmentStore {
12
- db;
13
- table;
14
- putStmt;
15
- getStmt;
16
- delStmt;
17
- /**
18
- * @param db A better-sqlite3 handle (lifecycle owned by the caller).
19
- * @param prefix Table name prefix to match the Storage convention. Default "".
20
- */
21
- constructor(db, prefix = "") {
22
- this.db = db;
23
- this.table = `${prefix}mail_attachments`;
24
- this.migrate();
25
- this.putStmt = this.db.prepare(`INSERT INTO ${this.table} (ref, content_type, filename, bytes, size_bytes, created_at)
26
- VALUES (@ref, @content_type, @filename, @bytes, @size_bytes, @created_at)
27
- ON CONFLICT(ref) DO NOTHING`);
28
- this.getStmt = this.db.prepare(`SELECT bytes FROM ${this.table} WHERE ref = ?`);
29
- this.delStmt = this.db.prepare(`DELETE FROM ${this.table} WHERE ref = ?`);
30
- }
31
- migrate() {
32
- this.db.exec(`
33
- CREATE TABLE IF NOT EXISTS ${this.table} (
34
- ref TEXT PRIMARY KEY,
35
- content_type TEXT NOT NULL,
36
- filename TEXT,
37
- bytes BLOB NOT NULL,
38
- size_bytes INTEGER NOT NULL,
39
- created_at TEXT NOT NULL
40
- );
41
- `);
42
- }
43
- async put(bytes, meta) {
44
- const ref = "sha256:" + createHash("sha256").update(bytes).digest("hex");
45
- // ON CONFLICT DO NOTHING — content-addressed, so identical bytes are deduped.
46
- this.putStmt.run({
47
- ref,
48
- content_type: meta.contentType,
49
- filename: meta.filename ?? null,
50
- bytes,
51
- size_bytes: bytes.length,
52
- created_at: new Date().toISOString(),
53
- });
54
- return ref;
55
- }
56
- async get(ref) {
57
- assertValidRef(ref);
58
- const row = this.getStmt.get(ref);
59
- if (!row)
60
- throw new Error(`Attachment not found: ${ref}`);
61
- return row.bytes;
62
- }
63
- async delete(ref) {
64
- assertValidRef(ref);
65
- this.delStmt.run(ref);
66
- }
67
- }
68
- /** Refs are always sha256-of-content; reject anything else (parity with FS store). */
69
- function assertValidRef(ref) {
70
- if (!/^sha256:[0-9a-f]{64}$/.test(ref)) {
71
- throw new Error(`Invalid attachment ref: ${ref}`);
72
- }
73
- }
74
- //# sourceMappingURL=attachment-store.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"attachment-store.js","sourceRoot":"","sources":["../../src/mail/attachment-store.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AAGH,OAAO,EAAE,UAAU,EAAE,MAAM,aAAa,CAAC;AAGzC,MAAM,OAAO,qBAAqB;IAWtB;IAVF,KAAK,CAAS;IACd,OAAO,CAAqB;IAC5B,OAAO,CAAqB;IAC5B,OAAO,CAAqB;IAEpC;;;OAGG;IACH,YACU,EAAqB,EAC7B,SAAiB,EAAE;QADX,OAAE,GAAF,EAAE,CAAmB;QAG7B,IAAI,CAAC,KAAK,GAAG,GAAG,MAAM,kBAAkB,CAAC;QACzC,IAAI,CAAC,OAAO,EAAE,CAAC;QACf,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC,EAAE,CAAC,OAAO,CAC5B,eAAe,IAAI,CAAC,KAAK;;mCAEI,CAC9B,CAAC;QACF,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC,EAAE,CAAC,OAAO,CAC5B,qBAAqB,IAAI,CAAC,KAAK,gBAAgB,CAChD,CAAC;QACF,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC,EAAE,CAAC,OAAO,CAAC,eAAe,IAAI,CAAC,KAAK,gBAAgB,CAAC,CAAC;IAC5E,CAAC;IAEO,OAAO;QACb,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC;mCACkB,IAAI,CAAC,KAAK;;;;;;;;KAQxC,CAAC,CAAC;IACL,CAAC;IAED,KAAK,CAAC,GAAG,CACP,KAAa,EACb,IAAgD;QAEhD,MAAM,GAAG,GAAG,SAAS,GAAG,UAAU,CAAC,QAAQ,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;QACzE,8EAA8E;QAC9E,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC;YACf,GAAG;YACH,YAAY,EAAE,IAAI,CAAC,WAAW;YAC9B,QAAQ,EAAE,IAAI,CAAC,QAAQ,IAAI,IAAI;YAC/B,KAAK;YACL,UAAU,EAAE,KAAK,CAAC,MAAM;YACxB,UAAU,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;SACrC,CAAC,CAAC;QACH,OAAO,GAAG,CAAC;IACb,CAAC;IAED,KAAK,CAAC,GAAG,CAAC,GAAW;QACnB,cAAc,CAAC,GAAG,CAAC,CAAC;QACpB,MAAM,GAAG,GAAG,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,GAAG,CAAkC,CAAC;QACnE,IAAI,CAAC,GAAG;YAAE,MAAM,IAAI,KAAK,CAAC,yBAAyB,GAAG,EAAE,CAAC,CAAC;QAC1D,OAAO,GAAG,CAAC,KAAK,CAAC;IACnB,CAAC;IAED,KAAK,CAAC,MAAM,CAAC,GAAW;QACtB,cAAc,CAAC,GAAG,CAAC,CAAC;QACpB,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;IACxB,CAAC;CACF;AAED,sFAAsF;AACtF,SAAS,cAAc,CAAC,GAAW;IACjC,IAAI,CAAC,uBAAuB,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC;QACvC,MAAM,IAAI,KAAK,CAAC,2BAA2B,GAAG,EAAE,CAAC,CAAC;IACpD,CAAC;AACH,CAAC"}
@@ -1,41 +0,0 @@
1
- /**
2
- * Bidirectional mapping between RFC 5322 email and agent-inbox Messages.
3
- *
4
- * Parallels src/mesh/type-mapper.ts. Two pure functions:
5
- * - inboundMailToMessage: external email → inbox Message
6
- * - messageToOutboundMail: inbox Message → rendered OutboundMail
7
- *
8
- * Threading: RFC Message-ID / In-Reply-To / References map onto the inbox's
9
- * thread_tag + in_reply_to. The RFC Message-ID is preserved in metadata so a
10
- * later reply can reconstruct the In-Reply-To / References chain. See §5.
11
- */
12
- import type { Message } from "../types.js";
13
- import type { AddressBook } from "./address-book.js";
14
- import type { InboundMail, OutboundMail } from "./types.js";
15
- export interface InboundMapContext {
16
- addressBook: AddressBook;
17
- /** Scope assigned when plus-addressing does not specify one. */
18
- defaultScope?: string;
19
- /**
20
- * Resolve an RFC Message-ID we have already stored to its inbox Message id,
21
- * so an inbound reply can set `in_reply_to` to the real parent.
22
- */
23
- lookupInboxIdByRfcMessageId?: (rfcMessageId: string) => string | undefined;
24
- }
25
- export interface OutboundMapContext {
26
- addressBook: AddressBook;
27
- /** Domain used to mint Message-IDs. Defaults to the address book primary domain. */
28
- messageIdDomain?: string;
29
- /** The RFC Message-ID we recorded when the parent inbox message was sent/received. */
30
- lookupRfcMessageId?: (inboxId: string) => string | undefined;
31
- /** The References chain recorded for the parent inbox message. */
32
- lookupReferences?: (inboxId: string) => string[] | undefined;
33
- }
34
- /** Thrown when no inbound recipient resolves to a local agent. */
35
- export declare class UnknownRecipientError extends Error {
36
- readonly recipients: string[];
37
- constructor(recipients: string[]);
38
- }
39
- export declare function inboundMailToMessage(mail: InboundMail, ctx: InboundMapContext): Message;
40
- export declare function messageToOutboundMail(msg: Message, ctx: OutboundMapContext): OutboundMail;
41
- //# sourceMappingURL=email-mapper.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"email-mapper.d.ts","sourceRoot":"","sources":["../../src/mail/email-mapper.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;GAUG;AAKH,OAAO,KAAK,EAAE,OAAO,EAAwC,MAAM,aAAa,CAAC;AACjF,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,mBAAmB,CAAC;AACrD,OAAO,KAAK,EACV,WAAW,EACX,YAAY,EAGb,MAAM,YAAY,CAAC;AAMpB,MAAM,WAAW,iBAAiB;IAChC,WAAW,EAAE,WAAW,CAAC;IACzB,gEAAgE;IAChE,YAAY,CAAC,EAAE,MAAM,CAAC;IACtB;;;OAGG;IACH,2BAA2B,CAAC,EAAE,CAAC,YAAY,EAAE,MAAM,KAAK,MAAM,GAAG,SAAS,CAAC;CAC5E;AAED,MAAM,WAAW,kBAAkB;IACjC,WAAW,EAAE,WAAW,CAAC;IACzB,oFAAoF;IACpF,eAAe,CAAC,EAAE,MAAM,CAAC;IACzB,sFAAsF;IACtF,kBAAkB,CAAC,EAAE,CAAC,OAAO,EAAE,MAAM,KAAK,MAAM,GAAG,SAAS,CAAC;IAC7D,kEAAkE;IAClE,gBAAgB,CAAC,EAAE,CAAC,OAAO,EAAE,MAAM,KAAK,MAAM,EAAE,GAAG,SAAS,CAAC;CAC9D;AAED,kEAAkE;AAClE,qBAAa,qBAAsB,SAAQ,KAAK;aAClB,UAAU,EAAE,MAAM,EAAE;gBAApB,UAAU,EAAE,MAAM,EAAE;CAMjD;AAMD,wBAAgB,oBAAoB,CAClC,IAAI,EAAE,WAAW,EACjB,GAAG,EAAE,iBAAiB,GACrB,OAAO,CAkET;AAMD,wBAAgB,qBAAqB,CACnC,GAAG,EAAE,OAAO,EACZ,GAAG,EAAE,kBAAkB,GACtB,YAAY,CAyCd"}
@@ -1,216 +0,0 @@
1
- /**
2
- * Bidirectional mapping between RFC 5322 email and agent-inbox Messages.
3
- *
4
- * Parallels src/mesh/type-mapper.ts. Two pure functions:
5
- * - inboundMailToMessage: external email → inbox Message
6
- * - messageToOutboundMail: inbox Message → rendered OutboundMail
7
- *
8
- * Threading: RFC Message-ID / In-Reply-To / References map onto the inbox's
9
- * thread_tag + in_reply_to. The RFC Message-ID is preserved in metadata so a
10
- * later reply can reconstruct the In-Reply-To / References chain. See §5.
11
- */
12
- import { ulid } from "ulid";
13
- import { createHash } from "node:crypto";
14
- import { normalizeContent } from "../router/message-router.js";
15
- /** Thrown when no inbound recipient resolves to a local agent. */
16
- export class UnknownRecipientError extends Error {
17
- recipients;
18
- constructor(recipients) {
19
- super(`No inbound recipient resolved to a local agent: ${recipients.join(", ")}`);
20
- this.recipients = recipients;
21
- this.name = "UnknownRecipientError";
22
- }
23
- }
24
- // ---------------------------------------------------------------------------
25
- // Inbound: email → Message
26
- // ---------------------------------------------------------------------------
27
- export function inboundMailToMessage(mail, ctx) {
28
- const recipients = [];
29
- const unresolved = [];
30
- let scope;
31
- const addKind = (addrs, kind) => {
32
- for (const a of addrs ?? []) {
33
- const resolved = ctx.addressBook.resolveInbound(a.address);
34
- if (!resolved) {
35
- unresolved.push(a.address);
36
- continue;
37
- }
38
- recipients.push({ agent_id: resolved.agentId, kind });
39
- // First resolved recipient with a plus-address scope wins. "to" is
40
- // processed before "cc", so a to-scope still takes precedence, but a
41
- // cc-only scoped recipient is no longer silently dropped.
42
- if (scope === undefined && resolved.scope) {
43
- scope = resolved.scope;
44
- }
45
- }
46
- };
47
- addKind(mail.to, "to");
48
- addKind(mail.cc, "cc");
49
- if (recipients.length === 0) {
50
- throw new UnknownRecipientError(unresolved);
51
- }
52
- const rfcMessageId = mail.headers.messageId;
53
- const rfcInReplyTo = mail.headers.inReplyTo;
54
- const rfcReferences = mail.headers.references;
55
- const metadata = {};
56
- if (rfcMessageId)
57
- metadata.rfcMessageId = rfcMessageId;
58
- if (rfcInReplyTo)
59
- metadata.rfcInReplyTo = rfcInReplyTo;
60
- if (rfcReferences)
61
- metadata.rfcReferences = rfcReferences;
62
- if (mail.html)
63
- metadata.htmlBody = mail.html;
64
- if (mail.authResults)
65
- metadata.authResults = mail.authResults;
66
- if (mail.remote)
67
- metadata.remote = mail.remote;
68
- if (mail.envelopeFrom)
69
- metadata.envelopeFrom = mail.envelopeFrom;
70
- if (mail.bounce)
71
- metadata.bounce = mail.bounce;
72
- if (mail.attachments?.length)
73
- metadata.mailAttachments = mail.attachments;
74
- // Resolve in_reply_to to a real inbox id when we can; otherwise leave it unset
75
- // (the RFC id is retained in metadata for later correlation).
76
- let inReplyTo;
77
- if (rfcInReplyTo && ctx.lookupInboxIdByRfcMessageId) {
78
- inReplyTo = ctx.lookupInboxIdByRfcMessageId(rfcInReplyTo);
79
- }
80
- const text = mail.text ?? "";
81
- return {
82
- id: ulid(),
83
- scope: scope ?? ctx.defaultScope ?? "default",
84
- sender_id: mail.from.address,
85
- recipients,
86
- subject: mail.subject,
87
- content: normalizeContent(text),
88
- thread_tag: deriveThreadTag(rfcReferences, rfcInReplyTo, rfcMessageId),
89
- in_reply_to: inReplyTo,
90
- importance: importanceFromHeaders(mail.headers.raw),
91
- metadata,
92
- created_at: mail.receivedAt,
93
- };
94
- }
95
- // ---------------------------------------------------------------------------
96
- // Outbound: Message → email
97
- // ---------------------------------------------------------------------------
98
- export function messageToOutboundMail(msg, ctx) {
99
- const domain = ctx.messageIdDomain ?? ctx.addressBook.primaryDomain();
100
- const to = recipientsOfKind(msg, "to");
101
- const cc = recipientsOfKind(msg, "cc");
102
- const bcc = recipientsOfKind(msg, "bcc");
103
- const { inReplyTo, references } = buildThreadingHeaders(msg, ctx);
104
- const text = textFromContent(msg);
105
- const html = typeof msg.metadata.htmlBody === "string"
106
- ? msg.metadata.htmlBody
107
- : undefined;
108
- const attachments = validateAttachments(msg.metadata.mailAttachments);
109
- // Only forward an explicit, opt-in hints object to the backend. We do NOT
110
- // dump arbitrary internal message metadata to the external mail provider.
111
- const backendHints = msg.metadata.mailBackendHints &&
112
- typeof msg.metadata.mailBackendHints === "object"
113
- ? msg.metadata.mailBackendHints
114
- : undefined;
115
- return {
116
- idempotencyKey: msg.id,
117
- from: ctx.addressBook.resolveFrom(msg.sender_id),
118
- to,
119
- ...(cc.length ? { cc } : {}),
120
- ...(bcc.length ? { bcc } : {}),
121
- subject: msg.subject,
122
- text,
123
- ...(html ? { html } : {}),
124
- headers: {
125
- messageId: mintMessageId(msg.id, domain),
126
- ...(inReplyTo ? { inReplyTo } : {}),
127
- ...(references && references.length ? { references } : {}),
128
- },
129
- ...(attachments ? { attachments } : {}),
130
- ...(backendHints ? { backendHints } : {}),
131
- };
132
- }
133
- /** Validate untrusted attachment metadata, dropping malformed entries. */
134
- function validateAttachments(value) {
135
- if (!Array.isArray(value))
136
- return undefined;
137
- const out = [];
138
- for (const a of value) {
139
- if (a &&
140
- typeof a === "object" &&
141
- typeof a.contentRef === "string" &&
142
- typeof a.contentType === "string") {
143
- const att = a;
144
- out.push({
145
- contentRef: att.contentRef,
146
- contentType: att.contentType,
147
- sizeBytes: typeof att.sizeBytes === "number" ? att.sizeBytes : 0,
148
- ...(typeof att.filename === "string" ? { filename: att.filename } : {}),
149
- ...(typeof att.contentId === "string" ? { contentId: att.contentId } : {}),
150
- });
151
- }
152
- }
153
- return out.length ? out : undefined;
154
- }
155
- // ---------------------------------------------------------------------------
156
- // Helpers
157
- // ---------------------------------------------------------------------------
158
- function recipientsOfKind(msg, kind) {
159
- return msg.recipients
160
- .filter((r) => r.kind === kind)
161
- .map((r) => ({ address: r.agent_id }));
162
- }
163
- /** Stable thread tag from the root of the References chain. */
164
- function deriveThreadTag(references, inReplyTo, messageId) {
165
- const root = (references && references.length > 0 ? references[0] : undefined) ??
166
- inReplyTo ??
167
- messageId;
168
- if (!root)
169
- return undefined;
170
- const hash = createHash("sha1").update(root).digest("hex").slice(0, 12);
171
- return `mail-${hash}`;
172
- }
173
- function buildThreadingHeaders(msg, ctx) {
174
- if (!msg.in_reply_to) {
175
- // Fresh thread, but a recorded RFC References chain (e.g. from the inbound
176
- // side) should still be carried if present in metadata.
177
- const meta = msg.metadata.rfcReferences;
178
- return { references: Array.isArray(meta) ? meta : undefined };
179
- }
180
- const domain = ctx.messageIdDomain ?? ctx.addressBook.primaryDomain();
181
- const parentRfcId = ctx.lookupRfcMessageId?.(msg.in_reply_to) ??
182
- mintMessageId(msg.in_reply_to, domain);
183
- const parentRefs = ctx.lookupReferences?.(msg.in_reply_to) ??
184
- (Array.isArray(msg.metadata.rfcReferences)
185
- ? msg.metadata.rfcReferences
186
- : []);
187
- return {
188
- inReplyTo: parentRfcId,
189
- references: [...parentRefs, parentRfcId],
190
- };
191
- }
192
- function mintMessageId(inboxId, domain) {
193
- return `<${inboxId}@${domain}>`;
194
- }
195
- function textFromContent(msg) {
196
- const c = msg.content;
197
- if (c.type === "text" && typeof c.text === "string")
198
- return c.text;
199
- return JSON.stringify(c);
200
- }
201
- function importanceFromHeaders(raw) {
202
- const header = (k) => {
203
- const v = raw[k];
204
- return (Array.isArray(v) ? v[0] : v ?? "").toString().toLowerCase();
205
- };
206
- const importance = header("importance");
207
- const priority = header("x-priority");
208
- if (importance === "high" || priority.startsWith("1") || priority.startsWith("2")) {
209
- return "high";
210
- }
211
- if (importance === "low" || priority.startsWith("4") || priority.startsWith("5")) {
212
- return "low";
213
- }
214
- return "normal";
215
- }
216
- //# sourceMappingURL=email-mapper.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"email-mapper.js","sourceRoot":"","sources":["../../src/mail/email-mapper.ts"],"names":[],"mappings":"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"}
@@ -1,38 +0,0 @@
1
- /**
2
- * Filesystem-backed AttachmentStore — opt-in for large/high-volume deployments.
3
- *
4
- * Keeps attachment bytes out of the SQLite DB (no WAL bloat, no synchronous
5
- * blob I/O blocking the event loop). Content-addressed by sha256, so identical
6
- * bytes dedup. Files live at `<dir>/<aa>/<sha256>` (sharded by the first byte
7
- * to avoid one enormous directory).
8
- *
9
- * Durability: writes go to a unique temp file, are fsync'd, then atomically
10
- * renamed into place. Because the caller stores the attachment (this `put`)
11
- * BEFORE committing the message row that references it, a crash can only ever
12
- * leave an orphan file (reclaimable by GC) — never a row pointing at missing
13
- * bytes. See docs/MAIL-INTEROP-PLAN.md §4c.
14
- */
15
- import type { AttachmentStore } from "./types.js";
16
- export declare class FsAttachmentStore implements AttachmentStore {
17
- private dir;
18
- constructor(dir: string);
19
- put(bytes: Buffer, _meta: {
20
- contentType: string;
21
- filename?: string;
22
- }): Promise<string>;
23
- get(ref: string): Promise<Buffer>;
24
- delete(ref: string): Promise<void>;
25
- /**
26
- * Garbage-collect orphan files: delete any stored attachment whose ref is not
27
- * in `liveRefs` AND that is older than `minAgeMs` (default 1h).
28
- *
29
- * The age guard prevents a TOCTOU race: an attachment is `put` (file lands on
30
- * disk) before the message row that references it is committed, so a just-
31
- * written file may legitimately not appear in a `liveRefs` snapshot yet.
32
- * Skipping recently-written files avoids deleting such in-flight attachments.
33
- */
34
- gc(liveRefs: Set<string>, minAgeMs?: number): Promise<number>;
35
- private pathForHex;
36
- private pathForRef;
37
- }
38
- //# sourceMappingURL=fs-attachment-store.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"fs-attachment-store.d.ts","sourceRoot":"","sources":["../../src/mail/fs-attachment-store.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;GAaG;AAKH,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,YAAY,CAAC;AAIlD,qBAAa,iBAAkB,YAAW,eAAe;IAC3C,OAAO,CAAC,GAAG;gBAAH,GAAG,EAAE,MAAM;IAEzB,GAAG,CACP,KAAK,EAAE,MAAM,EACb,KAAK,EAAE;QAAE,WAAW,EAAE,MAAM,CAAC;QAAC,QAAQ,CAAC,EAAE,MAAM,CAAA;KAAE,GAChD,OAAO,CAAC,MAAM,CAAC;IAoCZ,GAAG,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,CAAC;IAYjC,MAAM,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IAIxC;;;;;;;;OAQG;IACG,EAAE,CAAC,QAAQ,EAAE,GAAG,CAAC,MAAM,CAAC,EAAE,QAAQ,GAAE,MAAkB,GAAG,OAAO,CAAC,MAAM,CAAC;IAoC9E,OAAO,CAAC,UAAU;IAKlB,OAAO,CAAC,UAAU;CASnB"}