@jjrawlins/cfn-drift-remediate 0.0.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/.tool-versions +3 -0
- package/CLAUDE.md +61 -0
- package/LICENSE +202 -0
- package/README.md +113 -0
- package/lib/cli.d.ts +6 -0
- package/lib/cli.js +391 -0
- package/lib/index.d.ts +10 -0
- package/lib/index.js +151 -0
- package/lib/lib/cfn-client.d.ts +75 -0
- package/lib/lib/cfn-client.js +258 -0
- package/lib/lib/eligible-resources.d.ts +27 -0
- package/lib/lib/eligible-resources.js +415 -0
- package/lib/lib/interactive.d.ts +30 -0
- package/lib/lib/interactive.js +238 -0
- package/lib/lib/plan.d.ts +21 -0
- package/lib/lib/plan.js +153 -0
- package/lib/lib/resource-identifier.d.ts +28 -0
- package/lib/lib/resource-identifier.js +231 -0
- package/lib/lib/resource-importer.d.ts +37 -0
- package/lib/lib/resource-importer.js +211 -0
- package/lib/lib/template-transformer.d.ts +64 -0
- package/lib/lib/template-transformer.js +531 -0
- package/lib/lib/types.d.ts +220 -0
- package/lib/lib/types.js +3 -0
- package/lib/lib/utils.d.ts +37 -0
- package/lib/lib/utils.js +51 -0
- package/package.json +122 -0
package/lib/cli.js
ADDED
|
@@ -0,0 +1,391 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
9
|
+
}) : (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
o[k2] = m[k];
|
|
12
|
+
}));
|
|
13
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
14
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
15
|
+
}) : function(o, v) {
|
|
16
|
+
o["default"] = v;
|
|
17
|
+
});
|
|
18
|
+
var __importStar = (this && this.__importStar) || (function () {
|
|
19
|
+
var ownKeys = function(o) {
|
|
20
|
+
ownKeys = Object.getOwnPropertyNames || function (o) {
|
|
21
|
+
var ar = [];
|
|
22
|
+
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
|
|
23
|
+
return ar;
|
|
24
|
+
};
|
|
25
|
+
return ownKeys(o);
|
|
26
|
+
};
|
|
27
|
+
return function (mod) {
|
|
28
|
+
if (mod && mod.__esModule) return mod;
|
|
29
|
+
var result = {};
|
|
30
|
+
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
|
|
31
|
+
__setModuleDefault(result, mod);
|
|
32
|
+
return result;
|
|
33
|
+
};
|
|
34
|
+
})();
|
|
35
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
36
|
+
exports.remediate = remediate;
|
|
37
|
+
const fs = __importStar(require("fs"));
|
|
38
|
+
const path = __importStar(require("path"));
|
|
39
|
+
const cfn_client_1 = require("./lib/cfn-client");
|
|
40
|
+
const eligible_resources_1 = require("./lib/eligible-resources");
|
|
41
|
+
const interactive_1 = require("./lib/interactive");
|
|
42
|
+
const plan_1 = require("./lib/plan");
|
|
43
|
+
const resource_importer_1 = require("./lib/resource-importer");
|
|
44
|
+
const template_transformer_1 = require("./lib/template-transformer");
|
|
45
|
+
const utils_1 = require("./lib/utils");
|
|
46
|
+
const DEFAULT_CAPABILITIES = ['CAPABILITY_IAM', 'CAPABILITY_NAMED_IAM', 'CAPABILITY_AUTO_EXPAND'];
|
|
47
|
+
function packageVersion() {
|
|
48
|
+
try {
|
|
49
|
+
const pkgPath = path.join(__dirname, '..', 'package.json');
|
|
50
|
+
return JSON.parse(fs.readFileSync(pkgPath, 'utf-8')).version;
|
|
51
|
+
}
|
|
52
|
+
catch {
|
|
53
|
+
return 'unknown';
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
/**
|
|
57
|
+
* Main remediation function that orchestrates the drift remediation process
|
|
58
|
+
*/
|
|
59
|
+
async function remediate(options, spinner) {
|
|
60
|
+
const client = new cfn_client_1.CfnClientWrapper({
|
|
61
|
+
region: options.region,
|
|
62
|
+
profile: options.profile,
|
|
63
|
+
});
|
|
64
|
+
const result = {
|
|
65
|
+
success: false,
|
|
66
|
+
remediatedResources: [],
|
|
67
|
+
skippedResources: [],
|
|
68
|
+
removedResources: [],
|
|
69
|
+
errors: [],
|
|
70
|
+
};
|
|
71
|
+
const log = (message) => {
|
|
72
|
+
if (spinner) {
|
|
73
|
+
spinner.text = message;
|
|
74
|
+
}
|
|
75
|
+
else if (options.verbose) {
|
|
76
|
+
console.log(message);
|
|
77
|
+
}
|
|
78
|
+
};
|
|
79
|
+
try {
|
|
80
|
+
// Step 1: Get stack info and original template
|
|
81
|
+
log('Fetching stack information...');
|
|
82
|
+
const stackInfo = await client.getStackInfo(options.stackName);
|
|
83
|
+
log('Fetching original template...');
|
|
84
|
+
const originalTemplateBody = await client.getTemplate(stackInfo.stackId, true);
|
|
85
|
+
const originalTemplate = (0, template_transformer_1.parseTemplate)(originalTemplateBody);
|
|
86
|
+
if (options.verbose) {
|
|
87
|
+
console.log(`Found stack: ${stackInfo.stackName} (${stackInfo.stackId})`);
|
|
88
|
+
}
|
|
89
|
+
// Steps 2-4: Detect drift and collect decisions (or load from plan)
|
|
90
|
+
let allDriftedResources;
|
|
91
|
+
let decisions;
|
|
92
|
+
if (options.applyPlan) {
|
|
93
|
+
// Apply a previously exported plan — skip drift detection and prompting
|
|
94
|
+
log('Loading remediation plan...');
|
|
95
|
+
const planJson = fs.readFileSync(path.resolve(options.applyPlan), 'utf-8');
|
|
96
|
+
const plan = (0, plan_1.loadPlan)(planJson, options.stackName);
|
|
97
|
+
const loaded = (0, plan_1.planToDecisions)(plan);
|
|
98
|
+
allDriftedResources = loaded.allDriftedResources;
|
|
99
|
+
decisions = loaded.decisions;
|
|
100
|
+
if (spinner)
|
|
101
|
+
spinner.start('Processing...');
|
|
102
|
+
}
|
|
103
|
+
else {
|
|
104
|
+
// Normal flow: detect drift and prompt interactively
|
|
105
|
+
// Step 2: Detect drift
|
|
106
|
+
log('Detecting stack drift...');
|
|
107
|
+
const detectionId = await client.detectDrift(stackInfo.stackId);
|
|
108
|
+
log('Waiting for drift detection to complete...');
|
|
109
|
+
const detectionResult = await client.waitForDriftDetection(detectionId);
|
|
110
|
+
if (detectionResult.status !== 'DETECTION_COMPLETE') {
|
|
111
|
+
result.errors.push(`Drift detection did not complete: ${detectionResult.status}`);
|
|
112
|
+
return result;
|
|
113
|
+
}
|
|
114
|
+
if (detectionResult.driftStatus === 'IN_SYNC') {
|
|
115
|
+
if (spinner)
|
|
116
|
+
spinner.succeed('Stack is in sync - no drift detected');
|
|
117
|
+
result.success = true;
|
|
118
|
+
return result;
|
|
119
|
+
}
|
|
120
|
+
// Step 3: Get drifted resources and separate by type
|
|
121
|
+
log('Analyzing drifted resources...');
|
|
122
|
+
allDriftedResources = await client.getDriftedResources(stackInfo.stackId);
|
|
123
|
+
if (allDriftedResources.length === 0) {
|
|
124
|
+
if (spinner)
|
|
125
|
+
spinner.succeed('No drifted resources found');
|
|
126
|
+
result.success = true;
|
|
127
|
+
return result;
|
|
128
|
+
}
|
|
129
|
+
// Filter to only importable resources
|
|
130
|
+
const modifiedResources = [];
|
|
131
|
+
const deletedResources = [];
|
|
132
|
+
for (const resource of allDriftedResources) {
|
|
133
|
+
if (!(0, eligible_resources_1.isResourceImportable)(resource.resourceType)) {
|
|
134
|
+
result.skippedResources.push(resource.logicalResourceId);
|
|
135
|
+
continue;
|
|
136
|
+
}
|
|
137
|
+
if (resource.stackResourceDriftStatus === 'DELETED') {
|
|
138
|
+
deletedResources.push(resource);
|
|
139
|
+
}
|
|
140
|
+
else {
|
|
141
|
+
modifiedResources.push(resource);
|
|
142
|
+
}
|
|
143
|
+
}
|
|
144
|
+
if (modifiedResources.length === 0 && deletedResources.length === 0) {
|
|
145
|
+
result.errors.push('All drifted resources are not eligible for import');
|
|
146
|
+
return result;
|
|
147
|
+
}
|
|
148
|
+
// Step 4: Interactive decisions
|
|
149
|
+
if (spinner)
|
|
150
|
+
spinner.stop();
|
|
151
|
+
decisions = await (0, interactive_1.promptForDecisions)(modifiedResources, deletedResources, options.yes ?? false);
|
|
152
|
+
if (spinner)
|
|
153
|
+
spinner.start('Processing...');
|
|
154
|
+
// Export plan if requested (exit without executing)
|
|
155
|
+
if (options.exportPlan) {
|
|
156
|
+
const planMetadata = {
|
|
157
|
+
stackName: stackInfo.stackName,
|
|
158
|
+
region: client.region,
|
|
159
|
+
createdAt: new Date().toISOString(),
|
|
160
|
+
toolVersion: packageVersion(),
|
|
161
|
+
driftDetectionId: detectionId,
|
|
162
|
+
};
|
|
163
|
+
const plan = (0, plan_1.buildPlan)(planMetadata, decisions);
|
|
164
|
+
const planPath = path.resolve(options.exportPlan);
|
|
165
|
+
fs.writeFileSync(planPath, (0, plan_1.serializePlan)(plan));
|
|
166
|
+
if (spinner)
|
|
167
|
+
spinner.succeed(`Plan exported to ${planPath}`);
|
|
168
|
+
result.success = true;
|
|
169
|
+
return result;
|
|
170
|
+
}
|
|
171
|
+
}
|
|
172
|
+
// Record skipped resources
|
|
173
|
+
for (const r of decisions.skip) {
|
|
174
|
+
result.skippedResources.push(r.logicalResourceId);
|
|
175
|
+
}
|
|
176
|
+
// If everything was skipped or cancelled, we're done
|
|
177
|
+
if (decisions.autofix.length === 0 && decisions.reimport.length === 0 && decisions.remove.length === 0) {
|
|
178
|
+
if (spinner)
|
|
179
|
+
spinner.succeed('No actions selected');
|
|
180
|
+
result.success = true;
|
|
181
|
+
return result;
|
|
182
|
+
}
|
|
183
|
+
// Step 5: Build resources to import from decisions
|
|
184
|
+
const resourceIdentifiers = await client.getResourceIdentifiers(originalTemplateBody);
|
|
185
|
+
// Autofix resources: use existing buildResourcesToImport
|
|
186
|
+
const { importable: autofixImportable, skipped: autofixSkipped } = (0, resource_importer_1.buildResourcesToImport)(decisions.autofix, resourceIdentifiers);
|
|
187
|
+
for (const s of autofixSkipped) {
|
|
188
|
+
if (!result.skippedResources.includes(s.logicalResourceId)) {
|
|
189
|
+
result.skippedResources.push(s.logicalResourceId);
|
|
190
|
+
}
|
|
191
|
+
}
|
|
192
|
+
// Reimport resources: build from user-provided physical IDs
|
|
193
|
+
const reimportImportable = [];
|
|
194
|
+
for (const { resource, physicalId } of decisions.reimport) {
|
|
195
|
+
const descriptor = (0, resource_importer_1.buildReimportDescriptor)(resource, physicalId, resourceIdentifiers);
|
|
196
|
+
if (descriptor) {
|
|
197
|
+
reimportImportable.push(descriptor);
|
|
198
|
+
}
|
|
199
|
+
else {
|
|
200
|
+
result.errors.push(`Could not determine import identifier for ${resource.logicalResourceId} from "${physicalId}"`);
|
|
201
|
+
result.skippedResources.push(resource.logicalResourceId);
|
|
202
|
+
}
|
|
203
|
+
}
|
|
204
|
+
// Combined importable list
|
|
205
|
+
const allImportable = [...autofixImportable, ...reimportImportable];
|
|
206
|
+
// If nothing to import AND nothing to remove, we're stuck
|
|
207
|
+
if (allImportable.length === 0 && decisions.remove.length === 0) {
|
|
208
|
+
result.errors.push('Could not determine import identifiers for any selected resources');
|
|
209
|
+
return result;
|
|
210
|
+
}
|
|
211
|
+
// Build the set of logical IDs that need removal from the template
|
|
212
|
+
// This includes everything being acted on (autofix, reimport, remove)
|
|
213
|
+
const logicalIdsToRemove = new Set([
|
|
214
|
+
...autofixImportable.map((r) => r.LogicalResourceId),
|
|
215
|
+
...reimportImportable.map((r) => r.LogicalResourceId),
|
|
216
|
+
...decisions.remove.map((r) => r.logicalResourceId),
|
|
217
|
+
]);
|
|
218
|
+
// Analyze cascade removals (resources with broken Ref/GetAtt to removed resources)
|
|
219
|
+
const cascadeRemovals = (0, template_transformer_1.analyzeCascadeRemovals)(originalTemplate, logicalIdsToRemove);
|
|
220
|
+
// Partition into permanent (depend on user-removed resources) vs temporary (depend on autofix/reimport)
|
|
221
|
+
const permanentlyRemovedIds = new Set(decisions.remove.map((r) => r.logicalResourceId));
|
|
222
|
+
const permanentCascade = cascadeRemovals.filter((c) => permanentlyRemovedIds.has(c.dependsOn));
|
|
223
|
+
const temporaryCascade = cascadeRemovals.filter((c) => !permanentlyRemovedIds.has(c.dependsOn));
|
|
224
|
+
if (cascadeRemovals.length > 0) {
|
|
225
|
+
if (spinner)
|
|
226
|
+
spinner.stop();
|
|
227
|
+
(0, interactive_1.displayCascadeWarning)(permanentCascade, temporaryCascade);
|
|
228
|
+
if (spinner)
|
|
229
|
+
spinner.start('Processing...');
|
|
230
|
+
}
|
|
231
|
+
// Dry run - show what would be done
|
|
232
|
+
if (options.dryRun) {
|
|
233
|
+
if (spinner)
|
|
234
|
+
spinner.info('Dry run - planned actions:');
|
|
235
|
+
if (allImportable.length > 0) {
|
|
236
|
+
console.log('\nResources to remediate:');
|
|
237
|
+
for (const resource of allImportable) {
|
|
238
|
+
console.log(` - ${resource.LogicalResourceId} (${resource.ResourceType})`);
|
|
239
|
+
console.log(` Identifier: ${JSON.stringify(resource.ResourceIdentifier)}`);
|
|
240
|
+
}
|
|
241
|
+
}
|
|
242
|
+
if (decisions.remove.length > 0) {
|
|
243
|
+
console.log('\nResources to remove from stack:');
|
|
244
|
+
for (const r of decisions.remove) {
|
|
245
|
+
console.log(` - ${r.logicalResourceId} (${r.resourceType})`);
|
|
246
|
+
}
|
|
247
|
+
}
|
|
248
|
+
if (permanentCascade.length > 0) {
|
|
249
|
+
console.log('\nResources permanently cascade-removed (broken references):');
|
|
250
|
+
for (const c of permanentCascade) {
|
|
251
|
+
console.log(` - ${c.logicalResourceId} (${c.resourceType}) -> depends on ${c.dependsOn}`);
|
|
252
|
+
}
|
|
253
|
+
}
|
|
254
|
+
if (temporaryCascade.length > 0) {
|
|
255
|
+
console.log('\nResources temporarily removed and recreated:');
|
|
256
|
+
for (const c of temporaryCascade) {
|
|
257
|
+
console.log(` - ${c.logicalResourceId} (${c.resourceType}) -> depends on ${c.dependsOn}`);
|
|
258
|
+
}
|
|
259
|
+
}
|
|
260
|
+
result.success = true;
|
|
261
|
+
result.remediatedResources = allImportable.map((r) => r.LogicalResourceId);
|
|
262
|
+
result.removedResources = [
|
|
263
|
+
...decisions.remove.map((r) => r.logicalResourceId),
|
|
264
|
+
...permanentCascade.map((c) => c.logicalResourceId),
|
|
265
|
+
];
|
|
266
|
+
return result;
|
|
267
|
+
}
|
|
268
|
+
// Determine required capabilities
|
|
269
|
+
const resourceTypes = allImportable.map((r) => r.ResourceType);
|
|
270
|
+
const additionalCaps = (0, eligible_resources_1.getAllRequiredCapabilities)(resourceTypes);
|
|
271
|
+
const capabilities = [...new Set([...DEFAULT_CAPABILITIES, ...additionalCaps])];
|
|
272
|
+
// Save recovery checkpoint before any stack mutations
|
|
273
|
+
const timestamp = new Date().toISOString().replace(/[:.]/g, '-');
|
|
274
|
+
const backupFileName = `.cfn-drift-remediate-backup-${stackInfo.stackName}-${timestamp}.json`;
|
|
275
|
+
const backupPath = path.resolve(process.cwd(), backupFileName);
|
|
276
|
+
const checkpoint = {
|
|
277
|
+
stackName: stackInfo.stackName,
|
|
278
|
+
stackId: stackInfo.stackId,
|
|
279
|
+
originalTemplateBody,
|
|
280
|
+
parameters: stackInfo.parameters,
|
|
281
|
+
driftedResourceIds: Array.from(logicalIdsToRemove),
|
|
282
|
+
timestamp: new Date().toISOString(),
|
|
283
|
+
};
|
|
284
|
+
fs.writeFileSync(backupPath, JSON.stringify(checkpoint, null, 2));
|
|
285
|
+
log(`Recovery checkpoint saved to: ${backupPath}`);
|
|
286
|
+
if (options.verbose) {
|
|
287
|
+
console.log(`Recovery checkpoint: ${backupPath}`);
|
|
288
|
+
}
|
|
289
|
+
// Step 6: Set DeletionPolicy: Retain on all resources
|
|
290
|
+
// DELETED resources must be removed from the template first — CloudFormation
|
|
291
|
+
// cannot update metadata on resources that no longer exist in AWS.
|
|
292
|
+
log('Setting DeletionPolicy: Retain on all resources...');
|
|
293
|
+
const deletedLogicalIds = new Set(allDriftedResources
|
|
294
|
+
.filter((r) => r.stackResourceDriftStatus === 'DELETED' && logicalIdsToRemove.has(r.logicalResourceId))
|
|
295
|
+
.map((r) => r.logicalResourceId));
|
|
296
|
+
let retainTemplate = (0, template_transformer_1.setRetentionOnAllResources)(originalTemplate);
|
|
297
|
+
if (deletedLogicalIds.size > 0) {
|
|
298
|
+
// Remove deleted resources and clean up their dangling references
|
|
299
|
+
const { template: cleanedTemplate } = (0, template_transformer_1.transformTemplateForRemoval)(retainTemplate, deletedLogicalIds, new Map());
|
|
300
|
+
retainTemplate = cleanedTemplate;
|
|
301
|
+
}
|
|
302
|
+
await client.updateStack(stackInfo.stackId, (0, template_transformer_1.stringifyTemplate)(retainTemplate), retainTemplate.Parameters ? stackInfo.parameters : undefined, capabilities);
|
|
303
|
+
// Step 7: Resolve cross-references to MODIFIED resources being removed
|
|
304
|
+
// (DELETED resources are already removed from the template, so only MODIFIED refs remain)
|
|
305
|
+
const modifiedIdsToRemove = new Set([...logicalIdsToRemove].filter((id) => !deletedLogicalIds.has(id)));
|
|
306
|
+
const references = (0, template_transformer_1.collectReferences)(retainTemplate, modifiedIdsToRemove);
|
|
307
|
+
let resolvedValues = new Map();
|
|
308
|
+
if (references.size > 0) {
|
|
309
|
+
log('Resolving references to drifted resources...');
|
|
310
|
+
const outputTemplate = (0, template_transformer_1.addResolutionOutputs)(retainTemplate, references);
|
|
311
|
+
await client.updateStack(stackInfo.stackId, (0, template_transformer_1.stringifyTemplate)(outputTemplate), outputTemplate.Parameters ? stackInfo.parameters : undefined, capabilities);
|
|
312
|
+
const updatedStackInfo = await client.getStackInfo(stackInfo.stackId);
|
|
313
|
+
resolvedValues = (0, template_transformer_1.parseResolvedOutputs)(updatedStackInfo.outputs || [], references);
|
|
314
|
+
}
|
|
315
|
+
// Step 8: Remove remaining (MODIFIED) resources from template
|
|
316
|
+
log('Removing resources from stack (resources retained in AWS)...');
|
|
317
|
+
const { template: removalTemplate } = (0, template_transformer_1.transformTemplateForRemoval)(retainTemplate, modifiedIdsToRemove, resolvedValues);
|
|
318
|
+
const removalParams = removalTemplate.Parameters ? stackInfo.parameters : undefined;
|
|
319
|
+
await client.updateStack(stackInfo.stackId, (0, template_transformer_1.stringifyTemplate)(removalTemplate), removalParams, capabilities);
|
|
320
|
+
// Step 9: Import resources (only if there are resources to import)
|
|
321
|
+
if (allImportable.length > 0) {
|
|
322
|
+
log('Preparing import template with actual resource state...');
|
|
323
|
+
// Build import template from the removal template (current stack state)
|
|
324
|
+
// and add back the resources being imported with their actual properties
|
|
325
|
+
const importTemplate = (0, utils_1.deepClone)(removalTemplate);
|
|
326
|
+
for (const importable of allImportable) {
|
|
327
|
+
const logicalId = importable.LogicalResourceId;
|
|
328
|
+
const originalResource = originalTemplate.Resources?.[logicalId];
|
|
329
|
+
if (!originalResource)
|
|
330
|
+
continue;
|
|
331
|
+
// Start with original resource definition
|
|
332
|
+
importTemplate.Resources[logicalId] = (0, utils_1.deepClone)(originalResource);
|
|
333
|
+
// For autofix resources, override with actual (drifted) properties
|
|
334
|
+
const autofixResource = decisions.autofix.find((r) => r.logicalResourceId === logicalId);
|
|
335
|
+
if (autofixResource?.actualProperties && Object.keys(autofixResource.actualProperties).length > 0) {
|
|
336
|
+
importTemplate.Resources[logicalId].Properties = autofixResource.actualProperties;
|
|
337
|
+
}
|
|
338
|
+
importTemplate.Resources[logicalId].DeletionPolicy = 'Retain';
|
|
339
|
+
}
|
|
340
|
+
// Ensure Retain on all resources in import template
|
|
341
|
+
for (const logicalId of Object.keys(importTemplate.Resources)) {
|
|
342
|
+
importTemplate.Resources[logicalId].DeletionPolicy = 'Retain';
|
|
343
|
+
}
|
|
344
|
+
log('Creating import change set...');
|
|
345
|
+
const changeSetName = await client.createImportChangeSet(stackInfo.stackName, (0, template_transformer_1.stringifyTemplate)(importTemplate), allImportable, capabilities);
|
|
346
|
+
log('Executing import...');
|
|
347
|
+
await client.executeChangeSet(stackInfo.stackName, changeSetName);
|
|
348
|
+
}
|
|
349
|
+
// Step 10: Restore template
|
|
350
|
+
if (decisions.remove.length > 0) {
|
|
351
|
+
// Some resources permanently removed — restore original MINUS removed resources
|
|
352
|
+
log('Restoring template (excluding removed resources)...');
|
|
353
|
+
const restoredTemplate = (0, utils_1.deepClone)(originalTemplate);
|
|
354
|
+
for (const r of decisions.remove) {
|
|
355
|
+
delete restoredTemplate.Resources[r.logicalResourceId];
|
|
356
|
+
}
|
|
357
|
+
// Clean up references and outputs pointing to removed resources
|
|
358
|
+
const { template: cleanedTemplate } = (0, template_transformer_1.transformTemplateForRemoval)(restoredTemplate, new Set(decisions.remove.map((r) => r.logicalResourceId)), resolvedValues);
|
|
359
|
+
// Restore original DeletionPolicy values (transformTemplateForRemoval sets Retain on all)
|
|
360
|
+
for (const [logicalId, resource] of Object.entries(cleanedTemplate.Resources || {})) {
|
|
361
|
+
const originalResource = originalTemplate.Resources?.[logicalId];
|
|
362
|
+
if (originalResource?.DeletionPolicy) {
|
|
363
|
+
resource.DeletionPolicy = originalResource.DeletionPolicy;
|
|
364
|
+
}
|
|
365
|
+
else {
|
|
366
|
+
delete resource.DeletionPolicy;
|
|
367
|
+
}
|
|
368
|
+
}
|
|
369
|
+
await client.updateStack(stackInfo.stackName, (0, template_transformer_1.stringifyTemplate)(cleanedTemplate), cleanedTemplate.Parameters ? stackInfo.parameters : undefined, capabilities);
|
|
370
|
+
}
|
|
371
|
+
else {
|
|
372
|
+
// No removals — restore exact original
|
|
373
|
+
log('Restoring original template...');
|
|
374
|
+
await client.updateStack(stackInfo.stackName, originalTemplateBody, stackInfo.parameters, capabilities);
|
|
375
|
+
}
|
|
376
|
+
result.success = true;
|
|
377
|
+
result.remediatedResources = allImportable.map((r) => r.LogicalResourceId);
|
|
378
|
+
result.removedResources = [
|
|
379
|
+
...decisions.remove.map((r) => r.logicalResourceId),
|
|
380
|
+
...permanentCascade.map((c) => c.logicalResourceId),
|
|
381
|
+
];
|
|
382
|
+
if (options.verbose) {
|
|
383
|
+
console.log(`Remediation complete. Recovery checkpoint can be removed: ${backupPath}`);
|
|
384
|
+
}
|
|
385
|
+
}
|
|
386
|
+
catch (error) {
|
|
387
|
+
result.errors.push(error instanceof Error ? error.message : String(error));
|
|
388
|
+
}
|
|
389
|
+
return result;
|
|
390
|
+
}
|
|
391
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cli.js","sourceRoot":"","sources":["../src/cli.ts"],"names":[],"mappings":"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","sourcesContent":["import * as fs from 'fs';\nimport * as path from 'path';\nimport { Ora } from 'ora';\nimport { CfnClientWrapper } from './lib/cfn-client';\nimport { isResourceImportable, getAllRequiredCapabilities } from './lib/eligible-resources';\nimport { displayCascadeWarning, promptForDecisions } from './lib/interactive';\nimport { buildPlan, serializePlan, loadPlan, planToDecisions } from './lib/plan';\nimport { buildResourcesToImport, buildReimportDescriptor } from './lib/resource-importer';\nimport {\n  parseTemplate,\n  stringifyTemplate,\n  collectReferences,\n  addResolutionOutputs,\n  parseResolvedOutputs,\n  setRetentionOnAllResources,\n  transformTemplateForRemoval,\n  analyzeCascadeRemovals,\n} from './lib/template-transformer';\nimport {\n  RemediationOptions,\n  RemediationResult,\n  RecoveryCheckpoint,\n  DriftedResource,\n  InteractiveDecisions,\n  ResourceToImport,\n} from './lib/types';\nimport { deepClone } from './lib/utils';\n\nconst DEFAULT_CAPABILITIES = ['CAPABILITY_IAM', 'CAPABILITY_NAMED_IAM', 'CAPABILITY_AUTO_EXPAND'];\n\nfunction packageVersion(): string {\n  try {\n    const pkgPath = path.join(__dirname, '..', 'package.json');\n    return JSON.parse(fs.readFileSync(pkgPath, 'utf-8')).version;\n  } catch {\n    return 'unknown';\n  }\n}\n\n/**\n * Main remediation function that orchestrates the drift remediation process\n */\nexport async function remediate(\n  options: RemediationOptions,\n  spinner?: Ora,\n): Promise<RemediationResult> {\n  const client = new CfnClientWrapper({\n    region: options.region,\n    profile: options.profile,\n  });\n\n  const result: RemediationResult = {\n    success: false,\n    remediatedResources: [],\n    skippedResources: [],\n    removedResources: [],\n    errors: [],\n  };\n\n  const log = (message: string) => {\n    if (spinner) {\n      spinner.text = message;\n    } else if (options.verbose) {\n      console.log(message);\n    }\n  };\n\n  try {\n    // Step 1: Get stack info and original template\n    log('Fetching stack information...');\n    const stackInfo = await client.getStackInfo(options.stackName);\n\n    log('Fetching original template...');\n    const originalTemplateBody = await client.getTemplate(stackInfo.stackId, true);\n    const originalTemplate = parseTemplate(originalTemplateBody);\n\n    if (options.verbose) {\n      console.log(`Found stack: ${stackInfo.stackName} (${stackInfo.stackId})`);\n    }\n\n    // Steps 2-4: Detect drift and collect decisions (or load from plan)\n    let allDriftedResources: DriftedResource[];\n    let decisions: InteractiveDecisions;\n\n    if (options.applyPlan) {\n      // Apply a previously exported plan — skip drift detection and prompting\n      log('Loading remediation plan...');\n      const planJson = fs.readFileSync(path.resolve(options.applyPlan), 'utf-8');\n      const plan = loadPlan(planJson, options.stackName);\n      const loaded = planToDecisions(plan);\n      allDriftedResources = loaded.allDriftedResources;\n      decisions = loaded.decisions;\n\n      if (spinner) spinner.start('Processing...');\n    } else {\n      // Normal flow: detect drift and prompt interactively\n\n      // Step 2: Detect drift\n      log('Detecting stack drift...');\n      const detectionId = await client.detectDrift(stackInfo.stackId);\n\n      log('Waiting for drift detection to complete...');\n      const detectionResult = await client.waitForDriftDetection(detectionId);\n\n      if (detectionResult.status !== 'DETECTION_COMPLETE') {\n        result.errors.push(`Drift detection did not complete: ${detectionResult.status}`);\n        return result;\n      }\n\n      if (detectionResult.driftStatus === 'IN_SYNC') {\n        if (spinner) spinner.succeed('Stack is in sync - no drift detected');\n        result.success = true;\n        return result;\n      }\n\n      // Step 3: Get drifted resources and separate by type\n      log('Analyzing drifted resources...');\n      allDriftedResources = await client.getDriftedResources(stackInfo.stackId);\n\n      if (allDriftedResources.length === 0) {\n        if (spinner) spinner.succeed('No drifted resources found');\n        result.success = true;\n        return result;\n      }\n\n      // Filter to only importable resources\n      const modifiedResources: DriftedResource[] = [];\n      const deletedResources: DriftedResource[] = [];\n\n      for (const resource of allDriftedResources) {\n        if (!isResourceImportable(resource.resourceType)) {\n          result.skippedResources.push(resource.logicalResourceId);\n          continue;\n        }\n        if (resource.stackResourceDriftStatus === 'DELETED') {\n          deletedResources.push(resource);\n        } else {\n          modifiedResources.push(resource);\n        }\n      }\n\n      if (modifiedResources.length === 0 && deletedResources.length === 0) {\n        result.errors.push('All drifted resources are not eligible for import');\n        return result;\n      }\n\n      // Step 4: Interactive decisions\n      if (spinner) spinner.stop();\n\n      decisions = await promptForDecisions(\n        modifiedResources,\n        deletedResources,\n        options.yes ?? false,\n      );\n\n      if (spinner) spinner.start('Processing...');\n\n      // Export plan if requested (exit without executing)\n      if (options.exportPlan) {\n        const planMetadata = {\n          stackName: stackInfo.stackName,\n          region: client.region,\n          createdAt: new Date().toISOString(),\n          toolVersion: packageVersion(),\n          driftDetectionId: detectionId,\n        };\n        const plan = buildPlan(planMetadata, decisions);\n        const planPath = path.resolve(options.exportPlan);\n        fs.writeFileSync(planPath, serializePlan(plan));\n        if (spinner) spinner.succeed(`Plan exported to ${planPath}`);\n        result.success = true;\n        return result;\n      }\n    }\n\n    // Record skipped resources\n    for (const r of decisions.skip) {\n      result.skippedResources.push(r.logicalResourceId);\n    }\n\n    // If everything was skipped or cancelled, we're done\n    if (decisions.autofix.length === 0 && decisions.reimport.length === 0 && decisions.remove.length === 0) {\n      if (spinner) spinner.succeed('No actions selected');\n      result.success = true;\n      return result;\n    }\n\n    // Step 5: Build resources to import from decisions\n    const resourceIdentifiers = await client.getResourceIdentifiers(originalTemplateBody);\n\n    // Autofix resources: use existing buildResourcesToImport\n    const { importable: autofixImportable, skipped: autofixSkipped } =\n      buildResourcesToImport(decisions.autofix, resourceIdentifiers);\n\n    for (const s of autofixSkipped) {\n      if (!result.skippedResources.includes(s.logicalResourceId)) {\n        result.skippedResources.push(s.logicalResourceId);\n      }\n    }\n\n    // Reimport resources: build from user-provided physical IDs\n    const reimportImportable: ResourceToImport[] = [];\n    for (const { resource, physicalId } of decisions.reimport) {\n      const descriptor = buildReimportDescriptor(resource, physicalId, resourceIdentifiers);\n      if (descriptor) {\n        reimportImportable.push(descriptor);\n      } else {\n        result.errors.push(\n          `Could not determine import identifier for ${resource.logicalResourceId} from \"${physicalId}\"`,\n        );\n        result.skippedResources.push(resource.logicalResourceId);\n      }\n    }\n\n    // Combined importable list\n    const allImportable = [...autofixImportable, ...reimportImportable];\n\n    // If nothing to import AND nothing to remove, we're stuck\n    if (allImportable.length === 0 && decisions.remove.length === 0) {\n      result.errors.push('Could not determine import identifiers for any selected resources');\n      return result;\n    }\n\n    // Build the set of logical IDs that need removal from the template\n    // This includes everything being acted on (autofix, reimport, remove)\n    const logicalIdsToRemove = new Set([\n      ...autofixImportable.map((r) => r.LogicalResourceId),\n      ...reimportImportable.map((r) => r.LogicalResourceId),\n      ...decisions.remove.map((r) => r.logicalResourceId),\n    ]);\n\n    // Analyze cascade removals (resources with broken Ref/GetAtt to removed resources)\n    const cascadeRemovals = analyzeCascadeRemovals(originalTemplate, logicalIdsToRemove);\n\n    // Partition into permanent (depend on user-removed resources) vs temporary (depend on autofix/reimport)\n    const permanentlyRemovedIds = new Set(decisions.remove.map((r) => r.logicalResourceId));\n    const permanentCascade = cascadeRemovals.filter((c) => permanentlyRemovedIds.has(c.dependsOn));\n    const temporaryCascade = cascadeRemovals.filter((c) => !permanentlyRemovedIds.has(c.dependsOn));\n\n    if (cascadeRemovals.length > 0) {\n      if (spinner) spinner.stop();\n      displayCascadeWarning(permanentCascade, temporaryCascade);\n      if (spinner) spinner.start('Processing...');\n    }\n\n    // Dry run - show what would be done\n    if (options.dryRun) {\n      if (spinner) spinner.info('Dry run - planned actions:');\n      if (allImportable.length > 0) {\n        console.log('\\nResources to remediate:');\n        for (const resource of allImportable) {\n          console.log(`  - ${resource.LogicalResourceId} (${resource.ResourceType})`);\n          console.log(`    Identifier: ${JSON.stringify(resource.ResourceIdentifier)}`);\n        }\n      }\n      if (decisions.remove.length > 0) {\n        console.log('\\nResources to remove from stack:');\n        for (const r of decisions.remove) {\n          console.log(`  - ${r.logicalResourceId} (${r.resourceType})`);\n        }\n      }\n      if (permanentCascade.length > 0) {\n        console.log('\\nResources permanently cascade-removed (broken references):');\n        for (const c of permanentCascade) {\n          console.log(`  - ${c.logicalResourceId} (${c.resourceType}) -> depends on ${c.dependsOn}`);\n        }\n      }\n      if (temporaryCascade.length > 0) {\n        console.log('\\nResources temporarily removed and recreated:');\n        for (const c of temporaryCascade) {\n          console.log(`  - ${c.logicalResourceId} (${c.resourceType}) -> depends on ${c.dependsOn}`);\n        }\n      }\n      result.success = true;\n      result.remediatedResources = allImportable.map((r) => r.LogicalResourceId);\n      result.removedResources = [\n        ...decisions.remove.map((r) => r.logicalResourceId),\n        ...permanentCascade.map((c) => c.logicalResourceId),\n      ];\n      return result;\n    }\n\n    // Determine required capabilities\n    const resourceTypes = allImportable.map((r) => r.ResourceType);\n    const additionalCaps = getAllRequiredCapabilities(resourceTypes);\n    const capabilities = [...new Set([...DEFAULT_CAPABILITIES, ...additionalCaps])];\n\n    // Save recovery checkpoint before any stack mutations\n    const timestamp = new Date().toISOString().replace(/[:.]/g, '-');\n    const backupFileName = `.cfn-drift-remediate-backup-${stackInfo.stackName}-${timestamp}.json`;\n    const backupPath = path.resolve(process.cwd(), backupFileName);\n    const checkpoint: RecoveryCheckpoint = {\n      stackName: stackInfo.stackName,\n      stackId: stackInfo.stackId,\n      originalTemplateBody,\n      parameters: stackInfo.parameters,\n      driftedResourceIds: Array.from(logicalIdsToRemove),\n      timestamp: new Date().toISOString(),\n    };\n    fs.writeFileSync(backupPath, JSON.stringify(checkpoint, null, 2));\n    log(`Recovery checkpoint saved to: ${backupPath}`);\n    if (options.verbose) {\n      console.log(`Recovery checkpoint: ${backupPath}`);\n    }\n\n    // Step 6: Set DeletionPolicy: Retain on all resources\n    // DELETED resources must be removed from the template first — CloudFormation\n    // cannot update metadata on resources that no longer exist in AWS.\n    log('Setting DeletionPolicy: Retain on all resources...');\n    const deletedLogicalIds = new Set(\n      allDriftedResources\n        .filter((r) => r.stackResourceDriftStatus === 'DELETED' && logicalIdsToRemove.has(r.logicalResourceId))\n        .map((r) => r.logicalResourceId),\n    );\n\n    let retainTemplate = setRetentionOnAllResources(originalTemplate);\n\n    if (deletedLogicalIds.size > 0) {\n      // Remove deleted resources and clean up their dangling references\n      const { template: cleanedTemplate } = transformTemplateForRemoval(\n        retainTemplate,\n        deletedLogicalIds,\n        new Map(), // no resolved values — just strip unresolvable references\n      );\n      retainTemplate = cleanedTemplate;\n    }\n\n    await client.updateStack(\n      stackInfo.stackId,\n      stringifyTemplate(retainTemplate),\n      retainTemplate.Parameters ? stackInfo.parameters : undefined,\n      capabilities,\n    );\n\n    // Step 7: Resolve cross-references to MODIFIED resources being removed\n    // (DELETED resources are already removed from the template, so only MODIFIED refs remain)\n    const modifiedIdsToRemove = new Set(\n      [...logicalIdsToRemove].filter((id) => !deletedLogicalIds.has(id)),\n    );\n    const references = collectReferences(retainTemplate, modifiedIdsToRemove);\n\n    let resolvedValues = new Map<string, unknown>();\n    if (references.size > 0) {\n      log('Resolving references to drifted resources...');\n\n      const outputTemplate = addResolutionOutputs(retainTemplate, references);\n      await client.updateStack(\n        stackInfo.stackId,\n        stringifyTemplate(outputTemplate),\n        outputTemplate.Parameters ? stackInfo.parameters : undefined,\n        capabilities,\n      );\n\n      const updatedStackInfo = await client.getStackInfo(stackInfo.stackId);\n      resolvedValues = parseResolvedOutputs(updatedStackInfo.outputs || [], references);\n    }\n\n    // Step 8: Remove remaining (MODIFIED) resources from template\n    log('Removing resources from stack (resources retained in AWS)...');\n    const { template: removalTemplate } = transformTemplateForRemoval(\n      retainTemplate,\n      modifiedIdsToRemove,\n      resolvedValues,\n    );\n\n    const removalParams = removalTemplate.Parameters ? stackInfo.parameters : undefined;\n    await client.updateStack(\n      stackInfo.stackId,\n      stringifyTemplate(removalTemplate),\n      removalParams,\n      capabilities,\n    );\n\n    // Step 9: Import resources (only if there are resources to import)\n    if (allImportable.length > 0) {\n      log('Preparing import template with actual resource state...');\n\n      // Build import template from the removal template (current stack state)\n      // and add back the resources being imported with their actual properties\n      const importTemplate = deepClone(removalTemplate);\n\n      for (const importable of allImportable) {\n        const logicalId = importable.LogicalResourceId;\n        const originalResource = originalTemplate.Resources?.[logicalId];\n        if (!originalResource) continue;\n\n        // Start with original resource definition\n        importTemplate.Resources[logicalId] = deepClone(originalResource);\n\n        // For autofix resources, override with actual (drifted) properties\n        const autofixResource = decisions.autofix.find((r) => r.logicalResourceId === logicalId);\n        if (autofixResource?.actualProperties && Object.keys(autofixResource.actualProperties).length > 0) {\n          importTemplate.Resources[logicalId].Properties = autofixResource.actualProperties;\n        }\n\n        importTemplate.Resources[logicalId].DeletionPolicy = 'Retain';\n      }\n\n      // Ensure Retain on all resources in import template\n      for (const logicalId of Object.keys(importTemplate.Resources)) {\n        importTemplate.Resources[logicalId].DeletionPolicy = 'Retain';\n      }\n\n      log('Creating import change set...');\n      const changeSetName = await client.createImportChangeSet(\n        stackInfo.stackName,\n        stringifyTemplate(importTemplate),\n        allImportable,\n        capabilities,\n      );\n\n      log('Executing import...');\n      await client.executeChangeSet(stackInfo.stackName, changeSetName);\n    }\n\n    // Step 10: Restore template\n    if (decisions.remove.length > 0) {\n      // Some resources permanently removed — restore original MINUS removed resources\n      log('Restoring template (excluding removed resources)...');\n      const restoredTemplate = deepClone(originalTemplate);\n      for (const r of decisions.remove) {\n        delete restoredTemplate.Resources[r.logicalResourceId];\n      }\n      // Clean up references and outputs pointing to removed resources\n      const { template: cleanedTemplate } = transformTemplateForRemoval(\n        restoredTemplate,\n        new Set(decisions.remove.map((r) => r.logicalResourceId)),\n        resolvedValues,\n      );\n      // Restore original DeletionPolicy values (transformTemplateForRemoval sets Retain on all)\n      for (const [logicalId, resource] of Object.entries(cleanedTemplate.Resources || {})) {\n        const originalResource = originalTemplate.Resources?.[logicalId];\n        if (originalResource?.DeletionPolicy) {\n          resource.DeletionPolicy = originalResource.DeletionPolicy;\n        } else {\n          delete resource.DeletionPolicy;\n        }\n      }\n      await client.updateStack(\n        stackInfo.stackName,\n        stringifyTemplate(cleanedTemplate),\n        cleanedTemplate.Parameters ? stackInfo.parameters : undefined,\n        capabilities,\n      );\n    } else {\n      // No removals — restore exact original\n      log('Restoring original template...');\n      await client.updateStack(\n        stackInfo.stackName,\n        originalTemplateBody,\n        stackInfo.parameters,\n        capabilities,\n      );\n    }\n\n    result.success = true;\n    result.remediatedResources = allImportable.map((r) => r.LogicalResourceId);\n    result.removedResources = [\n      ...decisions.remove.map((r) => r.logicalResourceId),\n      ...permanentCascade.map((c) => c.logicalResourceId),\n    ];\n\n    if (options.verbose) {\n      console.log(`Remediation complete. Recovery checkpoint can be removed: ${backupPath}`);\n    }\n\n  } catch (error) {\n    result.errors.push(error instanceof Error ? error.message : String(error));\n  }\n\n  return result;\n}\n"]}
|
package/lib/index.d.ts
ADDED
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
export { remediate } from './cli';
|
|
3
|
+
export { CfnClientWrapper } from './lib/cfn-client';
|
|
4
|
+
export * from './lib/types';
|
|
5
|
+
export * from './lib/eligible-resources';
|
|
6
|
+
export * from './lib/template-transformer';
|
|
7
|
+
export * from './lib/resource-importer';
|
|
8
|
+
export * from './lib/resource-identifier';
|
|
9
|
+
export { promptForDecisions, formatDriftDiff, displayCascadeWarning } from './lib/interactive';
|
|
10
|
+
export { buildPlan, serializePlan, loadPlan, planToDecisions } from './lib/plan';
|