@geekmidas/cli 1.5.0 → 1.6.0

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 (119) hide show
  1. package/CHANGELOG.md +17 -0
  2. package/dist/{HostingerProvider-B9N-TKbp.mjs → HostingerProvider-402UdK89.mjs} +34 -1
  3. package/dist/HostingerProvider-402UdK89.mjs.map +1 -0
  4. package/dist/{HostingerProvider-DUV9-Tzg.cjs → HostingerProvider-BiXdHjiq.cjs} +34 -1
  5. package/dist/HostingerProvider-BiXdHjiq.cjs.map +1 -0
  6. package/dist/{Route53Provider-C8mS0zY6.mjs → Route53Provider-DbBo7Uz5.mjs} +53 -1
  7. package/dist/Route53Provider-DbBo7Uz5.mjs.map +1 -0
  8. package/dist/{Route53Provider-Bs7Arms9.cjs → Route53Provider-kfJ77LmL.cjs} +53 -1
  9. package/dist/Route53Provider-kfJ77LmL.cjs.map +1 -0
  10. package/dist/backup-provisioner-B5e-F6zX.cjs +164 -0
  11. package/dist/backup-provisioner-B5e-F6zX.cjs.map +1 -0
  12. package/dist/backup-provisioner-BIArpmTr.mjs +163 -0
  13. package/dist/backup-provisioner-BIArpmTr.mjs.map +1 -0
  14. package/dist/{config-ZQM1vBoz.cjs → config-6JHOwLCx.cjs} +30 -2
  15. package/dist/{config-ZQM1vBoz.cjs.map → config-6JHOwLCx.cjs.map} +1 -1
  16. package/dist/{config-DfCJ29PQ.mjs → config-DxASSNjr.mjs} +25 -3
  17. package/dist/{config-DfCJ29PQ.mjs.map → config-DxASSNjr.mjs.map} +1 -1
  18. package/dist/config.cjs +3 -2
  19. package/dist/config.d.cts +14 -2
  20. package/dist/config.d.cts.map +1 -1
  21. package/dist/config.d.mts +15 -3
  22. package/dist/config.d.mts.map +1 -1
  23. package/dist/config.mjs +3 -3
  24. package/dist/{dokploy-api-z0833e7r.mjs → dokploy-api-2ldYoN3i.mjs} +131 -1
  25. package/dist/dokploy-api-2ldYoN3i.mjs.map +1 -0
  26. package/dist/dokploy-api-C93pveuy.mjs +3 -0
  27. package/dist/dokploy-api-CbDh4o93.cjs +3 -0
  28. package/dist/{dokploy-api-CQvhV6Hd.cjs → dokploy-api-DLgvEQlr.cjs} +131 -1
  29. package/dist/dokploy-api-DLgvEQlr.cjs.map +1 -0
  30. package/dist/{index-C0SpUT9Y.d.mts → index-C-KxSGGK.d.mts} +133 -31
  31. package/dist/index-C-KxSGGK.d.mts.map +1 -0
  32. package/dist/{index-B58qjyBd.d.cts → index-Cyk2rTyj.d.cts} +132 -30
  33. package/dist/index-Cyk2rTyj.d.cts.map +1 -0
  34. package/dist/index.cjs +662 -152
  35. package/dist/index.cjs.map +1 -1
  36. package/dist/index.mjs +626 -116
  37. package/dist/index.mjs.map +1 -1
  38. package/dist/{openapi-BcSjLfWq.mjs → openapi-BYlyAbH3.mjs} +6 -5
  39. package/dist/openapi-BYlyAbH3.mjs.map +1 -0
  40. package/dist/{openapi-D6Hcfov0.cjs → openapi-CnvwSRDU.cjs} +6 -5
  41. package/dist/openapi-CnvwSRDU.cjs.map +1 -0
  42. package/dist/openapi.cjs +3 -3
  43. package/dist/openapi.d.cts +1 -0
  44. package/dist/openapi.d.cts.map +1 -1
  45. package/dist/openapi.d.mts +2 -1
  46. package/dist/openapi.d.mts.map +1 -1
  47. package/dist/openapi.mjs +3 -3
  48. package/dist/{types-B9UZ7fOG.d.mts → types-CZg5iUgD.d.mts} +1 -1
  49. package/dist/{types-B9UZ7fOG.d.mts.map → types-CZg5iUgD.d.mts.map} +1 -1
  50. package/dist/workspace/index.cjs +1 -1
  51. package/dist/workspace/index.d.cts +1 -1
  52. package/dist/workspace/index.d.mts +2 -2
  53. package/dist/workspace/index.mjs +1 -1
  54. package/dist/{workspace-BW2iU37P.mjs → workspace-9IQIjwkQ.mjs} +20 -4
  55. package/dist/workspace-9IQIjwkQ.mjs.map +1 -0
  56. package/dist/{workspace-2Do2YcGZ.cjs → workspace-D2ocAlpl.cjs} +20 -4
  57. package/dist/workspace-D2ocAlpl.cjs.map +1 -0
  58. package/examples/cron-example.ts +6 -6
  59. package/examples/function-example.ts +1 -1
  60. package/package.json +6 -3
  61. package/src/config.ts +44 -0
  62. package/src/deploy/__tests__/backup-provisioner.spec.ts +428 -0
  63. package/src/deploy/__tests__/createDnsProvider.spec.ts +23 -0
  64. package/src/deploy/__tests__/env-resolver.spec.ts +1 -1
  65. package/src/deploy/__tests__/undeploy.spec.ts +758 -0
  66. package/src/deploy/backup-provisioner.ts +316 -0
  67. package/src/deploy/dns/DnsProvider.ts +39 -1
  68. package/src/deploy/dns/HostingerProvider.ts +74 -0
  69. package/src/deploy/dns/Route53Provider.ts +81 -0
  70. package/src/deploy/dns/index.ts +25 -0
  71. package/src/deploy/dokploy-api.ts +237 -0
  72. package/src/deploy/index.ts +71 -13
  73. package/src/deploy/state.ts +171 -0
  74. package/src/deploy/undeploy.ts +407 -0
  75. package/src/dev/__tests__/index.spec.ts +490 -0
  76. package/src/dev/index.ts +313 -18
  77. package/src/generators/FunctionGenerator.ts +1 -1
  78. package/src/generators/Generator.ts +4 -1
  79. package/src/init/__tests__/generators.spec.ts +167 -18
  80. package/src/init/__tests__/init.spec.ts +66 -3
  81. package/src/init/generators/auth.ts +6 -5
  82. package/src/init/generators/config.ts +49 -7
  83. package/src/init/generators/docker.ts +8 -8
  84. package/src/init/generators/index.ts +1 -0
  85. package/src/init/generators/models.ts +3 -5
  86. package/src/init/generators/package.ts +4 -0
  87. package/src/init/generators/test.ts +133 -0
  88. package/src/init/generators/ui.ts +13 -12
  89. package/src/init/generators/web.ts +9 -8
  90. package/src/init/index.ts +2 -0
  91. package/src/init/templates/api.ts +6 -6
  92. package/src/init/templates/minimal.ts +2 -2
  93. package/src/init/templates/worker.ts +2 -2
  94. package/src/init/versions.ts +3 -3
  95. package/src/openapi.ts +6 -2
  96. package/src/test/__tests__/__fixtures__/workspace.ts +104 -0
  97. package/src/test/__tests__/api.spec.ts +199 -0
  98. package/src/test/__tests__/auth.spec.ts +162 -0
  99. package/src/test/__tests__/index.spec.ts +323 -0
  100. package/src/test/__tests__/web.spec.ts +210 -0
  101. package/src/test/index.ts +165 -14
  102. package/src/workspace/__tests__/index.spec.ts +3 -0
  103. package/src/workspace/index.ts +4 -2
  104. package/src/workspace/schema.ts +26 -0
  105. package/src/workspace/types.ts +14 -37
  106. package/dist/HostingerProvider-B9N-TKbp.mjs.map +0 -1
  107. package/dist/HostingerProvider-DUV9-Tzg.cjs.map +0 -1
  108. package/dist/Route53Provider-Bs7Arms9.cjs.map +0 -1
  109. package/dist/Route53Provider-C8mS0zY6.mjs.map +0 -1
  110. package/dist/dokploy-api-CQvhV6Hd.cjs.map +0 -1
  111. package/dist/dokploy-api-CWc02yyg.cjs +0 -3
  112. package/dist/dokploy-api-DSJYNx88.mjs +0 -3
  113. package/dist/dokploy-api-z0833e7r.mjs.map +0 -1
  114. package/dist/index-B58qjyBd.d.cts.map +0 -1
  115. package/dist/index-C0SpUT9Y.d.mts.map +0 -1
  116. package/dist/openapi-BcSjLfWq.mjs.map +0 -1
  117. package/dist/openapi-D6Hcfov0.cjs.map +0 -1
  118. package/dist/workspace-2Do2YcGZ.cjs.map +0 -1
  119. package/dist/workspace-BW2iU37P.mjs.map +0 -1
