@stacksjs/ts-cloud-core 0.1.1

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 (251) hide show
  1. package/LICENSE.md +21 -0
  2. package/README.md +321 -0
  3. package/package.json +31 -0
  4. package/src/advanced-features.test.ts +465 -0
  5. package/src/aws/cloudformation.ts +421 -0
  6. package/src/aws/cloudfront.ts +158 -0
  7. package/src/aws/credentials.test.ts +132 -0
  8. package/src/aws/credentials.ts +545 -0
  9. package/src/aws/index.ts +87 -0
  10. package/src/aws/s3.test.ts +188 -0
  11. package/src/aws/s3.ts +1088 -0
  12. package/src/aws/signature.test.ts +670 -0
  13. package/src/aws/signature.ts +1155 -0
  14. package/src/backup/disaster-recovery.test.ts +726 -0
  15. package/src/backup/disaster-recovery.ts +500 -0
  16. package/src/backup/index.ts +34 -0
  17. package/src/backup/manager.test.ts +498 -0
  18. package/src/backup/manager.ts +432 -0
  19. package/src/cicd/circleci.ts +430 -0
  20. package/src/cicd/github-actions.ts +424 -0
  21. package/src/cicd/gitlab-ci.ts +255 -0
  22. package/src/cicd/index.ts +8 -0
  23. package/src/cli/history.ts +396 -0
  24. package/src/cli/index.ts +10 -0
  25. package/src/cli/progress.ts +458 -0
  26. package/src/cli/repl.ts +454 -0
  27. package/src/cli/suggestions.ts +327 -0
  28. package/src/cli/table.test.ts +319 -0
  29. package/src/cli/table.ts +332 -0
  30. package/src/cloudformation/builder.test.ts +327 -0
  31. package/src/cloudformation/builder.ts +378 -0
  32. package/src/cloudformation/builders/api-gateway.ts +449 -0
  33. package/src/cloudformation/builders/cache.ts +334 -0
  34. package/src/cloudformation/builders/cdn.ts +278 -0
  35. package/src/cloudformation/builders/compute.ts +485 -0
  36. package/src/cloudformation/builders/database.ts +392 -0
  37. package/src/cloudformation/builders/functions.ts +343 -0
  38. package/src/cloudformation/builders/messaging.ts +140 -0
  39. package/src/cloudformation/builders/monitoring.ts +300 -0
  40. package/src/cloudformation/builders/network.ts +264 -0
  41. package/src/cloudformation/builders/queue.ts +147 -0
  42. package/src/cloudformation/builders/security.ts +399 -0
  43. package/src/cloudformation/builders/storage.ts +285 -0
  44. package/src/cloudformation/index.ts +30 -0
  45. package/src/cloudformation/types.ts +173 -0
  46. package/src/compliance/aws-config.ts +543 -0
  47. package/src/compliance/cloudtrail.ts +376 -0
  48. package/src/compliance/compliance.test.ts +423 -0
  49. package/src/compliance/guardduty.ts +446 -0
  50. package/src/compliance/index.ts +66 -0
  51. package/src/compliance/security-hub.ts +456 -0
  52. package/src/containers/build-optimization.ts +416 -0
  53. package/src/containers/containers.test.ts +508 -0
  54. package/src/containers/image-scanning.ts +360 -0
  55. package/src/containers/index.ts +9 -0
  56. package/src/containers/registry.ts +293 -0
  57. package/src/containers/service-mesh.ts +520 -0
  58. package/src/database/database.test.ts +762 -0
  59. package/src/database/index.ts +9 -0
  60. package/src/database/migrations.ts +444 -0
  61. package/src/database/performance.ts +528 -0
  62. package/src/database/replicas.ts +534 -0
  63. package/src/database/users.ts +494 -0
  64. package/src/dependency-graph.ts +143 -0
  65. package/src/deployment/ab-testing.ts +582 -0
  66. package/src/deployment/blue-green.ts +452 -0
  67. package/src/deployment/canary.ts +500 -0
  68. package/src/deployment/deployment.test.ts +526 -0
  69. package/src/deployment/index.ts +61 -0
  70. package/src/deployment/progressive.ts +62 -0
  71. package/src/dns/dns.test.ts +641 -0
  72. package/src/dns/dnssec.ts +315 -0
  73. package/src/dns/index.ts +8 -0
  74. package/src/dns/resolver.ts +496 -0
  75. package/src/dns/routing.ts +593 -0
  76. package/src/email/advanced/analytics.ts +445 -0
  77. package/src/email/advanced/index.ts +11 -0
  78. package/src/email/advanced/rules.ts +465 -0
  79. package/src/email/advanced/scheduling.ts +352 -0
  80. package/src/email/advanced/search.ts +412 -0
  81. package/src/email/advanced/shared-mailboxes.ts +404 -0
  82. package/src/email/advanced/templates.ts +455 -0
  83. package/src/email/advanced/threading.ts +281 -0
  84. package/src/email/analytics.ts +467 -0
  85. package/src/email/bounce-handling.ts +425 -0
  86. package/src/email/email.test.ts +431 -0
  87. package/src/email/handlers/__tests__/inbound.test.ts +38 -0
  88. package/src/email/handlers/__tests__/outbound.test.ts +37 -0
  89. package/src/email/handlers/converter.ts +227 -0
  90. package/src/email/handlers/feedback.ts +228 -0
  91. package/src/email/handlers/inbound.ts +169 -0
  92. package/src/email/handlers/outbound.ts +178 -0
  93. package/src/email/index.ts +15 -0
  94. package/src/email/reputation.ts +303 -0
  95. package/src/email/templates.ts +352 -0
  96. package/src/errors/index.test.ts +434 -0
  97. package/src/errors/index.ts +416 -0
  98. package/src/health-checks/index.ts +40 -0
  99. package/src/index.ts +360 -0
  100. package/src/intrinsic-functions.ts +118 -0
  101. package/src/lambda/concurrency.ts +330 -0
  102. package/src/lambda/destinations.ts +345 -0
  103. package/src/lambda/dlq.ts +425 -0
  104. package/src/lambda/index.ts +11 -0
  105. package/src/lambda/lambda.test.ts +840 -0
  106. package/src/lambda/layers.ts +263 -0
  107. package/src/lambda/versions.ts +376 -0
  108. package/src/lambda/vpc.ts +399 -0
  109. package/src/local/config.ts +114 -0
  110. package/src/local/index.ts +6 -0
  111. package/src/local/mock-aws.ts +351 -0
  112. package/src/modules/ai.ts +340 -0
  113. package/src/modules/api.ts +478 -0
  114. package/src/modules/auth.ts +805 -0
  115. package/src/modules/cache.ts +417 -0
  116. package/src/modules/cdn.ts +1062 -0
  117. package/src/modules/communication.ts +1094 -0
  118. package/src/modules/compute.ts +3348 -0
  119. package/src/modules/database.ts +554 -0
  120. package/src/modules/deployment.ts +1079 -0
  121. package/src/modules/dns.ts +337 -0
  122. package/src/modules/email.ts +1538 -0
  123. package/src/modules/filesystem.ts +515 -0
  124. package/src/modules/index.ts +32 -0
  125. package/src/modules/messaging.ts +486 -0
  126. package/src/modules/monitoring.ts +2086 -0
  127. package/src/modules/network.ts +664 -0
  128. package/src/modules/parameter-store.ts +325 -0
  129. package/src/modules/permissions.ts +1081 -0
  130. package/src/modules/phone.ts +494 -0
  131. package/src/modules/queue.ts +1260 -0
  132. package/src/modules/redirects.ts +464 -0
  133. package/src/modules/registry.ts +699 -0
  134. package/src/modules/search.ts +401 -0
  135. package/src/modules/secrets.ts +416 -0
  136. package/src/modules/security.ts +731 -0
  137. package/src/modules/sms.ts +389 -0
  138. package/src/modules/storage.ts +1120 -0
  139. package/src/modules/workflow.ts +680 -0
  140. package/src/multi-account/config.ts +521 -0
  141. package/src/multi-account/index.ts +7 -0
  142. package/src/multi-account/manager.ts +427 -0
  143. package/src/multi-region/cross-region.ts +410 -0
  144. package/src/multi-region/index.ts +8 -0
  145. package/src/multi-region/manager.ts +483 -0
  146. package/src/multi-region/regions.ts +435 -0
  147. package/src/network-security/index.ts +48 -0
  148. package/src/observability/index.ts +9 -0
  149. package/src/observability/logs.ts +522 -0
  150. package/src/observability/metrics.ts +460 -0
  151. package/src/observability/observability.test.ts +782 -0
  152. package/src/observability/synthetics.ts +568 -0
  153. package/src/observability/xray.ts +358 -0
  154. package/src/phone/advanced/analytics.ts +349 -0
  155. package/src/phone/advanced/callbacks.ts +428 -0
  156. package/src/phone/advanced/index.ts +8 -0
  157. package/src/phone/advanced/ivr-builder.ts +504 -0
  158. package/src/phone/advanced/recording.ts +310 -0
  159. package/src/phone/handlers/__tests__/incoming-call.test.ts +40 -0
  160. package/src/phone/handlers/incoming-call.ts +117 -0
  161. package/src/phone/handlers/missed-call.ts +116 -0
  162. package/src/phone/handlers/voicemail.ts +179 -0
  163. package/src/phone/index.ts +9 -0
  164. package/src/presets/api-backend.ts +134 -0
  165. package/src/presets/data-pipeline.ts +204 -0
  166. package/src/presets/extend.test.ts +295 -0
  167. package/src/presets/extend.ts +297 -0
  168. package/src/presets/fullstack-app.ts +144 -0
  169. package/src/presets/index.ts +27 -0
  170. package/src/presets/jamstack.ts +135 -0
  171. package/src/presets/microservices.ts +167 -0
  172. package/src/presets/ml-api.ts +208 -0
  173. package/src/presets/nodejs-server.ts +104 -0
  174. package/src/presets/nodejs-serverless.ts +114 -0
  175. package/src/presets/realtime-app.ts +184 -0
  176. package/src/presets/static-site.ts +64 -0
  177. package/src/presets/traditional-web-app.ts +339 -0
  178. package/src/presets/wordpress.ts +138 -0
  179. package/src/preview/github.test.ts +249 -0
  180. package/src/preview/github.ts +297 -0
  181. package/src/preview/index.ts +37 -0
  182. package/src/preview/manager.test.ts +440 -0
  183. package/src/preview/manager.ts +326 -0
  184. package/src/preview/notifications.test.ts +582 -0
  185. package/src/preview/notifications.ts +341 -0
  186. package/src/queue/batch-processing.ts +402 -0
  187. package/src/queue/dlq-monitoring.ts +402 -0
  188. package/src/queue/fifo.ts +342 -0
  189. package/src/queue/index.ts +9 -0
  190. package/src/queue/management.ts +428 -0
  191. package/src/queue/queue.test.ts +429 -0
  192. package/src/resource-mgmt/index.ts +39 -0
  193. package/src/resource-naming.ts +62 -0
  194. package/src/s3/index.ts +523 -0
  195. package/src/schema/cloud-config.schema.json +554 -0
  196. package/src/schema/index.ts +68 -0
  197. package/src/security/certificate-manager.ts +492 -0
  198. package/src/security/index.ts +9 -0
  199. package/src/security/scanning.ts +545 -0
  200. package/src/security/secrets-manager.ts +476 -0
  201. package/src/security/secrets-rotation.ts +456 -0
  202. package/src/security/security.test.ts +738 -0
  203. package/src/sms/advanced/ab-testing.ts +389 -0
  204. package/src/sms/advanced/analytics.ts +336 -0
  205. package/src/sms/advanced/campaigns.ts +523 -0
  206. package/src/sms/advanced/chatbot.ts +224 -0
  207. package/src/sms/advanced/index.ts +10 -0
  208. package/src/sms/advanced/link-tracking.ts +248 -0
  209. package/src/sms/advanced/mms.ts +308 -0
  210. package/src/sms/handlers/__tests__/send.test.ts +40 -0
  211. package/src/sms/handlers/delivery-status.ts +133 -0
  212. package/src/sms/handlers/receive.ts +162 -0
  213. package/src/sms/handlers/send.ts +174 -0
  214. package/src/sms/index.ts +9 -0
  215. package/src/stack-diff.ts +389 -0
  216. package/src/static-site/index.ts +85 -0
  217. package/src/template-builder.ts +110 -0
  218. package/src/template-validator.ts +574 -0
  219. package/src/utils/cache.ts +291 -0
  220. package/src/utils/diff.ts +269 -0
  221. package/src/utils/hash.ts +227 -0
  222. package/src/utils/index.ts +8 -0
  223. package/src/utils/parallel.ts +294 -0
  224. package/src/validators/credentials.test.ts +274 -0
  225. package/src/validators/credentials.ts +233 -0
  226. package/src/validators/quotas.test.ts +434 -0
  227. package/src/validators/quotas.ts +217 -0
  228. package/test/ai.test.ts +327 -0
  229. package/test/api.test.ts +511 -0
  230. package/test/auth.test.ts +632 -0
  231. package/test/cache.test.ts +406 -0
  232. package/test/cdn.test.ts +247 -0
  233. package/test/compute.test.ts +861 -0
  234. package/test/database.test.ts +523 -0
  235. package/test/deployment.test.ts +499 -0
  236. package/test/dns.test.ts +270 -0
  237. package/test/email.test.ts +439 -0
  238. package/test/filesystem.test.ts +382 -0
  239. package/test/integration.test.ts +350 -0
  240. package/test/messaging.test.ts +514 -0
  241. package/test/monitoring.test.ts +634 -0
  242. package/test/network.test.ts +425 -0
  243. package/test/permissions.test.ts +488 -0
  244. package/test/queue.test.ts +484 -0
  245. package/test/registry.test.ts +306 -0
  246. package/test/security.test.ts +462 -0
  247. package/test/storage.test.ts +463 -0
  248. package/test/template-validator.test.ts +559 -0
  249. package/test/workflow.test.ts +592 -0
  250. package/tsconfig.json +16 -0
  251. package/tsconfig.tsbuildinfo +1 -0
