UNPKG

@aws-cdk/cx-api

Version:

Cloud executable protocol

223 lines 25.3 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.CloudFormationStackArtifact = void 0; const jsiiDeprecationWarnings = require("../../.warnings.jsii.js"); const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti"); const fs = require("fs"); const path = require("path"); const cxschema = require("@aws-cdk/cloud-assembly-schema"); const cloud_artifact_1 = require("../cloud-artifact"); const environment_1 = require("../environment"); const error_1 = require("../private/error"); const CLOUDFORMATION_STACK_ARTIFACT_SYM = Symbol.for('@aws-cdk/cx-api.CloudFormationStackArtifact'); class CloudFormationStackArtifact extends cloud_artifact_1.CloudArtifact { static [JSII_RTTI_SYMBOL_1] = { fqn: "@aws-cdk/cx-api.CloudFormationStackArtifact", version: "2.233.0" }; /** * Checks if `art` is an instance of this class. * * Use this method instead of `instanceof` to properly detect `CloudFormationStackArtifact` * instances, even when the construct library is symlinked. * * Explanation: in JavaScript, multiple copies of the `cx-api` library on * disk are seen as independent, completely different libraries. As a * consequence, the class `CloudFormationStackArtifact` in each copy of the `cx-api` library * is seen as a different class, and an instance of one class will not test as * `instanceof` the other class. `npm install` will not create installations * like this, but users may manually symlink construct libraries together or * use a monorepo tool: in those cases, multiple copies of the `cx-api` * library can be accidentally installed, and `instanceof` will behave * unpredictably. It is safest to avoid using `instanceof`, and using * this type-testing method instead. */ static isCloudFormationStackArtifact(art) { return art && typeof art === 'object' && art[CLOUDFORMATION_STACK_ARTIFACT_SYM]; } /** * The file name of the template. */ templateFile; /** * The original name as defined in the CDK app. */ originalName; /** * Any assets associated with this stack. */ assets; /** * CloudFormation parameters to pass to the stack. */ parameters; /** * CloudFormation tags to pass to the stack. */ tags; /** * SNS Topics that will receive stack events. */ notificationArns; /** * The physical name of this stack. */ stackName; /** * A string that represents this stack. Should only be used in user * interfaces. If the stackName has not been set explicitly, or has been set * to artifactId, it will return the hierarchicalId of the stack. Otherwise, * it will return something like "<hierarchicalId> (<stackName>)" */ displayName; /** * The physical name of this stack. * @deprecated renamed to `stackName` */ name; /** * The environment into which to deploy this artifact. */ environment; /** * The role that needs to be assumed to deploy the stack * * @default - No role is assumed (current credentials are used) */ assumeRoleArn; /** * External ID to use when assuming role for cloudformation deployments * * @default - No external ID */ assumeRoleExternalId; /** * Additional options to pass to STS when assuming the role for cloudformation deployments. * * - `RoleArn` should not be used. Use the dedicated `assumeRoleArn` property instead. * - `ExternalId` should not be used. Use the dedicated `assumeRoleExternalId` instead. * - `TransitiveTagKeys` defaults to use all keys (if any) specified in `Tags`. E.g, all tags are transitive by default. * * @see https://docs.aws.amazon.com/AWSJavaScriptSDK/latest/AWS/STS.html#assumeRole-property * @default - No additional options. */ assumeRoleAdditionalOptions; /** * The role that is passed to CloudFormation to execute the change set * * @default - No role is passed (currently assumed role/credentials are used) */ cloudFormationExecutionRoleArn; /** * The role to use to look up values from the target AWS account * * @default - No role is assumed (current credentials are used) */ lookupRole; /** * If the stack template has already been included in the asset manifest, its asset URL * * @default - Not uploaded yet, upload just before deploying */ stackTemplateAssetObjectUrl; /** * Version of bootstrap stack required to deploy this stack * * @default - No bootstrap stack required */ requiresBootstrapStackVersion; /** * Name of SSM parameter with bootstrap stack version * * @default - Discover SSM parameter by reading stack */ bootstrapStackVersionSsmParameter; /** * Whether termination protection is enabled for this stack. */ terminationProtection; /** * Whether this stack should be validated by the CLI after synthesis * * @default - false */ validateOnSynth; _template; constructor(assembly, artifactId, artifact) { super(assembly, artifactId, artifact); try { jsiiDeprecationWarnings._aws_cdk_cx_api_CloudAssembly(assembly); } catch (error) { if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") { Error.captureStackTrace(error, CloudFormationStackArtifact); } throw error; } const properties = (this.manifest.properties || {}); if (!properties.templateFile) { throw new error_1.CloudAssemblyError('Invalid CloudFormation stack artifact. Missing "templateFile" property in cloud assembly manifest'); } if (!artifact.environment) { throw new error_1.CloudAssemblyError('Invalid CloudFormation stack artifact. Missing environment'); } this.environment = environment_1.EnvironmentUtils.parse(artifact.environment); this.templateFile = properties.templateFile; this.parameters = properties.parameters ?? {}; // We get the tags from 'properties' if available (cloud assembly format >= 6.0.0), otherwise // from the stack metadata this.tags = properties.tags ?? this.tagsFromMetadata(); this.notificationArns = properties.notificationArns; this.assumeRoleArn = properties.assumeRoleArn; this.assumeRoleExternalId = properties.assumeRoleExternalId; this.assumeRoleAdditionalOptions = properties.assumeRoleAdditionalOptions; this.cloudFormationExecutionRoleArn = properties.cloudFormationExecutionRoleArn; this.stackTemplateAssetObjectUrl = properties.stackTemplateAssetObjectUrl; this.requiresBootstrapStackVersion = properties.requiresBootstrapStackVersion; this.bootstrapStackVersionSsmParameter = properties.bootstrapStackVersionSsmParameter; this.terminationProtection = properties.terminationProtection; this.validateOnSynth = properties.validateOnSynth; this.lookupRole = properties.lookupRole; this.stackName = properties.stackName || artifactId; this.assets = this.findMetadataByType(cxschema.ArtifactMetadataEntryType.ASSET).map(e => e.data); this.displayName = this.stackName === artifactId ? this.hierarchicalId : `${this.hierarchicalId} (${this.stackName})`; this.name = this.stackName; // backwards compat this.originalName = this.stackName; } /** * Full path to the template file */ get templateFullPath() { return path.join(this.assembly.directory, this.templateFile); } /** * The CloudFormation template for this stack. */ get template() { if (this._template === undefined) { this._template = JSON.parse(fs.readFileSync(this.templateFullPath, 'utf-8')); } return this._template; } tagsFromMetadata() { const ret = {}; for (const metadataEntry of this.findMetadataByType(cxschema.ArtifactMetadataEntryType.STACK_TAGS)) { for (const tag of (metadataEntry.data ?? [])) { ret[tag.key] = tag.value; } } return ret; } } exports.CloudFormationStackArtifact = CloudFormationStackArtifact; /** * Mark all instances of 'CloudFormationStackArtifact' * * Why not put this in the constructor? Because this is a class property, * not an instance property. It applies to all instances of the class. */ Object.defineProperty(CloudFormationStackArtifact.prototype, CLOUDFORMATION_STACK_ARTIFACT_SYM, { value: true, enumerable: false, writable: false, }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cloudformation-artifact.js","sourceRoot":"","sources":["cloudformation-artifact.ts"],"names":[],"mappings":"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","sourcesContent":["import * as fs from 'fs';\nimport * as path from 'path';\nimport * as cxschema from '@aws-cdk/cloud-assembly-schema';\nimport { CloudArtifact } from '../cloud-artifact';\nimport type { CloudAssembly } from '../cloud-assembly';\nimport { Environment, EnvironmentUtils } from '../environment';\nimport { CloudAssemblyError } from '../private/error';\nconst CLOUDFORMATION_STACK_ARTIFACT_SYM = Symbol.for('@aws-cdk/cx-api.CloudFormationStackArtifact');\n\nexport class CloudFormationStackArtifact extends CloudArtifact {\n  /**\n   * Checks if `art` is an instance of this class.\n   *\n   * Use this method instead of `instanceof` to properly detect `CloudFormationStackArtifact`\n   * instances, even when the construct library is symlinked.\n   *\n   * Explanation: in JavaScript, multiple copies of the `cx-api` library on\n   * disk are seen as independent, completely different libraries. As a\n   * consequence, the class `CloudFormationStackArtifact` in each copy of the `cx-api` library\n   * is seen as a different class, and an instance of one class will not test as\n   * `instanceof` the other class. `npm install` will not create installations\n   * like this, but users may manually symlink construct libraries together or\n   * use a monorepo tool: in those cases, multiple copies of the `cx-api`\n   * library can be accidentally installed, and `instanceof` will behave\n   * unpredictably. It is safest to avoid using `instanceof`, and using\n   * this type-testing method instead.\n   */\n  public static isCloudFormationStackArtifact(art: any): art is CloudFormationStackArtifact {\n    return art && typeof art === 'object' && art[CLOUDFORMATION_STACK_ARTIFACT_SYM];\n  }\n\n  /**\n   * The file name of the template.\n   */\n  public readonly templateFile: string;\n\n  /**\n   * The original name as defined in the CDK app.\n   */\n  public readonly originalName: string;\n\n  /**\n   * Any assets associated with this stack.\n   */\n  public readonly assets: cxschema.AssetMetadataEntry[];\n\n  /**\n   * CloudFormation parameters to pass to the stack.\n   */\n  public readonly parameters: { [id: string]: string };\n\n  /**\n   * CloudFormation tags to pass to the stack.\n   */\n  public readonly tags: { [id: string]: string };\n\n  /**\n   * SNS Topics that will receive stack events.\n   */\n  public readonly notificationArns?: string[];\n\n  /**\n   * The physical name of this stack.\n   */\n  public readonly stackName: string;\n\n  /**\n   * A string that represents this stack. Should only be used in user\n   * interfaces. If the stackName has not been set explicitly, or has been set\n   * to artifactId, it will return the hierarchicalId of the stack. Otherwise,\n   * it will return something like \"<hierarchicalId> (<stackName>)\"\n   */\n  public readonly displayName: string;\n\n  /**\n   * The physical name of this stack.\n   * @deprecated renamed to `stackName`\n   */\n  public readonly name: string;\n\n  /**\n   * The environment into which to deploy this artifact.\n   */\n  public readonly environment: Environment;\n\n  /**\n   * The role that needs to be assumed to deploy the stack\n   *\n   * @default - No role is assumed (current credentials are used)\n   */\n  public readonly assumeRoleArn?: string;\n\n  /**\n   * External ID to use when assuming role for cloudformation deployments\n   *\n   * @default - No external ID\n   */\n  readonly assumeRoleExternalId?: string;\n\n  /**\n   * Additional options to pass to STS when assuming the role for cloudformation deployments.\n   *\n   * - `RoleArn` should not be used. Use the dedicated `assumeRoleArn` property instead.\n   * - `ExternalId` should not be used. Use the dedicated `assumeRoleExternalId` instead.\n   * - `TransitiveTagKeys` defaults to use all keys (if any) specified in `Tags`. E.g, all tags are transitive by default.\n   *\n   * @see https://docs.aws.amazon.com/AWSJavaScriptSDK/latest/AWS/STS.html#assumeRole-property\n   * @default - No additional options.\n   */\n  readonly assumeRoleAdditionalOptions?: { [key: string]: any };\n\n  /**\n   * The role that is passed to CloudFormation to execute the change set\n   *\n   * @default - No role is passed (currently assumed role/credentials are used)\n   */\n  public readonly cloudFormationExecutionRoleArn?: string;\n\n  /**\n   * The role to use to look up values from the target AWS account\n   *\n   * @default - No role is assumed (current credentials are used)\n   */\n  public readonly lookupRole?: cxschema.BootstrapRole;\n\n  /**\n   * If the stack template has already been included in the asset manifest, its asset URL\n   *\n   * @default - Not uploaded yet, upload just before deploying\n   */\n  public readonly stackTemplateAssetObjectUrl?: string;\n\n  /**\n   * Version of bootstrap stack required to deploy this stack\n   *\n   * @default - No bootstrap stack required\n   */\n  public readonly requiresBootstrapStackVersion?: number;\n\n  /**\n   * Name of SSM parameter with bootstrap stack version\n   *\n   * @default - Discover SSM parameter by reading stack\n   */\n  public readonly bootstrapStackVersionSsmParameter?: string;\n\n  /**\n   * Whether termination protection is enabled for this stack.\n   */\n  public readonly terminationProtection?: boolean;\n\n  /**\n   * Whether this stack should be validated by the CLI after synthesis\n   *\n   * @default - false\n   */\n  public readonly validateOnSynth?: boolean;\n\n  private _template: any | undefined;\n\n  constructor(assembly: CloudAssembly, artifactId: string, artifact: cxschema.ArtifactManifest) {\n    super(assembly, artifactId, artifact);\n\n    const properties = (this.manifest.properties || {}) as cxschema.AwsCloudFormationStackProperties;\n    if (!properties.templateFile) {\n      throw new CloudAssemblyError('Invalid CloudFormation stack artifact. Missing \"templateFile\" property in cloud assembly manifest');\n    }\n    if (!artifact.environment) {\n      throw new CloudAssemblyError('Invalid CloudFormation stack artifact. Missing environment');\n    }\n    this.environment = EnvironmentUtils.parse(artifact.environment);\n    this.templateFile = properties.templateFile;\n    this.parameters = properties.parameters ?? {};\n\n    // We get the tags from 'properties' if available (cloud assembly format >= 6.0.0), otherwise\n    // from the stack metadata\n    this.tags = properties.tags ?? this.tagsFromMetadata();\n    this.notificationArns = properties.notificationArns;\n    this.assumeRoleArn = properties.assumeRoleArn;\n    this.assumeRoleExternalId = properties.assumeRoleExternalId;\n    this.assumeRoleAdditionalOptions = properties.assumeRoleAdditionalOptions;\n    this.cloudFormationExecutionRoleArn = properties.cloudFormationExecutionRoleArn;\n    this.stackTemplateAssetObjectUrl = properties.stackTemplateAssetObjectUrl;\n    this.requiresBootstrapStackVersion = properties.requiresBootstrapStackVersion;\n    this.bootstrapStackVersionSsmParameter = properties.bootstrapStackVersionSsmParameter;\n    this.terminationProtection = properties.terminationProtection;\n    this.validateOnSynth = properties.validateOnSynth;\n    this.lookupRole = properties.lookupRole;\n\n    this.stackName = properties.stackName || artifactId;\n    this.assets = this.findMetadataByType(cxschema.ArtifactMetadataEntryType.ASSET).map(e => e.data as cxschema.AssetMetadataEntry);\n\n    this.displayName = this.stackName === artifactId\n      ? this.hierarchicalId\n      : `${this.hierarchicalId} (${this.stackName})`;\n\n    this.name = this.stackName; // backwards compat\n    this.originalName = this.stackName;\n  }\n\n  /**\n   * Full path to the template file\n   */\n  public get templateFullPath() {\n    return path.join(this.assembly.directory, this.templateFile);\n  }\n\n  /**\n   * The CloudFormation template for this stack.\n   */\n  public get template(): any {\n    if (this._template === undefined) {\n      this._template = JSON.parse(fs.readFileSync(this.templateFullPath, 'utf-8'));\n    }\n    return this._template;\n  }\n\n  private tagsFromMetadata() {\n    const ret: Record<string, string> = {};\n    for (const metadataEntry of this.findMetadataByType(cxschema.ArtifactMetadataEntryType.STACK_TAGS)) {\n      for (const tag of (metadataEntry.data ?? []) as cxschema.StackTagsMetadataEntry) {\n        ret[tag.key] = tag.value;\n      }\n    }\n    return ret;\n  }\n}\n\n/**\n * Mark all instances of 'CloudFormationStackArtifact'\n *\n * Why not put this in the constructor? Because this is a class property,\n * not an instance property. It applies to all instances of the class.\n */\nObject.defineProperty(CloudFormationStackArtifact.prototype, CLOUDFORMATION_STACK_ARTIFACT_SYM, {\n  value: true,\n  enumerable: false,\n  writable: false,\n});\n"]}