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.124.0 to 2.125.0

4

build-info.json
{
"comment": "Generated at 2024-01-26T20:35:52Z by generate.sh",
"commit": "4b6724c"
"comment": "Generated at 2024-01-31T23:43:58Z by generate.sh",
"commit": "5e3c3c6"
}

@@ -124,2 +124,3 @@ /// <reference types="node" />

};
resourcesToImport?: ResourcesToImport;
};

@@ -137,2 +138,4 @@ export type CreateChangeSetOptions = {

};
resourcesToImport?: ResourcesToImport;
role?: string;
};

@@ -143,3 +146,3 @@ /**

export declare function createDiffChangeSet(options: PrepareChangeSetOptions): Promise<CloudFormation.DescribeChangeSetOutput | undefined>;
export declare function cleanupOldChangeset(exists: boolean, changeSetName: string, stackName: string, cfn: CloudFormation): Promise<void>;
export declare function cleanupOldChangeset(stackExists: boolean, changeSetName: string, stackName: string, cfn: CloudFormation): Promise<void>;
/**

@@ -146,0 +149,0 @@ * Return true if the given change set has no changes

@@ -144,2 +144,7 @@ import { DeployOptions } from '@aws-cdk/cloud-assembly-schema';

/**
* Removes CDKMetadata and Outputs in the template so that only resources for importing are left.
* @returns template with import resources only
*/
export declare function removeNonImportResources(stack: cxapi.CloudFormationStackArtifact): any;
/**
* Information about a resource in the template that is importable

@@ -146,0 +151,0 @@ */

"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
exports.ResourceImporter = void 0;
exports.removeNonImportResources = exports.ResourceImporter = void 0;
const cfnDiff = require("@aws-cdk/cloudformation-diff");

@@ -294,9 +294,17 @@ const chalk = require("chalk");

