@stacksjs/ts-cloud 0.1.2 → 0.1.5

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 (187) hide show
  1. package/README.md +98 -13
  2. package/dist/aws/acm.d.ts +129 -0
  3. package/dist/aws/application-autoscaling.d.ts +282 -0
  4. package/dist/aws/bedrock.d.ts +2292 -0
  5. package/dist/aws/client.d.ts +79 -0
  6. package/dist/aws/cloudformation.d.ts +105 -0
  7. package/dist/aws/cloudfront.d.ts +265 -0
  8. package/dist/aws/cloudwatch-logs.d.ts +48 -0
  9. package/dist/aws/comprehend.d.ts +505 -0
  10. package/dist/aws/connect.d.ts +377 -0
  11. package/dist/aws/deploy-imap.d.ts +14 -0
  12. package/dist/aws/dynamodb.d.ts +176 -0
  13. package/dist/aws/ec2.d.ts +272 -0
  14. package/dist/aws/ecr.d.ts +149 -0
  15. package/dist/aws/ecs.d.ts +162 -0
  16. package/dist/aws/elasticache.d.ts +71 -0
  17. package/dist/aws/elbv2.d.ts +248 -0
  18. package/dist/aws/email.d.ts +175 -0
  19. package/dist/aws/eventbridge.d.ts +142 -0
  20. package/dist/aws/iam.d.ts +638 -0
  21. package/dist/aws/imap-server.d.ts +119 -0
  22. package/{src/aws/index.ts → dist/aws/index.d.ts} +62 -83
  23. package/{src/aws/kendra.ts → dist/aws/kendra.d.ts} +71 -386
  24. package/dist/aws/lambda.d.ts +232 -0
  25. package/dist/aws/opensearch.d.ts +87 -0
  26. package/dist/aws/personalize.d.ts +516 -0
  27. package/dist/aws/polly.d.ts +214 -0
  28. package/dist/aws/rds.d.ts +240 -0
  29. package/dist/aws/rekognition.d.ts +543 -0
  30. package/dist/aws/route53-domains.d.ts +113 -0
  31. package/dist/aws/route53.d.ts +215 -0
  32. package/dist/aws/s3.d.ts +212 -0
  33. package/dist/aws/scheduler.d.ts +140 -0
  34. package/dist/aws/secrets-manager.d.ts +170 -0
  35. package/dist/aws/ses.d.ts +288 -0
  36. package/dist/aws/setup-phone.d.ts +0 -0
  37. package/dist/aws/setup-sms.d.ts +115 -0
  38. package/dist/aws/sms.d.ts +304 -0
  39. package/dist/aws/smtp-server.d.ts +61 -0
  40. package/dist/aws/sns.d.ts +117 -0
  41. package/dist/aws/sqs.d.ts +65 -0
  42. package/dist/aws/ssm.d.ts +179 -0
  43. package/dist/aws/sts.d.ts +15 -0
  44. package/dist/aws/support.d.ts +104 -0
  45. package/dist/aws/test-imap.d.ts +0 -0
  46. package/dist/aws/textract.d.ts +403 -0
  47. package/dist/aws/transcribe.d.ts +60 -0
  48. package/dist/aws/translate.d.ts +358 -0
  49. package/dist/aws/voice.d.ts +219 -0
  50. package/dist/bin/cli.js +1724 -0
  51. package/dist/config.d.ts +7 -0
  52. package/dist/deploy/index.d.ts +2 -0
  53. package/dist/deploy/static-site-external-dns.d.ts +51 -0
  54. package/dist/deploy/static-site.d.ts +71 -0
  55. package/dist/dns/cloudflare.d.ts +52 -0
  56. package/dist/dns/godaddy.d.ts +38 -0
  57. package/dist/dns/index.d.ts +45 -0
  58. package/dist/dns/porkbun.d.ts +18 -0
  59. package/dist/dns/route53-adapter.d.ts +38 -0
  60. package/{src/dns/types.ts → dist/dns/types.d.ts} +26 -63
  61. package/dist/dns/validator.d.ts +78 -0
  62. package/dist/generators/index.d.ts +1 -0
  63. package/dist/generators/infrastructure.d.ts +30 -0
  64. package/{src/index.ts → dist/index.d.ts} +70 -93
  65. package/dist/index.js +7881 -0
  66. package/dist/push/apns.d.ts +60 -0
  67. package/dist/push/fcm.d.ts +117 -0
  68. package/dist/push/index.d.ts +14 -0
  69. package/dist/security/pre-deploy-scanner.d.ts +69 -0
  70. package/dist/ssl/acme-client.d.ts +67 -0
  71. package/dist/ssl/index.d.ts +2 -0
  72. package/dist/ssl/letsencrypt.d.ts +48 -0
  73. package/dist/types.d.ts +1 -0
  74. package/dist/utils/cli.d.ts +123 -0
  75. package/dist/validation/index.d.ts +1 -0
  76. package/dist/validation/template.d.ts +23 -0
  77. package/package.json +8 -8
  78. package/bin/cli.ts +0 -133
  79. package/bin/commands/analytics.ts +0 -328
  80. package/bin/commands/api.ts +0 -379
  81. package/bin/commands/assets.ts +0 -221
  82. package/bin/commands/audit.ts +0 -501
  83. package/bin/commands/backup.ts +0 -682
  84. package/bin/commands/cache.ts +0 -294
  85. package/bin/commands/cdn.ts +0 -281
  86. package/bin/commands/config.ts +0 -202
  87. package/bin/commands/container.ts +0 -105
  88. package/bin/commands/cost.ts +0 -208
  89. package/bin/commands/database.ts +0 -401
  90. package/bin/commands/deploy.ts +0 -674
  91. package/bin/commands/domain.ts +0 -397
  92. package/bin/commands/email.ts +0 -423
  93. package/bin/commands/environment.ts +0 -285
  94. package/bin/commands/events.ts +0 -424
  95. package/bin/commands/firewall.ts +0 -145
  96. package/bin/commands/function.ts +0 -116
  97. package/bin/commands/generate.ts +0 -280
  98. package/bin/commands/git.ts +0 -139
  99. package/bin/commands/iam.ts +0 -464
  100. package/bin/commands/index.ts +0 -48
  101. package/bin/commands/init.ts +0 -120
  102. package/bin/commands/logs.ts +0 -148
  103. package/bin/commands/network.ts +0 -579
  104. package/bin/commands/notify.ts +0 -489
  105. package/bin/commands/queue.ts +0 -407
  106. package/bin/commands/scheduler.ts +0 -370
  107. package/bin/commands/secrets.ts +0 -54
  108. package/bin/commands/server.ts +0 -629
  109. package/bin/commands/shared.ts +0 -97
  110. package/bin/commands/ssl.ts +0 -138
  111. package/bin/commands/stack.ts +0 -325
  112. package/bin/commands/status.ts +0 -385
  113. package/bin/commands/storage.ts +0 -450
  114. package/bin/commands/team.ts +0 -96
  115. package/bin/commands/tunnel.ts +0 -489
  116. package/bin/commands/utils.ts +0 -202
  117. package/build.ts +0 -15
  118. package/cloud +0 -2
  119. package/src/aws/acm.ts +0 -768
  120. package/src/aws/application-autoscaling.ts +0 -845
  121. package/src/aws/bedrock.ts +0 -4074
  122. package/src/aws/client.ts +0 -878
  123. package/src/aws/cloudformation.ts +0 -896
  124. package/src/aws/cloudfront.ts +0 -1531
  125. package/src/aws/cloudwatch-logs.ts +0 -154
  126. package/src/aws/comprehend.ts +0 -839
  127. package/src/aws/connect.ts +0 -1056
  128. package/src/aws/deploy-imap.ts +0 -384
  129. package/src/aws/dynamodb.ts +0 -340
  130. package/src/aws/ec2.ts +0 -1385
  131. package/src/aws/ecr.ts +0 -621
  132. package/src/aws/ecs.ts +0 -615
  133. package/src/aws/elasticache.ts +0 -301
  134. package/src/aws/elbv2.ts +0 -942
  135. package/src/aws/email.ts +0 -928
  136. package/src/aws/eventbridge.ts +0 -248
  137. package/src/aws/iam.ts +0 -1689
  138. package/src/aws/imap-server.ts +0 -2100
  139. package/src/aws/lambda.ts +0 -786
  140. package/src/aws/opensearch.ts +0 -158
  141. package/src/aws/personalize.ts +0 -977
  142. package/src/aws/polly.ts +0 -559
  143. package/src/aws/rds.ts +0 -888
  144. package/src/aws/rekognition.ts +0 -846
  145. package/src/aws/route53-domains.ts +0 -359
  146. package/src/aws/route53.ts +0 -1046
  147. package/src/aws/s3.ts +0 -2318
  148. package/src/aws/scheduler.ts +0 -571
  149. package/src/aws/secrets-manager.ts +0 -769
  150. package/src/aws/ses.ts +0 -1081
  151. package/src/aws/setup-phone.ts +0 -104
  152. package/src/aws/setup-sms.ts +0 -580
  153. package/src/aws/sms.ts +0 -1735
  154. package/src/aws/smtp-server.ts +0 -531
  155. package/src/aws/sns.ts +0 -758
  156. package/src/aws/sqs.ts +0 -382
  157. package/src/aws/ssm.ts +0 -807
  158. package/src/aws/sts.ts +0 -92
  159. package/src/aws/support.ts +0 -391
  160. package/src/aws/test-imap.ts +0 -86
  161. package/src/aws/textract.ts +0 -780
  162. package/src/aws/transcribe.ts +0 -108
  163. package/src/aws/translate.ts +0 -641
  164. package/src/aws/voice.ts +0 -1379
  165. package/src/config.ts +0 -35
  166. package/src/deploy/index.ts +0 -7
  167. package/src/deploy/static-site-external-dns.ts +0 -906
  168. package/src/deploy/static-site.ts +0 -1125
  169. package/src/dns/godaddy.ts +0 -412
  170. package/src/dns/index.ts +0 -183
  171. package/src/dns/porkbun.ts +0 -362
  172. package/src/dns/route53-adapter.ts +0 -414
  173. package/src/dns/validator.ts +0 -369
  174. package/src/generators/index.ts +0 -5
  175. package/src/generators/infrastructure.ts +0 -1660
  176. package/src/push/apns.ts +0 -452
  177. package/src/push/fcm.ts +0 -506
  178. package/src/push/index.ts +0 -58
  179. package/src/ssl/acme-client.ts +0 -478
  180. package/src/ssl/index.ts +0 -7
  181. package/src/ssl/letsencrypt.ts +0 -747
  182. package/src/types.ts +0 -2
  183. package/src/utils/cli.ts +0 -398
  184. package/src/validation/index.ts +0 -5
  185. package/src/validation/template.ts +0 -405
  186. package/test/index.test.ts +0 -128
  187. package/tsconfig.json +0 -18
