UNPKG

aws-cdk

Version:

AWS CDK CLI, the command line tool for CDK apps

158 lines 16.2 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.CloudFormationStack = void 0; const api_1 = require("../../../../@aws-cdk/tmp-toolkit-helpers/src/api"); const util_1 = require("../../util"); const stack_events_1 = require("../stack-events"); /** * Represents an (existing) Stack in CloudFormation * * Bundle and cache some information that we need during deployment (so we don't have to make * repeated calls to CloudFormation). */ class CloudFormationStack { static async lookup(cfn, stackName, retrieveProcessedTemplate = false) { try { const response = await cfn.describeStacks({ StackName: stackName }); return new CloudFormationStack(cfn, stackName, response.Stacks && response.Stacks[0], retrieveProcessedTemplate); } catch (e) { if (e.name === 'ValidationError' && (0, util_1.formatErrorMessage)(e) === `Stack with id ${stackName} does not exist`) { return new CloudFormationStack(cfn, stackName, undefined); } throw e; } } /** * Return a copy of the given stack that does not exist * * It's a little silly that it needs arguments to do that, but there we go. */ static doesNotExist(cfn, stackName) { return new CloudFormationStack(cfn, stackName); } /** * From static information (for testing) */ static fromStaticInformation(cfn, stackName, stack) { return new CloudFormationStack(cfn, stackName, stack); } constructor(cfn, stackName, stack, retrieveProcessedTemplate = false) { this.cfn = cfn; this.stackName = stackName; this.stack = stack; this.retrieveProcessedTemplate = retrieveProcessedTemplate; } /** * Retrieve the stack's deployed template * * Cached, so will only be retrieved once. Will return an empty * structure if the stack does not exist. */ async template() { if (!this.exists) { return {}; } if (this._template === undefined) { const response = await this.cfn.getTemplate({ StackName: this.stackName, TemplateStage: this.retrieveProcessedTemplate ? 'Processed' : 'Original', }); this._template = (response.TemplateBody && (0, util_1.deserializeStructure)(response.TemplateBody)) || {}; } return this._template; } /** * Whether the stack exists */ get exists() { return this.stack !== undefined; } /** * The stack's ID * * Throws if the stack doesn't exist. */ get stackId() { this.assertExists(); return this.stack.StackId; } /** * The stack's current outputs * * Empty object if the stack doesn't exist */ get outputs() { if (!this.exists) { return {}; } const result = {}; (this.stack.Outputs || []).forEach((output) => { result[output.OutputKey] = output.OutputValue; }); return result; } /** * The stack's status * * Special status NOT_FOUND if the stack does not exist. */ get stackStatus() { if (!this.exists) { return new stack_events_1.StackStatus('NOT_FOUND', 'Stack not found during lookup'); } return stack_events_1.StackStatus.fromStackDescription(this.stack); } /** * The stack's current tags * * Empty list if the stack does not exist */ get tags() { return this.stack?.Tags || []; } /** * SNS Topic ARNs that will receive stack events. * * Empty list if the stack does not exist */ get notificationArns() { return this.stack?.NotificationARNs ?? []; } /** * Return the names of all current parameters to the stack * * Empty list if the stack does not exist. */ get parameterNames() { return Object.keys(this.parameters); } /** * Return the names and values of all current parameters to the stack * * Empty object if the stack does not exist. */ get parameters() { if (!this.exists) { return {}; } const ret = {}; for (const param of this.stack.Parameters ?? []) { ret[param.ParameterKey] = param.ResolvedValue ?? param.ParameterValue; } return ret; } /** * Return the termination protection of the stack */ get terminationProtection() { return this.stack?.EnableTerminationProtection; } assertExists() { if (!this.exists) { throw new api_1.ToolkitError(`No stack named '${this.stackName}'`); } } } exports.CloudFormationStack = CloudFormationStack; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"stack-helpers.js","sourceRoot":"","sources":["stack-helpers.ts"],"names":[],"mappings":"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","sourcesContent":["import type { Stack, Tag } from '@aws-sdk/client-cloudformation';\nimport type { Template } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api';\nimport { ToolkitError } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api';\nimport { formatErrorMessage, deserializeStructure } from '../../util';\nimport type { ICloudFormationClient } from '../aws-auth';\nimport { StackStatus } from '../stack-events';\n\nexport { Template, TemplateParameter } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api';\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: ICloudFormationClient,\n    stackName: string,\n    retrieveProcessedTemplate: boolean = false,\n  ): Promise<CloudFormationStack> {\n    try {\n      const response = await cfn.describeStacks({ StackName: stackName });\n      return new CloudFormationStack(cfn, stackName, response.Stacks && response.Stacks[0], retrieveProcessedTemplate);\n    } catch (e: any) {\n      if (e.name === 'ValidationError' && formatErrorMessage(e) === `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: ICloudFormationClient, stackName: string) {\n    return new CloudFormationStack(cfn, stackName);\n  }\n\n  /**\n   * From static information (for testing)\n   */\n  public static fromStaticInformation(cfn: ICloudFormationClient, stackName: string, stack: Stack) {\n    return new CloudFormationStack(cfn, stackName, stack);\n  }\n\n  private _template: any;\n\n  protected constructor(\n    private readonly cfn: ICloudFormationClient,\n    public readonly stackName: string,\n    private readonly stack?: 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      });\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) {\n      return {};\n    }\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 if the stack does not exist\n   */\n  public get tags(): Tag[] {\n    return this.stack?.Tags || [];\n  }\n\n  /**\n   * SNS Topic ARNs that will receive stack events.\n   *\n   * Empty list if the stack does not exist\n   */\n  public get notificationArns(): string[] {\n    return this.stack?.NotificationARNs ?? [];\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) {\n      return {};\n    }\n    const ret: Record<string, string> = {};\n    for (const param of this.stack!.Parameters ?? []) {\n      ret[param.ParameterKey!] = param.ResolvedValue ?? 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 ToolkitError(`No stack named '${this.stackName}'`);\n    }\n  }\n}\n"]}