UNPKG

aws-cdk

Version:

AWS CDK CLI, the command line tool for CDK apps

438 lines 67.4 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.ParameterValues = exports.TemplateParameters = void 0; exports.waitForChangeSet = waitForChangeSet; exports.createDiffChangeSet = createDiffChangeSet; exports.uploadStackTemplateAssets = uploadStackTemplateAssets; exports.createChangeSet = createChangeSet; exports.changeSetHasNoChanges = changeSetHasNoChanges; exports.waitForStackDelete = waitForStackDelete; exports.waitForStackDeploy = waitForStackDeploy; exports.stabilizeStack = stabilizeStack; const util_1 = require("util"); const cxapi = require("@aws-cdk/cx-api"); const cx_api_1 = require("@aws-cdk/cx-api"); const client_cloudformation_1 = require("@aws-sdk/client-cloudformation"); const cdk_assets_1 = require("cdk-assets"); const asset_manifest_builder_1 = require("./asset-manifest-builder"); const api_1 = require("../../../../@aws-cdk/tmp-toolkit-helpers/src/api"); const private_1 = require("../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/private"); const cloudformation_1 = require("../cloudformation"); /** * Describe a changeset in CloudFormation, regardless of its current state. * * @param cfn a CloudFormation client * @param stackName the name of the Stack the ChangeSet belongs to * @param changeSetName the name of the ChangeSet * @param fetchAll if true, fetches all pages of the change set description. * * @returns CloudFormation information about the ChangeSet */ async function describeChangeSet(cfn, stackName, changeSetName, { fetchAll }) { const response = await cfn.describeChangeSet({ StackName: stackName, ChangeSetName: changeSetName, }); // If fetchAll is true, traverse all pages from the change set description. while (fetchAll && response.NextToken != null) { const nextPage = await cfn.describeChangeSet({ StackName: stackName, ChangeSetName: response.ChangeSetId ?? changeSetName, NextToken: response.NextToken, }); // Consolidate the changes if (nextPage.Changes != null) { response.Changes = response.Changes != null ? response.Changes.concat(nextPage.Changes) : nextPage.Changes; } // Forward the new NextToken response.NextToken = nextPage.NextToken; } return response; } /** * Waits for a function to return non-+undefined+ before returning. * * @param valueProvider a function that will return a value that is not +undefined+ once the wait should be over * @param timeout the time to wait between two calls to +valueProvider+ * * @returns the value that was returned by +valueProvider+ */ async function waitFor(valueProvider, timeout = 5000) { while (true) { const result = await valueProvider(); if (result === null) { return undefined; } else if (result !== undefined) { return result; } await new Promise((cb) => setTimeout(cb, timeout)); } } /** * Waits for a ChangeSet to be available for triggering a StackUpdate. * * Will return a changeset that is either ready to be executed or has no changes. * Will throw in other cases. * * @param cfn a CloudFormation client * @param stackName the name of the Stack that the ChangeSet belongs to * @param changeSetName the name of the ChangeSet * @param fetchAll if true, fetches all pages of the ChangeSet before returning. * * @returns the CloudFormation description of the ChangeSet */ async function waitForChangeSet(cfn, ioHelper, stackName, changeSetName, { fetchAll }) { await ioHelper.notify(private_1.IO.DEFAULT_TOOLKIT_DEBUG.msg((0, util_1.format)('Waiting for changeset %s on stack %s to finish creating...', changeSetName, stackName))); const ret = await waitFor(async () => { const description = await describeChangeSet(cfn, stackName, changeSetName, { fetchAll, }); // The following doesn't use a switch because tsc will not allow fall-through, UNLESS it is allows // EVERYWHERE that uses this library directly or indirectly, which is undesirable. if (description.Status === 'CREATE_PENDING' || description.Status === 'CREATE_IN_PROGRESS') { await ioHelper.notify(private_1.IO.DEFAULT_TOOLKIT_DEBUG.msg((0, util_1.format)('Changeset %s on stack %s is still creating', changeSetName, stackName))); return undefined; } if (description.Status === client_cloudformation_1.ChangeSetStatus.CREATE_COMPLETE || changeSetHasNoChanges(description)) { return description; } // eslint-disable-next-line max-len throw new api_1.ToolkitError(`Failed to create ChangeSet ${changeSetName} on ${stackName}: ${description.Status || 'NO_STATUS'}, ${description.StatusReason || 'no reason provided'}`); }); if (!ret) { throw new api_1.ToolkitError('Change set took too long to be created; aborting'); } return ret; } /** * Create a changeset for a diff operation */ async function createDiffChangeSet(ioHelper, options) { // `options.stack` has been modified to include any nested stack templates directly inline with its own template, under a special `NestedTemplate` property. // Thus the parent template's Resources section contains the nested template's CDK metadata check, which uses Fn::Equals. // This causes CreateChangeSet to fail with `Template Error: Fn::Equals cannot be partially collapsed`. for (const resource of Object.values(options.stack.template.Resources ?? {})) { if (resource.Type === 'AWS::CloudFormation::Stack') { await ioHelper.notify(private_1.IO.DEFAULT_TOOLKIT_DEBUG.msg('This stack contains one or more nested stacks, falling back to template-only diff...')); return undefined; } } return uploadBodyParameterAndCreateChangeSet(ioHelper, options); } /** * Returns all file entries from an AssetManifestArtifact that look like templates. * * This is used in the `uploadBodyParameterAndCreateChangeSet` function to find * all template asset files to build and publish. * * Returns a tuple of [AssetManifest, FileManifestEntry[]] */ function templatesFromAssetManifestArtifact(artifact) { const assets = []; const fileName = artifact.file; const assetManifest = cdk_assets_1.AssetManifest.fromFile(fileName); assetManifest.entries.forEach((entry) => { if (entry.type === 'file') { const source = entry.source; if (source.path && source.path.endsWith('.template.json')) { assets.push(entry); } } }); return [assetManifest, assets]; } async function uploadBodyParameterAndCreateChangeSet(ioHelper, options) { try { await uploadStackTemplateAssets(options.stack, options.deployments); const env = await options.deployments.envs.accessStackForMutableStackOperations(options.stack); const bodyParameter = await (0, cloudformation_1.makeBodyParameter)(ioHelper, options.stack, env.resolvedEnvironment, new asset_manifest_builder_1.AssetManifestBuilder(), env.resources); const cfn = env.sdk.cloudFormation(); const exists = (await cloudformation_1.CloudFormationStack.lookup(cfn, options.stack.stackName, false)).exists; const executionRoleArn = await env.replacePlaceholders(options.stack.cloudFormationExecutionRoleArn); await ioHelper.notify(private_1.IO.DEFAULT_TOOLKIT_INFO.msg('Hold on while we create a read-only change set to get a diff with accurate replacement information (use --no-change-set to use a less accurate but faster template-only diff)\n')); return await createChangeSet(ioHelper, { cfn, changeSetName: 'cdk-diff-change-set', stack: options.stack, exists, uuid: options.uuid, willExecute: options.willExecute, bodyParameter, parameters: options.parameters, resourcesToImport: options.resourcesToImport, role: executionRoleArn, }); } catch (e) { await ioHelper.notify(private_1.IO.DEFAULT_TOOLKIT_DEBUG.msg(String(e))); await ioHelper.notify(private_1.IO.DEFAULT_TOOLKIT_INFO.msg('Could not create a change set, will base the diff on template differences (run again with -v to see the reason)\n')); return undefined; } } /** * Uploads the assets that look like templates for this CloudFormation stack * * This is necessary for any CloudFormation call that needs the template, it may need * to be uploaded to an S3 bucket first. We have to follow the instructions in the * asset manifest, because technically that is the only place that knows about * bucket and assumed roles and such. */ async function uploadStackTemplateAssets(stack, deployments) { for (const artifact of stack.dependencies) { // Skip artifact if it is not an Asset Manifest Artifact if (!cxapi.AssetManifestArtifact.isAssetManifestArtifact(artifact)) { continue; } const [assetManifest, file_entries] = templatesFromAssetManifestArtifact(artifact); for (const entry of file_entries) { await deployments.buildSingleAsset(artifact, assetManifest, entry, { stack, }); await deployments.publishSingleAsset(assetManifest, entry, { stack, }); } } } async function createChangeSet(ioHelper, options) { await cleanupOldChangeset(options.cfn, ioHelper, options.changeSetName, options.stack.stackName); await ioHelper.notify(private_1.IO.DEFAULT_TOOLKIT_DEBUG.msg(`Attempting to create ChangeSet with name ${options.changeSetName} for stack ${options.stack.stackName}`)); const templateParams = TemplateParameters.fromTemplate(options.stack.template); const stackParams = templateParams.supplyAll(options.parameters); const changeSet = await options.cfn.createChangeSet({ StackName: options.stack.stackName, ChangeSetName: options.changeSetName, ChangeSetType: options.resourcesToImport ? 'IMPORT' : options.exists ? 'UPDATE' : 'CREATE', Description: `CDK Changeset for diff ${options.uuid}`, ClientToken: `diff${options.uuid}`, TemplateURL: options.bodyParameter.TemplateURL, TemplateBody: options.bodyParameter.TemplateBody, Parameters: stackParams.apiParameters, ResourcesToImport: options.resourcesToImport, RoleARN: options.role, Tags: toCfnTags(options.stack.tags), Capabilities: ['CAPABILITY_IAM', 'CAPABILITY_NAMED_IAM', 'CAPABILITY_AUTO_EXPAND'], }); await ioHelper.notify(private_1.IO.DEFAULT_TOOLKIT_DEBUG.msg((0, util_1.format)('Initiated creation of changeset: %s; waiting for it to finish creating...', changeSet.Id))); // Fetching all pages if we'll execute, so we can have the correct change count when monitoring. const createdChangeSet = await waitForChangeSet(options.cfn, ioHelper, options.stack.stackName, options.changeSetName, { fetchAll: options.willExecute, }); await cleanupOldChangeset(options.cfn, ioHelper, options.changeSetName, options.stack.stackName); return createdChangeSet; } function toCfnTags(tags) { return Object.entries(tags).map(([k, v]) => ({ Key: k, Value: v, })); } async function cleanupOldChangeset(cfn, ioHelper, changeSetName, stackName) { // Delete any existing change sets generated by CDK since change set names must be unique. // The delete request is successful as long as the stack exists (even if the change set does not exist). await ioHelper.notify(private_1.IO.DEFAULT_TOOLKIT_DEBUG.msg(`Removing existing change set with name ${changeSetName} if it exists`)); await cfn.deleteChangeSet({ StackName: stackName, ChangeSetName: changeSetName, }); } /** * Return true if the given change set has no changes * * This must be determined from the status, not the 'Changes' array on the * object; the latter can be empty because no resources were changed, but if * there are changes to Outputs, the change set can still be executed. */ function changeSetHasNoChanges(description) { const noChangeErrorPrefixes = [ // Error message for a regular template "The submitted information didn't contain changes.", // Error message when a Transform is involved (see #10650) 'No updates are to be performed.', ]; return (description.Status === 'FAILED' && noChangeErrorPrefixes.some((p) => (description.StatusReason ?? '').startsWith(p))); } /** * Waits for a CloudFormation stack to stabilize in a complete/available state * after a delete operation is issued. * * Fails if the stack is in a FAILED state. Will not fail if the stack was * already deleted. * * @param cfn a CloudFormation client * @param stackName the name of the stack to wait for after a delete * * @returns the CloudFormation description of the stabilized stack after the delete attempt */ async function waitForStackDelete(cfn, ioHelper, stackName) { const stack = await stabilizeStack(cfn, ioHelper, stackName); if (!stack) { return undefined; } const status = stack.stackStatus; if (status.isFailure) { throw new api_1.ToolkitError(`The stack named ${stackName} is in a failed state. You may need to delete it from the AWS console : ${status}`); } else if (status.isDeleted) { return undefined; } return stack; } /** * Waits for a CloudFormation stack to stabilize in a complete/available state * after an update/create operation is issued. * * Fails if the stack is in a FAILED state, ROLLBACK state, or DELETED state. * * @param cfn a CloudFormation client * @param stackName the name of the stack to wait for after an update * * @returns the CloudFormation description of the stabilized stack after the update attempt */ async function waitForStackDeploy(cfn, ioHelper, stackName) { const stack = await stabilizeStack(cfn, ioHelper, stackName); if (!stack) { return undefined; } const status = stack.stackStatus; if (status.isCreationFailure) { throw new api_1.ToolkitError(`The stack named ${stackName} failed creation, it may need to be manually deleted from the AWS console: ${status}`); } else if (!status.isDeploySuccess) { throw new api_1.ToolkitError(`The stack named ${stackName} failed to deploy: ${status}`); } return stack; } /** * Wait for a stack to become stable (no longer _IN_PROGRESS), returning it */ async function stabilizeStack(cfn, ioHelper, stackName) { await ioHelper.notify(private_1.IO.DEFAULT_TOOLKIT_DEBUG.msg((0, util_1.format)('Waiting for stack %s to finish creating or updating...', stackName))); return waitFor(async () => { const stack = await cloudformation_1.CloudFormationStack.lookup(cfn, stackName); if (!stack.exists) { await ioHelper.notify(private_1.IO.DEFAULT_TOOLKIT_DEBUG.msg((0, util_1.format)('Stack %s does not exist', stackName))); return null; } const status = stack.stackStatus; if (status.isInProgress) { await ioHelper.notify(private_1.IO.DEFAULT_TOOLKIT_DEBUG.msg((0, util_1.format)('Stack %s has an ongoing operation in progress and is not stable (%s)', stackName, status))); return undefined; } else if (status.isReviewInProgress) { // This may happen if a stack creation operation is interrupted before the ChangeSet execution starts. Recovering // from this would requiring manual intervention (deleting or executing the pending ChangeSet), and failing to do // so will result in an endless wait here (the ChangeSet wont delete or execute itself). Instead of blocking // "forever" we proceed as if the stack was existing and stable. If there is a concurrent operation that just // hasn't finished proceeding just yet, either this operation or the concurrent one may fail due to the other one // having made progress. Which is fine. I guess. await ioHelper.notify(private_1.IO.DEFAULT_TOOLKIT_DEBUG.msg((0, util_1.format)('Stack %s is in REVIEW_IN_PROGRESS state. Considering this is a stable status (%s)', stackName, status))); } return stack; }); } /** * The set of (formal) parameters that have been declared in a template */ class TemplateParameters { static fromTemplate(template) { return new TemplateParameters(template.Parameters || {}); } constructor(params) { this.params = params; } /** * Calculate stack parameters to pass from the given desired parameter values * * Will throw if parameters without a Default value or a Previous value are not * supplied. */ supplyAll(updates) { return new ParameterValues(this.params, updates); } /** * From the template, the given desired values and the current values, calculate the changes to the stack parameters * * Will take into account parameters already set on the template (will emit * 'UsePreviousValue: true' for those unless the value is changed), and will * throw if parameters without a Default value or a Previous value are not * supplied. */ updateExisting(updates, previousValues) { return new ParameterValues(this.params, updates, previousValues); } } exports.TemplateParameters = TemplateParameters; /** * The set of parameters we're going to pass to a Stack */ class ParameterValues { constructor(formalParams, updates, previousValues = {}) { this.formalParams = formalParams; this.values = {}; this.apiParameters = []; const missingRequired = new Array(); for (const [key, formalParam] of Object.entries(this.formalParams)) { // Check updates first, then use the previous value (if available), then use // the default (if available). // // If we don't find a parameter value using any of these methods, then that's an error. const updatedValue = updates[key]; if (updatedValue !== undefined) { this.values[key] = updatedValue; this.apiParameters.push({ ParameterKey: key, ParameterValue: updates[key], }); continue; } if (key in previousValues) { this.values[key] = previousValues[key]; this.apiParameters.push({ ParameterKey: key, UsePreviousValue: true }); continue; } if (formalParam.Default !== undefined) { this.values[key] = formalParam.Default; continue; } // Oh no missingRequired.push(key); } if (missingRequired.length > 0) { throw new api_1.ToolkitError(`The following CloudFormation Parameters are missing a value: ${missingRequired.join(', ')}`); } // Just append all supplied overrides that aren't really expected (this // will fail CFN but maybe people made typos that they want to be notified // of) const unknownParam = ([key, _]) => this.formalParams[key] === undefined; const hasValue = ([_, value]) => !!value; for (const [key, value] of Object.entries(updates).filter(unknownParam).filter(hasValue)) { this.values[key] = value; this.apiParameters.push({ ParameterKey: key, ParameterValue: value }); } } /** * Whether this set of parameter updates will change the actual stack values */ hasChanges(currentValues) { // If any of the parameters are SSM parameters, deploying must always happen // because we can't predict what the values will be. We will allow some // parameters to opt out of this check by having a magic string in their description. if (Object.values(this.formalParams).some((p) => p.Type.startsWith('AWS::SSM::Parameter::') && !p.Description?.includes(cx_api_1.SSMPARAM_NO_INVALIDATE))) { return 'ssm'; } // Otherwise we're dirty if: // - any of the existing values are removed, or changed if (Object.entries(currentValues).some(([key, value]) => !(key in this.values) || value !== this.values[key])) { return true; } // - any of the values we're setting are new if (Object.keys(this.values).some((key) => !(key in currentValues))) { return true; } return false; } } exports.ParameterValues = ParameterValues; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cfn-api.js","sourceRoot":"","sources":["cfn-api.ts"],"names":[],"mappings":"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","sourcesContent":["import { format } from 'util';\nimport * as cxapi from '@aws-cdk/cx-api';\nimport { SSMPARAM_NO_INVALIDATE } from '@aws-cdk/cx-api';\nimport type {\n  DescribeChangeSetCommandOutput,\n  Parameter,\n  ResourceToImport,\n  Tag,\n} from '@aws-sdk/client-cloudformation';\nimport {\n  ChangeSetStatus,\n} from '@aws-sdk/client-cloudformation';\nimport type { FileManifestEntry } from 'cdk-assets';\nimport { AssetManifest } from 'cdk-assets';\nimport { AssetManifestBuilder } from './asset-manifest-builder';\nimport type { Deployments } from './deployments';\nimport { ToolkitError } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api';\nimport { IO, type IoHelper } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api/io/private';\nimport type { ICloudFormationClient, SdkProvider } from '../aws-auth';\nimport type { Template, TemplateBodyParameter, TemplateParameter } from '../cloudformation';\nimport { CloudFormationStack, makeBodyParameter } from '../cloudformation';\nimport type { ResourcesToImport } from '../resource-import';\n\n/**\n * Describe a changeset in CloudFormation, regardless of its current state.\n *\n * @param cfn           a CloudFormation client\n * @param stackName     the name of the Stack the ChangeSet belongs to\n * @param changeSetName the name of the ChangeSet\n * @param fetchAll      if true, fetches all pages of the change set description.\n *\n * @returns       CloudFormation information about the ChangeSet\n */\nasync function describeChangeSet(\n  cfn: ICloudFormationClient,\n  stackName: string,\n  changeSetName: string,\n  { fetchAll }: { fetchAll: boolean },\n): Promise<DescribeChangeSetCommandOutput> {\n  const response = await cfn.describeChangeSet({\n    StackName: stackName,\n    ChangeSetName: changeSetName,\n  });\n\n  // If fetchAll is true, traverse all pages from the change set description.\n  while (fetchAll && response.NextToken != null) {\n    const nextPage = await cfn.describeChangeSet({\n      StackName: stackName,\n      ChangeSetName: response.ChangeSetId ?? changeSetName,\n      NextToken: response.NextToken,\n    });\n\n    // Consolidate the changes\n    if (nextPage.Changes != null) {\n      response.Changes = response.Changes != null ? response.Changes.concat(nextPage.Changes) : nextPage.Changes;\n    }\n\n    // Forward the new NextToken\n    response.NextToken = nextPage.NextToken;\n  }\n\n  return response;\n}\n\n/**\n * Waits for a function to return non-+undefined+ before returning.\n *\n * @param valueProvider a function that will return a value that is not +undefined+ once the wait should be over\n * @param timeout     the time to wait between two calls to +valueProvider+\n *\n * @returns       the value that was returned by +valueProvider+\n */\nasync function waitFor<T>(\n  valueProvider: () => Promise<T | null | undefined>,\n  timeout: number = 5000,\n): Promise<T | undefined> {\n  while (true) {\n    const result = await valueProvider();\n    if (result === null) {\n      return undefined;\n    } else if (result !== undefined) {\n      return result;\n    }\n    await new Promise((cb) => setTimeout(cb, timeout));\n  }\n}\n\n/**\n * Waits for a ChangeSet to be available for triggering a StackUpdate.\n *\n * Will return a changeset that is either ready to be executed or has no changes.\n * Will throw in other cases.\n *\n * @param cfn           a CloudFormation client\n * @param stackName     the name of the Stack that the ChangeSet belongs to\n * @param changeSetName the name of the ChangeSet\n * @param fetchAll      if true, fetches all pages of the ChangeSet before returning.\n *\n * @returns       the CloudFormation description of the ChangeSet\n */\nexport async function waitForChangeSet(\n  cfn: ICloudFormationClient,\n  ioHelper: IoHelper,\n  stackName: string,\n  changeSetName: string,\n  { fetchAll }: { fetchAll: boolean },\n): Promise<DescribeChangeSetCommandOutput> {\n  await ioHelper.notify(IO.DEFAULT_TOOLKIT_DEBUG.msg(format('Waiting for changeset %s on stack %s to finish creating...', changeSetName, stackName)));\n  const ret = await waitFor(async () => {\n    const description = await describeChangeSet(cfn, stackName, changeSetName, {\n      fetchAll,\n    });\n    // The following doesn't use a switch because tsc will not allow fall-through, UNLESS it is allows\n    // EVERYWHERE that uses this library directly or indirectly, which is undesirable.\n    if (description.Status === 'CREATE_PENDING' || description.Status === 'CREATE_IN_PROGRESS') {\n      await ioHelper.notify(IO.DEFAULT_TOOLKIT_DEBUG.msg(format('Changeset %s on stack %s is still creating', changeSetName, stackName)));\n      return undefined;\n    }\n\n    if (description.Status === ChangeSetStatus.CREATE_COMPLETE || changeSetHasNoChanges(description)) {\n      return description;\n    }\n\n    // eslint-disable-next-line max-len\n    throw new ToolkitError(\n      `Failed to create ChangeSet ${changeSetName} on ${stackName}: ${description.Status || 'NO_STATUS'}, ${description.StatusReason || 'no reason provided'}`,\n    );\n  });\n\n  if (!ret) {\n    throw new ToolkitError('Change set took too long to be created; aborting');\n  }\n\n  return ret;\n}\n\nexport type PrepareChangeSetOptions = {\n  stack: cxapi.CloudFormationStackArtifact;\n  deployments: Deployments;\n  uuid: string;\n  willExecute: boolean;\n  sdkProvider: SdkProvider;\n  parameters: { [name: string]: string | undefined };\n  resourcesToImport?: ResourcesToImport;\n}\n\nexport type CreateChangeSetOptions = {\n  cfn: ICloudFormationClient;\n  changeSetName: string;\n  willExecute: boolean;\n  exists: boolean;\n  uuid: string;\n  stack: cxapi.CloudFormationStackArtifact;\n  bodyParameter: TemplateBodyParameter;\n  parameters: { [name: string]: string | undefined };\n  resourcesToImport?: ResourceToImport[];\n  role?: string;\n};\n\n/**\n * Create a changeset for a diff operation\n */\nexport async function createDiffChangeSet(\n  ioHelper: IoHelper,\n  options: PrepareChangeSetOptions,\n): Promise<DescribeChangeSetCommandOutput | undefined> {\n  // `options.stack` has been modified to include any nested stack templates directly inline with its own template, under a special `NestedTemplate` property.\n  // Thus the parent template's Resources section contains the nested template's CDK metadata check, which uses Fn::Equals.\n  // This causes CreateChangeSet to fail with `Template Error: Fn::Equals cannot be partially collapsed`.\n  for (const resource of Object.values(options.stack.template.Resources ?? {})) {\n    if ((resource as any).Type === 'AWS::CloudFormation::Stack') {\n      await ioHelper.notify(IO.DEFAULT_TOOLKIT_DEBUG.msg('This stack contains one or more nested stacks, falling back to template-only diff...'));\n\n      return undefined;\n    }\n  }\n\n  return uploadBodyParameterAndCreateChangeSet(ioHelper, options);\n}\n\n/**\n * Returns all file entries from an AssetManifestArtifact that look like templates.\n *\n * This is used in the `uploadBodyParameterAndCreateChangeSet` function to find\n * all template asset files to build and publish.\n *\n * Returns a tuple of [AssetManifest, FileManifestEntry[]]\n */\nfunction templatesFromAssetManifestArtifact(\n  artifact: cxapi.AssetManifestArtifact,\n): [AssetManifest, FileManifestEntry[]] {\n  const assets: FileManifestEntry[] = [];\n  const fileName = artifact.file;\n  const assetManifest = AssetManifest.fromFile(fileName);\n\n  assetManifest.entries.forEach((entry) => {\n    if (entry.type === 'file') {\n      const source = (entry as FileManifestEntry).source;\n      if (source.path && source.path.endsWith('.template.json')) {\n        assets.push(entry as FileManifestEntry);\n      }\n    }\n  });\n  return [assetManifest, assets];\n}\n\nasync function uploadBodyParameterAndCreateChangeSet(\n  ioHelper: IoHelper,\n  options: PrepareChangeSetOptions,\n): Promise<DescribeChangeSetCommandOutput | undefined> {\n  try {\n    await uploadStackTemplateAssets(options.stack, options.deployments);\n    const env = await options.deployments.envs.accessStackForMutableStackOperations(options.stack);\n\n    const bodyParameter = await makeBodyParameter(\n      ioHelper,\n      options.stack,\n      env.resolvedEnvironment,\n      new AssetManifestBuilder(),\n      env.resources,\n    );\n    const cfn = env.sdk.cloudFormation();\n    const exists = (await CloudFormationStack.lookup(cfn, options.stack.stackName, false)).exists;\n\n    const executionRoleArn = await env.replacePlaceholders(options.stack.cloudFormationExecutionRoleArn);\n    await ioHelper.notify(IO.DEFAULT_TOOLKIT_INFO.msg(\n      'Hold on while we create a read-only change set to get a diff with accurate replacement information (use --no-change-set to use a less accurate but faster template-only diff)\\n',\n    ));\n\n    return await createChangeSet(ioHelper, {\n      cfn,\n      changeSetName: 'cdk-diff-change-set',\n      stack: options.stack,\n      exists,\n      uuid: options.uuid,\n      willExecute: options.willExecute,\n      bodyParameter,\n      parameters: options.parameters,\n      resourcesToImport: options.resourcesToImport,\n      role: executionRoleArn,\n    });\n  } catch (e: any) {\n    await ioHelper.notify(IO.DEFAULT_TOOLKIT_DEBUG.msg(String(e)));\n    await ioHelper.notify(IO.DEFAU