@stacksjs/ts-cloud-core 0.1.3 → 0.1.6

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 (250) hide show
  1. package/README.md +98 -13
  2. package/package.json +12 -3
  3. package/src/advanced-features.test.ts +0 -465
  4. package/src/aws/cloudformation.ts +0 -421
  5. package/src/aws/cloudfront.ts +0 -158
  6. package/src/aws/credentials.test.ts +0 -132
  7. package/src/aws/credentials.ts +0 -545
  8. package/src/aws/index.ts +0 -87
  9. package/src/aws/s3.test.ts +0 -188
  10. package/src/aws/s3.ts +0 -1088
  11. package/src/aws/signature.test.ts +0 -670
  12. package/src/aws/signature.ts +0 -1155
  13. package/src/backup/disaster-recovery.test.ts +0 -726
  14. package/src/backup/disaster-recovery.ts +0 -500
  15. package/src/backup/index.ts +0 -34
  16. package/src/backup/manager.test.ts +0 -498
  17. package/src/backup/manager.ts +0 -432
  18. package/src/cicd/circleci.ts +0 -430
  19. package/src/cicd/github-actions.ts +0 -424
  20. package/src/cicd/gitlab-ci.ts +0 -255
  21. package/src/cicd/index.ts +0 -8
  22. package/src/cli/history.ts +0 -396
  23. package/src/cli/index.ts +0 -10
  24. package/src/cli/progress.ts +0 -458
  25. package/src/cli/repl.ts +0 -454
  26. package/src/cli/suggestions.ts +0 -327
  27. package/src/cli/table.test.ts +0 -319
  28. package/src/cli/table.ts +0 -332
  29. package/src/cloudformation/builder.test.ts +0 -327
  30. package/src/cloudformation/builder.ts +0 -378
  31. package/src/cloudformation/builders/api-gateway.ts +0 -449
  32. package/src/cloudformation/builders/cache.ts +0 -334
  33. package/src/cloudformation/builders/cdn.ts +0 -278
  34. package/src/cloudformation/builders/compute.ts +0 -485
  35. package/src/cloudformation/builders/database.ts +0 -392
  36. package/src/cloudformation/builders/functions.ts +0 -343
  37. package/src/cloudformation/builders/messaging.ts +0 -140
  38. package/src/cloudformation/builders/monitoring.ts +0 -300
  39. package/src/cloudformation/builders/network.ts +0 -264
  40. package/src/cloudformation/builders/queue.ts +0 -147
  41. package/src/cloudformation/builders/security.ts +0 -399
  42. package/src/cloudformation/builders/storage.ts +0 -285
  43. package/src/cloudformation/index.ts +0 -30
  44. package/src/cloudformation/types.ts +0 -173
  45. package/src/compliance/aws-config.ts +0 -543
  46. package/src/compliance/cloudtrail.ts +0 -376
  47. package/src/compliance/compliance.test.ts +0 -423
  48. package/src/compliance/guardduty.ts +0 -446
  49. package/src/compliance/index.ts +0 -66
  50. package/src/compliance/security-hub.ts +0 -456
  51. package/src/containers/build-optimization.ts +0 -416
  52. package/src/containers/containers.test.ts +0 -508
  53. package/src/containers/image-scanning.ts +0 -360
  54. package/src/containers/index.ts +0 -9
  55. package/src/containers/registry.ts +0 -293
  56. package/src/containers/service-mesh.ts +0 -520
  57. package/src/database/database.test.ts +0 -762
  58. package/src/database/index.ts +0 -9
  59. package/src/database/migrations.ts +0 -444
  60. package/src/database/performance.ts +0 -528
  61. package/src/database/replicas.ts +0 -534
  62. package/src/database/users.ts +0 -494
  63. package/src/dependency-graph.ts +0 -143
  64. package/src/deployment/ab-testing.ts +0 -582
  65. package/src/deployment/blue-green.ts +0 -452
  66. package/src/deployment/canary.ts +0 -500
  67. package/src/deployment/deployment.test.ts +0 -526
  68. package/src/deployment/index.ts +0 -61
  69. package/src/deployment/progressive.ts +0 -62
  70. package/src/dns/dns.test.ts +0 -641
  71. package/src/dns/dnssec.ts +0 -315
  72. package/src/dns/index.ts +0 -8
  73. package/src/dns/resolver.ts +0 -496
  74. package/src/dns/routing.ts +0 -593
  75. package/src/email/advanced/analytics.ts +0 -445
  76. package/src/email/advanced/index.ts +0 -11
  77. package/src/email/advanced/rules.ts +0 -465
  78. package/src/email/advanced/scheduling.ts +0 -352
  79. package/src/email/advanced/search.ts +0 -412
  80. package/src/email/advanced/shared-mailboxes.ts +0 -404
  81. package/src/email/advanced/templates.ts +0 -455
  82. package/src/email/advanced/threading.ts +0 -281
  83. package/src/email/analytics.ts +0 -467
  84. package/src/email/bounce-handling.ts +0 -425
  85. package/src/email/email.test.ts +0 -431
  86. package/src/email/handlers/__tests__/inbound.test.ts +0 -38
  87. package/src/email/handlers/__tests__/outbound.test.ts +0 -37
  88. package/src/email/handlers/converter.ts +0 -227
  89. package/src/email/handlers/feedback.ts +0 -228
  90. package/src/email/handlers/inbound.ts +0 -169
  91. package/src/email/handlers/outbound.ts +0 -178
  92. package/src/email/index.ts +0 -15
  93. package/src/email/reputation.ts +0 -303
  94. package/src/email/templates.ts +0 -352
  95. package/src/errors/index.test.ts +0 -434
  96. package/src/errors/index.ts +0 -416
  97. package/src/health-checks/index.ts +0 -40
  98. package/src/index.ts +0 -360
  99. package/src/intrinsic-functions.ts +0 -118
  100. package/src/lambda/concurrency.ts +0 -330
  101. package/src/lambda/destinations.ts +0 -345
  102. package/src/lambda/dlq.ts +0 -425
  103. package/src/lambda/index.ts +0 -11
  104. package/src/lambda/lambda.test.ts +0 -840
  105. package/src/lambda/layers.ts +0 -263
  106. package/src/lambda/versions.ts +0 -376
  107. package/src/lambda/vpc.ts +0 -399
  108. package/src/local/config.ts +0 -114
  109. package/src/local/index.ts +0 -6
  110. package/src/local/mock-aws.ts +0 -351
  111. package/src/modules/ai.ts +0 -340
  112. package/src/modules/api.ts +0 -478
  113. package/src/modules/auth.ts +0 -805
  114. package/src/modules/cache.ts +0 -417
  115. package/src/modules/cdn.ts +0 -1062
  116. package/src/modules/communication.ts +0 -1094
  117. package/src/modules/compute.ts +0 -3348
  118. package/src/modules/database.ts +0 -554
  119. package/src/modules/deployment.ts +0 -1079
  120. package/src/modules/dns.ts +0 -337
  121. package/src/modules/email.ts +0 -1538
  122. package/src/modules/filesystem.ts +0 -515
  123. package/src/modules/index.ts +0 -32
  124. package/src/modules/messaging.ts +0 -486
  125. package/src/modules/monitoring.ts +0 -2086
  126. package/src/modules/network.ts +0 -664
  127. package/src/modules/parameter-store.ts +0 -325
  128. package/src/modules/permissions.ts +0 -1081
  129. package/src/modules/phone.ts +0 -494
  130. package/src/modules/queue.ts +0 -1260
  131. package/src/modules/redirects.ts +0 -464
  132. package/src/modules/registry.ts +0 -699
  133. package/src/modules/search.ts +0 -401
  134. package/src/modules/secrets.ts +0 -416
  135. package/src/modules/security.ts +0 -731
  136. package/src/modules/sms.ts +0 -389
  137. package/src/modules/storage.ts +0 -1120
  138. package/src/modules/workflow.ts +0 -680
  139. package/src/multi-account/config.ts +0 -521
  140. package/src/multi-account/index.ts +0 -7
  141. package/src/multi-account/manager.ts +0 -427
  142. package/src/multi-region/cross-region.ts +0 -410
  143. package/src/multi-region/index.ts +0 -8
  144. package/src/multi-region/manager.ts +0 -483
  145. package/src/multi-region/regions.ts +0 -435
  146. package/src/network-security/index.ts +0 -48
  147. package/src/observability/index.ts +0 -9
  148. package/src/observability/logs.ts +0 -522
  149. package/src/observability/metrics.ts +0 -460
  150. package/src/observability/observability.test.ts +0 -782
  151. package/src/observability/synthetics.ts +0 -568
  152. package/src/observability/xray.ts +0 -358
  153. package/src/phone/advanced/analytics.ts +0 -349
  154. package/src/phone/advanced/callbacks.ts +0 -428
  155. package/src/phone/advanced/index.ts +0 -8
  156. package/src/phone/advanced/ivr-builder.ts +0 -504
  157. package/src/phone/advanced/recording.ts +0 -310
  158. package/src/phone/handlers/__tests__/incoming-call.test.ts +0 -40
  159. package/src/phone/handlers/incoming-call.ts +0 -117
  160. package/src/phone/handlers/missed-call.ts +0 -116
  161. package/src/phone/handlers/voicemail.ts +0 -179
  162. package/src/phone/index.ts +0 -9
  163. package/src/presets/api-backend.ts +0 -134
  164. package/src/presets/data-pipeline.ts +0 -204
  165. package/src/presets/extend.test.ts +0 -295
  166. package/src/presets/extend.ts +0 -297
  167. package/src/presets/fullstack-app.ts +0 -144
  168. package/src/presets/index.ts +0 -27
  169. package/src/presets/jamstack.ts +0 -135
  170. package/src/presets/microservices.ts +0 -167
  171. package/src/presets/ml-api.ts +0 -208
  172. package/src/presets/nodejs-server.ts +0 -104
  173. package/src/presets/nodejs-serverless.ts +0 -114
  174. package/src/presets/realtime-app.ts +0 -184
  175. package/src/presets/static-site.ts +0 -64
  176. package/src/presets/traditional-web-app.ts +0 -339
  177. package/src/presets/wordpress.ts +0 -138
  178. package/src/preview/github.test.ts +0 -249
  179. package/src/preview/github.ts +0 -297
  180. package/src/preview/index.ts +0 -37
  181. package/src/preview/manager.test.ts +0 -440
  182. package/src/preview/manager.ts +0 -326
  183. package/src/preview/notifications.test.ts +0 -582
  184. package/src/preview/notifications.ts +0 -341
  185. package/src/queue/batch-processing.ts +0 -402
  186. package/src/queue/dlq-monitoring.ts +0 -402
  187. package/src/queue/fifo.ts +0 -342
  188. package/src/queue/index.ts +0 -9
  189. package/src/queue/management.ts +0 -428
  190. package/src/queue/queue.test.ts +0 -429
  191. package/src/resource-mgmt/index.ts +0 -39
  192. package/src/resource-naming.ts +0 -62
  193. package/src/s3/index.ts +0 -523
  194. package/src/schema/cloud-config.schema.json +0 -554
  195. package/src/schema/index.ts +0 -68
  196. package/src/security/certificate-manager.ts +0 -492
  197. package/src/security/index.ts +0 -9
  198. package/src/security/scanning.ts +0 -545
  199. package/src/security/secrets-manager.ts +0 -476
  200. package/src/security/secrets-rotation.ts +0 -456
  201. package/src/security/security.test.ts +0 -738
  202. package/src/sms/advanced/ab-testing.ts +0 -389
  203. package/src/sms/advanced/analytics.ts +0 -336
  204. package/src/sms/advanced/campaigns.ts +0 -523
  205. package/src/sms/advanced/chatbot.ts +0 -224
  206. package/src/sms/advanced/index.ts +0 -10
  207. package/src/sms/advanced/link-tracking.ts +0 -248
  208. package/src/sms/advanced/mms.ts +0 -308
  209. package/src/sms/handlers/__tests__/send.test.ts +0 -40
  210. package/src/sms/handlers/delivery-status.ts +0 -133
  211. package/src/sms/handlers/receive.ts +0 -162
  212. package/src/sms/handlers/send.ts +0 -174
  213. package/src/sms/index.ts +0 -9
  214. package/src/stack-diff.ts +0 -389
  215. package/src/static-site/index.ts +0 -85
  216. package/src/template-builder.ts +0 -110
  217. package/src/template-validator.ts +0 -574
  218. package/src/utils/cache.ts +0 -291
  219. package/src/utils/diff.ts +0 -269
  220. package/src/utils/hash.ts +0 -227
  221. package/src/utils/index.ts +0 -8
  222. package/src/utils/parallel.ts +0 -294
  223. package/src/validators/credentials.test.ts +0 -274
  224. package/src/validators/credentials.ts +0 -233
  225. package/src/validators/quotas.test.ts +0 -434
  226. package/src/validators/quotas.ts +0 -217
  227. package/test/ai.test.ts +0 -327
  228. package/test/api.test.ts +0 -511
  229. package/test/auth.test.ts +0 -632
  230. package/test/cache.test.ts +0 -406
  231. package/test/cdn.test.ts +0 -247
  232. package/test/compute.test.ts +0 -861
  233. package/test/database.test.ts +0 -523
  234. package/test/deployment.test.ts +0 -499
  235. package/test/dns.test.ts +0 -270
  236. package/test/email.test.ts +0 -439
  237. package/test/filesystem.test.ts +0 -382
  238. package/test/integration.test.ts +0 -350
  239. package/test/messaging.test.ts +0 -514
  240. package/test/monitoring.test.ts +0 -634
  241. package/test/network.test.ts +0 -425
  242. package/test/permissions.test.ts +0 -488
  243. package/test/queue.test.ts +0 -484
  244. package/test/registry.test.ts +0 -306
  245. package/test/security.test.ts +0 -462
  246. package/test/storage.test.ts +0 -463
  247. package/test/template-validator.test.ts +0 -559
  248. package/test/workflow.test.ts +0 -592
  249. package/tsconfig.json +0 -16
  250. package/tsconfig.tsbuildinfo +0 -1
