@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,332 @@
1
+ /**
2
+ * Table formatting utilities for CLI output
3
+ * Better table rendering with borders, alignment, colors
4
+ */
5
+
6
+ export interface TableColumn {
7
+ key: string
8
+ label: string
9
+ width?: number
10
+ align?: 'left' | 'right' | 'center'
11
+ formatter?: (value: any) => string
12
+ }
13
+
14
+ export interface TableOptions {
15
+ columns: TableColumn[]
16
+ data: Record<string, any>[]
17
+ border?: boolean
18
+ header?: boolean
19
+ compact?: boolean
20
+ maxWidth?: number
21
+ }
22
+
23
+ /**
24
+ * Format data as a table
25
+ */
26
+ export function formatTable(options: TableOptions): string {
27
+ const { columns, data, border = true, header = true, compact = false, maxWidth } = options
28
+
29
+ if (data.length === 0) {
30
+ return 'No data to display'
31
+ }
32
+
33
+ // Calculate column widths
34
+ const colWidths = columns.map((col) => {
35
+ const labelWidth = col.label.length
36
+ const dataWidth = Math.max(
37
+ ...data.map((row) => {
38
+ const value = col.formatter ? col.formatter(row[col.key]) : String(row[col.key] || '')
39
+ return value.length
40
+ }),
41
+ )
42
+
43
+ let width = col.width || Math.max(labelWidth, dataWidth)
44
+
45
+ // Apply max width if specified
46
+ if (maxWidth && width > maxWidth) {
47
+ width = maxWidth
48
+ }
49
+
50
+ return width
51
+ })
52
+
53
+ const lines: string[] = []
54
+
55
+ // Top border
56
+ if (border) {
57
+ lines.push(createBorder(colWidths, 'top', compact))
58
+ }
59
+
60
+ // Header
61
+ if (header) {
62
+ lines.push(createRow(columns.map(col => col.label), colWidths, columns.map(col => col.align || 'left'), border))
63
+
64
+ // Header separator
65
+ if (border) {
66
+ lines.push(createBorder(colWidths, 'middle', compact))
67
+ }
68
+ }
69
+
70
+ // Data rows
71
+ for (const row of data) {
72
+ const values = columns.map((col) => {
73
+ const value = col.formatter ? col.formatter(row[col.key]) : String(row[col.key] || '')
74
+ return truncate(value, colWidths[columns.indexOf(col)])
75
+ })
76
+
77
+ lines.push(createRow(values, colWidths, columns.map(col => col.align || 'left'), border))
78
+ }
79
+
80
+ // Bottom border
81
+ if (border) {
82
+ lines.push(createBorder(colWidths, 'bottom', compact))
83
+ }
84
+
85
+ return lines.join('\n')
86
+ }
87
+
88
+ /**
89
+ * Create a table row
90
+ */
91
+ function createRow(
92
+ values: string[],
93
+ widths: number[],
94
+ alignments: Array<'left' | 'right' | 'center'>,
95
+ border: boolean,
96
+ ): string {
97
+ const cells = values.map((value, i) => {
98
+ const width = widths[i]
99
+ const align = alignments[i]
100
+
101
+ return alignText(value, width, align)
102
+ })
103
+
104
+ if (border) {
105
+ return `│ ${cells.join(' │ ')} │`
106
+ }
107
+
108
+ return cells.join(' ')
109
+ }
110
+
111
+ /**
112
+ * Create a table border
113
+ */
114
+ function createBorder(widths: number[], position: 'top' | 'middle' | 'bottom', compact: boolean): string {
115
+ const left = position === 'top' ? '┌' : position === 'middle' ? '├' : '└'
116
+ const right = position === 'top' ? '┐' : position === 'middle' ? '┤' : '┘'
117
+ const cross = position === 'top' ? '┬' : position === 'middle' ? '┼' : '┴'
118
+ const horizontal = '─'
119
+
120
+ if (compact) {
121
+ return left + widths.map(w => horizontal.repeat(w + 2)).join(cross) + right
122
+ }
123
+
124
+ return left + widths.map(w => horizontal.repeat(w + 2)).join(cross) + right
125
+ }
126
+
127
+ /**
128
+ * Align text within a fixed width
129
+ */
130
+ function alignText(text: string, width: number, align: 'left' | 'right' | 'center'): string {
131
+ const textWidth = stripAnsi(text).length
132
+
133
+ if (textWidth >= width) {
134
+ return text
135
+ }
136
+
137
+ const padding = width - textWidth
138
+
139
+ switch (align) {
140
+ case 'right':
141
+ return ' '.repeat(padding) + text
142
+ case 'center': {
143
+ const leftPad = Math.floor(padding / 2)
144
+ const rightPad = padding - leftPad
145
+ return ' '.repeat(leftPad) + text + ' '.repeat(rightPad)
146
+ }
147
+ default:
148
+ return text + ' '.repeat(padding)
149
+ }
150
+ }
151
+
152
+ /**
153
+ * Truncate text to fit width
154
+ */
155
+ function truncate(text: string, width: number): string {
156
+ const textWidth = stripAnsi(text).length
157
+
158
+ if (textWidth <= width) {
159
+ return text
160
+ }
161
+
162
+ // Truncate and add ellipsis
163
+ return text.substring(0, width - 1) + '…'
164
+ }
165
+
166
+ /**
167
+ * Strip ANSI color codes from string
168
+ */
169
+ function stripAnsi(text: string): string {
170
+ // eslint-disable-next-line no-control-regex
171
+ return text.replace(/\x1B\[[0-9;]*m/g, '')
172
+ }
173
+
174
+ /**
175
+ * Format data as a tree structure
176
+ */
177
+ export interface TreeNode {
178
+ label: string
179
+ children?: TreeNode[]
180
+ metadata?: Record<string, any>
181
+ }
182
+
183
+ export interface TreeOptions {
184
+ indent?: string
185
+ showMetadata?: boolean
186
+ }
187
+
188
+ /**
189
+ * Format data as a tree
190
+ */
191
+ export function formatTree(nodes: TreeNode[], options: TreeOptions = {}): string {
192
+ const { indent = ' ', showMetadata = false } = options
193
+
194
+ const lines: string[] = []
195
+
196
+ function renderNode(node: TreeNode, prefix: string, isLast: boolean): void {
197
+ const connector = isLast ? '└─ ' : '├─ '
198
+ const line = prefix + connector + node.label
199
+
200
+ lines.push(line)
201
+
202
+ // Render metadata if enabled
203
+ if (showMetadata && node.metadata) {
204
+ const metadataPrefix = prefix + (isLast ? ' ' : '│ ')
205
+ for (const [key, value] of Object.entries(node.metadata)) {
206
+ lines.push(`${metadataPrefix}${key}: ${value}`)
207
+ }
208
+ }
209
+
210
+ // Render children
211
+ if (node.children && node.children.length > 0) {
212
+ const childPrefix = prefix + (isLast ? ' ' : '│ ')
213
+ node.children.forEach((child, index) => {
214
+ renderNode(child, childPrefix, index === node.children!.length - 1)
215
+ })
216
+ }
217
+ }
218
+
219
+ nodes.forEach((node, index) => {
220
+ renderNode(node, '', index === nodes.length - 1)
221
+ })
222
+
223
+ return lines.join('\n')
224
+ }
225
+
226
+ /**
227
+ * Create a simple progress bar
228
+ */
229
+ export interface ProgressBarOptions {
230
+ total: number
231
+ current: number
232
+ width?: number
233
+ format?: string
234
+ complete?: string
235
+ incomplete?: string
236
+ }
237
+
238
+ /**
239
+ * Format a progress bar
240
+ */
241
+ export function formatProgressBar(options: ProgressBarOptions): string {
242
+ const {
243
+ total,
244
+ current,
245
+ width = 40,
246
+ format = ':bar :percent :current/:total',
247
+ complete = '█',
248
+ incomplete = '░',
249
+ } = options
250
+
251
+ const percentage = Math.min(100, Math.max(0, (current / total) * 100))
252
+ const completed = Math.floor((width * current) / total)
253
+ const remaining = width - completed
254
+
255
+ const bar = complete.repeat(completed) + incomplete.repeat(remaining)
256
+
257
+ return format
258
+ .replace(':bar', bar)
259
+ .replace(':percent', `${percentage.toFixed(0)}%`)
260
+ .replace(':current', String(current))
261
+ .replace(':total', String(total))
262
+ }
263
+
264
+ /**
265
+ * Format bytes as human-readable size
266
+ */
267
+ export function formatBytes(bytes: number, decimals = 2): string {
268
+ if (bytes === 0) return '0 Bytes'
269
+
270
+ const k = 1024
271
+ const dm = decimals < 0 ? 0 : decimals
272
+ const sizes = ['Bytes', 'KB', 'MB', 'GB', 'TB', 'PB', 'EB', 'ZB', 'YB']
273
+
274
+ const i = Math.floor(Math.log(bytes) / Math.log(k))
275
+
276
+ return `${Number.parseFloat((bytes / Math.pow(k, i)).toFixed(dm))} ${sizes[i]}`
277
+ }
278
+
279
+ /**
280
+ * Format duration as human-readable time
281
+ */
282
+ export function formatDuration(ms: number): string {
283
+ if (ms < 1000) return `${ms}ms`
284
+
285
+ const seconds = Math.floor(ms / 1000)
286
+ if (seconds < 60) return `${seconds}s`
287
+
288
+ const minutes = Math.floor(seconds / 60)
289
+ const remainingSeconds = seconds % 60
290
+
291
+ if (minutes < 60) {
292
+ return remainingSeconds > 0 ? `${minutes}m ${remainingSeconds}s` : `${minutes}m`
293
+ }
294
+
295
+ const hours = Math.floor(minutes / 60)
296
+ const remainingMinutes = minutes % 60
297
+
298
+ if (hours < 24) {
299
+ return remainingMinutes > 0 ? `${hours}h ${remainingMinutes}m` : `${hours}h`
300
+ }
301
+
302
+ const days = Math.floor(hours / 24)
303
+ const remainingHours = hours % 24
304
+
305
+ return remainingHours > 0 ? `${days}d ${remainingHours}h` : `${days}d`
306
+ }
307
+
308
+ /**
309
+ * Format list with bullets
310
+ */
311
+ export function formatList(items: string[], bullet = '•'): string {
312
+ return items.map(item => `${bullet} ${item}`).join('\n')
313
+ }
314
+
315
+ /**
316
+ * Format key-value pairs
317
+ */
318
+ export function formatKeyValue(
319
+ data: Record<string, any>,
320
+ options: { indent?: string; separator?: string } = {},
321
+ ): string {
322
+ const { indent = '', separator = ': ' } = options
323
+
324
+ const maxKeyLength = Math.max(...Object.keys(data).map(k => k.length))
325
+
326
+ return Object.entries(data)
327
+ .map(([key, value]) => {
328
+ const paddedKey = key.padEnd(maxKeyLength)
329
+ return `${indent}${paddedKey}${separator}${value}`
330
+ })
331
+ .join('\n')
332
+ }
@@ -0,0 +1,327 @@
1
+ /**
2
+ * CloudFormation Builder Tests
3
+ */
4
+
5
+ import { describe, expect, it } from 'bun:test'
6
+ import { CloudFormationBuilder } from './builder'
7
+ import { Fn } from './types'
8
+
9
+ describe('CloudFormationBuilder', () => {
10
+ it('should initialize with empty template', () => {
11
+ const builder = new CloudFormationBuilder({
12
+ project: { name: 'Test', slug: 'test', region: 'us-east-1' },
13
+ environments: { production: { type: 'production' } },
14
+ })
15
+
16
+ const template = builder.build()
17
+
18
+ expect(template.AWSTemplateFormatVersion).toBe('2010-09-09')
19
+ expect(template.Description).toContain('Test')
20
+ expect(template.Resources).toEqual({})
21
+ })
22
+
23
+ it('should add resources to template', () => {
24
+ const builder = new CloudFormationBuilder({
25
+ project: { name: 'Test', slug: 'test', region: 'us-east-1' },
26
+ environments: { production: { type: 'production' } },
27
+ })
28
+
29
+ builder.addResource('MyBucket', 'AWS::S3::Bucket', {
30
+ BucketName: 'my-test-bucket',
31
+ })
32
+
33
+ const template = builder.build()
34
+
35
+ expect(template.Resources.MyBucket).toBeDefined()
36
+ expect(template.Resources.MyBucket.Type).toBe('AWS::S3::Bucket')
37
+ expect(template.Resources.MyBucket.Properties!.BucketName).toBe('my-test-bucket')
38
+ })
39
+
40
+ it('should add multiple resources', () => {
41
+ const builder = new CloudFormationBuilder({
42
+ project: { name: 'Test', slug: 'test', region: 'us-east-1' },
43
+ environments: { production: { type: 'production' } },
44
+ })
45
+
46
+ builder.addResource('Bucket1', 'AWS::S3::Bucket', { BucketName: 'bucket1' })
47
+ builder.addResource('Bucket2', 'AWS::S3::Bucket', { BucketName: 'bucket2' })
48
+
49
+ const template = builder.build()
50
+
51
+ expect(template.Resources.Bucket1).toBeDefined()
52
+ expect(template.Resources.Bucket2).toBeDefined()
53
+ })
54
+
55
+ it('should handle resource with DependsOn', () => {
56
+ const builder = new CloudFormationBuilder({
57
+ project: { name: 'Test', slug: 'test', region: 'us-east-1' },
58
+ environments: { production: { type: 'production' } },
59
+ })
60
+
61
+ builder.addResource('Bucket', 'AWS::S3::Bucket', { BucketName: 'bucket' })
62
+ builder.addResource('BucketPolicy', 'AWS::S3::BucketPolicy', {
63
+ Bucket: Fn.ref('Bucket'),
64
+ }, {
65
+ dependsOn: 'Bucket',
66
+ })
67
+
68
+ const template = builder.build()
69
+
70
+ expect(template.Resources.BucketPolicy.DependsOn).toBe('Bucket')
71
+ })
72
+
73
+ it('should handle resource with multiple DependsOn', () => {
74
+ const builder = new CloudFormationBuilder({
75
+ project: { name: 'Test', slug: 'test', region: 'us-east-1' },
76
+ environments: { production: { type: 'production' } },
77
+ })
78
+
79
+ builder.addResource('Resource1', 'AWS::S3::Bucket', {})
80
+ builder.addResource('Resource2', 'AWS::S3::Bucket', {})
81
+ builder.addResource('Resource3', 'AWS::S3::Bucket', {}, {
82
+ dependsOn: ['Resource1', 'Resource2'],
83
+ })
84
+
85
+ const template = builder.build()
86
+
87
+ expect(template.Resources.Resource3.DependsOn).toEqual(['Resource1', 'Resource2'])
88
+ })
89
+
90
+ it('should handle deletion policy', () => {
91
+ const builder = new CloudFormationBuilder({
92
+ project: { name: 'Test', slug: 'test', region: 'us-east-1' },
93
+ environments: { production: { type: 'production' } },
94
+ })
95
+
96
+ builder.addResource('Database', 'AWS::RDS::DBInstance', {
97
+ DBInstanceIdentifier: 'mydb',
98
+ }, {
99
+ deletionPolicy: 'Snapshot',
100
+ })
101
+
102
+ const template = builder.build()
103
+
104
+ expect(template.Resources.Database.DeletionPolicy).toBe('Snapshot')
105
+ })
106
+
107
+ it('should include parameters in built template', () => {
108
+ const builder = new CloudFormationBuilder({
109
+ project: { name: 'Test', slug: 'test', region: 'us-east-1' },
110
+ environments: { production: { type: 'production' } },
111
+ })
112
+
113
+ const template = builder.build()
114
+
115
+ expect(template.Parameters).toBeDefined()
116
+ expect(template.Parameters?.Environment).toBeDefined()
117
+ })
118
+
119
+ it('should include conditions in built template', () => {
120
+ const builder = new CloudFormationBuilder({
121
+ project: { name: 'Test', slug: 'test', region: 'us-east-1' },
122
+ environments: { production: { type: 'production' } },
123
+ })
124
+
125
+ const template = builder.build()
126
+
127
+ expect(template.Conditions).toBeDefined()
128
+ })
129
+
130
+ it('should throw error for circular dependencies', () => {
131
+ const builder = new CloudFormationBuilder({
132
+ project: { name: 'Test', slug: 'test', region: 'us-east-1' },
133
+ environments: { production: { type: 'production' } },
134
+ })
135
+
136
+ builder.addResource('Resource1', 'AWS::S3::Bucket', {}, { dependsOn: 'Resource2' })
137
+ builder.addResource('Resource2', 'AWS::S3::Bucket', {}, { dependsOn: 'Resource1' })
138
+
139
+ expect(() => builder.build()).toThrow('Circular dependency detected')
140
+ })
141
+
142
+ it('should detect complex circular dependencies', () => {
143
+ const builder = new CloudFormationBuilder({
144
+ project: { name: 'Test', slug: 'test', region: 'us-east-1' },
145
+ environments: { production: { type: 'production' } },
146
+ })
147
+
148
+ builder.addResource('A', 'AWS::S3::Bucket', {}, { dependsOn: 'B' })
149
+ builder.addResource('B', 'AWS::S3::Bucket', {}, { dependsOn: 'C' })
150
+ builder.addResource('C', 'AWS::S3::Bucket', {}, { dependsOn: 'A' })
151
+
152
+ expect(() => builder.build()).toThrow('Circular dependency detected')
153
+ })
154
+
155
+ it('should handle valid dependency chains', () => {
156
+ const builder = new CloudFormationBuilder({
157
+ project: { name: 'Test', slug: 'test', region: 'us-east-1' },
158
+ environments: { production: { type: 'production' } },
159
+ })
160
+
161
+ builder.addResource('A', 'AWS::S3::Bucket', {})
162
+ builder.addResource('B', 'AWS::S3::Bucket', {}, { dependsOn: 'A' })
163
+ builder.addResource('C', 'AWS::S3::Bucket', {}, { dependsOn: 'B' })
164
+
165
+ expect(() => builder.build()).not.toThrow()
166
+ })
167
+
168
+ it('should handle tags', () => {
169
+ const builder = new CloudFormationBuilder({
170
+ project: { name: 'Test', slug: 'test', region: 'us-east-1' },
171
+ environments: { production: { type: 'production' } },
172
+ })
173
+
174
+ builder.addResource('Bucket', 'AWS::S3::Bucket', {
175
+ BucketName: 'bucket',
176
+ Tags: [
177
+ { Key: 'Environment', Value: 'production' },
178
+ { Key: 'Project', Value: 'test' },
179
+ ],
180
+ })
181
+
182
+ const template = builder.build()
183
+
184
+ expect(template.Resources.Bucket.Properties!.Tags).toHaveLength(2)
185
+ expect(template.Resources.Bucket.Properties!.Tags![0]).toEqual({
186
+ Key: 'Environment',
187
+ Value: 'production',
188
+ })
189
+ })
190
+ })
191
+
192
+ describe('CloudFormation Intrinsic Functions', () => {
193
+ it('should create Ref function', () => {
194
+ const ref = Fn.ref('MyBucket')
195
+
196
+ expect(ref).toEqual({ Ref: 'MyBucket' })
197
+ })
198
+
199
+ it('should create GetAtt function', () => {
200
+ const getAtt = Fn.getAtt('MyBucket', 'Arn')
201
+
202
+ expect(getAtt).toEqual({ 'Fn::GetAtt': ['MyBucket', 'Arn'] })
203
+ })
204
+
205
+ it('should create Join function', () => {
206
+ const join = Fn.join('-', ['prefix', 'middle', 'suffix'])
207
+
208
+ expect(join).toEqual({ 'Fn::Join': ['-', ['prefix', 'middle', 'suffix']] })
209
+ })
210
+
211
+ it('should create Sub function with template only', () => {
212
+ const sub = Fn.sub('arn:aws:s3:::${BucketName}')
213
+
214
+ expect(sub).toEqual({ 'Fn::Sub': 'arn:aws:s3:::${BucketName}' })
215
+ })
216
+
217
+ it('should create Sub function with variables', () => {
218
+ const sub = Fn.sub('arn:aws:s3:::${Bucket}', {
219
+ Bucket: Fn.ref('MyBucket'),
220
+ })
221
+
222
+ expect(sub).toEqual({
223
+ 'Fn::Sub': ['arn:aws:s3:::${Bucket}', { Bucket: { Ref: 'MyBucket' } }],
224
+ })
225
+ })
226
+
227
+ it('should create Select function', () => {
228
+ const select = Fn.select(0, ['a', 'b', 'c'])
229
+
230
+ expect(select).toEqual({ 'Fn::Select': [0, ['a', 'b', 'c']] })
231
+ })
232
+
233
+ it('should create Split function', () => {
234
+ const split = Fn.split(',', 'a,b,c')
235
+
236
+ expect(split).toEqual({ 'Fn::Split': [',', 'a,b,c'] })
237
+ })
238
+
239
+ it('should create If function', () => {
240
+ const ifFunc = Fn.if('IsProduction', 'prod-value', 'dev-value')
241
+
242
+ expect(ifFunc).toEqual({ 'Fn::If': ['IsProduction', 'prod-value', 'dev-value'] })
243
+ })
244
+
245
+ it('should create Equals function', () => {
246
+ const equals = Fn.equals('value1', 'value2')
247
+
248
+ expect(equals).toEqual({ 'Fn::Equals': ['value1', 'value2'] })
249
+ })
250
+
251
+ it('should create Not function', () => {
252
+ const not = Fn.not(Fn.equals('a', 'b'))
253
+
254
+ expect(not).toEqual({ 'Fn::Not': [{ 'Fn::Equals': ['a', 'b'] }] })
255
+ })
256
+
257
+ it('should create And function', () => {
258
+ const and = Fn.and(
259
+ Fn.equals('a', 'a'),
260
+ Fn.equals('b', 'b'),
261
+ )
262
+
263
+ expect(and).toEqual({
264
+ 'Fn::And': [{ 'Fn::Equals': ['a', 'a'] }, { 'Fn::Equals': ['b', 'b'] }],
265
+ })
266
+ })
267
+
268
+ it('should create Or function', () => {
269
+ const or = Fn.or(
270
+ Fn.equals('a', 'b'),
271
+ Fn.equals('c', 'c'),
272
+ )
273
+
274
+ expect(or).toEqual({
275
+ 'Fn::Or': [{ 'Fn::Equals': ['a', 'b'] }, { 'Fn::Equals': ['c', 'c'] }],
276
+ })
277
+ })
278
+
279
+ it('should create Base64 function', () => {
280
+ const base64 = Fn.base64('user data script')
281
+
282
+ expect(base64).toEqual({ 'Fn::Base64': 'user data script' })
283
+ })
284
+
285
+ it('should create Cidr function', () => {
286
+ const cidr = Fn.cidr('10.0.0.0/16', 6, 8)
287
+
288
+ expect(cidr).toEqual({ 'Fn::Cidr': ['10.0.0.0/16', 6, 8] })
289
+ })
290
+
291
+ it('should create GetAZs function', () => {
292
+ const getAZs = Fn.getAZs('us-east-1')
293
+
294
+ expect(getAZs).toEqual({ 'Fn::GetAZs': 'us-east-1' })
295
+ })
296
+
297
+ it('should create GetAZs for current region', () => {
298
+ const getAZs = Fn.getAZs()
299
+
300
+ expect(getAZs).toEqual({ 'Fn::GetAZs': '' })
301
+ })
302
+
303
+ it('should create ImportValue function', () => {
304
+ const importValue = Fn.importValue('NetworkStackVpcId')
305
+
306
+ expect(importValue).toEqual({ 'Fn::ImportValue': 'NetworkStackVpcId' })
307
+ })
308
+
309
+ it('should nest intrinsic functions', () => {
310
+ const nested = Fn.join('-', [
311
+ Fn.ref('AWS::StackName'),
312
+ 'bucket',
313
+ Fn.select(0, Fn.getAZs()),
314
+ ])
315
+
316
+ expect(nested).toEqual({
317
+ 'Fn::Join': [
318
+ '-',
319
+ [
320
+ { Ref: 'AWS::StackName' },
321
+ 'bucket',
322
+ { 'Fn::Select': [0, { 'Fn::GetAZs': '' }] },
323
+ ],
324
+ ],
325
+ })
326
+ })
327
+ })