removeNonImportResources() {
const template = this.stack.template;
delete template.Resources.CDKMetadata;
delete template.Outputs;
return template;
return removeNonImportResources(this.stack);
}
}
exports.ResourceImporter = ResourceImporter;
/**
* Removes CDKMetadata and Outputs in the template so that only resources for importing are left.
* @returns template with import resources only
*/
function removeNonImportResources(stack) {
const template = stack.template;
delete template.Resources.CDKMetadata;
delete template.Outputs;
return template;
}
exports.removeNonImportResources = removeNonImportResources;
function fmtdict(xs) {

@@ -319,2 +327,2 @@ return Object.entries(xs).map(([k, v]) => `${k}=${v}`).join(', ');

}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"import.js","sourceRoot":"","sources":["import.ts"],"names":[],"mappings":"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","sourcesContent":["import { DeployOptions } from '@aws-cdk/cloud-assembly-schema';\nimport * 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 { DeploymentMethod } from './api';\nimport { Deployments } from './api/deployments';\nimport { ResourceIdentifierProperties, ResourcesToImport } from './api/util/cloudformation';\nimport { StackActivityProgress } from './api/util/cloudformation/stack-activity-monitor';\nimport { Tag } from './cdk-toolkit';\nimport { error, print, success, warning } from './logging';\n\nexport interface ImportDeploymentOptions extends DeployOptions {\n  deploymentMethod?: DeploymentMethod;\n  progress?: StackActivityProgress;\n  tags?: Tag[];\n}\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 importResourcesFromMap(importMap: ImportMap, options: ImportDeploymentOptions) {\n    const resourcesToImport: ResourcesToImport = await this.makeResourcesToImport(importMap);\n    const updatedTemplate = await this.currentTemplateWithAdditions(importMap.importResources);\n\n    await this.importResources(updatedTemplate, resourcesToImport, options);\n  }\n\n  /**\n   * Based on the app and resources file generated by cdk migrate. Removes all items from the template that\n   * cannot be included in an import change-set for new stacks and performs the import operation,\n   * creating the new stack.\n   *\n   * @param resourcesToImport The mapping created by cdk migrate\n   * @param options Options to pass to CloudFormation deploy operation\n   */\n  public async importResourcesFromMigrate(resourcesToImport: ResourcesToImport, options: ImportDeploymentOptions) {\n    const updatedTemplate = this.removeNonImportResources();\n\n    await this.importResources(updatedTemplate, resourcesToImport, options);\n  }\n\n  private async importResources(overrideTemplate: any, resourcesToImport: ResourcesToImport, options: ImportDeploymentOptions) {\n    try {\n      const result = await this.cfn.deployStack({\n        stack: this.stack,\n        deployName: this.stack.stackName,\n        ...options,\n        overrideTemplate,\n        resourcesToImport,\n      });\n\n      const message = result.noOp\n        ? ' ✅  %s (no changes)'\n        : ' ✅  %s';\n\n      success('\\n' + message, this.stack.displayName);\n    } catch (e) {\n      error('\\n ❌  %s failed: %s', chalk.bold(this.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   * Resolves the environment of a stack.\n   */\n  public async resolveEnvironment(): Promise<cxapi.Environment> {\n    return this.cfn.resolveEnvironment(this.stack);\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   * Removes CDKMetadata and Outputs in the template so that only resources for importing are left.\n   * @returns template with import resources only\n   */\n  private removeNonImportResources() {\n    const template = this.stack.template;\n    delete template.Resources.CDKMetadata;\n    delete template.Outputs;\n    return template;\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 { DeployOptions } from '@aws-cdk/cloud-assembly-schema';\nimport * 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 { DeploymentMethod } from './api';\nimport { Deployments } from './api/deployments';\nimport { ResourceIdentifierProperties, ResourcesToImport } from './api/util/cloudformation';\nimport { StackActivityProgress } from './api/util/cloudformation/stack-activity-monitor';\nimport { Tag } from './cdk-toolkit';\nimport { error, print, success, warning } from './logging';\n\nexport interface ImportDeploymentOptions extends DeployOptions {\n  deploymentMethod?: DeploymentMethod;\n  progress?: StackActivityProgress;\n  tags?: Tag[];\n}\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 importResourcesFromMap(importMap: ImportMap, options: ImportDeploymentOptions) {\n    const resourcesToImport: ResourcesToImport = await this.makeResourcesToImport(importMap);\n    const updatedTemplate = await this.currentTemplateWithAdditions(importMap.importResources);\n\n    await this.importResources(updatedTemplate, resourcesToImport, options);\n  }\n\n  /**\n   * Based on the app and resources file generated by cdk migrate. Removes all items from the template that\n   * cannot be included in an import change-set for new stacks and performs the import operation,\n   * creating the new stack.\n   *\n   * @param resourcesToImport The mapping created by cdk migrate\n   * @param options Options to pass to CloudFormation deploy operation\n   */\n  public async importResourcesFromMigrate(resourcesToImport: ResourcesToImport, options: ImportDeploymentOptions) {\n    const updatedTemplate = this.removeNonImportResources();\n\n    await this.importResources(updatedTemplate, resourcesToImport, options);\n  }\n\n  private async importResources(overrideTemplate: any, resourcesToImport: ResourcesToImport, options: ImportDeploymentOptions) {\n    try {\n      const result = await this.cfn.deployStack({\n        stack: this.stack,\n        deployName: this.stack.stackName,\n        ...options,\n        overrideTemplate,\n        resourcesToImport,\n      });\n\n      const message = result.noOp\n        ? ' ✅  %s (no changes)'\n        : ' ✅  %s';\n\n      success('\\n' + message, this.stack.displayName);\n    } catch (e) {\n      error('\\n ❌  %s failed: %s', chalk.bold(this.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   * Resolves the environment of a stack.\n   */\n  public async resolveEnvironment(): Promise<cxapi.Environment> {\n    return this.cfn.resolveEnvironment(this.stack);\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   * Removes CDKMetadata and Outputs in the template so that only resources for importing are left.\n   * @returns template with import resources only\n   */\n  private removeNonImportResources() {\n    return removeNonImportResources(this.stack);\n  }\n}\n\n/**\n * Removes CDKMetadata and Outputs in the template so that only resources for importing are left.\n * @returns template with import resources only\n */\nexport function removeNonImportResources(stack: cxapi.CloudFormationStackArtifact) {\n  const template = stack.template;\n  delete template.Resources.CDKMetadata;\n  delete template.Outputs;\n  return template;\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.11.6",
"@types/node": "20.11.13",
"jest": "^29.7.0",

@@ -18,0 +18,0 @@ "ts-jest": "^29.1.2",

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

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

@@ -16,0 +16,0 @@ "constructs": "%constructs-version%",

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

"@types/jest": "^29.5.11",
"@types/node": "20.11.6",
"@types/node": "20.11.13",
"jest": "^29.7.0",

@@ -18,0 +18,0 @@ "ts-jest": "^29.1.2",

{
"name": "aws-cdk",
"description": "CDK Toolkit, the command line tool for CDK apps",
"version": "2.124.0",
"version": "2.125.0",
"bin": {

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

"devDependencies": {
"@aws-cdk/cdk-build-tools": "2.124.0-alpha.0",
"@aws-cdk/pkglint": "2.124.0-alpha.0",
"@aws-cdk/cdk-build-tools": "2.125.0-alpha.0",
"@aws-cdk/pkglint": "2.125.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.124.0",
"aws-cdk-lib": "2.125.0",
"aws-sdk-mock": "5.6.0",
"axios": "^1.6.5",
"axios": "^1.6.7",
"constructs": "^10.0.0",
"fast-check": "^3.15.0",
"fast-check": "^3.15.1",
"jest": "^29.7.0",

@@ -94,3 +94,3 @@ "jest-mock": "^29.7.0",

"make-runnable": "^1.4.1",
"nock": "^13.5.0",
"nock": "^13.5.1",
"sinon": "^9.2.4",

@@ -100,11 +100,11 @@ "ts-jest": "^29.1.2",

"xml-js": "^1.6.11",
"@aws-cdk/cloud-assembly-schema": "2.124.0",
"@aws-cdk/cloudformation-diff": "2.124.0",
"@aws-cdk/cx-api": "2.124.0",
"@aws-cdk/region-info": "2.124.0",
"@aws-cdk/cloud-assembly-schema": "2.125.0",
"@aws-cdk/cloudformation-diff": "2.125.0",
"@aws-cdk/cx-api": "2.125.0",
"@aws-cdk/region-info": "2.125.0",
"@jsii/check-node": "1.94.0",
"archiver": "^5.3.2",
"aws-sdk": "^2.1542.0",
"aws-sdk": "^2.1547.0",
"camelcase": "^6.3.0",
"cdk-assets": "2.124.0",
"cdk-assets": "2.125.0",
"cdk-from-cfn": "^0.122.0",

@@ -111,0 +111,0 @@ "chalk": "^4",

@@ -14,2 +14,88 @@ "use strict";

let toolkit;
describe('imports', () => {
beforeEach(() => {
jest.spyOn(cfn, 'createDiffChangeSet').mockImplementation(async () => {
return {
Changes: [
{
ResourceChange: {
Action: 'Import',
LogicalResourceId: 'Queue',
},
},
{
ResourceChange: {
Action: 'Import',
LogicalResourceId: 'Bucket',
},
},
{
ResourceChange: {
Action: 'Import',
LogicalResourceId: 'Queue2',
},
},
],
};
});
cloudExecutable = new util_1.MockCloudExecutable({
stacks: [{
stackName: 'A',
template: {
Resources: {
Queue: {
Type: 'AWS::SQS::Queue',
},
Queue2: {
Type: 'AWS::SQS::Queue',
},
Bucket: {
Type: 'AWS::S3::Bucket',
},
},
},
}],
});
cloudFormation = (0, util_1.instanceMockFrom)(deployments_1.Deployments);
toolkit = new cdk_toolkit_1.CdkToolkit({
cloudExecutable,
deployments: cloudFormation,
configuration: cloudExecutable.configuration,
sdkProvider: cloudExecutable.sdkProvider,
});
// Default implementations
cloudFormation.readCurrentTemplateWithNestedStacks.mockImplementation((_stackArtifact) => {
return Promise.resolve({
deployedTemplate: {},
nestedStackCount: 0,
});
});
cloudFormation.deployStack.mockImplementation((options) => Promise.resolve({
noOp: true,
outputs: {},
stackArn: '',
stackArtifact: options.stack,
}));
});
test('imports', async () => {
// GIVEN
const buffer = new StringWritable();
// WHEN
const exitCode = await toolkit.diff({
stackNames: ['A'],
stream: buffer,
changeSet: true,
});
// THEN
const plainTextOutput = buffer.data.replace(/\x1B\[[0-?]*[ -/]*[@-~]/g, '');
expect(plainTextOutput).toContain(`Stack A
Resources
[←] AWS::SQS::Queue Queue import
[←] AWS::SQS::Queue Queue2 import
[←] AWS::S3::Bucket Bucket import
`);
expect(buffer.data.trim()).toContain('✨ Number of stacks with differences: 1');
expect(exitCode).toBe(0);
});
});
describe('non-nested stacks', () => {

@@ -357,2 +443,2 @@ beforeEach(() => {

}
//# 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"]}
//# 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('imports', () => {\n  beforeEach(() => {\n    jest.spyOn(cfn, 'createDiffChangeSet').mockImplementation(async () => {\n      return {\n        Changes: [\n          {\n            ResourceChange: {\n              Action: 'Import',\n              LogicalResourceId: 'Queue',\n            },\n          },\n          {\n            ResourceChange: {\n              Action: 'Import',\n              LogicalResourceId: 'Bucket',\n            },\n          },\n          {\n            ResourceChange: {\n              Action: 'Import',\n              LogicalResourceId: 'Queue2',\n            },\n          },\n        ],\n      };\n    });\n    cloudExecutable = new MockCloudExecutable({\n      stacks: [{\n        stackName: 'A',\n        template: {\n          Resources: {\n            Queue: {\n              Type: 'AWS::SQS::Queue',\n            },\n            Queue2: {\n              Type: 'AWS::SQS::Queue',\n            },\n            Bucket: {\n              Type: 'AWS::S3::Bucket',\n            },\n          },\n        },\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      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('imports', async () => {\n    // GIVEN\n    const buffer = new StringWritable();\n\n    // WHEN\n    const exitCode = await toolkit.diff({\n      stackNames: ['A'],\n      stream: buffer,\n      changeSet: true,\n    });\n\n    // THEN\n    const plainTextOutput = buffer.data.replace(/\\x1B\\[[0-?]*[ -/]*[@-~]/g, '');\n    expect(plainTextOutput).toContain(`Stack A\nResources\n[←] AWS::SQS::Queue Queue import\n[←] AWS::SQS::Queue Queue2 import\n[←] AWS::S3::Bucket Bucket import\n`);\n\n    expect(buffer.data.trim()).toContain('✨  Number of stacks with differences: 1');\n    expect(exitCode).toBe(0);\n  });\n});\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 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 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