@@ -1,421 +0,0 @@
1
- /**
2
- * AWS CloudFormation API Client
3
- * Direct API calls without AWS SDK dependency
4
- */
5
-
6
- import type { AWSCredentials } from './credentials'
7
- import { resolveCredentials } from './credentials'
8
- import { makeAWSRequest, parseXMLResponse } from './signature'
9
-
10
- export interface CloudFormationStack {
11
- StackName: string
12
- StackId?: string
13
- StackStatus?: string
14
- CreationTime?: string
15
- LastUpdatedTime?: string
16
- StackStatusReason?: string
17
- Description?: string
18
- Parameters?: Array<{ ParameterKey: string, ParameterValue: string }>
19
- Outputs?: Array<{ OutputKey: string, OutputValue: string, Description?: string }>
20
- Tags?: Array<{ Key: string, Value: string }>
21
- }
22
-
23
- export interface CreateStackOptions {
24
- stackName: string
25
- templateBody?: string
26
- templateURL?: string
27
- parameters?: Record<string, string>
28
- capabilities?: string[]
29
- tags?: Record<string, string>
30
- timeoutInMinutes?: number
31
- onFailure?: 'DO_NOTHING' | 'ROLLBACK' | 'DELETE'
32
- }
33
-
34
- export interface UpdateStackOptions {
35
- stackName: string
36
- templateBody?: string
37
- templateURL?: string
38
- parameters?: Record<string, string>
39
- capabilities?: string[]
40
- tags?: Record<string, string>
41
- }
42
-
43
- export interface StackEvent {
44
- EventId: string
45
- StackName: string
46
- LogicalResourceId: string
47
- PhysicalResourceId?: string
48
- ResourceType: string
49
- Timestamp: string
50
- ResourceStatus: string
51
- ResourceStatusReason?: string
52
- }
53
-
54
- /**
55
- * CloudFormation API Client
56
- */
57
- export class CloudFormationClient {
58
- private credentials: AWSCredentials | null = null
59
- private region: string
60
-
61
- constructor(
62
- region: string = 'us-east-1',
63
- private readonly profile: string = 'default',
64
- ) {
65
- this.region = region
66
- }
67
-
68
- /**
69
- * Initialize client with credentials
70
- */
71
- async init(): Promise<void> {
72
- this.credentials = await resolveCredentials(this.profile)
73
- if (this.credentials.region) {
74
- this.region = this.credentials.region
75
- }
76
- }
77
-
78
- /**
79
- * Ensure credentials are loaded
80
- */
81
- private async ensureCredentials(): Promise<AWSCredentials> {
82
- if (!this.credentials) {
83
- await this.init()
84
- }
85
- return this.credentials!
86
- }
87
-
88
- /**
89
- * Make a CloudFormation API request
90
- */
91
- private async request(action: string, params: Record<string, any>): Promise<any> {
92
- const credentials = await this.ensureCredentials()
93
-
94
- // Build query string
95
- const queryParams: Record<string, string> = {
96
- Action: action,
97
- Version: '2010-05-15',
98
- ...flattenParams(params),
99
- }
100
-
101
- const queryString = new URLSearchParams(queryParams).toString()
102
-
103
- const response = await makeAWSRequest({
104
- method: 'POST',
105
- url: `https://cloudformation.${this.region}.amazonaws.com/`,
106
- service: 'cloudformation',
107
- region: this.region,
108
- headers: {
109
- 'Content-Type': 'application/x-www-form-urlencoded',
110
- },
111
- body: queryString,
112
- accessKeyId: credentials.accessKeyId,
113
- secretAccessKey: credentials.secretAccessKey,
114
- sessionToken: credentials.sessionToken,
115
- })
116
-
117
- return await parseXMLResponse(response)
118
- }
119
-
120
- /**
121
- * Create a new CloudFormation stack
122
- */
123
- async createStack(options: CreateStackOptions): Promise<string> {
124
- const params: Record<string, any> = {
125
- StackName: options.stackName,
126
- }
127
-
128
- if (options.templateBody) {
129
- params.TemplateBody = options.templateBody
130
- }
131
- else if (options.templateURL) {
132
- params.TemplateURL = options.templateURL
133
- }
134
- else {
135
- throw new Error('Either templateBody or templateURL must be provided')
136
- }
137
-
138
- if (options.parameters) {
139
- params.Parameters = Object.entries(options.parameters).map(([key, value], index) => ({
140
- [`Parameters.member.${index + 1}.ParameterKey`]: key,
141
- [`Parameters.member.${index + 1}.ParameterValue`]: value,
142
- }))
143
- }
144
-
145
- if (options.capabilities) {
146
- params.Capabilities = options.capabilities.map((cap, index) => ({
147
- [`Capabilities.member.${index + 1}`]: cap,
148
- }))
149
- }
150
-
151
- if (options.tags) {
152
- params.Tags = Object.entries(options.tags).map(([key, value], index) => ({
153
- [`Tags.member.${index + 1}.Key`]: key,
154
- [`Tags.member.${index + 1}.Value`]: value,
155
- }))
156
- }
157
-
158
- if (options.timeoutInMinutes) {
159
- params.TimeoutInMinutes = options.timeoutInMinutes
160
- }
161
-
162
- if (options.onFailure) {
163
- params.OnFailure = options.onFailure
164
- }
165
-
166
- const result = await this.request('CreateStack', params)
167
- return result.StackId
168
- }
169
-
170
- /**
171
- * Update an existing CloudFormation stack
172
- */
173
- async updateStack(options: UpdateStackOptions): Promise<string> {
174
- const params: Record<string, any> = {
175
- StackName: options.stackName,
176
- }
177
-
178
- if (options.templateBody) {
179
- params.TemplateBody = options.templateBody
180
- }
181
- else if (options.templateURL) {
182
- params.TemplateURL = options.templateURL
183
- }
184
-
185
- if (options.parameters) {
186
- params.Parameters = Object.entries(options.parameters).map(([key, value], index) => ({
187
- [`Parameters.member.${index + 1}.ParameterKey`]: key,
188
- [`Parameters.member.${index + 1}.ParameterValue`]: value,
189
- }))
190
- }
191
-
192
- if (options.capabilities) {
193
- params.Capabilities = options.capabilities.map((cap, index) => ({
194
- [`Capabilities.member.${index + 1}`]: cap,
195
- }))
196
- }
197
-
198
- if (options.tags) {
199
- params.Tags = Object.entries(options.tags).map(([key, value], index) => ({
200
- [`Tags.member.${index + 1}.Key`]: key,
201
- [`Tags.member.${index + 1}.Value`]: value,
202
- }))
203
- }
204
-
205
- const result = await this.request('UpdateStack', params)
206
- return result.StackId
207
- }
208
-
209
- /**
210
- * Delete a CloudFormation stack
211
- */
212
- async deleteStack(stackName: string): Promise<void> {
213
- await this.request('DeleteStack', {
214
- StackName: stackName,
215
- })
216
- }
217
-
218
- /**
219
- * Describe a CloudFormation stack
220
- */
221
- async describeStack(stackName: string): Promise<CloudFormationStack> {
222
- const result = await this.request('DescribeStacks', {
223
- StackName: stackName,
224
- })
225
-
226
- // Parse stack from XML response
227
- return parseStack(result)
228
- }
229
-
230
- /**
231
- * List all CloudFormation stacks
232
- */
233
- async listStacks(statusFilter?: string[]): Promise<CloudFormationStack[]> {
234
- const params: Record<string, any> = {}
235
-
236
- if (statusFilter) {
237
- params.StackStatusFilter = statusFilter.map((status, index) => ({
238
- [`StackStatusFilter.member.${index + 1}`]: status,
239
- }))
240
- }
241
-
242
- const result = await this.request('ListStacks', params)
243
- return parseStackList(result)
244
- }
245
-
246
- /**
247
- * Get stack events
248
- */
249
- async describeStackEvents(stackName: string): Promise<StackEvent[]> {
250
- const result = await this.request('DescribeStackEvents', {
251
- StackName: stackName,
252
- })
253
-
254
- return parseStackEvents(result)
255
- }
256
-
257
- /**
258
- * Wait for stack to reach a terminal state
259
- */
260
- async waitForStack(
261
- stackName: string,
262
- desiredStates: string[],
263
- onProgress?: (event: StackEvent) => void,
264
- ): Promise<CloudFormationStack> {
265
- const pollInterval = 5000 // 5 seconds
266
- const maxAttempts = 360 // 30 minutes maximum
267
- let attempts = 0
268
- let lastEventId: string | null = null
269
-
270
- while (attempts < maxAttempts) {
271
- attempts++
272
-
273
- const stack = await this.describeStack(stackName)
274
-
275
- // Get latest events
276
- if (onProgress) {
277
- const events = await this.describeStackEvents(stackName)
278
- const newEvents = lastEventId
279
- ? events.filter(e => e.EventId !== lastEventId).reverse()
280
- : [events[0]]
281
-
282
- newEvents.forEach(event => onProgress(event))
283
-
284
- if (events.length > 0) {
285
- lastEventId = events[0].EventId
286
- }
287
- }
288
-
289
- // Check if stack reached desired state
290
- if (stack.StackStatus && desiredStates.includes(stack.StackStatus)) {
291
- return stack
292
- }
293
-
294
- // Check for failure states
295
- if (stack.StackStatus?.includes('FAILED') || stack.StackStatus?.includes('ROLLBACK')) {
296
- throw new Error(
297
- `Stack reached failed state: ${stack.StackStatus}\n`
298
- + `Reason: ${stack.StackStatusReason || 'Unknown'}`,
299
- )
300
- }
301
-
302
- // Wait before next poll
303
- await new Promise(resolve => setTimeout(resolve, pollInterval))
304
- }
305
-
306
- throw new Error(`Timeout waiting for stack ${stackName} to complete`)
307
- }
308
-
309
- /**
310
- * Create a change set
311
- */
312
- async createChangeSet(options: {
313
- stackName: string
314
- changeSetName: string
315
- templateBody?: string
316
- templateURL?: string
317
- parameters?: Record<string, string>
318
- capabilities?: string[]
319
- }): Promise<string> {
320
- const params: Record<string, any> = {
321
- StackName: options.stackName,
322
- ChangeSetName: options.changeSetName,
323
- ChangeSetType: 'UPDATE',
324
- }
325
-
326
- if (options.templateBody) {
327
- params.TemplateBody = options.templateBody
328
- }
329
- else if (options.templateURL) {
330
- params.TemplateURL = options.templateURL
331
- }
332
-
333
- if (options.parameters) {
334
- params.Parameters = Object.entries(options.parameters).map(([key, value], index) => ({
335
- [`Parameters.member.${index + 1}.ParameterKey`]: key,
336
- [`Parameters.member.${index + 1}.ParameterValue`]: value,
337
- }))
338
- }
339
-
340
- if (options.capabilities) {
341
- params.Capabilities = options.capabilities.map((cap, index) => ({
342
- [`Capabilities.member.${index + 1}`]: cap,
343
- }))
344
- }
345
-
346
- const result = await this.request('CreateChangeSet', params)
347
- return result.ChangeSetId
348
- }
349
-
350
- /**
351
- * Execute a change set
352
- */
353
- async executeChangeSet(changeSetName: string, stackName: string): Promise<void> {
354
- await this.request('ExecuteChangeSet', {
355
- ChangeSetName: changeSetName,
356
- StackName: stackName,
357
- })
358
- }
359
- }
360
-
361
- /**
362
- * Flatten nested parameters for AWS API query string
363
- */
364
- function flattenParams(params: Record<string, any>, prefix: string = ''): Record<string, string> {
365
- const result: Record<string, string> = {}
366
-
367
- for (const [key, value] of Object.entries(params)) {
368
- const fullKey = prefix ? `${prefix}.${key}` : key
369
-
370
- if (Array.isArray(value)) {
371
- value.forEach((item, index) => {
372
- if (typeof item === 'object') {
373
- Object.assign(result, flattenParams(item, `${fullKey}.${index + 1}`))
374
- }
375
- else {
376
- result[`${fullKey}.${index + 1}`] = String(item)
377
- }
378
- })
379
- }
380
- else if (typeof value === 'object' && value !== null) {
381
- Object.assign(result, flattenParams(value, fullKey))
382
- }
383
- else if (value !== undefined && value !== null) {
384
- result[fullKey] = String(value)
385
- }
386
- }
387
-
388
- return result
389
- }
390
-
391
- /**
392
- * Parse stack from XML response
393
- */
394
- function parseStack(data: any): CloudFormationStack {
395
- // Simplified parsing - in production, use a proper XML parser
396
- return {
397
- StackName: data.StackName || '',
398
- StackId: data.StackId,
399
- StackStatus: data.StackStatus,
400
- CreationTime: data.CreationTime,
401
- LastUpdatedTime: data.LastUpdatedTime,
402
- StackStatusReason: data.StackStatusReason,
403
- Description: data.Description,
404
- }
405
- }
406
-
407
- /**
408
- * Parse stack list from XML response
409
- */
410
- function parseStackList(data: any): CloudFormationStack[] {
411
- // Simplified parsing
412
- return []
413
- }
414
-
415
- /**
416
- * Parse stack events from XML response
417
- */
418
- function parseStackEvents(data: any): StackEvent[] {
419
- // Simplified parsing
420
- return []
421
- }
@@ -1,158 +0,0 @@
1
- /**
2
- * AWS CloudFront API Client
3
- * Direct API calls for CloudFront invalidations without AWS SDK
4
- */
5
-
6
- import type { AWSCredentials } from './credentials'
7
- import { resolveCredentials } from './credentials'
8
- import { makeAWSRequest, parseXMLResponse } from './signature'
9
-
10
- export interface InvalidationOptions {
11
- distributionId: string
12
- paths: string[]
13
- callerReference?: string
14
- }
15
-
16
- /**
17
- * CloudFront API Client
18
- */
19
- export class CloudFrontClient {
20
- private credentials: AWSCredentials | null = null
21
-
22
- constructor(
23
- private readonly profile: string = 'default',
24
- ) {}
25
-
26
- /**
27
- * Initialize client with credentials
28
- */
29
- async init(): Promise<void> {
30
- this.credentials = await resolveCredentials(this.profile)
31
- }
32
-
33
- /**
34
- * Ensure credentials are loaded
35
- */
36
- private async ensureCredentials(): Promise<AWSCredentials> {
37
- if (!this.credentials) {
38
- await this.init()
39
- }
40
- return this.credentials!
41
- }
42
-
43
- /**
44
- * Create a cache invalidation
45
- */
46
- async createInvalidation(options: InvalidationOptions): Promise<string> {
47
- const credentials = await this.ensureCredentials()
48
-
49
- const callerReference = options.callerReference || `invalidation-${Date.now()}`
50
-
51
- const body = `<?xml version="1.0" encoding="UTF-8"?>
52
- <InvalidationBatch>
53
- <Paths>
54
- <Quantity>${options.paths.length}</Quantity>
55
- <Items>
56
- ${options.paths.map(path => `<Path>${path}</Path>`).join('')}
57
- </Items>
58
- </Paths>
59
- <CallerReference>${callerReference}</CallerReference>
60
- </InvalidationBatch>`
61
-
62
- const url = `https://cloudfront.amazonaws.com/2020-05-31/distribution/${options.distributionId}/invalidation`
63
-
64
- const response = await makeAWSRequest({
65
- method: 'POST',
66
- url,
67
- service: 'cloudfront',
68
- region: 'us-east-1', // CloudFront is global, but uses us-east-1
69
- headers: {
70
- 'Content-Type': 'text/xml',
71
- },
72
- body,
73
- accessKeyId: credentials.accessKeyId,
74
- secretAccessKey: credentials.secretAccessKey,
75
- sessionToken: credentials.sessionToken,
76
- })
77
-
78
- const data = await parseXMLResponse(response)
79
- return data.Id
80
- }
81
-
82
- /**
83
- * Get invalidation status
84
- */
85
- async getInvalidation(distributionId: string, invalidationId: string): Promise<any> {
86
- const credentials = await this.ensureCredentials()
87
-
88
- const url = `https://cloudfront.amazonaws.com/2020-05-31/distribution/${distributionId}/invalidation/${invalidationId}`
89
-
90
- const response = await makeAWSRequest({
91
- method: 'GET',
92
- url,
93
- service: 'cloudfront',
94
- region: 'us-east-1',
95
- accessKeyId: credentials.accessKeyId,
96
- secretAccessKey: credentials.secretAccessKey,
97
- sessionToken: credentials.sessionToken,
98
- })
99
-
100
- return await parseXMLResponse(response)
101
- }
102
-
103
- /**
104
- * List invalidations for a distribution
105
- */
106
- async listInvalidations(distributionId: string): Promise<any[]> {
107
- const credentials = await this.ensureCredentials()
108
-
109
- const url = `https://cloudfront.amazonaws.com/2020-05-31/distribution/${distributionId}/invalidation`
110
-
111
- const response = await makeAWSRequest({
112
- method: 'GET',
113
- url,
114
- service: 'cloudfront',
115
- region: 'us-east-1',
116
- accessKeyId: credentials.accessKeyId,
117
- secretAccessKey: credentials.secretAccessKey,
118
- sessionToken: credentials.sessionToken,
119
- })
120
-
121
- const data = await parseXMLResponse(response)
122
- return data.Items || []
123
- }
124
-
125
- /**
126
- * Wait for invalidation to complete
127
- */
128
- async waitForInvalidation(
129
- distributionId: string,
130
- invalidationId: string,
131
- maxAttempts: number = 60,
132
- pollInterval: number = 5000,
133
- ): Promise<void> {
134
- for (let i = 0; i < maxAttempts; i++) {
135
- const invalidation = await this.getInvalidation(distributionId, invalidationId)
136
-
137
- if (invalidation.Status === 'Completed') {
138
- return
139
- }
140
-
141
- if (i < maxAttempts - 1) {
142
- await new Promise(resolve => setTimeout(resolve, pollInterval))
143
- }
144
- }
145
-
146
- throw new Error(`Invalidation ${invalidationId} did not complete within the expected time`)
147
- }
148
-
149
- /**
150
- * Invalidate all files in a distribution
151
- */
152
- async invalidateAll(distributionId: string): Promise<string> {
153
- return await this.createInvalidation({
154
- distributionId,
155
- paths: ['/*'],
156
- })
157
- }
158
- }
@@ -1,132 +0,0 @@
1
- /**
2
- * AWS Credentials Provider Tests
3
- */
4
-
5
- import { describe, expect, it, beforeEach, afterEach } from 'bun:test'
6
- import {
7
- fromEnvironment,
8
- fromSharedCredentials,
9
- createCredentialProvider,
10
- } from './credentials'
11
-
12
- describe('Credential Providers', () => {
13
- // Store original env
14
- const originalEnv = { ...process.env }
15
-
16
- beforeEach(() => {
17
- // Reset env before each test
18
- process.env = { ...originalEnv }
19
- })
20
-
21
- afterEach(() => {
22
- // Restore original env
23
- process.env = originalEnv
24
- })
25
-
26
- describe('fromEnvironment', () => {
27
- it('should return credentials from environment variables', () => {
28
- process.env.AWS_ACCESS_KEY_ID = 'AKIATEST123'
29
- process.env.AWS_SECRET_ACCESS_KEY = 'secret123'
30
-
31
- const creds = fromEnvironment()
32
-
33
- expect(creds).not.toBeNull()
34
- expect(creds?.accessKeyId).toBe('AKIATEST123')
35
- expect(creds?.secretAccessKey).toBe('secret123')
36
- expect(creds?.sessionToken).toBeUndefined()
37
- })
38
-
39
- it('should include session token if present', () => {
40
- process.env.AWS_ACCESS_KEY_ID = 'AKIATEST123'
41
- process.env.AWS_SECRET_ACCESS_KEY = 'secret123'
42
- process.env.AWS_SESSION_TOKEN = 'token123'
43
-
44
- const creds = fromEnvironment()
45
-
46
- expect(creds).not.toBeNull()
47
- expect(creds?.sessionToken).toBe('token123')
48
- })
49
-
50
- it('should return null if access key is missing', () => {
51
- process.env.AWS_SECRET_ACCESS_KEY = 'secret123'
52
- delete process.env.AWS_ACCESS_KEY_ID
53
-
54
- const creds = fromEnvironment()
55
-
56
- expect(creds).toBeNull()
57
- })
58
-
59
- it('should return null if secret key is missing', () => {
60
- process.env.AWS_ACCESS_KEY_ID = 'AKIATEST123'
61
- delete process.env.AWS_SECRET_ACCESS_KEY
62
-
63
- const creds = fromEnvironment()
64
-
65
- expect(creds).toBeNull()
66
- })
67
- })
68
-
69
- describe('fromSharedCredentials', () => {
70
- it('should return null for non-existent file', () => {
71
- const creds = fromSharedCredentials({
72
- credentialsFile: '/nonexistent/path/credentials',
73
- })
74
-
75
- expect(creds).toBeNull()
76
- })
77
-
78
- it('should parse credentials from a valid file', () => {
79
- // This test would need a mock file or temp file
80
- // For now, just verify the function doesn't crash
81
- const creds = fromSharedCredentials({
82
- credentialsFile: '/tmp/test-aws-credentials-nonexistent',
83
- })
84
-
85
- expect(creds).toBeNull()
86
- })
87
- })
88
-
89
- describe('createCredentialProvider', () => {
90
- it('should create a provider function', () => {
91
- const provider = createCredentialProvider()
92
- expect(typeof provider).toBe('function')
93
- })
94
-
95
- it('should cache credentials', async () => {
96
- process.env.AWS_ACCESS_KEY_ID = 'AKIATEST123'
97
- process.env.AWS_SECRET_ACCESS_KEY = 'secret123'
98
-
99
- const provider = createCredentialProvider()
100
-
101
- const creds1 = await provider()
102
- const creds2 = await provider()
103
-
104
- expect(creds1).toEqual(creds2)
105
- expect(creds1.accessKeyId).toBe('AKIATEST123')
106
- })
107
-
108
- it('should throw if no credentials found', async () => {
109
- delete process.env.AWS_ACCESS_KEY_ID
110
- delete process.env.AWS_SECRET_ACCESS_KEY
111
- delete process.env.AWS_PROFILE
112
-
113
- const provider = createCredentialProvider({
114
- credentialsFile: '/nonexistent/path',
115
- })
116
-
117
- // This will try all providers and fail
118
- // In real tests, you'd mock the metadata endpoints
119
- await expect(provider()).rejects.toThrow('Could not find AWS credentials')
120
- })
121
- })
122
- })
123
-
124
- describe('Credential File Parsing', () => {
125
- it('should handle empty credentials gracefully', () => {
126
- delete process.env.AWS_ACCESS_KEY_ID
127
- delete process.env.AWS_SECRET_ACCESS_KEY
128
-
129
- const creds = fromEnvironment()
130
- expect(creds).toBeNull()
131
- })
132
- })