@@ -0,0 +1,407 @@
1
+ /**
2
+ * Undeploy - Remove deployed resources from Dokploy
3
+ *
4
+ * Deletes applications, services (postgres, redis), DNS records, and optionally the project.
5
+ * Also handles cleanup of backup resources if configured.
6
+ */
7
+
8
+ import type { DnsProvider } from './dns/DnsProvider.js';
9
+ import type { DokployApi } from './dokploy-api.js';
10
+ import type { BackupState, DokployStageState } from './state.js';
11
+ import { getAllDnsRecords } from './state.js';
12
+
13
+ export interface UndeployOptions {
14
+ /** Dokploy API client */
15
+ api: DokployApi;
16
+ /** Deploy state for the stage */
17
+ state: DokployStageState;
18
+ /** DNS provider for deleting DNS records (optional) */
19
+ dnsProvider?: DnsProvider;
20
+ /** Whether to delete the Dokploy project (default: false) */
21
+ deleteProject?: boolean;
22
+ /** Whether to delete backup resources (S3 bucket, IAM user) - default: false */
23
+ deleteBackups?: boolean;
24
+ /** AWS endpoint override (for testing with LocalStack) */
25
+ awsEndpoint?: string;
26
+ /** Logger for progress output */
27
+ logger: { log: (msg: string) => void };
28
+ }
29
+
30
+ export interface UndeployResult {
31
+ /** Applications that were deleted */
32
+ deletedApplications: string[];
33
+ /** Whether postgres was deleted */
34
+ deletedPostgres: boolean;
35
+ /** Whether redis was deleted */
36
+ deletedRedis: boolean;
37
+ /** Whether the project was deleted */
38
+ deletedProject: boolean;
39
+ /** Whether backup destination was deleted */
40
+ deletedBackupDestination: boolean;
41
+ /** Whether AWS backup resources were deleted */
42
+ deletedAwsBackupResources: boolean;
43
+ /** DNS records that were deleted (name:type format) */
44
+ deletedDnsRecords: string[];
45
+ /** Updated state after undeploy (with deleted resources removed) */
46
+ updatedState: DokployStageState;
47
+ /** Errors encountered during undeploy (non-fatal) */
48
+ errors: string[];
49
+ }
50
+
51
+ /**
52
+ * Undeploy resources from Dokploy
53
+ *
54
+ * Executes in order:
55
+ * 1. Run final backup (if backup is configured)
56
+ * 2. Delete DNS records (if DNS provider is available)
57
+ * 3. Delete backup schedules
58
+ * 4. Delete applications
59
+ * 5. Delete postgres database
60
+ * 6. Delete redis instance
61
+ * 7. Delete backup destination
62
+ * 8. Delete AWS backup resources (if deleteBackups is true)
63
+ * 9. Delete project (if deleteProject is true)
64
+ *
65
+ * Returns the updated state with deleted resources removed.
66
+ */
67
+ export async function undeploy(
68
+ options: UndeployOptions,
69
+ ): Promise<UndeployResult> {
70
+ const {
71
+ api,
72
+ state,
73
+ dnsProvider,
74
+ deleteProject = false,
75
+ deleteBackups = false,
76
+ awsEndpoint,
77
+ logger,
78
+ } = options;
79
+
80
+ // Create a mutable copy of the state to track deletions
81
+ const updatedState: DokployStageState = JSON.parse(JSON.stringify(state));
82
+
83
+ const result: UndeployResult = {
84
+ deletedApplications: [],
85
+ deletedPostgres: false,
86
+ deletedRedis: false,
87
+ deletedProject: false,
88
+ deletedBackupDestination: false,
89
+ deletedAwsBackupResources: false,
90
+ deletedDnsRecords: [],
91
+ updatedState,
92
+ errors: [],
93
+ };
94
+
95
+ // 1. Run a final backup before undeploying (if backup is configured)
96
+ if (state.backups?.postgresBackupId) {
97
+ try {
98
+ logger.log(' Running final postgres backup before undeploy...');
99
+ await api.runBackupManually(state.backups.postgresBackupId);
100
+ logger.log(' ✓ Final backup triggered');
101
+ } catch (error) {
102
+ const msg = `Failed to run final backup: ${error}`;
103
+ logger.log(` ⚠ ${msg}`);
104
+ result.errors.push(msg);
105
+ }
106
+ }
107
+
108
+ // 2. Delete DNS records (if DNS provider is available)
109
+ if (dnsProvider) {
110
+ const dnsRecords = getAllDnsRecords(state);
111
+ if (dnsRecords.length > 0) {
112
+ // Group records by domain
113
+ const recordsByDomain = new Map<
114
+ string,
115
+ Array<{ name: string; type: string }>
116
+ >();
117
+ for (const record of dnsRecords) {
118
+ const existing = recordsByDomain.get(record.domain) ?? [];
119
+ existing.push({ name: record.name, type: record.type });
120
+ recordsByDomain.set(record.domain, existing);
121
+ }
122
+
123
+ for (const [domain, records] of recordsByDomain) {
124
+ try {
125
+ logger.log(
126
+ ` Deleting ${records.length} DNS record(s) for ${domain}...`,
127
+ );
128
+ const deleteResults = await dnsProvider.deleteRecords(
129
+ domain,
130
+ records.map((r) => ({
131
+ name: r.name,
132
+ type: r.type as
133
+ | 'A'
134
+ | 'AAAA'
135
+ | 'CNAME'
136
+ | 'MX'
137
+ | 'TXT'
138
+ | 'SRV'
139
+ | 'CAA',
140
+ })),
141
+ );
142
+
143
+ for (const deleteResult of deleteResults) {
144
+ const key = `${deleteResult.record.name}:${deleteResult.record.type}`;
145
+ if (deleteResult.deleted || deleteResult.notFound) {
146
+ result.deletedDnsRecords.push(key);
147
+ // Remove from state
148
+ if (updatedState.dnsRecords) {
149
+ delete updatedState.dnsRecords[key];
150
+ }
151
+ if (updatedState.dnsVerified) {
152
+ // Find and remove hostname from dnsVerified
153
+ const hostname =
154
+ deleteResult.record.name === '@'
155
+ ? domain
156
+ : `${deleteResult.record.name}.${domain}`;
157
+ delete updatedState.dnsVerified[hostname];
158
+ }
159
+ logger.log(` ✓ DNS record ${key} deleted`);
160
+ } else if (deleteResult.error) {
161
+ const msg = `Failed to delete DNS record ${key}: ${deleteResult.error}`;
162
+ logger.log(` ⚠ ${msg}`);
163
+ result.errors.push(msg);
164
+ }
165
+ }
166
+ } catch (error) {
167
+ const msg = `Failed to delete DNS records for ${domain}: ${error}`;
168
+ logger.log(` ⚠ ${msg}`);
169
+ result.errors.push(msg);
170
+ }
171
+ }
172
+ }
173
+ }
174
+
175
+ // 3. Delete backup schedules (before deleting postgres)
176
+ if (state.backups?.postgresBackupId) {
177
+ try {
178
+ logger.log(' Deleting postgres backup schedule...');
179
+ await api.deleteBackup(state.backups.postgresBackupId);
180
+ if (updatedState.backups) {
181
+ delete updatedState.backups.postgresBackupId;
182
+ }
183
+ logger.log(' ✓ Backup schedule deleted');
184
+ } catch (error) {
185
+ const msg = `Failed to delete backup schedule: ${error}`;
186
+ logger.log(` ⚠ ${msg}`);
187
+ result.errors.push(msg);
188
+ }
189
+ }
190
+
191
+ // 4. Delete all applications
192
+ for (const [appName, applicationId] of Object.entries(state.applications)) {
193
+ try {
194
+ logger.log(` Deleting application: ${appName}...`);
195
+ await api.deleteApplication(applicationId);
196
+ result.deletedApplications.push(appName);
197
+ delete updatedState.applications[appName];
198
+ // Also remove app credentials and generated secrets
199
+ if (updatedState.appCredentials) {
200
+ delete updatedState.appCredentials[appName];
201
+ }
202
+ if (updatedState.generatedSecrets) {
203
+ delete updatedState.generatedSecrets[appName];
204
+ }
205
+ logger.log(` ✓ Application ${appName} deleted`);
206
+ } catch (error) {
207
+ const msg = `Failed to delete application ${appName}: ${error}`;
208
+ logger.log(` ⚠ ${msg}`);
209
+ result.errors.push(msg);
210
+ }
211
+ }
212
+
213
+ // 5. Delete postgres if exists
214
+ if (state.services.postgresId) {
215
+ try {
216
+ logger.log(' Deleting postgres database...');
217
+ await api.deletePostgres(state.services.postgresId);
218
+ result.deletedPostgres = true;
219
+ delete updatedState.services.postgresId;
220
+ logger.log(' ✓ Postgres deleted');
221
+ } catch (error) {
222
+ const msg = `Failed to delete postgres: ${error}`;
223
+ logger.log(` ⚠ ${msg}`);
224
+ result.errors.push(msg);
225
+ }
226
+ }
227
+
228
+ // 6. Delete redis if exists
229
+ if (state.services.redisId) {
230
+ try {
231
+ logger.log(' Deleting redis instance...');
232
+ await api.deleteRedis(state.services.redisId);
233
+ result.deletedRedis = true;
234
+ delete updatedState.services.redisId;
235
+ logger.log(' ✓ Redis deleted');
236
+ } catch (error) {
237
+ const msg = `Failed to delete redis: ${error}`;
238
+ logger.log(` ⚠ ${msg}`);
239
+ result.errors.push(msg);
240
+ }
241
+ }
242
+
243
+ // 7. Delete backup destination from Dokploy
244
+ if (state.backups?.destinationId) {
245
+ try {
246
+ logger.log(' Deleting backup destination...');
247
+ await api.deleteDestination(state.backups.destinationId);
248
+ result.deletedBackupDestination = true;
249
+ logger.log(' ✓ Backup destination deleted');
250
+ } catch (error) {
251
+ const msg = `Failed to delete backup destination: ${error}`;
252
+ logger.log(` ⚠ ${msg}`);
253
+ result.errors.push(msg);
254
+ }
255
+ }
256
+
257
+ // 8. Delete AWS backup resources if requested
258
+ if (deleteBackups && state.backups) {
259
+ try {
260
+ logger.log(' Deleting AWS backup resources...');
261
+ await deleteAwsBackupResources(state.backups, awsEndpoint, logger);
262
+ result.deletedAwsBackupResources = true;
263
+ // Clear backup state entirely
264
+ delete updatedState.backups;
265
+ logger.log(' ✓ AWS backup resources deleted');
266
+ } catch (error) {
267
+ const msg = `Failed to delete AWS backup resources: ${error}`;
268
+ logger.log(` ⚠ ${msg}`);
269
+ result.errors.push(msg);
270
+ }
271
+ }
272
+
273
+ // 9. Delete project if requested
274
+ if (deleteProject) {
275
+ try {
276
+ logger.log(' Deleting Dokploy project...');
277
+ await api.deleteProject(state.projectId);
278
+ result.deletedProject = true;
279
+ logger.log(' ✓ Project deleted');
280
+ } catch (error) {
281
+ const msg = `Failed to delete project: ${error}`;
282
+ logger.log(` ⚠ ${msg}`);
283
+ result.errors.push(msg);
284
+ }
285
+ }
286
+
287
+ return result;
288
+ }
289
+
290
+ /**
291
+ * Delete AWS backup resources (S3 bucket, IAM user)
292
+ */
293
+ async function deleteAwsBackupResources(
294
+ backupState: BackupState,
295
+ awsEndpoint?: string,
296
+ logger?: { log: (msg: string) => void },
297
+ ): Promise<void> {
298
+ const {
299
+ S3Client,
300
+ DeleteBucketCommand,
301
+ DeleteObjectsCommand,
302
+ ListObjectsV2Command,
303
+ } = await import('@aws-sdk/client-s3');
304
+
305
+ const {
306
+ IAMClient,
307
+ DeleteAccessKeyCommand,
308
+ DeleteUserCommand,
309
+ DeleteUserPolicyCommand,
310
+ ListAccessKeysCommand,
311
+ } = await import('@aws-sdk/client-iam');
312
+
313
+ const clientConfig: {
314
+ region: string;
315
+ endpoint?: string;
316
+ forcePathStyle?: boolean;
317
+ credentials?: { accessKeyId: string; secretAccessKey: string };
318
+ } = {
319
+ region: backupState.region,
320
+ };
321
+
322
+ if (awsEndpoint) {
323
+ clientConfig.endpoint = awsEndpoint;
324
+ clientConfig.forcePathStyle = true;
325
+ clientConfig.credentials = {
326
+ accessKeyId: 'test',
327
+ secretAccessKey: 'test',
328
+ };
329
+ }
330
+
331
+ const s3 = new S3Client(clientConfig);
332
+ const iam = new IAMClient(clientConfig);
333
+
334
+ try {
335
+ // Delete all objects in the bucket first
336
+ logger?.log(` Emptying bucket: ${backupState.bucketName}`);
337
+ let continuationToken: string | undefined;
338
+ do {
339
+ const listResult = await s3.send(
340
+ new ListObjectsV2Command({
341
+ Bucket: backupState.bucketName,
342
+ ContinuationToken: continuationToken,
343
+ }),
344
+ );
345
+
346
+ if (listResult.Contents?.length) {
347
+ await s3.send(
348
+ new DeleteObjectsCommand({
349
+ Bucket: backupState.bucketName,
350
+ Delete: {
351
+ Objects: listResult.Contents.map((o) => ({ Key: o.Key })),
352
+ },
353
+ }),
354
+ );
355
+ }
356
+
357
+ continuationToken = listResult.NextContinuationToken;
358
+ } while (continuationToken);
359
+
360
+ // Delete the bucket
361
+ logger?.log(` Deleting bucket: ${backupState.bucketName}`);
362
+ await s3.send(new DeleteBucketCommand({ Bucket: backupState.bucketName }));
363
+ } catch (error) {
364
+ // Bucket might not exist, continue with IAM cleanup
365
+ logger?.log(` Warning: Could not delete bucket: ${error}`);
366
+ }
367
+
368
+ try {
369
+ // Delete all access keys for the IAM user
370
+ logger?.log(
371
+ ` Deleting IAM access keys for: ${backupState.iamUserName}`,
372
+ );
373
+ const keysResult = await iam.send(
374
+ new ListAccessKeysCommand({ UserName: backupState.iamUserName }),
375
+ );
376
+
377
+ for (const key of keysResult.AccessKeyMetadata ?? []) {
378
+ await iam.send(
379
+ new DeleteAccessKeyCommand({
380
+ UserName: backupState.iamUserName,
381
+ AccessKeyId: key.AccessKeyId,
382
+ }),
383
+ );
384
+ }
385
+
386
+ // Delete the user policy
387
+ logger?.log(` Deleting IAM policy for: ${backupState.iamUserName}`);
388
+ await iam.send(
389
+ new DeleteUserPolicyCommand({
390
+ UserName: backupState.iamUserName,
391
+ PolicyName: 'DokployBackupAccess',
392
+ }),
393
+ );
394
+
395
+ // Delete the IAM user
396
+ logger?.log(` Deleting IAM user: ${backupState.iamUserName}`);
397
+ await iam.send(
398
+ new DeleteUserCommand({ UserName: backupState.iamUserName }),
399
+ );
400
+ } catch (error) {
401
+ // IAM user might not exist
402
+ logger?.log(` Warning: Could not delete IAM user: ${error}`);
403
+ }
404
+
405
+ s3.destroy();
406
+ iam.destroy();
407
+ }