@mailshot/cdk 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 (41) hide show
  1. package/dist/bin/app.d.ts +3 -0
  2. package/dist/bin/app.d.ts.map +1 -0
  3. package/dist/bin/app.js +94 -0
  4. package/dist/bin/app.js.map +1 -0
  5. package/dist/lib/constructs/event-bus.d.ts +16 -0
  6. package/dist/lib/constructs/event-bus.d.ts.map +1 -0
  7. package/dist/lib/constructs/event-bus.js +119 -0
  8. package/dist/lib/constructs/event-bus.js.map +1 -0
  9. package/dist/lib/constructs/lambdas.d.ts +26 -0
  10. package/dist/lib/constructs/lambdas.d.ts.map +1 -0
  11. package/dist/lib/constructs/lambdas.js +178 -0
  12. package/dist/lib/constructs/lambdas.js.map +1 -0
  13. package/dist/lib/constructs/ses-config.d.ts +14 -0
  14. package/dist/lib/constructs/ses-config.d.ts.map +1 -0
  15. package/dist/lib/constructs/ses-config.js +75 -0
  16. package/dist/lib/constructs/ses-config.js.map +1 -0
  17. package/dist/lib/constructs/ssm-params.d.ts +17 -0
  18. package/dist/lib/constructs/ssm-params.d.ts.map +1 -0
  19. package/dist/lib/constructs/ssm-params.js +63 -0
  20. package/dist/lib/constructs/ssm-params.js.map +1 -0
  21. package/dist/lib/constructs/state-machines.d.ts +21 -0
  22. package/dist/lib/constructs/state-machines.d.ts.map +1 -0
  23. package/dist/lib/constructs/state-machines.js +216 -0
  24. package/dist/lib/constructs/state-machines.js.map +1 -0
  25. package/dist/lib/constructs/storage.d.ts +17 -0
  26. package/dist/lib/constructs/storage.d.ts.map +1 -0
  27. package/dist/lib/constructs/storage.js +91 -0
  28. package/dist/lib/constructs/storage.js.map +1 -0
  29. package/dist/lib/load-sequences.d.ts +7 -0
  30. package/dist/lib/load-sequences.d.ts.map +1 -0
  31. package/dist/lib/load-sequences.js +76 -0
  32. package/dist/lib/load-sequences.js.map +1 -0
  33. package/dist/lib/mailshot-stack.d.ts +13 -0
  34. package/dist/lib/mailshot-stack.d.ts.map +1 -0
  35. package/dist/lib/mailshot-stack.js +155 -0
  36. package/dist/lib/mailshot-stack.js.map +1 -0
  37. package/dist/lib/step-func-emailer-stack.d.ts +13 -0
  38. package/dist/lib/step-func-emailer-stack.d.ts.map +1 -0
  39. package/dist/lib/step-func-emailer-stack.js +155 -0
  40. package/dist/lib/step-func-emailer-stack.js.map +1 -0
  41. package/package.json +55 -0
