@aws-cdk/toolkit-lib 1.6.0 → 1.7.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.
@@ -33,10 +33,9 @@ class DriftFormatter {
33
33
  */
34
34
  formatStackDrift() {
35
35
  const formatterOutput = this.formatStackDriftChanges(this.buildLogicalToPathMap());
36
- // we are only interested in actual drifts and always ignore the metadata resource
37
- const actualDrifts = this.resourceDriftResults.filter(d => d.StackResourceDriftStatus === 'MODIFIED' ||
38
- d.StackResourceDriftStatus === 'DELETED' ||
39
- d.ResourceType === 'AWS::CDK::Metadata');
36
+ // we are only interested in actual drifts (and ignore the metadata resource)
37
+ const actualDrifts = this.resourceDriftResults.filter(d => (d.StackResourceDriftStatus === 'MODIFIED' || d.StackResourceDriftStatus === 'DELETED')
38
+ && d.ResourceType !== 'AWS::CDK::Metadata');
40
39
  // must output the stack name if there are drifts
41
40
  const stackHeader = (0, node_util_1.format)(`Stack ${chalk.bold(this.stackName)}\n`);
42
41
  if (actualDrifts.length === 0) {
@@ -45,6 +44,7 @@ class DriftFormatter {
45
44
  numResourcesWithDrift: 0,
46
45
  numResourcesUnchecked: this.allStackResources.size - this.resourceDriftResults.length,
47
46
  stackHeader,
47
+ unchecked: formatterOutput.unchecked,
48
48
  summary: finalResult,
49
49
  };
50
50
  }
@@ -68,11 +68,8 @@ class DriftFormatter {
68
68
  return map;
69
69
  }
70
70
  /**
71
- * Renders stack drift information to the given stream
71
+ * Renders stack drift information
72
72
  *
73
- * @param driftResults - The stack resource drifts from CloudFormation
74
- * @param allStackResources - A map of all stack resources
75
- * @param verbose - Whether to output more verbose text (include undrifted resources)
76
73
  * @param logicalToPathMap - A map from logical ID to construct path
77
74
  */
78
75
  formatStackDriftChanges(logicalToPathMap = {}) {
@@ -91,34 +88,32 @@ class DriftFormatter {
91
88
  for (const drift of unchangedResources) {
92
89
  if (!drift.LogicalResourceId || !drift.ResourceType)
93
90
  continue;
94
- unchanged += `${CONTEXT} ${this.formatValue(drift.ResourceType, chalk.cyan)} ${this.formatLogicalId(logicalToPathMap, drift.LogicalResourceId)}\n`;
91
+ unchanged += `${CONTEXT} ${chalk.cyan(drift.ResourceType)} ${this.formatLogicalId(logicalToPathMap, drift.LogicalResourceId)}\n`;
95
92
  }
96
93
  unchanged += this.printSectionFooter();
97
94
  }
98
- // Process all unchecked resources
99
- if (this.allStackResources) {
100
- const uncheckedResources = Array.from(this.allStackResources.keys()).filter((logicalId) => {
101
- return !drifts.find((drift) => drift.LogicalResourceId === logicalId);
102
- });
103
- if (uncheckedResources.length > 0) {
104
- unchecked = this.printSectionHeader('Unchecked Resources');
105
- for (const logicalId of uncheckedResources) {
106
- const resourceType = this.allStackResources.get(logicalId);
107
- unchecked += `${CONTEXT} ${this.formatValue(resourceType, chalk.cyan)} ${this.formatLogicalId(logicalToPathMap, logicalId)}\n`;
108
- }
109
- unchecked += this.printSectionFooter();
95
+ // Process all unchecked and unknown resources
96
+ const uncheckedResources = Array.from(this.allStackResources.keys()).filter((logicalId) => {
97
+ const drift = drifts.find((d) => d.LogicalResourceId === logicalId);
98
+ return !drift || drift.StackResourceDriftStatus === client_cloudformation_1.StackResourceDriftStatus.UNKNOWN;
99
+ });
100
+ if (uncheckedResources.length > 0) {
101
+ unchecked = this.printSectionHeader('Unchecked Resources');
102
+ for (const logicalId of uncheckedResources) {
103
+ const resourceType = this.allStackResources.get(logicalId);
104
+ unchecked += `${CONTEXT} ${chalk.cyan(resourceType)} ${this.formatLogicalId(logicalToPathMap, logicalId)}\n`;
110
105
  }
106
+ unchecked += this.printSectionFooter();
111
107
  }
112
- // Process modified resources
113
- const modifiedResources = drifts.filter(d => d.StackResourceDriftStatus === client_cloudformation_1.StackResourceDriftStatus.MODIFIED);
108
+ // Process modified resources (exclude AWS::CDK::Metadata)
109
+ const modifiedResources = drifts.filter(d => d.StackResourceDriftStatus === client_cloudformation_1.StackResourceDriftStatus.MODIFIED
110
+ && d.ResourceType !== 'AWS::CDK::Metadata');
114
111
  if (modifiedResources.length > 0) {
115
112
  modified = this.printSectionHeader('Modified Resources');
116
113
  for (const drift of modifiedResources) {
117
114
  if (!drift.LogicalResourceId || !drift.ResourceType)
118
115
  continue;
119
- if (modified === undefined)
120
- modified = '';
121
- modified += `${UPDATE} ${this.formatValue(drift.ResourceType, chalk.cyan)} ${this.formatLogicalId(logicalToPathMap, drift.LogicalResourceId)}\n`;
116
+ modified += `${UPDATE} ${chalk.cyan(drift.ResourceType)} ${this.formatLogicalId(logicalToPathMap, drift.LogicalResourceId)}\n`;
122
117
  if (drift.PropertyDifferences) {
123
118
  const propDiffs = drift.PropertyDifferences;
124
119
  for (let i = 0; i < propDiffs.length; i++) {
@@ -132,14 +127,15 @@ class DriftFormatter {
132
127
  }
133
128
  modified += this.printSectionFooter();
134
129
  }
135
- // Process deleted resources
136
- const deletedResources = drifts.filter(d => d.StackResourceDriftStatus === client_cloudformation_1.StackResourceDriftStatus.DELETED);
130
+ // Process deleted resources (exclude AWS::CDK::Metadata)
131
+ const deletedResources = drifts.filter(d => d.StackResourceDriftStatus === client_cloudformation_1.StackResourceDriftStatus.DELETED
132
+ && d.ResourceType !== 'AWS::CDK::Metadata');
137
133
  if (deletedResources.length > 0) {
138
134
  deleted = this.printSectionHeader('Deleted Resources');
139
135
  for (const drift of deletedResources) {
140
136
  if (!drift.LogicalResourceId || !drift.ResourceType)
141
137
  continue;
142
- deleted += `${REMOVAL} ${this.formatValue(drift.ResourceType, chalk.cyan)} ${this.formatLogicalId(logicalToPathMap, drift.LogicalResourceId)}\n`;
138
+ deleted += `${REMOVAL} ${chalk.cyan(drift.ResourceType)} ${this.formatLogicalId(logicalToPathMap, drift.LogicalResourceId)}\n`;
143
139
  }
144
140
  deleted += this.printSectionFooter();
145
141
  }
@@ -167,15 +163,6 @@ class DriftFormatter {
167
163
  }
168
164
  return `${normalizedPath} ${chalk.gray(logicalId)}`;
169
165
  }
170
- formatValue(value, colorFn) {
171
- if (value == null) {
172
- return '';
173
- }
174
- if (typeof value === 'string') {
175
- return colorFn(value);
176
- }
177
- return colorFn(JSON.stringify(value));
178
- }
179
166
  printSectionHeader(title) {
180
167
  return `${chalk.underline(chalk.bold(title))}\n`;
181
168
  }
@@ -187,8 +174,8 @@ class DriftFormatter {
187
174
  difference.isRemoval ? REMOVAL :
188
175
  UPDATE, propertyPath);
189
176
  if (difference.isUpdate) {
190
- result += (0, node_util_1.format)(' ├─ %s %s\n', REMOVAL, this.formatValue(difference.oldValue, chalk.red));
191
- result += (0, node_util_1.format)(' └─ %s %s\n', ADDITION, this.formatValue(difference.newValue, chalk.green));
177
+ result += (0, node_util_1.format)(' ├─ %s %s\n', REMOVAL, chalk.red(difference.oldValue));
178
+ result += (0, node_util_1.format)(' └─ %s %s\n', ADDITION, chalk.green(difference.newValue));
192
179
  }
193
180
  return result;
194
181
  }
@@ -198,4 +185,4 @@ const ADDITION = chalk.green('[+]');
198
185
  const CONTEXT = chalk.grey('[ ]');
199
186
  const UPDATE = chalk.yellow('[~]');
200
187
  const REMOVAL = chalk.red('[-]');
201
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"drift-formatter.js","sourceRoot":"","sources":["drift-formatter.ts"],"names":[],"mappings":"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","sourcesContent":["import { format } from 'node:util';\nimport * as cxschema from '@aws-cdk/cloud-assembly-schema';\nimport { Difference } from '@aws-cdk/cloudformation-diff';\nimport type * as cxapi from '@aws-cdk/cx-api';\nimport type { StackResourceDrift } from '@aws-sdk/client-cloudformation';\nimport { StackResourceDriftStatus } from '@aws-sdk/client-cloudformation';\nimport * as chalk from 'chalk';\nimport type { FormattedDrift } from '../../actions/drift';\n\n/**\n * Props for the Drift Formatter\n */\nexport interface DriftFormatterProps {\n  /**\n   * The CloudFormation stack artifact\n   */\n  readonly stack: cxapi.CloudFormationStackArtifact;\n\n  /**\n   * The results of stack drift detection\n   */\n  readonly resourceDrifts: StackResourceDrift[];\n}\n\ninterface DriftFormatterOutput {\n  /**\n   * Number of resources with drift. If undefined, then an error occurred\n   * and resources were not properly checked for drift.\n   */\n  readonly numResourcesWithDrift: number;\n\n  /**\n   * How many resources were not checked for drift. If undefined, then an\n   * error occurred and resources were not properly checked for drift.\n   */\n  readonly numResourcesUnchecked: number;\n\n  /**\n   * Resources that have not changed\n   */\n  readonly unchanged?: string;\n\n  /**\n   * Resources that were not checked for drift\n   */\n  readonly unchecked?: string;\n\n  /**\n   * Resources with drift\n   */\n  readonly modified?: string;\n\n  /**\n   * Resources that have been deleted (drift)\n   */\n  readonly deleted?: string;\n\n  /**\n   * The header, containing the stack name\n   */\n  readonly stackHeader: string;\n\n  /**\n   * The final results (summary) of the drift results\n   */\n  readonly summary: string;\n}\n\n/**\n * Class for formatting drift detection output\n */\nexport class DriftFormatter {\n  public readonly stackName: string;\n\n  private readonly stack: cxapi.CloudFormationStackArtifact;\n  private readonly resourceDriftResults: StackResourceDrift[];\n  private readonly allStackResources: Map<string, string>;\n\n  constructor(props: DriftFormatterProps) {\n    this.stack = props.stack;\n    this.stackName = props.stack.displayName ?? props.stack.stackName;\n    this.resourceDriftResults = props.resourceDrifts;\n\n    this.allStackResources = new Map<string, string>();\n    Object.keys(this.stack.template.Resources || {}).forEach(id => {\n      const resource = this.stack.template.Resources[id];\n      // always ignore the metadata resource\n      if (resource.Type === 'AWS::CDK::Metadata') {\n        return;\n      }\n      this.allStackResources.set(id, resource.Type);\n    });\n  }\n\n  /**\n   * Format the stack drift detection results\n   */\n  public formatStackDrift(): DriftFormatterOutput {\n    const formatterOutput = this.formatStackDriftChanges(this.buildLogicalToPathMap());\n\n    // we are only interested in actual drifts and always ignore the metadata resource\n    const actualDrifts = this.resourceDriftResults.filter(d =>\n      d.StackResourceDriftStatus === 'MODIFIED' ||\n      d.StackResourceDriftStatus === 'DELETED' ||\n      d.ResourceType === 'AWS::CDK::Metadata',\n    );\n\n    // must output the stack name if there are drifts\n    const stackHeader = format(`Stack ${chalk.bold(this.stackName)}\\n`);\n\n    if (actualDrifts.length === 0) {\n      const finalResult = chalk.green('No drift detected\\n');\n      return {\n        numResourcesWithDrift: 0,\n        numResourcesUnchecked: this.allStackResources.size - this.resourceDriftResults.length,\n        stackHeader,\n        summary: finalResult,\n      };\n    }\n\n    const finalResult = chalk.yellow(`\\n${actualDrifts.length} resource${actualDrifts.length === 1 ? '' : 's'} ${actualDrifts.length === 1 ? 'has' : 'have'} drifted from their expected configuration\\n`);\n    return {\n      numResourcesWithDrift: actualDrifts.length,\n      numResourcesUnchecked: this.allStackResources.size - this.resourceDriftResults.length,\n      stackHeader,\n      unchanged: formatterOutput.unchanged,\n      unchecked: formatterOutput.unchecked,\n      modified: formatterOutput.modified,\n      deleted: formatterOutput.deleted,\n      summary: finalResult,\n    };\n  }\n\n  private buildLogicalToPathMap() {\n    const map: { [id: string]: string } = {};\n    for (const md of this.stack.findMetadataByType(cxschema.ArtifactMetadataEntryType.LOGICAL_ID)) {\n      map[md.data as string] = md.path;\n    }\n    return map;\n  }\n\n  /**\n   * Renders stack drift information to the given stream\n   *\n   * @param driftResults - The stack resource drifts from CloudFormation\n   * @param allStackResources - A map of all stack resources\n   * @param verbose - Whether to output more verbose text (include undrifted resources)\n   * @param logicalToPathMap - A map from logical ID to construct path\n   */\n  private formatStackDriftChanges(\n    logicalToPathMap: { [logicalId: string]: string } = {}): FormattedDrift {\n    if (this.resourceDriftResults.length === 0) {\n      return {};\n    }\n\n    let unchanged;\n    let unchecked;\n    let modified;\n    let deleted;\n\n    const drifts = this.resourceDriftResults;\n\n    // Process unchanged resources\n    const unchangedResources = drifts.filter(d => d.StackResourceDriftStatus === StackResourceDriftStatus.IN_SYNC);\n    if (unchangedResources.length > 0) {\n      unchanged = this.printSectionHeader('Resources In Sync');\n\n      for (const drift of unchangedResources) {\n        if (!drift.LogicalResourceId || !drift.ResourceType) continue;\n        unchanged += `${CONTEXT} ${this.formatValue(drift.ResourceType, chalk.cyan)} ${this.formatLogicalId(logicalToPathMap, drift.LogicalResourceId)}\\n`;\n      }\n      unchanged += this.printSectionFooter();\n    }\n\n    // Process all unchecked resources\n    if (this.allStackResources) {\n      const uncheckedResources = Array.from(this.allStackResources.keys()).filter((logicalId) => {\n        return !drifts.find((drift) => drift.LogicalResourceId === logicalId);\n      });\n      if (uncheckedResources.length > 0) {\n        unchecked = this.printSectionHeader('Unchecked Resources');\n        for (const logicalId of uncheckedResources) {\n          const resourceType = this.allStackResources.get(logicalId);\n          unchecked += `${CONTEXT} ${this.formatValue(resourceType, chalk.cyan)} ${this.formatLogicalId(logicalToPathMap, logicalId)}\\n`;\n        }\n        unchecked += this.printSectionFooter();\n      }\n    }\n\n    // Process modified resources\n    const modifiedResources = drifts.filter(d => d.StackResourceDriftStatus === StackResourceDriftStatus.MODIFIED);\n    if (modifiedResources.length > 0) {\n      modified = this.printSectionHeader('Modified Resources');\n\n      for (const drift of modifiedResources) {\n        if (!drift.LogicalResourceId || !drift.ResourceType) continue;\n        if (modified === undefined) modified = '';\n        modified += `${UPDATE} ${this.formatValue(drift.ResourceType, chalk.cyan)} ${this.formatLogicalId(logicalToPathMap, drift.LogicalResourceId)}\\n`;\n        if (drift.PropertyDifferences) {\n          const propDiffs = drift.PropertyDifferences;\n          for (let i = 0; i < propDiffs.length; i++) {\n            const diff = propDiffs[i];\n            if (!diff.PropertyPath) continue;\n            const difference = new Difference(diff.ExpectedValue, diff.ActualValue);\n            modified += this.formatTreeDiff(diff.PropertyPath, difference, i === propDiffs.length - 1);\n          }\n        }\n      }\n      modified += this.printSectionFooter();\n    }\n\n    // Process deleted resources\n    const deletedResources = drifts.filter(d => d.StackResourceDriftStatus === StackResourceDriftStatus.DELETED);\n    if (deletedResources.length > 0) {\n      deleted = this.printSectionHeader('Deleted Resources');\n      for (const drift of deletedResources) {\n        if (!drift.LogicalResourceId || !drift.ResourceType) continue;\n        deleted += `${REMOVAL} ${this.formatValue(drift.ResourceType, chalk.cyan)} ${this.formatLogicalId(logicalToPathMap, drift.LogicalResourceId)}\\n`;\n      }\n      deleted += this.printSectionFooter();\n    }\n\n    return { unchanged, unchecked, modified, deleted };\n  }\n\n  private formatLogicalId(logicalToPathMap: { [logicalId: string]: string }, logicalId: string): string {\n    const path = logicalToPathMap[logicalId];\n    if (!path) return logicalId;\n\n    let normalizedPath = path;\n    if (normalizedPath.startsWith('/')) {\n      normalizedPath = normalizedPath.slice(1);\n    }\n\n    let parts = normalizedPath.split('/');\n    if (parts.length > 1) {\n      parts = parts.slice(1);\n\n      // remove the last component if it's \"Resource\" or \"Default\" (if we have more than a single component)\n      if (parts.length > 1) {\n        const last = parts[parts.length - 1];\n        if (last === 'Resource' || last === 'Default') {\n          parts = parts.slice(0, parts.length - 1);\n        }\n      }\n\n      normalizedPath = parts.join('/');\n    }\n\n    return `${normalizedPath} ${chalk.gray(logicalId)}`;\n  }\n\n  private formatValue(value: any, colorFn: (str: string) => string): string {\n    if (value == null) {\n      return '';\n    }\n    if (typeof value === 'string') {\n      return colorFn(value);\n    }\n    return colorFn(JSON.stringify(value));\n  }\n\n  private printSectionHeader(title: string): string {\n    return `${chalk.underline(chalk.bold(title))}\\n`;\n  }\n\n  private printSectionFooter(): string {\n    return '\\n';\n  }\n\n  private formatTreeDiff(propertyPath: string, difference: Difference<any>, isLast: boolean): string {\n    let result = format(' %s─ %s %s\\n', isLast ? '└' : '├',\n      difference.isAddition ? ADDITION :\n        difference.isRemoval ? REMOVAL :\n          UPDATE,\n      propertyPath,\n    );\n    if (difference.isUpdate) {\n      result += format('     ├─ %s %s\\n', REMOVAL, this.formatValue(difference.oldValue, chalk.red));\n      result += format('     └─ %s %s\\n', ADDITION, this.formatValue(difference.newValue, chalk.green));\n    }\n    return result;\n  }\n}\n\nconst ADDITION = chalk.green('[+]');\nconst CONTEXT = chalk.grey('[ ]');\nconst UPDATE = chalk.yellow('[~]');\nconst REMOVAL = chalk.red('[-]');\n"]}
188
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"drift-formatter.js","sourceRoot":"","sources":["drift-formatter.ts"],"names":[],"mappings":"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","sourcesContent":["import { format } from 'node:util';\nimport * as cxschema from '@aws-cdk/cloud-assembly-schema';\nimport { Difference } from '@aws-cdk/cloudformation-diff';\nimport type * as cxapi from '@aws-cdk/cx-api';\nimport type { StackResourceDrift } from '@aws-sdk/client-cloudformation';\nimport { StackResourceDriftStatus } from '@aws-sdk/client-cloudformation';\nimport * as chalk from 'chalk';\nimport type { FormattedDrift } from '../../actions/drift';\n\n/**\n * Props for the Drift Formatter\n */\nexport interface DriftFormatterProps {\n  /**\n   * The CloudFormation stack artifact\n   */\n  readonly stack: cxapi.CloudFormationStackArtifact;\n\n  /**\n   * The results of stack drift detection\n   */\n  readonly resourceDrifts: StackResourceDrift[];\n}\n\ninterface DriftFormatterOutput {\n  /**\n   * Number of resources with drift. If undefined, then an error occurred\n   * and resources were not properly checked for drift.\n   */\n  readonly numResourcesWithDrift: number;\n\n  /**\n   * How many resources were not checked for drift. If undefined, then an\n   * error occurred and resources were not properly checked for drift.\n   */\n  readonly numResourcesUnchecked: number;\n\n  /**\n   * Resources that have not changed\n   */\n  readonly unchanged?: string;\n\n  /**\n   * Resources that were not checked for drift or have an UNKNOWN drift status\n   */\n  readonly unchecked?: string;\n\n  /**\n   * Resources with drift\n   */\n  readonly modified?: string;\n\n  /**\n   * Resources that have been deleted (drift)\n   */\n  readonly deleted?: string;\n\n  /**\n   * The header, containing the stack name\n   */\n  readonly stackHeader: string;\n\n  /**\n   * The final results (summary) of the drift results\n   */\n  readonly summary: string;\n}\n\n/**\n * Class for formatting drift detection output\n */\nexport class DriftFormatter {\n  public readonly stackName: string;\n\n  private readonly stack: cxapi.CloudFormationStackArtifact;\n  private readonly resourceDriftResults: StackResourceDrift[];\n  private readonly allStackResources: Map<string, string>;\n\n  constructor(props: DriftFormatterProps) {\n    this.stack = props.stack;\n    this.stackName = props.stack.displayName ?? props.stack.stackName;\n    this.resourceDriftResults = props.resourceDrifts;\n\n    this.allStackResources = new Map<string, string>();\n    Object.keys(this.stack.template.Resources || {}).forEach(id => {\n      const resource = this.stack.template.Resources[id];\n      // always ignore the metadata resource\n      if (resource.Type === 'AWS::CDK::Metadata') {\n        return;\n      }\n      this.allStackResources.set(id, resource.Type);\n    });\n  }\n\n  /**\n   * Format the stack drift detection results\n   */\n  public formatStackDrift(): DriftFormatterOutput {\n    const formatterOutput = this.formatStackDriftChanges(this.buildLogicalToPathMap());\n\n    // we are only interested in actual drifts (and ignore the metadata resource)\n    const actualDrifts = this.resourceDriftResults.filter(d =>\n      (d.StackResourceDriftStatus === 'MODIFIED' || d.StackResourceDriftStatus === 'DELETED')\n      && d.ResourceType !== 'AWS::CDK::Metadata');\n\n    // must output the stack name if there are drifts\n    const stackHeader = format(`Stack ${chalk.bold(this.stackName)}\\n`);\n\n    if (actualDrifts.length === 0) {\n      const finalResult = chalk.green('No drift detected\\n');\n      return {\n        numResourcesWithDrift: 0,\n        numResourcesUnchecked: this.allStackResources.size - this.resourceDriftResults.length,\n        stackHeader,\n        unchecked: formatterOutput.unchecked,\n        summary: finalResult,\n      };\n    }\n\n    const finalResult = chalk.yellow(`\\n${actualDrifts.length} resource${actualDrifts.length === 1 ? '' : 's'} ${actualDrifts.length === 1 ? 'has' : 'have'} drifted from their expected configuration\\n`);\n    return {\n      numResourcesWithDrift: actualDrifts.length,\n      numResourcesUnchecked: this.allStackResources.size - this.resourceDriftResults.length,\n      stackHeader,\n      unchanged: formatterOutput.unchanged,\n      unchecked: formatterOutput.unchecked,\n      modified: formatterOutput.modified,\n      deleted: formatterOutput.deleted,\n      summary: finalResult,\n    };\n  }\n\n  private buildLogicalToPathMap() {\n    const map: { [id: string]: string } = {};\n    for (const md of this.stack.findMetadataByType(cxschema.ArtifactMetadataEntryType.LOGICAL_ID)) {\n      map[md.data as string] = md.path;\n    }\n    return map;\n  }\n\n  /**\n   * Renders stack drift information\n   *\n   * @param logicalToPathMap - A map from logical ID to construct path\n   */\n  private formatStackDriftChanges(\n    logicalToPathMap: { [logicalId: string]: string } = {}): FormattedDrift {\n    if (this.resourceDriftResults.length === 0) {\n      return {};\n    }\n\n    let unchanged;\n    let unchecked;\n    let modified;\n    let deleted;\n\n    const drifts = this.resourceDriftResults;\n\n    // Process unchanged resources\n    const unchangedResources = drifts.filter(d => d.StackResourceDriftStatus === StackResourceDriftStatus.IN_SYNC);\n    if (unchangedResources.length > 0) {\n      unchanged = this.printSectionHeader('Resources In Sync');\n\n      for (const drift of unchangedResources) {\n        if (!drift.LogicalResourceId || !drift.ResourceType) continue;\n        unchanged += `${CONTEXT} ${chalk.cyan(drift.ResourceType)} ${this.formatLogicalId(logicalToPathMap, drift.LogicalResourceId)}\\n`;\n      }\n      unchanged += this.printSectionFooter();\n    }\n\n    // Process all unchecked and unknown resources\n    const uncheckedResources = Array.from(this.allStackResources.keys()).filter((logicalId) => {\n      const drift = drifts.find((d) => d.LogicalResourceId === logicalId);\n      return !drift || drift.StackResourceDriftStatus === StackResourceDriftStatus.UNKNOWN;\n    });\n    if (uncheckedResources.length > 0) {\n      unchecked = this.printSectionHeader('Unchecked Resources');\n      for (const logicalId of uncheckedResources) {\n        const resourceType = this.allStackResources.get(logicalId);\n        unchecked += `${CONTEXT} ${chalk.cyan(resourceType)} ${this.formatLogicalId(logicalToPathMap, logicalId)}\\n`;\n      }\n      unchecked += this.printSectionFooter();\n    }\n\n    // Process modified resources (exclude AWS::CDK::Metadata)\n    const modifiedResources = drifts.filter(d =>\n      d.StackResourceDriftStatus === StackResourceDriftStatus.MODIFIED\n      && d.ResourceType !== 'AWS::CDK::Metadata');\n    if (modifiedResources.length > 0) {\n      modified = this.printSectionHeader('Modified Resources');\n\n      for (const drift of modifiedResources) {\n        if (!drift.LogicalResourceId || !drift.ResourceType) continue;\n        modified += `${UPDATE} ${chalk.cyan(drift.ResourceType)} ${this.formatLogicalId(logicalToPathMap, drift.LogicalResourceId)}\\n`;\n        if (drift.PropertyDifferences) {\n          const propDiffs = drift.PropertyDifferences;\n          for (let i = 0; i < propDiffs.length; i++) {\n            const diff = propDiffs[i];\n            if (!diff.PropertyPath) continue;\n            const difference = new Difference(diff.ExpectedValue, diff.ActualValue);\n            modified += this.formatTreeDiff(diff.PropertyPath, difference, i === propDiffs.length - 1);\n          }\n        }\n      }\n      modified += this.printSectionFooter();\n    }\n\n    // Process deleted resources (exclude AWS::CDK::Metadata)\n    const deletedResources = drifts.filter(d =>\n      d.StackResourceDriftStatus === StackResourceDriftStatus.DELETED\n      && d.ResourceType !== 'AWS::CDK::Metadata');\n    if (deletedResources.length > 0) {\n      deleted = this.printSectionHeader('Deleted Resources');\n      for (const drift of deletedResources) {\n        if (!drift.LogicalResourceId || !drift.ResourceType) continue;\n        deleted += `${REMOVAL} ${chalk.cyan(drift.ResourceType)} ${this.formatLogicalId(logicalToPathMap, drift.LogicalResourceId)}\\n`;\n      }\n      deleted += this.printSectionFooter();\n    }\n\n    return { unchanged, unchecked, modified, deleted };\n  }\n\n  private formatLogicalId(logicalToPathMap: { [logicalId: string]: string }, logicalId: string): string {\n    const path = logicalToPathMap[logicalId];\n    if (!path) return logicalId;\n\n    let normalizedPath = path;\n    if (normalizedPath.startsWith('/')) {\n      normalizedPath = normalizedPath.slice(1);\n    }\n\n    let parts = normalizedPath.split('/');\n    if (parts.length > 1) {\n      parts = parts.slice(1);\n\n      // remove the last component if it's \"Resource\" or \"Default\" (if we have more than a single component)\n      if (parts.length > 1) {\n        const last = parts[parts.length - 1];\n        if (last === 'Resource' || last === 'Default') {\n          parts = parts.slice(0, parts.length - 1);\n        }\n      }\n\n      normalizedPath = parts.join('/');\n    }\n\n    return `${normalizedPath} ${chalk.gray(logicalId)}`;\n  }\n\n  private printSectionHeader(title: string): string {\n    return `${chalk.underline(chalk.bold(title))}\\n`;\n  }\n\n  private printSectionFooter(): string {\n    return '\\n';\n  }\n\n  private formatTreeDiff(propertyPath: string, difference: Difference<string>, isLast: boolean): string {\n    let result = format(' %s─ %s %s\\n', isLast ? '└' : '├',\n      difference.isAddition ? ADDITION :\n        difference.isRemoval ? REMOVAL :\n          UPDATE,\n      propertyPath,\n    );\n    if (difference.isUpdate) {\n      result += format('     ├─ %s %s\\n', REMOVAL, chalk.red(difference.oldValue));\n      result += format('     └─ %s %s\\n', ADDITION, chalk.green(difference.newValue));\n    }\n    return result;\n  }\n}\n\nconst ADDITION = chalk.green('[+]');\nconst CONTEXT = chalk.grey('[ ]');\nconst UPDATE = chalk.yellow('[~]');\nconst REMOVAL = chalk.red('[-]');\n"]}
@@ -1,8 +1,9 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
3
  exports.detectStackDrift = detectStackDrift;
4
- const util_1 = require("util");
4
+ const node_util_1 = require("node:util");
5
5
  const toolkit_error_1 = require("../../toolkit/toolkit-error");
6
+ const string_manipulation_1 = require("../../util/string-manipulation");
6
7
  /**
7
8
  * Detect drift for a CloudFormation stack and wait for the detection to complete
8
9
  *
@@ -16,19 +17,25 @@ async function detectStackDrift(cfn, ioHelper, stackName) {
16
17
  const driftDetection = await cfn.detectStackDrift({
17
18
  StackName: stackName,
18
19
  });
19
- await ioHelper.defaults.trace((0, util_1.format)('Detecting drift with ID %s for stack %s...', driftDetection.StackDriftDetectionId, stackName));
20
+ await ioHelper.defaults.trace((0, node_util_1.format)('Detecting drift with ID %s for stack %s...', driftDetection.StackDriftDetectionId, stackName));
20
21
  // Wait for drift detection to complete
21
22
  const driftStatus = await waitForDriftDetection(cfn, ioHelper, driftDetection.StackDriftDetectionId);
22
- if (!driftStatus) {
23
- throw new toolkit_error_1.ToolkitError('Drift detection took too long to complete. Aborting');
23
+ // Handle UNKNOWN stack drift status
24
+ if (driftStatus?.StackDriftStatus === 'UNKNOWN') {
25
+ await ioHelper.defaults.trace('Stack drift status is UNKNOWN. This may occur when CloudFormation is unable to detect drift for at least one resource and all other resources are IN_SYNC.\n' +
26
+ `Reason: ${(0, string_manipulation_1.formatReason)(driftStatus.DetectionStatusReason)}`);
24
27
  }
25
- if (driftStatus?.DetectionStatus === 'DETECTION_FAILED') {
26
- throw new toolkit_error_1.ToolkitError(`Failed to detect drift: ${driftStatus.DetectionStatusReason || 'No reason provided'}`);
27
- }
28
- // Get the drift results
29
- return cfn.describeStackResourceDrifts({
28
+ // Get the drift results, including resources with UNKNOWN status
29
+ const driftResults = await cfn.describeStackResourceDrifts({
30
30
  StackName: stackName,
31
31
  });
32
+ // Log warning for any resources with UNKNOWN status
33
+ const unknownResources = driftResults.StackResourceDrifts?.filter(drift => drift.StackResourceDriftStatus === 'UNKNOWN');
34
+ if (unknownResources && unknownResources.length > 0) {
35
+ await ioHelper.defaults.trace('Some resources have UNKNOWN drift status. This may be due to insufficient permissions or throttling:\n' +
36
+ unknownResources.map(r => ` - ${r.LogicalResourceId}: ${(0, string_manipulation_1.formatReason)(r.DriftStatusReason)}`).join('\n'));
37
+ }
38
+ return driftResults;
32
39
  }
33
40
  /**
34
41
  * Wait for a drift detection operation to complete
@@ -47,7 +54,7 @@ async function waitForDriftDetection(cfn, ioHelper, driftDetectionId) {
47
54
  return response;
48
55
  }
49
56
  if (response.DetectionStatus === 'DETECTION_FAILED') {
50
- throw new toolkit_error_1.ToolkitError(`Drift detection failed: ${response.DetectionStatusReason}`);
57
+ throw new toolkit_error_1.ToolkitError(`Drift detection failed: ${(0, string_manipulation_1.formatReason)(response.DetectionStatusReason)}`);
51
58
  }
52
59
  if (Date.now() > deadline) {
53
60
  throw new toolkit_error_1.ToolkitError(`Drift detection failed: Timed out after ${maxWaitForDrift / 1000} seconds.`);
@@ -60,4 +67,4 @@ async function waitForDriftDetection(cfn, ioHelper, driftDetectionId) {
60
67
  await new Promise(resolve => setTimeout(resolve, timeBetweenApiCalls));
61
68
  }
62
69
  }
63
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZHJpZnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJkcmlmdC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOztBQWNBLDRDQStCQztBQTdDRCwrQkFBOEI7QUFFOUIsK0RBQTJEO0FBSTNEOzs7Ozs7O0dBT0c7QUFDSSxLQUFLLFVBQVUsZ0JBQWdCLENBQ3BDLEdBQTBCLEVBQzFCLFFBQWtCLEVBQ2xCLFNBQWlCO0lBRWpCLHdCQUF3QjtJQUN4QixNQUFNLGNBQWMsR0FBRyxNQUFNLEdBQUcsQ0FBQyxnQkFBZ0IsQ0FBQztRQUNoRCxTQUFTLEVBQUUsU0FBUztLQUNyQixDQUFDLENBQUM7SUFFSCxNQUFNLFFBQVEsQ0FBQyxRQUFRLENBQUMsS0FBSyxDQUMzQixJQUFBLGFBQU0sRUFBQyw0Q0FBNEMsRUFBRSxjQUFjLENBQUMscUJBQXFCLEVBQUUsU0FBUyxDQUFDLENBQ3RHLENBQUM7SUFFRix1Q0FBdUM7SUFDdkMsTUFBTSxXQUFXLEdBQUcsTUFBTSxxQkFBcUIsQ0FBQyxHQUFHLEVBQUUsUUFBUSxFQUFFLGNBQWMsQ0FBQyxxQkFBc0IsQ0FBQyxDQUFDO0lBRXRHLElBQUksQ0FBQyxXQUFXLEVBQUUsQ0FBQztRQUNqQixNQUFNLElBQUksNEJBQVksQ0FBQyxxREFBcUQsQ0FBQyxDQUFDO0lBQ2hGLENBQUM7SUFFRCxJQUFJLFdBQVcsRUFBRSxlQUFlLEtBQUssa0JBQWtCLEVBQUUsQ0FBQztRQUN4RCxNQUFNLElBQUksNEJBQVksQ0FDcEIsMkJBQTJCLFdBQVcsQ0FBQyxxQkFBcUIsSUFBSSxvQkFBb0IsRUFBRSxDQUN2RixDQUFDO0lBQ0osQ0FBQztJQUVELHdCQUF3QjtJQUN4QixPQUFPLEdBQUcsQ0FBQywyQkFBMkIsQ0FBQztRQUNyQyxTQUFTLEVBQUUsU0FBUztLQUNyQixDQUFDLENBQUM7QUFDTCxDQUFDO0FBRUQ7O0dBRUc7QUFDSCxLQUFLLFVBQVUscUJBQXFCLENBQ2xDLEdBQTBCLEVBQzFCLFFBQWtCLEVBQ2xCLGdCQUF3QjtJQUV4QixNQUFNLGVBQWUsR0FBRyxPQUFPLENBQUMsQ0FBQyxrQ0FBa0M7SUFDbkUsTUFBTSxrQkFBa0IsR0FBRyxNQUFNLENBQUMsQ0FBQyw0REFBNEQ7SUFDL0YsTUFBTSxtQkFBbUIsR0FBRyxLQUFLLENBQUMsQ0FBQyx1QkFBdUI7SUFDMUQsTUFBTSxRQUFRLEdBQUcsSUFBSSxDQUFDLEdBQUcsRUFBRSxHQUFHLGVBQWUsQ0FBQztJQUM5QyxJQUFJLE9BQU8sR0FBRyxJQUFJLENBQUMsR0FBRyxFQUFFLEdBQUcsa0JBQWtCLENBQUM7SUFFOUMsT0FBTyxJQUFJLEVBQUUsQ0FBQztRQUNaLE1BQU0sUUFBUSxHQUFHLE1BQU0sR0FBRyxDQUFDLGlDQUFpQyxDQUFDO1lBQzNELHFCQUFxQixFQUFFLGdCQUFnQjtTQUN4QyxDQUFDLENBQUM7UUFFSCxJQUFJLFFBQVEsQ0FBQyxlQUFlLEtBQUssb0JBQW9CLEVBQUUsQ0FBQztZQUN0RCxPQUFPLFFBQVEsQ0FBQztRQUNsQixDQUFDO1FBRUQsSUFBSSxRQUFRLENBQUMsZUFBZSxLQUFLLGtCQUFrQixFQUFFLENBQUM7WUFDcEQsTUFBTSxJQUFJLDRCQUFZLENBQUMsMkJBQTJCLFFBQVEsQ0FBQyxxQkFBcUIsRUFBRSxDQUFDLENBQUM7UUFDdEYsQ0FBQztRQUVELElBQUksSUFBSSxDQUFDLEdBQUcsRUFBRSxHQUFHLFFBQVEsRUFBRSxDQUFDO1lBQzFCLE1BQU0sSUFBSSw0QkFBWSxDQUFDLDJDQUEyQyxlQUFlLEdBQUcsSUFBSSxXQUFXLENBQUMsQ0FBQztRQUN2RyxDQUFDO1FBRUQsSUFBSSxJQUFJLENBQUMsR0FBRyxFQUFFLEdBQUcsT0FBTyxFQUFFLENBQUM7WUFDekIsTUFBTSxRQUFRLENBQUMsUUFBUSxDQUFDLEtBQUssQ0FBQyw0Q0FBNEMsQ0FBQyxDQUFDO1lBQzVFLE9BQU8sR0FBRyxJQUFJLENBQUMsR0FBRyxFQUFFLEdBQUcsa0JBQWtCLENBQUM7UUFDNUMsQ0FBQztRQUVELGtFQUFrRTtRQUNsRSxNQUFNLElBQUksT0FBTyxDQUFDLE9BQU8sQ0FBQyxFQUFFLENBQUMsVUFBVSxDQUFDLE9BQU8sRUFBRSxtQkFBbUIsQ0FBQyxDQUFDLENBQUM7SUFDekUsQ0FBQztBQUNILENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBmb3JtYXQgfSBmcm9tICd1dGlsJztcbmltcG9ydCB0eXBlIHsgRGVzY3JpYmVTdGFja0RyaWZ0RGV0ZWN0aW9uU3RhdHVzQ29tbWFuZE91dHB1dCwgRGVzY3JpYmVTdGFja1Jlc291cmNlRHJpZnRzQ29tbWFuZE91dHB1dCB9IGZyb20gJ0Bhd3Mtc2RrL2NsaWVudC1jbG91ZGZvcm1hdGlvbic7XG5pbXBvcnQgeyBUb29sa2l0RXJyb3IgfSBmcm9tICcuLi8uLi90b29sa2l0L3Rvb2xraXQtZXJyb3InO1xuaW1wb3J0IHR5cGUgeyBJQ2xvdWRGb3JtYXRpb25DbGllbnQgfSBmcm9tICcuLi9hd3MtYXV0aC9wcml2YXRlJztcbmltcG9ydCB0eXBlIHsgSW9IZWxwZXIgfSBmcm9tICcuLi9pby9wcml2YXRlJztcblxuLyoqXG4gKiBEZXRlY3QgZHJpZnQgZm9yIGEgQ2xvdWRGb3JtYXRpb24gc3RhY2sgYW5kIHdhaXQgZm9yIHRoZSBkZXRlY3Rpb24gdG8gY29tcGxldGVcbiAqXG4gKiBAcGFyYW0gY2ZuIC0gYSBDbG91ZEZvcm1hdGlvbiBjbGllbnRcbiAqIEBwYXJhbSBpb0hlbHBlciAtIGhlbHBlciBmb3IgSU8gb3BlcmF0aW9uc1xuICogQHBhcmFtIHN0YWNrTmFtZSAtIHRoZSBuYW1lIG9mIHRoZSBzdGFjayB0byBjaGVjayBmb3IgZHJpZnRcbiAqIEByZXR1cm5zIHRoZSBDbG91ZEZvcm1hdGlvbiBkZXNjcmlwdGlvbiBvZiB0aGUgZHJpZnQgZGV0ZWN0aW9uIHJlc3VsdHNcbiAqL1xuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIGRldGVjdFN0YWNrRHJpZnQoXG4gIGNmbjogSUNsb3VkRm9ybWF0aW9uQ2xpZW50LFxuICBpb0hlbHBlcjogSW9IZWxwZXIsXG4gIHN0YWNrTmFtZTogc3RyaW5nLFxuKTogUHJvbWlzZTxEZXNjcmliZVN0YWNrUmVzb3VyY2VEcmlmdHNDb21tYW5kT3V0cHV0PiB7XG4gIC8vIFN0YXJ0IGRyaWZ0IGRldGVjdGlvblxuICBjb25zdCBkcmlmdERldGVjdGlvbiA9IGF3YWl0IGNmbi5kZXRlY3RTdGFja0RyaWZ0KHtcbiAgICBTdGFja05hbWU6IHN0YWNrTmFtZSxcbiAgfSk7XG5cbiAgYXdhaXQgaW9IZWxwZXIuZGVmYXVsdHMudHJhY2UoXG4gICAgZm9ybWF0KCdEZXRlY3RpbmcgZHJpZnQgd2l0aCBJRCAlcyBmb3Igc3RhY2sgJXMuLi4nLCBkcmlmdERldGVjdGlvbi5TdGFja0RyaWZ0RGV0ZWN0aW9uSWQsIHN0YWNrTmFtZSksXG4gICk7XG5cbiAgLy8gV2FpdCBmb3IgZHJpZnQgZGV0ZWN0aW9uIHRvIGNvbXBsZXRlXG4gIGNvbnN0IGRyaWZ0U3RhdHVzID0gYXdhaXQgd2FpdEZvckRyaWZ0RGV0ZWN0aW9uKGNmbiwgaW9IZWxwZXIsIGRyaWZ0RGV0ZWN0aW9uLlN0YWNrRHJpZnREZXRlY3Rpb25JZCEpO1xuXG4gIGlmICghZHJpZnRTdGF0dXMpIHtcbiAgICB0aHJvdyBuZXcgVG9vbGtpdEVycm9yKCdEcmlmdCBkZXRlY3Rpb24gdG9vayB0b28gbG9uZyB0byBjb21wbGV0ZS4gQWJvcnRpbmcnKTtcbiAgfVxuXG4gIGlmIChkcmlmdFN0YXR1cz8uRGV0ZWN0aW9uU3RhdHVzID09PSAnREVURUNUSU9OX0ZBSUxFRCcpIHtcbiAgICB0aHJvdyBuZXcgVG9vbGtpdEVycm9yKFxuICAgICAgYEZhaWxlZCB0byBkZXRlY3QgZHJpZnQ6ICR7ZHJpZnRTdGF0dXMuRGV0ZWN0aW9uU3RhdHVzUmVhc29uIHx8ICdObyByZWFzb24gcHJvdmlkZWQnfWAsXG4gICAgKTtcbiAgfVxuXG4gIC8vIEdldCB0aGUgZHJpZnQgcmVzdWx0c1xuICByZXR1cm4gY2ZuLmRlc2NyaWJlU3RhY2tSZXNvdXJjZURyaWZ0cyh7XG4gICAgU3RhY2tOYW1lOiBzdGFja05hbWUsXG4gIH0pO1xufVxuXG4vKipcbiAqIFdhaXQgZm9yIGEgZHJpZnQgZGV0ZWN0aW9uIG9wZXJhdGlvbiB0byBjb21wbGV0ZVxuICovXG5hc3luYyBmdW5jdGlvbiB3YWl0Rm9yRHJpZnREZXRlY3Rpb24oXG4gIGNmbjogSUNsb3VkRm9ybWF0aW9uQ2xpZW50LFxuICBpb0hlbHBlcjogSW9IZWxwZXIsXG4gIGRyaWZ0RGV0ZWN0aW9uSWQ6IHN0cmluZyxcbik6IFByb21pc2U8RGVzY3JpYmVTdGFja0RyaWZ0RGV0ZWN0aW9uU3RhdHVzQ29tbWFuZE91dHB1dCB8IHVuZGVmaW5lZD4ge1xuICBjb25zdCBtYXhXYWl0Rm9yRHJpZnQgPSAzMDBfMDAwOyAvLyBpZiB0YWtlcyBsb25nZXIgdGhhbiA1bWluLCBmYWlsXG4gIGNvbnN0IHRpbWVCZXR3ZWVuT3V0cHV0cyA9IDEwXzAwMDsgLy8gaG93IGxvbmcgdG8gd2FpdCBiZWZvcmUgdGVsbGluZyB1c2VyIHdlJ3JlIHN0aWxsIGNoZWNraW5nXG4gIGNvbnN0IHRpbWVCZXR3ZWVuQXBpQ2FsbHMgPSAyXzAwMDsgLy8gd2FpdCAycyBwZXIgQVBJIGNhbGxcbiAgY29uc3QgZGVhZGxpbmUgPSBEYXRlLm5vdygpICsgbWF4V2FpdEZvckRyaWZ0O1xuICBsZXQgY2hlY2tJbiA9IERhdGUubm93KCkgKyB0aW1lQmV0d2Vlbk91dHB1dHM7XG5cbiAgd2hpbGUgKHRydWUpIHtcbiAgICBjb25zdCByZXNwb25zZSA9IGF3YWl0IGNmbi5kZXNjcmliZVN0YWNrRHJpZnREZXRlY3Rpb25TdGF0dXMoe1xuICAgICAgU3RhY2tEcmlmdERldGVjdGlvbklkOiBkcmlmdERldGVjdGlvbklkLFxuICAgIH0pO1xuXG4gICAgaWYgKHJlc3BvbnNlLkRldGVjdGlvblN0YXR1cyA9PT0gJ0RFVEVDVElPTl9DT01QTEVURScpIHtcbiAgICAgIHJldHVybiByZXNwb25zZTtcbiAgICB9XG5cbiAgICBpZiAocmVzcG9uc2UuRGV0ZWN0aW9uU3RhdHVzID09PSAnREVURUNUSU9OX0ZBSUxFRCcpIHtcbiAgICAgIHRocm93IG5ldyBUb29sa2l0RXJyb3IoYERyaWZ0IGRldGVjdGlvbiBmYWlsZWQ6ICR7cmVzcG9uc2UuRGV0ZWN0aW9uU3RhdHVzUmVhc29ufWApO1xuICAgIH1cblxuICAgIGlmIChEYXRlLm5vdygpID4gZGVhZGxpbmUpIHtcbiAgICAgIHRocm93IG5ldyBUb29sa2l0RXJyb3IoYERyaWZ0IGRldGVjdGlvbiBmYWlsZWQ6IFRpbWVkIG91dCBhZnRlciAke21heFdhaXRGb3JEcmlmdCAvIDEwMDB9IHNlY29uZHMuYCk7XG4gICAgfVxuXG4gICAgaWYgKERhdGUubm93KCkgPiBjaGVja0luKSB7XG4gICAgICBhd2FpdCBpb0hlbHBlci5kZWZhdWx0cy50cmFjZSgnV2FpdGluZyBmb3IgZHJpZnQgZGV0ZWN0aW9uIHRvIGNvbXBsZXRlLi4uJyk7XG4gICAgICBjaGVja0luID0gRGF0ZS5ub3coKSArIHRpbWVCZXR3ZWVuT3V0cHV0cztcbiAgICB9XG5cbiAgICAvLyBXYWl0IGEgc2hvcnQgd2hpbGUgYmV0d2VlbiBBUEkgY2FsbHMgc28gd2UgZG9uJ3QgY3JlYXRlIGEgZmxvb2RcbiAgICBhd2FpdCBuZXcgUHJvbWlzZShyZXNvbHZlID0+IHNldFRpbWVvdXQocmVzb2x2ZSwgdGltZUJldHdlZW5BcGlDYWxscykpO1xuICB9XG59XG4iXX0=
70
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"drift.js","sourceRoot":"","sources":["drift.ts"],"names":[],"mappings":"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","sourcesContent":["import { format } from 'node:util';\nimport type { DescribeStackDriftDetectionStatusCommandOutput, DescribeStackResourceDriftsCommandOutput } from '@aws-sdk/client-cloudformation';\nimport { ToolkitError } from '../../toolkit/toolkit-error';\nimport { formatReason } from '../../util/string-manipulation';\nimport type { ICloudFormationClient } from '../aws-auth/private';\nimport type { IoHelper } from '../io/private';\n\n/**\n * Detect drift for a CloudFormation stack and wait for the detection to complete\n *\n * @param cfn - a CloudFormation client\n * @param ioHelper - helper for IO operations\n * @param stackName - the name of the stack to check for drift\n * @returns the CloudFormation description of the drift detection results\n */\nexport async function detectStackDrift(\n  cfn: ICloudFormationClient,\n  ioHelper: IoHelper,\n  stackName: string,\n): Promise<DescribeStackResourceDriftsCommandOutput> {\n  // Start drift detection\n  const driftDetection = await cfn.detectStackDrift({\n    StackName: stackName,\n  });\n\n  await ioHelper.defaults.trace(\n    format('Detecting drift with ID %s for stack %s...', driftDetection.StackDriftDetectionId, stackName),\n  );\n\n  // Wait for drift detection to complete\n  const driftStatus = await waitForDriftDetection(cfn, ioHelper, driftDetection.StackDriftDetectionId!);\n\n  // Handle UNKNOWN stack drift status\n  if (driftStatus?.StackDriftStatus === 'UNKNOWN') {\n    await ioHelper.defaults.trace(\n      'Stack drift status is UNKNOWN. This may occur when CloudFormation is unable to detect drift for at least one resource and all other resources are IN_SYNC.\\n' +\n      `Reason: ${formatReason(driftStatus.DetectionStatusReason)}`,\n    );\n  }\n\n  // Get the drift results, including resources with UNKNOWN status\n  const driftResults = await cfn.describeStackResourceDrifts({\n    StackName: stackName,\n  });\n\n  // Log warning for any resources with UNKNOWN status\n  const unknownResources = driftResults.StackResourceDrifts?.filter(\n    drift => drift.StackResourceDriftStatus === 'UNKNOWN',\n  );\n\n  if (unknownResources && unknownResources.length > 0) {\n    await ioHelper.defaults.trace(\n      'Some resources have UNKNOWN drift status. This may be due to insufficient permissions or throttling:\\n' +\n      unknownResources.map(r =>\n        `  - ${r.LogicalResourceId}: ${formatReason(r.DriftStatusReason)}`,\n      ).join('\\n'),\n    );\n  }\n\n  return driftResults;\n}\n\n/**\n * Wait for a drift detection operation to complete\n */\nasync function waitForDriftDetection(\n  cfn: ICloudFormationClient,\n  ioHelper: IoHelper,\n  driftDetectionId: string,\n): Promise<DescribeStackDriftDetectionStatusCommandOutput | undefined> {\n  const maxWaitForDrift = 300_000; // if takes longer than 5min, fail\n  const timeBetweenOutputs = 10_000; // how long to wait before telling user we're still checking\n  const timeBetweenApiCalls = 2_000; // wait 2s per API call\n  const deadline = Date.now() + maxWaitForDrift;\n  let checkIn = Date.now() + timeBetweenOutputs;\n\n  while (true) {\n    const response = await cfn.describeStackDriftDetectionStatus({\n      StackDriftDetectionId: driftDetectionId,\n    });\n\n    if (response.DetectionStatus === 'DETECTION_COMPLETE') {\n      return response;\n    }\n\n    if (response.DetectionStatus === 'DETECTION_FAILED') {\n      throw new ToolkitError(`Drift detection failed: ${formatReason(response.DetectionStatusReason)}`);\n    }\n\n    if (Date.now() > deadline) {\n      throw new ToolkitError(`Drift detection failed: Timed out after ${maxWaitForDrift / 1000} seconds.`);\n    }\n\n    if (Date.now() > checkIn) {\n      await ioHelper.defaults.trace('Waiting for drift detection to complete...');\n      checkIn = Date.now() + timeBetweenOutputs;\n    }\n\n    // Wait a short while between API calls so we don't create a flood\n    await new Promise(resolve => setTimeout(resolve, timeBetweenApiCalls));\n  }\n}\n"]}
package/lib/index_bg.wasm CHANGED
Binary file