@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,593 @@
1
+ /**
2
+ * Route53 Advanced Routing
3
+ * Health-based, geolocation, weighted, failover, and latency-based routing
4
+ */
5
+
6
+ export interface RoutingPolicy {
7
+ id: string
8
+ name: string
9
+ type: 'simple' | 'weighted' | 'latency' | 'failover' | 'geolocation' | 'geoproximity' | 'multivalue'
10
+ recordSetId?: string
11
+ }
12
+
13
+ export interface WeightedRoutingPolicy extends RoutingPolicy {
14
+ type: 'weighted'
15
+ weight: number
16
+ setIdentifier: string
17
+ healthCheckId?: string
18
+ }
19
+
20
+ export interface LatencyRoutingPolicy extends RoutingPolicy {
21
+ type: 'latency'
22
+ region: string
23
+ setIdentifier: string
24
+ healthCheckId?: string
25
+ }
26
+
27
+ export interface FailoverRoutingPolicy extends RoutingPolicy {
28
+ type: 'failover'
29
+ failoverType: 'PRIMARY' | 'SECONDARY'
30
+ setIdentifier: string
31
+ healthCheckId: string
32
+ }
33
+
34
+ export interface GeolocationRoutingPolicy extends RoutingPolicy {
35
+ type: 'geolocation'
36
+ continent?: string
37
+ country?: string
38
+ subdivision?: string
39
+ setIdentifier: string
40
+ healthCheckId?: string
41
+ }
42
+
43
+ export interface GeoproximityRoutingPolicy extends RoutingPolicy {
44
+ type: 'geoproximity'
45
+ coordinates?: {
46
+ latitude: number
47
+ longitude: number
48
+ }
49
+ awsRegion?: string
50
+ bias?: number
51
+ setIdentifier: string
52
+ healthCheckId?: string
53
+ }
54
+
55
+ export interface HealthCheck {
56
+ id: string
57
+ name: string
58
+ type: 'http' | 'https' | 'tcp' | 'calculated' | 'cloudwatch_metric'
59
+ resourcePath?: string
60
+ fullyQualifiedDomainName?: string
61
+ ipAddress?: string
62
+ port?: number
63
+ requestInterval: number // seconds (10 or 30)
64
+ failureThreshold: number
65
+ healthCheckStatus: 'Healthy' | 'Unhealthy' | 'Unknown'
66
+ measureLatency?: boolean
67
+ enableSNI?: boolean
68
+ }
69
+
70
+ export interface CalculatedHealthCheck extends HealthCheck {
71
+ type: 'calculated'
72
+ childHealthChecks: string[]
73
+ healthThreshold: number
74
+ }
75
+
76
+ export interface TrafficPolicy {
77
+ id: string
78
+ name: string
79
+ version: number
80
+ document: TrafficPolicyDocument
81
+ }
82
+
83
+ export interface TrafficPolicyDocument {
84
+ recordType: 'A' | 'AAAA' | 'CNAME'
85
+ startRule: string
86
+ endpoints: Record<string, TrafficPolicyEndpoint>
87
+ rules: Record<string, TrafficPolicyRule>
88
+ }
89
+
90
+ export interface TrafficPolicyEndpoint {
91
+ type: 'value' | 'cloudfront' | 'elastic_load_balancer' | 's3_website'
92
+ value?: string
93
+ region?: string
94
+ }
95
+
96
+ export interface TrafficPolicyRule {
97
+ ruleType: 'failover' | 'geoproximity' | 'latency' | 'weighted' | 'multivalue'
98
+ primary?: string
99
+ secondary?: string
100
+ locations?: Array<{
101
+ endpointReference: string
102
+ region?: string
103
+ latitude?: number
104
+ longitude?: number
105
+ bias?: number
106
+ }>
107
+ items?: Array<{
108
+ endpointReference: string
109
+ weight?: number
110
+ }>
111
+ }
112
+
113
+ /**
114
+ * Route53 routing policy manager
115
+ */
116
+ export class Route53RoutingManager {
117
+ private policies: Map<string, RoutingPolicy> = new Map()
118
+ private healthChecks: Map<string, HealthCheck> = new Map()
119
+ private trafficPolicies: Map<string, TrafficPolicy> = new Map()
120
+ private policyCounter = 0
121
+ private healthCheckCounter = 0
122
+ private trafficPolicyCounter = 0
123
+
124
+ /**
125
+ * Create weighted routing policy
126
+ */
127
+ createWeightedPolicy(options: {
128
+ name: string
129
+ weight: number
130
+ setIdentifier: string
131
+ healthCheckId?: string
132
+ }): WeightedRoutingPolicy {
133
+ const id = `policy-${Date.now()}-${this.policyCounter++}`
134
+
135
+ const policy: WeightedRoutingPolicy = {
136
+ id,
137
+ type: 'weighted',
138
+ ...options,
139
+ }
140
+
141
+ this.policies.set(id, policy)
142
+
143
+ return policy
144
+ }
145
+
146
+ /**
147
+ * Create latency routing policy
148
+ */
149
+ createLatencyPolicy(options: {
150
+ name: string
151
+ region: string
152
+ setIdentifier: string
153
+ healthCheckId?: string
154
+ }): LatencyRoutingPolicy {
155
+ const id = `policy-${Date.now()}-${this.policyCounter++}`
156
+
157
+ const policy: LatencyRoutingPolicy = {
158
+ id,
159
+ type: 'latency',
160
+ ...options,
161
+ }
162
+
163
+ this.policies.set(id, policy)
164
+
165
+ return policy
166
+ }
167
+
168
+ /**
169
+ * Create failover routing policy
170
+ */
171
+ createFailoverPolicy(options: {
172
+ name: string
173
+ failoverType: 'PRIMARY' | 'SECONDARY'
174
+ setIdentifier: string
175
+ healthCheckId: string
176
+ }): FailoverRoutingPolicy {
177
+ const id = `policy-${Date.now()}-${this.policyCounter++}`
178
+
179
+ const policy: FailoverRoutingPolicy = {
180
+ id,
181
+ type: 'failover',
182
+ ...options,
183
+ }
184
+
185
+ this.policies.set(id, policy)
186
+
187
+ return policy
188
+ }
189
+
190
+ /**
191
+ * Create geolocation routing policy
192
+ */
193
+ createGeolocationPolicy(options: {
194
+ name: string
195
+ continent?: string
196
+ country?: string
197
+ subdivision?: string
198
+ setIdentifier: string
199
+ healthCheckId?: string
200
+ }): GeolocationRoutingPolicy {
201
+ const id = `policy-${Date.now()}-${this.policyCounter++}`
202
+
203
+ const policy: GeolocationRoutingPolicy = {
204
+ id,
205
+ type: 'geolocation',
206
+ ...options,
207
+ }
208
+
209
+ this.policies.set(id, policy)
210
+
211
+ return policy
212
+ }
213
+
214
+ /**
215
+ * Create geoproximity routing policy
216
+ */
217
+ createGeoproximityPolicy(options: {
218
+ name: string
219
+ coordinates?: {
220
+ latitude: number
221
+ longitude: number
222
+ }
223
+ awsRegion?: string
224
+ bias?: number
225
+ setIdentifier: string
226
+ healthCheckId?: string
227
+ }): GeoproximityRoutingPolicy {
228
+ const id = `policy-${Date.now()}-${this.policyCounter++}`
229
+
230
+ const policy: GeoproximityRoutingPolicy = {
231
+ id,
232
+ type: 'geoproximity',
233
+ ...options,
234
+ }
235
+
236
+ this.policies.set(id, policy)
237
+
238
+ return policy
239
+ }
240
+
241
+ /**
242
+ * Create HTTP health check
243
+ */
244
+ createHTTPHealthCheck(options: {
245
+ name: string
246
+ resourcePath: string
247
+ fullyQualifiedDomainName?: string
248
+ ipAddress?: string
249
+ port?: number
250
+ requestInterval?: number
251
+ failureThreshold?: number
252
+ enableSNI?: boolean
253
+ }): HealthCheck {
254
+ const id = `health-check-${Date.now()}-${this.healthCheckCounter++}`
255
+
256
+ const healthCheck: HealthCheck = {
257
+ id,
258
+ type: options.port === 443 ? 'https' : 'http',
259
+ requestInterval: options.requestInterval || 30,
260
+ failureThreshold: options.failureThreshold || 3,
261
+ healthCheckStatus: 'Unknown',
262
+ port: options.port || (options.port === 443 ? 443 : 80),
263
+ ...options,
264
+ }
265
+
266
+ this.healthChecks.set(id, healthCheck)
267
+
268
+ // Simulate health check execution
269
+ setTimeout(() => {
270
+ healthCheck.healthCheckStatus = Math.random() > 0.2 ? 'Healthy' : 'Unhealthy'
271
+ }, 100)
272
+
273
+ return healthCheck
274
+ }
275
+
276
+ /**
277
+ * Create TCP health check
278
+ */
279
+ createTCPHealthCheck(options: {
280
+ name: string
281
+ ipAddress: string
282
+ port: number
283
+ requestInterval?: number
284
+ failureThreshold?: number
285
+ }): HealthCheck {
286
+ const id = `health-check-${Date.now()}-${this.healthCheckCounter++}`
287
+
288
+ const healthCheck: HealthCheck = {
289
+ id,
290
+ type: 'tcp',
291
+ fullyQualifiedDomainName: options.ipAddress,
292
+ requestInterval: options.requestInterval || 30,
293
+ failureThreshold: options.failureThreshold || 3,
294
+ healthCheckStatus: 'Unknown',
295
+ ...options,
296
+ }
297
+
298
+ this.healthChecks.set(id, healthCheck)
299
+
300
+ setTimeout(() => {
301
+ healthCheck.healthCheckStatus = Math.random() > 0.1 ? 'Healthy' : 'Unhealthy'
302
+ }, 100)
303
+
304
+ return healthCheck
305
+ }
306
+
307
+ /**
308
+ * Create calculated health check
309
+ */
310
+ createCalculatedHealthCheck(options: {
311
+ name: string
312
+ childHealthChecks: string[]
313
+ healthThreshold: number
314
+ }): CalculatedHealthCheck {
315
+ const id = `health-check-${Date.now()}-${this.healthCheckCounter++}`
316
+
317
+ const healthCheck: CalculatedHealthCheck = {
318
+ id,
319
+ type: 'calculated',
320
+ requestInterval: 30,
321
+ failureThreshold: 0,
322
+ healthCheckStatus: 'Unknown',
323
+ ...options,
324
+ }
325
+
326
+ this.healthChecks.set(id, healthCheck)
327
+
328
+ // Calculate health based on child health checks
329
+ const updateCalculatedHealth = () => {
330
+ const healthyChildren = options.childHealthChecks.filter(childId => {
331
+ const child = this.healthChecks.get(childId)
332
+ return child?.healthCheckStatus === 'Healthy'
333
+ }).length
334
+
335
+ healthCheck.healthCheckStatus = healthyChildren >= options.healthThreshold ? 'Healthy' : 'Unhealthy'
336
+ }
337
+
338
+ setTimeout(updateCalculatedHealth, 150)
339
+
340
+ return healthCheck
341
+ }
342
+
343
+ /**
344
+ * Create traffic policy
345
+ */
346
+ createTrafficPolicy(policy: Omit<TrafficPolicy, 'id' | 'version'>): TrafficPolicy {
347
+ const id = `traffic-policy-${Date.now()}-${this.trafficPolicyCounter++}`
348
+
349
+ const trafficPolicy: TrafficPolicy = {
350
+ id,
351
+ version: 1,
352
+ ...policy,
353
+ }
354
+
355
+ this.trafficPolicies.set(id, trafficPolicy)
356
+
357
+ return trafficPolicy
358
+ }
359
+
360
+ /**
361
+ * Create failover traffic policy
362
+ */
363
+ createFailoverTrafficPolicy(options: {
364
+ name: string
365
+ primaryEndpoint: string
366
+ secondaryEndpoint: string
367
+ recordType?: 'A' | 'AAAA' | 'CNAME'
368
+ }): TrafficPolicy {
369
+ return this.createTrafficPolicy({
370
+ name: options.name,
371
+ document: {
372
+ recordType: options.recordType || 'A',
373
+ startRule: 'failover',
374
+ endpoints: {
375
+ primary: { type: 'value', value: options.primaryEndpoint },
376
+ secondary: { type: 'value', value: options.secondaryEndpoint },
377
+ },
378
+ rules: {
379
+ failover: {
380
+ ruleType: 'failover',
381
+ primary: 'primary',
382
+ secondary: 'secondary',
383
+ },
384
+ },
385
+ },
386
+ })
387
+ }
388
+
389
+ /**
390
+ * Create geoproximity traffic policy
391
+ */
392
+ createGeoproximityTrafficPolicy(options: {
393
+ name: string
394
+ locations: Array<{
395
+ endpoint: string
396
+ region?: string
397
+ latitude?: number
398
+ longitude?: number
399
+ bias?: number
400
+ }>
401
+ }): TrafficPolicy {
402
+ const endpoints: Record<string, TrafficPolicyEndpoint> = {}
403
+ const locations: Array<{
404
+ endpointReference: string
405
+ region?: string
406
+ latitude?: number
407
+ longitude?: number
408
+ bias?: number
409
+ }> = []
410
+
411
+ options.locations.forEach((loc, index) => {
412
+ const endpointRef = `endpoint${index}`
413
+ endpoints[endpointRef] = { type: 'value', value: loc.endpoint }
414
+ locations.push({
415
+ endpointReference: endpointRef,
416
+ region: loc.region,
417
+ latitude: loc.latitude,
418
+ longitude: loc.longitude,
419
+ bias: loc.bias,
420
+ })
421
+ })
422
+
423
+ return this.createTrafficPolicy({
424
+ name: options.name,
425
+ document: {
426
+ recordType: 'A',
427
+ startRule: 'geoproximity',
428
+ endpoints,
429
+ rules: {
430
+ geoproximity: {
431
+ ruleType: 'geoproximity',
432
+ locations,
433
+ },
434
+ },
435
+ },
436
+ })
437
+ }
438
+
439
+ /**
440
+ * Get routing policy
441
+ */
442
+ getPolicy(id: string): RoutingPolicy | undefined {
443
+ return this.policies.get(id)
444
+ }
445
+
446
+ /**
447
+ * List routing policies
448
+ */
449
+ listPolicies(): RoutingPolicy[] {
450
+ return Array.from(this.policies.values())
451
+ }
452
+
453
+ /**
454
+ * Get health check
455
+ */
456
+ getHealthCheck(id: string): HealthCheck | undefined {
457
+ return this.healthChecks.get(id)
458
+ }
459
+
460
+ /**
461
+ * List health checks
462
+ */
463
+ listHealthChecks(): HealthCheck[] {
464
+ return Array.from(this.healthChecks.values())
465
+ }
466
+
467
+ /**
468
+ * Get health check status
469
+ */
470
+ getHealthCheckStatus(id: string): 'Healthy' | 'Unhealthy' | 'Unknown' {
471
+ return this.healthChecks.get(id)?.healthCheckStatus || 'Unknown'
472
+ }
473
+
474
+ /**
475
+ * Generate CloudFormation for health check
476
+ */
477
+ generateHealthCheckCF(healthCheck: HealthCheck): any {
478
+ const config: any = {
479
+ Type: 'AWS::Route53::HealthCheck',
480
+ Properties: {
481
+ HealthCheckConfig: {
482
+ Type: healthCheck.type.toUpperCase(),
483
+ RequestInterval: healthCheck.requestInterval,
484
+ FailureThreshold: healthCheck.failureThreshold,
485
+ ...(healthCheck.measureLatency && { MeasureLatency: true }),
486
+ },
487
+ HealthCheckTags: [
488
+ {
489
+ Key: 'Name',
490
+ Value: healthCheck.name,
491
+ },
492
+ ],
493
+ },
494
+ }
495
+
496
+ if (healthCheck.type === 'calculated') {
497
+ const calculated = healthCheck as CalculatedHealthCheck
498
+ config.Properties.HealthCheckConfig.ChildHealthChecks = calculated.childHealthChecks
499
+ config.Properties.HealthCheckConfig.HealthThreshold = calculated.healthThreshold
500
+ } else {
501
+ if (healthCheck.ipAddress) {
502
+ config.Properties.HealthCheckConfig.IPAddress = healthCheck.ipAddress
503
+ }
504
+ if (healthCheck.fullyQualifiedDomainName) {
505
+ config.Properties.HealthCheckConfig.FullyQualifiedDomainName = healthCheck.fullyQualifiedDomainName
506
+ }
507
+ if (healthCheck.port) {
508
+ config.Properties.HealthCheckConfig.Port = healthCheck.port
509
+ }
510
+ if (healthCheck.resourcePath) {
511
+ config.Properties.HealthCheckConfig.ResourcePath = healthCheck.resourcePath
512
+ }
513
+ if (healthCheck.enableSNI) {
514
+ config.Properties.HealthCheckConfig.EnableSNI = true
515
+ }
516
+ }
517
+
518
+ return config
519
+ }
520
+
521
+ /**
522
+ * Generate CloudFormation for weighted record set
523
+ */
524
+ generateWeightedRecordSetCF(options: {
525
+ hostedZoneId: string
526
+ name: string
527
+ type: 'A' | 'AAAA' | 'CNAME'
528
+ ttl: number
529
+ resourceRecords: string[]
530
+ weight: number
531
+ setIdentifier: string
532
+ healthCheckId?: string
533
+ }): any {
534
+ return {
535
+ Type: 'AWS::Route53::RecordSet',
536
+ Properties: {
537
+ HostedZoneId: options.hostedZoneId,
538
+ Name: options.name,
539
+ Type: options.type,
540
+ TTL: options.ttl,
541
+ ResourceRecords: options.resourceRecords,
542
+ Weight: options.weight,
543
+ SetIdentifier: options.setIdentifier,
544
+ ...(options.healthCheckId && { HealthCheckId: options.healthCheckId }),
545
+ },
546
+ }
547
+ }
548
+
549
+ /**
550
+ * Generate CloudFormation for failover record set
551
+ */
552
+ generateFailoverRecordSetCF(options: {
553
+ hostedZoneId: string
554
+ name: string
555
+ type: 'A' | 'AAAA' | 'CNAME'
556
+ ttl: number
557
+ resourceRecords: string[]
558
+ failover: 'PRIMARY' | 'SECONDARY'
559
+ setIdentifier: string
560
+ healthCheckId?: string
561
+ }): any {
562
+ return {
563
+ Type: 'AWS::Route53::RecordSet',
564
+ Properties: {
565
+ HostedZoneId: options.hostedZoneId,
566
+ Name: options.name,
567
+ Type: options.type,
568
+ TTL: options.ttl,
569
+ ResourceRecords: options.resourceRecords,
570
+ Failover: options.failover,
571
+ SetIdentifier: options.setIdentifier,
572
+ ...(options.healthCheckId && { HealthCheckId: options.healthCheckId }),
573
+ },
574
+ }
575
+ }
576
+
577
+ /**
578
+ * Clear all data
579
+ */
580
+ clear(): void {
581
+ this.policies.clear()
582
+ this.healthChecks.clear()
583
+ this.trafficPolicies.clear()
584
+ this.policyCounter = 0
585
+ this.healthCheckCounter = 0
586
+ this.trafficPolicyCounter = 0
587
+ }
588
+ }
589
+
590
+ /**
591
+ * Global Route53 routing manager instance
592
+ */
593
+ export const route53RoutingManager: Route53RoutingManager = new Route53RoutingManager()