@step-func-emailer/handlers 0.2.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (136) hide show
  1. package/dist/handlers/__tests__/bounce-handler.test.d.ts +2 -0
  2. package/dist/handlers/__tests__/bounce-handler.test.d.ts.map +1 -0
  3. package/dist/handlers/__tests__/bounce-handler.test.js +126 -0
  4. package/dist/handlers/__tests__/bounce-handler.test.js.map +1 -0
  5. package/dist/handlers/__tests__/check-condition.test.d.ts +2 -0
  6. package/dist/handlers/__tests__/check-condition.test.d.ts.map +1 -0
  7. package/dist/handlers/__tests__/check-condition.test.js +140 -0
  8. package/dist/handlers/__tests__/check-condition.test.js.map +1 -0
  9. package/dist/handlers/__tests__/engagement-handler.test.d.ts +2 -0
  10. package/dist/handlers/__tests__/engagement-handler.test.d.ts.map +1 -0
  11. package/dist/handlers/__tests__/engagement-handler.test.js +139 -0
  12. package/dist/handlers/__tests__/engagement-handler.test.js.map +1 -0
  13. package/dist/handlers/__tests__/send-email.test.d.ts +2 -0
  14. package/dist/handlers/__tests__/send-email.test.d.ts.map +1 -0
  15. package/dist/handlers/__tests__/send-email.test.js +205 -0
  16. package/dist/handlers/__tests__/send-email.test.js.map +1 -0
  17. package/dist/handlers/__tests__/unsubscribe.test.d.ts +2 -0
  18. package/dist/handlers/__tests__/unsubscribe.test.d.ts.map +1 -0
  19. package/dist/handlers/__tests__/unsubscribe.test.js +79 -0
  20. package/dist/handlers/__tests__/unsubscribe.test.js.map +1 -0
  21. package/dist/handlers/bounce-handler.d.ts +3 -0
  22. package/dist/handlers/bounce-handler.d.ts.map +1 -0
  23. package/dist/handlers/bounce-handler.js +55 -0
  24. package/dist/handlers/bounce-handler.js.map +1 -0
  25. package/dist/handlers/check-condition.d.ts +3 -0
  26. package/dist/handlers/check-condition.d.ts.map +1 -0
  27. package/dist/handlers/check-condition.js +74 -0
  28. package/dist/handlers/check-condition.js.map +1 -0
  29. package/dist/handlers/engagement-handler.d.ts +3 -0
  30. package/dist/handlers/engagement-handler.d.ts.map +1 -0
  31. package/dist/handlers/engagement-handler.js +98 -0
  32. package/dist/handlers/engagement-handler.js.map +1 -0
  33. package/dist/handlers/send-email.d.ts +5 -0
  34. package/dist/handlers/send-email.d.ts.map +1 -0
  35. package/dist/handlers/send-email.js +137 -0
  36. package/dist/handlers/send-email.js.map +1 -0
  37. package/dist/handlers/unsubscribe.d.ts +5 -0
  38. package/dist/handlers/unsubscribe.d.ts.map +1 -0
  39. package/dist/handlers/unsubscribe.js +53 -0
  40. package/dist/handlers/unsubscribe.js.map +1 -0
  41. package/dist/index.d.ts +8 -0
  42. package/dist/index.d.ts.map +1 -0
  43. package/dist/index.js +50 -0
  44. package/dist/index.js.map +1 -0
  45. package/dist/lib/__tests__/display-names.test.d.ts +2 -0
  46. package/dist/lib/__tests__/display-names.test.d.ts.map +1 -0
  47. package/dist/lib/__tests__/display-names.test.js +49 -0
  48. package/dist/lib/__tests__/display-names.test.js.map +1 -0
  49. package/dist/lib/__tests__/dynamo-client.test.d.ts +2 -0
  50. package/dist/lib/__tests__/dynamo-client.test.d.ts.map +1 -0
  51. package/dist/lib/__tests__/dynamo-client.test.js +229 -0
  52. package/dist/lib/__tests__/dynamo-client.test.js.map +1 -0
  53. package/dist/lib/__tests__/execution-stopper.test.d.ts +2 -0
  54. package/dist/lib/__tests__/execution-stopper.test.d.ts.map +1 -0
  55. package/dist/lib/__tests__/execution-stopper.test.js +56 -0
  56. package/dist/lib/__tests__/execution-stopper.test.js.map +1 -0
  57. package/dist/lib/__tests__/ses-sender.test.d.ts +2 -0
  58. package/dist/lib/__tests__/ses-sender.test.d.ts.map +1 -0
  59. package/dist/lib/__tests__/ses-sender.test.js +66 -0
  60. package/dist/lib/__tests__/ses-sender.test.js.map +1 -0
  61. package/dist/lib/__tests__/ssm-config.test.d.ts +2 -0
  62. package/dist/lib/__tests__/ssm-config.test.d.ts.map +1 -0
  63. package/dist/lib/__tests__/ssm-config.test.js +73 -0
  64. package/dist/lib/__tests__/ssm-config.test.js.map +1 -0
  65. package/dist/lib/__tests__/template-renderer.test.d.ts +2 -0
  66. package/dist/lib/__tests__/template-renderer.test.d.ts.map +1 -0
  67. package/dist/lib/__tests__/template-renderer.test.js +79 -0
  68. package/dist/lib/__tests__/template-renderer.test.js.map +1 -0
  69. package/dist/lib/__tests__/unsubscribe-token.test.d.ts +2 -0
  70. package/dist/lib/__tests__/unsubscribe-token.test.d.ts.map +1 -0
  71. package/dist/lib/__tests__/unsubscribe-token.test.js +74 -0
  72. package/dist/lib/__tests__/unsubscribe-token.test.js.map +1 -0
  73. package/dist/lib/display-names.d.ts +5 -0
  74. package/dist/lib/display-names.d.ts.map +1 -0
  75. package/dist/lib/display-names.js +46 -0
  76. package/dist/lib/display-names.js.map +1 -0
  77. package/dist/lib/dynamo-client.d.ts +13 -0
  78. package/dist/lib/dynamo-client.d.ts.map +1 -0
  79. package/dist/lib/dynamo-client.js +211 -0
  80. package/dist/lib/dynamo-client.js.map +1 -0
  81. package/dist/lib/execution-stopper.d.ts +2 -0
  82. package/dist/lib/execution-stopper.d.ts.map +1 -0
  83. package/dist/lib/execution-stopper.js +38 -0
  84. package/dist/lib/execution-stopper.js.map +1 -0
  85. package/dist/lib/logger.d.ts +8 -0
  86. package/dist/lib/logger.d.ts.map +1 -0
  87. package/dist/lib/logger.js +13 -0
  88. package/dist/lib/logger.js.map +1 -0
  89. package/dist/lib/ses-sender.d.ts +13 -0
  90. package/dist/lib/ses-sender.d.ts.map +1 -0
  91. package/dist/lib/ses-sender.js +86 -0
  92. package/dist/lib/ses-sender.js.map +1 -0
  93. package/dist/lib/ses-suppression.d.ts +2 -0
  94. package/dist/lib/ses-suppression.d.ts.map +1 -0
  95. package/dist/lib/ses-suppression.js +24 -0
  96. package/dist/lib/ses-suppression.js.map +1 -0
  97. package/dist/lib/ssm-config.d.ts +14 -0
  98. package/dist/lib/ssm-config.d.ts.map +1 -0
  99. package/dist/lib/ssm-config.js +54 -0
  100. package/dist/lib/ssm-config.js.map +1 -0
  101. package/dist/lib/template-renderer.d.ts +9 -0
  102. package/dist/lib/template-renderer.d.ts.map +1 -0
  103. package/dist/lib/template-renderer.js +36 -0
  104. package/dist/lib/template-renderer.js.map +1 -0
  105. package/dist/lib/unsubscribe-token.d.ts +13 -0
  106. package/dist/lib/unsubscribe-token.d.ts.map +1 -0
  107. package/dist/lib/unsubscribe-token.js +37 -0
  108. package/dist/lib/unsubscribe-token.js.map +1 -0
  109. package/package.json +41 -0
  110. package/src/handlers/__tests__/bounce-handler.test.ts +173 -0
  111. package/src/handlers/__tests__/check-condition.test.ts +172 -0
  112. package/src/handlers/__tests__/engagement-handler.test.ts +161 -0
  113. package/src/handlers/__tests__/send-email.test.ts +279 -0
  114. package/src/handlers/__tests__/unsubscribe.test.ts +111 -0
  115. package/src/handlers/bounce-handler.ts +91 -0
  116. package/src/handlers/check-condition.ts +77 -0
  117. package/src/handlers/engagement-handler.ts +169 -0
  118. package/src/handlers/send-email.ts +184 -0
  119. package/src/handlers/unsubscribe.ts +70 -0
  120. package/src/index.ts +10 -0
  121. package/src/lib/__tests__/display-names.test.ts +52 -0
  122. package/src/lib/__tests__/dynamo-client.test.ts +346 -0
  123. package/src/lib/__tests__/execution-stopper.test.ts +68 -0
  124. package/src/lib/__tests__/ses-sender.test.ts +85 -0
  125. package/src/lib/__tests__/ssm-config.test.ts +85 -0
  126. package/src/lib/__tests__/template-renderer.test.ts +96 -0
  127. package/src/lib/__tests__/unsubscribe-token.test.ts +79 -0
  128. package/src/lib/display-names.ts +54 -0
  129. package/src/lib/dynamo-client.ts +301 -0
  130. package/src/lib/execution-stopper.ts +40 -0
  131. package/src/lib/logger.ts +10 -0
  132. package/src/lib/ses-sender.ts +102 -0
  133. package/src/lib/ses-suppression.ts +30 -0
  134. package/src/lib/ssm-config.ts +81 -0
  135. package/src/lib/template-renderer.ts +52 -0
  136. package/src/lib/unsubscribe-token.ts +53 -0