@@ -0,0 +1,680 @@
1
+ import type {
2
+ StepFunctionsStateMachine,
3
+ IAMRole,
4
+ } from '@stacksjs/ts-cloud-aws-types'
5
+ import type { EnvironmentType } from '@stacksjs/ts-cloud-types'
6
+ import { Fn } from '../intrinsic-functions'
7
+ import { generateLogicalId, generateResourceName } from '../resource-naming'
8
+
9
+ export interface StateMachineOptions {
10
+ slug: string
11
+ environment: EnvironmentType
12
+ stateMachineName?: string
13
+ type?: 'STANDARD' | 'EXPRESS'
14
+ definition: StateMachineDefinition
15
+ roleArn?: string
16
+ loggingConfiguration?: {
17
+ level: 'ALL' | 'ERROR' | 'FATAL' | 'OFF'
18
+ includeExecutionData?: boolean
19
+ destinations?: string[]
20
+ }
21
+ tracingConfiguration?: {
22
+ enabled: boolean
23
+ }
24
+ }
25
+
26
+ export interface StateMachineDefinition {
27
+ Comment?: string
28
+ StartAt: string
29
+ States: Record<string, State>
30
+ TimeoutSeconds?: number
31
+ Version?: string
32
+ }
33
+
34
+ export type State =
35
+ | TaskState
36
+ | PassState
37
+ | WaitState
38
+ | ChoiceState
39
+ | ParallelState
40
+ | MapState
41
+ | SucceedState
42
+ | FailState
43
+
44
+ export interface BaseState {
45
+ Type: 'Task' | 'Pass' | 'Wait' | 'Choice' | 'Parallel' | 'Map' | 'Succeed' | 'Fail'
46
+ Comment?: string
47
+ End?: boolean
48
+ Next?: string
49
+ }
50
+
51
+ export interface TaskState extends BaseState {
52
+ Type: 'Task'
53
+ Resource: string
54
+ Parameters?: Record<string, unknown>
55
+ ResultPath?: string | null
56
+ OutputPath?: string
57
+ InputPath?: string
58
+ TimeoutSeconds?: number
59
+ HeartbeatSeconds?: number
60
+ Retry?: RetryConfig[]
61
+ Catch?: CatchConfig[]
62
+ }
63
+
64
+ export interface PassState extends BaseState {
65
+ Type: 'Pass'
66
+ Result?: unknown
67
+ ResultPath?: string | null
68
+ Parameters?: Record<string, unknown>
69
+ }
70
+
71
+ export interface WaitState extends BaseState {
72
+ Type: 'Wait'
73
+ Seconds?: number
74
+ Timestamp?: string
75
+ SecondsPath?: string
76
+ TimestampPath?: string
77
+ }
78
+
79
+ export interface ChoiceState extends BaseState {
80
+ Type: 'Choice'
81
+ Choices: ChoiceRule[]
82
+ Default?: string
83
+ }
84
+
85
+ export interface ChoiceRule {
86
+ Variable: string
87
+ StringEquals?: string
88
+ StringLessThan?: string
89
+ StringGreaterThan?: string
90
+ NumericEquals?: number
91
+ NumericLessThan?: number
92
+ NumericGreaterThan?: number
93
+ BooleanEquals?: boolean
94
+ TimestampEquals?: string
95
+ TimestampLessThan?: string
96
+ TimestampGreaterThan?: string
97
+ IsPresent?: boolean
98
+ IsNull?: boolean
99
+ IsNumeric?: boolean
100
+ IsString?: boolean
101
+ IsBoolean?: boolean
102
+ IsTimestamp?: boolean
103
+ Next: string
104
+ And?: ChoiceRule[]
105
+ Or?: ChoiceRule[]
106
+ Not?: ChoiceRule
107
+ }
108
+
109
+ export interface ParallelState extends BaseState {
110
+ Type: 'Parallel'
111
+ Branches: StateMachineDefinition[]
112
+ ResultPath?: string | null
113
+ Retry?: RetryConfig[]
114
+ Catch?: CatchConfig[]
115
+ }
116
+
117
+ export interface MapState extends BaseState {
118
+ Type: 'Map'
119
+ ItemsPath?: string
120
+ Iterator: StateMachineDefinition
121
+ MaxConcurrency?: number
122
+ ResultPath?: string | null
123
+ Retry?: RetryConfig[]
124
+ Catch?: CatchConfig[]
125
+ }
126
+
127
+ export interface SucceedState extends BaseState {
128
+ Type: 'Succeed'
129
+ }
130
+
131
+ export interface FailState extends BaseState {
132
+ Type: 'Fail'
133
+ Error?: string
134
+ Cause?: string
135
+ }
136
+
137
+ export interface RetryConfig {
138
+ ErrorEquals: string[]
139
+ IntervalSeconds?: number
140
+ MaxAttempts?: number
141
+ BackoffRate?: number
142
+ }
143
+
144
+ export interface CatchConfig {
145
+ ErrorEquals: string[]
146
+ Next: string
147
+ ResultPath?: string
148
+ }
149
+
150
+ /**
151
+ * Workflow Module - Step Functions
152
+ * Provides clean API for orchestrating distributed applications and microservices
153
+ */
154
+ export class Workflow {
155
+ /**
156
+ * Create a Step Functions state machine
157
+ */
158
+ static createStateMachine(options: StateMachineOptions): {
159
+ stateMachine: StepFunctionsStateMachine
160
+ logicalId: string
161
+ role?: IAMRole
162
+ roleLogicalId?: string
163
+ } {
164
+ const {
165
+ slug,
166
+ environment,
167
+ stateMachineName,
168
+ type = 'STANDARD',
169
+ definition,
170
+ roleArn,
171
+ loggingConfiguration,
172
+ tracingConfiguration,
173
+ } = options
174
+
175
+ const resourceName = stateMachineName || generateResourceName({
176
+ slug,
177
+ environment,
178
+ resourceType: 'state-machine',
179
+ })
180
+
181
+ const logicalId = generateLogicalId(resourceName)
182
+
183
+ // Create role if not provided
184
+ let role: IAMRole | undefined
185
+ let roleLogicalId: string | undefined
186
+ let finalRoleArn: string
187
+
188
+ if (roleArn) {
189
+ finalRoleArn = roleArn
190
+ }
191
+ else {
192
+ const roleResourceName = generateResourceName({
193
+ slug,
194
+ environment,
195
+ resourceType: 'state-machine-role',
196
+ })
197
+ roleLogicalId = generateLogicalId(roleResourceName)
198
+
199
+ role = {
200
+ Type: 'AWS::IAM::Role',
201
+ Properties: {
202
+ RoleName: roleResourceName,
203
+ AssumeRolePolicyDocument: {
204
+ Version: '2012-10-17',
205
+ Statement: [
206
+ {
207
+ Effect: 'Allow',
208
+ Principal: {
209
+ Service: 'states.amazonaws.com',
210
+ },
211
+ Action: 'sts:AssumeRole',
212
+ },
213
+ ],
214
+ },
215
+ ManagedPolicyArns: [
216
+ 'arn:aws:iam::aws:policy/CloudWatchLogsFullAccess',
217
+ ],
218
+ Tags: [
219
+ { Key: 'Name', Value: roleResourceName },
220
+ { Key: 'Environment', Value: environment },
221
+ ],
222
+ },
223
+ }
224
+
225
+ finalRoleArn = Fn.GetAtt(roleLogicalId, 'Arn') as unknown as string
226
+ }
227
+
228
+ const stateMachine: StepFunctionsStateMachine = {
229
+ Type: 'AWS::StepFunctions::StateMachine',
230
+ Properties: {
231
+ StateMachineName: resourceName,
232
+ StateMachineType: type,
233
+ DefinitionString: JSON.stringify(definition),
234
+ RoleArn: finalRoleArn,
235
+ Tags: [
236
+ { Key: 'Name', Value: resourceName },
237
+ { Key: 'Environment', Value: environment },
238
+ ],
239
+ },
240
+ }
241
+
242
+ if (loggingConfiguration) {
243
+ stateMachine.Properties!.LoggingConfiguration = {
244
+ Level: loggingConfiguration.level,
245
+ IncludeExecutionData: loggingConfiguration.includeExecutionData,
246
+ Destinations: loggingConfiguration.destinations?.map(arn => ({ CloudWatchLogsLogGroup: { LogGroupArn: arn } })),
247
+ }
248
+ }
249
+
250
+ if (tracingConfiguration) {
251
+ stateMachine.Properties!.TracingConfiguration = {
252
+ Enabled: tracingConfiguration.enabled,
253
+ }
254
+ }
255
+
256
+ return { stateMachine, logicalId, role, roleLogicalId }
257
+ }
258
+
259
+ /**
260
+ * Create a Lambda task state
261
+ */
262
+ static createLambdaTask(
263
+ functionArn: string,
264
+ options?: {
265
+ parameters?: Record<string, unknown>
266
+ resultPath?: string | null
267
+ retry?: RetryConfig[]
268
+ catch?: CatchConfig[]
269
+ next?: string
270
+ end?: boolean
271
+ },
272
+ ): TaskState {
273
+ return {
274
+ Type: 'Task',
275
+ Resource: 'arn:aws:states:::lambda:invoke',
276
+ Parameters: {
277
+ FunctionName: functionArn,
278
+ Payload: options?.parameters || { 'Input.$': '$' },
279
+ },
280
+ ResultPath: options?.resultPath,
281
+ Retry: options?.retry,
282
+ Catch: options?.catch,
283
+ Next: options?.next,
284
+ End: options?.end,
285
+ }
286
+ }
287
+
288
+ /**
289
+ * Create a DynamoDB task state
290
+ */
291
+ static createDynamoDBTask(
292
+ action: 'GetItem' | 'PutItem' | 'UpdateItem' | 'DeleteItem',
293
+ tableName: string,
294
+ parameters: Record<string, unknown>,
295
+ options?: {
296
+ resultPath?: string | null
297
+ retry?: RetryConfig[]
298
+ catch?: CatchConfig[]
299
+ next?: string
300
+ end?: boolean
301
+ },
302
+ ): TaskState {
303
+ const resourceMap = {
304
+ GetItem: 'arn:aws:states:::dynamodb:getItem',
305
+ PutItem: 'arn:aws:states:::dynamodb:putItem',
306
+ UpdateItem: 'arn:aws:states:::dynamodb:updateItem',
307
+ DeleteItem: 'arn:aws:states:::dynamodb:deleteItem',
308
+ }
309
+
310
+ return {
311
+ Type: 'Task',
312
+ Resource: resourceMap[action],
313
+ Parameters: {
314
+ TableName: tableName,
315
+ ...parameters,
316
+ },
317
+ ResultPath: options?.resultPath,
318
+ Retry: options?.retry,
319
+ Catch: options?.catch,
320
+ Next: options?.next,
321
+ End: options?.end,
322
+ }
323
+ }
324
+
325
+ /**
326
+ * Create an SNS publish task state
327
+ */
328
+ static createSNSPublishTask(
329
+ topicArn: string,
330
+ message: Record<string, unknown>,
331
+ options?: {
332
+ resultPath?: string | null
333
+ retry?: RetryConfig[]
334
+ catch?: CatchConfig[]
335
+ next?: string
336
+ end?: boolean
337
+ },
338
+ ): TaskState {
339
+ return {
340
+ Type: 'Task',
341
+ Resource: 'arn:aws:states:::sns:publish',
342
+ Parameters: {
343
+ TopicArn: topicArn,
344
+ Message: message,
345
+ },
346
+ ResultPath: options?.resultPath,
347
+ Retry: options?.retry,
348
+ Catch: options?.catch,
349
+ Next: options?.next,
350
+ End: options?.end,
351
+ }
352
+ }
353
+
354
+ /**
355
+ * Create an SQS send message task state
356
+ */
357
+ static createSQSSendMessageTask(
358
+ queueUrl: string,
359
+ messageBody: Record<string, unknown>,
360
+ options?: {
361
+ resultPath?: string | null
362
+ retry?: RetryConfig[]
363
+ catch?: CatchConfig[]
364
+ next?: string
365
+ end?: boolean
366
+ },
367
+ ): TaskState {
368
+ return {
369
+ Type: 'Task',
370
+ Resource: 'arn:aws:states:::sqs:sendMessage',
371
+ Parameters: {
372
+ QueueUrl: queueUrl,
373
+ MessageBody: messageBody,
374
+ },
375
+ ResultPath: options?.resultPath,
376
+ Retry: options?.retry,
377
+ Catch: options?.catch,
378
+ Next: options?.next,
379
+ End: options?.end,
380
+ }
381
+ }
382
+
383
+ /**
384
+ * Create a Pass state
385
+ */
386
+ static createPassState(options?: {
387
+ result?: unknown
388
+ resultPath?: string | null
389
+ parameters?: Record<string, unknown>
390
+ next?: string
391
+ end?: boolean
392
+ }): PassState {
393
+ return {
394
+ Type: 'Pass',
395
+ Result: options?.result,
396
+ ResultPath: options?.resultPath,
397
+ Parameters: options?.parameters,
398
+ Next: options?.next,
399
+ End: options?.end,
400
+ }
401
+ }
402
+
403
+ /**
404
+ * Create a Wait state
405
+ */
406
+ static createWaitState(options: {
407
+ seconds?: number
408
+ timestamp?: string
409
+ secondsPath?: string
410
+ timestampPath?: string
411
+ next?: string
412
+ end?: boolean
413
+ }): WaitState {
414
+ return {
415
+ Type: 'Wait',
416
+ Seconds: options.seconds,
417
+ Timestamp: options.timestamp,
418
+ SecondsPath: options.secondsPath,
419
+ TimestampPath: options.timestampPath,
420
+ Next: options.next,
421
+ End: options.end,
422
+ }
423
+ }
424
+
425
+ /**
426
+ * Create a Choice state
427
+ */
428
+ static createChoiceState(
429
+ choices: ChoiceRule[],
430
+ defaultState?: string,
431
+ ): ChoiceState {
432
+ return {
433
+ Type: 'Choice',
434
+ Choices: choices,
435
+ Default: defaultState,
436
+ }
437
+ }
438
+
439
+ /**
440
+ * Create a Parallel state
441
+ */
442
+ static createParallelState(
443
+ branches: StateMachineDefinition[],
444
+ options?: {
445
+ resultPath?: string | null
446
+ retry?: RetryConfig[]
447
+ catch?: CatchConfig[]
448
+ next?: string
449
+ end?: boolean
450
+ },
451
+ ): ParallelState {
452
+ return {
453
+ Type: 'Parallel',
454
+ Branches: branches,
455
+ ResultPath: options?.resultPath,
456
+ Retry: options?.retry,
457
+ Catch: options?.catch,
458
+ Next: options?.next,
459
+ End: options?.end,
460
+ }
461
+ }
462
+
463
+ /**
464
+ * Create a Map state
465
+ */
466
+ static createMapState(
467
+ iterator: StateMachineDefinition,
468
+ options?: {
469
+ itemsPath?: string
470
+ maxConcurrency?: number
471
+ resultPath?: string | null
472
+ retry?: RetryConfig[]
473
+ catch?: CatchConfig[]
474
+ next?: string
475
+ end?: boolean
476
+ },
477
+ ): MapState {
478
+ return {
479
+ Type: 'Map',
480
+ ItemsPath: options?.itemsPath || '$.items',
481
+ Iterator: iterator,
482
+ MaxConcurrency: options?.maxConcurrency,
483
+ ResultPath: options?.resultPath,
484
+ Retry: options?.retry,
485
+ Catch: options?.catch,
486
+ Next: options?.next,
487
+ End: options?.end,
488
+ }
489
+ }
490
+
491
+ /**
492
+ * Create a Succeed state
493
+ */
494
+ static createSucceedState(): SucceedState {
495
+ return {
496
+ Type: 'Succeed',
497
+ }
498
+ }
499
+
500
+ /**
501
+ * Create a Fail state
502
+ */
503
+ static createFailState(error?: string, cause?: string): FailState {
504
+ return {
505
+ Type: 'Fail',
506
+ Error: error,
507
+ Cause: cause,
508
+ }
509
+ }
510
+
511
+ /**
512
+ * Common retry configurations
513
+ */
514
+ static readonly RetryPolicies = {
515
+ /**
516
+ * Standard retry with exponential backoff
517
+ */
518
+ standard: (): RetryConfig => ({
519
+ ErrorEquals: ['States.ALL'],
520
+ IntervalSeconds: 2,
521
+ MaxAttempts: 3,
522
+ BackoffRate: 2.0,
523
+ }),
524
+
525
+ /**
526
+ * Aggressive retry for transient errors
527
+ */
528
+ aggressive: (): RetryConfig => ({
529
+ ErrorEquals: ['States.TaskFailed', 'States.Timeout'],
530
+ IntervalSeconds: 1,
531
+ MaxAttempts: 5,
532
+ BackoffRate: 1.5,
533
+ }),
534
+
535
+ /**
536
+ * Custom retry configuration
537
+ */
538
+ custom: (
539
+ errorEquals: string[],
540
+ intervalSeconds: number,
541
+ maxAttempts: number,
542
+ backoffRate: number,
543
+ ): RetryConfig => ({
544
+ ErrorEquals: errorEquals,
545
+ IntervalSeconds: intervalSeconds,
546
+ MaxAttempts: maxAttempts,
547
+ BackoffRate: backoffRate,
548
+ }),
549
+ } as const
550
+
551
+ /**
552
+ * Common catch configurations
553
+ */
554
+ static readonly CatchPolicies = {
555
+ /**
556
+ * Catch all errors
557
+ */
558
+ all: (nextState: string, resultPath?: string): CatchConfig => ({
559
+ ErrorEquals: ['States.ALL'],
560
+ Next: nextState,
561
+ ResultPath: resultPath || '$.error',
562
+ }),
563
+
564
+ /**
565
+ * Catch specific errors
566
+ */
567
+ specific: (errors: string[], nextState: string, resultPath?: string): CatchConfig => ({
568
+ ErrorEquals: errors,
569
+ Next: nextState,
570
+ ResultPath: resultPath || '$.error',
571
+ }),
572
+ } as const
573
+
574
+ /**
575
+ * Common workflow patterns
576
+ */
577
+ static readonly Patterns = {
578
+ /**
579
+ * Simple sequential workflow
580
+ */
581
+ sequential: (
582
+ slug: string,
583
+ environment: EnvironmentType,
584
+ tasks: { name: string, state: State }[],
585
+ ): StateMachineDefinition => {
586
+ const states: Record<string, State> = {}
587
+
588
+ tasks.forEach((task, index) => {
589
+ const isLast = index === tasks.length - 1
590
+ states[task.name] = {
591
+ ...task.state,
592
+ Next: isLast ? undefined : tasks[index + 1].name,
593
+ End: isLast,
594
+ }
595
+ })
596
+
597
+ return {
598
+ Comment: 'Sequential workflow',
599
+ StartAt: tasks[0].name,
600
+ States: states,
601
+ }
602
+ },
603
+
604
+ /**
605
+ * Fan-out workflow (parallel execution)
606
+ */
607
+ fanout: (
608
+ slug: string,
609
+ environment: EnvironmentType,
610
+ branches: { name: string, definition: StateMachineDefinition }[],
611
+ ): StateMachineDefinition => {
612
+ return {
613
+ Comment: 'Fan-out workflow',
614
+ StartAt: 'Parallel',
615
+ States: {
616
+ Parallel: {
617
+ Type: 'Parallel',
618
+ Branches: branches.map(b => b.definition),
619
+ End: true,
620
+ },
621
+ },
622
+ }
623
+ },
624
+
625
+ /**
626
+ * Map workflow (process array of items)
627
+ */
628
+ map: (
629
+ slug: string,
630
+ environment: EnvironmentType,
631
+ itemProcessor: StateMachineDefinition,
632
+ maxConcurrency?: number,
633
+ ): StateMachineDefinition => {
634
+ return {
635
+ Comment: 'Map workflow',
636
+ StartAt: 'Map',
637
+ States: {
638
+ Map: {
639
+ Type: 'Map',
640
+ ItemsPath: '$.items',
641
+ Iterator: itemProcessor,
642
+ MaxConcurrency: maxConcurrency,
643
+ End: true,
644
+ },
645
+ },
646
+ }
647
+ },
648
+
649
+ /**
650
+ * Error handling workflow
651
+ */
652
+ withErrorHandling: (
653
+ slug: string,
654
+ environment: EnvironmentType,
655
+ mainTask: TaskState,
656
+ errorHandler: State,
657
+ ): StateMachineDefinition => {
658
+ return {
659
+ Comment: 'Workflow with error handling',
660
+ StartAt: 'Main',
661
+ States: {
662
+ Main: {
663
+ ...mainTask,
664
+ Catch: [
665
+ {
666
+ ErrorEquals: ['States.ALL'],
667
+ Next: 'ErrorHandler',
668
+ },
669
+ ],
670
+ Next: 'Success',
671
+ },
672
+ ErrorHandler: errorHandler,
673
+ Success: {
674
+ Type: 'Succeed',
675
+ },
676
+ },
677
+ }
678
+ },
679
+ } as const
680
+ }