Socket
Socket
Sign inDemoInstall

aws-cdk

Package Overview
Dependencies
Maintainers
3
Versions
553
Alerts
File Explorer

Advanced tools

Socket logo

Install Socket

Detect and block malicious and high-risk dependencies

Install

aws-cdk - npm Package Compare versions

Comparing version 2.118.0 to 2.119.0

lib/api/util/template-body-parameter.d.ts

4

build-info.json
{
"comment": "Generated at 2024-01-03T17:53:31Z by generate.sh",
"commit": "a40f2ec"
"comment": "Generated at 2024-01-11T20:09:20Z by generate.sh",
"commit": "0392e71"
}

@@ -29,5 +29,11 @@ import * as cxapi from '@aws-cdk/cx-api';

/**
* The behavior if if no selectors are privided.
* The behavior if if no selectors are provided.
*/
defaultBehavior: DefaultSelection;
/**
* Whether to deploy if the app contains no stacks.
*
* @default false
*/
ignoreNoStacks?: boolean;
}

@@ -34,0 +40,0 @@ /**

@@ -65,2 +65,5 @@ "use strict";

if (stacks.length === 0) {
if (options.ignoreNoStacks) {
return new StackCollection(this, []);
}
throw new Error('This app contains no stacks');

@@ -283,2 +286,2 @@ }

}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cloud-assembly.js","sourceRoot":"","sources":["cloud-assembly.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cxapi from '@aws-cdk/cx-api';\nimport * as chalk from 'chalk';\nimport { minimatch } from 'minimatch';\nimport * as semver from 'semver';\nimport { error, print, warning } from '../../logging';\nimport { flatten } from '../../util';\nimport { versionNumber } from '../../version';\n\nexport enum DefaultSelection {\n  /**\n   * Returns an empty selection in case there are no selectors.\n   */\n  None = 'none',\n\n  /**\n   * If the app includes a single stack, returns it. Otherwise throws an exception.\n   * This behavior is used by \"deploy\".\n   */\n  OnlySingle = 'single',\n\n  /**\n   * Returns all stacks in the main (top level) assembly only.\n   */\n  MainAssembly = 'main',\n\n  /**\n   * If no selectors are provided, returns all stacks in the app,\n   * including stacks inside nested assemblies.\n   */\n  AllStacks = 'all',\n}\n\nexport interface SelectStacksOptions {\n  /**\n   * Extend the selection to upstread/downstream stacks\n   * @default ExtendedStackSelection.None only select the specified stacks.\n   */\n  extend?: ExtendedStackSelection;\n\n  /**\n   * The behavior if if no selectors are privided.\n   */\n  defaultBehavior: DefaultSelection;\n}\n\n/**\n * When selecting stacks, what other stacks to include because of dependencies\n */\nexport enum ExtendedStackSelection {\n  /**\n   * Don't select any extra stacks\n   */\n  None,\n\n  /**\n   * Include stacks that this stack depends on\n   */\n  Upstream,\n\n  /**\n   * Include stacks that depend on this stack\n   */\n  Downstream\n}\n\n/**\n * A specification of which stacks should be selected\n */\nexport interface StackSelector {\n  /**\n   * Whether all stacks at the top level assembly should\n   * be selected and nothing else\n   */\n  allTopLevel?: boolean,\n\n  /**\n   * A list of patterns to match the stack hierarchical ids\n   */\n  patterns: string[],\n}\n\n/**\n * A single Cloud Assembly and the operations we do on it to deploy the artifacts inside\n */\nexport class CloudAssembly {\n  /**\n   * The directory this CloudAssembly was read from\n   */\n  public readonly directory: string;\n\n  constructor(public readonly assembly: cxapi.CloudAssembly) {\n    this.directory = assembly.directory;\n  }\n\n  public async selectStacks(selector: StackSelector, options: SelectStacksOptions): Promise<StackCollection> {\n    const asm = this.assembly;\n    const topLevelStacks = asm.stacks;\n    const stacks = semver.major(asm.version) < 10 ? asm.stacks : asm.stacksRecursively;\n    const allTopLevel = selector.allTopLevel ?? false;\n    const patterns = sanitizePatterns(selector.patterns);\n\n    if (stacks.length === 0) {\n      throw new Error('This app contains no stacks');\n    }\n\n    if (allTopLevel) {\n      return this.selectTopLevelStacks(stacks, topLevelStacks, options.extend);\n    } else if (patterns.length > 0) {\n      return this.selectMatchingStacks(stacks, patterns, options.extend);\n    } else {\n      return this.selectDefaultStacks(stacks, topLevelStacks, options.defaultBehavior);\n    }\n  }\n\n  private selectTopLevelStacks(stacks: cxapi.CloudFormationStackArtifact[],\n    topLevelStacks: cxapi.CloudFormationStackArtifact[],\n    extend: ExtendedStackSelection = ExtendedStackSelection.None): StackCollection {\n    if (topLevelStacks.length > 0) {\n      return this.extendStacks(topLevelStacks, stacks, extend);\n    } else {\n      throw new Error('No stack found in the main cloud assembly. Use \"list\" to print manifest');\n    }\n  }\n\n  private selectMatchingStacks(stacks: cxapi.CloudFormationStackArtifact[],\n    patterns: string[],\n    extend: ExtendedStackSelection = ExtendedStackSelection.None): StackCollection {\n\n    // cli tests use this to ensure tests do not depend on legacy behavior\n    // (otherwise they will fail in v2)\n    const disableLegacy = process.env.CXAPI_DISABLE_SELECT_BY_ID === '1';\n\n    const matchingPattern = (pattern: string) => (stack: cxapi.CloudFormationStackArtifact) => {\n      if (minimatch(stack.hierarchicalId, pattern)) {\n        return true;\n      } else if (!disableLegacy && stack.id === pattern && semver.major(versionNumber()) < 2) {\n        warning('Selecting stack by identifier \"%s\". This identifier is deprecated and will be removed in v2. Please use \"%s\" instead.', chalk.bold(stack.id), chalk.bold(stack.hierarchicalId));\n        warning('Run \"cdk ls\" to see a list of all stack identifiers');\n        return true;\n      }\n      return false;\n    };\n\n    const matchedStacks = flatten(patterns.map(pattern => stacks.filter(matchingPattern(pattern))));\n\n    return this.extendStacks(matchedStacks, stacks, extend);\n  }\n\n  private selectDefaultStacks(stacks: cxapi.CloudFormationStackArtifact[],\n    topLevelStacks: cxapi.CloudFormationStackArtifact[],\n    defaultSelection: DefaultSelection) {\n    switch (defaultSelection) {\n      case DefaultSelection.MainAssembly:\n        return new StackCollection(this, topLevelStacks);\n      case DefaultSelection.AllStacks:\n        return new StackCollection(this, stacks);\n      case DefaultSelection.None:\n        return new StackCollection(this, []);\n      case DefaultSelection.OnlySingle:\n        if (topLevelStacks.length === 1) {\n          return new StackCollection(this, topLevelStacks);\n        } else {\n          throw new Error('Since this app includes more than a single stack, specify which stacks to use (wildcards are supported) or specify `--all`\\n' +\n          `Stacks: ${stacks.map(x => x.hierarchicalId).join(' · ')}`);\n        }\n      default:\n        throw new Error(`invalid default behavior: ${defaultSelection}`);\n    }\n  }\n\n  private extendStacks(matched: cxapi.CloudFormationStackArtifact[],\n    all: cxapi.CloudFormationStackArtifact[],\n    extend: ExtendedStackSelection = ExtendedStackSelection.None) {\n    const allStacks = new Map<string, cxapi.CloudFormationStackArtifact>();\n    for (const stack of all) {\n      allStacks.set(stack.hierarchicalId, stack);\n    }\n\n    const index = indexByHierarchicalId(matched);\n\n    switch (extend) {\n      case ExtendedStackSelection.Downstream:\n        includeDownstreamStacks(index, allStacks);\n        break;\n      case ExtendedStackSelection.Upstream:\n        includeUpstreamStacks(index, allStacks);\n        break;\n    }\n\n    // Filter original array because it is in the right order\n    const selectedList = all.filter(s => index.has(s.hierarchicalId));\n\n    return new StackCollection(this, selectedList);\n  }\n\n  /**\n   * Select a single stack by its ID\n   */\n  public stackById(stackId: string) {\n    return new StackCollection(this, [this.assembly.getStackArtifact(stackId)]);\n  }\n}\n\n/**\n * A collection of stacks and related artifacts\n *\n * In practice, not all artifacts in the CloudAssembly are created equal;\n * stacks can be selected independently, but other artifacts such as asset\n * bundles cannot.\n */\nexport class StackCollection {\n  constructor(public readonly assembly: CloudAssembly, public readonly stackArtifacts: cxapi.CloudFormationStackArtifact[]) {\n  }\n\n  public get stackCount() {\n    return this.stackArtifacts.length;\n  }\n\n  public get firstStack() {\n    if (this.stackCount < 1) {\n      throw new Error('StackCollection contains no stack artifacts (trying to access the first one)');\n    }\n    return this.stackArtifacts[0];\n  }\n\n  public get stackIds(): string[] {\n    return this.stackArtifacts.map(s => s.id);\n  }\n\n  public reversed() {\n    const arts = [...this.stackArtifacts];\n    arts.reverse();\n    return new StackCollection(this.assembly, arts);\n  }\n\n  public filter(predicate: (art: cxapi.CloudFormationStackArtifact) => boolean): StackCollection {\n    return new StackCollection(this.assembly, this.stackArtifacts.filter(predicate));\n  }\n\n  public concat(other: StackCollection): StackCollection {\n    return new StackCollection(this.assembly, this.stackArtifacts.concat(other.stackArtifacts));\n  }\n\n  /**\n   * Extracts 'aws:cdk:warning|info|error' metadata entries from the stack synthesis\n   */\n  public processMetadataMessages(options: MetadataMessageOptions = {}) {\n    let warnings = false;\n    let errors = false;\n\n    for (const stack of this.stackArtifacts) {\n      for (const message of stack.messages) {\n        switch (message.level) {\n          case cxapi.SynthesisMessageLevel.WARNING:\n            warnings = true;\n            printMessage(warning, 'Warning', message.id, message.entry);\n            break;\n          case cxapi.SynthesisMessageLevel.ERROR:\n            errors = true;\n            printMessage(error, 'Error', message.id, message.entry);\n            break;\n          case cxapi.SynthesisMessageLevel.INFO:\n            printMessage(print, 'Info', message.id, message.entry);\n            break;\n        }\n      }\n    }\n\n    if (errors && !options.ignoreErrors) {\n      throw new Error('Found errors');\n    }\n\n    if (options.strict && warnings) {\n      throw new Error('Found warnings (--strict mode)');\n    }\n\n    function printMessage(logFn: (s: string) => void, prefix: string, id: string, entry: cxapi.MetadataEntry) {\n      logFn(`[${prefix} at ${id}] ${entry.data}`);\n\n      if (options.verbose && entry.trace) {\n        logFn(`  ${entry.trace.join('\\n  ')}`);\n      }\n    }\n  }\n}\n\nexport interface MetadataMessageOptions {\n  /**\n   * Whether to be verbose\n   *\n   * @default false\n   */\n  verbose?: boolean;\n\n  /**\n   * Don't stop on error metadata\n   *\n   * @default false\n   */\n  ignoreErrors?: boolean;\n\n  /**\n   * Treat warnings in metadata as errors\n   *\n   * @default false\n   */\n  strict?: boolean;\n}\n\nfunction indexByHierarchicalId(stacks: cxapi.CloudFormationStackArtifact[]): Map<string, cxapi.CloudFormationStackArtifact> {\n  const result = new Map<string, cxapi.CloudFormationStackArtifact>();\n\n  for (const stack of stacks) {\n    result.set(stack.hierarchicalId, stack);\n  }\n\n  return result;\n}\n\n/**\n * Calculate the transitive closure of stack dependents.\n *\n * Modifies `selectedStacks` in-place.\n */\nfunction includeDownstreamStacks(\n  selectedStacks: Map<string, cxapi.CloudFormationStackArtifact>,\n  allStacks: Map<string, cxapi.CloudFormationStackArtifact>) {\n  const added = new Array<string>();\n\n  let madeProgress;\n  do {\n    madeProgress = false;\n\n    for (const [id, stack] of allStacks) {\n      // Select this stack if it's not selected yet AND it depends on a stack that's in the selected set\n      if (!selectedStacks.has(id) && (stack.dependencies || []).some(dep => selectedStacks.has(dep.id))) {\n        selectedStacks.set(id, stack);\n        added.push(id);\n        madeProgress = true;\n      }\n    }\n  } while (madeProgress);\n\n  if (added.length > 0) {\n    print('Including depending stacks: %s', chalk.bold(added.join(', ')));\n  }\n}\n\n/**\n * Calculate the transitive closure of stack dependencies.\n *\n * Modifies `selectedStacks` in-place.\n */\nfunction includeUpstreamStacks(\n  selectedStacks: Map<string, cxapi.CloudFormationStackArtifact>,\n  allStacks: Map<string, cxapi.CloudFormationStackArtifact>) {\n  const added = new Array<string>();\n  let madeProgress = true;\n  while (madeProgress) {\n    madeProgress = false;\n\n    for (const stack of selectedStacks.values()) {\n      // Select an additional stack if it's not selected yet and a dependency of a selected stack (and exists, obviously)\n      for (const dependencyId of stack.dependencies.map(x => x.manifest.displayName ?? x.id)) {\n        if (!selectedStacks.has(dependencyId) && allStacks.has(dependencyId)) {\n          added.push(dependencyId);\n          selectedStacks.set(dependencyId, allStacks.get(dependencyId)!);\n          madeProgress = true;\n        }\n      }\n    }\n  }\n\n  if (added.length > 0) {\n    print('Including dependency stacks: %s', chalk.bold(added.join(', ')));\n  }\n}\n\nfunction sanitizePatterns(patterns: string[]): string[] {\n  let sanitized = patterns.filter(s => s != null); // filter null/undefined\n  sanitized = [...new Set(sanitized)]; // make them unique\n  return sanitized;\n}\n"]}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cloud-assembly.js","sourceRoot":"","sources":["cloud-assembly.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cxapi from '@aws-cdk/cx-api';\nimport * as chalk from 'chalk';\nimport { minimatch } from 'minimatch';\nimport * as semver from 'semver';\nimport { error, print, warning } from '../../logging';\nimport { flatten } from '../../util';\nimport { versionNumber } from '../../version';\n\nexport enum DefaultSelection {\n  /**\n   * Returns an empty selection in case there are no selectors.\n   */\n  None = 'none',\n\n  /**\n   * If the app includes a single stack, returns it. Otherwise throws an exception.\n   * This behavior is used by \"deploy\".\n   */\n  OnlySingle = 'single',\n\n  /**\n   * Returns all stacks in the main (top level) assembly only.\n   */\n  MainAssembly = 'main',\n\n  /**\n   * If no selectors are provided, returns all stacks in the app,\n   * including stacks inside nested assemblies.\n   */\n  AllStacks = 'all',\n}\n\nexport interface SelectStacksOptions {\n  /**\n   * Extend the selection to upstread/downstream stacks\n   * @default ExtendedStackSelection.None only select the specified stacks.\n   */\n  extend?: ExtendedStackSelection;\n\n  /**\n   * The behavior if if no selectors are provided.\n   */\n  defaultBehavior: DefaultSelection;\n\n  /**\n   * Whether to deploy if the app contains no stacks.\n   *\n   * @default false\n   */\n  ignoreNoStacks?: boolean;\n}\n\n/**\n * When selecting stacks, what other stacks to include because of dependencies\n */\nexport enum ExtendedStackSelection {\n  /**\n   * Don't select any extra stacks\n   */\n  None,\n\n  /**\n   * Include stacks that this stack depends on\n   */\n  Upstream,\n\n  /**\n   * Include stacks that depend on this stack\n   */\n  Downstream\n}\n\n/**\n * A specification of which stacks should be selected\n */\nexport interface StackSelector {\n  /**\n   * Whether all stacks at the top level assembly should\n   * be selected and nothing else\n   */\n  allTopLevel?: boolean,\n\n  /**\n   * A list of patterns to match the stack hierarchical ids\n   */\n  patterns: string[],\n}\n\n/**\n * A single Cloud Assembly and the operations we do on it to deploy the artifacts inside\n */\nexport class CloudAssembly {\n  /**\n   * The directory this CloudAssembly was read from\n   */\n  public readonly directory: string;\n\n  constructor(public readonly assembly: cxapi.CloudAssembly) {\n    this.directory = assembly.directory;\n  }\n\n  public async selectStacks(selector: StackSelector, options: SelectStacksOptions): Promise<StackCollection> {\n    const asm = this.assembly;\n    const topLevelStacks = asm.stacks;\n    const stacks = semver.major(asm.version) < 10 ? asm.stacks : asm.stacksRecursively;\n    const allTopLevel = selector.allTopLevel ?? false;\n    const patterns = sanitizePatterns(selector.patterns);\n\n    if (stacks.length === 0) {\n      if (options.ignoreNoStacks) {\n        return new StackCollection(this, []);\n      }\n      throw new Error('This app contains no stacks');\n    }\n\n    if (allTopLevel) {\n      return this.selectTopLevelStacks(stacks, topLevelStacks, options.extend);\n    } else if (patterns.length > 0) {\n      return this.selectMatchingStacks(stacks, patterns, options.extend);\n    } else {\n      return this.selectDefaultStacks(stacks, topLevelStacks, options.defaultBehavior);\n    }\n  }\n\n  private selectTopLevelStacks(stacks: cxapi.CloudFormationStackArtifact[],\n    topLevelStacks: cxapi.CloudFormationStackArtifact[],\n    extend: ExtendedStackSelection = ExtendedStackSelection.None): StackCollection {\n    if (topLevelStacks.length > 0) {\n      return this.extendStacks(topLevelStacks, stacks, extend);\n    } else {\n      throw new Error('No stack found in the main cloud assembly. Use \"list\" to print manifest');\n    }\n  }\n\n  private selectMatchingStacks(stacks: cxapi.CloudFormationStackArtifact[],\n    patterns: string[],\n    extend: ExtendedStackSelection = ExtendedStackSelection.None): StackCollection {\n\n    // cli tests use this to ensure tests do not depend on legacy behavior\n    // (otherwise they will fail in v2)\n    const disableLegacy = process.env.CXAPI_DISABLE_SELECT_BY_ID === '1';\n\n    const matchingPattern = (pattern: string) => (stack: cxapi.CloudFormationStackArtifact) => {\n      if (minimatch(stack.hierarchicalId, pattern)) {\n        return true;\n      } else if (!disableLegacy && stack.id === pattern && semver.major(versionNumber()) < 2) {\n        warning('Selecting stack by identifier \"%s\". This identifier is deprecated and will be removed in v2. Please use \"%s\" instead.', chalk.bold(stack.id), chalk.bold(stack.hierarchicalId));\n        warning('Run \"cdk ls\" to see a list of all stack identifiers');\n        return true;\n      }\n      return false;\n    };\n\n    const matchedStacks = flatten(patterns.map(pattern => stacks.filter(matchingPattern(pattern))));\n\n    return this.extendStacks(matchedStacks, stacks, extend);\n  }\n\n  private selectDefaultStacks(stacks: cxapi.CloudFormationStackArtifact[],\n    topLevelStacks: cxapi.CloudFormationStackArtifact[],\n    defaultSelection: DefaultSelection) {\n    switch (defaultSelection) {\n      case DefaultSelection.MainAssembly:\n        return new StackCollection(this, topLevelStacks);\n      case DefaultSelection.AllStacks:\n        return new StackCollection(this, stacks);\n      case DefaultSelection.None:\n        return new StackCollection(this, []);\n      case DefaultSelection.OnlySingle:\n        if (topLevelStacks.length === 1) {\n          return new StackCollection(this, topLevelStacks);\n        } else {\n          throw new Error('Since this app includes more than a single stack, specify which stacks to use (wildcards are supported) or specify `--all`\\n' +\n          `Stacks: ${stacks.map(x => x.hierarchicalId).join(' · ')}`);\n        }\n      default:\n        throw new Error(`invalid default behavior: ${defaultSelection}`);\n    }\n  }\n\n  private extendStacks(matched: cxapi.CloudFormationStackArtifact[],\n    all: cxapi.CloudFormationStackArtifact[],\n    extend: ExtendedStackSelection = ExtendedStackSelection.None) {\n    const allStacks = new Map<string, cxapi.CloudFormationStackArtifact>();\n    for (const stack of all) {\n      allStacks.set(stack.hierarchicalId, stack);\n    }\n\n    const index = indexByHierarchicalId(matched);\n\n    switch (extend) {\n      case ExtendedStackSelection.Downstream:\n        includeDownstreamStacks(index, allStacks);\n        break;\n      case ExtendedStackSelection.Upstream:\n        includeUpstreamStacks(index, allStacks);\n        break;\n    }\n\n    // Filter original array because it is in the right order\n    const selectedList = all.filter(s => index.has(s.hierarchicalId));\n\n    return new StackCollection(this, selectedList);\n  }\n\n  /**\n   * Select a single stack by its ID\n   */\n  public stackById(stackId: string) {\n    return new StackCollection(this, [this.assembly.getStackArtifact(stackId)]);\n  }\n}\n\n/**\n * A collection of stacks and related artifacts\n *\n * In practice, not all artifacts in the CloudAssembly are created equal;\n * stacks can be selected independently, but other artifacts such as asset\n * bundles cannot.\n */\nexport class StackCollection {\n  constructor(public readonly assembly: CloudAssembly, public readonly stackArtifacts: cxapi.CloudFormationStackArtifact[]) {\n  }\n\n  public get stackCount() {\n    return this.stackArtifacts.length;\n  }\n\n  public get firstStack() {\n    if (this.stackCount < 1) {\n      throw new Error('StackCollection contains no stack artifacts (trying to access the first one)');\n    }\n    return this.stackArtifacts[0];\n  }\n\n  public get stackIds(): string[] {\n    return this.stackArtifacts.map(s => s.id);\n  }\n\n  public reversed() {\n    const arts = [...this.stackArtifacts];\n    arts.reverse();\n    return new StackCollection(this.assembly, arts);\n  }\n\n  public filter(predicate: (art: cxapi.CloudFormationStackArtifact) => boolean): StackCollection {\n    return new StackCollection(this.assembly, this.stackArtifacts.filter(predicate));\n  }\n\n  public concat(other: StackCollection): StackCollection {\n    return new StackCollection(this.assembly, this.stackArtifacts.concat(other.stackArtifacts));\n  }\n\n  /**\n   * Extracts 'aws:cdk:warning|info|error' metadata entries from the stack synthesis\n   */\n  public processMetadataMessages(options: MetadataMessageOptions = {}) {\n    let warnings = false;\n    let errors = false;\n\n    for (const stack of this.stackArtifacts) {\n      for (const message of stack.messages) {\n        switch (message.level) {\n          case cxapi.SynthesisMessageLevel.WARNING:\n            warnings = true;\n            printMessage(warning, 'Warning', message.id, message.entry);\n            break;\n          case cxapi.SynthesisMessageLevel.ERROR:\n            errors = true;\n            printMessage(error, 'Error', message.id, message.entry);\n            break;\n          case cxapi.SynthesisMessageLevel.INFO:\n            printMessage(print, 'Info', message.id, message.entry);\n            break;\n        }\n      }\n    }\n\n    if (errors && !options.ignoreErrors) {\n      throw new Error('Found errors');\n    }\n\n    if (options.strict && warnings) {\n      throw new Error('Found warnings (--strict mode)');\n    }\n\n    function printMessage(logFn: (s: string) => void, prefix: string, id: string, entry: cxapi.MetadataEntry) {\n      logFn(`[${prefix} at ${id}] ${entry.data}`);\n\n      if (options.verbose && entry.trace) {\n        logFn(`  ${entry.trace.join('\\n  ')}`);\n      }\n    }\n  }\n}\n\nexport interface MetadataMessageOptions {\n  /**\n   * Whether to be verbose\n   *\n   * @default false\n   */\n  verbose?: boolean;\n\n  /**\n   * Don't stop on error metadata\n   *\n   * @default false\n   */\n  ignoreErrors?: boolean;\n\n  /**\n   * Treat warnings in metadata as errors\n   *\n   * @default false\n   */\n  strict?: boolean;\n}\n\nfunction indexByHierarchicalId(stacks: cxapi.CloudFormationStackArtifact[]): Map<string, cxapi.CloudFormationStackArtifact> {\n  const result = new Map<string, cxapi.CloudFormationStackArtifact>();\n\n  for (const stack of stacks) {\n    result.set(stack.hierarchicalId, stack);\n  }\n\n  return result;\n}\n\n/**\n * Calculate the transitive closure of stack dependents.\n *\n * Modifies `selectedStacks` in-place.\n */\nfunction includeDownstreamStacks(\n  selectedStacks: Map<string, cxapi.CloudFormationStackArtifact>,\n  allStacks: Map<string, cxapi.CloudFormationStackArtifact>) {\n  const added = new Array<string>();\n\n  let madeProgress;\n  do {\n    madeProgress = false;\n\n    for (const [id, stack] of allStacks) {\n      // Select this stack if it's not selected yet AND it depends on a stack that's in the selected set\n      if (!selectedStacks.has(id) && (stack.dependencies || []).some(dep => selectedStacks.has(dep.id))) {\n        selectedStacks.set(id, stack);\n        added.push(id);\n        madeProgress = true;\n      }\n    }\n  } while (madeProgress);\n\n  if (added.length > 0) {\n    print('Including depending stacks: %s', chalk.bold(added.join(', ')));\n  }\n}\n\n/**\n * Calculate the transitive closure of stack dependencies.\n *\n * Modifies `selectedStacks` in-place.\n */\nfunction includeUpstreamStacks(\n  selectedStacks: Map<string, cxapi.CloudFormationStackArtifact>,\n  allStacks: Map<string, cxapi.CloudFormationStackArtifact>) {\n  const added = new Array<string>();\n  let madeProgress = true;\n  while (madeProgress) {\n    madeProgress = false;\n\n    for (const stack of selectedStacks.values()) {\n      // Select an additional stack if it's not selected yet and a dependency of a selected stack (and exists, obviously)\n      for (const dependencyId of stack.dependencies.map(x => x.manifest.displayName ?? x.id)) {\n        if (!selectedStacks.has(dependencyId) && allStacks.has(dependencyId)) {\n          added.push(dependencyId);\n          selectedStacks.set(dependencyId, allStacks.get(dependencyId)!);\n          madeProgress = true;\n        }\n      }\n    }\n  }\n\n  if (added.length > 0) {\n    print('Including dependency stacks: %s', chalk.bold(added.join(', ')));\n  }\n}\n\nfunction sanitizePatterns(patterns: string[]): string[] {\n  let sanitized = patterns.filter(s => s != null); // filter null/undefined\n  sanitized = [...new Set(sanitized)]; // make them unique\n  return sanitized;\n}\n"]}

@@ -8,6 +8,2 @@ import * as cxapi from '@aws-cdk/cx-api';

import { Tag } from '../cdk-toolkit';
type TemplateBodyParameter = {
TemplateBody?: string;
TemplateURL?: string;
};
export interface DeployStackResult {

@@ -186,9 +182,2 @@ readonly noOp: boolean;

export declare function deployStack(options: DeployStackOptions): Promise<DeployStackResult>;
/**
* Prepare a body parameter for CFN, performing the upload
*
* Return it as-is if it is small enough to pass in the API call,
* upload to S3 and return the coordinates if it is not.
*/
export declare function makeBodyParameterAndUpload(stack: cxapi.CloudFormationStackArtifact, resolvedEnvironment: cxapi.Environment, resources: EnvironmentResources, sdkProvider: SdkProvider, sdk: ISDK, overrideTemplate?: any): Promise<TemplateBodyParameter>;
export interface DestroyStackOptions {

@@ -206,2 +195,1 @@ /**

export declare function destroyStack(options: DestroyStackOptions): Promise<void>;
export {};

@@ -166,2 +166,8 @@ import * as cxapi from '@aws-cdk/cx-api';

readonly assetParallelism?: boolean;
/**
* Whether to deploy if the app contains no stacks.
*
* @default false
*/
ignoreNoStacks?: boolean;
}

@@ -266,2 +272,3 @@ interface AssetOptions {

stackExists(options: StackExistsOptions): Promise<boolean>;
prepareSdkWithDeployRole(stackArtifact: cxapi.CloudFormationStackArtifact): Promise<PreparedSdkForEnvironment>;
private prepareSdkWithLookupOrDeployRole;

@@ -268,0 +275,0 @@ /**

@@ -65,3 +65,3 @@ "use strict";

});
const stackChanges = cfn_diff.diffTemplate(currentTemplate.deployedTemplate, stackArtifact.template);
const stackChanges = cfn_diff.fullDiff(currentTemplate.deployedTemplate, stackArtifact.template);
const { hotswappableChanges, nonHotswappableChanges } = await classifyResourceChanges(stackChanges, evaluateCfnTemplate, sdk, currentTemplate.nestedStackNames);

@@ -201,3 +201,3 @@ logNonHotswappableChanges(nonHotswappableChanges, hotswapMode);

const evaluateNestedCfnTemplate = await evaluateCfnTemplate.createNestedEvaluateCloudFormationTemplate(nestedStackName, change.newValue?.Properties?.NestedTemplate, change.newValue?.Properties?.Parameters);
const nestedDiff = cfn_diff.diffTemplate(change.oldValue?.Properties?.NestedTemplate, change.newValue?.Properties?.NestedTemplate);
const nestedDiff = cfn_diff.fullDiff(change.oldValue?.Properties?.NestedTemplate, change.newValue?.Properties?.NestedTemplate);
return classifyResourceChanges(nestedDiff, evaluateNestedCfnTemplate, sdk, nestedStackNames[logicalId].nestedChildStackNames);

@@ -318,2 +318,2 @@ }

}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"hotswap-deployments.js","sourceRoot":"","sources":["hotswap-deployments.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cfn_diff from '@aws-cdk/cloudformation-diff';\nimport * as cxapi from '@aws-cdk/cx-api';\nimport * as chalk from 'chalk';\nimport { ISDK, Mode, SdkProvider } from './aws-auth';\nimport { DeployStackResult } from './deploy-stack';\nimport { EvaluateCloudFormationTemplate } from './evaluate-cloudformation-template';\nimport { isHotswappableAppSyncChange } from './hotswap/appsync-mapping-templates';\nimport { isHotswappableCodeBuildProjectChange } from './hotswap/code-build-projects';\nimport { ICON, ChangeHotswapResult, HotswapMode, HotswappableChange, NonHotswappableChange, HotswappableChangeCandidate, ClassifiedResourceChanges, reportNonHotswappableChange, reportNonHotswappableResource } from './hotswap/common';\nimport { isHotswappableEcsServiceChange } from './hotswap/ecs-services';\nimport { isHotswappableLambdaFunctionChange } from './hotswap/lambda-functions';\nimport { skipChangeForS3DeployCustomResourcePolicy, isHotswappableS3BucketDeploymentChange } from './hotswap/s3-bucket-deployments';\nimport { isHotswappableStateMachineChange } from './hotswap/stepfunctions-state-machines';\nimport { loadCurrentTemplateWithNestedStacks, NestedStackNames } from './nested-stack-helpers';\nimport { CloudFormationStack } from './util/cloudformation';\nimport { print } from '../logging';\n\ntype HotswapDetector = (\n  logicalId: string, change: HotswappableChangeCandidate, evaluateCfnTemplate: EvaluateCloudFormationTemplate\n) => Promise<ChangeHotswapResult>;\n\nconst RESOURCE_DETECTORS: { [key: string]: HotswapDetector } = {\n  // Lambda\n  'AWS::Lambda::Function': isHotswappableLambdaFunctionChange,\n  'AWS::Lambda::Version': isHotswappableLambdaFunctionChange,\n  'AWS::Lambda::Alias': isHotswappableLambdaFunctionChange,\n\n  // AppSync\n  'AWS::AppSync::Resolver': isHotswappableAppSyncChange,\n  'AWS::AppSync::FunctionConfiguration': isHotswappableAppSyncChange,\n  'AWS::AppSync::GraphQLSchema': isHotswappableAppSyncChange,\n  'AWS::AppSync::ApiKey': isHotswappableAppSyncChange,\n\n  'AWS::ECS::TaskDefinition': isHotswappableEcsServiceChange,\n  'AWS::CodeBuild::Project': isHotswappableCodeBuildProjectChange,\n  'AWS::StepFunctions::StateMachine': isHotswappableStateMachineChange,\n  'Custom::CDKBucketDeployment': isHotswappableS3BucketDeploymentChange,\n  'AWS::IAM::Policy': async (\n    logicalId: string, change: HotswappableChangeCandidate, evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n  ): Promise<ChangeHotswapResult> => {\n    // If the policy is for a S3BucketDeploymentChange, we can ignore the change\n    if (await skipChangeForS3DeployCustomResourcePolicy(logicalId, change, evaluateCfnTemplate)) {\n      return [];\n    }\n\n    return reportNonHotswappableResource(change, 'This resource type is not supported for hotswap deployments');\n  },\n\n  'AWS::CDK::Metadata': async () => [],\n};\n\n/**\n * Perform a hotswap deployment,\n * short-circuiting CloudFormation if possible.\n * If it's not possible to short-circuit the deployment\n * (because the CDK Stack contains changes that cannot be deployed without CloudFormation),\n * returns `undefined`.\n */\nexport async function tryHotswapDeployment(\n  sdkProvider: SdkProvider, assetParams: { [key: string]: string },\n  cloudFormationStack: CloudFormationStack, stackArtifact: cxapi.CloudFormationStackArtifact,\n  hotswapMode: HotswapMode,\n): Promise<DeployStackResult | undefined> {\n  // resolve the environment, so we can substitute things like AWS::Region in CFN expressions\n  const resolvedEnv = await sdkProvider.resolveEnvironment(stackArtifact.environment);\n  // create a new SDK using the CLI credentials, because the default one will not work for new-style synthesis -\n  // it assumes the bootstrap deploy Role, which doesn't have permissions to update Lambda functions\n  const sdk = (await sdkProvider.forEnvironment(resolvedEnv, Mode.ForWriting)).sdk;\n\n  const currentTemplate = await loadCurrentTemplateWithNestedStacks(stackArtifact, sdk);\n\n  const evaluateCfnTemplate = new EvaluateCloudFormationTemplate({\n    stackName: stackArtifact.stackName,\n    template: stackArtifact.template,\n    parameters: assetParams,\n    account: resolvedEnv.account,\n    region: resolvedEnv.region,\n    partition: (await sdk.currentAccount()).partition,\n    urlSuffix: (region) => sdk.getEndpointSuffix(region),\n    sdk,\n    nestedStackNames: currentTemplate.nestedStackNames,\n  });\n\n  const stackChanges = cfn_diff.diffTemplate(currentTemplate.deployedTemplate, stackArtifact.template);\n  const { hotswappableChanges, nonHotswappableChanges } = await classifyResourceChanges(\n    stackChanges, evaluateCfnTemplate, sdk, currentTemplate.nestedStackNames,\n  );\n\n  logNonHotswappableChanges(nonHotswappableChanges, hotswapMode);\n\n  // preserve classic hotswap behavior\n  if (hotswapMode === HotswapMode.FALL_BACK) {\n    if (nonHotswappableChanges.length > 0) {\n      return undefined;\n    }\n  }\n\n  // apply the short-circuitable changes\n  await applyAllHotswappableChanges(sdk, hotswappableChanges);\n\n  return { noOp: hotswappableChanges.length === 0, stackArn: cloudFormationStack.stackId, outputs: cloudFormationStack.outputs };\n}\n\n/**\n * Classifies all changes to all resources as either hotswappable or not.\n * Metadata changes are excluded from the list of (non)hotswappable resources.\n */\nasync function classifyResourceChanges(\n  stackChanges: cfn_diff.TemplateDiff,\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n  sdk: ISDK,\n  nestedStackNames: { [nestedStackName: string]: NestedStackNames },\n): Promise<ClassifiedResourceChanges> {\n  const resourceDifferences = getStackResourceDifferences(stackChanges);\n\n  const promises: Array<() => Promise<ChangeHotswapResult>> = [];\n  const hotswappableResources = new Array<HotswappableChange>();\n  const nonHotswappableResources = new Array<NonHotswappableChange>();\n  for (const logicalId of Object.keys(stackChanges.outputs.changes)) {\n    nonHotswappableResources.push({\n      hotswappable: false,\n      reason: 'output was changed',\n      logicalId,\n      rejectedChanges: [],\n      resourceType: 'Stack Output',\n    });\n  }\n  // gather the results of the detector functions\n  for (const [logicalId, change] of Object.entries(resourceDifferences)) {\n    if (change.newValue?.Type === 'AWS::CloudFormation::Stack' && change.oldValue?.Type === 'AWS::CloudFormation::Stack') {\n      const nestedHotswappableResources = await findNestedHotswappableChanges(logicalId, change, nestedStackNames, evaluateCfnTemplate, sdk);\n      hotswappableResources.push(...nestedHotswappableResources.hotswappableChanges);\n      nonHotswappableResources.push(...nestedHotswappableResources.nonHotswappableChanges);\n\n      continue;\n    }\n\n    const hotswappableChangeCandidate = isCandidateForHotswapping(change, logicalId);\n    // we don't need to run this through the detector functions, we can already judge this\n    if ('hotswappable' in hotswappableChangeCandidate) {\n      if (!hotswappableChangeCandidate.hotswappable) {\n        nonHotswappableResources.push(hotswappableChangeCandidate);\n      }\n\n      continue;\n    }\n\n    const resourceType: string = hotswappableChangeCandidate.newValue.Type;\n    if (resourceType in RESOURCE_DETECTORS) {\n      // run detector functions lazily to prevent unhandled promise rejections\n      promises.push(() => RESOURCE_DETECTORS[resourceType](logicalId, hotswappableChangeCandidate, evaluateCfnTemplate));\n    } else {\n      reportNonHotswappableChange(nonHotswappableResources, hotswappableChangeCandidate, undefined, 'This resource type is not supported for hotswap deployments');\n    }\n  }\n\n  // resolve all detector results\n  const changesDetectionResults: Array<ChangeHotswapResult> = [];\n  for (const detectorResultPromises of promises) {\n    const hotswapDetectionResults = await Promise.all(await detectorResultPromises());\n    changesDetectionResults.push(hotswapDetectionResults);\n  }\n\n  for (const resourceDetectionResults of changesDetectionResults) {\n    for (const propertyResult of resourceDetectionResults) {\n      propertyResult.hotswappable ?\n        hotswappableResources.push(propertyResult) :\n        nonHotswappableResources.push(propertyResult);\n    }\n  }\n\n  return {\n    hotswappableChanges: hotswappableResources,\n    nonHotswappableChanges: nonHotswappableResources,\n  };\n}\n\n/**\n * Returns all changes to resources in the given Stack.\n *\n * @param stackChanges the collection of all changes to a given Stack\n */\nfunction getStackResourceDifferences(stackChanges: cfn_diff.TemplateDiff): { [logicalId: string]: cfn_diff.ResourceDifference } {\n  // we need to collapse logical ID rename changes into one change,\n  // as they are represented in stackChanges as a pair of two changes: one addition and one removal\n  const allResourceChanges: { [logId: string]: cfn_diff.ResourceDifference } = stackChanges.resources.changes;\n  const allRemovalChanges = filterDict(allResourceChanges, resChange => resChange.isRemoval);\n  const allNonRemovalChanges = filterDict(allResourceChanges, resChange => !resChange.isRemoval);\n  for (const [logId, nonRemovalChange] of Object.entries(allNonRemovalChanges)) {\n    if (nonRemovalChange.isAddition) {\n      const addChange = nonRemovalChange;\n      // search for an identical removal change\n      const identicalRemovalChange = Object.entries(allRemovalChanges).find(([_, remChange]) => {\n        return changesAreForSameResource(remChange, addChange);\n      });\n      // if we found one, then this means this is a rename change\n      if (identicalRemovalChange) {\n        const [removedLogId, removedResourceChange] = identicalRemovalChange;\n        allNonRemovalChanges[logId] = makeRenameDifference(removedResourceChange, addChange);\n        // delete the removal change that forms the rename pair\n        delete allRemovalChanges[removedLogId];\n      }\n    }\n  }\n  // the final result are all of the remaining removal changes,\n  // plus all of the non-removal changes\n  // (we saved the rename changes in that object already)\n  return {\n    ...allRemovalChanges,\n    ...allNonRemovalChanges,\n  };\n}\n\n/** Filters an object with string keys based on whether the callback returns 'true' for the given value in the object. */\nfunction filterDict<T>(dict: { [key: string]: T }, func: (t: T) => boolean): { [key: string]: T } {\n  return Object.entries(dict).reduce((acc, [key, t]) => {\n    if (func(t)) {\n      acc[key] = t;\n    }\n    return acc;\n  }, {} as { [key: string]: T });\n}\n\n/** Finds any hotswappable changes in all nested stacks. */\nasync function findNestedHotswappableChanges(\n  logicalId: string,\n  change: cfn_diff.ResourceDifference,\n  nestedStackNames: { [nestedStackName: string]: NestedStackNames },\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n  sdk: ISDK,\n): Promise<ClassifiedResourceChanges> {\n  const nestedStackName = nestedStackNames[logicalId].nestedStackPhysicalName;\n  if (!nestedStackName) {\n    return {\n      hotswappableChanges: [],\n      nonHotswappableChanges: [{\n        hotswappable: false,\n        logicalId,\n        reason: `physical name for AWS::CloudFormation::Stack '${logicalId}' could not be found in CloudFormation, so this is a newly created nested stack and cannot be hotswapped`,\n        rejectedChanges: [],\n        resourceType: 'AWS::CloudFormation::Stack',\n      }],\n    };\n  }\n\n  const evaluateNestedCfnTemplate = await evaluateCfnTemplate.createNestedEvaluateCloudFormationTemplate(\n    nestedStackName, change.newValue?.Properties?.NestedTemplate, change.newValue?.Properties?.Parameters,\n  );\n\n  const nestedDiff = cfn_diff.diffTemplate(\n    change.oldValue?.Properties?.NestedTemplate, change.newValue?.Properties?.NestedTemplate,\n  );\n\n  return classifyResourceChanges(nestedDiff, evaluateNestedCfnTemplate, sdk, nestedStackNames[logicalId].nestedChildStackNames);\n}\n\n/** Returns 'true' if a pair of changes is for the same resource. */\nfunction changesAreForSameResource(oldChange: cfn_diff.ResourceDifference, newChange: cfn_diff.ResourceDifference): boolean {\n  return oldChange.oldResourceType === newChange.newResourceType &&\n    // this isn't great, but I don't want to bring in something like underscore just for this comparison\n    JSON.stringify(oldChange.oldProperties) === JSON.stringify(newChange.newProperties);\n}\n\nfunction makeRenameDifference(\n  remChange: cfn_diff.ResourceDifference,\n  addChange: cfn_diff.ResourceDifference,\n): cfn_diff.ResourceDifference {\n  return new cfn_diff.ResourceDifference(\n    // we have to fill in the old value, because otherwise this will be classified as a non-hotswappable change\n    remChange.oldValue,\n    addChange.newValue,\n    {\n      resourceType: {\n        oldType: remChange.oldResourceType,\n        newType: addChange.newResourceType,\n      },\n      propertyDiffs: (addChange as any).propertyDiffs,\n      otherDiffs: (addChange as any).otherDiffs,\n    },\n  );\n}\n\n/**\n * Returns a `HotswappableChangeCandidate` if the change is hotswappable\n * Returns an empty `HotswappableChange` if the change is to CDK::Metadata\n * Returns a `NonHotswappableChange` if the change is not hotswappable\n */\nfunction isCandidateForHotswapping(\n  change: cfn_diff.ResourceDifference, logicalId: string,\n): HotswappableChange | NonHotswappableChange | HotswappableChangeCandidate {\n  // a resource has been removed OR a resource has been added; we can't short-circuit that change\n  if (!change.oldValue) {\n    return {\n      hotswappable: false,\n      resourceType: change.newValue!.Type,\n      logicalId,\n      rejectedChanges: [],\n      reason: `resource '${logicalId}' was created by this deployment`,\n    };\n  } else if (!change.newValue) {\n    return {\n      hotswappable: false,\n      resourceType: change.oldValue!.Type,\n      logicalId,\n      rejectedChanges: [],\n      reason: `resource '${logicalId}' was destroyed by this deployment`,\n    };\n  }\n\n  // a resource has had its type changed\n  if (change.newValue?.Type !== change.oldValue?.Type) {\n    return {\n      hotswappable: false,\n      resourceType: change.newValue?.Type,\n      logicalId,\n      rejectedChanges: [],\n      reason: `resource '${logicalId}' had its type changed from '${change.oldValue?.Type}' to '${change.newValue?.Type}'`,\n    };\n  }\n\n  return {\n    logicalId,\n    oldValue: change.oldValue,\n    newValue: change.newValue,\n    propertyUpdates: change.propertyUpdates,\n  };\n}\n\nasync function applyAllHotswappableChanges(sdk: ISDK, hotswappableChanges: HotswappableChange[]): Promise<void[]> {\n  if (hotswappableChanges.length > 0) {\n    print(`\\n${ICON} hotswapping resources:`);\n  }\n  return Promise.all(hotswappableChanges.map(hotswapOperation => {\n    return applyHotswappableChange(sdk, hotswapOperation);\n  }));\n}\n\nasync function applyHotswappableChange(sdk: ISDK, hotswapOperation: HotswappableChange): Promise<void> {\n  // note the type of service that was successfully hotswapped in the User-Agent\n  const customUserAgent = `cdk-hotswap/success-${hotswapOperation.service}`;\n  sdk.appendCustomUserAgent(customUserAgent);\n\n  for (const name of hotswapOperation.resourceNames) {\n    print(`   ${ICON} %s`, chalk.bold(name));\n  }\n\n  // if the SDK call fails, an error will be thrown by the SDK\n  // and will prevent the green 'hotswapped!' text from being displayed\n  await hotswapOperation.apply(sdk);\n\n  for (const name of hotswapOperation.resourceNames) {\n    print(`${ICON} %s %s`, chalk.bold(name), chalk.green('hotswapped!'));\n  }\n\n  sdk.removeCustomUserAgent(customUserAgent);\n}\n\nfunction logNonHotswappableChanges(nonHotswappableChanges: NonHotswappableChange[], hotswapMode: HotswapMode): void {\n  if (nonHotswappableChanges.length === 0) {\n    return;\n  }\n  /**\n   * EKS Services can have a task definition that doesn't refer to the task definition being updated.\n   * We have to log this as a non-hotswappable change to the task definition, but when we do,\n   * we wind up hotswapping the task definition and logging it as a non-hotswappable change.\n   *\n   * This logic prevents us from logging that change as non-hotswappable when we hotswap it.\n   */\n  if (hotswapMode === HotswapMode.HOTSWAP_ONLY) {\n    nonHotswappableChanges = nonHotswappableChanges.filter((change) => change.hotswapOnlyVisible === true);\n\n    if (nonHotswappableChanges.length === 0) {\n      return;\n    }\n  }\n  if (hotswapMode === HotswapMode.HOTSWAP_ONLY) {\n    print('\\n%s %s', chalk.red('⚠️'), chalk.red('The following non-hotswappable changes were found. To reconcile these using CloudFormation, specify --hotswap-fallback'));\n  } else {\n    print('\\n%s %s', chalk.red('⚠️'), chalk.red('The following non-hotswappable changes were found:'));\n  }\n\n  for (const change of nonHotswappableChanges) {\n    change.rejectedChanges.length > 0 ?\n      print('    logicalID: %s, type: %s, rejected changes: %s, reason: %s', chalk.bold(change.logicalId), chalk.bold(change.resourceType), chalk.bold(change.rejectedChanges), chalk.red(change.reason)) :\n      print('    logicalID: %s, type: %s, reason: %s', chalk.bold(change.logicalId), chalk.bold(change.resourceType), chalk.red(change.reason));\n  }\n\n  print(''); // newline\n}\n"]}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"hotswap-deployments.js","sourceRoot":"","sources":["hotswap-deployments.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cfn_diff from '@aws-cdk/cloudformation-diff';\nimport * as cxapi from '@aws-cdk/cx-api';\nimport * as chalk from 'chalk';\nimport { ISDK, Mode, SdkProvider } from './aws-auth';\nimport { DeployStackResult } from './deploy-stack';\nimport { EvaluateCloudFormationTemplate } from './evaluate-cloudformation-template';\nimport { isHotswappableAppSyncChange } from './hotswap/appsync-mapping-templates';\nimport { isHotswappableCodeBuildProjectChange } from './hotswap/code-build-projects';\nimport { ICON, ChangeHotswapResult, HotswapMode, HotswappableChange, NonHotswappableChange, HotswappableChangeCandidate, ClassifiedResourceChanges, reportNonHotswappableChange, reportNonHotswappableResource } from './hotswap/common';\nimport { isHotswappableEcsServiceChange } from './hotswap/ecs-services';\nimport { isHotswappableLambdaFunctionChange } from './hotswap/lambda-functions';\nimport { skipChangeForS3DeployCustomResourcePolicy, isHotswappableS3BucketDeploymentChange } from './hotswap/s3-bucket-deployments';\nimport { isHotswappableStateMachineChange } from './hotswap/stepfunctions-state-machines';\nimport { loadCurrentTemplateWithNestedStacks, NestedStackNames } from './nested-stack-helpers';\nimport { CloudFormationStack } from './util/cloudformation';\nimport { print } from '../logging';\n\ntype HotswapDetector = (\n  logicalId: string, change: HotswappableChangeCandidate, evaluateCfnTemplate: EvaluateCloudFormationTemplate\n) => Promise<ChangeHotswapResult>;\n\nconst RESOURCE_DETECTORS: { [key: string]: HotswapDetector } = {\n  // Lambda\n  'AWS::Lambda::Function': isHotswappableLambdaFunctionChange,\n  'AWS::Lambda::Version': isHotswappableLambdaFunctionChange,\n  'AWS::Lambda::Alias': isHotswappableLambdaFunctionChange,\n\n  // AppSync\n  'AWS::AppSync::Resolver': isHotswappableAppSyncChange,\n  'AWS::AppSync::FunctionConfiguration': isHotswappableAppSyncChange,\n  'AWS::AppSync::GraphQLSchema': isHotswappableAppSyncChange,\n  'AWS::AppSync::ApiKey': isHotswappableAppSyncChange,\n\n  'AWS::ECS::TaskDefinition': isHotswappableEcsServiceChange,\n  'AWS::CodeBuild::Project': isHotswappableCodeBuildProjectChange,\n  'AWS::StepFunctions::StateMachine': isHotswappableStateMachineChange,\n  'Custom::CDKBucketDeployment': isHotswappableS3BucketDeploymentChange,\n  'AWS::IAM::Policy': async (\n    logicalId: string, change: HotswappableChangeCandidate, evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n  ): Promise<ChangeHotswapResult> => {\n    // If the policy is for a S3BucketDeploymentChange, we can ignore the change\n    if (await skipChangeForS3DeployCustomResourcePolicy(logicalId, change, evaluateCfnTemplate)) {\n      return [];\n    }\n\n    return reportNonHotswappableResource(change, 'This resource type is not supported for hotswap deployments');\n  },\n\n  'AWS::CDK::Metadata': async () => [],\n};\n\n/**\n * Perform a hotswap deployment,\n * short-circuiting CloudFormation if possible.\n * If it's not possible to short-circuit the deployment\n * (because the CDK Stack contains changes that cannot be deployed without CloudFormation),\n * returns `undefined`.\n */\nexport async function tryHotswapDeployment(\n  sdkProvider: SdkProvider, assetParams: { [key: string]: string },\n  cloudFormationStack: CloudFormationStack, stackArtifact: cxapi.CloudFormationStackArtifact,\n  hotswapMode: HotswapMode,\n): Promise<DeployStackResult | undefined> {\n  // resolve the environment, so we can substitute things like AWS::Region in CFN expressions\n  const resolvedEnv = await sdkProvider.resolveEnvironment(stackArtifact.environment);\n  // create a new SDK using the CLI credentials, because the default one will not work for new-style synthesis -\n  // it assumes the bootstrap deploy Role, which doesn't have permissions to update Lambda functions\n  const sdk = (await sdkProvider.forEnvironment(resolvedEnv, Mode.ForWriting)).sdk;\n\n  const currentTemplate = await loadCurrentTemplateWithNestedStacks(stackArtifact, sdk);\n\n  const evaluateCfnTemplate = new EvaluateCloudFormationTemplate({\n    stackName: stackArtifact.stackName,\n    template: stackArtifact.template,\n    parameters: assetParams,\n    account: resolvedEnv.account,\n    region: resolvedEnv.region,\n    partition: (await sdk.currentAccount()).partition,\n    urlSuffix: (region) => sdk.getEndpointSuffix(region),\n    sdk,\n    nestedStackNames: currentTemplate.nestedStackNames,\n  });\n\n  const stackChanges = cfn_diff.fullDiff(currentTemplate.deployedTemplate, stackArtifact.template);\n  const { hotswappableChanges, nonHotswappableChanges } = await classifyResourceChanges(\n    stackChanges, evaluateCfnTemplate, sdk, currentTemplate.nestedStackNames,\n  );\n\n  logNonHotswappableChanges(nonHotswappableChanges, hotswapMode);\n\n  // preserve classic hotswap behavior\n  if (hotswapMode === HotswapMode.FALL_BACK) {\n    if (nonHotswappableChanges.length > 0) {\n      return undefined;\n    }\n  }\n\n  // apply the short-circuitable changes\n  await applyAllHotswappableChanges(sdk, hotswappableChanges);\n\n  return { noOp: hotswappableChanges.length === 0, stackArn: cloudFormationStack.stackId, outputs: cloudFormationStack.outputs };\n}\n\n/**\n * Classifies all changes to all resources as either hotswappable or not.\n * Metadata changes are excluded from the list of (non)hotswappable resources.\n */\nasync function classifyResourceChanges(\n  stackChanges: cfn_diff.TemplateDiff,\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n  sdk: ISDK,\n  nestedStackNames: { [nestedStackName: string]: NestedStackNames },\n): Promise<ClassifiedResourceChanges> {\n  const resourceDifferences = getStackResourceDifferences(stackChanges);\n\n  const promises: Array<() => Promise<ChangeHotswapResult>> = [];\n  const hotswappableResources = new Array<HotswappableChange>();\n  const nonHotswappableResources = new Array<NonHotswappableChange>();\n  for (const logicalId of Object.keys(stackChanges.outputs.changes)) {\n    nonHotswappableResources.push({\n      hotswappable: false,\n      reason: 'output was changed',\n      logicalId,\n      rejectedChanges: [],\n      resourceType: 'Stack Output',\n    });\n  }\n  // gather the results of the detector functions\n  for (const [logicalId, change] of Object.entries(resourceDifferences)) {\n    if (change.newValue?.Type === 'AWS::CloudFormation::Stack' && change.oldValue?.Type === 'AWS::CloudFormation::Stack') {\n      const nestedHotswappableResources = await findNestedHotswappableChanges(logicalId, change, nestedStackNames, evaluateCfnTemplate, sdk);\n      hotswappableResources.push(...nestedHotswappableResources.hotswappableChanges);\n      nonHotswappableResources.push(...nestedHotswappableResources.nonHotswappableChanges);\n\n      continue;\n    }\n\n    const hotswappableChangeCandidate = isCandidateForHotswapping(change, logicalId);\n    // we don't need to run this through the detector functions, we can already judge this\n    if ('hotswappable' in hotswappableChangeCandidate) {\n      if (!hotswappableChangeCandidate.hotswappable) {\n        nonHotswappableResources.push(hotswappableChangeCandidate);\n      }\n\n      continue;\n    }\n\n    const resourceType: string = hotswappableChangeCandidate.newValue.Type;\n    if (resourceType in RESOURCE_DETECTORS) {\n      // run detector functions lazily to prevent unhandled promise rejections\n      promises.push(() => RESOURCE_DETECTORS[resourceType](logicalId, hotswappableChangeCandidate, evaluateCfnTemplate));\n    } else {\n      reportNonHotswappableChange(nonHotswappableResources, hotswappableChangeCandidate, undefined, 'This resource type is not supported for hotswap deployments');\n    }\n  }\n\n  // resolve all detector results\n  const changesDetectionResults: Array<ChangeHotswapResult> = [];\n  for (const detectorResultPromises of promises) {\n    const hotswapDetectionResults = await Promise.all(await detectorResultPromises());\n    changesDetectionResults.push(hotswapDetectionResults);\n  }\n\n  for (const resourceDetectionResults of changesDetectionResults) {\n    for (const propertyResult of resourceDetectionResults) {\n      propertyResult.hotswappable ?\n        hotswappableResources.push(propertyResult) :\n        nonHotswappableResources.push(propertyResult);\n    }\n  }\n\n  return {\n    hotswappableChanges: hotswappableResources,\n    nonHotswappableChanges: nonHotswappableResources,\n  };\n}\n\n/**\n * Returns all changes to resources in the given Stack.\n *\n * @param stackChanges the collection of all changes to a given Stack\n */\nfunction getStackResourceDifferences(stackChanges: cfn_diff.TemplateDiff): { [logicalId: string]: cfn_diff.ResourceDifference } {\n  // we need to collapse logical ID rename changes into one change,\n  // as they are represented in stackChanges as a pair of two changes: one addition and one removal\n  const allResourceChanges: { [logId: string]: cfn_diff.ResourceDifference } = stackChanges.resources.changes;\n  const allRemovalChanges = filterDict(allResourceChanges, resChange => resChange.isRemoval);\n  const allNonRemovalChanges = filterDict(allResourceChanges, resChange => !resChange.isRemoval);\n  for (const [logId, nonRemovalChange] of Object.entries(allNonRemovalChanges)) {\n    if (nonRemovalChange.isAddition) {\n      const addChange = nonRemovalChange;\n      // search for an identical removal change\n      const identicalRemovalChange = Object.entries(allRemovalChanges).find(([_, remChange]) => {\n        return changesAreForSameResource(remChange, addChange);\n      });\n      // if we found one, then this means this is a rename change\n      if (identicalRemovalChange) {\n        const [removedLogId, removedResourceChange] = identicalRemovalChange;\n        allNonRemovalChanges[logId] = makeRenameDifference(removedResourceChange, addChange);\n        // delete the removal change that forms the rename pair\n        delete allRemovalChanges[removedLogId];\n      }\n    }\n  }\n  // the final result are all of the remaining removal changes,\n  // plus all of the non-removal changes\n  // (we saved the rename changes in that object already)\n  return {\n    ...allRemovalChanges,\n    ...allNonRemovalChanges,\n  };\n}\n\n/** Filters an object with string keys based on whether the callback returns 'true' for the given value in the object. */\nfunction filterDict<T>(dict: { [key: string]: T }, func: (t: T) => boolean): { [key: string]: T } {\n  return Object.entries(dict).reduce((acc, [key, t]) => {\n    if (func(t)) {\n      acc[key] = t;\n    }\n    return acc;\n  }, {} as { [key: string]: T });\n}\n\n/** Finds any hotswappable changes in all nested stacks. */\nasync function findNestedHotswappableChanges(\n  logicalId: string,\n  change: cfn_diff.ResourceDifference,\n  nestedStackNames: { [nestedStackName: string]: NestedStackNames },\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n  sdk: ISDK,\n): Promise<ClassifiedResourceChanges> {\n  const nestedStackName = nestedStackNames[logicalId].nestedStackPhysicalName;\n  if (!nestedStackName) {\n    return {\n      hotswappableChanges: [],\n      nonHotswappableChanges: [{\n        hotswappable: false,\n        logicalId,\n        reason: `physical name for AWS::CloudFormation::Stack '${logicalId}' could not be found in CloudFormation, so this is a newly created nested stack and cannot be hotswapped`,\n        rejectedChanges: [],\n        resourceType: 'AWS::CloudFormation::Stack',\n      }],\n    };\n  }\n\n  const evaluateNestedCfnTemplate = await evaluateCfnTemplate.createNestedEvaluateCloudFormationTemplate(\n    nestedStackName, change.newValue?.Properties?.NestedTemplate, change.newValue?.Properties?.Parameters,\n  );\n\n  const nestedDiff = cfn_diff.fullDiff(\n    change.oldValue?.Properties?.NestedTemplate, change.newValue?.Properties?.NestedTemplate,\n  );\n\n  return classifyResourceChanges(nestedDiff, evaluateNestedCfnTemplate, sdk, nestedStackNames[logicalId].nestedChildStackNames);\n}\n\n/** Returns 'true' if a pair of changes is for the same resource. */\nfunction changesAreForSameResource(oldChange: cfn_diff.ResourceDifference, newChange: cfn_diff.ResourceDifference): boolean {\n  return oldChange.oldResourceType === newChange.newResourceType &&\n    // this isn't great, but I don't want to bring in something like underscore just for this comparison\n    JSON.stringify(oldChange.oldProperties) === JSON.stringify(newChange.newProperties);\n}\n\nfunction makeRenameDifference(\n  remChange: cfn_diff.ResourceDifference,\n  addChange: cfn_diff.ResourceDifference,\n): cfn_diff.ResourceDifference {\n  return new cfn_diff.ResourceDifference(\n    // we have to fill in the old value, because otherwise this will be classified as a non-hotswappable change\n    remChange.oldValue,\n    addChange.newValue,\n    {\n      resourceType: {\n        oldType: remChange.oldResourceType,\n        newType: addChange.newResourceType,\n      },\n      propertyDiffs: (addChange as any).propertyDiffs,\n      otherDiffs: (addChange as any).otherDiffs,\n    },\n  );\n}\n\n/**\n * Returns a `HotswappableChangeCandidate` if the change is hotswappable\n * Returns an empty `HotswappableChange` if the change is to CDK::Metadata\n * Returns a `NonHotswappableChange` if the change is not hotswappable\n */\nfunction isCandidateForHotswapping(\n  change: cfn_diff.ResourceDifference, logicalId: string,\n): HotswappableChange | NonHotswappableChange | HotswappableChangeCandidate {\n  // a resource has been removed OR a resource has been added; we can't short-circuit that change\n  if (!change.oldValue) {\n    return {\n      hotswappable: false,\n      resourceType: change.newValue!.Type,\n      logicalId,\n      rejectedChanges: [],\n      reason: `resource '${logicalId}' was created by this deployment`,\n    };\n  } else if (!change.newValue) {\n    return {\n      hotswappable: false,\n      resourceType: change.oldValue!.Type,\n      logicalId,\n      rejectedChanges: [],\n      reason: `resource '${logicalId}' was destroyed by this deployment`,\n    };\n  }\n\n  // a resource has had its type changed\n  if (change.newValue?.Type !== change.oldValue?.Type) {\n    return {\n      hotswappable: false,\n      resourceType: change.newValue?.Type,\n      logicalId,\n      rejectedChanges: [],\n      reason: `resource '${logicalId}' had its type changed from '${change.oldValue?.Type}' to '${change.newValue?.Type}'`,\n    };\n  }\n\n  return {\n    logicalId,\n    oldValue: change.oldValue,\n    newValue: change.newValue,\n    propertyUpdates: change.propertyUpdates,\n  };\n}\n\nasync function applyAllHotswappableChanges(sdk: ISDK, hotswappableChanges: HotswappableChange[]): Promise<void[]> {\n  if (hotswappableChanges.length > 0) {\n    print(`\\n${ICON} hotswapping resources:`);\n  }\n  return Promise.all(hotswappableChanges.map(hotswapOperation => {\n    return applyHotswappableChange(sdk, hotswapOperation);\n  }));\n}\n\nasync function applyHotswappableChange(sdk: ISDK, hotswapOperation: HotswappableChange): Promise<void> {\n  // note the type of service that was successfully hotswapped in the User-Agent\n  const customUserAgent = `cdk-hotswap/success-${hotswapOperation.service}`;\n  sdk.appendCustomUserAgent(customUserAgent);\n\n  for (const name of hotswapOperation.resourceNames) {\n    print(`   ${ICON} %s`, chalk.bold(name));\n  }\n\n  // if the SDK call fails, an error will be thrown by the SDK\n  // and will prevent the green 'hotswapped!' text from being displayed\n  await hotswapOperation.apply(sdk);\n\n  for (const name of hotswapOperation.resourceNames) {\n    print(`${ICON} %s %s`, chalk.bold(name), chalk.green('hotswapped!'));\n  }\n\n  sdk.removeCustomUserAgent(customUserAgent);\n}\n\nfunction logNonHotswappableChanges(nonHotswappableChanges: NonHotswappableChange[], hotswapMode: HotswapMode): void {\n  if (nonHotswappableChanges.length === 0) {\n    return;\n  }\n  /**\n   * EKS Services can have a task definition that doesn't refer to the task definition being updated.\n   * We have to log this as a non-hotswappable change to the task definition, but when we do,\n   * we wind up hotswapping the task definition and logging it as a non-hotswappable change.\n   *\n   * This logic prevents us from logging that change as non-hotswappable when we hotswap it.\n   */\n  if (hotswapMode === HotswapMode.HOTSWAP_ONLY) {\n    nonHotswappableChanges = nonHotswappableChanges.filter((change) => change.hotswapOnlyVisible === true);\n\n    if (nonHotswappableChanges.length === 0) {\n      return;\n    }\n  }\n  if (hotswapMode === HotswapMode.HOTSWAP_ONLY) {\n    print('\\n%s %s', chalk.red('⚠️'), chalk.red('The following non-hotswappable changes were found. To reconcile these using CloudFormation, specify --hotswap-fallback'));\n  } else {\n    print('\\n%s %s', chalk.red('⚠️'), chalk.red('The following non-hotswappable changes were found:'));\n  }\n\n  for (const change of nonHotswappableChanges) {\n    change.rejectedChanges.length > 0 ?\n      print('    logicalID: %s, type: %s, rejected changes: %s, reason: %s', chalk.bold(change.logicalId), chalk.bold(change.resourceType), chalk.bold(change.rejectedChanges), chalk.red(change.reason)) :\n      print('    logicalID: %s, type: %s, reason: %s', chalk.bold(change.logicalId), chalk.bold(change.resourceType), chalk.red(change.reason));\n  }\n\n  print(''); // newline\n}\n"]}

@@ -0,3 +1,8 @@

/// <reference types="node" />
import * as cxapi from '@aws-cdk/cx-api';
import { CloudFormation } from 'aws-sdk';
import { StackStatus } from './cloudformation/stack-status';
import { TemplateBodyParameter } from './template-body-parameter';
import { SdkProvider } from '../aws-auth';
import { Deployments } from '../deployments';
export type Template = {

@@ -109,3 +114,31 @@ Parameters?: Record<string, TemplateParameter>;

}): Promise<CloudFormation.DescribeChangeSetOutput>;
export type PrepareChangeSetOptions = {
stack: cxapi.CloudFormationStackArtifact;
deployments: Deployments;
uuid: string;
willExecute: boolean;
sdkProvider: SdkProvider;
stream: NodeJS.WritableStream;
parameters: {
[name: string]: string | undefined;
};
};
export type CreateChangeSetOptions = {
cfn: CloudFormation;
changeSetName: string;
willExecute: boolean;
exists: boolean;
uuid: string;
stack: cxapi.CloudFormationStackArtifact;
bodyParameter: TemplateBodyParameter;
parameters: {
[name: string]: string | undefined;
};
};
/**
* Create a changeset for a diff operation
*/
export declare function createDiffChangeSet(options: PrepareChangeSetOptions): Promise<CloudFormation.DescribeChangeSetOutput | undefined>;
export declare function cleanupOldChangeset(exists: boolean, changeSetName: string, stackName: string, cfn: CloudFormation): Promise<void>;
/**
* Return true if the given change set has no changes

@@ -112,0 +145,0 @@ *

@@ -188,2 +188,15 @@ /// <reference types="node" />

quiet?: boolean;
/**
* Additional parameters for CloudFormation at diff time, used to create a change set
* @default {}
*/
parameters?: {
[name: string]: string | undefined;
};
/**
* Whether or not to create, analyze, and subsequently delete a changeset
*
* @default true
*/
changeSet?: boolean;
}

@@ -368,2 +381,8 @@ interface CfnDeployOptions {

readonly assetBuildTime?: AssetBuildTime;
/**
* Whether to deploy if the app contains no stacks.
*
* @default false
*/
readonly ignoreNoStacks?: boolean;
}

@@ -370,0 +389,0 @@ export interface ImportOptions extends CfnDeployOptions {

import * as cfnDiff from '@aws-cdk/cloudformation-diff';
import * as cxapi from '@aws-cdk/cx-api';
import { CloudFormation } from 'aws-sdk';
/**

@@ -14,3 +15,3 @@ * Pretty-prints the differences between two template states to the console.

*/
export declare function printStackDiff(oldTemplate: any, newTemplate: cxapi.CloudFormationStackArtifact, strict: boolean, context: number, quiet: boolean, stream?: cfnDiff.FormatStream): number;
export declare function printStackDiff(oldTemplate: any, newTemplate: cxapi.CloudFormationStackArtifact, strict: boolean, context: number, quiet: boolean, changeSet?: CloudFormation.DescribeChangeSetOutput, stream?: cfnDiff.FormatStream): number;
export declare enum RequireApproval {

@@ -26,2 +27,2 @@ Never = "never",

*/
export declare function printSecurityDiff(oldTemplate: any, newTemplate: cxapi.CloudFormationStackArtifact, requireApproval: RequireApproval): boolean;
export declare function printSecurityDiff(oldTemplate: any, newTemplate: cxapi.CloudFormationStackArtifact, requireApproval: RequireApproval, changeSet?: CloudFormation.DescribeChangeSetOutput): boolean;

@@ -19,4 +19,4 @@ "use strict";

*/
function printStackDiff(oldTemplate, newTemplate, strict, context, quiet, stream) {
let diff = cfnDiff.diffTemplate(oldTemplate, newTemplate.template);
function printStackDiff(oldTemplate, newTemplate, strict, context, quiet, changeSet, stream) {
let diff = cfnDiff.fullDiff(oldTemplate, newTemplate.template, changeSet);
// detect and filter out mangled characters from the diff

@@ -26,3 +26,3 @@ let filteredChangesCount = 0;

const mangledNewTemplate = JSON.parse(cfnDiff.mangleLikeCloudFormation(JSON.stringify(newTemplate.template)));
const mangledDiff = cfnDiff.diffTemplate(oldTemplate, mangledNewTemplate);
const mangledDiff = cfnDiff.fullDiff(oldTemplate, mangledNewTemplate, changeSet);
filteredChangesCount = Math.max(0, diff.differenceCount - mangledDiff.differenceCount);

@@ -74,4 +74,4 @@ if (filteredChangesCount > 0) {

*/
function printSecurityDiff(oldTemplate, newTemplate, requireApproval) {
const diff = cfnDiff.diffTemplate(oldTemplate, newTemplate.template);
function printSecurityDiff(oldTemplate, newTemplate, requireApproval, changeSet) {
const diff = cfnDiff.fullDiff(oldTemplate, newTemplate.template, changeSet);
if (difRequiresApproval(diff, requireApproval)) {

@@ -118,2 +118,2 @@ // eslint-disable-next-line max-len

}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"diff.js","sourceRoot":"","sources":["diff.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cxschema from '@aws-cdk/cloud-assembly-schema';\nimport * as cfnDiff from '@aws-cdk/cloudformation-diff';\nimport * as cxapi from '@aws-cdk/cx-api';\nimport * as chalk from 'chalk';\nimport { print, warning } from './logging';\n\n/**\n * Pretty-prints the differences between two template states to the console.\n *\n * @param oldTemplate the old/current state of the stack.\n * @param newTemplate the new/target state of the stack.\n * @param strict      do not filter out AWS::CDK::Metadata\n * @param context     lines of context to use in arbitrary JSON diff\n * @param quiet       silences \\'There were no differences\\' messages\n *\n * @returns the count of differences that were rendered.\n */\nexport function printStackDiff(\n  oldTemplate: any,\n  newTemplate: cxapi.CloudFormationStackArtifact,\n  strict: boolean,\n  context: number,\n  quiet: boolean,\n  stream?: cfnDiff.FormatStream): number {\n\n  let diff = cfnDiff.diffTemplate(oldTemplate, newTemplate.template);\n\n  // detect and filter out mangled characters from the diff\n  let filteredChangesCount = 0;\n  if (diff.differenceCount && !strict) {\n    const mangledNewTemplate = JSON.parse(cfnDiff.mangleLikeCloudFormation(JSON.stringify(newTemplate.template)));\n    const mangledDiff = cfnDiff.diffTemplate(oldTemplate, mangledNewTemplate);\n    filteredChangesCount = Math.max(0, diff.differenceCount - mangledDiff.differenceCount);\n    if (filteredChangesCount > 0) {\n      diff = mangledDiff;\n    }\n  }\n\n  // filter out 'AWS::CDK::Metadata' resources from the template\n  if (diff.resources && !strict) {\n    diff.resources = diff.resources.filter(change => {\n      if (!change) { return true; }\n      if (change.newResourceType === 'AWS::CDK::Metadata') { return false; }\n      if (change.oldResourceType === 'AWS::CDK::Metadata') { return false; }\n      return true;\n    });\n  }\n\n  if (!diff.isEmpty) {\n    cfnDiff.formatDifferences(stream || process.stderr, diff, {\n      ...logicalIdMapFromTemplate(oldTemplate),\n      ...buildLogicalToPathMap(newTemplate),\n    }, context);\n  } else if (!quiet) {\n    print(chalk.green('There were no differences'));\n  }\n  if (filteredChangesCount > 0) {\n    print(chalk.yellow(`Omitted ${filteredChangesCount} changes because they are likely mangled non-ASCII characters. Use --strict to print them.`));\n  }\n\n  return diff.differenceCount;\n}\n\nexport enum RequireApproval {\n  Never = 'never',\n\n  AnyChange = 'any-change',\n\n  Broadening = 'broadening'\n}\n\n/**\n * Print the security changes of this diff, if the change is impactful enough according to the approval level\n *\n * Returns true if the changes are prompt-worthy, false otherwise.\n */\nexport function printSecurityDiff(oldTemplate: any, newTemplate: cxapi.CloudFormationStackArtifact, requireApproval: RequireApproval): boolean {\n  const diff = cfnDiff.diffTemplate(oldTemplate, newTemplate.template);\n\n  if (difRequiresApproval(diff, requireApproval)) {\n    // eslint-disable-next-line max-len\n    warning(`This deployment will make potentially sensitive changes according to your current security approval level (--require-approval ${requireApproval}).`);\n    warning('Please confirm you intend to make the following modifications:\\n');\n\n    cfnDiff.formatSecurityChanges(process.stdout, diff, buildLogicalToPathMap(newTemplate));\n    return true;\n  }\n  return false;\n}\n\n/**\n * Return whether the diff has security-impacting changes that need confirmation\n *\n * TODO: Filter the security impact determination based off of an enum that allows\n * us to pick minimum \"severities\" to alert on.\n */\nfunction difRequiresApproval(diff: cfnDiff.TemplateDiff, requireApproval: RequireApproval) {\n  switch (requireApproval) {\n    case RequireApproval.Never: return false;\n    case RequireApproval.AnyChange: return diff.permissionsAnyChanges;\n    case RequireApproval.Broadening: return diff.permissionsBroadened;\n    default: throw new Error(`Unrecognized approval level: ${requireApproval}`);\n  }\n}\n\nfunction buildLogicalToPathMap(stack: cxapi.CloudFormationStackArtifact) {\n  const map: { [id: string]: string } = {};\n  for (const md of stack.findMetadataByType(cxschema.ArtifactMetadataEntryType.LOGICAL_ID)) {\n    map[md.data as string] = md.path;\n  }\n  return map;\n}\n\nfunction logicalIdMapFromTemplate(template: any) {\n  const ret: Record<string, string> = {};\n\n  for (const [logicalId, resource] of Object.entries(template.Resources ?? {})) {\n    const path = (resource as any)?.Metadata?.['aws:cdk:path'];\n    if (path) {\n      ret[logicalId] = path;\n    }\n  }\n  return ret;\n}\n"]}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"diff.js","sourceRoot":"","sources":["diff.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cxschema from '@aws-cdk/cloud-assembly-schema';\nimport * as cfnDiff from '@aws-cdk/cloudformation-diff';\nimport * as cxapi from '@aws-cdk/cx-api';\nimport { CloudFormation } from 'aws-sdk';\nimport * as chalk from 'chalk';\nimport { print, warning } from './logging';\n\n/**\n * Pretty-prints the differences between two template states to the console.\n *\n * @param oldTemplate the old/current state of the stack.\n * @param newTemplate the new/target state of the stack.\n * @param strict      do not filter out AWS::CDK::Metadata\n * @param context     lines of context to use in arbitrary JSON diff\n * @param quiet       silences \\'There were no differences\\' messages\n *\n * @returns the count of differences that were rendered.\n */\nexport function printStackDiff(\n  oldTemplate: any,\n  newTemplate: cxapi.CloudFormationStackArtifact,\n  strict: boolean,\n  context: number,\n  quiet: boolean,\n  changeSet?: CloudFormation.DescribeChangeSetOutput,\n  stream?: cfnDiff.FormatStream): number {\n\n  let diff = cfnDiff.fullDiff(oldTemplate, newTemplate.template, changeSet);\n\n  // detect and filter out mangled characters from the diff\n  let filteredChangesCount = 0;\n  if (diff.differenceCount && !strict) {\n    const mangledNewTemplate = JSON.parse(cfnDiff.mangleLikeCloudFormation(JSON.stringify(newTemplate.template)));\n    const mangledDiff = cfnDiff.fullDiff(oldTemplate, mangledNewTemplate, changeSet);\n    filteredChangesCount = Math.max(0, diff.differenceCount - mangledDiff.differenceCount);\n    if (filteredChangesCount > 0) {\n      diff = mangledDiff;\n    }\n  }\n\n  // filter out 'AWS::CDK::Metadata' resources from the template\n  if (diff.resources && !strict) {\n    diff.resources = diff.resources.filter(change => {\n      if (!change) { return true; }\n      if (change.newResourceType === 'AWS::CDK::Metadata') { return false; }\n      if (change.oldResourceType === 'AWS::CDK::Metadata') { return false; }\n      return true;\n    });\n  }\n\n  if (!diff.isEmpty) {\n    cfnDiff.formatDifferences(stream || process.stderr, diff, {\n      ...logicalIdMapFromTemplate(oldTemplate),\n      ...buildLogicalToPathMap(newTemplate),\n    }, context);\n  } else if (!quiet) {\n    print(chalk.green('There were no differences'));\n  }\n  if (filteredChangesCount > 0) {\n    print(chalk.yellow(`Omitted ${filteredChangesCount} changes because they are likely mangled non-ASCII characters. Use --strict to print them.`));\n  }\n\n  return diff.differenceCount;\n}\n\nexport enum RequireApproval {\n  Never = 'never',\n\n  AnyChange = 'any-change',\n\n  Broadening = 'broadening'\n}\n\n/**\n * Print the security changes of this diff, if the change is impactful enough according to the approval level\n *\n * Returns true if the changes are prompt-worthy, false otherwise.\n */\nexport function printSecurityDiff(\n  oldTemplate: any,\n  newTemplate: cxapi.CloudFormationStackArtifact,\n  requireApproval: RequireApproval,\n  changeSet?: CloudFormation.DescribeChangeSetOutput,\n): boolean {\n  const diff = cfnDiff.fullDiff(oldTemplate, newTemplate.template, changeSet);\n\n  if (difRequiresApproval(diff, requireApproval)) {\n    // eslint-disable-next-line max-len\n    warning(`This deployment will make potentially sensitive changes according to your current security approval level (--require-approval ${requireApproval}).`);\n    warning('Please confirm you intend to make the following modifications:\\n');\n\n    cfnDiff.formatSecurityChanges(process.stdout, diff, buildLogicalToPathMap(newTemplate));\n    return true;\n  }\n  return false;\n}\n\n/**\n * Return whether the diff has security-impacting changes that need confirmation\n *\n * TODO: Filter the security impact determination based off of an enum that allows\n * us to pick minimum \"severities\" to alert on.\n */\nfunction difRequiresApproval(diff: cfnDiff.TemplateDiff, requireApproval: RequireApproval) {\n  switch (requireApproval) {\n    case RequireApproval.Never: return false;\n    case RequireApproval.AnyChange: return diff.permissionsAnyChanges;\n    case RequireApproval.Broadening: return diff.permissionsBroadened;\n    default: throw new Error(`Unrecognized approval level: ${requireApproval}`);\n  }\n}\n\nfunction buildLogicalToPathMap(stack: cxapi.CloudFormationStackArtifact) {\n  const map: { [id: string]: string } = {};\n  for (const md of stack.findMetadataByType(cxschema.ArtifactMetadataEntryType.LOGICAL_ID)) {\n    map[md.data as string] = md.path;\n  }\n  return map;\n}\n\nfunction logicalIdMapFromTemplate(template: any) {\n  const ret: Record<string, string> = {};\n\n  for (const [logicalId, resource] of Object.entries(template.Resources ?? {})) {\n    const path = (resource as any)?.Metadata?.['aws:cdk:path'];\n    if (path) {\n      ret[logicalId] = path;\n    }\n  }\n  return ret;\n}\n"]}

@@ -99,3 +99,3 @@ "use strict";

const currentTemplate = await this.currentTemplate();
const diff = cfnDiff.diffTemplate(currentTemplate, this.stack.template);
const diff = cfnDiff.fullDiff(currentTemplate, this.stack.template);
// Ignore changes to CDKMetadata

@@ -286,2 +286,2 @@ const resourceChanges = Object.entries(diff.resources.changes)

}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"import.js","sourceRoot":"","sources":["import.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cfnDiff from '@aws-cdk/cloudformation-diff';\nimport { ResourceDifference } from '@aws-cdk/cloudformation-diff';\nimport * as cxapi from '@aws-cdk/cx-api';\nimport * as chalk from 'chalk';\nimport * as fs from 'fs-extra';\nimport * as promptly from 'promptly';\nimport { Deployments, DeployStackOptions } from './api/deployments';\nimport { ResourceIdentifierProperties, ResourcesToImport } from './api/util/cloudformation';\nimport { error, print, success, warning } from './logging';\n\n/**\n * Set of parameters that uniquely identify a physical resource of a given type\n * for the import operation, example:\n *\n * ```\n * {\n *   \"AWS::S3::Bucket\": [[\"BucketName\"]],\n *   \"AWS::DynamoDB::GlobalTable\": [[\"TableName\"], [\"TableArn\"], [\"TableStreamArn\"]],\n *   \"AWS::Route53::KeySigningKey\": [[\"HostedZoneId\", \"Name\"]],\n * }\n * ```\n */\nexport type ResourceIdentifiers = { [resourceType: string]: string[][] };\n\n/**\n * Mapping of CDK resources (L1 constructs) to physical resources to be imported\n * in their place, example:\n *\n * ```\n * {\n *   \"MyStack/MyS3Bucket/Resource\": {\n *     \"BucketName\": \"my-manually-created-s3-bucket\"\n *   },\n *   \"MyStack/MyVpc/Resource\": {\n *     \"VpcId\": \"vpc-123456789\"\n *   }\n * }\n * ```\n */\nexport type ResourceMap = { [logicalResource: string]: ResourceIdentifierProperties };\n\n/**\n * Resource importing utility class\n *\n * - Determines the resources added to a template (compared to the deployed version)\n * - Look up the identification information\n *   - Load them from a file, or\n *   - Ask the user, based on information supplied to us by CloudFormation's GetTemplateSummary\n * - Translate the input to a structure expected by CloudFormation, update the template to add the\n *   importable resources, then run an IMPORT changeset.\n */\nexport class ResourceImporter {\n  private _currentTemplate: any;\n\n  constructor(\n    private readonly stack: cxapi.CloudFormationStackArtifact,\n    private readonly cfn: Deployments) { }\n\n  /**\n   * Ask the user for resources to import\n   */\n  public async askForResourceIdentifiers(available: ImportableResource[]): Promise<ImportMap> {\n    const ret: ImportMap = { importResources: [], resourceMap: {} };\n    const resourceIdentifiers = await this.resourceIdentifiers();\n\n    for (const resource of available) {\n      const identifier = await this.askForResourceIdentifier(resourceIdentifiers, resource);\n      if (!identifier) {\n        continue;\n      }\n\n      ret.importResources.push(resource);\n      ret.resourceMap[resource.logicalId] = identifier;\n    }\n\n    return ret;\n  }\n\n  /**\n   * Load the resources to import from a file\n   */\n  public async loadResourceIdentifiers(available: ImportableResource[], filename: string): Promise<ImportMap> {\n    const contents = await fs.readJson(filename);\n\n    const ret: ImportMap = { importResources: [], resourceMap: {} };\n    for (const resource of available) {\n      const descr = this.describeResource(resource.logicalId);\n      const idProps = contents[resource.logicalId];\n      if (idProps) {\n        print('%s: importing using %s', chalk.blue(descr), chalk.blue(fmtdict(idProps)));\n\n        ret.importResources.push(resource);\n        ret.resourceMap[resource.logicalId] = idProps;\n        delete contents[resource.logicalId];\n      } else {\n        print('%s: skipping', chalk.blue(descr));\n      }\n    }\n\n    const unknown = Object.keys(contents);\n    if (unknown.length > 0) {\n      warning(`Unrecognized resource identifiers in mapping file: ${unknown.join(', ')}`);\n    }\n\n    return ret;\n  }\n\n  /**\n   * Based on the provided resource mapping, prepare CFN structures for import (template,\n   * ResourcesToImport structure) and perform the import operation (CloudFormation deployment)\n   *\n   * @param importMap Mapping from CDK construct tree path to physical resource import identifiers\n   * @param options Options to pass to CloudFormation deploy operation\n   */\n  public async importResources(importMap: ImportMap, options: DeployStackOptions) {\n    const resourcesToImport: ResourcesToImport = await this.makeResourcesToImport(importMap);\n    const updatedTemplate = await this.currentTemplateWithAdditions(importMap.importResources);\n\n    try {\n      const result = await this.cfn.deployStack({\n        ...options,\n        overrideTemplate: updatedTemplate,\n        resourcesToImport,\n      });\n\n      const message = result.noOp\n        ? ' ✅  %s (no changes)'\n        : ' ✅  %s';\n\n      success('\\n' + message, options.stack.displayName);\n    } catch (e) {\n      error('\\n ❌  %s failed: %s', chalk.bold(options.stack.displayName), e);\n      throw e;\n    }\n  }\n\n  /**\n   * Perform a diff between the currently running and the new template, ensure that it is valid\n   * for importing and return a list of resources that are being added in the new version\n   *\n   * @return mapping logicalResourceId -> resourceDifference\n   */\n  public async discoverImportableResources(allowNonAdditions = false): Promise<DiscoverImportableResourcesResult> {\n    const currentTemplate = await this.currentTemplate();\n\n    const diff = cfnDiff.diffTemplate(currentTemplate, this.stack.template);\n\n    // Ignore changes to CDKMetadata\n    const resourceChanges = Object.entries(diff.resources.changes)\n      .filter(([logicalId, _]) => logicalId !== 'CDKMetadata');\n\n    // Split the changes into additions and non-additions. Imports only make sense\n    // for newly-added resources.\n    const nonAdditions = resourceChanges.filter(([_, dif]) => !dif.isAddition);\n    const additions = resourceChanges.filter(([_, dif]) => dif.isAddition);\n\n    if (nonAdditions.length) {\n      const offendingResources = nonAdditions.map(([logId, _]) => this.describeResource(logId));\n\n      if (allowNonAdditions) {\n        warning(`Ignoring updated/deleted resources (--force): ${offendingResources.join(', ')}`);\n      } else {\n        throw new Error('No resource updates or deletes are allowed on import operation. Make sure to resolve pending changes ' +\n          `to existing resources, before attempting an import. Updated/deleted resources: ${offendingResources.join(', ')} (--force to override)`);\n      }\n    }\n\n    // Resources in the new template, that are not present in the current template, are a potential import candidates\n    return {\n      additions: additions.map(([logicalId, resourceDiff]) => ({\n        logicalId,\n        resourceDiff,\n        resourceDefinition: addDefaultDeletionPolicy(this.stack.template?.Resources?.[logicalId] ?? {}),\n      })),\n      hasNonAdditions: nonAdditions.length > 0,\n    };\n  }\n\n  /**\n   * Get currently deployed template of the given stack (SINGLETON)\n   *\n   * @returns Currently deployed CloudFormation template\n   */\n  private async currentTemplate(): Promise<any> {\n    if (!this._currentTemplate) {\n      this._currentTemplate = await this.cfn.readCurrentTemplate(this.stack);\n    }\n    return this._currentTemplate;\n  }\n\n  /**\n   * Return the current template, with the given resources added to it\n   */\n  private async currentTemplateWithAdditions(additions: ImportableResource[]): Promise<any> {\n    const template = await this.currentTemplate();\n    if (!template.Resources) {\n      template.Resources = {};\n    }\n\n    for (const add of additions) {\n      template.Resources[add.logicalId] = add.resourceDefinition;\n    }\n\n    return template;\n  }\n\n  /**\n   * Get a list of import identifiers for all resource types used in the given\n   * template that do support the import operation (SINGLETON)\n   *\n   * @returns a mapping from a resource type to a list of property names that together identify the resource for import\n   */\n  private async resourceIdentifiers(): Promise<ResourceIdentifiers> {\n    const ret: ResourceIdentifiers = {};\n    const resourceIdentifierSummaries = await this.cfn.resourceIdentifierSummaries(this.stack);\n    for (const summary of resourceIdentifierSummaries) {\n      if ('ResourceType' in summary && summary.ResourceType && 'ResourceIdentifiers' in summary && summary.ResourceIdentifiers) {\n        ret[summary.ResourceType] = (summary.ResourceIdentifiers ?? [])?.map(x => x.split(','));\n      }\n    }\n    return ret;\n  }\n\n  /**\n   * Ask for the importable identifier for the given resource\n   *\n   * There may be more than one identifier under which a resource can be imported. The `import`\n   * operation needs exactly one of them.\n   *\n   * - If we can get one from the template, we will use one.\n   * - Otherwise, we will ask the user for one of them.\n   */\n  private async askForResourceIdentifier(\n    resourceIdentifiers: ResourceIdentifiers,\n    chg: ImportableResource,\n  ): Promise<ResourceIdentifierProperties | undefined> {\n    const resourceName = this.describeResource(chg.logicalId);\n\n    // Skip resources that do not support importing\n    const resourceType = chg.resourceDiff.newResourceType;\n    if (resourceType === undefined || !(resourceType in resourceIdentifiers)) {\n      warning(`${resourceName}: unsupported resource type ${resourceType}, skipping import.`);\n      return undefined;\n    }\n\n    const idPropSets = resourceIdentifiers[resourceType];\n\n    // Retain only literal strings: strip potential CFN intrinsics\n    const resourceProps = Object.fromEntries(Object.entries(chg.resourceDefinition.Properties ?? {})\n      .filter(([_, v]) => typeof v === 'string')) as Record<string, string>;\n\n    // Find property sets that are fully satisfied in the template, ask the user to confirm them\n    const satisfiedPropSets = idPropSets.filter(ps => ps.every(p => resourceProps[p]));\n    for (const satisfiedPropSet of satisfiedPropSets) {\n      const candidateProps = Object.fromEntries(satisfiedPropSet.map(p => [p, resourceProps[p]]));\n      const displayCandidateProps = fmtdict(candidateProps);\n\n      if (await promptly.confirm(\n        `${chalk.blue(resourceName)} (${resourceType}): import with ${chalk.yellow(displayCandidateProps)} (yes/no) [default: yes]? `,\n        { default: 'yes' },\n      )) {\n        return candidateProps;\n      }\n    }\n\n    // If we got here and the user rejected any available identifiers, then apparently they don't want the resource at all\n    if (satisfiedPropSets.length > 0) {\n      print(chalk.grey(`Skipping import of ${resourceName}`));\n      return undefined;\n    }\n\n    // We cannot auto-import this, ask the user for one of the props\n    // The only difference between these cases is what we print: for multiple properties, we print a preamble\n    const prefix = `${chalk.blue(resourceName)} (${resourceType})`;\n    let preamble;\n    let promptPattern;\n    if (idPropSets.length > 1) {\n      preamble = `${prefix}: enter one of ${idPropSets.map(x => chalk.blue(x.join('+'))).join(', ')} to import (all empty to skip)`;\n      promptPattern = `${prefix}: enter %`;\n    } else {\n      promptPattern = `${prefix}: enter %`;\n    }\n\n    // Do the input loop here\n    if (preamble) {\n      print(preamble);\n    }\n    for (const idProps of idPropSets) {\n      const input: Record<string, string> = {};\n      for (const idProp of idProps) {\n        // If we have a value from the template, use it as default. This will only be a partial\n        // identifier if present, otherwise we would have done the import already above.\n        const defaultValue = resourceProps[idProp] ?? '';\n\n        const prompt = [\n          promptPattern.replace(/%/g, chalk.blue(idProp)),\n          defaultValue\n            ? `[${defaultValue}]`\n            : '(empty to skip)',\n        ].join(' ') + ':';\n        const response = await promptly.prompt(prompt,\n          { default: defaultValue, trim: true },\n        );\n\n        if (!response) {\n          break;\n        }\n\n        input[idProp] = response;\n        // Also stick this property into 'resourceProps', so that it may be reused by a subsequent question\n        // (for a different compound identifier that involves the same property). Just a small UX enhancement.\n        resourceProps[idProp] = response;\n      }\n\n      // If the user gave inputs for all values, we are complete\n      if (Object.keys(input).length === idProps.length) {\n        return input;\n      }\n    }\n\n    print(chalk.grey(`Skipping import of ${resourceName}`));\n    return undefined;\n  }\n\n  /**\n   * Convert the internal \"resource mapping\" structure to CloudFormation accepted \"ResourcesToImport\" structure\n   */\n  private async makeResourcesToImport(resourceMap: ImportMap): Promise<ResourcesToImport> {\n    return resourceMap.importResources.map(res => ({\n      LogicalResourceId: res.logicalId,\n      ResourceType: res.resourceDiff.newResourceType!,\n      ResourceIdentifier: resourceMap.resourceMap[res.logicalId],\n    }));\n  }\n\n  /**\n   * Convert CloudFormation logical resource ID to CDK construct tree path\n   *\n   * @param logicalId CloudFormation logical ID of the resource (the key in the template's Resources section)\n   * @returns Forward-slash separated path of the resource in CDK construct tree, e.g. MyStack/MyBucket/Resource\n   */\n  private describeResource(logicalId: string): string {\n    return this.stack.template?.Resources?.[logicalId]?.Metadata?.['aws:cdk:path'] ?? logicalId;\n  }\n}\n\n/**\n * Information about a resource in the template that is importable\n */\nexport interface ImportableResource {\n  /**\n   * The logical ID of the resource\n   */\n  readonly logicalId: string;\n\n  /**\n   * The resource definition in the new template\n   */\n  readonly resourceDefinition: any;\n\n  /**\n   * The diff as reported by `cloudformation-diff`.\n   */\n  readonly resourceDiff: ResourceDifference;\n}\n\n/**\n * The information necessary to execute an import operation\n */\nexport interface ImportMap {\n  /**\n   * Mapping logical IDs to physical names\n   */\n  readonly resourceMap: ResourceMap;\n\n  /**\n   * The selection of resources we are actually importing\n   *\n   * For each of the resources in this list, there is a corresponding entry in\n   * the `resourceMap` map.\n   */\n  readonly importResources: ImportableResource[];\n}\n\nfunction fmtdict<A>(xs: Record<string, A>) {\n  return Object.entries(xs).map(([k, v]) => `${k}=${v}`).join(', ');\n}\n\n/**\n * Add a default `DeletionPolicy` policy.\n * The default value is set to 'Retain', to lower risk of unintentionally\n * deleting stateful resources in the process of importing to CDK.\n */\nfunction addDefaultDeletionPolicy(resource: any): any {\n  if (resource.DeletionPolicy) { return resource; }\n\n  return {\n    ...resource,\n    DeletionPolicy: 'Retain',\n  };\n}\n\nexport interface DiscoverImportableResourcesResult {\n  readonly additions: ImportableResource[];\n  readonly hasNonAdditions: boolean;\n}\n"]}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"import.js","sourceRoot":"","sources":["import.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cfnDiff from '@aws-cdk/cloudformation-diff';\nimport { ResourceDifference } from '@aws-cdk/cloudformation-diff';\nimport * as cxapi from '@aws-cdk/cx-api';\nimport * as chalk from 'chalk';\nimport * as fs from 'fs-extra';\nimport * as promptly from 'promptly';\nimport { Deployments, DeployStackOptions } from './api/deployments';\nimport { ResourceIdentifierProperties, ResourcesToImport } from './api/util/cloudformation';\nimport { error, print, success, warning } from './logging';\n\n/**\n * Set of parameters that uniquely identify a physical resource of a given type\n * for the import operation, example:\n *\n * ```\n * {\n *   \"AWS::S3::Bucket\": [[\"BucketName\"]],\n *   \"AWS::DynamoDB::GlobalTable\": [[\"TableName\"], [\"TableArn\"], [\"TableStreamArn\"]],\n *   \"AWS::Route53::KeySigningKey\": [[\"HostedZoneId\", \"Name\"]],\n * }\n * ```\n */\nexport type ResourceIdentifiers = { [resourceType: string]: string[][] };\n\n/**\n * Mapping of CDK resources (L1 constructs) to physical resources to be imported\n * in their place, example:\n *\n * ```\n * {\n *   \"MyStack/MyS3Bucket/Resource\": {\n *     \"BucketName\": \"my-manually-created-s3-bucket\"\n *   },\n *   \"MyStack/MyVpc/Resource\": {\n *     \"VpcId\": \"vpc-123456789\"\n *   }\n * }\n * ```\n */\nexport type ResourceMap = { [logicalResource: string]: ResourceIdentifierProperties };\n\n/**\n * Resource importing utility class\n *\n * - Determines the resources added to a template (compared to the deployed version)\n * - Look up the identification information\n *   - Load them from a file, or\n *   - Ask the user, based on information supplied to us by CloudFormation's GetTemplateSummary\n * - Translate the input to a structure expected by CloudFormation, update the template to add the\n *   importable resources, then run an IMPORT changeset.\n */\nexport class ResourceImporter {\n  private _currentTemplate: any;\n\n  constructor(\n    private readonly stack: cxapi.CloudFormationStackArtifact,\n    private readonly cfn: Deployments) { }\n\n  /**\n   * Ask the user for resources to import\n   */\n  public async askForResourceIdentifiers(available: ImportableResource[]): Promise<ImportMap> {\n    const ret: ImportMap = { importResources: [], resourceMap: {} };\n    const resourceIdentifiers = await this.resourceIdentifiers();\n\n    for (const resource of available) {\n      const identifier = await this.askForResourceIdentifier(resourceIdentifiers, resource);\n      if (!identifier) {\n        continue;\n      }\n\n      ret.importResources.push(resource);\n      ret.resourceMap[resource.logicalId] = identifier;\n    }\n\n    return ret;\n  }\n\n  /**\n   * Load the resources to import from a file\n   */\n  public async loadResourceIdentifiers(available: ImportableResource[], filename: string): Promise<ImportMap> {\n    const contents = await fs.readJson(filename);\n\n    const ret: ImportMap = { importResources: [], resourceMap: {} };\n    for (const resource of available) {\n      const descr = this.describeResource(resource.logicalId);\n      const idProps = contents[resource.logicalId];\n      if (idProps) {\n        print('%s: importing using %s', chalk.blue(descr), chalk.blue(fmtdict(idProps)));\n\n        ret.importResources.push(resource);\n        ret.resourceMap[resource.logicalId] = idProps;\n        delete contents[resource.logicalId];\n      } else {\n        print('%s: skipping', chalk.blue(descr));\n      }\n    }\n\n    const unknown = Object.keys(contents);\n    if (unknown.length > 0) {\n      warning(`Unrecognized resource identifiers in mapping file: ${unknown.join(', ')}`);\n    }\n\n    return ret;\n  }\n\n  /**\n   * Based on the provided resource mapping, prepare CFN structures for import (template,\n   * ResourcesToImport structure) and perform the import operation (CloudFormation deployment)\n   *\n   * @param importMap Mapping from CDK construct tree path to physical resource import identifiers\n   * @param options Options to pass to CloudFormation deploy operation\n   */\n  public async importResources(importMap: ImportMap, options: DeployStackOptions) {\n    const resourcesToImport: ResourcesToImport = await this.makeResourcesToImport(importMap);\n    const updatedTemplate = await this.currentTemplateWithAdditions(importMap.importResources);\n\n    try {\n      const result = await this.cfn.deployStack({\n        ...options,\n        overrideTemplate: updatedTemplate,\n        resourcesToImport,\n      });\n\n      const message = result.noOp\n        ? ' ✅  %s (no changes)'\n        : ' ✅  %s';\n\n      success('\\n' + message, options.stack.displayName);\n    } catch (e) {\n      error('\\n ❌  %s failed: %s', chalk.bold(options.stack.displayName), e);\n      throw e;\n    }\n  }\n\n  /**\n   * Perform a diff between the currently running and the new template, ensure that it is valid\n   * for importing and return a list of resources that are being added in the new version\n   *\n   * @return mapping logicalResourceId -> resourceDifference\n   */\n  public async discoverImportableResources(allowNonAdditions = false): Promise<DiscoverImportableResourcesResult> {\n    const currentTemplate = await this.currentTemplate();\n\n    const diff = cfnDiff.fullDiff(currentTemplate, this.stack.template);\n\n    // Ignore changes to CDKMetadata\n    const resourceChanges = Object.entries(diff.resources.changes)\n      .filter(([logicalId, _]) => logicalId !== 'CDKMetadata');\n\n    // Split the changes into additions and non-additions. Imports only make sense\n    // for newly-added resources.\n    const nonAdditions = resourceChanges.filter(([_, dif]) => !dif.isAddition);\n    const additions = resourceChanges.filter(([_, dif]) => dif.isAddition);\n\n    if (nonAdditions.length) {\n      const offendingResources = nonAdditions.map(([logId, _]) => this.describeResource(logId));\n\n      if (allowNonAdditions) {\n        warning(`Ignoring updated/deleted resources (--force): ${offendingResources.join(', ')}`);\n      } else {\n        throw new Error('No resource updates or deletes are allowed on import operation. Make sure to resolve pending changes ' +\n          `to existing resources, before attempting an import. Updated/deleted resources: ${offendingResources.join(', ')} (--force to override)`);\n      }\n    }\n\n    // Resources in the new template, that are not present in the current template, are a potential import candidates\n    return {\n      additions: additions.map(([logicalId, resourceDiff]) => ({\n        logicalId,\n        resourceDiff,\n        resourceDefinition: addDefaultDeletionPolicy(this.stack.template?.Resources?.[logicalId] ?? {}),\n      })),\n      hasNonAdditions: nonAdditions.length > 0,\n    };\n  }\n\n  /**\n   * Get currently deployed template of the given stack (SINGLETON)\n   *\n   * @returns Currently deployed CloudFormation template\n   */\n  private async currentTemplate(): Promise<any> {\n    if (!this._currentTemplate) {\n      this._currentTemplate = await this.cfn.readCurrentTemplate(this.stack);\n    }\n    return this._currentTemplate;\n  }\n\n  /**\n   * Return the current template, with the given resources added to it\n   */\n  private async currentTemplateWithAdditions(additions: ImportableResource[]): Promise<any> {\n    const template = await this.currentTemplate();\n    if (!template.Resources) {\n      template.Resources = {};\n    }\n\n    for (const add of additions) {\n      template.Resources[add.logicalId] = add.resourceDefinition;\n    }\n\n    return template;\n  }\n\n  /**\n   * Get a list of import identifiers for all resource types used in the given\n   * template that do support the import operation (SINGLETON)\n   *\n   * @returns a mapping from a resource type to a list of property names that together identify the resource for import\n   */\n  private async resourceIdentifiers(): Promise<ResourceIdentifiers> {\n    const ret: ResourceIdentifiers = {};\n    const resourceIdentifierSummaries = await this.cfn.resourceIdentifierSummaries(this.stack);\n    for (const summary of resourceIdentifierSummaries) {\n      if ('ResourceType' in summary && summary.ResourceType && 'ResourceIdentifiers' in summary && summary.ResourceIdentifiers) {\n        ret[summary.ResourceType] = (summary.ResourceIdentifiers ?? [])?.map(x => x.split(','));\n      }\n    }\n    return ret;\n  }\n\n  /**\n   * Ask for the importable identifier for the given resource\n   *\n   * There may be more than one identifier under which a resource can be imported. The `import`\n   * operation needs exactly one of them.\n   *\n   * - If we can get one from the template, we will use one.\n   * - Otherwise, we will ask the user for one of them.\n   */\n  private async askForResourceIdentifier(\n    resourceIdentifiers: ResourceIdentifiers,\n    chg: ImportableResource,\n  ): Promise<ResourceIdentifierProperties | undefined> {\n    const resourceName = this.describeResource(chg.logicalId);\n\n    // Skip resources that do not support importing\n    const resourceType = chg.resourceDiff.newResourceType;\n    if (resourceType === undefined || !(resourceType in resourceIdentifiers)) {\n      warning(`${resourceName}: unsupported resource type ${resourceType}, skipping import.`);\n      return undefined;\n    }\n\n    const idPropSets = resourceIdentifiers[resourceType];\n\n    // Retain only literal strings: strip potential CFN intrinsics\n    const resourceProps = Object.fromEntries(Object.entries(chg.resourceDefinition.Properties ?? {})\n      .filter(([_, v]) => typeof v === 'string')) as Record<string, string>;\n\n    // Find property sets that are fully satisfied in the template, ask the user to confirm them\n    const satisfiedPropSets = idPropSets.filter(ps => ps.every(p => resourceProps[p]));\n    for (const satisfiedPropSet of satisfiedPropSets) {\n      const candidateProps = Object.fromEntries(satisfiedPropSet.map(p => [p, resourceProps[p]]));\n      const displayCandidateProps = fmtdict(candidateProps);\n\n      if (await promptly.confirm(\n        `${chalk.blue(resourceName)} (${resourceType}): import with ${chalk.yellow(displayCandidateProps)} (yes/no) [default: yes]? `,\n        { default: 'yes' },\n      )) {\n        return candidateProps;\n      }\n    }\n\n    // If we got here and the user rejected any available identifiers, then apparently they don't want the resource at all\n    if (satisfiedPropSets.length > 0) {\n      print(chalk.grey(`Skipping import of ${resourceName}`));\n      return undefined;\n    }\n\n    // We cannot auto-import this, ask the user for one of the props\n    // The only difference between these cases is what we print: for multiple properties, we print a preamble\n    const prefix = `${chalk.blue(resourceName)} (${resourceType})`;\n    let preamble;\n    let promptPattern;\n    if (idPropSets.length > 1) {\n      preamble = `${prefix}: enter one of ${idPropSets.map(x => chalk.blue(x.join('+'))).join(', ')} to import (all empty to skip)`;\n      promptPattern = `${prefix}: enter %`;\n    } else {\n      promptPattern = `${prefix}: enter %`;\n    }\n\n    // Do the input loop here\n    if (preamble) {\n      print(preamble);\n    }\n    for (const idProps of idPropSets) {\n      const input: Record<string, string> = {};\n      for (const idProp of idProps) {\n        // If we have a value from the template, use it as default. This will only be a partial\n        // identifier if present, otherwise we would have done the import already above.\n        const defaultValue = resourceProps[idProp] ?? '';\n\n        const prompt = [\n          promptPattern.replace(/%/g, chalk.blue(idProp)),\n          defaultValue\n            ? `[${defaultValue}]`\n            : '(empty to skip)',\n        ].join(' ') + ':';\n        const response = await promptly.prompt(prompt,\n          { default: defaultValue, trim: true },\n        );\n\n        if (!response) {\n          break;\n        }\n\n        input[idProp] = response;\n        // Also stick this property into 'resourceProps', so that it may be reused by a subsequent question\n        // (for a different compound identifier that involves the same property). Just a small UX enhancement.\n        resourceProps[idProp] = response;\n      }\n\n      // If the user gave inputs for all values, we are complete\n      if (Object.keys(input).length === idProps.length) {\n        return input;\n      }\n    }\n\n    print(chalk.grey(`Skipping import of ${resourceName}`));\n    return undefined;\n  }\n\n  /**\n   * Convert the internal \"resource mapping\" structure to CloudFormation accepted \"ResourcesToImport\" structure\n   */\n  private async makeResourcesToImport(resourceMap: ImportMap): Promise<ResourcesToImport> {\n    return resourceMap.importResources.map(res => ({\n      LogicalResourceId: res.logicalId,\n      ResourceType: res.resourceDiff.newResourceType!,\n      ResourceIdentifier: resourceMap.resourceMap[res.logicalId],\n    }));\n  }\n\n  /**\n   * Convert CloudFormation logical resource ID to CDK construct tree path\n   *\n   * @param logicalId CloudFormation logical ID of the resource (the key in the template's Resources section)\n   * @returns Forward-slash separated path of the resource in CDK construct tree, e.g. MyStack/MyBucket/Resource\n   */\n  private describeResource(logicalId: string): string {\n    return this.stack.template?.Resources?.[logicalId]?.Metadata?.['aws:cdk:path'] ?? logicalId;\n  }\n}\n\n/**\n * Information about a resource in the template that is importable\n */\nexport interface ImportableResource {\n  /**\n   * The logical ID of the resource\n   */\n  readonly logicalId: string;\n\n  /**\n   * The resource definition in the new template\n   */\n  readonly resourceDefinition: any;\n\n  /**\n   * The diff as reported by `cloudformation-diff`.\n   */\n  readonly resourceDiff: ResourceDifference;\n}\n\n/**\n * The information necessary to execute an import operation\n */\nexport interface ImportMap {\n  /**\n   * Mapping logical IDs to physical names\n   */\n  readonly resourceMap: ResourceMap;\n\n  /**\n   * The selection of resources we are actually importing\n   *\n   * For each of the resources in this list, there is a corresponding entry in\n   * the `resourceMap` map.\n   */\n  readonly importResources: ImportableResource[];\n}\n\nfunction fmtdict<A>(xs: Record<string, A>) {\n  return Object.entries(xs).map(([k, v]) => `${k}=${v}`).join(', ');\n}\n\n/**\n * Add a default `DeletionPolicy` policy.\n * The default value is set to 'Retain', to lower risk of unintentionally\n * deleting stateful resources in the process of importing to CDK.\n */\nfunction addDefaultDeletionPolicy(resource: any): any {\n  if (resource.DeletionPolicy) { return resource; }\n\n  return {\n    ...resource,\n    DeletionPolicy: 'Retain',\n  };\n}\n\nexport interface DiscoverImportableResourcesResult {\n  readonly additions: ImportableResource[];\n  readonly hasNonAdditions: boolean;\n}\n"]}

@@ -15,3 +15,3 @@ {

"@types/jest": "^29.5.11",
"@types/node": "20.10.4",
"@types/node": "20.10.8",
"jest": "^29.7.0",

@@ -28,2 +28,2 @@ "ts-jest": "^29.1.1",

}
}
}

@@ -13,3 +13,3 @@ {

"@types/jest": "^29.5.11",
"@types/node": "20.10.4",
"@types/node": "20.10.8",
"aws-cdk-lib": "%cdk-version%",

@@ -25,2 +25,2 @@ "constructs": "%constructs-version%",

}
}
}

@@ -15,3 +15,3 @@ {

"@types/jest": "^29.5.11",
"@types/node": "20.10.4",
"@types/node": "20.10.8",
"jest": "^29.7.0",

@@ -27,2 +27,2 @@ "ts-jest": "^29.1.1",

}
}
}

