@stacksjs/ts-cloud-core 0.1.7 → 0.1.9

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 (225) hide show
  1. package/package.json +7 -6
  2. package/src/advanced-features.test.ts +465 -0
  3. package/src/aws/cloudformation.ts +421 -0
  4. package/src/aws/cloudfront.ts +158 -0
  5. package/src/aws/credentials.test.ts +132 -0
  6. package/src/aws/credentials.ts +545 -0
  7. package/src/aws/index.ts +87 -0
  8. package/src/aws/s3.test.ts +188 -0
  9. package/src/aws/s3.ts +1088 -0
  10. package/src/aws/signature.test.ts +670 -0
  11. package/src/aws/signature.ts +1155 -0
  12. package/src/backup/disaster-recovery.test.ts +726 -0
  13. package/src/backup/disaster-recovery.ts +500 -0
  14. package/src/backup/index.ts +34 -0
  15. package/src/backup/manager.test.ts +498 -0
  16. package/src/backup/manager.ts +432 -0
  17. package/src/cicd/circleci.ts +430 -0
  18. package/src/cicd/github-actions.ts +424 -0
  19. package/src/cicd/gitlab-ci.ts +255 -0
  20. package/src/cicd/index.ts +8 -0
  21. package/src/cli/history.ts +396 -0
  22. package/src/cli/index.ts +10 -0
  23. package/src/cli/progress.ts +458 -0
  24. package/src/cli/repl.ts +454 -0
  25. package/src/cli/suggestions.ts +327 -0
  26. package/src/cli/table.test.ts +319 -0
  27. package/src/cli/table.ts +332 -0
  28. package/src/cloudformation/builder.test.ts +327 -0
  29. package/src/cloudformation/builder.ts +378 -0
  30. package/src/cloudformation/builders/api-gateway.ts +449 -0
  31. package/src/cloudformation/builders/cache.ts +334 -0
  32. package/src/cloudformation/builders/cdn.ts +278 -0
  33. package/src/cloudformation/builders/compute.ts +485 -0
  34. package/src/cloudformation/builders/database.ts +392 -0
  35. package/src/cloudformation/builders/functions.ts +343 -0
  36. package/src/cloudformation/builders/messaging.ts +140 -0
  37. package/src/cloudformation/builders/monitoring.ts +300 -0
  38. package/src/cloudformation/builders/network.ts +264 -0
  39. package/src/cloudformation/builders/queue.ts +147 -0
  40. package/src/cloudformation/builders/security.ts +399 -0
  41. package/src/cloudformation/builders/storage.ts +285 -0
  42. package/src/cloudformation/index.ts +30 -0
  43. package/src/cloudformation/types.ts +173 -0
  44. package/src/compliance/aws-config.ts +543 -0
  45. package/src/compliance/cloudtrail.ts +376 -0
  46. package/src/compliance/compliance.test.ts +423 -0
  47. package/src/compliance/guardduty.ts +446 -0
  48. package/src/compliance/index.ts +66 -0
  49. package/src/compliance/security-hub.ts +456 -0
  50. package/src/containers/build-optimization.ts +416 -0
  51. package/src/containers/containers.test.ts +508 -0
  52. package/src/containers/image-scanning.ts +360 -0
  53. package/src/containers/index.ts +9 -0
  54. package/src/containers/registry.ts +293 -0
  55. package/src/containers/service-mesh.ts +520 -0
  56. package/src/database/database.test.ts +762 -0
  57. package/src/database/index.ts +9 -0
  58. package/src/database/migrations.ts +444 -0
  59. package/src/database/performance.ts +528 -0
  60. package/src/database/replicas.ts +534 -0
  61. package/src/database/users.ts +494 -0
  62. package/src/dependency-graph.ts +143 -0
  63. package/src/deployment/ab-testing.ts +582 -0
  64. package/src/deployment/blue-green.ts +452 -0
  65. package/src/deployment/canary.ts +500 -0
  66. package/src/deployment/deployment.test.ts +526 -0
  67. package/src/deployment/index.ts +61 -0
  68. package/src/deployment/progressive.ts +62 -0
  69. package/src/dns/dns.test.ts +641 -0
  70. package/src/dns/dnssec.ts +315 -0
  71. package/src/dns/index.ts +8 -0
  72. package/src/dns/resolver.ts +496 -0
  73. package/src/dns/routing.ts +593 -0
  74. package/src/email/advanced/analytics.ts +445 -0
  75. package/src/email/advanced/index.ts +11 -0
  76. package/src/email/advanced/rules.ts +465 -0
  77. package/src/email/advanced/scheduling.ts +352 -0
  78. package/src/email/advanced/search.ts +412 -0
  79. package/src/email/advanced/shared-mailboxes.ts +404 -0
  80. package/src/email/advanced/templates.ts +455 -0
  81. package/src/email/advanced/threading.ts +281 -0
  82. package/src/email/analytics.ts +467 -0
  83. package/src/email/bounce-handling.ts +425 -0
  84. package/src/email/email.test.ts +431 -0
  85. package/src/email/handlers/__tests__/inbound.test.ts +38 -0
  86. package/src/email/handlers/__tests__/outbound.test.ts +37 -0
  87. package/src/email/handlers/converter.ts +227 -0
  88. package/src/email/handlers/feedback.ts +228 -0
  89. package/src/email/handlers/inbound.ts +169 -0
  90. package/src/email/handlers/outbound.ts +178 -0
  91. package/src/email/index.ts +15 -0
  92. package/src/email/reputation.ts +303 -0
  93. package/src/email/templates.ts +352 -0
  94. package/src/errors/index.test.ts +434 -0
  95. package/src/errors/index.ts +416 -0
  96. package/src/health-checks/index.ts +40 -0
  97. package/src/index.ts +360 -0
  98. package/src/intrinsic-functions.ts +118 -0
  99. package/src/lambda/concurrency.ts +330 -0
  100. package/src/lambda/destinations.ts +345 -0
  101. package/src/lambda/dlq.ts +425 -0
  102. package/src/lambda/index.ts +11 -0
  103. package/src/lambda/lambda.test.ts +840 -0
  104. package/src/lambda/layers.ts +263 -0
  105. package/src/lambda/versions.ts +376 -0
  106. package/src/lambda/vpc.ts +399 -0
  107. package/src/local/config.ts +114 -0
  108. package/src/local/index.ts +6 -0
  109. package/src/local/mock-aws.ts +351 -0
  110. package/src/modules/ai.ts +340 -0
  111. package/src/modules/api.ts +478 -0
  112. package/src/modules/auth.ts +805 -0
  113. package/src/modules/cache.ts +417 -0
  114. package/src/modules/cdn.ts +1062 -0
  115. package/src/modules/communication.ts +1094 -0
  116. package/src/modules/compute.ts +3348 -0
  117. package/src/modules/database.ts +554 -0
  118. package/src/modules/deployment.ts +1079 -0
  119. package/src/modules/dns.ts +337 -0
  120. package/src/modules/email.ts +1538 -0
  121. package/src/modules/filesystem.ts +515 -0
  122. package/src/modules/index.ts +32 -0
  123. package/src/modules/messaging.ts +486 -0
  124. package/src/modules/monitoring.ts +2086 -0
  125. package/src/modules/network.ts +664 -0
  126. package/src/modules/parameter-store.ts +325 -0
  127. package/src/modules/permissions.ts +1081 -0
  128. package/src/modules/phone.ts +494 -0
  129. package/src/modules/queue.ts +1260 -0
  130. package/src/modules/redirects.ts +464 -0
  131. package/src/modules/registry.ts +699 -0
  132. package/src/modules/search.ts +401 -0
  133. package/src/modules/secrets.ts +416 -0
  134. package/src/modules/security.ts +731 -0
  135. package/src/modules/sms.ts +389 -0
  136. package/src/modules/storage.ts +1120 -0
  137. package/src/modules/workflow.ts +680 -0
  138. package/src/multi-account/config.ts +521 -0
  139. package/src/multi-account/index.ts +7 -0
  140. package/src/multi-account/manager.ts +427 -0
  141. package/src/multi-region/cross-region.ts +410 -0
  142. package/src/multi-region/index.ts +8 -0
  143. package/src/multi-region/manager.ts +483 -0
  144. package/src/multi-region/regions.ts +435 -0
  145. package/src/network-security/index.ts +48 -0
  146. package/src/observability/index.ts +9 -0
  147. package/src/observability/logs.ts +522 -0
  148. package/src/observability/metrics.ts +460 -0
  149. package/src/observability/observability.test.ts +782 -0
  150. package/src/observability/synthetics.ts +568 -0
  151. package/src/observability/xray.ts +358 -0
  152. package/src/phone/advanced/analytics.ts +349 -0
  153. package/src/phone/advanced/callbacks.ts +428 -0
  154. package/src/phone/advanced/index.ts +8 -0
  155. package/src/phone/advanced/ivr-builder.ts +504 -0
  156. package/src/phone/advanced/recording.ts +310 -0
  157. package/src/phone/handlers/__tests__/incoming-call.test.ts +40 -0
  158. package/src/phone/handlers/incoming-call.ts +117 -0
  159. package/src/phone/handlers/missed-call.ts +116 -0
  160. package/src/phone/handlers/voicemail.ts +179 -0
  161. package/src/phone/index.ts +9 -0
  162. package/src/presets/api-backend.ts +134 -0
  163. package/src/presets/data-pipeline.ts +204 -0
  164. package/src/presets/extend.test.ts +295 -0
  165. package/src/presets/extend.ts +297 -0
  166. package/src/presets/fullstack-app.ts +144 -0
  167. package/src/presets/index.ts +27 -0
  168. package/src/presets/jamstack.ts +135 -0
  169. package/src/presets/microservices.ts +167 -0
  170. package/src/presets/ml-api.ts +208 -0
  171. package/src/presets/nodejs-server.ts +104 -0
  172. package/src/presets/nodejs-serverless.ts +114 -0
  173. package/src/presets/realtime-app.ts +184 -0
  174. package/src/presets/static-site.ts +64 -0
  175. package/src/presets/traditional-web-app.ts +339 -0
  176. package/src/presets/wordpress.ts +138 -0
  177. package/src/preview/github.test.ts +249 -0
  178. package/src/preview/github.ts +297 -0
  179. package/src/preview/index.ts +37 -0
  180. package/src/preview/manager.test.ts +440 -0
  181. package/src/preview/manager.ts +326 -0
  182. package/src/preview/notifications.test.ts +582 -0
  183. package/src/preview/notifications.ts +341 -0
  184. package/src/queue/batch-processing.ts +402 -0
  185. package/src/queue/dlq-monitoring.ts +402 -0
  186. package/src/queue/fifo.ts +342 -0
  187. package/src/queue/index.ts +9 -0
  188. package/src/queue/management.ts +428 -0
  189. package/src/queue/queue.test.ts +429 -0
  190. package/src/resource-mgmt/index.ts +39 -0
  191. package/src/resource-naming.ts +62 -0
  192. package/src/s3/index.ts +523 -0
  193. package/src/schema/cloud-config.schema.json +554 -0
  194. package/src/schema/index.ts +68 -0
  195. package/src/security/certificate-manager.ts +492 -0
  196. package/src/security/index.ts +9 -0
  197. package/src/security/scanning.ts +545 -0
  198. package/src/security/secrets-manager.ts +476 -0
  199. package/src/security/secrets-rotation.ts +456 -0
  200. package/src/security/security.test.ts +738 -0
  201. package/src/sms/advanced/ab-testing.ts +389 -0
  202. package/src/sms/advanced/analytics.ts +336 -0
  203. package/src/sms/advanced/campaigns.ts +523 -0
  204. package/src/sms/advanced/chatbot.ts +224 -0
  205. package/src/sms/advanced/index.ts +10 -0
  206. package/src/sms/advanced/link-tracking.ts +248 -0
  207. package/src/sms/advanced/mms.ts +308 -0
  208. package/src/sms/handlers/__tests__/send.test.ts +40 -0
  209. package/src/sms/handlers/delivery-status.ts +133 -0
  210. package/src/sms/handlers/receive.ts +162 -0
  211. package/src/sms/handlers/send.ts +174 -0
  212. package/src/sms/index.ts +9 -0
  213. package/src/stack-diff.ts +389 -0
  214. package/src/static-site/index.ts +85 -0
  215. package/src/template-builder.ts +110 -0
  216. package/src/template-validator.ts +574 -0
  217. package/src/utils/cache.ts +291 -0
  218. package/src/utils/diff.ts +269 -0
  219. package/src/utils/hash.ts +227 -0
  220. package/src/utils/index.ts +8 -0
  221. package/src/utils/parallel.ts +294 -0
  222. package/src/validators/credentials.test.ts +274 -0
  223. package/src/validators/credentials.ts +233 -0
  224. package/src/validators/quotas.test.ts +434 -0
  225. package/src/validators/quotas.ts +217 -0
