@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,351 @@
1
+ /**
2
+ * Mock AWS services for unit testing
3
+ * Provides in-memory implementations of AWS services without external dependencies
4
+ */
5
+
6
+ export interface MockAWSConfig {
7
+ region?: string
8
+ credentials?: {
9
+ accessKeyId: string
10
+ secretAccessKey: string
11
+ }
12
+ }
13
+
14
+ /**
15
+ * Mock CloudFormation service
16
+ */
17
+ export class MockCloudFormation {
18
+ private stacks: Map<string, any> = new Map()
19
+ private stackEvents: Map<string, any[]> = new Map()
20
+
21
+ async createStack(params: any): Promise<any> {
22
+ const stackId = `arn:aws:cloudformation:${params.region || 'us-east-1'}:123456789012:stack/${params.StackName}/${Date.now()}`
23
+
24
+ this.stacks.set(params.StackName, {
25
+ StackId: stackId,
26
+ StackName: params.StackName,
27
+ StackStatus: 'CREATE_IN_PROGRESS',
28
+ CreationTime: new Date().toISOString(),
29
+ TemplateBody: params.TemplateBody,
30
+ Parameters: params.Parameters || [],
31
+ })
32
+
33
+ this.addEvent(params.StackName, {
34
+ EventId: Date.now().toString(),
35
+ StackName: params.StackName,
36
+ LogicalResourceId: params.StackName,
37
+ ResourceType: 'AWS::CloudFormation::Stack',
38
+ Timestamp: new Date().toISOString(),
39
+ ResourceStatus: 'CREATE_IN_PROGRESS',
40
+ })
41
+
42
+ // Simulate async stack creation
43
+ setTimeout(() => {
44
+ const stack = this.stacks.get(params.StackName)
45
+ if (stack) {
46
+ stack.StackStatus = 'CREATE_COMPLETE'
47
+ this.addEvent(params.StackName, {
48
+ EventId: (Date.now() + 1).toString(),
49
+ StackName: params.StackName,
50
+ LogicalResourceId: params.StackName,
51
+ ResourceType: 'AWS::CloudFormation::Stack',
52
+ Timestamp: new Date().toISOString(),
53
+ ResourceStatus: 'CREATE_COMPLETE',
54
+ })
55
+ }
56
+ }, 100)
57
+
58
+ return { StackId: stackId }
59
+ }
60
+
61
+ async updateStack(params: any): Promise<any> {
62
+ const stack = this.stacks.get(params.StackName)
63
+
64
+ if (!stack) {
65
+ throw new Error(`Stack ${params.StackName} does not exist`)
66
+ }
67
+
68
+ stack.StackStatus = 'UPDATE_IN_PROGRESS'
69
+ stack.TemplateBody = params.TemplateBody || stack.TemplateBody
70
+
71
+ this.addEvent(params.StackName, {
72
+ EventId: Date.now().toString(),
73
+ StackName: params.StackName,
74
+ LogicalResourceId: params.StackName,
75
+ ResourceType: 'AWS::CloudFormation::Stack',
76
+ Timestamp: new Date().toISOString(),
77
+ ResourceStatus: 'UPDATE_IN_PROGRESS',
78
+ })
79
+
80
+ setTimeout(() => {
81
+ stack.StackStatus = 'UPDATE_COMPLETE'
82
+ this.addEvent(params.StackName, {
83
+ EventId: (Date.now() + 1).toString(),
84
+ StackName: params.StackName,
85
+ LogicalResourceId: params.StackName,
86
+ ResourceType: 'AWS::CloudFormation::Stack',
87
+ Timestamp: new Date().toISOString(),
88
+ ResourceStatus: 'UPDATE_COMPLETE',
89
+ })
90
+ }, 100)
91
+
92
+ return { StackId: stack.StackId }
93
+ }
94
+
95
+ async deleteStack(params: any): Promise<void> {
96
+ const stack = this.stacks.get(params.StackName)
97
+
98
+ if (!stack) {
99
+ throw new Error(`Stack ${params.StackName} does not exist`)
100
+ }
101
+
102
+ stack.StackStatus = 'DELETE_IN_PROGRESS'
103
+
104
+ this.addEvent(params.StackName, {
105
+ EventId: Date.now().toString(),
106
+ StackName: params.StackName,
107
+ LogicalResourceId: params.StackName,
108
+ ResourceType: 'AWS::CloudFormation::Stack',
109
+ Timestamp: new Date().toISOString(),
110
+ ResourceStatus: 'DELETE_IN_PROGRESS',
111
+ })
112
+
113
+ setTimeout(() => {
114
+ this.stacks.delete(params.StackName)
115
+ this.addEvent(params.StackName, {
116
+ EventId: (Date.now() + 1).toString(),
117
+ StackName: params.StackName,
118
+ LogicalResourceId: params.StackName,
119
+ ResourceType: 'AWS::CloudFormation::Stack',
120
+ Timestamp: new Date().toISOString(),
121
+ ResourceStatus: 'DELETE_COMPLETE',
122
+ })
123
+ }, 100)
124
+ }
125
+
126
+ async describeStacks(params: any): Promise<any> {
127
+ if (params.StackName) {
128
+ const stack = this.stacks.get(params.StackName)
129
+
130
+ if (!stack) {
131
+ throw new Error(`Stack ${params.StackName} does not exist`)
132
+ }
133
+
134
+ return {
135
+ Stacks: [stack],
136
+ }
137
+ }
138
+
139
+ return {
140
+ Stacks: Array.from(this.stacks.values()),
141
+ }
142
+ }
143
+
144
+ async describeStackEvents(params: any): Promise<any> {
145
+ const events = this.stackEvents.get(params.StackName) || []
146
+
147
+ return {
148
+ StackEvents: events,
149
+ }
150
+ }
151
+
152
+ private addEvent(stackName: string, event: any): void {
153
+ if (!this.stackEvents.has(stackName)) {
154
+ this.stackEvents.set(stackName, [])
155
+ }
156
+
157
+ this.stackEvents.get(stackName)!.unshift(event)
158
+ }
159
+
160
+ /**
161
+ * Reset mock state (useful for testing)
162
+ */
163
+ reset(): void {
164
+ this.stacks.clear()
165
+ this.stackEvents.clear()
166
+ }
167
+ }
168
+
169
+ /**
170
+ * Mock S3 service
171
+ */
172
+ export class MockS3 {
173
+ private buckets: Map<string, Map<string, Buffer>> = new Map()
174
+
175
+ async createBucket(params: any): Promise<any> {
176
+ this.buckets.set(params.Bucket, new Map())
177
+
178
+ return {
179
+ Location: `/${params.Bucket}`,
180
+ }
181
+ }
182
+
183
+ async deleteBucket(params: any): Promise<void> {
184
+ this.buckets.delete(params.Bucket)
185
+ }
186
+
187
+ async putObject(params: any): Promise<any> {
188
+ let bucket = this.buckets.get(params.Bucket)
189
+
190
+ if (!bucket) {
191
+ bucket = new Map()
192
+ this.buckets.set(params.Bucket, bucket)
193
+ }
194
+
195
+ bucket.set(params.Key, Buffer.from(params.Body || ''))
196
+
197
+ return {
198
+ ETag: `"${Date.now()}"`,
199
+ }
200
+ }
201
+
202
+ async getObject(params: any): Promise<any> {
203
+ const bucket = this.buckets.get(params.Bucket)
204
+
205
+ if (!bucket) {
206
+ throw new Error(`Bucket ${params.Bucket} does not exist`)
207
+ }
208
+
209
+ const data = bucket.get(params.Key)
210
+
211
+ if (!data) {
212
+ throw new Error(`Object ${params.Key} does not exist`)
213
+ }
214
+
215
+ return {
216
+ Body: data,
217
+ ContentLength: data.length,
218
+ }
219
+ }
220
+
221
+ async deleteObject(params: any): Promise<void> {
222
+ const bucket = this.buckets.get(params.Bucket)
223
+
224
+ if (bucket) {
225
+ bucket.delete(params.Key)
226
+ }
227
+ }
228
+
229
+ async listObjects(params: any): Promise<any> {
230
+ const bucket = this.buckets.get(params.Bucket)
231
+
232
+ if (!bucket) {
233
+ throw new Error(`Bucket ${params.Bucket} does not exist`)
234
+ }
235
+
236
+ const objects = Array.from(bucket.keys()).map(key => ({
237
+ Key: key,
238
+ Size: bucket.get(key)!.length,
239
+ LastModified: new Date().toISOString(),
240
+ }))
241
+
242
+ return {
243
+ Contents: objects,
244
+ }
245
+ }
246
+
247
+ /**
248
+ * Reset mock state
249
+ */
250
+ reset(): void {
251
+ this.buckets.clear()
252
+ }
253
+ }
254
+
255
+ /**
256
+ * Mock DynamoDB service
257
+ */
258
+ export class MockDynamoDB {
259
+ private tables: Map<string, any> = new Map()
260
+ private data: Map<string, Map<string, any>> = new Map()
261
+
262
+ async createTable(params: any): Promise<any> {
263
+ this.tables.set(params.TableName, {
264
+ TableName: params.TableName,
265
+ KeySchema: params.KeySchema,
266
+ AttributeDefinitions: params.AttributeDefinitions,
267
+ TableStatus: 'ACTIVE',
268
+ CreationDateTime: new Date().toISOString(),
269
+ })
270
+
271
+ this.data.set(params.TableName, new Map())
272
+
273
+ return {
274
+ TableDescription: this.tables.get(params.TableName),
275
+ }
276
+ }
277
+
278
+ async deleteTable(params: any): Promise<void> {
279
+ this.tables.delete(params.TableName)
280
+ this.data.delete(params.TableName)
281
+ }
282
+
283
+ async putItem(params: any): Promise<void> {
284
+ let tableData = this.data.get(params.TableName)
285
+
286
+ if (!tableData) {
287
+ tableData = new Map()
288
+ this.data.set(params.TableName, tableData)
289
+ }
290
+
291
+ const key = JSON.stringify(params.Item)
292
+ tableData.set(key, params.Item)
293
+ }
294
+
295
+ async getItem(params: any): Promise<any> {
296
+ const tableData = this.data.get(params.TableName)
297
+
298
+ if (!tableData) {
299
+ return {}
300
+ }
301
+
302
+ const key = JSON.stringify(params.Key)
303
+ const item = tableData.get(key)
304
+
305
+ return item ? { Item: item } : {}
306
+ }
307
+
308
+ async deleteItem(params: any): Promise<void> {
309
+ const tableData = this.data.get(params.TableName)
310
+
311
+ if (tableData) {
312
+ const key = JSON.stringify(params.Key)
313
+ tableData.delete(key)
314
+ }
315
+ }
316
+
317
+ async scan(params: any): Promise<any> {
318
+ const tableData = this.data.get(params.TableName)
319
+
320
+ if (!tableData) {
321
+ return { Items: [] }
322
+ }
323
+
324
+ return {
325
+ Items: Array.from(tableData.values()),
326
+ }
327
+ }
328
+
329
+ /**
330
+ * Reset mock state
331
+ */
332
+ reset(): void {
333
+ this.tables.clear()
334
+ this.data.clear()
335
+ }
336
+ }
337
+
338
+ /**
339
+ * Create mock AWS services
340
+ */
341
+ export function createMockAWS(config?: MockAWSConfig): {
342
+ cloudformation: MockCloudFormation
343
+ s3: MockS3
344
+ dynamodb: MockDynamoDB
345
+ } {
346
+ return {
347
+ cloudformation: new MockCloudFormation(),
348
+ s3: new MockS3(),
349
+ dynamodb: new MockDynamoDB(),
350
+ }
351
+ }
@@ -0,0 +1,340 @@
1
+ import type {
2
+ IAMRole,
3
+ IAMManagedPolicy,
4
+ } from '@stacksjs/ts-cloud-aws-types'
5
+ import type { EnvironmentType } from '@stacksjs/ts-cloud-types'
6
+ import { generateLogicalId, generateResourceName } from '../resource-naming'
7
+
8
+ export interface BedrockRoleOptions {
9
+ slug: string
10
+ environment: EnvironmentType
11
+ name?: string
12
+ models?: string[]
13
+ allowStreaming?: boolean
14
+ }
15
+
16
+ export interface BedrockPolicyOptions {
17
+ slug: string
18
+ environment: EnvironmentType
19
+ name?: string
20
+ models?: string[]
21
+ allowInvoke?: boolean
22
+ allowStreaming?: boolean
23
+ allowAsync?: boolean
24
+ }
25
+
26
+ /**
27
+ * AI Module - Amazon Bedrock
28
+ * Provides clean API for setting up Bedrock permissions and roles
29
+ */
30
+ export class AI {
31
+ /**
32
+ * Create an IAM role for Bedrock access
33
+ */
34
+ static createBedrockRole(
35
+ servicePrincipal: string,
36
+ options: BedrockRoleOptions,
37
+ ): {
38
+ role: IAMRole
39
+ logicalId: string
40
+ } {
41
+ const {
42
+ slug,
43
+ environment,
44
+ name,
45
+ models = ['*'],
46
+ allowStreaming = true,
47
+ } = options
48
+
49
+ const resourceName = name || generateResourceName({
50
+ slug,
51
+ environment,
52
+ resourceType: 'bedrock-role',
53
+ })
54
+
55
+ const logicalId = generateLogicalId(resourceName)
56
+
57
+ // Build actions array
58
+ const actions = ['bedrock:InvokeModel']
59
+ if (allowStreaming) {
60
+ actions.push('bedrock:InvokeModelWithResponseStream')
61
+ }
62
+
63
+ // Build resource ARNs for models
64
+ const modelArns = models.map(model =>
65
+ model === '*'
66
+ ? 'arn:aws:bedrock:*::foundation-model/*'
67
+ : `arn:aws:bedrock:*::foundation-model/${model}`,
68
+ )
69
+
70
+ const role: IAMRole = {
71
+ Type: 'AWS::IAM::Role',
72
+ Properties: {
73
+ RoleName: resourceName,
74
+ AssumeRolePolicyDocument: {
75
+ Version: '2012-10-17',
76
+ Statement: [
77
+ {
78
+ Effect: 'Allow',
79
+ Principal: {
80
+ Service: servicePrincipal,
81
+ },
82
+ Action: 'sts:AssumeRole',
83
+ },
84
+ ],
85
+ },
86
+ Policies: [
87
+ {
88
+ PolicyName: `${resourceName}-policy`,
89
+ PolicyDocument: {
90
+ Version: '2012-10-17',
91
+ Statement: [
92
+ {
93
+ Effect: 'Allow',
94
+ Action: actions,
95
+ Resource: modelArns,
96
+ },
97
+ ],
98
+ },
99
+ },
100
+ ],
101
+ Tags: [
102
+ { Key: 'Name', Value: resourceName },
103
+ { Key: 'Environment', Value: environment },
104
+ ],
105
+ },
106
+ }
107
+
108
+ return { role, logicalId }
109
+ }
110
+
111
+ /**
112
+ * Create an IAM policy for Bedrock model invocation
113
+ */
114
+ static createBedrockPolicy(options: BedrockPolicyOptions): {
115
+ policy: IAMManagedPolicy
116
+ logicalId: string
117
+ } {
118
+ const {
119
+ slug,
120
+ environment,
121
+ name,
122
+ models = ['*'],
123
+ allowInvoke = true,
124
+ allowStreaming = true,
125
+ allowAsync = false,
126
+ } = options
127
+
128
+ const resourceName = name || generateResourceName({
129
+ slug,
130
+ environment,
131
+ resourceType: 'bedrock-policy',
132
+ })
133
+
134
+ const logicalId = generateLogicalId(resourceName)
135
+
136
+ // Build actions array
137
+ const actions: string[] = []
138
+ if (allowInvoke) {
139
+ actions.push('bedrock:InvokeModel')
140
+ }
141
+ if (allowStreaming) {
142
+ actions.push('bedrock:InvokeModelWithResponseStream')
143
+ }
144
+ if (allowAsync) {
145
+ actions.push('bedrock:InvokeModelAsync')
146
+ }
147
+
148
+ // Build resource ARNs for models
149
+ const modelArns = models.map(model =>
150
+ model === '*'
151
+ ? 'arn:aws:bedrock:*::foundation-model/*'
152
+ : `arn:aws:bedrock:*::foundation-model/${model}`,
153
+ )
154
+
155
+ const policy: IAMManagedPolicy = {
156
+ Type: 'AWS::IAM::ManagedPolicy',
157
+ Properties: {
158
+ ManagedPolicyName: resourceName,
159
+ PolicyDocument: {
160
+ Version: '2012-10-17',
161
+ Statement: [
162
+ {
163
+ Sid: 'BedrockModelInvocation',
164
+ Effect: 'Allow',
165
+ Action: actions,
166
+ Resource: modelArns,
167
+ },
168
+ ],
169
+ },
170
+ },
171
+ }
172
+
173
+ return { policy, logicalId }
174
+ }
175
+
176
+ /**
177
+ * Enable Bedrock for Lambda function
178
+ * Returns a role with Bedrock permissions
179
+ */
180
+ static enableBedrockForLambda(options: BedrockRoleOptions): {
181
+ role: IAMRole
182
+ logicalId: string
183
+ } {
184
+ return AI.createBedrockRole('lambda.amazonaws.com', options)
185
+ }
186
+
187
+ /**
188
+ * Enable Bedrock for ECS task
189
+ * Returns a role with Bedrock permissions
190
+ */
191
+ static enableBedrockForEcs(options: BedrockRoleOptions): {
192
+ role: IAMRole
193
+ logicalId: string
194
+ } {
195
+ return AI.createBedrockRole('ecs-tasks.amazonaws.com', options)
196
+ }
197
+
198
+ /**
199
+ * Enable Bedrock for EC2 instance
200
+ * Returns a role with Bedrock permissions
201
+ */
202
+ static enableBedrockForEc2(options: BedrockRoleOptions): {
203
+ role: IAMRole
204
+ logicalId: string
205
+ } {
206
+ return AI.createBedrockRole('ec2.amazonaws.com', options)
207
+ }
208
+
209
+ /**
210
+ * Add Bedrock permissions to an existing role
211
+ */
212
+ static addBedrockPermissions(
213
+ role: IAMRole,
214
+ models: string[] = ['*'],
215
+ allowStreaming = true,
216
+ ): IAMRole {
217
+ // Build actions array
218
+ const actions = ['bedrock:InvokeModel']
219
+ if (allowStreaming) {
220
+ actions.push('bedrock:InvokeModelWithResponseStream')
221
+ }
222
+
223
+ // Build resource ARNs for models
224
+ const modelArns = models.map(model =>
225
+ model === '*'
226
+ ? 'arn:aws:bedrock:*::foundation-model/*'
227
+ : `arn:aws:bedrock:*::foundation-model/${model}`,
228
+ )
229
+
230
+ if (!role.Properties.Policies) {
231
+ role.Properties.Policies = []
232
+ }
233
+
234
+ // Add Bedrock policy
235
+ role.Properties.Policies.push({
236
+ PolicyName: 'bedrock-permissions',
237
+ PolicyDocument: {
238
+ Version: '2012-10-17',
239
+ Statement: [
240
+ {
241
+ Effect: 'Allow',
242
+ Action: actions,
243
+ Resource: modelArns,
244
+ },
245
+ ],
246
+ },
247
+ })
248
+
249
+ return role
250
+ }
251
+
252
+ /**
253
+ * Common Bedrock model IDs
254
+ */
255
+ static readonly Models = {
256
+ // Anthropic Claude Models
257
+ Claude3_5_Sonnet: 'anthropic.claude-3-5-sonnet-20241022-v2:0',
258
+ Claude3_5_Haiku: 'anthropic.claude-3-5-haiku-20241022-v1:0',
259
+ Claude3_Opus: 'anthropic.claude-3-opus-20240229-v1:0',
260
+ Claude3_Sonnet: 'anthropic.claude-3-sonnet-20240229-v1:0',
261
+ Claude3_Haiku: 'anthropic.claude-3-haiku-20240307-v1:0',
262
+
263
+ // Amazon Titan Models
264
+ TitanTextG1Express: 'amazon.titan-text-express-v1',
265
+ TitanTextG1Lite: 'amazon.titan-text-lite-v1',
266
+ TitanEmbedG1Text: 'amazon.titan-embed-text-v1',
267
+ TitanImageG1: 'amazon.titan-image-generator-v1',
268
+
269
+ // AI21 Labs Models
270
+ JurassicUltra: 'ai21.j2-ultra-v1',
271
+ JurassicMid: 'ai21.j2-mid-v1',
272
+
273
+ // Cohere Models
274
+ CommandText: 'cohere.command-text-v14',
275
+ CommandLight: 'cohere.command-light-text-v14',
276
+ EmbedEnglish: 'cohere.embed-english-v3',
277
+ EmbedMultilingual: 'cohere.embed-multilingual-v3',
278
+
279
+ // Meta Llama Models
280
+ Llama3_2_1B: 'meta.llama3-2-1b-instruct-v1:0',
281
+ Llama3_2_3B: 'meta.llama3-2-3b-instruct-v1:0',
282
+ Llama3_2_11B: 'meta.llama3-2-11b-instruct-v1:0',
283
+ Llama3_2_90B: 'meta.llama3-2-90b-instruct-v1:0',
284
+ Llama3_1_8B: 'meta.llama3-1-8b-instruct-v1:0',
285
+ Llama3_1_70B: 'meta.llama3-1-70b-instruct-v1:0',
286
+ Llama3_1_405B: 'meta.llama3-1-405b-instruct-v1:0',
287
+
288
+ // Mistral AI Models
289
+ Mistral7B: 'mistral.mistral-7b-instruct-v0:2',
290
+ Mixtral8x7B: 'mistral.mixtral-8x7b-instruct-v0:1',
291
+ MistralLarge: 'mistral.mistral-large-2402-v1:0',
292
+
293
+ // Stability AI Models
294
+ StableDiffusionXL: 'stability.stable-diffusion-xl-v1',
295
+ } as const
296
+
297
+ /**
298
+ * Common model groups for easier permission management
299
+ */
300
+ static readonly ModelGroups = {
301
+ AllClaude: [
302
+ 'anthropic.claude-3-5-sonnet-20241022-v2:0',
303
+ 'anthropic.claude-3-5-haiku-20241022-v1:0',
304
+ 'anthropic.claude-3-opus-20240229-v1:0',
305
+ 'anthropic.claude-3-sonnet-20240229-v1:0',
306
+ 'anthropic.claude-3-haiku-20240307-v1:0',
307
+ ],
308
+ AllTitan: [
309
+ 'amazon.titan-text-express-v1',
310
+ 'amazon.titan-text-lite-v1',
311
+ 'amazon.titan-embed-text-v1',
312
+ 'amazon.titan-image-generator-v1',
313
+ ],
314
+ AllLlama: [
315
+ 'meta.llama3-2-1b-instruct-v1:0',
316
+ 'meta.llama3-2-3b-instruct-v1:0',
317
+ 'meta.llama3-2-11b-instruct-v1:0',
318
+ 'meta.llama3-2-90b-instruct-v1:0',
319
+ 'meta.llama3-1-8b-instruct-v1:0',
320
+ 'meta.llama3-1-70b-instruct-v1:0',
321
+ 'meta.llama3-1-405b-instruct-v1:0',
322
+ ],
323
+ TextModels: [
324
+ 'anthropic.claude-3-5-sonnet-20241022-v2:0',
325
+ 'anthropic.claude-3-5-haiku-20241022-v1:0',
326
+ 'amazon.titan-text-express-v1',
327
+ 'meta.llama3-2-11b-instruct-v1:0',
328
+ 'mistral.mistral-7b-instruct-v0:2',
329
+ ],
330
+ EmbeddingModels: [
331
+ 'amazon.titan-embed-text-v1',
332
+ 'cohere.embed-english-v3',
333
+ 'cohere.embed-multilingual-v3',
334
+ ],
335
+ ImageModels: [
336
+ 'amazon.titan-image-generator-v1',
337
+ 'stability.stable-diffusion-xl-v1',
338
+ ],
339
+ } as const
340
+ }