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.
- package/THIRD_PARTY_LICENSES +10 -6
- package/build-info.json +2 -2
- package/lib/api/bootstrap/bootstrap-template.yaml +1 -3
- package/lib/api/cloudformation-deployments.d.ts +1 -1
- package/lib/api/cloudformation-deployments.js +3 -3
- package/lib/api/nested-stack-helpers.d.ts +13 -13
- package/lib/api/nested-stack-helpers.js +7 -7
- package/lib/api/util/cloudformation.d.ts +3 -2
- package/lib/api/util/cloudformation.js +9 -5
- package/lib/bridge.js +25 -15
- package/lib/cdk-toolkit.d.ts +7 -0
- package/lib/cdk-toolkit.js +10 -2
- package/lib/cli.js +4 -2
- package/lib/index.js +19340 -6602
- package/lib/init-templates/app/typescript/package.json +1 -1
- package/lib/init-templates/sample-app/typescript/package.json +1 -1
- package/lib/notices.js +9 -15
- package/lib/settings.d.ts +9 -0
- package/lib/settings.js +31 -2
- package/lib/setup-sandbox.js +6 -3
- package/lib/util/validate-notification-arn.d.ts +4 -0
- package/lib/util/validate-notification-arn.js +11 -0
- package/package.json +14 -13
- package/test/cdk-toolkit.test.js +22 -2
- package/test/integ/cli/app/app.js +15 -0
- package/test/integ/cli/cli.integtest.js +7 -1
- package/test/integ/cli/sam_cdk_integ_app/src/python/Function/requirements.txt +1 -1
- package/test/integ/cli/sam_cdk_integ_app/src/python/Layer/requirements.txt +1 -1
- package/test/settings.test.js +19 -1
- package/test/util/cloudformation.test.js +31 -4
- package/test/util/validate-notification-arn.test.d.ts +1 -0
- package/test/util/validate-notification-arn.test.js +26 -0
|
@@ -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({
|
|
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.
|
|
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.
|
|
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.
|
|
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.
|
|
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.
|
|
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.
|
|
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.
|
|
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.
|
|
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.
|
|
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.
|
|
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.
|
|
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.
|
|
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.
|
|
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.
|
|
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.
|
|
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];
|
package/lib/cdk-toolkit.d.ts
CHANGED
|
@@ -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
|
/**
|