@@ -0,0 +1,417 @@
1
+ import type {
2
+ ElastiCacheCluster,
3
+ ElastiCacheParameterGroup,
4
+ ElastiCacheReplicationGroup,
5
+ ElastiCacheSubnetGroup,
6
+ } from '@stacksjs/ts-cloud-aws-types'
7
+ import type { EnvironmentType } from '@stacksjs/ts-cloud-types'
8
+ import { Fn } from '../intrinsic-functions'
9
+ import { generateLogicalId, generateResourceName } from '../resource-naming'
10
+
11
+ export interface RedisOptions {
12
+ slug: string
13
+ environment: EnvironmentType
14
+ nodeType?: string
15
+ engineVersion?: string
16
+ port?: number
17
+ subnetIds?: string[]
18
+ securityGroupIds?: string[]
19
+ numCacheClusters?: number
20
+ automaticFailover?: boolean
21
+ multiAz?: boolean
22
+ clusterMode?: boolean
23
+ numNodeGroups?: number
24
+ replicasPerNodeGroup?: number
25
+ atRestEncryption?: boolean
26
+ transitEncryption?: boolean
27
+ authToken?: string
28
+ kmsKeyId?: string
29
+ snapshotRetentionDays?: number
30
+ snapshotWindow?: string
31
+ maintenanceWindow?: string
32
+ }
33
+
34
+ export interface MemcachedOptions {
35
+ slug: string
36
+ environment: EnvironmentType
37
+ nodeType?: string
38
+ engineVersion?: string
39
+ port?: number
40
+ numCacheNodes?: number
41
+ subnetIds?: string[]
42
+ securityGroupIds?: string[]
43
+ azMode?: 'single-az' | 'cross-az'
44
+ preferredAzs?: string[]
45
+ maintenanceWindow?: string
46
+ }
47
+
48
+ /**
49
+ * Cache Module - ElastiCache (Redis + Memcached)
50
+ * Provides clean API for creating Redis and Memcached clusters
51
+ */
52
+ export class Cache {
53
+ /**
54
+ * Create a Redis cluster
55
+ */
56
+ static createRedis(options: RedisOptions): {
57
+ replicationGroup: ElastiCacheReplicationGroup
58
+ subnetGroup?: ElastiCacheSubnetGroup
59
+ logicalId: string
60
+ subnetGroupId?: string
61
+ } {
62
+ const {
63
+ slug,
64
+ environment,
65
+ nodeType = 'cache.t3.micro',
66
+ engineVersion = '7.1',
67
+ port = 6379,
68
+ subnetIds,
69
+ securityGroupIds,
70
+ numCacheClusters = 2,
71
+ automaticFailover = true,
72
+ multiAz = true,
73
+ clusterMode = false,
74
+ numNodeGroups = 1,
75
+ replicasPerNodeGroup = 1,
76
+ atRestEncryption = true,
77
+ transitEncryption = true,
78
+ authToken,
79
+ kmsKeyId,
80
+ snapshotRetentionDays = 7,
81
+ snapshotWindow,
82
+ maintenanceWindow,
83
+ } = options
84
+
85
+ const resourceName = generateResourceName({
86
+ slug,
87
+ environment,
88
+ resourceType: 'redis',
89
+ })
90
+
91
+ const logicalId = generateLogicalId(resourceName)
92
+
93
+ // Create subnet group if subnets provided
94
+ let subnetGroup: ElastiCacheSubnetGroup | undefined
95
+ let subnetGroupId: string | undefined
96
+
97
+ if (subnetIds && subnetIds.length > 0) {
98
+ const subnetGroupName = generateResourceName({
99
+ slug,
100
+ environment,
101
+ resourceType: 'cache-subnet-group',
102
+ })
103
+
104
+ subnetGroupId = generateLogicalId(subnetGroupName)
105
+
106
+ subnetGroup = {
107
+ Type: 'AWS::ElastiCache::SubnetGroup',
108
+ Properties: {
109
+ CacheSubnetGroupName: subnetGroupName,
110
+ Description: `Subnet group for ${resourceName}`,
111
+ SubnetIds: subnetIds,
112
+ Tags: [
113
+ { Key: 'Name', Value: subnetGroupName },
114
+ { Key: 'Environment', Value: environment },
115
+ ],
116
+ },
117
+ }
118
+ }
119
+
120
+ const replicationGroup: ElastiCacheReplicationGroup = {
121
+ Type: 'AWS::ElastiCache::ReplicationGroup',
122
+ Properties: {
123
+ ReplicationGroupId: resourceName,
124
+ ReplicationGroupDescription: `Redis cluster for ${slug} ${environment}`,
125
+ Engine: 'redis',
126
+ EngineVersion: engineVersion,
127
+ CacheNodeType: nodeType,
128
+ Port: port,
129
+ AutomaticFailoverEnabled: automaticFailover,
130
+ MultiAZEnabled: multiAz,
131
+ AtRestEncryptionEnabled: atRestEncryption,
132
+ TransitEncryptionEnabled: transitEncryption,
133
+ SnapshotRetentionLimit: snapshotRetentionDays,
134
+ AutoMinorVersionUpgrade: true,
135
+ Tags: [
136
+ { Key: 'Name', Value: resourceName },
137
+ { Key: 'Environment', Value: environment },
138
+ ],
139
+ },
140
+ }
141
+
142
+ // Configure cluster mode or replication mode
143
+ if (clusterMode) {
144
+ replicationGroup.Properties.NumNodeGroups = numNodeGroups
145
+ replicationGroup.Properties.ReplicasPerNodeGroup = replicasPerNodeGroup
146
+ }
147
+ else {
148
+ replicationGroup.Properties.NumCacheClusters = numCacheClusters
149
+ }
150
+
151
+ if (authToken) {
152
+ replicationGroup.Properties.AuthToken = authToken
153
+ }
154
+
155
+ if (kmsKeyId) {
156
+ replicationGroup.Properties.KmsKeyId = kmsKeyId
157
+ }
158
+
159
+ if (subnetGroupId) {
160
+ replicationGroup.Properties.CacheSubnetGroupName = Fn.Ref(subnetGroupId) as unknown as string
161
+ }
162
+
163
+ if (securityGroupIds && securityGroupIds.length > 0) {
164
+ replicationGroup.Properties.SecurityGroupIds = securityGroupIds
165
+ }
166
+
167
+ if (snapshotWindow) {
168
+ replicationGroup.Properties.SnapshotWindow = snapshotWindow
169
+ }
170
+
171
+ if (maintenanceWindow) {
172
+ replicationGroup.Properties.PreferredMaintenanceWindow = maintenanceWindow
173
+ }
174
+
175
+ return {
176
+ replicationGroup,
177
+ subnetGroup,
178
+ logicalId,
179
+ subnetGroupId,
180
+ }
181
+ }
182
+
183
+ /**
184
+ * Create a Memcached cluster
185
+ */
186
+ static createMemcached(options: MemcachedOptions): {
187
+ cluster: ElastiCacheCluster
188
+ subnetGroup?: ElastiCacheSubnetGroup
189
+ logicalId: string
190
+ subnetGroupId?: string
191
+ } {
192
+ const {
193
+ slug,
194
+ environment,
195
+ nodeType = 'cache.t3.micro',
196
+ engineVersion = '1.6.22',
197
+ port = 11211,
198
+ numCacheNodes = 2,
199
+ subnetIds,
200
+ securityGroupIds,
201
+ azMode = 'cross-az',
202
+ preferredAzs,
203
+ maintenanceWindow,
204
+ } = options
205
+
206
+ const resourceName = generateResourceName({
207
+ slug,
208
+ environment,
209
+ resourceType: 'memcached',
210
+ })
211
+
212
+ const logicalId = generateLogicalId(resourceName)
213
+
214
+ // Create subnet group if subnets provided
215
+ let subnetGroup: ElastiCacheSubnetGroup | undefined
216
+ let subnetGroupId: string | undefined
217
+
218
+ if (subnetIds && subnetIds.length > 0) {
219
+ const subnetGroupName = generateResourceName({
220
+ slug,
221
+ environment,
222
+ resourceType: 'cache-subnet-group',
223
+ })
224
+
225
+ subnetGroupId = generateLogicalId(subnetGroupName)
226
+
227
+ subnetGroup = {
228
+ Type: 'AWS::ElastiCache::SubnetGroup',
229
+ Properties: {
230
+ CacheSubnetGroupName: subnetGroupName,
231
+ Description: `Subnet group for ${resourceName}`,
232
+ SubnetIds: subnetIds,
233
+ Tags: [
234
+ { Key: 'Name', Value: subnetGroupName },
235
+ { Key: 'Environment', Value: environment },
236
+ ],
237
+ },
238
+ }
239
+ }
240
+
241
+ const cluster: ElastiCacheCluster = {
242
+ Type: 'AWS::ElastiCache::CacheCluster',
243
+ Properties: {
244
+ ClusterName: resourceName,
245
+ CacheNodeType: nodeType,
246
+ Engine: 'memcached',
247
+ EngineVersion: engineVersion,
248
+ NumCacheNodes: numCacheNodes,
249
+ Port: port,
250
+ AZMode: azMode,
251
+ AutoMinorVersionUpgrade: true,
252
+ Tags: [
253
+ { Key: 'Name', Value: resourceName },
254
+ { Key: 'Environment', Value: environment },
255
+ ],
256
+ },
257
+ }
258
+
259
+ if (subnetGroupId) {
260
+ cluster.Properties.CacheSubnetGroupName = Fn.Ref(subnetGroupId) as unknown as string
261
+ }
262
+
263
+ if (securityGroupIds && securityGroupIds.length > 0) {
264
+ cluster.Properties.VpcSecurityGroupIds = securityGroupIds
265
+ }
266
+
267
+ if (preferredAzs && preferredAzs.length > 0) {
268
+ cluster.Properties.PreferredAvailabilityZones = preferredAzs
269
+ }
270
+
271
+ if (maintenanceWindow) {
272
+ cluster.Properties.PreferredMaintenanceWindow = maintenanceWindow
273
+ }
274
+
275
+ return {
276
+ cluster,
277
+ subnetGroup,
278
+ logicalId,
279
+ subnetGroupId,
280
+ }
281
+ }
282
+
283
+ /**
284
+ * Enable cluster mode for Redis (returns new configuration)
285
+ */
286
+ static enableClusterMode(
287
+ replicationGroup: ElastiCacheReplicationGroup,
288
+ numNodeGroups: number = 3,
289
+ replicasPerNodeGroup: number = 2,
290
+ ): ElastiCacheReplicationGroup {
291
+ // Remove NumCacheClusters (used for non-cluster mode)
292
+ delete replicationGroup.Properties.NumCacheClusters
293
+
294
+ // Set cluster mode parameters
295
+ replicationGroup.Properties.NumNodeGroups = numNodeGroups
296
+ replicationGroup.Properties.ReplicasPerNodeGroup = replicasPerNodeGroup
297
+
298
+ return replicationGroup
299
+ }
300
+
301
+ /**
302
+ * Create a parameter group for Redis
303
+ */
304
+ static createRedisParameterGroup(
305
+ version: string,
306
+ options: {
307
+ slug: string
308
+ environment: EnvironmentType
309
+ parameters?: Record<string, string>
310
+ },
311
+ ): {
312
+ parameterGroup: ElastiCacheParameterGroup
313
+ logicalId: string
314
+ } {
315
+ const { slug, environment, parameters = {} } = options
316
+
317
+ const resourceName = generateResourceName({
318
+ slug,
319
+ environment,
320
+ resourceType: 'redis-params',
321
+ })
322
+
323
+ const logicalId = generateLogicalId(resourceName)
324
+
325
+ // Determine parameter group family
326
+ const family = `redis${version.split('.')[0]}.x`
327
+
328
+ const parameterGroup: ElastiCacheParameterGroup = {
329
+ Type: 'AWS::ElastiCache::ParameterGroup',
330
+ Properties: {
331
+ CacheParameterGroupFamily: family,
332
+ Description: `Parameter group for ${resourceName}`,
333
+ Properties: parameters,
334
+ Tags: [
335
+ { Key: 'Name', Value: resourceName },
336
+ { Key: 'Environment', Value: environment },
337
+ ],
338
+ },
339
+ }
340
+
341
+ return { parameterGroup, logicalId }
342
+ }
343
+
344
+ /**
345
+ * Create a parameter group for Memcached
346
+ */
347
+ static createMemcachedParameterGroup(
348
+ version: string,
349
+ options: {
350
+ slug: string
351
+ environment: EnvironmentType
352
+ parameters?: Record<string, string>
353
+ },
354
+ ): {
355
+ parameterGroup: ElastiCacheParameterGroup
356
+ logicalId: string
357
+ } {
358
+ const { slug, environment, parameters = {} } = options
359
+
360
+ const resourceName = generateResourceName({
361
+ slug,
362
+ environment,
363
+ resourceType: 'memcached-params',
364
+ })
365
+
366
+ const logicalId = generateLogicalId(resourceName)
367
+
368
+ // Determine parameter group family
369
+ const family = `memcached${version.split('.')[0]}.${version.split('.')[1]}`
370
+
371
+ const parameterGroup: ElastiCacheParameterGroup = {
372
+ Type: 'AWS::ElastiCache::ParameterGroup',
373
+ Properties: {
374
+ CacheParameterGroupFamily: family,
375
+ Description: `Parameter group for ${resourceName}`,
376
+ Properties: parameters,
377
+ Tags: [
378
+ { Key: 'Name', Value: resourceName },
379
+ { Key: 'Environment', Value: environment },
380
+ ],
381
+ },
382
+ }
383
+
384
+ return { parameterGroup, logicalId }
385
+ }
386
+
387
+ /**
388
+ * Common ElastiCache node types
389
+ */
390
+ static readonly NodeTypes = {
391
+ // T3 - Burstable performance
392
+ T3_Micro: 'cache.t3.micro',
393
+ T3_Small: 'cache.t3.small',
394
+ T3_Medium: 'cache.t3.medium',
395
+
396
+ // T4g - Arm-based burstable
397
+ T4g_Micro: 'cache.t4g.micro',
398
+ T4g_Small: 'cache.t4g.small',
399
+ T4g_Medium: 'cache.t4g.medium',
400
+
401
+ // M5 - General purpose
402
+ M5_Large: 'cache.m5.large',
403
+ M5_XLarge: 'cache.m5.xlarge',
404
+ M5_2XLarge: 'cache.m5.2xlarge',
405
+
406
+ // R5 - Memory optimized
407
+ R5_Large: 'cache.r5.large',
408
+ R5_XLarge: 'cache.r5.xlarge',
409
+ R5_2XLarge: 'cache.r5.2xlarge',
410
+ R5_4XLarge: 'cache.r5.4xlarge',
411
+
412
+ // R6g - Arm-based memory optimized
413
+ R6g_Large: 'cache.r6g.large',
414
+ R6g_XLarge: 'cache.r6g.xlarge',
415
+ R6g_2XLarge: 'cache.r6g.2xlarge',
416
+ } as const
417
+ }