@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.
- package/CHANGELOG.md +17 -0
- package/dist/{HostingerProvider-B9N-TKbp.mjs → HostingerProvider-402UdK89.mjs} +34 -1
- package/dist/HostingerProvider-402UdK89.mjs.map +1 -0
- package/dist/{HostingerProvider-DUV9-Tzg.cjs → HostingerProvider-BiXdHjiq.cjs} +34 -1
- package/dist/HostingerProvider-BiXdHjiq.cjs.map +1 -0
- package/dist/{Route53Provider-C8mS0zY6.mjs → Route53Provider-DbBo7Uz5.mjs} +53 -1
- package/dist/Route53Provider-DbBo7Uz5.mjs.map +1 -0
- package/dist/{Route53Provider-Bs7Arms9.cjs → Route53Provider-kfJ77LmL.cjs} +53 -1
- package/dist/Route53Provider-kfJ77LmL.cjs.map +1 -0
- package/dist/backup-provisioner-B5e-F6zX.cjs +164 -0
- package/dist/backup-provisioner-B5e-F6zX.cjs.map +1 -0
- package/dist/backup-provisioner-BIArpmTr.mjs +163 -0
- package/dist/backup-provisioner-BIArpmTr.mjs.map +1 -0
- package/dist/{config-ZQM1vBoz.cjs → config-6JHOwLCx.cjs} +30 -2
- package/dist/{config-ZQM1vBoz.cjs.map → config-6JHOwLCx.cjs.map} +1 -1
- package/dist/{config-DfCJ29PQ.mjs → config-DxASSNjr.mjs} +25 -3
- package/dist/{config-DfCJ29PQ.mjs.map → config-DxASSNjr.mjs.map} +1 -1
- package/dist/config.cjs +3 -2
- package/dist/config.d.cts +14 -2
- package/dist/config.d.cts.map +1 -1
- package/dist/config.d.mts +15 -3
- package/dist/config.d.mts.map +1 -1
- package/dist/config.mjs +3 -3
- package/dist/{dokploy-api-z0833e7r.mjs → dokploy-api-2ldYoN3i.mjs} +131 -1
- package/dist/dokploy-api-2ldYoN3i.mjs.map +1 -0
- package/dist/dokploy-api-C93pveuy.mjs +3 -0
- package/dist/dokploy-api-CbDh4o93.cjs +3 -0
- package/dist/{dokploy-api-CQvhV6Hd.cjs → dokploy-api-DLgvEQlr.cjs} +131 -1
- package/dist/dokploy-api-DLgvEQlr.cjs.map +1 -0
- package/dist/{index-C0SpUT9Y.d.mts → index-C-KxSGGK.d.mts} +133 -31
- package/dist/index-C-KxSGGK.d.mts.map +1 -0
- package/dist/{index-B58qjyBd.d.cts → index-Cyk2rTyj.d.cts} +132 -30
- package/dist/index-Cyk2rTyj.d.cts.map +1 -0
- package/dist/index.cjs +662 -152
- package/dist/index.cjs.map +1 -1
- package/dist/index.mjs +626 -116
- package/dist/index.mjs.map +1 -1
- package/dist/{openapi-BcSjLfWq.mjs → openapi-BYlyAbH3.mjs} +6 -5
- package/dist/openapi-BYlyAbH3.mjs.map +1 -0
- package/dist/{openapi-D6Hcfov0.cjs → openapi-CnvwSRDU.cjs} +6 -5
- package/dist/openapi-CnvwSRDU.cjs.map +1 -0
- package/dist/openapi.cjs +3 -3
- package/dist/openapi.d.cts +1 -0
- package/dist/openapi.d.cts.map +1 -1
- package/dist/openapi.d.mts +2 -1
- package/dist/openapi.d.mts.map +1 -1
- package/dist/openapi.mjs +3 -3
- package/dist/{types-B9UZ7fOG.d.mts → types-CZg5iUgD.d.mts} +1 -1
- package/dist/{types-B9UZ7fOG.d.mts.map → types-CZg5iUgD.d.mts.map} +1 -1
- package/dist/workspace/index.cjs +1 -1
- package/dist/workspace/index.d.cts +1 -1
- package/dist/workspace/index.d.mts +2 -2
- package/dist/workspace/index.mjs +1 -1
- package/dist/{workspace-BW2iU37P.mjs → workspace-9IQIjwkQ.mjs} +20 -4
- package/dist/workspace-9IQIjwkQ.mjs.map +1 -0
- package/dist/{workspace-2Do2YcGZ.cjs → workspace-D2ocAlpl.cjs} +20 -4
- package/dist/workspace-D2ocAlpl.cjs.map +1 -0
- package/examples/cron-example.ts +6 -6
- package/examples/function-example.ts +1 -1
- package/package.json +6 -3
- package/src/config.ts +44 -0
- package/src/deploy/__tests__/backup-provisioner.spec.ts +428 -0
- package/src/deploy/__tests__/createDnsProvider.spec.ts +23 -0
- package/src/deploy/__tests__/env-resolver.spec.ts +1 -1
- package/src/deploy/__tests__/undeploy.spec.ts +758 -0
- package/src/deploy/backup-provisioner.ts +316 -0
- package/src/deploy/dns/DnsProvider.ts +39 -1
- package/src/deploy/dns/HostingerProvider.ts +74 -0
- package/src/deploy/dns/Route53Provider.ts +81 -0
- package/src/deploy/dns/index.ts +25 -0
- package/src/deploy/dokploy-api.ts +237 -0
- package/src/deploy/index.ts +71 -13
- package/src/deploy/state.ts +171 -0
- package/src/deploy/undeploy.ts +407 -0
- package/src/dev/__tests__/index.spec.ts +490 -0
- package/src/dev/index.ts +313 -18
- package/src/generators/FunctionGenerator.ts +1 -1
- package/src/generators/Generator.ts +4 -1
- package/src/init/__tests__/generators.spec.ts +167 -18
- package/src/init/__tests__/init.spec.ts +66 -3
- package/src/init/generators/auth.ts +6 -5
- package/src/init/generators/config.ts +49 -7
- package/src/init/generators/docker.ts +8 -8
- package/src/init/generators/index.ts +1 -0
- package/src/init/generators/models.ts +3 -5
- package/src/init/generators/package.ts +4 -0
- package/src/init/generators/test.ts +133 -0
- package/src/init/generators/ui.ts +13 -12
- package/src/init/generators/web.ts +9 -8
- package/src/init/index.ts +2 -0
- package/src/init/templates/api.ts +6 -6
- package/src/init/templates/minimal.ts +2 -2
- package/src/init/templates/worker.ts +2 -2
- package/src/init/versions.ts +3 -3
- package/src/openapi.ts +6 -2
- package/src/test/__tests__/__fixtures__/workspace.ts +104 -0
- package/src/test/__tests__/api.spec.ts +199 -0
- package/src/test/__tests__/auth.spec.ts +162 -0
- package/src/test/__tests__/index.spec.ts +323 -0
- package/src/test/__tests__/web.spec.ts +210 -0
- package/src/test/index.ts +165 -14
- package/src/workspace/__tests__/index.spec.ts +3 -0
- package/src/workspace/index.ts +4 -2
- package/src/workspace/schema.ts +26 -0
- package/src/workspace/types.ts +14 -37
- package/dist/HostingerProvider-B9N-TKbp.mjs.map +0 -1
- package/dist/HostingerProvider-DUV9-Tzg.cjs.map +0 -1
- package/dist/Route53Provider-Bs7Arms9.cjs.map +0 -1
- package/dist/Route53Provider-C8mS0zY6.mjs.map +0 -1
- package/dist/dokploy-api-CQvhV6Hd.cjs.map +0 -1
- package/dist/dokploy-api-CWc02yyg.cjs +0 -3
- package/dist/dokploy-api-DSJYNx88.mjs +0 -3
- package/dist/dokploy-api-z0833e7r.mjs.map +0 -1
- package/dist/index-B58qjyBd.d.cts.map +0 -1
- package/dist/index-C0SpUT9Y.d.mts.map +0 -1
- package/dist/openapi-BcSjLfWq.mjs.map +0 -1
- package/dist/openapi-D6Hcfov0.cjs.map +0 -1
- package/dist/workspace-2Do2YcGZ.cjs.map +0 -1
- 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
|
+
}
|