@@ -1,384 +0,0 @@
1
- #!/usr/bin/env bun
2
- /**
3
- * Deploy IMAP-to-S3 bridge server to EC2 via SSM
4
- * Embeds all code directly in SSM commands to avoid S3 permission issues
5
- *
6
- * Security:
7
- * - Uses EC2 instance IAM role for AWS credentials (no hardcoded keys)
8
- * - Fetches IMAP passwords from AWS Secrets Manager at startup
9
- * - Secret name: stacks/mail-server/credentials
10
- * - Credentials are read from email config and synced to Secrets Manager
11
- */
12
-
13
- import { SSMClient } from './ssm'
14
- import { AWSClient } from './client'
15
- import * as fs from 'node:fs'
16
- import * as path from 'node:path'
17
-
18
- export interface MailboxConfig {
19
- email: string
20
- password?: string
21
- }
22
-
23
- export interface MailServerDeployConfig {
24
- instanceId: string
25
- region: string
26
- secretName: string
27
- domain: string
28
- bucket: string
29
- prefix: string
30
- /** Mailboxes can be simple strings or objects with optional passwords */
31
- mailboxes: Array<string | MailboxConfig>
32
- }
33
-
34
- /**
35
- * Normalize mailbox config to object format with password lookup
36
- * Supports:
37
- * - Simple usernames: 'chris' -> 'chris@{domain}', looks up MAIL_PASSWORD_CHRIS
38
- * - Full email strings: 'chris@stacksjs.com' -> looks up MAIL_PASSWORD_CHRIS
39
- * - Objects with email: { email: 'chris@stacksjs.com', password: '...' }
40
- * - Objects with address (deprecated): { address: 'chris@stacksjs.com' }
41
- */
42
- function normalizeMailbox(mailbox: string | MailboxConfig | { address: string, password?: string }, domain: string): MailboxConfig {
43
- if (typeof mailbox === 'string') {
44
- // If it's just a username (no @), append the domain
45
- const email = mailbox.includes('@') ? mailbox : `${mailbox}@${domain}`
46
- const username = email.split('@')[0].toUpperCase()
47
- const envKey = `MAIL_PASSWORD_${username}`
48
- const password = process.env[envKey]
49
- return { email, password }
50
- }
51
-
52
- // Handle both 'email' and 'address' fields (address is deprecated)
53
- let email = 'email' in mailbox ? mailbox.email : (mailbox as { address: string }).address
54
- if (!email) {
55
- throw new Error('Mailbox must have either "email" or "address" field')
56
- }
57
-
58
- // If it's just a username (no @), append the domain
59
- if (!email.includes('@')) {
60
- email = `${email}@${domain}`
61
- }
62
-
63
- // If object format but no password, try env lookup
64
- if (!mailbox.password) {
65
- const username = email.split('@')[0].toUpperCase()
66
- const envKey = `MAIL_PASSWORD_${username}`
67
- const password = process.env[envKey]
68
- return { email, password }
69
- }
70
- return { email, password: mailbox.password }
71
- }
72
-
73
- const defaultConfig: MailServerDeployConfig = {
74
- instanceId: 'i-032233d3e9839b78b',
75
- region: 'us-east-1',
76
- secretName: 'stacks/mail-server/credentials',
77
- domain: 'stacksjs.com',
78
- bucket: 'stacks-production-email',
79
- prefix: 'incoming/',
80
- mailboxes: [],
81
- }
82
-
83
- export async function deployImapServer(config: MailServerDeployConfig = defaultConfig): Promise<void> {
84
- console.log('Deploying IMAP-to-S3 bridge server to EC2...')
85
- console.log('')
86
-
87
- const ssm = new SSMClient(config.region)
88
- const awsClient = new AWSClient()
89
-
90
- // Normalize all mailboxes to object format with password lookup
91
- const normalizedMailboxes = config.mailboxes.map((m) => normalizeMailbox(m, config.domain))
92
-
93
- // Build credentials from normalized mailboxes
94
- const credentials: Record<string, string> = {}
95
- for (const mailbox of normalizedMailboxes) {
96
- // Extract username from email (chris@stacksjs.com -> chris)
97
- const username = mailbox.email.split('@')[0]
98
- if (mailbox.password) {
99
- credentials[username] = mailbox.password
100
- }
101
- }
102
-
103
- // Ensure the secret exists with IMAP credentials from config
104
- console.log('0. Ensuring credentials secret exists in Secrets Manager...')
105
- try {
106
- const existingSecretResult = await awsClient.request({
107
- service: 'secretsmanager',
108
- region: config.region,
109
- method: 'POST',
110
- path: '/',
111
- headers: {
112
- 'Content-Type': 'application/x-amz-json-1.1',
113
- 'X-Amz-Target': 'secretsmanager.GetSecretValue',
114
- },
115
- body: JSON.stringify({ SecretId: config.secretName }),
116
- })
117
- const existingSecret = existingSecretResult.SecretString || '{}'
118
- console.log(' Secret already exists')
119
-
120
- // Update if we have new credentials from config
121
- if (Object.keys(credentials).length > 0) {
122
- const existingCreds = JSON.parse(existingSecret)
123
- const mergedCreds = { ...existingCreds, ...credentials }
124
- await awsClient.request({
125
- service: 'secretsmanager',
126
- region: config.region,
127
- method: 'POST',
128
- path: '/',
129
- headers: {
130
- 'Content-Type': 'application/x-amz-json-1.1',
131
- 'X-Amz-Target': 'secretsmanager.PutSecretValue',
132
- },
133
- body: JSON.stringify({
134
- SecretId: config.secretName,
135
- SecretString: JSON.stringify(mergedCreds),
136
- }),
137
- })
138
- console.log(' Secret updated with config credentials')
139
- }
140
- }
141
- catch {
142
- // Create the secret with credentials from config
143
- console.log(' Creating secret...')
144
- if (Object.keys(credentials).length === 0) {
145
- console.warn(' WARNING: No passwords configured in mailboxes - logins will fail')
146
- console.warn(' Set MAIL_PASSWORD_<USER> environment variables in your config')
147
- }
148
- await awsClient.request({
149
- service: 'secretsmanager',
150
- region: config.region,
151
- method: 'POST',
152
- path: '/',
153
- headers: {
154
- 'Content-Type': 'application/x-amz-json-1.1',
155
- 'X-Amz-Target': 'secretsmanager.CreateSecret',
156
- },
157
- body: JSON.stringify({
158
- Name: config.secretName,
159
- Description: `IMAP mail server credentials for ${config.domain}`,
160
- SecretString: JSON.stringify(credentials),
161
- ClientRequestToken: crypto.randomUUID(),
162
- }),
163
- })
164
- console.log(' Secret created')
165
- }
166
-
167
- // Read the source files
168
- const imapServerCode = fs.readFileSync(path.join(__dirname, 'imap-server.ts'), 'utf-8')
169
- const s3ClientCode = fs.readFileSync(path.join(__dirname, 's3.ts'), 'utf-8')
170
- const clientCode = fs.readFileSync(path.join(__dirname, 'client.ts'), 'utf-8')
171
-
172
- // Build users config for server script from normalized mailboxes
173
- const usersConfig = normalizedMailboxes.map((m) => {
174
- const username = m.email.split('@')[0]
175
- return ` ${username}: {
176
- password: passwords.${username} || 'changeme',
177
- email: '${m.email}',
178
- },`
179
- }).join('\n')
180
-
181
- // Create the server startup script - fetches credentials from Secrets Manager using AWSClient directly
182
- const serverScript = `#!/usr/bin/env bun
183
- import * as fs from 'node:fs'
184
- import { startImapServer } from './imap-server'
185
- import { AWSClient } from './client'
186
-
187
- const SECRET_NAME = '${config.secretName}'
188
- const REGION = '${config.region}'
189
-
190
- async function main() {
191
- console.log('Starting IMAP-to-S3 bridge server...')
192
-
193
- // Fetch credentials from Secrets Manager using AWSClient directly (uses EC2 instance IAM role)
194
- console.log('Fetching credentials from Secrets Manager...')
195
- const client = new AWSClient()
196
- let passwords: Record<string, string> = {}
197
-
198
- try {
199
- const result = await client.request({
200
- service: 'secretsmanager',
201
- region: REGION,
202
- method: 'POST',
203
- path: '/',
204
- headers: {
205
- 'Content-Type': 'application/x-amz-json-1.1',
206
- 'X-Amz-Target': 'secretsmanager.GetSecretValue'
207
- },
208
- body: JSON.stringify({ SecretId: SECRET_NAME })
209
- })
210
- passwords = JSON.parse(result.SecretString || '{}')
211
- console.log('Credentials loaded for:', Object.keys(passwords).join(', '))
212
- }
213
- catch (error) {
214
- console.error('Failed to fetch credentials from Secrets Manager:', error)
215
- console.error('Using fallback empty passwords - logins will fail')
216
- }
217
-
218
- const hasTlsCerts = fs.existsSync('/etc/letsencrypt/live/mail.${config.domain}/privkey.pem')
219
- console.log('TLS certificates available:', hasTlsCerts)
220
-
221
- const server = await startImapServer({
222
- port: 143,
223
- sslPort: 993,
224
- host: '0.0.0.0',
225
- region: REGION,
226
- bucket: '${config.bucket}',
227
- prefix: '${config.prefix}',
228
- domain: '${config.domain}',
229
- users: {
230
- ${usersConfig}
231
- },
232
- tls: hasTlsCerts ? {
233
- key: '/etc/letsencrypt/live/mail.${config.domain}/privkey.pem',
234
- cert: '/etc/letsencrypt/live/mail.${config.domain}/fullchain.pem',
235
- } : undefined,
236
- })
237
-
238
- console.log('IMAP server running on port 143' + (hasTlsCerts ? ' and 993 (TLS)' : ''))
239
-
240
- process.on('SIGINT', async () => {
241
- console.log('Shutting down...')
242
- await server.stop()
243
- process.exit(0)
244
- })
245
-
246
- process.on('SIGTERM', async () => {
247
- console.log('Shutting down...')
248
- await server.stop()
249
- process.exit(0)
250
- })
251
- }
252
-
253
- main().catch(console.error)
254
- `
255
-
256
- // Create systemd service file - NO hardcoded credentials, uses IAM role
257
- const systemdService = `[Unit]
258
- Description=IMAP-to-S3 Bridge Server
259
- After=network.target
260
-
261
- [Service]
262
- Type=simple
263
- User=root
264
- WorkingDirectory=/opt/imap-server
265
- # AWS credentials come from EC2 instance IAM role - no hardcoded keys needed
266
- Environment="AWS_REGION=${config.region}"
267
- ExecStart=/root/.bun/bin/bun run /opt/imap-server/server.ts
268
- Restart=always
269
- RestartSec=10
270
-
271
- [Install]
272
- WantedBy=multi-user.target
273
- `
274
-
275
- // Step 1: Create directory structure
276
- console.log('1. Creating directory structure on EC2...')
277
- let result = await ssm.runShellCommand(config.instanceId, [
278
- 'mkdir -p /opt/imap-server',
279
- 'ls -la /opt/imap-server',
280
- ], { maxWaitMs: 60000 })
281
-
282
- if (!result.success) {
283
- console.error('Failed to create directory:', result.error)
284
- process.exit(1)
285
- }
286
- console.log(' Directory created')
287
-
288
- // Step 2: Write client.ts (base64 encode to handle special chars)
289
- console.log('2. Writing client.ts...')
290
- const clientBase64 = Buffer.from(clientCode).toString('base64')
291
- result = await ssm.runShellCommand(config.instanceId, [
292
- `echo '${clientBase64}' | base64 -d > /opt/imap-server/client.ts`,
293
- 'wc -l /opt/imap-server/client.ts',
294
- ], { maxWaitMs: 60000 })
295
-
296
- if (!result.success) {
297
- console.error('Failed to write client.ts:', result.error)
298
- process.exit(1)
299
- }
300
- console.log(' client.ts written')
301
-
302
- // Step 3: Write s3.ts
303
- console.log('3. Writing s3.ts...')
304
- const s3Base64 = Buffer.from(s3ClientCode).toString('base64')
305
- result = await ssm.runShellCommand(config.instanceId, [
306
- `echo '${s3Base64}' | base64 -d > /opt/imap-server/s3.ts`,
307
- 'wc -l /opt/imap-server/s3.ts',
308
- ], { maxWaitMs: 60000 })
309
-
310
- if (!result.success) {
311
- console.error('Failed to write s3.ts:', result.error)
312
- process.exit(1)
313
- }
314
- console.log(' s3.ts written')
315
-
316
- // Step 4: Write imap-server.ts
317
- console.log('4. Writing imap-server.ts...')
318
- const imapBase64 = Buffer.from(imapServerCode).toString('base64')
319
- result = await ssm.runShellCommand(config.instanceId, [
320
- `echo '${imapBase64}' | base64 -d > /opt/imap-server/imap-server.ts`,
321
- 'wc -l /opt/imap-server/imap-server.ts',
322
- ], { maxWaitMs: 60000 })
323
-
324
- if (!result.success) {
325
- console.error('Failed to write imap-server.ts:', result.error)
326
- process.exit(1)
327
- }
328
- console.log(' imap-server.ts written')
329
-
330
- // Step 5: Write server.ts
331
- console.log('5. Writing server.ts...')
332
- const serverBase64 = Buffer.from(serverScript).toString('base64')
333
- result = await ssm.runShellCommand(config.instanceId, [
334
- `echo '${serverBase64}' | base64 -d > /opt/imap-server/server.ts`,
335
- 'wc -l /opt/imap-server/server.ts',
336
- ], { maxWaitMs: 60000 })
337
-
338
- if (!result.success) {
339
- console.error('Failed to write server.ts:', result.error)
340
- process.exit(1)
341
- }
342
- console.log(' server.ts written')
343
-
344
- // Step 6: Write systemd service and start
345
- console.log('6. Setting up systemd service...')
346
- const serviceBase64 = Buffer.from(systemdService).toString('base64')
347
- result = await ssm.runShellCommand(config.instanceId, [
348
- `echo '${serviceBase64}' | base64 -d > /etc/systemd/system/imap-server.service`,
349
- 'systemctl daemon-reload',
350
- 'systemctl stop imap-server 2>/dev/null || true',
351
- 'systemctl enable imap-server',
352
- 'systemctl start imap-server',
353
- 'sleep 3',
354
- 'systemctl status imap-server --no-pager || true',
355
- 'ss -tlnp | grep -E ":143|:993" || netstat -tlnp | grep -E ":143|:993" || echo "Ports not yet listening"',
356
- ], { maxWaitMs: 120000 })
357
-
358
- console.log('')
359
- console.log('Service status:')
360
- console.log(result.output || result.error)
361
-
362
- console.log('')
363
- console.log('='.repeat(60))
364
- console.log('IMAP Server Deployment Complete!')
365
- console.log('='.repeat(60))
366
- console.log('')
367
- console.log('Mail.app Settings:')
368
- console.log(' Account Type: IMAP')
369
- console.log(` Incoming Server: mail.${config.domain}`)
370
- console.log(' Port: 143 (or 993 with SSL)')
371
- console.log(' Username: <email username>')
372
- console.log(' Password: <from Secrets Manager>')
373
- console.log('')
374
- console.log('Credentials are stored in AWS Secrets Manager:')
375
- console.log(` Secret: ${config.secretName}`)
376
- console.log('')
377
- }
378
-
379
- // Main entry point for standalone execution
380
- async function main() {
381
- await deployImapServer(defaultConfig)
382
- }
383
-
384
- main().catch(console.error)