@@ -0,0 +1,63 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || (function () {
19
+ var ownKeys = function(o) {
20
+ ownKeys = Object.getOwnPropertyNames || function (o) {
21
+ var ar = [];
22
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
23
+ return ar;
24
+ };
25
+ return ownKeys(o);
26
+ };
27
+ return function (mod) {
28
+ if (mod && mod.__esModule) return mod;
29
+ var result = {};
30
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
31
+ __setModuleDefault(result, mod);
32
+ return result;
33
+ };
34
+ })();
35
+ Object.defineProperty(exports, "__esModule", { value: true });
36
+ exports.SsmConstruct = void 0;
37
+ const ssm = __importStar(require("aws-cdk-lib/aws-ssm"));
38
+ const constructs_1 = require("constructs");
39
+ class SsmConstruct extends constructs_1.Construct {
40
+ constructor(scope, id, props) {
41
+ super(scope, id);
42
+ const params = {
43
+ "table-name": props.tableName,
44
+ "events-table-name": props.eventsTableName,
45
+ "template-bucket": props.templateBucketName,
46
+ "default-from-email": props.defaultFromEmail,
47
+ "default-from-name": props.defaultFromName,
48
+ ...(props.replyToEmail ? { "reply-to-email": props.replyToEmail } : {}),
49
+ "ses-config-set": props.sesConfigSetName,
50
+ "unsubscribe-base-url": props.unsubscribeBaseUrl,
51
+ "unsubscribe-secret": props.unsubscribeSecret,
52
+ };
53
+ for (const [key, value] of Object.entries(params)) {
54
+ new ssm.StringParameter(this, `Param-${key}`, {
55
+ parameterName: `${props.prefix}/${key}`,
56
+ stringValue: value,
57
+ description: `Email Sequencer: ${key}`,
58
+ });
59
+ }
60
+ }
61
+ }
62
+ exports.SsmConstruct = SsmConstruct;
63
+ //# sourceMappingURL=ssm-params.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ssm-params.js","sourceRoot":"","sources":["../../../lib/constructs/ssm-params.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA,yDAA2C;AAC3C,2CAAuC;AAevC,MAAa,YAAa,SAAQ,sBAAS;IACzC,YAAY,KAAgB,EAAE,EAAU,EAAE,KAAwB;QAChE,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAEjB,MAAM,MAAM,GAA2B;YACrC,YAAY,EAAE,KAAK,CAAC,SAAS;YAC7B,mBAAmB,EAAE,KAAK,CAAC,eAAe;YAC1C,iBAAiB,EAAE,KAAK,CAAC,kBAAkB;YAC3C,oBAAoB,EAAE,KAAK,CAAC,gBAAgB;YAC5C,mBAAmB,EAAE,KAAK,CAAC,eAAe;YAC1C,GAAG,CAAC,KAAK,CAAC,YAAY,CAAC,CAAC,CAAC,EAAE,gBAAgB,EAAE,KAAK,CAAC,YAAY,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;YACvE,gBAAgB,EAAE,KAAK,CAAC,gBAAgB;YACxC,sBAAsB,EAAE,KAAK,CAAC,kBAAkB;YAChD,oBAAoB,EAAE,KAAK,CAAC,iBAAiB;SAC9C,CAAC;QAEF,KAAK,MAAM,CAAC,GAAG,EAAE,KAAK,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE,CAAC;YAClD,IAAI,GAAG,CAAC,eAAe,CAAC,IAAI,EAAE,SAAS,GAAG,EAAE,EAAE;gBAC5C,aAAa,EAAE,GAAG,KAAK,CAAC,MAAM,IAAI,GAAG,EAAE;gBACvC,WAAW,EAAE,KAAK;gBAClB,WAAW,EAAE,oBAAoB,GAAG,EAAE;aACvC,CAAC,CAAC;QACL,CAAC;IACH,CAAC;CACF;AAxBD,oCAwBC"}
@@ -0,0 +1,21 @@
1
+ import * as sfn from "aws-cdk-lib/aws-stepfunctions";
2
+ import type * as lambda from "aws-cdk-lib/aws-lambda";
3
+ import { Construct } from "constructs";
4
+ import type { SequenceDefinition } from "@mailshot/shared";
5
+ export interface StateMachinesProps {
6
+ sendEmailFn: lambda.IFunction;
7
+ checkConditionFn: lambda.IFunction;
8
+ sequences: SequenceDefinition[];
9
+ }
10
+ export declare class StateMachinesConstruct extends Construct {
11
+ readonly stateMachines: Map<string, sfn.StateMachine>;
12
+ private readonly retryConfig;
13
+ constructor(scope: Construct, id: string, props: StateMachinesProps);
14
+ private buildSequence;
15
+ private buildChain;
16
+ private buildSendStep;
17
+ private buildWaitStep;
18
+ private buildConditionStep;
19
+ private buildChoiceStep;
20
+ }
21
+ //# sourceMappingURL=state-machines.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"state-machines.d.ts","sourceRoot":"","sources":["../../../lib/constructs/state-machines.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,GAAG,MAAM,+BAA+B,CAAC;AAErD,OAAO,KAAK,KAAK,MAAM,MAAM,wBAAwB,CAAC;AACtD,OAAO,EAAE,SAAS,EAAE,MAAM,YAAY,CAAC;AACvC,OAAO,KAAK,EACV,kBAAkB,EAMnB,MAAM,kBAAkB,CAAC;AAE1B,MAAM,WAAW,kBAAkB;IACjC,WAAW,EAAE,MAAM,CAAC,SAAS,CAAC;IAC9B,gBAAgB,EAAE,MAAM,CAAC,SAAS,CAAC;IACnC,SAAS,EAAE,kBAAkB,EAAE,CAAC;CACjC;AASD,qBAAa,sBAAuB,SAAQ,SAAS;IACnD,SAAgB,aAAa,EAAE,GAAG,CAAC,MAAM,EAAE,GAAG,CAAC,YAAY,CAAC,CAAC;IAE7D,OAAO,CAAC,QAAQ,CAAC,WAAW,CAK1B;gBAEU,KAAK,EAAE,SAAS,EAAE,EAAE,EAAE,MAAM,EAAE,KAAK,EAAE,kBAAkB;IAWnE,OAAO,CAAC,aAAa;IA4DrB,OAAO,CAAC,UAAU;IAoDlB,OAAO,CAAC,aAAa;IAuBrB,OAAO,CAAC,aAAa;IAUrB,OAAO,CAAC,kBAAkB;IA2D1B,OAAO,CAAC,eAAe;CAsDxB"}
@@ -0,0 +1,216 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || (function () {
19
+ var ownKeys = function(o) {
20
+ ownKeys = Object.getOwnPropertyNames || function (o) {
21
+ var ar = [];
22
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
23
+ return ar;
24
+ };
25
+ return ownKeys(o);
26
+ };
27
+ return function (mod) {
28
+ if (mod && mod.__esModule) return mod;
29
+ var result = {};
30
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
31
+ __setModuleDefault(result, mod);
32
+ return result;
33
+ };
34
+ })();
35
+ Object.defineProperty(exports, "__esModule", { value: true });
36
+ exports.StateMachinesConstruct = void 0;
37
+ const cdk = __importStar(require("aws-cdk-lib"));
38
+ const sfn = __importStar(require("aws-cdk-lib/aws-stepfunctions"));
39
+ const tasks = __importStar(require("aws-cdk-lib/aws-stepfunctions-tasks"));
40
+ const constructs_1 = require("constructs");
41
+ function pascalCase(id) {
42
+ return id
43
+ .split(/[-_]/)
44
+ .map((s) => s.charAt(0).toUpperCase() + s.slice(1))
45
+ .join("");
46
+ }
47
+ class StateMachinesConstruct extends constructs_1.Construct {
48
+ stateMachines;
49
+ retryConfig = {
50
+ errors: ["States.TaskFailed"],
51
+ interval: cdk.Duration.seconds(60),
52
+ maxAttempts: 3,
53
+ backoffRate: 2,
54
+ };
55
+ constructor(scope, id, props) {
56
+ super(scope, id);
57
+ this.stateMachines = new Map();
58
+ for (const seq of props.sequences) {
59
+ const sm = this.buildSequence(seq, props.sendEmailFn, props.checkConditionFn);
60
+ this.stateMachines.set(seq.id, sm);
61
+ }
62
+ }
63
+ buildSequence(def, sendEmailFn, checkConditionFn) {
64
+ const prefix = pascalCase(def.id);
65
+ // Register task
66
+ const register = new tasks.LambdaInvoke(this, `${prefix}-Register`, {
67
+ lambdaFunction: sendEmailFn,
68
+ payload: sfn.TaskInput.fromObject({
69
+ action: "register",
70
+ sequenceId: def.id,
71
+ "subscriber.$": "$.subscriber",
72
+ "executionArn.$": "$$.Execution.Id",
73
+ }),
74
+ resultPath: "$.context",
75
+ payloadResponseOnly: true,
76
+ });
77
+ // Build step chain
78
+ const chain = this.buildChain(prefix, def.steps, sendEmailFn, checkConditionFn, { counter: 0 }, def.id);
79
+ // Complete task
80
+ const complete = new tasks.LambdaInvoke(this, `${prefix}-Complete`, {
81
+ lambdaFunction: sendEmailFn,
82
+ payload: sfn.TaskInput.fromObject({
83
+ action: "complete",
84
+ sequenceId: def.id,
85
+ "subscriber.$": "$.subscriber",
86
+ "executionArn.$": "$$.Execution.Id",
87
+ }),
88
+ resultPath: sfn.JsonPath.DISCARD,
89
+ payloadResponseOnly: true,
90
+ });
91
+ const succeed = new sfn.Succeed(this, `${prefix}-Done`);
92
+ // Chain: Register → steps → Complete → Succeed
93
+ let definition;
94
+ if (chain) {
95
+ definition = register.next(chain).next(complete).next(succeed);
96
+ }
97
+ else {
98
+ definition = register.next(complete).next(succeed);
99
+ }
100
+ return new sfn.StateMachine(this, `${prefix}Sequence`, {
101
+ stateMachineName: `${prefix}Sequence`,
102
+ definitionBody: sfn.DefinitionBody.fromChainable(definition),
103
+ timeout: cdk.Duration.minutes(def.timeoutMinutes),
104
+ });
105
+ }
106
+ buildChain(prefix, steps, sendEmailFn, checkConditionFn, ctx, sequenceId) {
107
+ let chain = null;
108
+ for (const step of steps) {
109
+ ctx.counter++;
110
+ const n = ctx.counter;
111
+ let state;
112
+ switch (step.type) {
113
+ case "send":
114
+ state = this.buildSendStep(prefix, n, step, sendEmailFn, sequenceId);
115
+ break;
116
+ case "wait":
117
+ state = this.buildWaitStep(prefix, n, step);
118
+ break;
119
+ case "condition":
120
+ state = this.buildConditionStep(prefix, n, step, sendEmailFn, checkConditionFn, ctx, sequenceId);
121
+ break;
122
+ case "choice":
123
+ state = this.buildChoiceStep(prefix, n, step, sendEmailFn, checkConditionFn, ctx, sequenceId);
124
+ break;
125
+ }
126
+ chain = chain ? chain.next(state) : sfn.Chain.start(state);
127
+ }
128
+ return chain;
129
+ }
130
+ buildSendStep(prefix, n, step, sendEmailFn, sequenceId) {
131
+ const task = new tasks.LambdaInvoke(this, `${prefix}-Send${n}`, {
132
+ lambdaFunction: sendEmailFn,
133
+ payload: sfn.TaskInput.fromObject({
134
+ action: "send",
135
+ templateKey: step.templateKey,
136
+ subject: step.subject,
137
+ sequenceId,
138
+ "subscriber.$": "$.subscriber",
139
+ }),
140
+ resultPath: "$.sendResult",
141
+ payloadResponseOnly: true,
142
+ });
143
+ task.addRetry(this.retryConfig);
144
+ return task;
145
+ }
146
+ buildWaitStep(prefix, n, step) {
147
+ const totalSeconds = (step.days ?? 0) * 86400 + (step.hours ?? 0) * 3600 + (step.minutes ?? 0) * 60;
148
+ return new sfn.Wait(this, `${prefix}-Wait${n}`, {
149
+ time: sfn.WaitTime.duration(cdk.Duration.seconds(totalSeconds)),
150
+ });
151
+ }
152
+ // Lambda-based condition check (reads from DynamoDB — for has_been_sent etc.)
153
+ buildConditionStep(prefix, n, step, sendEmailFn, checkConditionFn, ctx, sequenceId) {
154
+ const checkTask = new tasks.LambdaInvoke(this, `${prefix}-Check${n}`, {
155
+ lambdaFunction: checkConditionFn,
156
+ payload: sfn.TaskInput.fromObject({
157
+ check: step.check,
158
+ ...(step.field !== null && step.field !== undefined && { field: step.field }),
159
+ ...(step.value !== null && step.value !== undefined && { value: step.value }),
160
+ ...(step.templateKey !== null &&
161
+ step.templateKey !== undefined && { templateKey: step.templateKey }),
162
+ "subscriber.$": "$.subscriber",
163
+ }),
164
+ resultPath: "$.conditionResult",
165
+ payloadResponseOnly: true,
166
+ });
167
+ checkTask.addRetry(this.retryConfig);
168
+ const thenChain = this.buildChain(prefix, step.then, sendEmailFn, checkConditionFn, ctx, sequenceId);
169
+ const elseChain = this.buildChain(prefix, step.else ?? [], sendEmailFn, checkConditionFn, ctx, sequenceId);
170
+ const choice = new sfn.Choice(this, `${prefix}-Cond${n}`);
171
+ const thenState = thenChain ?? new sfn.Pass(this, `${prefix}-ThenPass${n}`);
172
+ const elseState = elseChain ?? new sfn.Pass(this, `${prefix}-ElsePass${n}`);
173
+ choice
174
+ .when(sfn.Condition.booleanEquals("$.conditionResult.result", true), thenState)
175
+ .otherwise(elseState);
176
+ const converge = new sfn.Pass(this, `${prefix}-CondMerge${n}`);
177
+ choice.afterwards().next(converge);
178
+ // Chain: checkTask → choice → (branches) → converge
179
+ // Use custom chain so the end state is converge (chainable), not choice
180
+ const checkChain = sfn.Chain.start(checkTask).next(choice);
181
+ return sfn.Chain.custom(checkChain.startState, [converge], converge);
182
+ }
183
+ // Native Step Functions Choice — no Lambda, evaluates JSONPath directly
184
+ buildChoiceStep(prefix, n, step, sendEmailFn, checkConditionFn, ctx, sequenceId) {
185
+ const choice = new sfn.Choice(this, `${prefix}-Choice${n}`);
186
+ // Build each branch and wire it to the Choice
187
+ for (const branch of step.branches) {
188
+ const branchChain = this.buildChain(prefix, branch.steps, sendEmailFn, checkConditionFn, ctx, sequenceId);
189
+ if (branchChain) {
190
+ choice.when(sfn.Condition.stringEquals(step.field, branch.value), branchChain);
191
+ }
192
+ }
193
+ // Default/otherwise branch
194
+ if (step.default && step.default.length > 0) {
195
+ const defaultChain = this.buildChain(prefix, step.default, sendEmailFn, checkConditionFn, ctx, sequenceId);
196
+ if (defaultChain) {
197
+ choice.otherwise(defaultChain);
198
+ }
199
+ else {
200
+ ctx.counter++;
201
+ choice.otherwise(new sfn.Pass(this, `${prefix}-DefaultPass${ctx.counter}`));
202
+ }
203
+ }
204
+ else {
205
+ ctx.counter++;
206
+ choice.otherwise(new sfn.Pass(this, `${prefix}-DefaultPass${ctx.counter}`));
207
+ }
208
+ // Converge all branches into a single Pass so the chain can continue
209
+ const converge = new sfn.Pass(this, `${prefix}-ChoiceMerge${n}`);
210
+ choice.afterwards().next(converge);
211
+ // Return a Chain that starts at choice but ends at converge (chainable)
212
+ return sfn.Chain.custom(choice.startState, [converge], converge);
213
+ }
214
+ }
215
+ exports.StateMachinesConstruct = StateMachinesConstruct;
216
+ //# sourceMappingURL=state-machines.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"state-machines.js","sourceRoot":"","sources":["../../../lib/constructs/state-machines.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA,iDAAmC;AACnC,mEAAqD;AACrD,2EAA6D;AAE7D,2CAAuC;AAgBvC,SAAS,UAAU,CAAC,EAAU;IAC5B,OAAO,EAAE;SACN,KAAK,CAAC,MAAM,CAAC;SACb,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,WAAW,EAAE,GAAG,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;SAClD,IAAI,CAAC,EAAE,CAAC,CAAC;AACd,CAAC;AAED,MAAa,sBAAuB,SAAQ,sBAAS;IACnC,aAAa,CAAgC;IAE5C,WAAW,GAAmB;QAC7C,MAAM,EAAE,CAAC,mBAAmB,CAAC;QAC7B,QAAQ,EAAE,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;QAClC,WAAW,EAAE,CAAC;QACd,WAAW,EAAE,CAAC;KACf,CAAC;IAEF,YAAY,KAAgB,EAAE,EAAU,EAAE,KAAyB;QACjE,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAEjB,IAAI,CAAC,aAAa,GAAG,IAAI,GAAG,EAAE,CAAC;QAE/B,KAAK,MAAM,GAAG,IAAI,KAAK,CAAC,SAAS,EAAE,CAAC;YAClC,MAAM,EAAE,GAAG,IAAI,CAAC,aAAa,CAAC,GAAG,EAAE,KAAK,CAAC,WAAW,EAAE,KAAK,CAAC,gBAAgB,CAAC,CAAC;YAC9E,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,EAAE,EAAE,CAAC,CAAC;QACrC,CAAC;IACH,CAAC;IAEO,aAAa,CACnB,GAAuB,EACvB,WAA6B,EAC7B,gBAAkC;QAElC,MAAM,MAAM,GAAG,UAAU,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;QAElC,gBAAgB;QAChB,MAAM,QAAQ,GAAG,IAAI,KAAK,CAAC,YAAY,CAAC,IAAI,EAAE,GAAG,MAAM,WAAW,EAAE;YAClE,cAAc,EAAE,WAAW;YAC3B,OAAO,EAAE,GAAG,CAAC,SAAS,CAAC,UAAU,CAAC;gBAChC,MAAM,EAAE,UAAU;gBAClB,UAAU,EAAE,GAAG,CAAC,EAAE;gBAClB,cAAc,EAAE,cAAc;gBAC9B,gBAAgB,EAAE,iBAAiB;aACpC,CAAC;YACF,UAAU,EAAE,WAAW;YACvB,mBAAmB,EAAE,IAAI;SAC1B,CAAC,CAAC;QAEH,mBAAmB;QACnB,MAAM,KAAK,GAAG,IAAI,CAAC,UAAU,CAC3B,MAAM,EACN,GAAG,CAAC,KAAK,EACT,WAAW,EACX,gBAAgB,EAChB,EAAE,OAAO,EAAE,CAAC,EAAE,EACd,GAAG,CAAC,EAAE,CACP,CAAC;QAEF,gBAAgB;QAChB,MAAM,QAAQ,GAAG,IAAI,KAAK,CAAC,YAAY,CAAC,IAAI,EAAE,GAAG,MAAM,WAAW,EAAE;YAClE,cAAc,EAAE,WAAW;YAC3B,OAAO,EAAE,GAAG,CAAC,SAAS,CAAC,UAAU,CAAC;gBAChC,MAAM,EAAE,UAAU;gBAClB,UAAU,EAAE,GAAG,CAAC,EAAE;gBAClB,cAAc,EAAE,cAAc;gBAC9B,gBAAgB,EAAE,iBAAiB;aACpC,CAAC;YACF,UAAU,EAAE,GAAG,CAAC,QAAQ,CAAC,OAAO;YAChC,mBAAmB,EAAE,IAAI;SAC1B,CAAC,CAAC;QAEH,MAAM,OAAO,GAAG,IAAI,GAAG,CAAC,OAAO,CAAC,IAAI,EAAE,GAAG,MAAM,OAAO,CAAC,CAAC;QAExD,+CAA+C;QAC/C,IAAI,UAA0B,CAAC;QAC/B,IAAI,KAAK,EAAE,CAAC;YACV,UAAU,GAAG,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QACjE,CAAC;aAAM,CAAC;YACN,UAAU,GAAG,QAAQ,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QACrD,CAAC;QAED,OAAO,IAAI,GAAG,CAAC,YAAY,CAAC,IAAI,EAAE,GAAG,MAAM,UAAU,EAAE;YACrD,gBAAgB,EAAE,GAAG,MAAM,UAAU;YACrC,cAAc,EAAE,GAAG,CAAC,cAAc,CAAC,aAAa,CAAC,UAAU,CAAC;YAC5D,OAAO,EAAE,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,cAAc,CAAC;SAClD,CAAC,CAAC;IACL,CAAC;IAEO,UAAU,CAChB,MAAc,EACd,KAAqB,EACrB,WAA6B,EAC7B,gBAAkC,EAClC,GAAwB,EACxB,UAAkB;QAElB,IAAI,KAAK,GAAqB,IAAI,CAAC;QAEnC,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE,CAAC;YACzB,GAAG,CAAC,OAAO,EAAE,CAAC;YACd,MAAM,CAAC,GAAG,GAAG,CAAC,OAAO,CAAC;YACtB,IAAI,KAAqB,CAAC;YAE1B,QAAQ,IAAI,CAAC,IAAI,EAAE,CAAC;gBAClB,KAAK,MAAM;oBACT,KAAK,GAAG,IAAI,CAAC,aAAa,CAAC,MAAM,EAAE,CAAC,EAAE,IAAI,EAAE,WAAW,EAAE,UAAU,CAAC,CAAC;oBACrE,MAAM;gBACR,KAAK,MAAM;oBACT,KAAK,GAAG,IAAI,CAAC,aAAa,CAAC,MAAM,EAAE,CAAC,EAAE,IAAI,CAAC,CAAC;oBAC5C,MAAM;gBACR,KAAK,WAAW;oBACd,KAAK,GAAG,IAAI,CAAC,kBAAkB,CAC7B,MAAM,EACN,CAAC,EACD,IAAI,EACJ,WAAW,EACX,gBAAgB,EAChB,GAAG,EACH,UAAU,CACX,CAAC;oBACF,MAAM;gBACR,KAAK,QAAQ;oBACX,KAAK,GAAG,IAAI,CAAC,eAAe,CAC1B,MAAM,EACN,CAAC,EACD,IAAI,EACJ,WAAW,EACX,gBAAgB,EAChB,GAAG,EACH,UAAU,CACX,CAAC;oBACF,MAAM;YACV,CAAC;YAED,KAAK,GAAG,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,KAAK,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;QAC7D,CAAC;QAED,OAAO,KAAK,CAAC;IACf,CAAC;IAEO,aAAa,CACnB,MAAc,EACd,CAAS,EACT,IAAc,EACd,WAA6B,EAC7B,UAAkB;QAElB,MAAM,IAAI,GAAG,IAAI,KAAK,CAAC,YAAY,CAAC,IAAI,EAAE,GAAG,MAAM,QAAQ,CAAC,EAAE,EAAE;YAC9D,cAAc,EAAE,WAAW;YAC3B,OAAO,EAAE,GAAG,CAAC,SAAS,CAAC,UAAU,CAAC;gBAChC,MAAM,EAAE,MAAM;gBACd,WAAW,EAAE,IAAI,CAAC,WAAW;gBAC7B,OAAO,EAAE,IAAI,CAAC,OAAO;gBACrB,UAAU;gBACV,cAAc,EAAE,cAAc;aAC/B,CAAC;YACF,UAAU,EAAE,cAAc;YAC1B,mBAAmB,EAAE,IAAI;SAC1B,CAAC,CAAC;QACH,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;QAChC,OAAO,IAAI,CAAC;IACd,CAAC;IAEO,aAAa,CAAC,MAAc,EAAE,CAAS,EAAE,IAAc;QAC7D,MAAM,YAAY,GAChB,CAAC,IAAI,CAAC,IAAI,IAAI,CAAC,CAAC,GAAG,KAAK,GAAG,CAAC,IAAI,CAAC,KAAK,IAAI,CAAC,CAAC,GAAG,IAAI,GAAG,CAAC,IAAI,CAAC,OAAO,IAAI,CAAC,CAAC,GAAG,EAAE,CAAC;QAEjF,OAAO,IAAI,GAAG,CAAC,IAAI,CAAC,IAAI,EAAE,GAAG,MAAM,QAAQ,CAAC,EAAE,EAAE;YAC9C,IAAI,EAAE,GAAG,CAAC,QAAQ,CAAC,QAAQ,CAAC,GAAG,CAAC,QAAQ,CAAC,OAAO,CAAC,YAAY,CAAC,CAAC;SAChE,CAAC,CAAC;IACL,CAAC;IAED,8EAA8E;IACtE,kBAAkB,CACxB,MAAc,EACd,CAAS,EACT,IAAmB,EACnB,WAA6B,EAC7B,gBAAkC,EAClC,GAAwB,EACxB,UAAkB;QAElB,MAAM,SAAS,GAAG,IAAI,KAAK,CAAC,YAAY,CAAC,IAAI,EAAE,GAAG,MAAM,SAAS,CAAC,EAAE,EAAE;YACpE,cAAc,EAAE,gBAAgB;YAChC,OAAO,EAAE,GAAG,CAAC,SAAS,CAAC,UAAU,CAAC;gBAChC,KAAK,EAAE,IAAI,CAAC,KAAK;gBACjB,GAAG,CAAC,IAAI,CAAC,KAAK,KAAK,IAAI,IAAI,IAAI,CAAC,KAAK,KAAK,SAAS,IAAI,EAAE,KAAK,EAAE,IAAI,CAAC,KAAK,EAAE,CAAC;gBAC7E,GAAG,CAAC,IAAI,CAAC,KAAK,KAAK,IAAI,IAAI,IAAI,CAAC,KAAK,KAAK,SAAS,IAAI,EAAE,KAAK,EAAE,IAAI,CAAC,KAAK,EAAE,CAAC;gBAC7E,GAAG,CAAC,IAAI,CAAC,WAAW,KAAK,IAAI;oBAC3B,IAAI,CAAC,WAAW,KAAK,SAAS,IAAI,EAAE,WAAW,EAAE,IAAI,CAAC,WAAW,EAAE,CAAC;gBACtE,cAAc,EAAE,cAAc;aAC/B,CAAC;YACF,UAAU,EAAE,mBAAmB;YAC/B,mBAAmB,EAAE,IAAI;SAC1B,CAAC,CAAC;QACH,SAAS,CAAC,QAAQ,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;QAErC,MAAM,SAAS,GAAG,IAAI,CAAC,UAAU,CAC/B,MAAM,EACN,IAAI,CAAC,IAAI,EACT,WAAW,EACX,gBAAgB,EAChB,GAAG,EACH,UAAU,CACX,CAAC;QACF,MAAM,SAAS,GAAG,IAAI,CAAC,UAAU,CAC/B,MAAM,EACN,IAAI,CAAC,IAAI,IAAI,EAAE,EACf,WAAW,EACX,gBAAgB,EAChB,GAAG,EACH,UAAU,CACX,CAAC;QAEF,MAAM,MAAM,GAAG,IAAI,GAAG,CAAC,MAAM,CAAC,IAAI,EAAE,GAAG,MAAM,QAAQ,CAAC,EAAE,CAAC,CAAC;QAC1D,MAAM,SAAS,GAAG,SAAS,IAAI,IAAI,GAAG,CAAC,IAAI,CAAC,IAAI,EAAE,GAAG,MAAM,YAAY,CAAC,EAAE,CAAC,CAAC;QAC5E,MAAM,SAAS,GAAG,SAAS,IAAI,IAAI,GAAG,CAAC,IAAI,CAAC,IAAI,EAAE,GAAG,MAAM,YAAY,CAAC,EAAE,CAAC,CAAC;QAE5E,MAAM;aACH,IAAI,CAAC,GAAG,CAAC,SAAS,CAAC,aAAa,CAAC,0BAA0B,EAAE,IAAI,CAAC,EAAE,SAAS,CAAC;aAC9E,SAAS,CAAC,SAAS,CAAC,CAAC;QAExB,MAAM,QAAQ,GAAG,IAAI,GAAG,CAAC,IAAI,CAAC,IAAI,EAAE,GAAG,MAAM,aAAa,CAAC,EAAE,CAAC,CAAC;QAC/D,MAAM,CAAC,UAAU,EAAE,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;QAEnC,oDAAoD;QACpD,wEAAwE;QACxE,MAAM,UAAU,GAAG,GAAG,CAAC,KAAK,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QAC3D,OAAO,GAAG,CAAC,KAAK,CAAC,MAAM,CAAC,UAAU,CAAC,UAAU,EAAE,CAAC,QAAQ,CAAC,EAAE,QAAQ,CAAC,CAAC;IACvE,CAAC;IAED,wEAAwE;IAChE,eAAe,CACrB,MAAc,EACd,CAAS,EACT,IAAgB,EAChB,WAA6B,EAC7B,gBAAkC,EAClC,GAAwB,EACxB,UAAkB;QAElB,MAAM,MAAM,GAAG,IAAI,GAAG,CAAC,MAAM,CAAC,IAAI,EAAE,GAAG,MAAM,UAAU,CAAC,EAAE,CAAC,CAAC;QAE5D,8CAA8C;QAC9C,KAAK,MAAM,MAAM,IAAI,IAAI,CAAC,QAAQ,EAAE,CAAC;YACnC,MAAM,WAAW,GAAG,IAAI,CAAC,UAAU,CACjC,MAAM,EACN,MAAM,CAAC,KAAK,EACZ,WAAW,EACX,gBAAgB,EAChB,GAAG,EACH,UAAU,CACX,CAAC;YACF,IAAI,WAAW,EAAE,CAAC;gBAChB,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,SAAS,CAAC,YAAY,CAAC,IAAI,CAAC,KAAK,EAAE,MAAM,CAAC,KAAK,CAAC,EAAE,WAAW,CAAC,CAAC;YACjF,CAAC;QACH,CAAC;QAED,2BAA2B;QAC3B,IAAI,IAAI,CAAC,OAAO,IAAI,IAAI,CAAC,OAAO,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YAC5C,MAAM,YAAY,GAAG,IAAI,CAAC,UAAU,CAClC,MAAM,EACN,IAAI,CAAC,OAAO,EACZ,WAAW,EACX,gBAAgB,EAChB,GAAG,EACH,UAAU,CACX,CAAC;YACF,IAAI,YAAY,EAAE,CAAC;gBACjB,MAAM,CAAC,SAAS,CAAC,YAAY,CAAC,CAAC;YACjC,CAAC;iBAAM,CAAC;gBACN,GAAG,CAAC,OAAO,EAAE,CAAC;gBACd,MAAM,CAAC,SAAS,CAAC,IAAI,GAAG,CAAC,IAAI,CAAC,IAAI,EAAE,GAAG,MAAM,eAAe,GAAG,CAAC,OAAO,EAAE,CAAC,CAAC,CAAC;YAC9E,CAAC;QACH,CAAC;aAAM,CAAC;YACN,GAAG,CAAC,OAAO,EAAE,CAAC;YACd,MAAM,CAAC,SAAS,CAAC,IAAI,GAAG,CAAC,IAAI,CAAC,IAAI,EAAE,GAAG,MAAM,eAAe,GAAG,CAAC,OAAO,EAAE,CAAC,CAAC,CAAC;QAC9E,CAAC;QAED,qEAAqE;QACrE,MAAM,QAAQ,GAAG,IAAI,GAAG,CAAC,IAAI,CAAC,IAAI,EAAE,GAAG,MAAM,eAAe,CAAC,EAAE,CAAC,CAAC;QACjE,MAAM,CAAC,UAAU,EAAE,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;QAEnC,wEAAwE;QACxE,OAAO,GAAG,CAAC,KAAK,CAAC,MAAM,CAAC,MAAM,CAAC,UAAU,EAAE,CAAC,QAAQ,CAAC,EAAE,QAAQ,CAAC,CAAC;IACnE,CAAC;CACF;AAvRD,wDAuRC"}
@@ -0,0 +1,17 @@
1
+ import * as dynamodb from "aws-cdk-lib/aws-dynamodb";
2
+ import * as s3 from "aws-cdk-lib/aws-s3";
3
+ import { Construct } from "constructs";
4
+ export interface StorageProps {
5
+ tableName: string;
6
+ eventsTableName: string;
7
+ templateBucketName: string;
8
+ sequenceIds: string[];
9
+ templateBuildDir: string;
10
+ }
11
+ export declare class StorageConstruct extends Construct {
12
+ readonly table: dynamodb.Table;
13
+ readonly eventsTable: dynamodb.Table;
14
+ readonly templateBucket: s3.Bucket;
15
+ constructor(scope: Construct, id: string, props: StorageProps);
16
+ }
17
+ //# sourceMappingURL=storage.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"storage.d.ts","sourceRoot":"","sources":["../../../lib/constructs/storage.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,QAAQ,MAAM,0BAA0B,CAAC;AACrD,OAAO,KAAK,EAAE,MAAM,oBAAoB,CAAC;AAEzC,OAAO,EAAE,SAAS,EAAE,MAAM,YAAY,CAAC;AAEvC,MAAM,WAAW,YAAY;IAC3B,SAAS,EAAE,MAAM,CAAC;IAClB,eAAe,EAAE,MAAM,CAAC;IACxB,kBAAkB,EAAE,MAAM,CAAC;IAC3B,WAAW,EAAE,MAAM,EAAE,CAAC;IACtB,gBAAgB,EAAE,MAAM,CAAC;CAC1B;AAED,qBAAa,gBAAiB,SAAQ,SAAS;IAC7C,SAAgB,KAAK,EAAE,QAAQ,CAAC,KAAK,CAAC;IACtC,SAAgB,WAAW,EAAE,QAAQ,CAAC,KAAK,CAAC;IAC5C,SAAgB,cAAc,EAAE,EAAE,CAAC,MAAM,CAAC;gBAE9B,KAAK,EAAE,SAAS,EAAE,EAAE,EAAE,MAAM,EAAE,KAAK,EAAE,YAAY;CA+C9D"}
@@ -0,0 +1,91 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || (function () {
19
+ var ownKeys = function(o) {
20
+ ownKeys = Object.getOwnPropertyNames || function (o) {
21
+ var ar = [];
22
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
23
+ return ar;
24
+ };
25
+ return ownKeys(o);
26
+ };
27
+ return function (mod) {
28
+ if (mod && mod.__esModule) return mod;
29
+ var result = {};
30
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
31
+ __setModuleDefault(result, mod);
32
+ return result;
33
+ };
34
+ })();
35
+ Object.defineProperty(exports, "__esModule", { value: true });
36
+ exports.StorageConstruct = void 0;
37
+ const path = __importStar(require("node:path"));
38
+ const cdk = __importStar(require("aws-cdk-lib"));
39
+ const dynamodb = __importStar(require("aws-cdk-lib/aws-dynamodb"));
40
+ const s3 = __importStar(require("aws-cdk-lib/aws-s3"));
41
+ const s3deploy = __importStar(require("aws-cdk-lib/aws-s3-deployment"));
42
+ const constructs_1 = require("constructs");
43
+ class StorageConstruct extends constructs_1.Construct {
44
+ table;
45
+ eventsTable;
46
+ templateBucket;
47
+ constructor(scope, id, props) {
48
+ super(scope, id);
49
+ this.table = new dynamodb.Table(this, "Table", {
50
+ tableName: props.tableName,
51
+ partitionKey: { name: "PK", type: dynamodb.AttributeType.STRING },
52
+ sortKey: { name: "SK", type: dynamodb.AttributeType.STRING },
53
+ billingMode: dynamodb.BillingMode.PAY_PER_REQUEST,
54
+ timeToLiveAttribute: "ttl",
55
+ removalPolicy: cdk.RemovalPolicy.RETAIN,
56
+ pointInTimeRecoverySpecification: { pointInTimeRecoveryEnabled: true },
57
+ });
58
+ // ── Events table (engagement tracking) ─────────────────────────────
59
+ this.eventsTable = new dynamodb.Table(this, "EventsTable", {
60
+ tableName: props.eventsTableName,
61
+ partitionKey: { name: "PK", type: dynamodb.AttributeType.STRING },
62
+ sortKey: { name: "SK", type: dynamodb.AttributeType.STRING },
63
+ billingMode: dynamodb.BillingMode.PAY_PER_REQUEST,
64
+ timeToLiveAttribute: "ttl",
65
+ removalPolicy: cdk.RemovalPolicy.RETAIN,
66
+ pointInTimeRecoverySpecification: { pointInTimeRecoveryEnabled: true },
67
+ });
68
+ this.eventsTable.addGlobalSecondaryIndex({
69
+ indexName: "TemplateIndex",
70
+ partitionKey: { name: "templateKey", type: dynamodb.AttributeType.STRING },
71
+ sortKey: { name: "SK", type: dynamodb.AttributeType.STRING },
72
+ projectionType: dynamodb.ProjectionType.ALL,
73
+ });
74
+ this.templateBucket = new s3.Bucket(this, "TemplateBucket", {
75
+ bucketName: props.templateBucketName,
76
+ removalPolicy: cdk.RemovalPolicy.RETAIN,
77
+ blockPublicAccess: s3.BlockPublicAccess.BLOCK_ALL,
78
+ encryption: s3.BucketEncryption.S3_MANAGED,
79
+ });
80
+ // Deploy rendered templates from build/<sequenceId>/templates/ directories
81
+ for (const seqId of props.sequenceIds) {
82
+ new s3deploy.BucketDeployment(this, `DeployTemplates-${seqId}`, {
83
+ sources: [s3deploy.Source.asset(path.join(props.templateBuildDir, seqId, "templates"))],
84
+ destinationBucket: this.templateBucket,
85
+ destinationKeyPrefix: `${seqId}/`,
86
+ });
87
+ }
88
+ }
89
+ }
90
+ exports.StorageConstruct = StorageConstruct;
91
+ //# sourceMappingURL=storage.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"storage.js","sourceRoot":"","sources":["../../../lib/constructs/storage.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,7 @@
1
+ import type { SequenceDefinition } from "@mailshot/shared";
2
+ /**
3
+ * Scans sequences/ * /sequence.config.ts and loads each definition.
4
+ * Works because CDK is invoked via tsx which registers the TS loader.
5
+ */
6
+ export declare function loadSequenceConfigs(sequencesDir?: string): SequenceDefinition[];
7
+ //# sourceMappingURL=load-sequences.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"load-sequences.d.ts","sourceRoot":"","sources":["../../lib/load-sequences.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,kBAAkB,EAAE,MAAM,kBAAkB,CAAC;AAE3D;;;GAGG;AACH,wBAAgB,mBAAmB,CAAC,YAAY,CAAC,EAAE,MAAM,GAAG,kBAAkB,EAAE,CAuC/E"}
@@ -0,0 +1,76 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || (function () {
19
+ var ownKeys = function(o) {
20
+ ownKeys = Object.getOwnPropertyNames || function (o) {
21
+ var ar = [];
22
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
23
+ return ar;
24
+ };
25
+ return ownKeys(o);
26
+ };
27
+ return function (mod) {
28
+ if (mod && mod.__esModule) return mod;
29
+ var result = {};
30
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
31
+ __setModuleDefault(result, mod);
32
+ return result;
33
+ };
34
+ })();
35
+ Object.defineProperty(exports, "__esModule", { value: true });
36
+ exports.loadSequenceConfigs = loadSequenceConfigs;
37
+ const fs = __importStar(require("node:fs"));
38
+ const path = __importStar(require("node:path"));
39
+ /**
40
+ * Scans sequences/ * /sequence.config.ts and loads each definition.
41
+ * Works because CDK is invoked via tsx which registers the TS loader.
42
+ */
43
+ function loadSequenceConfigs(sequencesDir) {
44
+ const dir = sequencesDir ?? path.resolve(process.cwd(), "sequences");
45
+ if (!fs.existsSync(dir)) {
46
+ return [];
47
+ }
48
+ const definitions = [];
49
+ for (const entry of fs.readdirSync(dir, { withFileTypes: true })) {
50
+ if (!entry.isDirectory())
51
+ continue;
52
+ const configPath = path.join(dir, entry.name, "sequence.config.ts");
53
+ if (!fs.existsSync(configPath))
54
+ continue;
55
+ const mod = require(configPath);
56
+ const def = mod.default ?? mod.sequence;
57
+ if (!def) {
58
+ throw new Error(`${configPath}: must export a default or named 'sequence' export`);
59
+ }
60
+ if (!def.id) {
61
+ throw new Error(`${configPath}: missing 'id'`);
62
+ }
63
+ if (!def.trigger?.detailType) {
64
+ throw new Error(`${configPath}: missing 'trigger.detailType'`);
65
+ }
66
+ if (!Array.isArray(def.steps)) {
67
+ throw new Error(`${configPath}: missing 'steps' array`);
68
+ }
69
+ if (!def.timeoutMinutes) {
70
+ throw new Error(`${configPath}: missing 'timeoutMinutes'`);
71
+ }
72
+ definitions.push(def);
73
+ }
74
+ return definitions;
75
+ }
76
+ //# sourceMappingURL=load-sequences.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"load-sequences.js","sourceRoot":"","sources":["../../lib/load-sequences.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAQA,kDAuCC;AA/CD,4CAA8B;AAC9B,gDAAkC;AAGlC;;;GAGG;AACH,SAAgB,mBAAmB,CAAC,YAAqB;IACvD,MAAM,GAAG,GAAG,YAAY,IAAI,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,GAAG,EAAE,EAAE,WAAW,CAAC,CAAC;IAErE,IAAI,CAAC,EAAE,CAAC,UAAU,CAAC,GAAG,CAAC,EAAE,CAAC;QACxB,OAAO,EAAE,CAAC;IACZ,CAAC;IAED,MAAM,WAAW,GAAyB,EAAE,CAAC;IAE7C,KAAK,MAAM,KAAK,IAAI,EAAE,CAAC,WAAW,CAAC,GAAG,EAAE,EAAE,aAAa,EAAE,IAAI,EAAE,CAAC,EAAE,CAAC;QACjE,IAAI,CAAC,KAAK,CAAC,WAAW,EAAE;YAAE,SAAS;QAEnC,MAAM,UAAU,GAAG,IAAI,CAAC,IAAI,CAAC,GAAG,EAAE,KAAK,CAAC,IAAI,EAAE,oBAAoB,CAAC,CAAC;QACpE,IAAI,CAAC,EAAE,CAAC,UAAU,CAAC,UAAU,CAAC;YAAE,SAAS;QAEzC,MAAM,GAAG,GAAG,OAAO,CAAC,UAAU,CAAC,CAAC;QAChC,MAAM,GAAG,GAAuB,GAAG,CAAC,OAAO,IAAI,GAAG,CAAC,QAAQ,CAAC;QAE5D,IAAI,CAAC,GAAG,EAAE,CAAC;YACT,MAAM,IAAI,KAAK,CAAC,GAAG,UAAU,oDAAoD,CAAC,CAAC;QACrF,CAAC;QACD,IAAI,CAAC,GAAG,CAAC,EAAE,EAAE,CAAC;YACZ,MAAM,IAAI,KAAK,CAAC,GAAG,UAAU,gBAAgB,CAAC,CAAC;QACjD,CAAC;QACD,IAAI,CAAC,GAAG,CAAC,OAAO,EAAE,UAAU,EAAE,CAAC;YAC7B,MAAM,IAAI,KAAK,CAAC,GAAG,UAAU,gCAAgC,CAAC,CAAC;QACjE,CAAC;QAED,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC;YAC9B,MAAM,IAAI,KAAK,CAAC,GAAG,UAAU,yBAAyB,CAAC,CAAC;QAC1D,CAAC;QACD,IAAI,CAAC,GAAG,CAAC,cAAc,EAAE,CAAC;YACxB,MAAM,IAAI,KAAK,CAAC,GAAG,UAAU,4BAA4B,CAAC,CAAC;QAC7D,CAAC;QAED,WAAW,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;IACxB,CAAC;IAED,OAAO,WAAW,CAAC;AACrB,CAAC"}
@@ -0,0 +1,13 @@
1
+ import * as cdk from "aws-cdk-lib";
2
+ import type { Construct } from "constructs";
3
+ import type { MailshotConfig, SequenceDefinition } from "@mailshot/shared";
4
+ export interface MailshotStackProps extends cdk.StackProps {
5
+ config: MailshotConfig;
6
+ definitions: SequenceDefinition[];
7
+ handlersPath?: string;
8
+ templateBuildDir?: string;
9
+ }
10
+ export declare class MailshotStack extends cdk.Stack {
11
+ constructor(scope: Construct, id: string, props: MailshotStackProps);
12
+ }
13
+ //# sourceMappingURL=mailshot-stack.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"mailshot-stack.d.ts","sourceRoot":"","sources":["../../lib/mailshot-stack.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,GAAG,MAAM,aAAa,CAAC;AACnC,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,YAAY,CAAC;AAC5C,OAAO,KAAK,EAAE,cAAc,EAAE,kBAAkB,EAAE,MAAM,kBAAkB,CAAC;AAQ3E,MAAM,WAAW,kBAAmB,SAAQ,GAAG,CAAC,UAAU;IACxD,MAAM,EAAE,cAAc,CAAC;IACvB,WAAW,EAAE,kBAAkB,EAAE,CAAC;IAClC,YAAY,CAAC,EAAE,MAAM,CAAC;IACtB,gBAAgB,CAAC,EAAE,MAAM,CAAC;CAC3B;AAED,qBAAa,aAAc,SAAQ,GAAG,CAAC,KAAK;gBAC9B,KAAK,EAAE,SAAS,EAAE,EAAE,EAAE,MAAM,EAAE,KAAK,EAAE,kBAAkB;CAmIpE"}