@@ -237,2 +237,3 @@ "use strict";

assetPrebuild: argv['asset-prebuild'],
ignoreNoStacks: argv['ignore-no-stacks'],
});

@@ -422,2 +423,2 @@ }

}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"settings.js","sourceRoot":"","sources":["settings.ts"],"names":[],"mappings":"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","sourcesContent":["import * as os from 'os';\nimport * as fs_path from 'path';\nimport * as fs from 'fs-extra';\nimport { Tag } from './cdk-toolkit';\nimport { debug, warning } from './logging';\nimport * as util from './util';\n\nexport type SettingsMap = {[key: string]: any};\n\nexport const PROJECT_CONFIG = 'cdk.json';\nexport const PROJECT_CONTEXT = 'cdk.context.json';\nexport const USER_DEFAULTS = '~/.cdk.json';\n\n/**\n * If a context value is an object with this key set to a truthy value, it won't be saved to cdk.context.json\n */\nexport const TRANSIENT_CONTEXT_KEY = '$dontSaveContext';\n\nconst CONTEXT_KEY = 'context';\n\nexport enum Command {\n  LS = 'ls',\n  LIST = 'list',\n  DIFF = 'diff',\n  BOOTSTRAP = 'bootstrap',\n  DEPLOY = 'deploy',\n  DESTROY = 'destroy',\n  SYNTHESIZE = 'synthesize',\n  SYNTH = 'synth',\n  METADATA = 'metadata',\n  INIT = 'init',\n  VERSION = 'version',\n  WATCH = 'watch',\n}\n\nconst BUNDLING_COMMANDS = [\n  Command.DEPLOY,\n  Command.DIFF,\n  Command.SYNTH,\n  Command.SYNTHESIZE,\n  Command.WATCH,\n];\n\nexport type Arguments = {\n  readonly _: [Command, ...string[]];\n  readonly exclusively?: boolean;\n  readonly STACKS?: string[];\n  readonly lookups?: boolean;\n  readonly [name: string]: unknown;\n};\n\nexport interface ConfigurationProps {\n  /**\n   * Configuration passed via command line arguments\n   *\n   * @default - Nothing passed\n   */\n  readonly commandLineArguments?: Arguments;\n\n  /**\n   * Whether or not to use context from `.cdk.json` in user home directory\n   *\n   * @default true\n   */\n  readonly readUserContext?: boolean;\n}\n\n/**\n * All sources of settings combined\n */\nexport class Configuration {\n  public settings = new Settings();\n  public context = new Context();\n\n  public readonly defaultConfig = new Settings({\n    versionReporting: true,\n    assetMetadata: true,\n    pathMetadata: true,\n    output: 'cdk.out',\n  });\n\n  private readonly commandLineArguments: Settings;\n  private readonly commandLineContext: Settings;\n  private _projectConfig?: Settings;\n  private _projectContext?: Settings;\n  private loaded = false;\n\n  constructor(private readonly props: ConfigurationProps = {}) {\n    this.commandLineArguments = props.commandLineArguments\n      ? Settings.fromCommandLineArguments(props.commandLineArguments)\n      : new Settings();\n    this.commandLineContext = this.commandLineArguments.subSettings([CONTEXT_KEY]).makeReadOnly();\n  }\n\n  private get projectConfig() {\n    if (!this._projectConfig) {\n      throw new Error('#load has not been called yet!');\n    }\n    return this._projectConfig;\n  }\n\n  private get projectContext() {\n    if (!this._projectContext) {\n      throw new Error('#load has not been called yet!');\n    }\n    return this._projectContext;\n  }\n\n  /**\n   * Load all config\n   */\n  public async load(): Promise<this> {\n    const userConfig = await loadAndLog(USER_DEFAULTS);\n    this._projectConfig = await loadAndLog(PROJECT_CONFIG);\n    this._projectContext = await loadAndLog(PROJECT_CONTEXT);\n\n    const readUserContext = this.props.readUserContext ?? true;\n\n    if (userConfig.get(['build'])) {\n      throw new Error('The `build` key cannot be specified in the user config (~/.cdk.json), specify it in the project config (cdk.json) instead');\n    }\n\n    const contextSources = [\n      this.commandLineContext,\n      this.projectConfig.subSettings([CONTEXT_KEY]).makeReadOnly(),\n      this.projectContext,\n    ];\n    if (readUserContext) {\n      contextSources.push(userConfig.subSettings([CONTEXT_KEY]).makeReadOnly());\n    }\n\n    this.context = new Context(...contextSources);\n\n    // Build settings from what's left\n    this.settings = this.defaultConfig\n      .merge(userConfig)\n      .merge(this.projectConfig)\n      .merge(this.commandLineArguments)\n      .makeReadOnly();\n\n    debug('merged settings:', this.settings.all);\n\n    this.loaded = true;\n\n    return this;\n  }\n\n  /**\n   * Save the project context\n   */\n  public async saveContext(): Promise<this> {\n    if (!this.loaded) { return this; } // Avoid overwriting files with nothing\n\n    await this.projectContext.save(PROJECT_CONTEXT);\n\n    return this;\n  }\n}\n\nasync function loadAndLog(fileName: string): Promise<Settings> {\n  const ret = new Settings();\n  await ret.load(fileName);\n  if (!ret.empty) {\n    debug(fileName + ':', JSON.stringify(ret.all, undefined, 2));\n  }\n  return ret;\n}\n\n/**\n * Class that supports overlaying property bags\n *\n * Reads come from the first property bag that can has the given key,\n * writes go to the first property bag that is not readonly. A write\n * will remove the value from all property bags after the first\n * writable one.\n */\nexport class Context {\n  private readonly bags: Settings[];\n\n  constructor(...bags: Settings[]) {\n    this.bags = bags.length > 0 ? bags : [new Settings()];\n  }\n\n  public get keys(): string[] {\n    return Object.keys(this.all);\n  }\n\n  public has(key: string) {\n    return this.keys.indexOf(key) > -1;\n  }\n\n  public get all(): {[key: string]: any} {\n    let ret = new Settings();\n\n    // In reverse order so keys to the left overwrite keys to the right of them\n    for (const bag of [...this.bags].reverse()) {\n      ret = ret.merge(bag);\n    }\n\n    return ret.all;\n  }\n\n  public get(key: string): any {\n    for (const bag of this.bags) {\n      const v = bag.get([key]);\n      if (v !== undefined) { return v; }\n    }\n    return undefined;\n  }\n\n  public set(key: string, value: any) {\n    for (const bag of this.bags) {\n      if (bag.readOnly) { continue; }\n\n      // All bags past the first one have the value erased\n      bag.set([key], value);\n      value = undefined;\n    }\n  }\n\n  public unset(key: string) {\n    this.set(key, undefined);\n  }\n\n  public clear() {\n    for (const key of this.keys) {\n      this.unset(key);\n    }\n  }\n}\n\n/**\n * A single bag of settings\n */\nexport class Settings {\n\n  /**\n   * Parse Settings out of CLI arguments.\n   *\n   * CLI arguments in must be accessed in the CLI code via\n   * `configuration.settings.get(['argName'])` instead of via `args.argName`.\n   *\n   * The advantage is that they can be configured via `cdk.json` and\n   * `$HOME/.cdk.json`. Arguments not listed below and accessed via this object\n   * can only be specified on the command line.\n   *\n   * @param argv the received CLI arguments.\n   * @returns a new Settings object.\n   */\n  public static fromCommandLineArguments(argv: Arguments): Settings {\n    const context = this.parseStringContextListToObject(argv);\n    const tags = this.parseStringTagsListToObject(expectStringList(argv.tags));\n\n    // Determine bundling stacks\n    let bundlingStacks: string[];\n    if (BUNDLING_COMMANDS.includes(argv._[0])) {\n    // If we deploy, diff, synth or watch a list of stacks exclusively we skip\n    // bundling for all other stacks.\n      bundlingStacks = argv.exclusively\n        ? argv.STACKS ?? ['**']\n        : ['**'];\n    } else { // Skip bundling for all stacks\n      bundlingStacks = [];\n    }\n\n    return new Settings({\n      app: argv.app,\n      browser: argv.browser,\n      build: argv.build,\n      context,\n      debug: argv.debug,\n      tags,\n      language: argv.language,\n      pathMetadata: argv.pathMetadata,\n      assetMetadata: argv.assetMetadata,\n      profile: argv.profile,\n      plugin: argv.plugin,\n      requireApproval: argv.requireApproval,\n      toolkitStackName: argv.toolkitStackName,\n      toolkitBucket: {\n        bucketName: argv.bootstrapBucketName,\n        kmsKeyId: argv.bootstrapKmsKeyId,\n      },\n      versionReporting: argv.versionReporting,\n      staging: argv.staging,\n      output: argv.output,\n      outputsFile: argv.outputsFile,\n      progress: argv.progress,\n      bundlingStacks,\n      lookups: argv.lookups,\n      rollback: argv.rollback,\n      notices: argv.notices,\n      assetParallelism: argv['asset-parallelism'],\n      assetPrebuild: argv['asset-prebuild'],\n    });\n  }\n\n  public static mergeAll(...settings: Settings[]): Settings {\n    let ret = new Settings();\n    for (const setting of settings) {\n      ret = ret.merge(setting);\n    }\n    return ret;\n  }\n\n  private static parseStringContextListToObject(argv: Arguments): any {\n    const context: any = {};\n\n    for (const assignment of ((argv as any).context || [])) {\n      const parts = assignment.split(/=(.*)/, 2);\n      if (parts.length === 2) {\n        debug('CLI argument context: %s=%s', parts[0], parts[1]);\n        if (parts[0].match(/^aws:.+/)) {\n          throw new Error(`User-provided context cannot use keys prefixed with 'aws:', but ${parts[0]} was provided.`);\n        }\n        context[parts[0]] = parts[1];\n      } else {\n        warning('Context argument is not an assignment (key=value): %s', assignment);\n      }\n    }\n    return context;\n  }\n\n  /**\n   * Parse tags out of arguments\n   *\n   * Return undefined if no tags were provided, return an empty array if only empty\n   * strings were provided\n   */\n  private static parseStringTagsListToObject(argTags: string[] | undefined): Tag[] | undefined {\n    if (argTags === undefined) { return undefined; }\n    if (argTags.length === 0) { return undefined; }\n    const nonEmptyTags = argTags.filter(t => t !== '');\n    if (nonEmptyTags.length === 0) { return []; }\n\n    const tags: Tag[] = [];\n\n    for (const assignment of nonEmptyTags) {\n      const parts = assignment.split(/=(.*)/, 2);\n      if (parts.length === 2) {\n        debug('CLI argument tags: %s=%s', parts[0], parts[1]);\n        tags.push({\n          Key: parts[0],\n          Value: parts[1],\n        });\n      } else {\n        warning('Tags argument is not an assignment (key=value): %s', assignment);\n      }\n    }\n    return tags.length > 0 ? tags : undefined;\n  }\n\n  constructor(private settings: SettingsMap = {}, public readonly readOnly = false) {}\n\n  public async load(fileName: string): Promise<this> {\n    if (this.readOnly) {\n      throw new Error(`Can't load ${fileName}: settings object is readonly`);\n    }\n    this.settings = {};\n\n    const expanded = expandHomeDir(fileName);\n    if (await fs.pathExists(expanded)) {\n      this.settings = await fs.readJson(expanded);\n    }\n\n    // See https://github.com/aws/aws-cdk/issues/59\n    this.prohibitContextKey('default-account', fileName);\n    this.prohibitContextKey('default-region', fileName);\n    this.warnAboutContextKey('aws:', fileName);\n\n    return this;\n  }\n\n  public async save(fileName: string): Promise<this> {\n    const expanded = expandHomeDir(fileName);\n    await fs.writeJson(expanded, stripTransientValues(this.settings), { spaces: 2 });\n    return this;\n  }\n\n  public get all(): any {\n    return this.get([]);\n  }\n\n  public merge(other: Settings): Settings {\n    return new Settings(util.deepMerge(this.settings, other.settings));\n  }\n\n  public subSettings(keyPrefix: string[]) {\n    return new Settings(this.get(keyPrefix) || {}, false);\n  }\n\n  public makeReadOnly(): Settings {\n    return new Settings(this.settings, true);\n  }\n\n  public clear() {\n    if (this.readOnly) {\n      throw new Error('Cannot clear(): settings are readonly');\n    }\n    this.settings = {};\n  }\n\n  public get empty(): boolean {\n    return Object.keys(this.settings).length === 0;\n  }\n\n  public get(path: string[]): any {\n    return util.deepClone(util.deepGet(this.settings, path));\n  }\n\n  public set(path: string[], value: any): Settings {\n    if (this.readOnly) {\n      throw new Error(`Can't set ${path}: settings object is readonly`);\n    }\n    if (path.length === 0) {\n      // deepSet can't handle this case\n      this.settings = value;\n    } else {\n      util.deepSet(this.settings, path, value);\n    }\n    return this;\n  }\n\n  public unset(path: string[]) {\n    this.set(path, undefined);\n  }\n\n  private prohibitContextKey(key: string, fileName: string) {\n    if (!this.settings.context) { return; }\n    if (key in this.settings.context) {\n      // eslint-disable-next-line max-len\n      throw new Error(`The 'context.${key}' key was found in ${fs_path.resolve(fileName)}, but it is no longer supported. Please remove it.`);\n    }\n  }\n\n  private warnAboutContextKey(prefix: string, fileName: string) {\n    if (!this.settings.context) { return; }\n    for (const contextKey of Object.keys(this.settings.context)) {\n      if (contextKey.startsWith(prefix)) {\n        // eslint-disable-next-line max-len\n        warning(`A reserved context key ('context.${prefix}') key was found in ${fs_path.resolve(fileName)}, it might cause surprising behavior and should be removed.`);\n      }\n    }\n  }\n}\n\nfunction expandHomeDir(x: string) {\n  if (x.startsWith('~')) {\n    return fs_path.join(os.homedir(), x.slice(1));\n  }\n  return x;\n}\n\n/**\n * Return all context value that are not transient context values\n */\nfunction stripTransientValues(obj: {[key: string]: any}) {\n  const ret: any = {};\n  for (const [key, value] of Object.entries(obj)) {\n    if (!isTransientValue(value)) {\n      ret[key] = value;\n    }\n  }\n  return ret;\n}\n\n/**\n * Return whether the given value is a transient context value\n *\n * Values that are objects with a magic key set to a truthy value are considered transient.\n */\nfunction isTransientValue(value: any) {\n  return typeof value === 'object' && value !== null && (value as any)[TRANSIENT_CONTEXT_KEY];\n}\n\nfunction expectStringList(x: unknown): string[] | undefined {\n  if (x === undefined) { return undefined; }\n  if (!Array.isArray(x)) {\n    throw new Error(`Expected array, got '${x}'`);\n  }\n  const nonStrings = x.filter(e => typeof e !== 'string');\n  if (nonStrings.length > 0) {\n    throw new Error(`Expected list of strings, found ${nonStrings}`);\n  }\n  return x;\n}\n"]}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"settings.js","sourceRoot":"","sources":["settings.ts"],"names":[],"mappings":"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","sourcesContent":["import * as os from 'os';\nimport * as fs_path from 'path';\nimport * as fs from 'fs-extra';\nimport { Tag } from './cdk-toolkit';\nimport { debug, warning } from './logging';\nimport * as util from './util';\n\nexport type SettingsMap = {[key: string]: any};\n\nexport const PROJECT_CONFIG = 'cdk.json';\nexport const PROJECT_CONTEXT = 'cdk.context.json';\nexport const USER_DEFAULTS = '~/.cdk.json';\n\n/**\n * If a context value is an object with this key set to a truthy value, it won't be saved to cdk.context.json\n */\nexport const TRANSIENT_CONTEXT_KEY = '$dontSaveContext';\n\nconst CONTEXT_KEY = 'context';\n\nexport enum Command {\n  LS = 'ls',\n  LIST = 'list',\n  DIFF = 'diff',\n  BOOTSTRAP = 'bootstrap',\n  DEPLOY = 'deploy',\n  DESTROY = 'destroy',\n  SYNTHESIZE = 'synthesize',\n  SYNTH = 'synth',\n  METADATA = 'metadata',\n  INIT = 'init',\n  VERSION = 'version',\n  WATCH = 'watch',\n}\n\nconst BUNDLING_COMMANDS = [\n  Command.DEPLOY,\n  Command.DIFF,\n  Command.SYNTH,\n  Command.SYNTHESIZE,\n  Command.WATCH,\n];\n\nexport type Arguments = {\n  readonly _: [Command, ...string[]];\n  readonly exclusively?: boolean;\n  readonly STACKS?: string[];\n  readonly lookups?: boolean;\n  readonly [name: string]: unknown;\n};\n\nexport interface ConfigurationProps {\n  /**\n   * Configuration passed via command line arguments\n   *\n   * @default - Nothing passed\n   */\n  readonly commandLineArguments?: Arguments;\n\n  /**\n   * Whether or not to use context from `.cdk.json` in user home directory\n   *\n   * @default true\n   */\n  readonly readUserContext?: boolean;\n}\n\n/**\n * All sources of settings combined\n */\nexport class Configuration {\n  public settings = new Settings();\n  public context = new Context();\n\n  public readonly defaultConfig = new Settings({\n    versionReporting: true,\n    assetMetadata: true,\n    pathMetadata: true,\n    output: 'cdk.out',\n  });\n\n  private readonly commandLineArguments: Settings;\n  private readonly commandLineContext: Settings;\n  private _projectConfig?: Settings;\n  private _projectContext?: Settings;\n  private loaded = false;\n\n  constructor(private readonly props: ConfigurationProps = {}) {\n    this.commandLineArguments = props.commandLineArguments\n      ? Settings.fromCommandLineArguments(props.commandLineArguments)\n      : new Settings();\n    this.commandLineContext = this.commandLineArguments.subSettings([CONTEXT_KEY]).makeReadOnly();\n  }\n\n  private get projectConfig() {\n    if (!this._projectConfig) {\n      throw new Error('#load has not been called yet!');\n    }\n    return this._projectConfig;\n  }\n\n  private get projectContext() {\n    if (!this._projectContext) {\n      throw new Error('#load has not been called yet!');\n    }\n    return this._projectContext;\n  }\n\n  /**\n   * Load all config\n   */\n  public async load(): Promise<this> {\n    const userConfig = await loadAndLog(USER_DEFAULTS);\n    this._projectConfig = await loadAndLog(PROJECT_CONFIG);\n    this._projectContext = await loadAndLog(PROJECT_CONTEXT);\n\n    const readUserContext = this.props.readUserContext ?? true;\n\n    if (userConfig.get(['build'])) {\n      throw new Error('The `build` key cannot be specified in the user config (~/.cdk.json), specify it in the project config (cdk.json) instead');\n    }\n\n    const contextSources = [\n      this.commandLineContext,\n      this.projectConfig.subSettings([CONTEXT_KEY]).makeReadOnly(),\n      this.projectContext,\n    ];\n    if (readUserContext) {\n      contextSources.push(userConfig.subSettings([CONTEXT_KEY]).makeReadOnly());\n    }\n\n    this.context = new Context(...contextSources);\n\n    // Build settings from what's left\n    this.settings = this.defaultConfig\n      .merge(userConfig)\n      .merge(this.projectConfig)\n      .merge(this.commandLineArguments)\n      .makeReadOnly();\n\n    debug('merged settings:', this.settings.all);\n\n    this.loaded = true;\n\n    return this;\n  }\n\n  /**\n   * Save the project context\n   */\n  public async saveContext(): Promise<this> {\n    if (!this.loaded) { return this; } // Avoid overwriting files with nothing\n\n    await this.projectContext.save(PROJECT_CONTEXT);\n\n    return this;\n  }\n}\n\nasync function loadAndLog(fileName: string): Promise<Settings> {\n  const ret = new Settings();\n  await ret.load(fileName);\n  if (!ret.empty) {\n    debug(fileName + ':', JSON.stringify(ret.all, undefined, 2));\n  }\n  return ret;\n}\n\n/**\n * Class that supports overlaying property bags\n *\n * Reads come from the first property bag that can has the given key,\n * writes go to the first property bag that is not readonly. A write\n * will remove the value from all property bags after the first\n * writable one.\n */\nexport class Context {\n  private readonly bags: Settings[];\n\n  constructor(...bags: Settings[]) {\n    this.bags = bags.length > 0 ? bags : [new Settings()];\n  }\n\n  public get keys(): string[] {\n    return Object.keys(this.all);\n  }\n\n  public has(key: string) {\n    return this.keys.indexOf(key) > -1;\n  }\n\n  public get all(): {[key: string]: any} {\n    let ret = new Settings();\n\n    // In reverse order so keys to the left overwrite keys to the right of them\n    for (const bag of [...this.bags].reverse()) {\n      ret = ret.merge(bag);\n    }\n\n    return ret.all;\n  }\n\n  public get(key: string): any {\n    for (const bag of this.bags) {\n      const v = bag.get([key]);\n      if (v !== undefined) { return v; }\n    }\n    return undefined;\n  }\n\n  public set(key: string, value: any) {\n    for (const bag of this.bags) {\n      if (bag.readOnly) { continue; }\n\n      // All bags past the first one have the value erased\n      bag.set([key], value);\n      value = undefined;\n    }\n  }\n\n  public unset(key: string) {\n    this.set(key, undefined);\n  }\n\n  public clear() {\n    for (const key of this.keys) {\n      this.unset(key);\n    }\n  }\n}\n\n/**\n * A single bag of settings\n */\nexport class Settings {\n\n  /**\n   * Parse Settings out of CLI arguments.\n   *\n   * CLI arguments in must be accessed in the CLI code via\n   * `configuration.settings.get(['argName'])` instead of via `args.argName`.\n   *\n   * The advantage is that they can be configured via `cdk.json` and\n   * `$HOME/.cdk.json`. Arguments not listed below and accessed via this object\n   * can only be specified on the command line.\n   *\n   * @param argv the received CLI arguments.\n   * @returns a new Settings object.\n   */\n  public static fromCommandLineArguments(argv: Arguments): Settings {\n    const context = this.parseStringContextListToObject(argv);\n    const tags = this.parseStringTagsListToObject(expectStringList(argv.tags));\n\n    // Determine bundling stacks\n    let bundlingStacks: string[];\n    if (BUNDLING_COMMANDS.includes(argv._[0])) {\n    // If we deploy, diff, synth or watch a list of stacks exclusively we skip\n    // bundling for all other stacks.\n      bundlingStacks = argv.exclusively\n        ? argv.STACKS ?? ['**']\n        : ['**'];\n    } else { // Skip bundling for all stacks\n      bundlingStacks = [];\n    }\n\n    return new Settings({\n      app: argv.app,\n      browser: argv.browser,\n      build: argv.build,\n      context,\n      debug: argv.debug,\n      tags,\n      language: argv.language,\n      pathMetadata: argv.pathMetadata,\n      assetMetadata: argv.assetMetadata,\n      profile: argv.profile,\n      plugin: argv.plugin,\n      requireApproval: argv.requireApproval,\n      toolkitStackName: argv.toolkitStackName,\n      toolkitBucket: {\n        bucketName: argv.bootstrapBucketName,\n        kmsKeyId: argv.bootstrapKmsKeyId,\n      },\n      versionReporting: argv.versionReporting,\n      staging: argv.staging,\n      output: argv.output,\n      outputsFile: argv.outputsFile,\n      progress: argv.progress,\n      bundlingStacks,\n      lookups: argv.lookups,\n      rollback: argv.rollback,\n      notices: argv.notices,\n      assetParallelism: argv['asset-parallelism'],\n      assetPrebuild: argv['asset-prebuild'],\n      ignoreNoStacks: argv['ignore-no-stacks'],\n    });\n  }\n\n  public static mergeAll(...settings: Settings[]): Settings {\n    let ret = new Settings();\n    for (const setting of settings) {\n      ret = ret.merge(setting);\n    }\n    return ret;\n  }\n\n  private static parseStringContextListToObject(argv: Arguments): any {\n    const context: any = {};\n\n    for (const assignment of ((argv as any).context || [])) {\n      const parts = assignment.split(/=(.*)/, 2);\n      if (parts.length === 2) {\n        debug('CLI argument context: %s=%s', parts[0], parts[1]);\n        if (parts[0].match(/^aws:.+/)) {\n          throw new Error(`User-provided context cannot use keys prefixed with 'aws:', but ${parts[0]} was provided.`);\n        }\n        context[parts[0]] = parts[1];\n      } else {\n        warning('Context argument is not an assignment (key=value): %s', assignment);\n      }\n    }\n    return context;\n  }\n\n  /**\n   * Parse tags out of arguments\n   *\n   * Return undefined if no tags were provided, return an empty array if only empty\n   * strings were provided\n   */\n  private static parseStringTagsListToObject(argTags: string[] | undefined): Tag[] | undefined {\n    if (argTags === undefined) { return undefined; }\n    if (argTags.length === 0) { return undefined; }\n    const nonEmptyTags = argTags.filter(t => t !== '');\n    if (nonEmptyTags.length === 0) { return []; }\n\n    const tags: Tag[] = [];\n\n    for (const assignment of nonEmptyTags) {\n      const parts = assignment.split(/=(.*)/, 2);\n      if (parts.length === 2) {\n        debug('CLI argument tags: %s=%s', parts[0], parts[1]);\n        tags.push({\n          Key: parts[0],\n          Value: parts[1],\n        });\n      } else {\n        warning('Tags argument is not an assignment (key=value): %s', assignment);\n      }\n    }\n    return tags.length > 0 ? tags : undefined;\n  }\n\n  constructor(private settings: SettingsMap = {}, public readonly readOnly = false) {}\n\n  public async load(fileName: string): Promise<this> {\n    if (this.readOnly) {\n      throw new Error(`Can't load ${fileName}: settings object is readonly`);\n    }\n    this.settings = {};\n\n    const expanded = expandHomeDir(fileName);\n    if (await fs.pathExists(expanded)) {\n      this.settings = await fs.readJson(expanded);\n    }\n\n    // See https://github.com/aws/aws-cdk/issues/59\n    this.prohibitContextKey('default-account', fileName);\n    this.prohibitContextKey('default-region', fileName);\n    this.warnAboutContextKey('aws:', fileName);\n\n    return this;\n  }\n\n  public async save(fileName: string): Promise<this> {\n    const expanded = expandHomeDir(fileName);\n    await fs.writeJson(expanded, stripTransientValues(this.settings), { spaces: 2 });\n    return this;\n  }\n\n  public get all(): any {\n    return this.get([]);\n  }\n\n  public merge(other: Settings): Settings {\n    return new Settings(util.deepMerge(this.settings, other.settings));\n  }\n\n  public subSettings(keyPrefix: string[]) {\n    return new Settings(this.get(keyPrefix) || {}, false);\n  }\n\n  public makeReadOnly(): Settings {\n    return new Settings(this.settings, true);\n  }\n\n  public clear() {\n    if (this.readOnly) {\n      throw new Error('Cannot clear(): settings are readonly');\n    }\n    this.settings = {};\n  }\n\n  public get empty(): boolean {\n    return Object.keys(this.settings).length === 0;\n  }\n\n  public get(path: string[]): any {\n    return util.deepClone(util.deepGet(this.settings, path));\n  }\n\n  public set(path: string[], value: any): Settings {\n    if (this.readOnly) {\n      throw new Error(`Can't set ${path}: settings object is readonly`);\n    }\n    if (path.length === 0) {\n      // deepSet can't handle this case\n      this.settings = value;\n    } else {\n      util.deepSet(this.settings, path, value);\n    }\n    return this;\n  }\n\n  public unset(path: string[]) {\n    this.set(path, undefined);\n  }\n\n  private prohibitContextKey(key: string, fileName: string) {\n    if (!this.settings.context) { return; }\n    if (key in this.settings.context) {\n      // eslint-disable-next-line max-len\n      throw new Error(`The 'context.${key}' key was found in ${fs_path.resolve(fileName)}, but it is no longer supported. Please remove it.`);\n    }\n  }\n\n  private warnAboutContextKey(prefix: string, fileName: string) {\n    if (!this.settings.context) { return; }\n    for (const contextKey of Object.keys(this.settings.context)) {\n      if (contextKey.startsWith(prefix)) {\n        // eslint-disable-next-line max-len\n        warning(`A reserved context key ('context.${prefix}') key was found in ${fs_path.resolve(fileName)}, it might cause surprising behavior and should be removed.`);\n      }\n    }\n  }\n}\n\nfunction expandHomeDir(x: string) {\n  if (x.startsWith('~')) {\n    return fs_path.join(os.homedir(), x.slice(1));\n  }\n  return x;\n}\n\n/**\n * Return all context value that are not transient context values\n */\nfunction stripTransientValues(obj: {[key: string]: any}) {\n  const ret: any = {};\n  for (const [key, value] of Object.entries(obj)) {\n    if (!isTransientValue(value)) {\n      ret[key] = value;\n    }\n  }\n  return ret;\n}\n\n/**\n * Return whether the given value is a transient context value\n *\n * Values that are objects with a magic key set to a truthy value are considered transient.\n */\nfunction isTransientValue(value: any) {\n  return typeof value === 'object' && value !== null && (value as any)[TRANSIENT_CONTEXT_KEY];\n}\n\nfunction expectStringList(x: unknown): string[] | undefined {\n  if (x === undefined) { return undefined; }\n  if (!Array.isArray(x)) {\n    throw new Error(`Expected array, got '${x}'`);\n  }\n  const nonStrings = x.filter(e => typeof e !== 'string');\n  if (nonStrings.length > 0) {\n    throw new Error(`Expected list of strings, found ${nonStrings}`);\n  }\n  return x;\n}\n"]}
{
"name": "aws-cdk",
"description": "CDK Toolkit, the command line tool for CDK apps",
"version": "2.118.0",
"version": "2.119.0",
"bin": {

@@ -67,4 +67,4 @@ "cdk": "bin/cdk"

"devDependencies": {
"@aws-cdk/cdk-build-tools": "2.118.0-alpha.0",
"@aws-cdk/pkglint": "2.118.0-alpha.0",
"@aws-cdk/cdk-build-tools": "2.119.0-alpha.0",
"@aws-cdk/pkglint": "2.119.0-alpha.0",
"@octokit/rest": "^18.12.0",

@@ -84,7 +84,7 @@ "@types/archiver": "^5.3.4",

"@types/yargs": "^15.0.19",
"aws-cdk-lib": "2.118.0",
"aws-cdk-lib": "2.119.0",
"aws-sdk-mock": "5.6.0",
"axios": "^1.6.2",
"axios": "^1.6.5",
"constructs": "^10.0.0",
"fast-check": "^3.14.0",
"fast-check": "^3.15.0",
"jest": "^29.7.0",

@@ -99,12 +99,12 @@ "jest-mock": "^29.7.0",

"xml-js": "^1.6.11",
"@aws-cdk/cloud-assembly-schema": "2.118.0",
"@aws-cdk/cloudformation-diff": "2.118.0",
"@aws-cdk/cx-api": "2.118.0",
"@aws-cdk/region-info": "2.118.0",
"@jsii/check-node": "1.93.0",
"@aws-cdk/cloud-assembly-schema": "2.119.0",
"@aws-cdk/cloudformation-diff": "2.119.0",
"@aws-cdk/cx-api": "2.119.0",
"@aws-cdk/region-info": "2.119.0",
"@jsii/check-node": "1.94.0",
"archiver": "^5.3.2",
"aws-sdk": "^2.1517.0",
"aws-sdk": "^2.1532.0",
"camelcase": "^6.3.0",
"cdk-assets": "2.118.0",
"cdk-from-cfn": "^0.91.0",
"cdk-assets": "2.119.0",
"cdk-from-cfn": "^0.116.0",
"chalk": "^4",

@@ -111,0 +111,0 @@ "chokidar": "^3.5.3",

@@ -168,2 +168,6 @@ # AWS CDK Toolkit

The `change-set` flag will make `diff` create a change set and extract resource replacement data from it. This is a bit slower, but will provide no false positives for resource replacement.
The `--no-change-set` mode will consider any change to a property that requires replacement to be a resource replacement,
even if the change is purely cosmetic (like replacing a resource reference with a hardcoded arn).
### `cdk deploy`

@@ -386,2 +390,16 @@

#### Ignore No Stacks
You may have an app with multiple environments, e.g., dev and prod. When starting
development, your prod app may not have any resources or the resources are commented
out. In this scenario, you will receive an error message stating that the app has no
stacks.
To bypass this error messages, you can pass the `--ignore-no-stacks` flag to the
`deploy` command:
```console
$ cdk deploy --ignore-no-stacks
```
#### Hotswap deployments for faster development

@@ -388,0 +406,0 @@

@@ -123,2 +123,27 @@ "use strict";

});
test('select behavior with no stacks and ignore stacks option', async () => {
// GIVEN
const cxasm = await testCloudAssemblyNoStacks();
// WHEN
const x = await cxasm.selectStacks({ patterns: [] }, {
defaultBehavior: cloud_assembly_1.DefaultSelection.AllStacks,
ignoreNoStacks: true,
});
// THEN
expect(x.stackCount).toBe(0);
});
test('select behavior with no stacks and no ignore stacks option', async () => {
// GIVEN
const cxasm = await testCloudAssemblyNoStacks();
// WHEN & THEN
await expect(cxasm.selectStacks({ patterns: [] }, { defaultBehavior: cloud_assembly_1.DefaultSelection.AllStacks, ignoreNoStacks: false }))
.rejects.toThrow('This app contains no stacks');
});
test('select behavior with no stacks and default ignore stacks options (false)', async () => {
// GIVEN
const cxasm = await testCloudAssemblyNoStacks();
// WHEN & THEN
await expect(cxasm.selectStacks({ patterns: [] }, { defaultBehavior: cloud_assembly_1.DefaultSelection.AllStacks }))
.rejects.toThrow('This app contains no stacks');
});
async function testCloudAssembly({ env } = {}) {

@@ -148,2 +173,8 @@ const cloudExec = new util_1.MockCloudExecutable({

}
async function testCloudAssemblyNoStacks() {
const cloudExec = new util_1.MockCloudExecutable({
stacks: [],
});
return cloudExec.synthesize();
}
async function testNestedCloudAssembly({ env } = {}) {

@@ -189,2 +220,2 @@ const cloudExec = new util_1.MockCloudExecutable({

}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cloud-assembly.test.js","sourceRoot":"","sources":["cloud-assembly.test.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable import/order */\nimport * as cxschema from '@aws-cdk/cloud-assembly-schema';\nimport { DefaultSelection } from '../../lib/api/cxapp/cloud-assembly';\nimport { MockCloudExecutable } from '../util';\n\n// behave like v2\nprocess.env.CXAPI_DISABLE_SELECT_BY_ID = '1';\n\ntest('do not throw when selecting stack without errors', async () => {\n  // GIVEN\n  const cxasm = await testCloudAssembly();\n\n  // WHEN\n  const selected = await cxasm.selectStacks( { patterns: ['withouterrorsNODEPATH'] }, {\n    defaultBehavior: DefaultSelection.AllStacks,\n  });\n  selected.processMetadataMessages();\n\n  // THEN\n  expect(selected.firstStack.template.resource).toBe('noerrorresource');\n});\n\ntest('do throw when selecting stack with errors', async () => {\n  // GIVEN\n  const cxasm = await testCloudAssembly();\n\n  // WHEN\n  const selected = await cxasm.selectStacks({ patterns: ['witherrors'] }, {\n    defaultBehavior: DefaultSelection.AllStacks,\n  });\n\n  // THEN\n  expect(() => selected.processMetadataMessages()).toThrow(/Found errors/);\n});\n\ntest('select all top level stacks in the presence of nested assemblies', async () => {\n  // GIVEN\n  const cxasm = await testNestedCloudAssembly();\n\n  // WHEN\n  const x = await cxasm.selectStacks({ allTopLevel: true, patterns: [] }, { defaultBehavior: DefaultSelection.AllStacks });\n\n  // THEN\n  expect(x.stackCount).toBe(2);\n  expect(x.stackIds).toContain('witherrors');\n  expect(x.stackIds).toContain('withouterrors');\n});\n\ntest('select stacks by glob pattern', async () => {\n  // GIVEN\n  const cxasm = await testCloudAssembly();\n\n  // WHEN\n  const x = await cxasm.selectStacks({ patterns: ['with*'] }, { defaultBehavior: DefaultSelection.AllStacks });\n\n  // THEN\n  expect(x.stackCount).toBe(2);\n  expect(x.stackIds).toContain('witherrors');\n  expect(x.stackIds).toContain('withouterrors');\n});\n\ntest('select behavior: all', async () => {\n  // GIVEN\n  const cxasm = await testCloudAssembly();\n\n  // WHEN\n  const x = await cxasm.selectStacks({ patterns: [] }, { defaultBehavior: DefaultSelection.AllStacks });\n\n  // THEN\n  expect(x.stackCount).toBe(2);\n});\n\ntest('select behavior: none', async () => {\n  // GIVEN\n  const cxasm = await testCloudAssembly();\n\n  // WHEN\n  const x = await cxasm.selectStacks({ patterns: [] }, { defaultBehavior: DefaultSelection.None });\n\n  // THEN\n  expect(x.stackCount).toBe(0);\n});\n\ntest('select behavior: single', async () => {\n  // GIVEN\n  const cxasm = await testCloudAssembly();\n\n  // WHEN\n  await expect(cxasm.selectStacks({ patterns: [] }, { defaultBehavior: DefaultSelection.OnlySingle }))\n    .rejects.toThrow('Since this app includes more than a single stack, specify which stacks to use (wildcards are supported) or specify `--all`');\n});\n\ntest('stack list error contains node paths', async () => {\n  // GIVEN\n  const cxasm = await testCloudAssembly();\n\n  // WHEN\n  await expect(cxasm.selectStacks({ patterns: [] }, { defaultBehavior: DefaultSelection.OnlySingle }))\n    .rejects.toThrow('withouterrorsNODEPATH');\n});\n\ntest('select behavior: repeat', async () => {\n  // GIVEN\n  const cxasm = await testCloudAssembly();\n\n  // WHEN\n  const x = await cxasm.selectStacks({ patterns: ['withouterrorsNODEPATH', 'withouterrorsNODEPATH'] }, {\n    defaultBehavior: DefaultSelection.AllStacks,\n  });\n\n  // THEN\n  expect(x.stackCount).toBe(1);\n});\n\ntest('select behavior with nested assemblies: all', async () => {\n  // GIVEN\n  const cxasm = await testNestedCloudAssembly();\n\n  // WHEN\n  const x = await cxasm.selectStacks({ patterns: [] }, { defaultBehavior: DefaultSelection.AllStacks });\n\n  // THEN\n  expect(x.stackCount).toBe(3);\n});\n\ntest('select behavior with nested assemblies: none', async () => {\n  // GIVEN\n  const cxasm = await testNestedCloudAssembly();\n\n  // WHEN\n  const x = await cxasm.selectStacks({ patterns: [] }, { defaultBehavior: DefaultSelection.None });\n\n  // THEN\n  expect(x.stackCount).toBe(0);\n});\n\ntest('select behavior with nested assemblies: single', async () => {\n  // GIVEN\n  const cxasm = await testNestedCloudAssembly();\n\n  // WHEN\n  await expect(cxasm.selectStacks({ patterns: [] }, { defaultBehavior: DefaultSelection.OnlySingle }))\n    .rejects.toThrow('Since this app includes more than a single stack, specify which stacks to use (wildcards are supported) or specify `--all`');\n});\n\ntest('select behavior with nested assemblies: repeat', async() => {\n  // GIVEN\n  const cxasm = await testNestedCloudAssembly();\n\n  // WHEN\n  const x = await cxasm.selectStacks({ patterns: ['deeply/hidden/withouterrors', 'nested'] }, {\n    defaultBehavior: DefaultSelection.AllStacks,\n  });\n\n  // THEN\n  expect(x.stackCount).toBe(2);\n});\n\nasync function testCloudAssembly({ env }: { env?: string, versionReporting?: boolean } = {}) {\n  const cloudExec = new MockCloudExecutable({\n    stacks: [{\n      stackName: 'withouterrors',\n      displayName: 'withouterrorsNODEPATH',\n      env,\n      template: { resource: 'noerrorresource' },\n    },\n    {\n      stackName: 'witherrors',\n      env,\n      template: { resource: 'errorresource' },\n      metadata: {\n        '/resource': [\n          {\n            type: cxschema.ArtifactMetadataEntryType.ERROR,\n            data: 'this is an error',\n          },\n        ],\n      },\n    }],\n  });\n\n  return cloudExec.synthesize();\n}\n\nasync function testNestedCloudAssembly({ env }: { env?: string, versionReporting?: boolean } = {}) {\n  const cloudExec = new MockCloudExecutable({\n    stacks: [{\n      stackName: 'withouterrors',\n      env,\n      template: { resource: 'noerrorresource' },\n      // The nesting in the path should be independent of the position in the tree\n      displayName: 'deeply/hidden/withouterrors',\n    },\n    {\n      stackName: 'witherrors',\n      env,\n      template: { resource: 'errorresource' },\n      metadata: {\n        '/resource': [\n          {\n            type: cxschema.ArtifactMetadataEntryType.ERROR,\n            data: 'this is an error',\n          },\n        ],\n      },\n    }],\n    nestedAssemblies: [{\n      stacks: [{\n        stackName: 'nested',\n        env,\n        template: { resource: 'nestederror' },\n        metadata: {\n          '/resource': [\n            {\n              type: cxschema.ArtifactMetadataEntryType.ERROR,\n              data: 'this is another error',\n            },\n          ],\n        },\n      }],\n    }],\n  });\n\n  return cloudExec.synthesize();\n}\n"]}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cloud-assembly.test.js","sourceRoot":"","sources":["cloud-assembly.test.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable import/order */\nimport * as cxschema from '@aws-cdk/cloud-assembly-schema';\nimport { DefaultSelection } from '../../lib/api/cxapp/cloud-assembly';\nimport { MockCloudExecutable } from '../util';\n\n// behave like v2\nprocess.env.CXAPI_DISABLE_SELECT_BY_ID = '1';\n\ntest('do not throw when selecting stack without errors', async () => {\n  // GIVEN\n  const cxasm = await testCloudAssembly();\n\n  // WHEN\n  const selected = await cxasm.selectStacks( { patterns: ['withouterrorsNODEPATH'] }, {\n    defaultBehavior: DefaultSelection.AllStacks,\n  });\n  selected.processMetadataMessages();\n\n  // THEN\n  expect(selected.firstStack.template.resource).toBe('noerrorresource');\n});\n\ntest('do throw when selecting stack with errors', async () => {\n  // GIVEN\n  const cxasm = await testCloudAssembly();\n\n  // WHEN\n  const selected = await cxasm.selectStacks({ patterns: ['witherrors'] }, {\n    defaultBehavior: DefaultSelection.AllStacks,\n  });\n\n  // THEN\n  expect(() => selected.processMetadataMessages()).toThrow(/Found errors/);\n});\n\ntest('select all top level stacks in the presence of nested assemblies', async () => {\n  // GIVEN\n  const cxasm = await testNestedCloudAssembly();\n\n  // WHEN\n  const x = await cxasm.selectStacks({ allTopLevel: true, patterns: [] }, { defaultBehavior: DefaultSelection.AllStacks });\n\n  // THEN\n  expect(x.stackCount).toBe(2);\n  expect(x.stackIds).toContain('witherrors');\n  expect(x.stackIds).toContain('withouterrors');\n});\n\ntest('select stacks by glob pattern', async () => {\n  // GIVEN\n  const cxasm = await testCloudAssembly();\n\n  // WHEN\n  const x = await cxasm.selectStacks({ patterns: ['with*'] }, { defaultBehavior: DefaultSelection.AllStacks });\n\n  // THEN\n  expect(x.stackCount).toBe(2);\n  expect(x.stackIds).toContain('witherrors');\n  expect(x.stackIds).toContain('withouterrors');\n});\n\ntest('select behavior: all', async () => {\n  // GIVEN\n  const cxasm = await testCloudAssembly();\n\n  // WHEN\n  const x = await cxasm.selectStacks({ patterns: [] }, { defaultBehavior: DefaultSelection.AllStacks });\n\n  // THEN\n  expect(x.stackCount).toBe(2);\n});\n\ntest('select behavior: none', async () => {\n  // GIVEN\n  const cxasm = await testCloudAssembly();\n\n  // WHEN\n  const x = await cxasm.selectStacks({ patterns: [] }, { defaultBehavior: DefaultSelection.None });\n\n  // THEN\n  expect(x.stackCount).toBe(0);\n});\n\ntest('select behavior: single', async () => {\n  // GIVEN\n  const cxasm = await testCloudAssembly();\n\n  // WHEN\n  await expect(cxasm.selectStacks({ patterns: [] }, { defaultBehavior: DefaultSelection.OnlySingle }))\n    .rejects.toThrow('Since this app includes more than a single stack, specify which stacks to use (wildcards are supported) or specify `--all`');\n});\n\ntest('stack list error contains node paths', async () => {\n  // GIVEN\n  const cxasm = await testCloudAssembly();\n\n  // WHEN\n  await expect(cxasm.selectStacks({ patterns: [] }, { defaultBehavior: DefaultSelection.OnlySingle }))\n    .rejects.toThrow('withouterrorsNODEPATH');\n});\n\ntest('select behavior: repeat', async () => {\n  // GIVEN\n  const cxasm = await testCloudAssembly();\n\n  // WHEN\n  const x = await cxasm.selectStacks({ patterns: ['withouterrorsNODEPATH', 'withouterrorsNODEPATH'] }, {\n    defaultBehavior: DefaultSelection.AllStacks,\n  });\n\n  // THEN\n  expect(x.stackCount).toBe(1);\n});\n\ntest('select behavior with nested assemblies: all', async () => {\n  // GIVEN\n  const cxasm = await testNestedCloudAssembly();\n\n  // WHEN\n  const x = await cxasm.selectStacks({ patterns: [] }, { defaultBehavior: DefaultSelection.AllStacks });\n\n  // THEN\n  expect(x.stackCount).toBe(3);\n});\n\ntest('select behavior with nested assemblies: none', async () => {\n  // GIVEN\n  const cxasm = await testNestedCloudAssembly();\n\n  // WHEN\n  const x = await cxasm.selectStacks({ patterns: [] }, { defaultBehavior: DefaultSelection.None });\n\n  // THEN\n  expect(x.stackCount).toBe(0);\n});\n\ntest('select behavior with nested assemblies: single', async () => {\n  // GIVEN\n  const cxasm = await testNestedCloudAssembly();\n\n  // WHEN\n  await expect(cxasm.selectStacks({ patterns: [] }, { defaultBehavior: DefaultSelection.OnlySingle }))\n    .rejects.toThrow('Since this app includes more than a single stack, specify which stacks to use (wildcards are supported) or specify `--all`');\n});\n\ntest('select behavior with nested assemblies: repeat', async() => {\n  // GIVEN\n  const cxasm = await testNestedCloudAssembly();\n\n  // WHEN\n  const x = await cxasm.selectStacks({ patterns: ['deeply/hidden/withouterrors', 'nested'] }, {\n    defaultBehavior: DefaultSelection.AllStacks,\n  });\n\n  // THEN\n  expect(x.stackCount).toBe(2);\n});\n\ntest('select behavior with no stacks and ignore stacks option', async() => {\n  // GIVEN\n  const cxasm = await testCloudAssemblyNoStacks();\n\n  // WHEN\n  const x = await cxasm.selectStacks({ patterns: [] }, {\n    defaultBehavior: DefaultSelection.AllStacks,\n    ignoreNoStacks: true,\n  });\n\n  // THEN\n  expect(x.stackCount).toBe(0);\n});\n\ntest('select behavior with no stacks and no ignore stacks option', async() => {\n  // GIVEN\n  const cxasm = await testCloudAssemblyNoStacks();\n\n  // WHEN & THEN\n  await expect(cxasm.selectStacks({ patterns: [] }, { defaultBehavior: DefaultSelection.AllStacks, ignoreNoStacks: false }))\n    .rejects.toThrow('This app contains no stacks');\n});\n\ntest('select behavior with no stacks and default ignore stacks options (false)', async() => {\n  // GIVEN\n  const cxasm = await testCloudAssemblyNoStacks();\n\n  // WHEN & THEN\n  await expect(cxasm.selectStacks({ patterns: [] }, { defaultBehavior: DefaultSelection.AllStacks }))\n    .rejects.toThrow('This app contains no stacks');\n});\n\nasync function testCloudAssembly({ env }: { env?: string, versionReporting?: boolean } = {}) {\n  const cloudExec = new MockCloudExecutable({\n    stacks: [{\n      stackName: 'withouterrors',\n      displayName: 'withouterrorsNODEPATH',\n      env,\n      template: { resource: 'noerrorresource' },\n    },\n    {\n      stackName: 'witherrors',\n      env,\n      template: { resource: 'errorresource' },\n      metadata: {\n        '/resource': [\n          {\n            type: cxschema.ArtifactMetadataEntryType.ERROR,\n            data: 'this is an error',\n          },\n        ],\n      },\n    }],\n  });\n\n  return cloudExec.synthesize();\n}\n\nasync function testCloudAssemblyNoStacks() {\n  const cloudExec = new MockCloudExecutable({\n    stacks: [],\n  });\n\n  return cloudExec.synthesize();\n}\n\nasync function testNestedCloudAssembly({ env }: { env?: string, versionReporting?: boolean } = {}) {\n  const cloudExec = new MockCloudExecutable({\n    stacks: [{\n      stackName: 'withouterrors',\n      env,\n      template: { resource: 'noerrorresource' },\n      // The nesting in the path should be independent of the position in the tree\n      displayName: 'deeply/hidden/withouterrors',\n    },\n    {\n      stackName: 'witherrors',\n      env,\n      template: { resource: 'errorresource' },\n      metadata: {\n        '/resource': [\n          {\n            type: cxschema.ArtifactMetadataEntryType.ERROR,\n            data: 'this is an error',\n          },\n        ],\n      },\n    }],\n    nestedAssemblies: [{\n      stacks: [{\n        stackName: 'nested',\n        env,\n        template: { resource: 'nestederror' },\n        metadata: {\n          '/resource': [\n            {\n              type: cxschema.ArtifactMetadataEntryType.ERROR,\n              data: 'this is another error',\n            },\n          ],\n        },\n      }],\n    }],\n  });\n\n  return cloudExec.synthesize();\n}\n"]}

@@ -176,3 +176,3 @@ "use strict";

const app = fs.readFileSync(path.join(workDir, 'GoodGo', 'good_go.go'), 'utf8').split('\n');
expect(app.map(line => line.match(/func NewGoodGoStack\(scope constructs.Construct, id string, props GoodGoStackProps\) \*GoodGoStack \{/)).filter(line => line).length).toEqual(1);
expect(app.map(line => line.match(/func NewGoodGoStack\(scope constructs.Construct, id string, props \*GoodGoStackProps\) \*GoodGoStack \{/)).filter(line => line).length).toEqual(1);
expect(app.map(line => line.match(/ NewGoodGoStack\(app, "GoodGo", &GoodGoStackProps\{/)));

@@ -209,2 +209,2 @@ });

}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"migrate.test.js","sourceRoot":"","sources":["migrate.test.ts"],"names":[],"mappings":"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","sourcesContent":["import { exec as _exec } from 'child_process';\nimport * as os from 'os';\nimport * as path from 'path';\nimport { promisify } from 'util';\nimport * as fs from 'fs-extra';\nimport { generateCdkApp, generateStack, readFromPath, readFromStack, setEnvironment, validateSourceOptions } from '../../lib/commands/migrate';\nimport { MockSdkProvider, MockedObject, SyncHandlerSubsetOf } from '../util/mock-sdk';\n\nconst exec = promisify(_exec);\n\ndescribe('Migrate Function Tests', () => {\n  let sdkProvider: MockSdkProvider;\n  let getTemplateMock: jest.Mock;\n  let describeStacksMock: jest.Mock;\n  let cfnMocks: MockedObject<SyncHandlerSubsetOf<AWS.CloudFormation>>;\n\n  const testResourcePath = [__dirname, 'test-resources'];\n  const templatePath = [...testResourcePath, 'templates'];\n  const stackPath = [...testResourcePath, 'stacks'];\n\n  const validTemplatePath = path.join(...templatePath, 's3-template.json');\n  const validTemplate = readFromPath(validTemplatePath)!;\n\n  beforeEach(async () => {\n    sdkProvider = new MockSdkProvider();\n    getTemplateMock = jest.fn();\n    describeStacksMock = jest.fn();\n    cfnMocks = { getTemplate: getTemplateMock, describeStacks: describeStacksMock };\n    sdkProvider.stubCloudFormation(cfnMocks as any);\n  });\n\n  test('validateSourceOptions throws if both --from-path and --from-stack is provided', () => {\n    expect(() => validateSourceOptions('any-value', true)).toThrowError('Only one of `--from-path` or `--from-stack` may be provided.');\n  });\n\n  test('validateSourceOptions throws if neither --from-path or --from-stack is provided', () => {\n    expect(() => validateSourceOptions(undefined, undefined)).toThrowError('Either `--from-path` or `--from-stack` must be used to provide the source of the CloudFormation template.');\n  });\n\n  test('validateSourceOptions does not throw when only --from-path is supplied', () => {\n    expect(() => validateSourceOptions('any-value', false)).not.toThrow();\n  });\n\n  test('validateSourceOptions does now throw when only --from-stack is provided', () => {\n    expect(() => validateSourceOptions(undefined, true)).not.toThrow();\n  });\n\n  test('readFromPath produces a string representation of the template at a given path', () => {\n    expect(readFromPath(validTemplatePath)).toEqual(fs.readFileSync(validTemplatePath, 'utf8'));\n  });\n\n  test('readFromPath returns undefined when template file is not provided', () => {\n    expect(readFromPath()).toEqual(undefined);\n  });\n\n  test('readFromPath throws error when template file does not exist at a given path', () => {\n    const badTemplatePath = './not-here.json';\n    expect(() => readFromPath(badTemplatePath)).toThrowError(`\\'${badTemplatePath}\\' is not a valid path.`);\n  });\n\n  test('readFromStack produces a string representation of the template retrieved from CloudFormation', async () => {\n    const template = fs.readFileSync(validTemplatePath, { encoding: 'utf-8' });\n    getTemplateMock.mockImplementation(() => ({\n      TemplateBody: template,\n    }));\n\n    describeStacksMock.mockImplementation(() => ({\n      Stacks: [\n        {\n          StackName: 'this-one',\n          StackStatus: 'CREATE_COMPLETE',\n        },\n      ],\n    }));\n\n    expect(await readFromStack('this-one', sdkProvider, { account: 'num', region: 'here', name: 'hello-my-name-is-what...' })).toEqual(JSON.stringify(JSON.parse(template)));\n  });\n\n  test('readFromStack throws error when no stack exists with the stack name in the account and region', async () => {\n    describeStacksMock.mockImplementation(() => { throw new Error('No stack. This did not go well.'); });\n    await expect(() => readFromStack('that-one', sdkProvider, { account: 'num', region: 'here', name: 'hello-my-name-is-who...' })).rejects.toThrow('No stack. This did not go well.');\n  });\n\n  test('readFromStack throws error when stack exists but the status is not healthy', async () => {\n    describeStacksMock.mockImplementation(() => ({\n      Stacks: [\n        {\n          StackName: 'this-one',\n          StackStatus: 'CREATE_FAILED',\n          StackStatusReason: 'Something went wrong',\n        },\n      ],\n    }));\n\n    await expect(() => readFromStack('that-one', sdkProvider, { account: 'num', region: 'here', name: 'hello-my-name-is-chicka-chicka...' })).rejects.toThrow('Stack \\'that-one\\' in account num and region here has a status of \\'CREATE_FAILED\\' due to \\'Something went wrong\\'. The stack cannot be migrated until it is in a healthy state.');\n  });\n\n  test('setEnvironment sets account and region when provided', () => {\n    expect(setEnvironment('my-account', 'somewhere')).toEqual({ account: 'my-account', region: 'somewhere', name: 'cdk-migrate-env' });\n  });\n\n  test('serEnvironment uses default account and region when not provided', () => {\n    expect(setEnvironment()).toEqual({ account: 'unknown-account', region: 'unknown-region', name: 'cdk-migrate-env' });\n  });\n\n  test('generateStack generates the expected stack string when called for typescript', () => {\n    const stack = generateStack(validTemplate, 'GoodTypeScript', 'typescript');\n    expect(stack).toEqual(fs.readFileSync(path.join(...stackPath, 's3-stack.ts'), 'utf8'));\n  });\n\n  test('generateStack generates the expected stack string when called for python', () => {\n    const stack = generateStack(validTemplate, 'GoodPython', 'python');\n    expect(stack).toEqual(fs.readFileSync(path.join(...stackPath, 's3_stack.py'), 'utf8'));\n  });\n\n  test('generateStack generates the expected stack string when called for java', () => {\n    const stack = generateStack(validTemplate, 'GoodJava', 'java');\n    expect(stack).toEqual(fs.readFileSync(path.join(...stackPath, 'S3Stack.java'), 'utf8'));\n  });\n\n  test('generateStack generates the expected stack string when called for csharp', () => {\n    const stack = generateStack(validTemplate, 'GoodCSharp', 'csharp');\n    expect(stack).toEqual(fs.readFileSync(path.join(...stackPath, 'S3Stack.cs'), 'utf8'));\n  });\n\n  // TODO: fix with actual go template\n  test('generateStack generates the expected stack string when called for go', () => {\n    const stack = generateStack(validTemplate, 'GoodGo', 'go');\n    expect(stack).toEqual(fs.readFileSync(path.join(...stackPath, 's3.go'), 'utf8'));\n  });\n\n  test('generateStack throws error when called for other language', () => {\n    expect(() => generateStack(validTemplate, 'BadBadBad', 'php')).toThrowError('stack generation failed due to error \\'unreachable\\'');\n  });\n\n  cliTest('generateCdkApp generates the expected cdk app when called for typescript', async (workDir) => {\n    const stack = generateStack(validTemplate, 'GoodTypeScript', 'typescript');\n    await generateCdkApp('GoodTypeScript', stack, 'typescript', workDir);\n\n    // Packages exist in the correct spot\n    expect(fs.pathExistsSync(path.join(workDir, 'GoodTypeScript', 'package.json'))).toBeTruthy();\n    expect(fs.pathExistsSync(path.join(workDir, 'GoodTypeScript', 'bin', 'good_type_script.ts'))).toBeTruthy();\n    expect(fs.pathExistsSync(path.join(workDir, 'GoodTypeScript', 'lib', 'good_type_script-stack.ts'))).toBeTruthy();\n\n    // Replaced stack file is referenced correctly in app file\n    const app = fs.readFileSync(path.join(workDir, 'GoodTypeScript', 'bin', 'good_type_script.ts'), 'utf8').split('\\n');\n    expect(app.map(line => line.match('import { GoodTypeScriptStack } from \\'../lib/good_type_script-stack\\';')).filter(line => line).length).toEqual(1);\n    expect(app.map(line => line.match(/new GoodTypeScriptStack\\(app, \\'GoodTypeScript\\', \\{/)).filter(line => line).length).toEqual(1);\n\n    // Replaced stack file is correctly generated\n    const replacedStack = fs.readFileSync(path.join(workDir, 'GoodTypeScript', 'lib', 'good_type_script-stack.ts'), 'utf8');\n    expect(replacedStack).toEqual(fs.readFileSync(path.join(...stackPath, 's3-stack.ts'), 'utf8'));\n  });\n\n  cliTest('generateCdkApp generates the expected cdk app when called for python', async (workDir) => {\n    const stack = generateStack(validTemplate, 'GoodPython', 'python');\n    await generateCdkApp('GoodPython', stack, 'python', workDir);\n\n    // Packages exist in the correct spot\n    expect(fs.pathExistsSync(path.join(workDir, 'GoodPython', 'requirements.txt'))).toBeTruthy();\n    expect(fs.pathExistsSync(path.join(workDir, 'GoodPython', 'app.py'))).toBeTruthy();\n    expect(fs.pathExistsSync(path.join(workDir, 'GoodPython', 'good_python', 'good_python_stack.py'))).toBeTruthy();\n\n    // Replaced stack file is referenced correctly in app file\n    const app = fs.readFileSync(path.join(workDir, 'GoodPython', 'app.py'), 'utf8').split('\\n');\n    expect(app.map(line => line.match('from good_python.good_python_stack import GoodPythonStack')).filter(line => line).length).toEqual(1);\n    expect(app.map(line => line.match(/GoodPythonStack\\(app, \"GoodPython\",/)).filter(line => line).length).toEqual(1);\n\n    // Replaced stack file is correctly generated\n    const replacedStack = fs.readFileSync(path.join(workDir, 'GoodPython', 'good_python', 'good_python_stack.py'), 'utf8');\n    expect(replacedStack).toEqual(fs.readFileSync(path.join(...stackPath, 's3_stack.py'), 'utf8'));\n  });\n\n  cliTest('generateCdkApp generates the expected cdk app when called for java', async (workDir) => {\n    const stack = generateStack(validTemplate, 'GoodJava', 'java');\n    await generateCdkApp('GoodJava', stack, 'java', workDir);\n\n    // Packages exist in the correct spot\n    expect(fs.pathExistsSync(path.join(workDir, 'GoodJava', 'pom.xml'))).toBeTruthy();\n    expect(fs.pathExistsSync(path.join(workDir, 'GoodJava', 'src', 'main', 'java', 'com', 'myorg', 'GoodJavaApp.java'))).toBeTruthy();\n    expect(fs.pathExistsSync(path.join(workDir, 'GoodJava', 'src', 'main', 'java', 'com', 'myorg', 'GoodJavaStack.java'))).toBeTruthy();\n\n    // Replaced stack file is referenced correctly in app file\n    const app = fs.readFileSync(path.join(workDir, 'GoodJava', 'src', 'main', 'java', 'com', 'myorg', 'GoodJavaApp.java'), 'utf8').split('\\n');\n    expect(app.map(line => line.match('public class GoodJavaApp {')).filter(line => line).length).toEqual(1);\n    expect(app.map(line => line.match(/        new GoodJavaStack\\(app, \"GoodJava\", StackProps.builder()/)).filter(line => line).length).toEqual(1);\n\n    // Replaced stack file is correctly generated\n    const replacedStack = fs.readFileSync(path.join(workDir, 'GoodJava', 'src', 'main', 'java', 'com', 'myorg', 'GoodJavaStack.java'), 'utf8');\n    expect(replacedStack).toEqual(fs.readFileSync(path.join(...stackPath, 'S3Stack.java'), 'utf8'));\n  });\n\n  cliTest('generateCdkApp generates the expected cdk app when called for csharp', async (workDir) => {\n    const stack = generateStack(validTemplate, 'GoodCSharp', 'csharp');\n    await generateCdkApp('GoodCSharp', stack, 'csharp', workDir);\n\n    // Packages exist in the correct spot\n    expect(fs.pathExistsSync(path.join(workDir, 'GoodCSharp', 'src', 'GoodCSharp.sln'))).toBeTruthy();\n    expect(fs.pathExistsSync(path.join(workDir, 'GoodCSharp', 'src', 'GoodCSharp', 'Program.cs'))).toBeTruthy();\n    expect(fs.pathExistsSync(path.join(workDir, 'GoodCSharp', 'src', 'GoodCSharp', 'GoodCSharpStack.cs'))).toBeTruthy();\n\n    // Replaced stack file is referenced correctly in app file\n    const app = fs.readFileSync(path.join(workDir, 'GoodCSharp', 'src', 'GoodCSharp', 'Program.cs'), 'utf8').split('\\n');\n    expect(app.map(line => line.match('namespace GoodCSharp')).filter(line => line).length).toEqual(1);\n    expect(app.map(line => line.match(/        new GoodCSharpStack\\(app, \"GoodCSharp\", new GoodCSharpStackProps/)).filter(line => line).length).toEqual(1);\n\n    // Replaced stack file is correctly generated\n    const replacedStack = fs.readFileSync(path.join(workDir, 'GoodCSharp', 'src', 'GoodCSharp', 'GoodCSharpStack.cs'), 'utf8');\n    expect(replacedStack).toEqual(fs.readFileSync(path.join(...stackPath, 'S3Stack.cs'), 'utf8'));\n  });\n\n  cliTest('generatedCdkApp generates the expected cdk app when called for go', async (workDir) => {\n    const stack = generateStack(validTemplate, 'GoodGo', 'go');\n    await generateCdkApp('GoodGo', stack, 'go', workDir);\n\n    expect(fs.pathExists(path.join(workDir, 's3.go'))).toBeTruthy();\n    const app = fs.readFileSync(path.join(workDir, 'GoodGo', 'good_go.go'), 'utf8').split('\\n');\n    expect(app.map(line => line.match(/func NewGoodGoStack\\(scope constructs.Construct, id string, props GoodGoStackProps\\) \\*GoodGoStack \\{/)).filter(line => line).length).toEqual(1);\n    expect(app.map(line => line.match(/    NewGoodGoStack\\(app, \"GoodGo\", &GoodGoStackProps\\{/)));\n  });\n\n  cliTest('generatedCdkApp generates a zip file when --compress is used', async (workDir) => {\n    const stack = generateStack(validTemplate, 'GoodTypeScript', 'typescript');\n    await generateCdkApp('GoodTypeScript', stack, 'typescript', workDir, true);\n\n    // Packages not in outDir\n    expect(fs.pathExistsSync(path.join(workDir, 'GoodTypeScript', 'package.json'))).toBeFalsy();\n    expect(fs.pathExistsSync(path.join(workDir, 'GoodTypeScript', 'bin', 'good_type_script.ts'))).toBeFalsy();\n    expect(fs.pathExistsSync(path.join(workDir, 'GoodTypeScript', 'lib', 'good_type_script-stack.ts'))).toBeFalsy();\n\n    // Zip file exists\n    expect(fs.pathExistsSync(path.join(workDir, 'GoodTypeScript.zip'))).toBeTruthy();\n\n    // Unzip it\n    await exec(`unzip ${path.join(workDir, 'GoodTypeScript.zip')}`, { cwd: workDir });\n\n    // Now the files should be there\n    expect(fs.pathExistsSync(path.join(workDir, 'package.json'))).toBeTruthy();\n    expect(fs.pathExistsSync(path.join(workDir, 'bin', 'good_type_script.ts'))).toBeTruthy();\n    expect(fs.pathExistsSync(path.join(workDir, 'lib', 'good_type_script-stack.ts'))).toBeTruthy();\n  });\n});\n\nfunction cliTest(name: string, handler: (dir: string) => void | Promise<any>): void {\n  test(name, () => withTempDir(handler));\n}\n\nasync function withTempDir(cb: (dir: string) => void | Promise<any>) {\n  const tmpDir = await fs.mkdtemp(path.join(os.tmpdir(), 'aws-cdk-test'));\n  try {\n    await cb(tmpDir);\n  } finally {\n    await fs.remove(tmpDir);\n  }\n}"]}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"migrate.test.js","sourceRoot":"","sources":["migrate.test.ts"],"names":[],"mappings":"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","sourcesContent":["import { exec as _exec } from 'child_process';\nimport * as os from 'os';\nimport * as path from 'path';\nimport { promisify } from 'util';\nimport * as fs from 'fs-extra';\nimport { generateCdkApp, generateStack, readFromPath, readFromStack, setEnvironment, validateSourceOptions } from '../../lib/commands/migrate';\nimport { MockSdkProvider, MockedObject, SyncHandlerSubsetOf } from '../util/mock-sdk';\n\nconst exec = promisify(_exec);\n\ndescribe('Migrate Function Tests', () => {\n  let sdkProvider: MockSdkProvider;\n  let getTemplateMock: jest.Mock;\n  let describeStacksMock: jest.Mock;\n  let cfnMocks: MockedObject<SyncHandlerSubsetOf<AWS.CloudFormation>>;\n\n  const testResourcePath = [__dirname, 'test-resources'];\n  const templatePath = [...testResourcePath, 'templates'];\n  const stackPath = [...testResourcePath, 'stacks'];\n\n  const validTemplatePath = path.join(...templatePath, 's3-template.json');\n  const validTemplate = readFromPath(validTemplatePath)!;\n\n  beforeEach(async () => {\n    sdkProvider = new MockSdkProvider();\n    getTemplateMock = jest.fn();\n    describeStacksMock = jest.fn();\n    cfnMocks = { getTemplate: getTemplateMock, describeStacks: describeStacksMock };\n    sdkProvider.stubCloudFormation(cfnMocks as any);\n  });\n\n  test('validateSourceOptions throws if both --from-path and --from-stack is provided', () => {\n    expect(() => validateSourceOptions('any-value', true)).toThrowError('Only one of `--from-path` or `--from-stack` may be provided.');\n  });\n\n  test('validateSourceOptions throws if neither --from-path or --from-stack is provided', () => {\n    expect(() => validateSourceOptions(undefined, undefined)).toThrowError('Either `--from-path` or `--from-stack` must be used to provide the source of the CloudFormation template.');\n  });\n\n  test('validateSourceOptions does not throw when only --from-path is supplied', () => {\n    expect(() => validateSourceOptions('any-value', false)).not.toThrow();\n  });\n\n  test('validateSourceOptions does now throw when only --from-stack is provided', () => {\n    expect(() => validateSourceOptions(undefined, true)).not.toThrow();\n  });\n\n  test('readFromPath produces a string representation of the template at a given path', () => {\n    expect(readFromPath(validTemplatePath)).toEqual(fs.readFileSync(validTemplatePath, 'utf8'));\n  });\n\n  test('readFromPath returns undefined when template file is not provided', () => {\n    expect(readFromPath()).toEqual(undefined);\n  });\n\n  test('readFromPath throws error when template file does not exist at a given path', () => {\n    const badTemplatePath = './not-here.json';\n    expect(() => readFromPath(badTemplatePath)).toThrowError(`\\'${badTemplatePath}\\' is not a valid path.`);\n  });\n\n  test('readFromStack produces a string representation of the template retrieved from CloudFormation', async () => {\n    const template = fs.readFileSync(validTemplatePath, { encoding: 'utf-8' });\n    getTemplateMock.mockImplementation(() => ({\n      TemplateBody: template,\n    }));\n\n    describeStacksMock.mockImplementation(() => ({\n      Stacks: [\n        {\n          StackName: 'this-one',\n          StackStatus: 'CREATE_COMPLETE',\n        },\n      ],\n    }));\n\n    expect(await readFromStack('this-one', sdkProvider, { account: 'num', region: 'here', name: 'hello-my-name-is-what...' })).toEqual(JSON.stringify(JSON.parse(template)));\n  });\n\n  test('readFromStack throws error when no stack exists with the stack name in the account and region', async () => {\n    describeStacksMock.mockImplementation(() => { throw new Error('No stack. This did not go well.'); });\n    await expect(() => readFromStack('that-one', sdkProvider, { account: 'num', region: 'here', name: 'hello-my-name-is-who...' })).rejects.toThrow('No stack. This did not go well.');\n  });\n\n  test('readFromStack throws error when stack exists but the status is not healthy', async () => {\n    describeStacksMock.mockImplementation(() => ({\n      Stacks: [\n        {\n          StackName: 'this-one',\n          StackStatus: 'CREATE_FAILED',\n          StackStatusReason: 'Something went wrong',\n        },\n      ],\n    }));\n\n    await expect(() => readFromStack('that-one', sdkProvider, { account: 'num', region: 'here', name: 'hello-my-name-is-chicka-chicka...' })).rejects.toThrow('Stack \\'that-one\\' in account num and region here has a status of \\'CREATE_FAILED\\' due to \\'Something went wrong\\'. The stack cannot be migrated until it is in a healthy state.');\n  });\n\n  test('setEnvironment sets account and region when provided', () => {\n    expect(setEnvironment('my-account', 'somewhere')).toEqual({ account: 'my-account', region: 'somewhere', name: 'cdk-migrate-env' });\n  });\n\n  test('serEnvironment uses default account and region when not provided', () => {\n    expect(setEnvironment()).toEqual({ account: 'unknown-account', region: 'unknown-region', name: 'cdk-migrate-env' });\n  });\n\n  test('generateStack generates the expected stack string when called for typescript', () => {\n    const stack = generateStack(validTemplate, 'GoodTypeScript', 'typescript');\n    expect(stack).toEqual(fs.readFileSync(path.join(...stackPath, 's3-stack.ts'), 'utf8'));\n  });\n\n  test('generateStack generates the expected stack string when called for python', () => {\n    const stack = generateStack(validTemplate, 'GoodPython', 'python');\n    expect(stack).toEqual(fs.readFileSync(path.join(...stackPath, 's3_stack.py'), 'utf8'));\n  });\n\n  test('generateStack generates the expected stack string when called for java', () => {\n    const stack = generateStack(validTemplate, 'GoodJava', 'java');\n    expect(stack).toEqual(fs.readFileSync(path.join(...stackPath, 'S3Stack.java'), 'utf8'));\n  });\n\n  test('generateStack generates the expected stack string when called for csharp', () => {\n    const stack = generateStack(validTemplate, 'GoodCSharp', 'csharp');\n    expect(stack).toEqual(fs.readFileSync(path.join(...stackPath, 'S3Stack.cs'), 'utf8'));\n  });\n\n  // TODO: fix with actual go template\n  test('generateStack generates the expected stack string when called for go', () => {\n    const stack = generateStack(validTemplate, 'GoodGo', 'go');\n    expect(stack).toEqual(fs.readFileSync(path.join(...stackPath, 's3.go'), 'utf8'));\n  });\n\n  test('generateStack throws error when called for other language', () => {\n    expect(() => generateStack(validTemplate, 'BadBadBad', 'php')).toThrowError('stack generation failed due to error \\'unreachable\\'');\n  });\n\n  cliTest('generateCdkApp generates the expected cdk app when called for typescript', async (workDir) => {\n    const stack = generateStack(validTemplate, 'GoodTypeScript', 'typescript');\n    await generateCdkApp('GoodTypeScript', stack, 'typescript', workDir);\n\n    // Packages exist in the correct spot\n    expect(fs.pathExistsSync(path.join(workDir, 'GoodTypeScript', 'package.json'))).toBeTruthy();\n    expect(fs.pathExistsSync(path.join(workDir, 'GoodTypeScript', 'bin', 'good_type_script.ts'))).toBeTruthy();\n    expect(fs.pathExistsSync(path.join(workDir, 'GoodTypeScript', 'lib', 'good_type_script-stack.ts'))).toBeTruthy();\n\n    // Replaced stack file is referenced correctly in app file\n    const app = fs.readFileSync(path.join(workDir, 'GoodTypeScript', 'bin', 'good_type_script.ts'), 'utf8').split('\\n');\n    expect(app.map(line => line.match('import { GoodTypeScriptStack } from \\'../lib/good_type_script-stack\\';')).filter(line => line).length).toEqual(1);\n    expect(app.map(line => line.match(/new GoodTypeScriptStack\\(app, \\'GoodTypeScript\\', \\{/)).filter(line => line).length).toEqual(1);\n\n    // Replaced stack file is correctly generated\n    const replacedStack = fs.readFileSync(path.join(workDir, 'GoodTypeScript', 'lib', 'good_type_script-stack.ts'), 'utf8');\n    expect(replacedStack).toEqual(fs.readFileSync(path.join(...stackPath, 's3-stack.ts'), 'utf8'));\n  });\n\n  cliTest('generateCdkApp generates the expected cdk app when called for python', async (workDir) => {\n    const stack = generateStack(validTemplate, 'GoodPython', 'python');\n    await generateCdkApp('GoodPython', stack, 'python', workDir);\n\n    // Packages exist in the correct spot\n    expect(fs.pathExistsSync(path.join(workDir, 'GoodPython', 'requirements.txt'))).toBeTruthy();\n    expect(fs.pathExistsSync(path.join(workDir, 'GoodPython', 'app.py'))).toBeTruthy();\n    expect(fs.pathExistsSync(path.join(workDir, 'GoodPython', 'good_python', 'good_python_stack.py'))).toBeTruthy();\n\n    // Replaced stack file is referenced correctly in app file\n    const app = fs.readFileSync(path.join(workDir, 'GoodPython', 'app.py'), 'utf8').split('\\n');\n    expect(app.map(line => line.match('from good_python.good_python_stack import GoodPythonStack')).filter(line => line).length).toEqual(1);\n    expect(app.map(line => line.match(/GoodPythonStack\\(app, \"GoodPython\",/)).filter(line => line).length).toEqual(1);\n\n    // Replaced stack file is correctly generated\n    const replacedStack = fs.readFileSync(path.join(workDir, 'GoodPython', 'good_python', 'good_python_stack.py'), 'utf8');\n    expect(replacedStack).toEqual(fs.readFileSync(path.join(...stackPath, 's3_stack.py'), 'utf8'));\n  });\n\n  cliTest('generateCdkApp generates the expected cdk app when called for java', async (workDir) => {\n    const stack = generateStack(validTemplate, 'GoodJava', 'java');\n    await generateCdkApp('GoodJava', stack, 'java', workDir);\n\n    // Packages exist in the correct spot\n    expect(fs.pathExistsSync(path.join(workDir, 'GoodJava', 'pom.xml'))).toBeTruthy();\n    expect(fs.pathExistsSync(path.join(workDir, 'GoodJava', 'src', 'main', 'java', 'com', 'myorg', 'GoodJavaApp.java'))).toBeTruthy();\n    expect(fs.pathExistsSync(path.join(workDir, 'GoodJava', 'src', 'main', 'java', 'com', 'myorg', 'GoodJavaStack.java'))).toBeTruthy();\n\n    // Replaced stack file is referenced correctly in app file\n    const app = fs.readFileSync(path.join(workDir, 'GoodJava', 'src', 'main', 'java', 'com', 'myorg', 'GoodJavaApp.java'), 'utf8').split('\\n');\n    expect(app.map(line => line.match('public class GoodJavaApp {')).filter(line => line).length).toEqual(1);\n    expect(app.map(line => line.match(/        new GoodJavaStack\\(app, \"GoodJava\", StackProps.builder()/)).filter(line => line).length).toEqual(1);\n\n    // Replaced stack file is correctly generated\n    const replacedStack = fs.readFileSync(path.join(workDir, 'GoodJava', 'src', 'main', 'java', 'com', 'myorg', 'GoodJavaStack.java'), 'utf8');\n    expect(replacedStack).toEqual(fs.readFileSync(path.join(...stackPath, 'S3Stack.java'), 'utf8'));\n  });\n\n  cliTest('generateCdkApp generates the expected cdk app when called for csharp', async (workDir) => {\n    const stack = generateStack(validTemplate, 'GoodCSharp', 'csharp');\n    await generateCdkApp('GoodCSharp', stack, 'csharp', workDir);\n\n    // Packages exist in the correct spot\n    expect(fs.pathExistsSync(path.join(workDir, 'GoodCSharp', 'src', 'GoodCSharp.sln'))).toBeTruthy();\n    expect(fs.pathExistsSync(path.join(workDir, 'GoodCSharp', 'src', 'GoodCSharp', 'Program.cs'))).toBeTruthy();\n    expect(fs.pathExistsSync(path.join(workDir, 'GoodCSharp', 'src', 'GoodCSharp', 'GoodCSharpStack.cs'))).toBeTruthy();\n\n    // Replaced stack file is referenced correctly in app file\n    const app = fs.readFileSync(path.join(workDir, 'GoodCSharp', 'src', 'GoodCSharp', 'Program.cs'), 'utf8').split('\\n');\n    expect(app.map(line => line.match('namespace GoodCSharp')).filter(line => line).length).toEqual(1);\n    expect(app.map(line => line.match(/        new GoodCSharpStack\\(app, \"GoodCSharp\", new GoodCSharpStackProps/)).filter(line => line).length).toEqual(1);\n\n    // Replaced stack file is correctly generated\n    const replacedStack = fs.readFileSync(path.join(workDir, 'GoodCSharp', 'src', 'GoodCSharp', 'GoodCSharpStack.cs'), 'utf8');\n    expect(replacedStack).toEqual(fs.readFileSync(path.join(...stackPath, 'S3Stack.cs'), 'utf8'));\n  });\n\n  cliTest('generatedCdkApp generates the expected cdk app when called for go', async (workDir) => {\n    const stack = generateStack(validTemplate, 'GoodGo', 'go');\n    await generateCdkApp('GoodGo', stack, 'go', workDir);\n\n    expect(fs.pathExists(path.join(workDir, 's3.go'))).toBeTruthy();\n    const app = fs.readFileSync(path.join(workDir, 'GoodGo', 'good_go.go'), 'utf8').split('\\n');\n    expect(app.map(line => line.match(/func NewGoodGoStack\\(scope constructs.Construct, id string, props \\*GoodGoStackProps\\) \\*GoodGoStack \\{/)).filter(line => line).length).toEqual(1);\n    expect(app.map(line => line.match(/    NewGoodGoStack\\(app, \"GoodGo\", &GoodGoStackProps\\{/)));\n  });\n\n  cliTest('generatedCdkApp generates a zip file when --compress is used', async (workDir) => {\n    const stack = generateStack(validTemplate, 'GoodTypeScript', 'typescript');\n    await generateCdkApp('GoodTypeScript', stack, 'typescript', workDir, true);\n\n    // Packages not in outDir\n    expect(fs.pathExistsSync(path.join(workDir, 'GoodTypeScript', 'package.json'))).toBeFalsy();\n    expect(fs.pathExistsSync(path.join(workDir, 'GoodTypeScript', 'bin', 'good_type_script.ts'))).toBeFalsy();\n    expect(fs.pathExistsSync(path.join(workDir, 'GoodTypeScript', 'lib', 'good_type_script-stack.ts'))).toBeFalsy();\n\n    // Zip file exists\n    expect(fs.pathExistsSync(path.join(workDir, 'GoodTypeScript.zip'))).toBeTruthy();\n\n    // Unzip it\n    await exec(`unzip ${path.join(workDir, 'GoodTypeScript.zip')}`, { cwd: workDir });\n\n    // Now the files should be there\n    expect(fs.pathExistsSync(path.join(workDir, 'package.json'))).toBeTruthy();\n    expect(fs.pathExistsSync(path.join(workDir, 'bin', 'good_type_script.ts'))).toBeTruthy();\n    expect(fs.pathExistsSync(path.join(workDir, 'lib', 'good_type_script-stack.ts'))).toBeTruthy();\n  });\n});\n\nfunction cliTest(name: string, handler: (dir: string) => void | Promise<any>): void {\n  test(name, () => withTempDir(handler));\n}\n\nasync function withTempDir(cb: (dir: string) => void | Promise<any>) {\n  const tmpDir = await fs.mkdtemp(path.join(os.tmpdir(), 'aws-cdk-test'));\n  try {\n    await cb(tmpDir);\n  } finally {\n    await fs.remove(tmpDir);\n  }\n}"]}

@@ -10,2 +10,3 @@ "use strict";

const cdk_toolkit_1 = require("../lib/cdk-toolkit");
const cfn = require("../lib/api/util/cloudformation");
let cloudExecutable;

@@ -301,2 +302,38 @@ let cloudFormation;

});
test('diff falls back to non-changeset diff for nested stacks', async () => {
// GIVEN
const changeSetSpy = jest.spyOn(cfn, 'waitForChangeSet');
const buffer = new StringWritable();
// WHEN
const exitCode = await toolkit.diff({
stackNames: ['Parent'],
stream: buffer,
changeSet: true,
});
// THEN
const plainTextOutput = buffer.data.replace(/\x1B\[[0-?]*[ -/]*[@-~]/g, '')
.replace(/[ \t]+$/mg, '');
expect(plainTextOutput.trim()).toEqual(`Stack Parent
Resources
[~] AWS::CloudFormation::Stack AdditionChild
└─ [~] Resources
└─ [~] .SomeResource:
└─ [+] Added: .Properties
[~] AWS::CloudFormation::Stack DeletionChild
└─ [~] Resources
└─ [~] .SomeResource:
└─ [-] Removed: .Properties
[~] AWS::CloudFormation::Stack ChangedChild
└─ [~] Resources
└─ [~] .SomeResource:
└─ [~] .Properties:
└─ [~] .Prop:
├─ [-] old-value
└─ [+] new-value
✨ Number of stacks with differences: 4`);
expect(exitCode).toBe(0);
expect(changeSetSpy).not.toHaveBeenCalled();
});
});

@@ -321,2 +358,2 @@ class StringWritable extends stream_1.Writable {

}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"diff.test.js","sourceRoot":"","sources":["diff.test.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable import/order */\nimport { Writable } from 'stream';\nimport { StringDecoder } from 'string_decoder';\nimport * as cxschema from '@aws-cdk/cloud-assembly-schema';\nimport { CloudFormationStackArtifact } from '@aws-cdk/cx-api';\nimport { instanceMockFrom, MockCloudExecutable } from './util';\nimport { Deployments } from '../lib/api/deployments';\nimport { CdkToolkit } from '../lib/cdk-toolkit';\n\nlet cloudExecutable: MockCloudExecutable;\nlet cloudFormation: jest.Mocked<Deployments>;\nlet toolkit: CdkToolkit;\n\ndescribe('non-nested stacks', () => {\n  beforeEach(() => {\n    cloudExecutable = new MockCloudExecutable({\n      stacks: [{\n        stackName: 'A',\n        template: { resource: 'A' },\n      },\n      {\n        stackName: 'B',\n        depends: ['A'],\n        template: { resource: 'B' },\n      },\n      {\n        stackName: 'C',\n        depends: ['A'],\n        template: { resource: 'C' },\n        metadata: {\n          '/resource': [\n            {\n              type: cxschema.ArtifactMetadataEntryType.ERROR,\n              data: 'this is an error',\n            },\n          ],\n        },\n      },\n      {\n        stackName: 'D',\n        template: { resource: 'D' },\n      }],\n    });\n\n    cloudFormation = instanceMockFrom(Deployments);\n\n    toolkit = new CdkToolkit({\n      cloudExecutable,\n      deployments: cloudFormation,\n      configuration: cloudExecutable.configuration,\n      sdkProvider: cloudExecutable.sdkProvider,\n    });\n\n    // Default implementations\n    cloudFormation.readCurrentTemplateWithNestedStacks.mockImplementation((stackArtifact: CloudFormationStackArtifact) => {\n      if (stackArtifact.stackName === 'D') {\n        return Promise.resolve({\n          deployedTemplate: { resource: 'D' },\n          nestedStackCount: 0,\n        });\n      }\n      return Promise.resolve({\n        deployedTemplate: {},\n        nestedStackCount: 0,\n      });\n    });\n    cloudFormation.deployStack.mockImplementation((options) => Promise.resolve({\n      noOp: true,\n      outputs: {},\n      stackArn: '',\n      stackArtifact: options.stack,\n    }));\n  });\n\n  test('diff can diff multiple stacks', async () => {\n    // GIVEN\n    const buffer = new StringWritable();\n\n    // WHEN\n    const exitCode = await toolkit.diff({\n      stackNames: ['B'],\n      stream: buffer,\n    });\n\n    // THEN\n    const plainTextOutput = buffer.data.replace(/\\x1B\\[[0-?]*[ -/]*[@-~]/g, '');\n    expect(plainTextOutput).toContain('Stack A');\n    expect(plainTextOutput).toContain('Stack B');\n\n    expect(buffer.data.trim()).toContain('✨  Number of stacks with differences: 2');\n    expect(exitCode).toBe(0);\n  });\n\n  test('diff number of stack diffs, not resource diffs', async () => {\n    // GIVEN\n    cloudExecutable = new MockCloudExecutable({\n      stacks: [{\n        stackName: 'A',\n        template: { resourceA: 'A', resourceB: 'B' },\n      },\n      {\n        stackName: 'B',\n        template: { resourceC: 'C' },\n      }],\n    });\n\n    toolkit = new CdkToolkit({\n      cloudExecutable,\n      deployments: cloudFormation,\n      configuration: cloudExecutable.configuration,\n      sdkProvider: cloudExecutable.sdkProvider,\n    });\n\n    const buffer = new StringWritable();\n\n    // WHEN\n    const exitCode = await toolkit.diff({\n      stackNames: ['A', 'B'],\n      stream: buffer,\n    });\n\n    // THEN\n    const plainTextOutput = buffer.data.replace(/\\x1B\\[[0-?]*[ -/]*[@-~]/g, '');\n    expect(plainTextOutput).toContain('Stack A');\n    expect(plainTextOutput).toContain('Stack B');\n\n    expect(buffer.data.trim()).toContain('✨  Number of stacks with differences: 2');\n    expect(exitCode).toBe(0);\n  });\n\n  test('exits with 1 with diffs and fail set to true', async () => {\n    // GIVEN\n    const buffer = new StringWritable();\n\n    // WHEN\n    const exitCode = await toolkit.diff({\n      stackNames: ['A'],\n      stream: buffer,\n      fail: true,\n    });\n\n    // THEN\n    expect(buffer.data.trim()).toContain('✨  Number of stacks with differences: 1');\n    expect(exitCode).toBe(1);\n  });\n\n  test('throws an error if no valid stack names given', async () => {\n    const buffer = new StringWritable();\n\n    // WHEN\n    await expect(() => toolkit.diff({\n      stackNames: ['X', 'Y', 'Z'],\n      stream: buffer,\n    })).rejects.toThrow('No stacks match the name(s) X,Y,Z');\n  });\n\n  test('exits with 1 with diff in first stack, but not in second stack and fail set to true', async () => {\n    // GIVEN\n    const buffer = new StringWritable();\n\n    // WHEN\n    const exitCode = await toolkit.diff({\n      stackNames: ['A', 'D'],\n      stream: buffer,\n      fail: true,\n    });\n\n    // THEN\n    expect(buffer.data.trim()).toContain('✨  Number of stacks with differences: 1');\n    expect(exitCode).toBe(1);\n  });\n\n  test('throws an error during diffs on stack with error metadata', async () => {\n    const buffer = new StringWritable();\n\n    // WHEN\n    await expect(() => toolkit.diff({\n      stackNames: ['C'],\n      stream: buffer,\n    })).rejects.toThrow(/Found errors/);\n  });\n\n  test('when quiet mode is enabled, stacks with no diffs should not print stack name & no differences to stdout', async () => {\n    // GIVEN\n    const buffer = new StringWritable();\n\n    // WHEN\n    const exitCode = await toolkit.diff({\n      stackNames: ['A', 'A'],\n      stream: buffer,\n      fail: false,\n      quiet: true,\n    });\n\n    // THEN\n    expect(buffer.data.trim()).not.toContain('Stack A');\n    expect(buffer.data.trim()).not.toContain('There were no differences');\n    expect(exitCode).toBe(0);\n  });\n});\n\ndescribe('nested stacks', () => {\n  beforeEach(() => {\n    cloudExecutable = new MockCloudExecutable({\n      stacks: [{\n        stackName: 'Parent',\n        template: {},\n      }],\n    });\n\n    cloudFormation = instanceMockFrom(Deployments);\n\n    toolkit = new CdkToolkit({\n      cloudExecutable,\n      deployments: cloudFormation,\n      configuration: cloudExecutable.configuration,\n      sdkProvider: cloudExecutable.sdkProvider,\n    });\n\n    cloudFormation.readCurrentTemplateWithNestedStacks.mockImplementation((stackArtifact: CloudFormationStackArtifact) => {\n      if (stackArtifact.stackName === 'Parent') {\n        stackArtifact.template.Resources = {\n          AdditionChild: {\n            Type: 'AWS::CloudFormation::Stack',\n            Resources: {\n              SomeResource: {\n                Type: 'AWS::Something',\n                Properties: {\n                  Prop: 'added-value',\n                },\n              },\n            },\n          },\n          DeletionChild: {\n            Type: 'AWS::CloudFormation::Stack',\n            Resources: {\n              SomeResource: {\n                Type: 'AWS::Something',\n              },\n            },\n          },\n          ChangedChild: {\n            Type: 'AWS::CloudFormation::Stack',\n            Resources: {\n              SomeResource: {\n                Type: 'AWS::Something',\n                Properties: {\n                  Prop: 'new-value',\n                },\n              },\n            },\n          },\n        };\n        return Promise.resolve({\n          deployedTemplate: {\n            Resources: {\n              AdditionChild: {\n                Type: 'AWS::CloudFormation::Stack',\n                Resources: {\n                  SomeResource: {\n                    Type: 'AWS::Something',\n                  },\n                },\n              },\n              DeletionChild: {\n                Type: 'AWS::CloudFormation::Stack',\n                Resources: {\n                  SomeResource: {\n                    Type: 'AWS::Something',\n                    Properties: {\n                      Prop: 'value-to-be-removed',\n                    },\n                  },\n                },\n              },\n              ChangedChild: {\n                Type: 'AWS::CloudFormation::Stack',\n                Resources: {\n                  SomeResource: {\n                    Type: 'AWS::Something',\n                    Properties: {\n                      Prop: 'old-value',\n                    },\n                  },\n                },\n              },\n            },\n          },\n          nestedStackCount: 3,\n        });\n      }\n      return Promise.resolve({\n        deployedTemplate: {},\n        nestedStackCount: 0,\n      });\n    });\n  });\n\n  test('diff can diff nested stacks', async () => {\n    // GIVEN\n    const buffer = new StringWritable();\n\n    // WHEN\n    const exitCode = await toolkit.diff({\n      stackNames: ['Parent'],\n      stream: buffer,\n    });\n\n    // THEN\n    const plainTextOutput = buffer.data.replace(/\\x1B\\[[0-?]*[ -/]*[@-~]/g, '')\n      .replace(/[ \\t]+$/mg, '');\n    expect(plainTextOutput.trim()).toEqual(`Stack Parent\nResources\n[~] AWS::CloudFormation::Stack AdditionChild\n └─ [~] Resources\n     └─ [~] .SomeResource:\n         └─ [+] Added: .Properties\n[~] AWS::CloudFormation::Stack DeletionChild\n └─ [~] Resources\n     └─ [~] .SomeResource:\n         └─ [-] Removed: .Properties\n[~] AWS::CloudFormation::Stack ChangedChild\n └─ [~] Resources\n     └─ [~] .SomeResource:\n         └─ [~] .Properties:\n             └─ [~] .Prop:\n                 ├─ [-] old-value\n                 └─ [+] new-value\n\n\n✨  Number of stacks with differences: 4`);\n\n    expect(exitCode).toBe(0);\n  });\n});\n\nclass StringWritable extends Writable {\n  public data: string;\n  private readonly _decoder: StringDecoder;\n\n  constructor(options: any = {}) {\n    super(options);\n    this._decoder = new StringDecoder(options && options.defaultEncoding);\n    this.data = '';\n  }\n\n  public _write(chunk: any, encoding: string, callback: (error?: Error | undefined) => void) {\n    if (encoding === 'buffer') {\n      chunk = this._decoder.write(chunk);\n    }\n    this.data += chunk;\n    callback();\n  }\n\n  public _final(callback: (error?: Error | null) => void) {\n    this.data += this._decoder.end();\n    callback();\n  }\n}\n"]}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"diff.test.js","sourceRoot":"","sources":["diff.test.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable import/order */\nimport { Writable } from 'stream';\nimport { StringDecoder } from 'string_decoder';\nimport * as cxschema from '@aws-cdk/cloud-assembly-schema';\nimport { CloudFormationStackArtifact } from '@aws-cdk/cx-api';\nimport { instanceMockFrom, MockCloudExecutable } from './util';\nimport { Deployments } from '../lib/api/deployments';\nimport { CdkToolkit } from '../lib/cdk-toolkit';\nimport * as cfn from '../lib/api/util/cloudformation';\n\nlet cloudExecutable: MockCloudExecutable;\nlet cloudFormation: jest.Mocked<Deployments>;\nlet toolkit: CdkToolkit;\n\ndescribe('non-nested stacks', () => {\n  beforeEach(() => {\n    cloudExecutable = new MockCloudExecutable({\n      stacks: [{\n        stackName: 'A',\n        template: { resource: 'A' },\n      },\n      {\n        stackName: 'B',\n        depends: ['A'],\n        template: { resource: 'B' },\n      },\n      {\n        stackName: 'C',\n        depends: ['A'],\n        template: { resource: 'C' },\n        metadata: {\n          '/resource': [\n            {\n              type: cxschema.ArtifactMetadataEntryType.ERROR,\n              data: 'this is an error',\n            },\n          ],\n        },\n      },\n      {\n        stackName: 'D',\n        template: { resource: 'D' },\n      }],\n    });\n\n    cloudFormation = instanceMockFrom(Deployments);\n\n    toolkit = new CdkToolkit({\n      cloudExecutable,\n      deployments: cloudFormation,\n      configuration: cloudExecutable.configuration,\n      sdkProvider: cloudExecutable.sdkProvider,\n    });\n\n    // Default implementations\n    cloudFormation.readCurrentTemplateWithNestedStacks.mockImplementation((stackArtifact: CloudFormationStackArtifact) => {\n      if (stackArtifact.stackName === 'D') {\n        return Promise.resolve({\n          deployedTemplate: { resource: 'D' },\n          nestedStackCount: 0,\n        });\n      }\n      return Promise.resolve({\n        deployedTemplate: {},\n        nestedStackCount: 0,\n      });\n    });\n    cloudFormation.deployStack.mockImplementation((options) => Promise.resolve({\n      noOp: true,\n      outputs: {},\n      stackArn: '',\n      stackArtifact: options.stack,\n    }));\n  });\n\n  test('diff can diff multiple stacks', async () => {\n    // GIVEN\n    const buffer = new StringWritable();\n\n    // WHEN\n    const exitCode = await toolkit.diff({\n      stackNames: ['B'],\n      stream: buffer,\n    });\n\n    // THEN\n    const plainTextOutput = buffer.data.replace(/\\x1B\\[[0-?]*[ -/]*[@-~]/g, '');\n    expect(plainTextOutput).toContain('Stack A');\n    expect(plainTextOutput).toContain('Stack B');\n\n    expect(buffer.data.trim()).toContain('✨  Number of stacks with differences: 2');\n    expect(exitCode).toBe(0);\n  });\n\n  test('diff number of stack diffs, not resource diffs', async () => {\n    // GIVEN\n    cloudExecutable = new MockCloudExecutable({\n      stacks: [{\n        stackName: 'A',\n        template: { resourceA: 'A', resourceB: 'B' },\n      },\n      {\n        stackName: 'B',\n        template: { resourceC: 'C' },\n      }],\n    });\n\n    toolkit = new CdkToolkit({\n      cloudExecutable,\n      deployments: cloudFormation,\n      configuration: cloudExecutable.configuration,\n      sdkProvider: cloudExecutable.sdkProvider,\n    });\n\n    const buffer = new StringWritable();\n\n    // WHEN\n    const exitCode = await toolkit.diff({\n      stackNames: ['A', 'B'],\n      stream: buffer,\n    });\n\n    // THEN\n    const plainTextOutput = buffer.data.replace(/\\x1B\\[[0-?]*[ -/]*[@-~]/g, '');\n    expect(plainTextOutput).toContain('Stack A');\n    expect(plainTextOutput).toContain('Stack B');\n\n    expect(buffer.data.trim()).toContain('✨  Number of stacks with differences: 2');\n    expect(exitCode).toBe(0);\n  });\n\n  test('exits with 1 with diffs and fail set to true', async () => {\n    // GIVEN\n    const buffer = new StringWritable();\n\n    // WHEN\n    const exitCode = await toolkit.diff({\n      stackNames: ['A'],\n      stream: buffer,\n      fail: true,\n    });\n\n    // THEN\n    expect(buffer.data.trim()).toContain('✨  Number of stacks with differences: 1');\n    expect(exitCode).toBe(1);\n  });\n\n  test('throws an error if no valid stack names given', async () => {\n    const buffer = new StringWritable();\n\n    // WHEN\n    await expect(() => toolkit.diff({\n      stackNames: ['X', 'Y', 'Z'],\n      stream: buffer,\n    })).rejects.toThrow('No stacks match the name(s) X,Y,Z');\n  });\n\n  test('exits with 1 with diff in first stack, but not in second stack and fail set to true', async () => {\n    // GIVEN\n    const buffer = new StringWritable();\n\n    // WHEN\n    const exitCode = await toolkit.diff({\n      stackNames: ['A', 'D'],\n      stream: buffer,\n      fail: true,\n    });\n\n    // THEN\n    expect(buffer.data.trim()).toContain('✨  Number of stacks with differences: 1');\n    expect(exitCode).toBe(1);\n  });\n\n  test('throws an error during diffs on stack with error metadata', async () => {\n    const buffer = new StringWritable();\n\n    // WHEN\n    await expect(() => toolkit.diff({\n      stackNames: ['C'],\n      stream: buffer,\n    })).rejects.toThrow(/Found errors/);\n  });\n\n  test('when quiet mode is enabled, stacks with no diffs should not print stack name & no differences to stdout', async () => {\n    // GIVEN\n    const buffer = new StringWritable();\n\n    // WHEN\n    const exitCode = await toolkit.diff({\n      stackNames: ['A', 'A'],\n      stream: buffer,\n      fail: false,\n      quiet: true,\n    });\n\n    // THEN\n    expect(buffer.data.trim()).not.toContain('Stack A');\n    expect(buffer.data.trim()).not.toContain('There were no differences');\n    expect(exitCode).toBe(0);\n  });\n});\n\ndescribe('nested stacks', () => {\n  beforeEach(() => {\n    cloudExecutable = new MockCloudExecutable({\n      stacks: [{\n        stackName: 'Parent',\n        template: {},\n      }],\n    });\n\n    cloudFormation = instanceMockFrom(Deployments);\n\n    toolkit = new CdkToolkit({\n      cloudExecutable,\n      deployments: cloudFormation,\n      configuration: cloudExecutable.configuration,\n      sdkProvider: cloudExecutable.sdkProvider,\n    });\n\n    cloudFormation.readCurrentTemplateWithNestedStacks.mockImplementation((stackArtifact: CloudFormationStackArtifact) => {\n      if (stackArtifact.stackName === 'Parent') {\n        stackArtifact.template.Resources = {\n          AdditionChild: {\n            Type: 'AWS::CloudFormation::Stack',\n            Resources: {\n              SomeResource: {\n                Type: 'AWS::Something',\n                Properties: {\n                  Prop: 'added-value',\n                },\n              },\n            },\n          },\n          DeletionChild: {\n            Type: 'AWS::CloudFormation::Stack',\n            Resources: {\n              SomeResource: {\n                Type: 'AWS::Something',\n              },\n            },\n          },\n          ChangedChild: {\n            Type: 'AWS::CloudFormation::Stack',\n            Resources: {\n              SomeResource: {\n                Type: 'AWS::Something',\n                Properties: {\n                  Prop: 'new-value',\n                },\n              },\n            },\n          },\n        };\n        return Promise.resolve({\n          deployedTemplate: {\n            Resources: {\n              AdditionChild: {\n                Type: 'AWS::CloudFormation::Stack',\n                Resources: {\n                  SomeResource: {\n                    Type: 'AWS::Something',\n                  },\n                },\n              },\n              DeletionChild: {\n                Type: 'AWS::CloudFormation::Stack',\n                Resources: {\n                  SomeResource: {\n                    Type: 'AWS::Something',\n                    Properties: {\n                      Prop: 'value-to-be-removed',\n                    },\n                  },\n                },\n              },\n              ChangedChild: {\n                Type: 'AWS::CloudFormation::Stack',\n                Resources: {\n                  SomeResource: {\n                    Type: 'AWS::Something',\n                    Properties: {\n                      Prop: 'old-value',\n                    },\n                  },\n                },\n              },\n            },\n          },\n          nestedStackCount: 3,\n        });\n      }\n      return Promise.resolve({\n        deployedTemplate: {},\n        nestedStackCount: 0,\n      });\n    });\n  });\n\n  test('diff can diff nested stacks', async () => {\n    // GIVEN\n    const buffer = new StringWritable();\n\n    // WHEN\n    const exitCode = await toolkit.diff({\n      stackNames: ['Parent'],\n      stream: buffer,\n    });\n\n    // THEN\n    const plainTextOutput = buffer.data.replace(/\\x1B\\[[0-?]*[ -/]*[@-~]/g, '')\n      .replace(/[ \\t]+$/mg, '');\n    expect(plainTextOutput.trim()).toEqual(`Stack Parent\nResources\n[~] AWS::CloudFormation::Stack AdditionChild\n └─ [~] Resources\n     └─ [~] .SomeResource:\n         └─ [+] Added: .Properties\n[~] AWS::CloudFormation::Stack DeletionChild\n └─ [~] Resources\n     └─ [~] .SomeResource:\n         └─ [-] Removed: .Properties\n[~] AWS::CloudFormation::Stack ChangedChild\n └─ [~] Resources\n     └─ [~] .SomeResource:\n         └─ [~] .Properties:\n             └─ [~] .Prop:\n                 ├─ [-] old-value\n                 └─ [+] new-value\n\n\n✨  Number of stacks with differences: 4`);\n\n    expect(exitCode).toBe(0);\n  });\n\n  test('diff falls back to non-changeset diff for nested stacks', async () => {\n    // GIVEN\n    const changeSetSpy = jest.spyOn(cfn, 'waitForChangeSet');\n    const buffer = new StringWritable();\n\n    // WHEN\n    const exitCode = await toolkit.diff({\n      stackNames: ['Parent'],\n      stream: buffer,\n      changeSet: true,\n    });\n\n    // THEN\n    const plainTextOutput = buffer.data.replace(/\\x1B\\[[0-?]*[ -/]*[@-~]/g, '')\n      .replace(/[ \\t]+$/mg, '');\n    expect(plainTextOutput.trim()).toEqual(`Stack Parent\nResources\n[~] AWS::CloudFormation::Stack AdditionChild\n └─ [~] Resources\n     └─ [~] .SomeResource:\n         └─ [+] Added: .Properties\n[~] AWS::CloudFormation::Stack DeletionChild\n └─ [~] Resources\n     └─ [~] .SomeResource:\n         └─ [-] Removed: .Properties\n[~] AWS::CloudFormation::Stack ChangedChild\n └─ [~] Resources\n     └─ [~] .SomeResource:\n         └─ [~] .Properties:\n             └─ [~] .Prop:\n                 ├─ [-] old-value\n                 └─ [+] new-value\n\n\n✨  Number of stacks with differences: 4`);\n\n    expect(exitCode).toBe(0);\n    expect(changeSetSpy).not.toHaveBeenCalled();\n  });\n});\n\nclass StringWritable extends Writable {\n  public data: string;\n  private readonly _decoder: StringDecoder;\n\n  constructor(options: any = {}) {\n    super(options);\n    this._decoder = new StringDecoder(options && options.defaultEncoding);\n    this.data = '';\n  }\n\n  public _write(chunk: any, encoding: string, callback: (error?: Error | undefined) => void) {\n    if (encoding === 'buffer') {\n      chunk = this._decoder.write(chunk);\n    }\n    this.data += chunk;\n    callback();\n  }\n\n  public _final(callback: (error?: Error | null) => void) {\n    this.data += this._decoder.end();\n    callback();\n  }\n}\n"]}

Sorry, the diff of this file is not supported yet

Sorry, the diff of this file is not supported yet

Sorry, the diff of this file is too big to display

Sorry, the diff of this file is too big to display

Sorry, the diff of this file is too big to display

Sorry, the diff of this file is too big to display

Sorry, the diff of this file is too big to display

Sorry, the diff of this file is not supported yet

Sorry, the diff of this file is too big to display

Sorry, the diff of this file is not supported yet

Sorry, the diff of this file is not supported yet

Sorry, the diff of this file is not supported yet

SocketSocket SOC 2 Logo

Product

  • Package Alerts
  • Integrations
  • Docs
  • Pricing
  • FAQ
  • Roadmap
  • Changelog

Packages

npm

Stay in touch

Get open source security insights delivered straight into your inbox.


  • Terms
  • Privacy
  • Security

Made with ⚡️ by Socket Inc