@@ -0,0 +1,56 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ const vitest_1 = require("vitest");
4
+ const mockSfnSend = vitest_1.vi.fn();
5
+ vitest_1.vi.mock("@aws-sdk/client-sfn", () => ({
6
+ SFNClient: class {
7
+ send = mockSfnSend;
8
+ },
9
+ StopExecutionCommand: class {
10
+ input;
11
+ constructor(input) { this.input = input; }
12
+ },
13
+ }));
14
+ const mockGetAllExecutions = vitest_1.vi.fn();
15
+ const mockDeleteExecution = vitest_1.vi.fn();
16
+ vitest_1.vi.mock("../dynamo-client.js", () => ({
17
+ getAllExecutions: (...args) => mockGetAllExecutions(...args),
18
+ deleteExecution: (...args) => mockDeleteExecution(...args),
19
+ }));
20
+ const { stopAllExecutions } = await import("../execution-stopper.js");
21
+ beforeEach(() => {
22
+ mockSfnSend.mockReset();
23
+ mockGetAllExecutions.mockReset();
24
+ mockDeleteExecution.mockReset();
25
+ });
26
+ describe("stopAllExecutions", () => {
27
+ it("stops and deletes all active executions", async () => {
28
+ mockGetAllExecutions.mockResolvedValueOnce([
29
+ { executionArn: "arn:1", sequenceId: "onboarding" },
30
+ { executionArn: "arn:2", sequenceId: "win-back" },
31
+ ]);
32
+ mockSfnSend.mockResolvedValue({});
33
+ mockDeleteExecution.mockResolvedValue(undefined);
34
+ await stopAllExecutions("TestTable", "user@example.com");
35
+ expect(mockSfnSend).toHaveBeenCalledTimes(2);
36
+ expect(mockDeleteExecution).toHaveBeenCalledTimes(2);
37
+ expect(mockDeleteExecution).toHaveBeenCalledWith("TestTable", "user@example.com", "onboarding");
38
+ expect(mockDeleteExecution).toHaveBeenCalledWith("TestTable", "user@example.com", "win-back");
39
+ });
40
+ it("does nothing when no executions exist", async () => {
41
+ mockGetAllExecutions.mockResolvedValueOnce([]);
42
+ await stopAllExecutions("TestTable", "user@example.com");
43
+ expect(mockSfnSend).not.toHaveBeenCalled();
44
+ expect(mockDeleteExecution).not.toHaveBeenCalled();
45
+ });
46
+ it("continues when SFN stop fails (already stopped)", async () => {
47
+ mockGetAllExecutions.mockResolvedValueOnce([
48
+ { executionArn: "arn:1", sequenceId: "onboarding" },
49
+ ]);
50
+ mockSfnSend.mockRejectedValueOnce(new Error("ExecutionDoesNotExist"));
51
+ mockDeleteExecution.mockResolvedValue(undefined);
52
+ await expect(stopAllExecutions("TestTable", "user@example.com")).resolves.toBeUndefined();
53
+ expect(mockDeleteExecution).toHaveBeenCalledOnce();
54
+ });
55
+ });
56
+ //# sourceMappingURL=execution-stopper.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"execution-stopper.test.js","sourceRoot":"","sources":["../../../src/lib/__tests__/execution-stopper.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=ses-sender.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ses-sender.test.d.ts","sourceRoot":"","sources":["../../../src/lib/__tests__/ses-sender.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,66 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ const vitest_1 = require("vitest");
4
+ const mockSend = vitest_1.vi.fn();
5
+ vitest_1.vi.mock("@aws-sdk/client-sesv2", () => ({
6
+ SESv2Client: class {
7
+ send = mockSend;
8
+ },
9
+ SendEmailCommand: class {
10
+ input;
11
+ constructor(input) { this.input = input; }
12
+ },
13
+ }));
14
+ const { sendEmail } = await import("../ses-sender.js");
15
+ beforeEach(() => {
16
+ mockSend.mockReset();
17
+ });
18
+ describe("sendEmail", () => {
19
+ const defaultParams = {
20
+ from: "Test <noreply@example.com>",
21
+ to: "user@example.com",
22
+ subject: "Welcome!",
23
+ htmlBody: "<h1>Hello</h1>",
24
+ configurationSetName: "my-config-set",
25
+ unsubscribeUrl: "https://unsub.example.com?token=abc",
26
+ };
27
+ it("sends email and returns messageId", async () => {
28
+ mockSend.mockResolvedValueOnce({ MessageId: "ses-msg-123" });
29
+ const messageId = await sendEmail(defaultParams);
30
+ expect(messageId).toBe("ses-msg-123");
31
+ expect(mockSend).toHaveBeenCalledOnce();
32
+ });
33
+ it("returns 'unknown' when MessageId is missing", async () => {
34
+ mockSend.mockResolvedValueOnce({});
35
+ const messageId = await sendEmail(defaultParams);
36
+ expect(messageId).toBe("unknown");
37
+ });
38
+ it("includes List-Unsubscribe headers", async () => {
39
+ mockSend.mockResolvedValueOnce({ MessageId: "msg-1" });
40
+ await sendEmail(defaultParams);
41
+ const cmd = mockSend.mock.calls[0][0];
42
+ const headers = cmd.input.Content.Simple.Headers;
43
+ expect(headers).toContainEqual({
44
+ Name: "List-Unsubscribe",
45
+ Value: `<${defaultParams.unsubscribeUrl}>`,
46
+ });
47
+ expect(headers).toContainEqual({
48
+ Name: "List-Unsubscribe-Post",
49
+ Value: "List-Unsubscribe=One-Click",
50
+ });
51
+ });
52
+ it("sets correct destination and from address", async () => {
53
+ mockSend.mockResolvedValueOnce({ MessageId: "msg-1" });
54
+ await sendEmail(defaultParams);
55
+ const cmd = mockSend.mock.calls[0][0];
56
+ expect(cmd.input.FromEmailAddress).toBe(defaultParams.from);
57
+ expect(cmd.input.Destination.ToAddresses).toEqual([defaultParams.to]);
58
+ });
59
+ it("uses the configuration set name", async () => {
60
+ mockSend.mockResolvedValueOnce({ MessageId: "msg-1" });
61
+ await sendEmail(defaultParams);
62
+ const cmd = mockSend.mock.calls[0][0];
63
+ expect(cmd.input.ConfigurationSetName).toBe("my-config-set");
64
+ });
65
+ });
66
+ //# sourceMappingURL=ses-sender.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ses-sender.test.js","sourceRoot":"","sources":["../../../src/lib/__tests__/ses-sender.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=ssm-config.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ssm-config.test.d.ts","sourceRoot":"","sources":["../../../src/lib/__tests__/ssm-config.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,73 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ const vitest_1 = require("vitest");
4
+ const mockSend = vitest_1.vi.fn();
5
+ vitest_1.vi.mock("@aws-sdk/client-ssm", () => ({
6
+ SSMClient: class {
7
+ send = mockSend;
8
+ },
9
+ GetParameterCommand: class {
10
+ input;
11
+ constructor(input) { this.input = input; }
12
+ },
13
+ }));
14
+ // Must import after mock setup
15
+ const { getParameter, resolveConfig } = await import("../ssm-config.js");
16
+ beforeEach(() => {
17
+ mockSend.mockReset();
18
+ });
19
+ describe("getParameter", () => {
20
+ it("fetches from SSM and returns value", async () => {
21
+ mockSend.mockResolvedValueOnce({
22
+ Parameter: { Value: "my-table" },
23
+ });
24
+ const value = await getParameter("/test/param");
25
+ expect(value).toBe("my-table");
26
+ expect(mockSend).toHaveBeenCalledOnce();
27
+ });
28
+ it("throws when parameter not found", async () => {
29
+ mockSend.mockResolvedValueOnce({ Parameter: {} });
30
+ await expect(getParameter("/test/missing")).rejects.toThrow("SSM parameter not found");
31
+ });
32
+ it("caches values for subsequent calls", async () => {
33
+ mockSend.mockResolvedValueOnce({
34
+ Parameter: { Value: "cached-value" },
35
+ });
36
+ const v1 = await getParameter("/test/cached");
37
+ const v2 = await getParameter("/test/cached");
38
+ expect(v1).toBe("cached-value");
39
+ expect(v2).toBe("cached-value");
40
+ expect(mockSend).toHaveBeenCalledOnce();
41
+ });
42
+ });
43
+ describe("resolveConfig", () => {
44
+ it("resolves all 8 SSM parameters", async () => {
45
+ const params = {
46
+ "table-name": "MainTable",
47
+ "events-table-name": "EventsTable",
48
+ "template-bucket": "my-bucket",
49
+ "default-from-email": "noreply@example.com",
50
+ "default-from-name": "Example",
51
+ "ses-config-set": "my-config-set",
52
+ "unsubscribe-base-url": "https://unsub.example.com",
53
+ "unsubscribe-secret": "secret123",
54
+ };
55
+ mockSend.mockImplementation(async (cmd) => {
56
+ const suffix = cmd.input.Name.split("/").pop();
57
+ const value = params[suffix];
58
+ if (!value)
59
+ throw new Error(`Unexpected param: ${cmd.input.Name}`);
60
+ return { Parameter: { Value: value } };
61
+ });
62
+ const config = await resolveConfig();
63
+ expect(config.tableName).toBe("MainTable");
64
+ expect(config.eventsTableName).toBe("EventsTable");
65
+ expect(config.templateBucket).toBe("my-bucket");
66
+ expect(config.defaultFromEmail).toBe("noreply@example.com");
67
+ expect(config.defaultFromName).toBe("Example");
68
+ expect(config.sesConfigSet).toBe("my-config-set");
69
+ expect(config.unsubscribeBaseUrl).toBe("https://unsub.example.com");
70
+ expect(config.unsubscribeSecret).toBe("secret123");
71
+ });
72
+ });
73
+ //# sourceMappingURL=ssm-config.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ssm-config.test.js","sourceRoot":"","sources":["../../../src/lib/__tests__/ssm-config.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=template-renderer.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"template-renderer.test.d.ts","sourceRoot":"","sources":["../../../src/lib/__tests__/template-renderer.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,79 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ const vitest_1 = require("vitest");
4
+ const mockSend = vitest_1.vi.fn();
5
+ vitest_1.vi.mock("@aws-sdk/client-s3", () => ({
6
+ S3Client: class {
7
+ send = mockSend;
8
+ },
9
+ GetObjectCommand: class {
10
+ input;
11
+ constructor(input) { this.input = input; }
12
+ },
13
+ }));
14
+ const { renderTemplate } = await import("../template-renderer.js");
15
+ beforeEach(() => {
16
+ mockSend.mockReset();
17
+ });
18
+ describe("renderTemplate", () => {
19
+ it("fetches template from S3 and renders with LiquidJS", async () => {
20
+ mockSend.mockResolvedValueOnce({
21
+ Body: {
22
+ transformToString: () => Promise.resolve("<h1>Hello {{ firstName }}</h1>"),
23
+ },
24
+ });
25
+ const result = await renderTemplate("my-bucket", "welcome", {
26
+ email: "user@example.com",
27
+ firstName: "Jane",
28
+ unsubscribeUrl: "https://unsub.example.com",
29
+ currentYear: 2026,
30
+ });
31
+ expect(result).toBe("<h1>Hello Jane</h1>");
32
+ });
33
+ it("fetches template with .html extension", async () => {
34
+ mockSend.mockResolvedValueOnce({
35
+ Body: {
36
+ transformToString: () => Promise.resolve("<p>Hi</p>"),
37
+ },
38
+ });
39
+ await renderTemplate("my-bucket", "onboarding/step1", {
40
+ email: "user@example.com",
41
+ firstName: "Jane",
42
+ unsubscribeUrl: "https://unsub.example.com",
43
+ currentYear: 2026,
44
+ });
45
+ const cmd = mockSend.mock.calls[0][0];
46
+ expect(cmd.input.Bucket).toBe("my-bucket");
47
+ expect(cmd.input.Key).toBe("onboarding/step1.html");
48
+ });
49
+ it("renders multiple Liquid variables", async () => {
50
+ mockSend.mockResolvedValueOnce({
51
+ Body: {
52
+ transformToString: () => Promise.resolve("Hi {{ firstName }}, your email is {{ email }}. Year: {{ currentYear }}. <a href='{{ unsubscribeUrl }}'>Unsub</a>"),
53
+ },
54
+ });
55
+ const result = await renderTemplate("my-bucket", "multi-var", {
56
+ email: "jane@example.com",
57
+ firstName: "Jane",
58
+ unsubscribeUrl: "https://unsub.example.com?token=abc",
59
+ currentYear: 2026,
60
+ });
61
+ expect(result).toContain("Jane");
62
+ expect(result).toContain("jane@example.com");
63
+ expect(result).toContain("2026");
64
+ expect(result).toContain("https://unsub.example.com?token=abc");
65
+ });
66
+ it("handles empty template body gracefully", async () => {
67
+ mockSend.mockResolvedValueOnce({
68
+ Body: { transformToString: () => Promise.resolve("") },
69
+ });
70
+ const result = await renderTemplate("my-bucket", "empty", {
71
+ email: "user@example.com",
72
+ firstName: "Jane",
73
+ unsubscribeUrl: "https://unsub.example.com",
74
+ currentYear: 2026,
75
+ });
76
+ expect(result).toBe("");
77
+ });
78
+ });
79
+ //# sourceMappingURL=template-renderer.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"template-renderer.test.js","sourceRoot":"","sources":["../../../src/lib/__tests__/template-renderer.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=unsubscribe-token.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"unsubscribe-token.test.d.ts","sourceRoot":"","sources":["../../../src/lib/__tests__/unsubscribe-token.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,74 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ const unsubscribe_token_js_1 = require("../unsubscribe-token.js");
4
+ const SECRET = "test-secret-key";
5
+ describe("unsubscribe-token", () => {
6
+ describe("generateToken", () => {
7
+ it("returns a base64url-encoded string", () => {
8
+ const token = (0, unsubscribe_token_js_1.generateToken)("user@example.com", SECRET);
9
+ expect(token).toBeTruthy();
10
+ // base64url chars only
11
+ expect(token).toMatch(/^[A-Za-z0-9_-]+$/);
12
+ });
13
+ it("produces different tokens for different emails", () => {
14
+ const t1 = (0, unsubscribe_token_js_1.generateToken)("a@example.com", SECRET);
15
+ const t2 = (0, unsubscribe_token_js_1.generateToken)("b@example.com", SECRET);
16
+ expect(t1).not.toBe(t2);
17
+ });
18
+ it("produces different tokens for different secrets", () => {
19
+ const t1 = (0, unsubscribe_token_js_1.generateToken)("user@example.com", "secret-1");
20
+ const t2 = (0, unsubscribe_token_js_1.generateToken)("user@example.com", "secret-2");
21
+ expect(t1).not.toBe(t2);
22
+ });
23
+ });
24
+ describe("validateToken", () => {
25
+ it("validates a freshly generated token", () => {
26
+ const token = (0, unsubscribe_token_js_1.generateToken)("user@example.com", SECRET);
27
+ const result = (0, unsubscribe_token_js_1.validateToken)(token, SECRET);
28
+ expect(result.valid).toBe(true);
29
+ if (result.valid) {
30
+ expect(result.email).toBe("user@example.com");
31
+ expect(result.sendTimestamp).toBeTruthy();
32
+ expect(result.expiryTimestamp).toBeTruthy();
33
+ }
34
+ });
35
+ it("rejects token with wrong secret", () => {
36
+ const token = (0, unsubscribe_token_js_1.generateToken)("user@example.com", SECRET);
37
+ const result = (0, unsubscribe_token_js_1.validateToken)(token, "wrong-secret");
38
+ expect(result.valid).toBe(false);
39
+ if (!result.valid) {
40
+ expect(result.reason).toBe("invalid signature");
41
+ }
42
+ });
43
+ it("rejects malformed token", () => {
44
+ const result = (0, unsubscribe_token_js_1.validateToken)("not-a-valid-token", SECRET);
45
+ expect(result.valid).toBe(false);
46
+ });
47
+ it("rejects expired token", () => {
48
+ // Manually craft an expired token
49
+ const { createHmac } = require("node:crypto");
50
+ const email = "user@example.com";
51
+ const sendTimestamp = "2020-01-01T00:00:00.000Z";
52
+ const expiryTimestamp = "2020-04-01T00:00:00.000Z"; // expired
53
+ const payload = `${email}|${sendTimestamp}|${expiryTimestamp}`;
54
+ const signature = createHmac("sha256", SECRET)
55
+ .update(payload)
56
+ .digest("hex");
57
+ const token = Buffer.from(`${payload}|${signature}`).toString("base64url");
58
+ const result = (0, unsubscribe_token_js_1.validateToken)(token, SECRET);
59
+ expect(result.valid).toBe(false);
60
+ if (!result.valid) {
61
+ expect(result.reason).toBe("token expired");
62
+ }
63
+ });
64
+ it("rejects token with wrong part count", () => {
65
+ const token = Buffer.from("a|b|c|d|e").toString("base64url");
66
+ const result = (0, unsubscribe_token_js_1.validateToken)(token, SECRET);
67
+ expect(result.valid).toBe(false);
68
+ if (!result.valid) {
69
+ expect(result.reason).toBe("malformed token");
70
+ }
71
+ });
72
+ });
73
+ });
74
+ //# sourceMappingURL=unsubscribe-token.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"unsubscribe-token.test.js","sourceRoot":"","sources":["../../../src/lib/__tests__/unsubscribe-token.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,5 @@
1
+ type DisplayNameMap = Record<string, Record<string, string>>;
2
+ export declare function loadDisplayNames(bucket: string): Promise<DisplayNameMap>;
3
+ export declare function resolveDisplayNames(displayNameMap: DisplayNameMap, attributes: Record<string, unknown>): Record<string, string>;
4
+ export {};
5
+ //# sourceMappingURL=display-names.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"display-names.d.ts","sourceRoot":"","sources":["../../src/lib/display-names.ts"],"names":[],"mappings":"AAMA,KAAK,cAAc,GAAG,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC,CAAC;AAK7D,wBAAsB,gBAAgB,CAAC,MAAM,EAAE,MAAM,GAAG,OAAO,CAAC,cAAc,CAAC,CAuB9E;AAED,wBAAgB,mBAAmB,CACjC,cAAc,EAAE,cAAc,EAC9B,UAAU,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,GAClC,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAcxB"}
@@ -0,0 +1,46 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.loadDisplayNames = loadDisplayNames;
4
+ exports.resolveDisplayNames = resolveDisplayNames;
5
+ const client_s3_1 = require("@aws-sdk/client-s3");
6
+ const logger_js_1 = require("./logger.js");
7
+ const logger = (0, logger_js_1.createLogger)("display-names");
8
+ const s3 = new client_s3_1.S3Client({});
9
+ let cachedMap = null;
10
+ const CACHE_TTL_MS = 10 * 60 * 1000;
11
+ async function loadDisplayNames(bucket) {
12
+ if (cachedMap && Date.now() - cachedMap.fetchedAt < CACHE_TTL_MS) {
13
+ logger.debug("Display names cache hit");
14
+ return cachedMap.data;
15
+ }
16
+ try {
17
+ logger.debug("Loading display names from S3", { bucket });
18
+ const result = await s3.send(new client_s3_1.GetObjectCommand({
19
+ Bucket: bucket,
20
+ Key: "display-names.json",
21
+ }));
22
+ const body = (await result.Body?.transformToString()) ?? "{}";
23
+ const data = JSON.parse(body);
24
+ cachedMap = { data, fetchedAt: Date.now() };
25
+ logger.debug("Display names loaded", { mappingCount: Object.keys(data).length });
26
+ return data;
27
+ }
28
+ catch {
29
+ logger.debug("Display names file not found, using empty map");
30
+ return {};
31
+ }
32
+ }
33
+ function resolveDisplayNames(displayNameMap, attributes) {
34
+ const resolved = {};
35
+ for (const [displayKey, valueMap] of Object.entries(displayNameMap)) {
36
+ // Convention: displayKey "platformName" maps from attribute "platform"
37
+ // The source attribute key is derived by removing "Name" suffix
38
+ const sourceKey = displayKey.replace(/Name$/, "");
39
+ const rawValue = attributes[sourceKey];
40
+ if (typeof rawValue === "string") {
41
+ resolved[displayKey] = valueMap[rawValue] ?? rawValue;
42
+ }
43
+ }
44
+ return resolved;
45
+ }
46
+ //# sourceMappingURL=display-names.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"display-names.js","sourceRoot":"","sources":["../../src/lib/display-names.ts"],"names":[],"mappings":";;AAWA,4CAuBC;AAED,kDAiBC;AArDD,kDAAgE;AAChE,2CAA2C;AAE3C,MAAM,MAAM,GAAG,IAAA,wBAAY,EAAC,eAAe,CAAC,CAAC;AAC7C,MAAM,EAAE,GAAG,IAAI,oBAAQ,CAAC,EAAE,CAAC,CAAC;AAI5B,IAAI,SAAS,GAAuD,IAAI,CAAC;AACzE,MAAM,YAAY,GAAG,EAAE,GAAG,EAAE,GAAG,IAAI,CAAC;AAE7B,KAAK,UAAU,gBAAgB,CAAC,MAAc;IACnD,IAAI,SAAS,IAAI,IAAI,CAAC,GAAG,EAAE,GAAG,SAAS,CAAC,SAAS,GAAG,YAAY,EAAE,CAAC;QACjE,MAAM,CAAC,KAAK,CAAC,yBAAyB,CAAC,CAAC;QACxC,OAAO,SAAS,CAAC,IAAI,CAAC;IACxB,CAAC;IAED,IAAI,CAAC;QACH,MAAM,CAAC,KAAK,CAAC,+BAA+B,EAAE,EAAE,MAAM,EAAE,CAAC,CAAC;QAC1D,MAAM,MAAM,GAAG,MAAM,EAAE,CAAC,IAAI,CAC1B,IAAI,4BAAgB,CAAC;YACnB,MAAM,EAAE,MAAM;YACd,GAAG,EAAE,oBAAoB;SAC1B,CAAC,CACH,CAAC;QACF,MAAM,IAAI,GAAG,CAAC,MAAM,MAAM,CAAC,IAAI,EAAE,iBAAiB,EAAE,CAAC,IAAI,IAAI,CAAC;QAC9D,MAAM,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAmB,CAAC;QAChD,SAAS,GAAG,EAAE,IAAI,EAAE,SAAS,EAAE,IAAI,CAAC,GAAG,EAAE,EAAE,CAAC;QAC5C,MAAM,CAAC,KAAK,CAAC,sBAAsB,EAAE,EAAE,YAAY,EAAE,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC;QACjF,OAAO,IAAI,CAAC;IACd,CAAC;IAAC,MAAM,CAAC;QACP,MAAM,CAAC,KAAK,CAAC,+CAA+C,CAAC,CAAC;QAC9D,OAAO,EAAE,CAAC;IACZ,CAAC;AACH,CAAC;AAED,SAAgB,mBAAmB,CACjC,cAA8B,EAC9B,UAAmC;IAEnC,MAAM,QAAQ,GAA2B,EAAE,CAAC;IAE5C,KAAK,MAAM,CAAC,UAAU,EAAE,QAAQ,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,cAAc,CAAC,EAAE,CAAC;QACpE,uEAAuE;QACvE,gEAAgE;QAChE,MAAM,SAAS,GAAG,UAAU,CAAC,OAAO,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC;QAClD,MAAM,QAAQ,GAAG,UAAU,CAAC,SAAS,CAAC,CAAC;QACvC,IAAI,OAAO,QAAQ,KAAK,QAAQ,EAAE,CAAC;YACjC,QAAQ,CAAC,UAAU,CAAC,GAAG,QAAQ,CAAC,QAAQ,CAAC,IAAI,QAAQ,CAAC;QACxD,CAAC;IACH,CAAC;IAED,OAAO,QAAQ,CAAC;AAClB,CAAC"}
@@ -0,0 +1,13 @@
1
+ import type { Subscriber, SubscriberProfile, ActiveExecution, SendLog } from "@step-func-emailer/shared";
2
+ export declare function getSubscriberProfile(tableName: string, email: string): Promise<SubscriberProfile | null>;
3
+ export declare function extractAttributes(profile: Record<string, unknown>): Record<string, unknown>;
4
+ export declare function upsertSubscriberProfile(tableName: string, subscriber: Subscriber): Promise<void>;
5
+ export declare function getExecution(tableName: string, email: string, sequenceId: string): Promise<ActiveExecution | null>;
6
+ export declare function putExecution(tableName: string, email: string, sequenceId: string, executionArn: string): Promise<void>;
7
+ export declare function deleteExecution(tableName: string, email: string, sequenceId: string): Promise<void>;
8
+ export declare function getAllExecutions(tableName: string, email: string): Promise<ActiveExecution[]>;
9
+ export declare function writeSendLog(tableName: string, email: string, log: Omit<SendLog, "PK" | "SK" | "ttl">): Promise<void>;
10
+ export declare function hasBeenSent(tableName: string, email: string, templateKey: string): Promise<boolean>;
11
+ export declare function writeSuppression(tableName: string, email: string, reason: "bounce" | "complaint", bounceType: string | undefined, sesNotificationId: string): Promise<void>;
12
+ export declare function setProfileFlag(tableName: string, email: string, flag: "unsubscribed" | "suppressed"): Promise<void>;
13
+ //# sourceMappingURL=dynamo-client.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"dynamo-client.d.ts","sourceRoot":"","sources":["../../src/lib/dynamo-client.ts"],"names":[],"mappings":"AAmBA,OAAO,KAAK,EACV,UAAU,EACV,iBAAiB,EACjB,eAAe,EACf,OAAO,EACR,MAAM,2BAA2B,CAAC;AAQnC,wBAAsB,oBAAoB,CACxC,SAAS,EAAE,MAAM,EACjB,KAAK,EAAE,MAAM,GACZ,OAAO,CAAC,iBAAiB,GAAG,IAAI,CAAC,CAgBnC;AAaD,wBAAgB,iBAAiB,CAAC,OAAO,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,GAAG,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAQ3F;AAED,wBAAsB,uBAAuB,CAC3C,SAAS,EAAE,MAAM,EACjB,UAAU,EAAE,UAAU,GACrB,OAAO,CAAC,IAAI,CAAC,CAgDf;AAID,wBAAsB,YAAY,CAChC,SAAS,EAAE,MAAM,EACjB,KAAK,EAAE,MAAM,EACb,UAAU,EAAE,MAAM,GACjB,OAAO,CAAC,eAAe,GAAG,IAAI,CAAC,CAYjC;AAED,wBAAsB,YAAY,CAChC,SAAS,EAAE,MAAM,EACjB,KAAK,EAAE,MAAM,EACb,UAAU,EAAE,MAAM,EAClB,YAAY,EAAE,MAAM,GACnB,OAAO,CAAC,IAAI,CAAC,CAcf;AAED,wBAAsB,eAAe,CACnC,SAAS,EAAE,MAAM,EACjB,KAAK,EAAE,MAAM,EACb,UAAU,EAAE,MAAM,GACjB,OAAO,CAAC,IAAI,CAAC,CAWf;AAED,wBAAsB,gBAAgB,CACpC,SAAS,EAAE,MAAM,EACjB,KAAK,EAAE,MAAM,GACZ,OAAO,CAAC,eAAe,EAAE,CAAC,CAe5B;AAID,wBAAsB,YAAY,CAChC,SAAS,EAAE,MAAM,EACjB,KAAK,EAAE,MAAM,EACb,GAAG,EAAE,IAAI,CAAC,OAAO,EAAE,IAAI,GAAG,IAAI,GAAG,KAAK,CAAC,GACtC,OAAO,CAAC,IAAI,CAAC,CAoBf;AAED,wBAAsB,WAAW,CAC/B,SAAS,EAAE,MAAM,EACjB,KAAK,EAAE,MAAM,EACb,WAAW,EAAE,MAAM,GAClB,OAAO,CAAC,OAAO,CAAC,CAkBlB;AAID,wBAAsB,gBAAgB,CACpC,SAAS,EAAE,MAAM,EACjB,KAAK,EAAE,MAAM,EACb,MAAM,EAAE,QAAQ,GAAG,WAAW,EAC9B,UAAU,EAAE,MAAM,GAAG,SAAS,EAC9B,iBAAiB,EAAE,MAAM,GACxB,OAAO,CAAC,IAAI,CAAC,CAef;AAED,wBAAsB,cAAc,CAClC,SAAS,EAAE,MAAM,EACjB,KAAK,EAAE,MAAM,EACb,IAAI,EAAE,cAAc,GAAG,YAAY,GAClC,OAAO,CAAC,IAAI,CAAC,CAcf"}