@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.
- package/LICENSE.md +21 -0
- package/README.md +321 -0
- package/package.json +31 -0
- package/src/advanced-features.test.ts +465 -0
- package/src/aws/cloudformation.ts +421 -0
- package/src/aws/cloudfront.ts +158 -0
- package/src/aws/credentials.test.ts +132 -0
- package/src/aws/credentials.ts +545 -0
- package/src/aws/index.ts +87 -0
- package/src/aws/s3.test.ts +188 -0
- package/src/aws/s3.ts +1088 -0
- package/src/aws/signature.test.ts +670 -0
- package/src/aws/signature.ts +1155 -0
- package/src/backup/disaster-recovery.test.ts +726 -0
- package/src/backup/disaster-recovery.ts +500 -0
- package/src/backup/index.ts +34 -0
- package/src/backup/manager.test.ts +498 -0
- package/src/backup/manager.ts +432 -0
- package/src/cicd/circleci.ts +430 -0
- package/src/cicd/github-actions.ts +424 -0
- package/src/cicd/gitlab-ci.ts +255 -0
- package/src/cicd/index.ts +8 -0
- package/src/cli/history.ts +396 -0
- package/src/cli/index.ts +10 -0
- package/src/cli/progress.ts +458 -0
- package/src/cli/repl.ts +454 -0
- package/src/cli/suggestions.ts +327 -0
- package/src/cli/table.test.ts +319 -0
- package/src/cli/table.ts +332 -0
- package/src/cloudformation/builder.test.ts +327 -0
- package/src/cloudformation/builder.ts +378 -0
- package/src/cloudformation/builders/api-gateway.ts +449 -0
- package/src/cloudformation/builders/cache.ts +334 -0
- package/src/cloudformation/builders/cdn.ts +278 -0
- package/src/cloudformation/builders/compute.ts +485 -0
- package/src/cloudformation/builders/database.ts +392 -0
- package/src/cloudformation/builders/functions.ts +343 -0
- package/src/cloudformation/builders/messaging.ts +140 -0
- package/src/cloudformation/builders/monitoring.ts +300 -0
- package/src/cloudformation/builders/network.ts +264 -0
- package/src/cloudformation/builders/queue.ts +147 -0
- package/src/cloudformation/builders/security.ts +399 -0
- package/src/cloudformation/builders/storage.ts +285 -0
- package/src/cloudformation/index.ts +30 -0
- package/src/cloudformation/types.ts +173 -0
- package/src/compliance/aws-config.ts +543 -0
- package/src/compliance/cloudtrail.ts +376 -0
- package/src/compliance/compliance.test.ts +423 -0
- package/src/compliance/guardduty.ts +446 -0
- package/src/compliance/index.ts +66 -0
- package/src/compliance/security-hub.ts +456 -0
- package/src/containers/build-optimization.ts +416 -0
- package/src/containers/containers.test.ts +508 -0
- package/src/containers/image-scanning.ts +360 -0
- package/src/containers/index.ts +9 -0
- package/src/containers/registry.ts +293 -0
- package/src/containers/service-mesh.ts +520 -0
- package/src/database/database.test.ts +762 -0
- package/src/database/index.ts +9 -0
- package/src/database/migrations.ts +444 -0
- package/src/database/performance.ts +528 -0
- package/src/database/replicas.ts +534 -0
- package/src/database/users.ts +494 -0
- package/src/dependency-graph.ts +143 -0
- package/src/deployment/ab-testing.ts +582 -0
- package/src/deployment/blue-green.ts +452 -0
- package/src/deployment/canary.ts +500 -0
- package/src/deployment/deployment.test.ts +526 -0
- package/src/deployment/index.ts +61 -0
- package/src/deployment/progressive.ts +62 -0
- package/src/dns/dns.test.ts +641 -0
- package/src/dns/dnssec.ts +315 -0
- package/src/dns/index.ts +8 -0
- package/src/dns/resolver.ts +496 -0
- package/src/dns/routing.ts +593 -0
- package/src/email/advanced/analytics.ts +445 -0
- package/src/email/advanced/index.ts +11 -0
- package/src/email/advanced/rules.ts +465 -0
- package/src/email/advanced/scheduling.ts +352 -0
- package/src/email/advanced/search.ts +412 -0
- package/src/email/advanced/shared-mailboxes.ts +404 -0
- package/src/email/advanced/templates.ts +455 -0
- package/src/email/advanced/threading.ts +281 -0
- package/src/email/analytics.ts +467 -0
- package/src/email/bounce-handling.ts +425 -0
- package/src/email/email.test.ts +431 -0
- package/src/email/handlers/__tests__/inbound.test.ts +38 -0
- package/src/email/handlers/__tests__/outbound.test.ts +37 -0
- package/src/email/handlers/converter.ts +227 -0
- package/src/email/handlers/feedback.ts +228 -0
- package/src/email/handlers/inbound.ts +169 -0
- package/src/email/handlers/outbound.ts +178 -0
- package/src/email/index.ts +15 -0
- package/src/email/reputation.ts +303 -0
- package/src/email/templates.ts +352 -0
- package/src/errors/index.test.ts +434 -0
- package/src/errors/index.ts +416 -0
- package/src/health-checks/index.ts +40 -0
- package/src/index.ts +360 -0
- package/src/intrinsic-functions.ts +118 -0
- package/src/lambda/concurrency.ts +330 -0
- package/src/lambda/destinations.ts +345 -0
- package/src/lambda/dlq.ts +425 -0
- package/src/lambda/index.ts +11 -0
- package/src/lambda/lambda.test.ts +840 -0
- package/src/lambda/layers.ts +263 -0
- package/src/lambda/versions.ts +376 -0
- package/src/lambda/vpc.ts +399 -0
- package/src/local/config.ts +114 -0
- package/src/local/index.ts +6 -0
- package/src/local/mock-aws.ts +351 -0
- package/src/modules/ai.ts +340 -0
- package/src/modules/api.ts +478 -0
- package/src/modules/auth.ts +805 -0
- package/src/modules/cache.ts +417 -0
- package/src/modules/cdn.ts +1062 -0
- package/src/modules/communication.ts +1094 -0
- package/src/modules/compute.ts +3348 -0
- package/src/modules/database.ts +554 -0
- package/src/modules/deployment.ts +1079 -0
- package/src/modules/dns.ts +337 -0
- package/src/modules/email.ts +1538 -0
- package/src/modules/filesystem.ts +515 -0
- package/src/modules/index.ts +32 -0
- package/src/modules/messaging.ts +486 -0
- package/src/modules/monitoring.ts +2086 -0
- package/src/modules/network.ts +664 -0
- package/src/modules/parameter-store.ts +325 -0
- package/src/modules/permissions.ts +1081 -0
- package/src/modules/phone.ts +494 -0
- package/src/modules/queue.ts +1260 -0
- package/src/modules/redirects.ts +464 -0
- package/src/modules/registry.ts +699 -0
- package/src/modules/search.ts +401 -0
- package/src/modules/secrets.ts +416 -0
- package/src/modules/security.ts +731 -0
- package/src/modules/sms.ts +389 -0
- package/src/modules/storage.ts +1120 -0
- package/src/modules/workflow.ts +680 -0
- package/src/multi-account/config.ts +521 -0
- package/src/multi-account/index.ts +7 -0
- package/src/multi-account/manager.ts +427 -0
- package/src/multi-region/cross-region.ts +410 -0
- package/src/multi-region/index.ts +8 -0
- package/src/multi-region/manager.ts +483 -0
- package/src/multi-region/regions.ts +435 -0
- package/src/network-security/index.ts +48 -0
- package/src/observability/index.ts +9 -0
- package/src/observability/logs.ts +522 -0
- package/src/observability/metrics.ts +460 -0
- package/src/observability/observability.test.ts +782 -0
- package/src/observability/synthetics.ts +568 -0
- package/src/observability/xray.ts +358 -0
- package/src/phone/advanced/analytics.ts +349 -0
- package/src/phone/advanced/callbacks.ts +428 -0
- package/src/phone/advanced/index.ts +8 -0
- package/src/phone/advanced/ivr-builder.ts +504 -0
- package/src/phone/advanced/recording.ts +310 -0
- package/src/phone/handlers/__tests__/incoming-call.test.ts +40 -0
- package/src/phone/handlers/incoming-call.ts +117 -0
- package/src/phone/handlers/missed-call.ts +116 -0
- package/src/phone/handlers/voicemail.ts +179 -0
- package/src/phone/index.ts +9 -0
- package/src/presets/api-backend.ts +134 -0
- package/src/presets/data-pipeline.ts +204 -0
- package/src/presets/extend.test.ts +295 -0
- package/src/presets/extend.ts +297 -0
- package/src/presets/fullstack-app.ts +144 -0
- package/src/presets/index.ts +27 -0
- package/src/presets/jamstack.ts +135 -0
- package/src/presets/microservices.ts +167 -0
- package/src/presets/ml-api.ts +208 -0
- package/src/presets/nodejs-server.ts +104 -0
- package/src/presets/nodejs-serverless.ts +114 -0
- package/src/presets/realtime-app.ts +184 -0
- package/src/presets/static-site.ts +64 -0
- package/src/presets/traditional-web-app.ts +339 -0
- package/src/presets/wordpress.ts +138 -0
- package/src/preview/github.test.ts +249 -0
- package/src/preview/github.ts +297 -0
- package/src/preview/index.ts +37 -0
- package/src/preview/manager.test.ts +440 -0
- package/src/preview/manager.ts +326 -0
- package/src/preview/notifications.test.ts +582 -0
- package/src/preview/notifications.ts +341 -0
- package/src/queue/batch-processing.ts +402 -0
- package/src/queue/dlq-monitoring.ts +402 -0
- package/src/queue/fifo.ts +342 -0
- package/src/queue/index.ts +9 -0
- package/src/queue/management.ts +428 -0
- package/src/queue/queue.test.ts +429 -0
- package/src/resource-mgmt/index.ts +39 -0
- package/src/resource-naming.ts +62 -0
- package/src/s3/index.ts +523 -0
- package/src/schema/cloud-config.schema.json +554 -0
- package/src/schema/index.ts +68 -0
- package/src/security/certificate-manager.ts +492 -0
- package/src/security/index.ts +9 -0
- package/src/security/scanning.ts +545 -0
- package/src/security/secrets-manager.ts +476 -0
- package/src/security/secrets-rotation.ts +456 -0
- package/src/security/security.test.ts +738 -0
- package/src/sms/advanced/ab-testing.ts +389 -0
- package/src/sms/advanced/analytics.ts +336 -0
- package/src/sms/advanced/campaigns.ts +523 -0
- package/src/sms/advanced/chatbot.ts +224 -0
- package/src/sms/advanced/index.ts +10 -0
- package/src/sms/advanced/link-tracking.ts +248 -0
- package/src/sms/advanced/mms.ts +308 -0
- package/src/sms/handlers/__tests__/send.test.ts +40 -0
- package/src/sms/handlers/delivery-status.ts +133 -0
- package/src/sms/handlers/receive.ts +162 -0
- package/src/sms/handlers/send.ts +174 -0
- package/src/sms/index.ts +9 -0
- package/src/stack-diff.ts +389 -0
- package/src/static-site/index.ts +85 -0
- package/src/template-builder.ts +110 -0
- package/src/template-validator.ts +574 -0
- package/src/utils/cache.ts +291 -0
- package/src/utils/diff.ts +269 -0
- package/src/utils/hash.ts +227 -0
- package/src/utils/index.ts +8 -0
- package/src/utils/parallel.ts +294 -0
- package/src/validators/credentials.test.ts +274 -0
- package/src/validators/credentials.ts +233 -0
- package/src/validators/quotas.test.ts +434 -0
- package/src/validators/quotas.ts +217 -0
- package/test/ai.test.ts +327 -0
- package/test/api.test.ts +511 -0
- package/test/auth.test.ts +632 -0
- package/test/cache.test.ts +406 -0
- package/test/cdn.test.ts +247 -0
- package/test/compute.test.ts +861 -0
- package/test/database.test.ts +523 -0
- package/test/deployment.test.ts +499 -0
- package/test/dns.test.ts +270 -0
- package/test/email.test.ts +439 -0
- package/test/filesystem.test.ts +382 -0
- package/test/integration.test.ts +350 -0
- package/test/messaging.test.ts +514 -0
- package/test/monitoring.test.ts +634 -0
- package/test/network.test.ts +425 -0
- package/test/permissions.test.ts +488 -0
- package/test/queue.test.ts +484 -0
- package/test/registry.test.ts +306 -0
- package/test/security.test.ts +462 -0
- package/test/storage.test.ts +463 -0
- package/test/template-validator.test.ts +559 -0
- package/test/workflow.test.ts +592 -0
- package/tsconfig.json +16 -0
- package/tsconfig.tsbuildinfo +1 -0
package/src/cli/table.ts
ADDED
|
@@ -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
|
+
})
|