aws-cdk 2.32.0 → 2.34.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -12,15 +12,16 @@ const stack_status_1 = require("./cloudformation/stack-status");
12
12
  * repeated calls to CloudFormation).
13
13
  */
14
14
  class CloudFormationStack {
15
- constructor(cfn, stackName, stack) {
15
+ constructor(cfn, stackName, stack, retrieveProcessedTemplate = false) {
16
16
  this.cfn = cfn;
17
17
  this.stackName = stackName;
18
18
  this.stack = stack;
19
+ this.retrieveProcessedTemplate = retrieveProcessedTemplate;
19
20
  }
20
- static async lookup(cfn, stackName) {
21
+ static async lookup(cfn, stackName, retrieveProcessedTemplate = false) {
21
22
  try {
22
23
  const response = await cfn.describeStacks({ StackName: stackName }).promise();
23
- return new CloudFormationStack(cfn, stackName, response.Stacks && response.Stacks[0]);
24
+ return new CloudFormationStack(cfn, stackName, response.Stacks && response.Stacks[0], retrieveProcessedTemplate);
24
25
  }
25
26
  catch (e) {
26
27
  if (e.code === 'ValidationError' && e.message === `Stack with id ${stackName} does not exist`) {
@@ -54,7 +55,10 @@ class CloudFormationStack {
54
55
  return {};
55
56
  }
56
57
  if (this._template === undefined) {
57
- const response = await this.cfn.getTemplate({ StackName: this.stackName, TemplateStage: 'Original' }).promise();
58
+ const response = await this.cfn.getTemplate({
59
+ StackName: this.stackName,
60
+ TemplateStage: this.retrieveProcessedTemplate ? 'Processed' : 'Original',
61
+ }).promise();
58
62
  this._template = (response.TemplateBody && serialize_1.deserializeStructure(response.TemplateBody)) || {};
59
63
  }
60
64
  return this._template;
@@ -425,4 +429,4 @@ class ParameterValues {
425
429
  }
426
430
  }
427
431
  exports.ParameterValues = ParameterValues;
428
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cloudformation.js","sourceRoot":"","sources":["cloudformation.ts"],"names":[],"mappings":"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","sourcesContent":["import { SSMPARAM_NO_INVALIDATE } from '@aws-cdk/cx-api';\nimport { CloudFormation } from 'aws-sdk';\nimport { debug } from '../../logging';\nimport { deserializeStructure } from '../../serialize';\nimport { StackStatus } from './cloudformation/stack-status';\n\nexport type Template = {\n  Parameters?: Record<string, TemplateParameter>;\n  [key: string]: any;\n};\n\ninterface TemplateParameter {\n  Type: string;\n  Default?: any;\n  Description?: string;\n  [key: string]: any;\n}\n\nexport type ResourceIdentifierProperties = CloudFormation.ResourceIdentifierProperties;\nexport type ResourceIdentifierSummaries = CloudFormation.ResourceIdentifierSummaries;\nexport type ResourcesToImport = CloudFormation.ResourcesToImport;\n\n/**\n * Represents an (existing) Stack in CloudFormation\n *\n * Bundle and cache some information that we need during deployment (so we don't have to make\n * repeated calls to CloudFormation).\n */\nexport class CloudFormationStack {\n  public static async lookup(cfn: CloudFormation, stackName: string): Promise<CloudFormationStack> {\n    try {\n      const response = await cfn.describeStacks({ StackName: stackName }).promise();\n      return new CloudFormationStack(cfn, stackName, response.Stacks && response.Stacks[0]);\n    } catch (e) {\n      if (e.code === 'ValidationError' && e.message === `Stack with id ${stackName} does not exist`) {\n        return new CloudFormationStack(cfn, stackName, undefined);\n      }\n      throw e;\n    }\n  }\n\n  /**\n   * Return a copy of the given stack that does not exist\n   *\n   * It's a little silly that it needs arguments to do that, but there we go.\n   */\n  public static doesNotExist(cfn: CloudFormation, stackName: string) {\n    return new CloudFormationStack(cfn, stackName);\n  }\n\n  /**\n   * From static information (for testing)\n   */\n  public static fromStaticInformation(cfn: CloudFormation, stackName: string, stack: CloudFormation.Stack) {\n    return new CloudFormationStack(cfn, stackName, stack);\n  }\n\n  private _template: any;\n\n  protected constructor(private readonly cfn: CloudFormation, public readonly stackName: string, private readonly stack?: CloudFormation.Stack) {\n  }\n\n  /**\n   * Retrieve the stack's deployed template\n   *\n   * Cached, so will only be retrieved once. Will return an empty\n   * structure if the stack does not exist.\n   */\n  public async template(): Promise<Template> {\n    if (!this.exists) {\n      return {};\n    }\n\n    if (this._template === undefined) {\n      const response = await this.cfn.getTemplate({ StackName: this.stackName, TemplateStage: 'Original' }).promise();\n      this._template = (response.TemplateBody && deserializeStructure(response.TemplateBody)) || {};\n    }\n    return this._template;\n  }\n\n  /**\n   * Whether the stack exists\n   */\n  public get exists() {\n    return this.stack !== undefined;\n  }\n\n  /**\n   * The stack's ID\n   *\n   * Throws if the stack doesn't exist.\n   */\n  public get stackId() {\n    this.assertExists();\n    return this.stack!.StackId!;\n  }\n\n  /**\n   * The stack's current outputs\n   *\n   * Empty object if the stack doesn't exist\n   */\n  public get outputs(): Record<string, string> {\n    if (!this.exists) { return {}; }\n    const result: { [name: string]: string } = {};\n    (this.stack!.Outputs || []).forEach(output => {\n      result[output.OutputKey!] = output.OutputValue!;\n    });\n    return result;\n  }\n\n  /**\n   * The stack's status\n   *\n   * Special status NOT_FOUND if the stack does not exist.\n   */\n  public get stackStatus(): StackStatus {\n    if (!this.exists) {\n      return new StackStatus('NOT_FOUND', 'Stack not found during lookup');\n    }\n    return StackStatus.fromStackDescription(this.stack!);\n  }\n\n  /**\n   * The stack's current tags\n   *\n   * Empty list of the stack does not exist\n   */\n  public get tags(): CloudFormation.Tags {\n    return this.stack?.Tags || [];\n  }\n\n  /**\n   * Return the names of all current parameters to the stack\n   *\n   * Empty list if the stack does not exist.\n   */\n  public get parameterNames(): string[] {\n    return Object.keys(this.parameters);\n  }\n\n  /**\n   * Return the names and values of all current parameters to the stack\n   *\n   * Empty object if the stack does not exist.\n   */\n  public get parameters(): Record<string, string> {\n    if (!this.exists) { return {}; }\n    const ret: Record<string, string> = {};\n    for (const param of this.stack!.Parameters ?? []) {\n      ret[param.ParameterKey!] = param.ParameterValue!;\n    }\n    return ret;\n  }\n\n  /**\n   * Return the termination protection of the stack\n   */\n  public get terminationProtection(): boolean | undefined {\n    return this.stack?.EnableTerminationProtection;\n  }\n\n  private assertExists() {\n    if (!this.exists) {\n      throw new Error(`No stack named '${this.stackName}'`);\n    }\n  }\n}\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: CloudFormation,\n  stackName: string,\n  changeSetName: string,\n  { fetchAll }: { fetchAll: boolean },\n): Promise<CloudFormation.DescribeChangeSetOutput> {\n  const response = await cfn.describeChangeSet({ StackName: stackName, ChangeSetName: changeSetName }).promise();\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    }).promise();\n\n    // Consolidate the changes\n    if (nextPage.Changes != null) {\n      response.Changes = response.Changes != null\n        ? response.Changes.concat(nextPage.Changes)\n        : 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>(valueProvider: () => Promise<T | null | undefined>, timeout: number = 5000): 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 */\n// eslint-disable-next-line max-len\nexport async function waitForChangeSet(\n  cfn: CloudFormation,\n  stackName: string,\n  changeSetName: string,\n  { fetchAll }: { fetchAll: boolean },\n): Promise<CloudFormation.DescribeChangeSetOutput> {\n  debug('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, { fetchAll });\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      debug('Changeset %s on stack %s is still creating', changeSetName, stackName);\n      return undefined;\n    }\n\n    if (description.Status === 'CREATE_COMPLETE' || changeSetHasNoChanges(description)) {\n      return description;\n    }\n\n    // eslint-disable-next-line max-len\n    throw new Error(`Failed to create ChangeSet ${changeSetName} on ${stackName}: ${description.Status || 'NO_STATUS'}, ${description.StatusReason || 'no reason provided'}`);\n  });\n\n  if (!ret) {\n    throw new Error('Change set took too long to be created; aborting');\n  }\n\n  return ret;\n}\n\n/**\n * Return true if the given change set has no changes\n *\n * This must be determined from the status, not the 'Changes' array on the\n * object; the latter can be empty because no resources were changed, but if\n * there are changes to Outputs, the change set can still be executed.\n */\nexport function changeSetHasNoChanges(description: CloudFormation.DescribeChangeSetOutput) {\n  const noChangeErrorPrefixes = [\n    // Error message for a regular template\n    'The submitted information didn\\'t contain changes.',\n    // Error message when a Transform is involved (see #10650)\n    'No updates are to be performed.',\n  ];\n\n  return description.Status === 'FAILED'\n    && noChangeErrorPrefixes.some(p => (description.StatusReason ?? '').startsWith(p));\n}\n\n/**\n * Waits for a CloudFormation stack to stabilize in a complete/available state\n * after a delete operation is issued.\n *\n * Fails if the stack is in a FAILED state. Will not fail if the stack was\n * already deleted.\n *\n * @param cfn        a CloudFormation client\n * @param stackName      the name of the stack to wait for after a delete\n *\n * @returns     the CloudFormation description of the stabilized stack after the delete attempt\n */\nexport async function waitForStackDelete(\n  cfn: CloudFormation,\n  stackName: string): Promise<CloudFormationStack | undefined> {\n\n  const stack = await stabilizeStack(cfn, stackName);\n  if (!stack) { return undefined; }\n\n  const status = stack.stackStatus;\n  if (status.isFailure) {\n    throw new Error(`The stack named ${stackName} is in a failed state. You may need to delete it from the AWS console : ${status}`);\n  } else if (status.isDeleted) {\n    return undefined;\n  }\n  return stack;\n}\n\n/**\n * Waits for a CloudFormation stack to stabilize in a complete/available state\n * after an update/create operation is issued.\n *\n * Fails if the stack is in a FAILED state, ROLLBACK state, or DELETED state.\n *\n * @param cfn        a CloudFormation client\n * @param stackName      the name of the stack to wait for after an update\n *\n * @returns     the CloudFormation description of the stabilized stack after the update attempt\n */\nexport async function waitForStackDeploy(\n  cfn: CloudFormation,\n  stackName: string): Promise<CloudFormationStack | undefined> {\n\n  const stack = await stabilizeStack(cfn, stackName);\n  if (!stack) { return undefined; }\n\n  const status = stack.stackStatus;\n\n  if (status.isCreationFailure) {\n    throw new Error(`The stack named ${stackName} failed creation, it may need to be manually deleted from the AWS console: ${status}`);\n  } else if (!status.isDeploySuccess) {\n    throw new Error(`The stack named ${stackName} failed to deploy: ${status}`);\n  }\n\n  return stack;\n}\n\n/**\n * Wait for a stack to become stable (no longer _IN_PROGRESS), returning it\n */\nexport async function stabilizeStack(cfn: CloudFormation, stackName: string) {\n  debug('Waiting for stack %s to finish creating or updating...', stackName);\n  return waitFor(async () => {\n    const stack = await CloudFormationStack.lookup(cfn, stackName);\n    if (!stack.exists) {\n      debug('Stack %s does not exist', stackName);\n      return null;\n    }\n    const status = stack.stackStatus;\n    if (status.isInProgress) {\n      debug('Stack %s has an ongoing operation in progress and is not stable (%s)', stackName, status);\n      return undefined;\n    }\n\n    return stack;\n  });\n}\n\n/**\n * The set of (formal) parameters that have been declared in a template\n */\nexport class TemplateParameters {\n  public static fromTemplate(template: Template) {\n    return new TemplateParameters(template.Parameters || {});\n  }\n\n  constructor(private readonly params: Record<string, TemplateParameter>) {\n  }\n\n  /**\n   * Calculate stack parameters to pass from the given desired parameter values\n   *\n   * Will throw if parameters without a Default value or a Previous value are not\n   * supplied.\n   */\n  public supplyAll(updates: Record<string, string | undefined>): ParameterValues {\n    return new ParameterValues(this.params, updates);\n  }\n\n  /**\n   * From the template, the given desired values and the current values, calculate the changes to the stack parameters\n   *\n   * Will take into account parameters already set on the template (will emit\n   * 'UsePreviousValue: true' for those unless the value is changed), and will\n   * throw if parameters without a Default value or a Previous value are not\n   * supplied.\n   */\n  public updateExisting(updates: Record<string, string | undefined>, previousValues: Record<string, string>): ParameterValues {\n    return new ParameterValues(this.params, updates, previousValues);\n  }\n}\n\n/**\n * The set of parameters we're going to pass to a Stack\n */\nexport class ParameterValues {\n  public readonly values: Record<string, string> = {};\n  public readonly apiParameters: CloudFormation.Parameter[] = [];\n\n  constructor(\n    private readonly formalParams: Record<string, TemplateParameter>,\n    updates: Record<string, string | undefined>,\n    previousValues: Record<string, string> = {}) {\n\n    const missingRequired = new Array<string>();\n\n    for (const [key, formalParam] of Object.entries(this.formalParams)) {\n      // Check updates first, then use the previous value (if available), then use\n      // the default (if available).\n      //\n      // If we don't find a parameter value using any of these methods, then that's an error.\n      const updatedValue = updates[key];\n      if (updatedValue !== undefined) {\n        this.values[key] = updatedValue;\n        this.apiParameters.push({ ParameterKey: key, ParameterValue: updates[key] });\n        continue;\n      }\n\n      if (key in previousValues) {\n        this.values[key] = previousValues[key];\n        this.apiParameters.push({ ParameterKey: key, UsePreviousValue: true });\n        continue;\n      }\n\n      if (formalParam.Default !== undefined) {\n        this.values[key] = formalParam.Default;\n        continue;\n      }\n\n      // Oh no\n      missingRequired.push(key);\n    }\n\n    if (missingRequired.length > 0) {\n      throw new Error(`The following CloudFormation Parameters are missing a value: ${missingRequired.join(', ')}`);\n    }\n\n    // Just append all supplied overrides that aren't really expected (this\n    // will fail CFN but maybe people made typos that they want to be notified\n    // of)\n    const unknownParam = ([key, _]: [string, any]) => this.formalParams[key] === undefined;\n    const hasValue = ([_, value]: [string, any]) => !!value;\n    for (const [key, value] of Object.entries(updates).filter(unknownParam).filter(hasValue)) {\n      this.values[key] = value!;\n      this.apiParameters.push({ ParameterKey: key, ParameterValue: value });\n    }\n  }\n\n  /**\n   * Whether this set of parameter updates will change the actual stack values\n   */\n  public hasChanges(currentValues: Record<string, string>): ParameterChanges {\n    // If any of the parameters are SSM parameters, deploying must always happen\n    // because we can't predict what the values will be. We will allow some\n    // parameters to opt out of this check by having a magic string in their description.\n    if (Object.values(this.formalParams).some(p => p.Type.startsWith('AWS::SSM::Parameter::') && !p.Description?.includes(SSMPARAM_NO_INVALIDATE))) {\n      return 'ssm';\n    }\n\n    // Otherwise we're dirty if:\n    // - any of the existing values are removed, or changed\n    if (Object.entries(currentValues).some(([key, value]) => !(key in this.values) || value !== this.values[key])) {\n      return true;\n    }\n\n    // - any of the values we're setting are new\n    if (Object.keys(this.values).some(key => !(key in currentValues))) {\n      return true;\n    }\n\n    return false;\n  }\n}\n\nexport type ParameterChanges = boolean | 'ssm';\n"]}
432
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cloudformation.js","sourceRoot":"","sources":["cloudformation.ts"],"names":[],"mappings":"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","sourcesContent":["import { SSMPARAM_NO_INVALIDATE } from '@aws-cdk/cx-api';\nimport { CloudFormation } from 'aws-sdk';\nimport { debug } from '../../logging';\nimport { deserializeStructure } from '../../serialize';\nimport { StackStatus } from './cloudformation/stack-status';\n\nexport type Template = {\n  Parameters?: Record<string, TemplateParameter>;\n  [key: string]: any;\n};\n\ninterface TemplateParameter {\n  Type: string;\n  Default?: any;\n  Description?: string;\n  [key: string]: any;\n}\n\nexport type ResourceIdentifierProperties = CloudFormation.ResourceIdentifierProperties;\nexport type ResourceIdentifierSummaries = CloudFormation.ResourceIdentifierSummaries;\nexport type ResourcesToImport = CloudFormation.ResourcesToImport;\n\n/**\n * Represents an (existing) Stack in CloudFormation\n *\n * Bundle and cache some information that we need during deployment (so we don't have to make\n * repeated calls to CloudFormation).\n */\nexport class CloudFormationStack {\n  public static async lookup(\n    cfn: CloudFormation, stackName: string, retrieveProcessedTemplate: boolean = false,\n  ): Promise<CloudFormationStack> {\n    try {\n      const response = await cfn.describeStacks({ StackName: stackName }).promise();\n      return new CloudFormationStack(cfn, stackName, response.Stacks && response.Stacks[0], retrieveProcessedTemplate);\n    } catch (e) {\n      if (e.code === 'ValidationError' && e.message === `Stack with id ${stackName} does not exist`) {\n        return new CloudFormationStack(cfn, stackName, undefined);\n      }\n      throw e;\n    }\n  }\n\n  /**\n   * Return a copy of the given stack that does not exist\n   *\n   * It's a little silly that it needs arguments to do that, but there we go.\n   */\n  public static doesNotExist(cfn: CloudFormation, stackName: string) {\n    return new CloudFormationStack(cfn, stackName);\n  }\n\n  /**\n   * From static information (for testing)\n   */\n  public static fromStaticInformation(cfn: CloudFormation, stackName: string, stack: CloudFormation.Stack) {\n    return new CloudFormationStack(cfn, stackName, stack);\n  }\n\n  private _template: any;\n\n  protected constructor(\n    private readonly cfn: CloudFormation, public readonly stackName: string, private readonly stack?: CloudFormation.Stack,\n    private readonly retrieveProcessedTemplate: boolean = false,\n  ) {\n  }\n\n  /**\n   * Retrieve the stack's deployed template\n   *\n   * Cached, so will only be retrieved once. Will return an empty\n   * structure if the stack does not exist.\n   */\n  public async template(): Promise<Template> {\n    if (!this.exists) {\n      return {};\n    }\n\n    if (this._template === undefined) {\n      const response = await this.cfn.getTemplate({\n        StackName: this.stackName,\n        TemplateStage: this.retrieveProcessedTemplate ? 'Processed' : 'Original',\n      }).promise();\n      this._template = (response.TemplateBody && deserializeStructure(response.TemplateBody)) || {};\n    }\n    return this._template;\n  }\n\n  /**\n   * Whether the stack exists\n   */\n  public get exists() {\n    return this.stack !== undefined;\n  }\n\n  /**\n   * The stack's ID\n   *\n   * Throws if the stack doesn't exist.\n   */\n  public get stackId() {\n    this.assertExists();\n    return this.stack!.StackId!;\n  }\n\n  /**\n   * The stack's current outputs\n   *\n   * Empty object if the stack doesn't exist\n   */\n  public get outputs(): Record<string, string> {\n    if (!this.exists) { return {}; }\n    const result: { [name: string]: string } = {};\n    (this.stack!.Outputs || []).forEach(output => {\n      result[output.OutputKey!] = output.OutputValue!;\n    });\n    return result;\n  }\n\n  /**\n   * The stack's status\n   *\n   * Special status NOT_FOUND if the stack does not exist.\n   */\n  public get stackStatus(): StackStatus {\n    if (!this.exists) {\n      return new StackStatus('NOT_FOUND', 'Stack not found during lookup');\n    }\n    return StackStatus.fromStackDescription(this.stack!);\n  }\n\n  /**\n   * The stack's current tags\n   *\n   * Empty list of the stack does not exist\n   */\n  public get tags(): CloudFormation.Tags {\n    return this.stack?.Tags || [];\n  }\n\n  /**\n   * Return the names of all current parameters to the stack\n   *\n   * Empty list if the stack does not exist.\n   */\n  public get parameterNames(): string[] {\n    return Object.keys(this.parameters);\n  }\n\n  /**\n   * Return the names and values of all current parameters to the stack\n   *\n   * Empty object if the stack does not exist.\n   */\n  public get parameters(): Record<string, string> {\n    if (!this.exists) { return {}; }\n    const ret: Record<string, string> = {};\n    for (const param of this.stack!.Parameters ?? []) {\n      ret[param.ParameterKey!] = param.ParameterValue!;\n    }\n    return ret;\n  }\n\n  /**\n   * Return the termination protection of the stack\n   */\n  public get terminationProtection(): boolean | undefined {\n    return this.stack?.EnableTerminationProtection;\n  }\n\n  private assertExists() {\n    if (!this.exists) {\n      throw new Error(`No stack named '${this.stackName}'`);\n    }\n  }\n}\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: CloudFormation,\n  stackName: string,\n  changeSetName: string,\n  { fetchAll }: { fetchAll: boolean },\n): Promise<CloudFormation.DescribeChangeSetOutput> {\n  const response = await cfn.describeChangeSet({ StackName: stackName, ChangeSetName: changeSetName }).promise();\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    }).promise();\n\n    // Consolidate the changes\n    if (nextPage.Changes != null) {\n      response.Changes = response.Changes != null\n        ? response.Changes.concat(nextPage.Changes)\n        : 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>(valueProvider: () => Promise<T | null | undefined>, timeout: number = 5000): 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 */\n// eslint-disable-next-line max-len\nexport async function waitForChangeSet(\n  cfn: CloudFormation,\n  stackName: string,\n  changeSetName: string,\n  { fetchAll }: { fetchAll: boolean },\n): Promise<CloudFormation.DescribeChangeSetOutput> {\n  debug('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, { fetchAll });\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      debug('Changeset %s on stack %s is still creating', changeSetName, stackName);\n      return undefined;\n    }\n\n    if (description.Status === 'CREATE_COMPLETE' || changeSetHasNoChanges(description)) {\n      return description;\n    }\n\n    // eslint-disable-next-line max-len\n    throw new Error(`Failed to create ChangeSet ${changeSetName} on ${stackName}: ${description.Status || 'NO_STATUS'}, ${description.StatusReason || 'no reason provided'}`);\n  });\n\n  if (!ret) {\n    throw new Error('Change set took too long to be created; aborting');\n  }\n\n  return ret;\n}\n\n/**\n * Return true if the given change set has no changes\n *\n * This must be determined from the status, not the 'Changes' array on the\n * object; the latter can be empty because no resources were changed, but if\n * there are changes to Outputs, the change set can still be executed.\n */\nexport function changeSetHasNoChanges(description: CloudFormation.DescribeChangeSetOutput) {\n  const noChangeErrorPrefixes = [\n    // Error message for a regular template\n    'The submitted information didn\\'t contain changes.',\n    // Error message when a Transform is involved (see #10650)\n    'No updates are to be performed.',\n  ];\n\n  return description.Status === 'FAILED'\n    && noChangeErrorPrefixes.some(p => (description.StatusReason ?? '').startsWith(p));\n}\n\n/**\n * Waits for a CloudFormation stack to stabilize in a complete/available state\n * after a delete operation is issued.\n *\n * Fails if the stack is in a FAILED state. Will not fail if the stack was\n * already deleted.\n *\n * @param cfn        a CloudFormation client\n * @param stackName      the name of the stack to wait for after a delete\n *\n * @returns     the CloudFormation description of the stabilized stack after the delete attempt\n */\nexport async function waitForStackDelete(\n  cfn: CloudFormation,\n  stackName: string): Promise<CloudFormationStack | undefined> {\n\n  const stack = await stabilizeStack(cfn, stackName);\n  if (!stack) { return undefined; }\n\n  const status = stack.stackStatus;\n  if (status.isFailure) {\n    throw new Error(`The stack named ${stackName} is in a failed state. You may need to delete it from the AWS console : ${status}`);\n  } else if (status.isDeleted) {\n    return undefined;\n  }\n  return stack;\n}\n\n/**\n * Waits for a CloudFormation stack to stabilize in a complete/available state\n * after an update/create operation is issued.\n *\n * Fails if the stack is in a FAILED state, ROLLBACK state, or DELETED state.\n *\n * @param cfn        a CloudFormation client\n * @param stackName      the name of the stack to wait for after an update\n *\n * @returns     the CloudFormation description of the stabilized stack after the update attempt\n */\nexport async function waitForStackDeploy(\n  cfn: CloudFormation,\n  stackName: string): Promise<CloudFormationStack | undefined> {\n\n  const stack = await stabilizeStack(cfn, stackName);\n  if (!stack) { return undefined; }\n\n  const status = stack.stackStatus;\n\n  if (status.isCreationFailure) {\n    throw new Error(`The stack named ${stackName} failed creation, it may need to be manually deleted from the AWS console: ${status}`);\n  } else if (!status.isDeploySuccess) {\n    throw new Error(`The stack named ${stackName} failed to deploy: ${status}`);\n  }\n\n  return stack;\n}\n\n/**\n * Wait for a stack to become stable (no longer _IN_PROGRESS), returning it\n */\nexport async function stabilizeStack(cfn: CloudFormation, stackName: string) {\n  debug('Waiting for stack %s to finish creating or updating...', stackName);\n  return waitFor(async () => {\n    const stack = await CloudFormationStack.lookup(cfn, stackName);\n    if (!stack.exists) {\n      debug('Stack %s does not exist', stackName);\n      return null;\n    }\n    const status = stack.stackStatus;\n    if (status.isInProgress) {\n      debug('Stack %s has an ongoing operation in progress and is not stable (%s)', stackName, status);\n      return undefined;\n    }\n\n    return stack;\n  });\n}\n\n/**\n * The set of (formal) parameters that have been declared in a template\n */\nexport class TemplateParameters {\n  public static fromTemplate(template: Template) {\n    return new TemplateParameters(template.Parameters || {});\n  }\n\n  constructor(private readonly params: Record<string, TemplateParameter>) {\n  }\n\n  /**\n   * Calculate stack parameters to pass from the given desired parameter values\n   *\n   * Will throw if parameters without a Default value or a Previous value are not\n   * supplied.\n   */\n  public supplyAll(updates: Record<string, string | undefined>): ParameterValues {\n    return new ParameterValues(this.params, updates);\n  }\n\n  /**\n   * From the template, the given desired values and the current values, calculate the changes to the stack parameters\n   *\n   * Will take into account parameters already set on the template (will emit\n   * 'UsePreviousValue: true' for those unless the value is changed), and will\n   * throw if parameters without a Default value or a Previous value are not\n   * supplied.\n   */\n  public updateExisting(updates: Record<string, string | undefined>, previousValues: Record<string, string>): ParameterValues {\n    return new ParameterValues(this.params, updates, previousValues);\n  }\n}\n\n/**\n * The set of parameters we're going to pass to a Stack\n */\nexport class ParameterValues {\n  public readonly values: Record<string, string> = {};\n  public readonly apiParameters: CloudFormation.Parameter[] = [];\n\n  constructor(\n    private readonly formalParams: Record<string, TemplateParameter>,\n    updates: Record<string, string | undefined>,\n    previousValues: Record<string, string> = {}) {\n\n    const missingRequired = new Array<string>();\n\n    for (const [key, formalParam] of Object.entries(this.formalParams)) {\n      // Check updates first, then use the previous value (if available), then use\n      // the default (if available).\n      //\n      // If we don't find a parameter value using any of these methods, then that's an error.\n      const updatedValue = updates[key];\n      if (updatedValue !== undefined) {\n        this.values[key] = updatedValue;\n        this.apiParameters.push({ ParameterKey: key, ParameterValue: updates[key] });\n        continue;\n      }\n\n      if (key in previousValues) {\n        this.values[key] = previousValues[key];\n        this.apiParameters.push({ ParameterKey: key, UsePreviousValue: true });\n        continue;\n      }\n\n      if (formalParam.Default !== undefined) {\n        this.values[key] = formalParam.Default;\n        continue;\n      }\n\n      // Oh no\n      missingRequired.push(key);\n    }\n\n    if (missingRequired.length > 0) {\n      throw new Error(`The following CloudFormation Parameters are missing a value: ${missingRequired.join(', ')}`);\n    }\n\n    // Just append all supplied overrides that aren't really expected (this\n    // will fail CFN but maybe people made typos that they want to be notified\n    // of)\n    const unknownParam = ([key, _]: [string, any]) => this.formalParams[key] === undefined;\n    const hasValue = ([_, value]: [string, any]) => !!value;\n    for (const [key, value] of Object.entries(updates).filter(unknownParam).filter(hasValue)) {\n      this.values[key] = value!;\n      this.apiParameters.push({ ParameterKey: key, ParameterValue: value });\n    }\n  }\n\n  /**\n   * Whether this set of parameter updates will change the actual stack values\n   */\n  public hasChanges(currentValues: Record<string, string>): ParameterChanges {\n    // If any of the parameters are SSM parameters, deploying must always happen\n    // because we can't predict what the values will be. We will allow some\n    // parameters to opt out of this check by having a magic string in their description.\n    if (Object.values(this.formalParams).some(p => p.Type.startsWith('AWS::SSM::Parameter::') && !p.Description?.includes(SSMPARAM_NO_INVALIDATE))) {\n      return 'ssm';\n    }\n\n    // Otherwise we're dirty if:\n    // - any of the existing values are removed, or changed\n    if (Object.entries(currentValues).some(([key, value]) => !(key in this.values) || value !== this.values[key])) {\n      return true;\n    }\n\n    // - any of the values we're setting are new\n    if (Object.keys(this.values).some(key => !(key in currentValues))) {\n      return true;\n    }\n\n    return false;\n  }\n}\n\nexport type ParameterChanges = boolean | 'ssm';\n"]}
package/lib/bridge.js CHANGED
@@ -102,6 +102,8 @@ const thisErrorCaptureStackTrace = Error.captureStackTrace;
102
102
 
103
103
  const thisSymbolToString = Symbol.prototype.toString;
104
104
  const thisSymbolToStringTag = Symbol.toStringTag;
105
+ const thisSymbolIterator = Symbol.iterator;
106
+ const thisSymbolNodeJSUtilInspectCustom = Symbol.for('nodejs.util.inspect.custom');
105
107
 
106
108
  /**
107
109
  * VMError.
@@ -348,7 +350,11 @@ function createBridge(otherInit, registerProxy) {
348
350
  constructor(object) {
349
351
  // Note: object@other(unsafe) throws@this(unsafe)
350
352
  super();
351
- this.object = object;
353
+ this.objectWrapper = () => object;
354
+ }
355
+
356
+ getObject() {
357
+ return this.objectWrapper();
352
358
  }
353
359
 
354
360
  getFactory() {
@@ -408,7 +414,7 @@ function createBridge(otherInit, registerProxy) {
408
414
 
409
415
  get(target, key, receiver) {
410
416
  // Note: target@this(unsafe) key@prim receiver@this(unsafe) throws@this(unsafe)
411
- const object = this.object; // @other(unsafe)
417
+ const object = this.getObject(); // @other(unsafe)
412
418
  switch (key) {
413
419
  case 'constructor': {
414
420
  const desc = otherSafeGetOwnPropertyDescriptor(object, key);
@@ -447,7 +453,7 @@ function createBridge(otherInit, registerProxy) {
447
453
 
448
454
  set(target, key, value, receiver) {
449
455
  // Note: target@this(unsafe) key@prim value@this(unsafe) receiver@this(unsafe) throws@this(unsafe)
450
- const object = this.object; // @other(unsafe)
456
+ const object = this.getObject(); // @other(unsafe)
451
457
  if (key === '__proto__' && !thisOtherHasOwnProperty(object, key)) {
452
458
  return this.setPrototypeOf(target, value);
453
459
  }
@@ -471,7 +477,7 @@ function createBridge(otherInit, registerProxy) {
471
477
 
472
478
  apply(target, context, args) {
473
479
  // Note: target@this(unsafe) context@this(unsafe) args@this(safe-array) throws@this(unsafe)
474
- const object = this.object; // @other(unsafe)
480
+ const object = this.getObject(); // @other(unsafe)
475
481
  let ret; // @other(unsafe)
476
482
  try {
477
483
  context = otherFromThis(context);
@@ -485,7 +491,7 @@ function createBridge(otherInit, registerProxy) {
485
491
 
486
492
  construct(target, args, newTarget) {
487
493
  // Note: target@this(unsafe) args@this(safe-array) newTarget@this(unsafe) throws@this(unsafe)
488
- const object = this.object; // @other(unsafe)
494
+ const object = this.getObject(); // @other(unsafe)
489
495
  let ret; // @other(unsafe)
490
496
  try {
491
497
  args = otherFromThisArguments(args);
@@ -498,14 +504,14 @@ function createBridge(otherInit, registerProxy) {
498
504
 
499
505
  getOwnPropertyDescriptorDesc(target, prop, desc) {
500
506
  // Note: target@this(unsafe) prop@prim desc@other{safe} throws@this(unsafe)
501
- const object = this.object; // @other(unsafe)
507
+ const object = this.getObject(); // @other(unsafe)
502
508
  if (desc && typeof object === 'function' && (prop === 'arguments' || prop === 'caller' || prop === 'callee')) desc.value = null;
503
509
  return desc;
504
510
  }
505
511
 
506
512
  getOwnPropertyDescriptor(target, prop) {
507
513
  // Note: target@this(unsafe) prop@prim throws@this(unsafe)
508
- const object = this.object; // @other(unsafe)
514
+ const object = this.getObject(); // @other(unsafe)
509
515
  let desc; // @other(safe)
510
516
  try {
511
517
  desc = otherSafeGetOwnPropertyDescriptor(object, prop);
@@ -551,7 +557,7 @@ function createBridge(otherInit, registerProxy) {
551
557
 
552
558
  defineProperty(target, prop, desc) {
553
559
  // Note: target@this(unsafe) prop@prim desc@this(unsafe) throws@this(unsafe)
554
- const object = this.object; // @other(unsafe)
560
+ const object = this.getObject(); // @other(unsafe)
555
561
  if (!thisReflectSetPrototypeOf(desc, null)) throw thisUnexpected();
556
562
 
557
563
  desc = this.definePropertyDesc(target, prop, desc);
@@ -604,7 +610,7 @@ function createBridge(otherInit, registerProxy) {
604
610
 
605
611
  deleteProperty(target, prop) {
606
612
  // Note: target@this(unsafe) prop@prim throws@this(unsafe)
607
- const object = this.object; // @other(unsafe)
613
+ const object = this.getObject(); // @other(unsafe)
608
614
  try {
609
615
  return otherReflectDeleteProperty(object, prop) === true;
610
616
  } catch (e) { // @other(unsafe)
@@ -614,7 +620,7 @@ function createBridge(otherInit, registerProxy) {
614
620
 
615
621
  has(target, key) {
616
622
  // Note: target@this(unsafe) key@prim throws@this(unsafe)
617
- const object = this.object; // @other(unsafe)
623
+ const object = this.getObject(); // @other(unsafe)
618
624
  try {
619
625
  return otherReflectHas(object, key) === true;
620
626
  } catch (e) { // @other(unsafe)
@@ -624,7 +630,7 @@ function createBridge(otherInit, registerProxy) {
624
630
 
625
631
  isExtensible(target) {
626
632
  // Note: target@this(unsafe) throws@this(unsafe)
627
- const object = this.object; // @other(unsafe)
633
+ const object = this.getObject(); // @other(unsafe)
628
634
  try {
629
635
  if (otherReflectIsExtensible(object)) return true;
630
636
  } catch (e) { // @other(unsafe)
@@ -638,7 +644,7 @@ function createBridge(otherInit, registerProxy) {
638
644
 
639
645
  ownKeys(target) {
640
646
  // Note: target@this(unsafe) throws@this(unsafe)
641
- const object = this.object; // @other(unsafe)
647
+ const object = this.getObject(); // @other(unsafe)
642
648
  let res; // @other(unsafe)
643
649
  try {
644
650
  res = otherReflectOwnKeys(object);
@@ -650,7 +656,7 @@ function createBridge(otherInit, registerProxy) {
650
656
 
651
657
  preventExtensions(target) {
652
658
  // Note: target@this(unsafe) throws@this(unsafe)
653
- const object = this.object; // @other(unsafe)
659
+ const object = this.getObject(); // @other(unsafe)
654
660
  try {
655
661
  if (!otherReflectPreventExtensions(object)) return false;
656
662
  } catch (e) { // @other(unsafe)
@@ -664,7 +670,7 @@ function createBridge(otherInit, registerProxy) {
664
670
 
665
671
  enumerate(target) {
666
672
  // Note: target@this(unsafe) throws@this(unsafe)
667
- const object = this.object; // @other(unsafe)
673
+ const object = this.getObject(); // @other(unsafe)
668
674
  let res; // @other(unsafe)
669
675
  try {
670
676
  res = otherReflectEnumerate(object);
@@ -676,6 +682,10 @@ function createBridge(otherInit, registerProxy) {
676
682
 
677
683
  }
678
684
 
685
+ BaseHandler.prototype[thisSymbolNodeJSUtilInspectCustom] = undefined;
686
+ BaseHandler.prototype[thisSymbolToStringTag] = 'VM2 Wrapper';
687
+ BaseHandler.prototype[thisSymbolIterator] = undefined;
688
+
679
689
  function defaultFactory(object) {
680
690
  // Note: other@other(unsafe) returns@this(unsafe) throws@this(unsafe)
681
691
  return new BaseHandler(object);
@@ -773,7 +783,7 @@ function createBridge(otherInit, registerProxy) {
773
783
 
774
784
  get(target, key, receiver) {
775
785
  // Note: target@this(unsafe) key@prim receiver@this(unsafe) throws@this(unsafe)
776
- const object = this.object; // @other(unsafe)
786
+ const object = this.getObject(); // @other(unsafe)
777
787
  const mock = this.mock;
778
788
  if (thisReflectApply(thisObjectHasOwnProperty, mock, key) && !thisOtherHasOwnProperty(object, key)) {
779
789
  return mock[key];
@@ -150,6 +150,13 @@ export interface DiffOptions {
150
150
  * @default false
151
151
  */
152
152
  securityOnly?: boolean;
153
+ /**
154
+ * Whether to run the diff against the template after the CloudFormation Transforms inside it have been executed
155
+ * (as opposed to the original template, the default, which contains the unprocessed Transforms).
156
+ *
157
+ * @default false
158
+ */
159
+ compareAgainstProcessedTemplate?: boolean;
153
160
  }
154
161
  interface CfnDeployOptions {
155
162
  /**