UNPKG

@aws-cdk/core

Version:

AWS Cloud Development Kit Core Library

306 lines 49.3 kB
"use strict"; var _a; Object.defineProperty(exports, "__esModule", { value: true }); exports.DefaultStackSynthesizer = exports.BOOTSTRAP_QUALIFIER_CONTEXT = void 0; const jsiiDeprecationWarnings = require("../../.warnings.jsii.js"); const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti"); const cxapi = require("@aws-cdk/cx-api"); const cfn_fn_1 = require("../cfn-fn"); const cfn_parameter_1 = require("../cfn-parameter"); const cfn_rule_1 = require("../cfn-rule"); const token_1 = require("../token"); const _asset_manifest_builder_1 = require("./_asset-manifest-builder"); const _shared_1 = require("./_shared"); const stack_synthesizer_1 = require("./stack-synthesizer"); exports.BOOTSTRAP_QUALIFIER_CONTEXT = '@aws-cdk/core:bootstrapQualifier'; /* eslint-disable max-len */ /** * The minimum bootstrap stack version required by this app. */ const MIN_BOOTSTRAP_STACK_VERSION = 6; /** * The minimum bootstrap stack version required * to use the lookup role. */ const MIN_LOOKUP_ROLE_BOOTSTRAP_STACK_VERSION = 8; /** * Uses conventionally named roles and asset storage locations * * This synthesizer: * * - Supports cross-account deployments (the CLI can have credentials to one * account, and you can still deploy to another account by assuming roles with * well-known names in the other account). * - Supports the **CDK Pipelines** library. * * Requires the environment to have been bootstrapped with Bootstrap Stack V2 * (also known as "modern bootstrap stack"). The synthesizer adds a version * check to the template, to make sure the bootstrap stack is recent enough * to support all features expected by this synthesizer. */ class DefaultStackSynthesizer extends stack_synthesizer_1.StackSynthesizer { constructor(props = {}) { super(); this.props = props; this.assetManifest = new _asset_manifest_builder_1.AssetManifestBuilder(); try { jsiiDeprecationWarnings._aws_cdk_core_DefaultStackSynthesizerProps(props); } catch (error) { if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") { Error.captureStackTrace(error, DefaultStackSynthesizer); } throw error; } this.useLookupRoleForStackOperations = props.useLookupRoleForStackOperations ?? true; for (const key in props) { if (props.hasOwnProperty(key)) { validateNoToken(key); } } function validateNoToken(key) { const prop = props[key]; if (typeof prop === 'string' && token_1.Token.isUnresolved(prop)) { throw new Error(`DefaultSynthesizer property '${key}' cannot contain tokens; only the following placeholder strings are allowed: ` + [ '${Qualifier}', cxapi.EnvironmentPlaceholders.CURRENT_REGION, cxapi.EnvironmentPlaceholders.CURRENT_ACCOUNT, cxapi.EnvironmentPlaceholders.CURRENT_PARTITION, ].join(', ')); } } } bind(stack) { try { jsiiDeprecationWarnings._aws_cdk_core_Stack(stack); } catch (error) { if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") { Error.captureStackTrace(error, this.bind); } throw error; } if (this._stack !== undefined) { throw new Error('A StackSynthesizer can only be used for one Stack: create a new instance to use with a different Stack'); } this._stack = stack; const qualifier = this.props.qualifier ?? stack.node.tryGetContext(exports.BOOTSTRAP_QUALIFIER_CONTEXT) ?? DefaultStackSynthesizer.DEFAULT_QUALIFIER; this.qualifier = qualifier; const spec = new _shared_1.StringSpecializer(stack, qualifier); /* eslint-disable max-len */ this.bucketName = spec.specialize(this.props.fileAssetsBucketName ?? DefaultStackSynthesizer.DEFAULT_FILE_ASSETS_BUCKET_NAME); this.repositoryName = spec.specialize(this.props.imageAssetsRepositoryName ?? DefaultStackSynthesizer.DEFAULT_IMAGE_ASSETS_REPOSITORY_NAME); this._deployRoleArn = spec.specialize(this.props.deployRoleArn ?? DefaultStackSynthesizer.DEFAULT_DEPLOY_ROLE_ARN); this._cloudFormationExecutionRoleArn = spec.specialize(this.props.cloudFormationExecutionRole ?? DefaultStackSynthesizer.DEFAULT_CLOUDFORMATION_ROLE_ARN); this.fileAssetPublishingRoleArn = spec.specialize(this.props.fileAssetPublishingRoleArn ?? DefaultStackSynthesizer.DEFAULT_FILE_ASSET_PUBLISHING_ROLE_ARN); this.imageAssetPublishingRoleArn = spec.specialize(this.props.imageAssetPublishingRoleArn ?? DefaultStackSynthesizer.DEFAULT_IMAGE_ASSET_PUBLISHING_ROLE_ARN); this.lookupRoleArn = spec.specialize(this.props.lookupRoleArn ?? DefaultStackSynthesizer.DEFAULT_LOOKUP_ROLE_ARN); this.bucketPrefix = spec.specialize(this.props.bucketPrefix ?? DefaultStackSynthesizer.DEFAULT_FILE_ASSET_PREFIX); this.dockerTagPrefix = spec.specialize(this.props.dockerTagPrefix ?? DefaultStackSynthesizer.DEFAULT_DOCKER_ASSET_PREFIX); this.bootstrapStackVersionSsmParameter = spec.qualifierOnly(this.props.bootstrapStackVersionSsmParameter ?? DefaultStackSynthesizer.DEFAULT_BOOTSTRAP_STACK_VERSION_SSM_PARAMETER); } addFileAsset(asset) { try { jsiiDeprecationWarnings._aws_cdk_core_FileAssetSource(asset); } catch (error) { if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") { Error.captureStackTrace(error, this.addFileAsset); } throw error; } _shared_1.assertBound(this.stack); _shared_1.assertBound(this.bucketName); _shared_1.assertBound(this.bucketPrefix); return this.assetManifest.addFileAssetDefault(asset, this.stack, this.bucketName, this.bucketPrefix, { assumeRoleArn: this.fileAssetPublishingRoleArn, assumeRoleExternalId: this.props.fileAssetPublishingExternalId, }); } addDockerImageAsset(asset) { try { jsiiDeprecationWarnings._aws_cdk_core_DockerImageAssetSource(asset); } catch (error) { if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") { Error.captureStackTrace(error, this.addDockerImageAsset); } throw error; } _shared_1.assertBound(this.stack); _shared_1.assertBound(this.repositoryName); _shared_1.assertBound(this.dockerTagPrefix); return this.assetManifest.addDockerImageAssetDefault(asset, this.stack, this.repositoryName, this.dockerTagPrefix, { assumeRoleArn: this.imageAssetPublishingRoleArn, assumeRoleExternalId: this.props.imageAssetPublishingExternalId, }); } synthesizeStackTemplate(stack, session) { try { jsiiDeprecationWarnings._aws_cdk_core_Stack(stack); jsiiDeprecationWarnings._aws_cdk_core_ISynthesisSession(session); } catch (error) { if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") { Error.captureStackTrace(error, this.synthesizeStackTemplate); } throw error; } stack._synthesizeTemplate(session, this.lookupRoleArn); } /** * Synthesize the associated stack to the session */ synthesize(session) { try { jsiiDeprecationWarnings._aws_cdk_core_ISynthesisSession(session); } catch (error) { if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") { Error.captureStackTrace(error, this.synthesize); } throw error; } _shared_1.assertBound(this.stack); _shared_1.assertBound(this.qualifier); // Must be done here -- if it's done in bind() (called in the Stack's constructor) // then it will become impossible to set context after that. // // If it's done AFTER _synthesizeTemplate(), then the template won't contain the // right constructs. if (this.props.generateBootstrapVersionRule ?? true) { addBootstrapVersionRule(this.stack, MIN_BOOTSTRAP_STACK_VERSION, this.bootstrapStackVersionSsmParameter); } this.synthesizeStackTemplate(this.stack, session); const templateAsset = this.addFileAsset(_shared_1.stackTemplateFileAsset(this.stack, session)); const assetManifestId = this.assetManifest.writeManifest(this.stack, session, { requiresBootstrapStackVersion: MIN_BOOTSTRAP_STACK_VERSION, bootstrapStackVersionSsmParameter: this.bootstrapStackVersionSsmParameter, }); this.emitStackArtifact(this.stack, session, { assumeRoleExternalId: this.props.deployRoleExternalId, assumeRoleArn: this._deployRoleArn, cloudFormationExecutionRoleArn: this._cloudFormationExecutionRoleArn, stackTemplateAssetObjectUrl: templateAsset.s3ObjectUrlWithPlaceholders, requiresBootstrapStackVersion: MIN_BOOTSTRAP_STACK_VERSION, bootstrapStackVersionSsmParameter: this.bootstrapStackVersionSsmParameter, additionalDependencies: [assetManifestId], lookupRole: this.useLookupRoleForStackOperations && this.lookupRoleArn ? { arn: this.lookupRoleArn, assumeRoleExternalId: this.props.lookupRoleExternalId, requiresBootstrapStackVersion: MIN_LOOKUP_ROLE_BOOTSTRAP_STACK_VERSION, bootstrapStackVersionSsmParameter: this.bootstrapStackVersionSsmParameter, } : undefined, }); } /** * Returns the ARN of the deploy Role. */ get deployRoleArn() { if (!this._deployRoleArn) { throw new Error('deployRoleArn getter can only be called after the synthesizer has been bound to a Stack'); } return this._deployRoleArn; } /** * Returns the ARN of the CFN execution Role. */ get cloudFormationExecutionRoleArn() { if (!this._cloudFormationExecutionRoleArn) { throw new Error('cloudFormationExecutionRoleArn getter can only be called after the synthesizer has been bound to a Stack'); } return this._cloudFormationExecutionRoleArn; } get stack() { return this._stack; } } exports.DefaultStackSynthesizer = DefaultStackSynthesizer; _a = JSII_RTTI_SYMBOL_1; DefaultStackSynthesizer[_a] = { fqn: "@aws-cdk/core.DefaultStackSynthesizer", version: "1.204.0" }; /** * Default ARN qualifier */ DefaultStackSynthesizer.DEFAULT_QUALIFIER = 'hnb659fds'; /** * Default CloudFormation role ARN. */ DefaultStackSynthesizer.DEFAULT_CLOUDFORMATION_ROLE_ARN = 'arn:${AWS::Partition}:iam::${AWS::AccountId}:role/cdk-${Qualifier}-cfn-exec-role-${AWS::AccountId}-${AWS::Region}'; /** * Default deploy role ARN. */ DefaultStackSynthesizer.DEFAULT_DEPLOY_ROLE_ARN = 'arn:${AWS::Partition}:iam::${AWS::AccountId}:role/cdk-${Qualifier}-deploy-role-${AWS::AccountId}-${AWS::Region}'; /** * Default asset publishing role ARN for file (S3) assets. */ DefaultStackSynthesizer.DEFAULT_FILE_ASSET_PUBLISHING_ROLE_ARN = 'arn:${AWS::Partition}:iam::${AWS::AccountId}:role/cdk-${Qualifier}-file-publishing-role-${AWS::AccountId}-${AWS::Region}'; /** * Default asset publishing role ARN for image (ECR) assets. */ DefaultStackSynthesizer.DEFAULT_IMAGE_ASSET_PUBLISHING_ROLE_ARN = 'arn:${AWS::Partition}:iam::${AWS::AccountId}:role/cdk-${Qualifier}-image-publishing-role-${AWS::AccountId}-${AWS::Region}'; /** * Default lookup role ARN for missing values. */ DefaultStackSynthesizer.DEFAULT_LOOKUP_ROLE_ARN = 'arn:${AWS::Partition}:iam::${AWS::AccountId}:role/cdk-${Qualifier}-lookup-role-${AWS::AccountId}-${AWS::Region}'; /** * Default image assets repository name */ DefaultStackSynthesizer.DEFAULT_IMAGE_ASSETS_REPOSITORY_NAME = 'cdk-${Qualifier}-container-assets-${AWS::AccountId}-${AWS::Region}'; /** * Default file assets bucket name */ DefaultStackSynthesizer.DEFAULT_FILE_ASSETS_BUCKET_NAME = 'cdk-${Qualifier}-assets-${AWS::AccountId}-${AWS::Region}'; /** * Name of the CloudFormation Export with the asset key name */ DefaultStackSynthesizer.DEFAULT_FILE_ASSET_KEY_ARN_EXPORT_NAME = 'CdkBootstrap-${Qualifier}-FileAssetKeyArn'; /** * Default file asset prefix */ DefaultStackSynthesizer.DEFAULT_FILE_ASSET_PREFIX = ''; /** * Default Docker asset prefix */ DefaultStackSynthesizer.DEFAULT_DOCKER_ASSET_PREFIX = ''; /** * Default bootstrap stack version SSM parameter. */ DefaultStackSynthesizer.DEFAULT_BOOTSTRAP_STACK_VERSION_SSM_PARAMETER = '/cdk-bootstrap/${Qualifier}/version'; /** * Add a CfnRule to the Stack which checks the current version of the bootstrap stack this template is targeting * * The CLI normally checks this, but in a pipeline the CLI is not involved * so we encode this rule into the template in a way that CloudFormation will check it. */ function addBootstrapVersionRule(stack, requiredVersion, bootstrapStackVersionSsmParameter) { // Because of https://github.com/aws/aws-cdk/blob/master/packages/assert-internal/lib/synth-utils.ts#L74 // synthesize() may be called more than once on a stack in unit tests, and the below would break // if we execute it a second time. Guard against the constructs already existing. if (stack.node.tryFindChild('BootstrapVersion')) { return; } const param = new cfn_parameter_1.CfnParameter(stack, 'BootstrapVersion', { type: 'AWS::SSM::Parameter::Value<String>', description: `Version of the CDK Bootstrap resources in this environment, automatically retrieved from SSM Parameter Store. ${cxapi.SSMPARAM_NO_INVALIDATE}`, default: bootstrapStackVersionSsmParameter, }); // There is no >= check in CloudFormation, so we have to check the number // is NOT in [1, 2, 3, ... <required> - 1] const oldVersions = range(1, requiredVersion).map(n => `${n}`); new cfn_rule_1.CfnRule(stack, 'CheckBootstrapVersion', { assertions: [ { assert: cfn_fn_1.Fn.conditionNot(cfn_fn_1.Fn.conditionContains(oldVersions, param.valueAsString)), assertDescription: `CDK bootstrap stack version ${requiredVersion} required. Please run 'cdk bootstrap' with a recent version of the CDK CLI.`, }, ], }); } function range(startIncl, endExcl) { const ret = new Array(); for (let i = startIncl; i < endExcl; i++) { ret.push(i); } return ret; } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"default-synthesizer.js","sourceRoot":"","sources":["default-synthesizer.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cxapi from '@aws-cdk/cx-api';\nimport { DockerImageAssetLocation, DockerImageAssetSource, FileAssetLocation, FileAssetSource } from '../assets';\nimport { Fn } from '../cfn-fn';\nimport { CfnParameter } from '../cfn-parameter';\nimport { CfnRule } from '../cfn-rule';\nimport { ISynthesisSession } from '../construct-compat';\nimport { Stack } from '../stack';\nimport { Token } from '../token';\nimport { AssetManifestBuilder } from './_asset-manifest-builder';\nimport { assertBound, StringSpecializer, stackTemplateFileAsset } from './_shared';\nimport { StackSynthesizer } from './stack-synthesizer';\n\nexport const BOOTSTRAP_QUALIFIER_CONTEXT = '@aws-cdk/core:bootstrapQualifier';\n\n/* eslint-disable max-len */\n\n/**\n * The minimum bootstrap stack version required by this app.\n */\nconst MIN_BOOTSTRAP_STACK_VERSION = 6;\n\n/**\n * The minimum bootstrap stack version required\n * to use the lookup role.\n */\nconst MIN_LOOKUP_ROLE_BOOTSTRAP_STACK_VERSION = 8;\n\n/**\n * Configuration properties for DefaultStackSynthesizer\n */\nexport interface DefaultStackSynthesizerProps {\n  /**\n   * Name of the S3 bucket to hold file assets\n   *\n   * You must supply this if you have given a non-standard name to the staging bucket.\n   *\n   * The placeholders `${Qualifier}`, `${AWS::AccountId}` and `${AWS::Region}` will\n   * be replaced with the values of qualifier and the stack's account and region,\n   * respectively.\n   *\n   * @default DefaultStackSynthesizer.DEFAULT_FILE_ASSETS_BUCKET_NAME\n   */\n  readonly fileAssetsBucketName?: string;\n\n  /**\n   * Name of the ECR repository to hold Docker Image assets\n   *\n   * You must supply this if you have given a non-standard name to the ECR repository.\n   *\n   * The placeholders `${Qualifier}`, `${AWS::AccountId}` and `${AWS::Region}` will\n   * be replaced with the values of qualifier and the stack's account and region,\n   * respectively.\n   *\n   * @default DefaultStackSynthesizer.DEFAULT_IMAGE_ASSETS_REPOSITORY_NAME\n   */\n  readonly imageAssetsRepositoryName?: string;\n\n  /**\n   * The role to use to publish file assets to the S3 bucket in this environment\n   *\n   * You must supply this if you have given a non-standard name to the publishing role.\n   *\n   * The placeholders `${Qualifier}`, `${AWS::AccountId}` and `${AWS::Region}` will\n   * be replaced with the values of qualifier and the stack's account and region,\n   * respectively.\n   *\n   * @default DefaultStackSynthesizer.DEFAULT_FILE_ASSET_PUBLISHING_ROLE_ARN\n   */\n  readonly fileAssetPublishingRoleArn?: string;\n\n  /**\n   * External ID to use when assuming role for file asset publishing\n   *\n   * @default - No external ID\n   */\n  readonly fileAssetPublishingExternalId?: string;\n\n  /**\n   * The role to use to publish image assets to the ECR repository in this environment\n   *\n   * You must supply this if you have given a non-standard name to the publishing role.\n   *\n   * The placeholders `${Qualifier}`, `${AWS::AccountId}` and `${AWS::Region}` will\n   * be replaced with the values of qualifier and the stack's account and region,\n   * respectively.\n   *\n   * @default DefaultStackSynthesizer.DEFAULT_IMAGE_ASSET_PUBLISHING_ROLE_ARN\n   */\n  readonly imageAssetPublishingRoleArn?: string;\n\n  /**\n   * The role to use to look up values from the target AWS account during synthesis\n   *\n   * @default - None\n   */\n  readonly lookupRoleArn?: string;\n\n  /**\n   * External ID to use when assuming lookup role\n   *\n   * @default - No external ID\n   */\n  readonly lookupRoleExternalId?: string;\n\n  /**\n   * Use the bootstrapped lookup role for (read-only) stack operations\n   *\n   * Use the lookup role when performing a `cdk diff`. If set to `false`, the\n   * `deploy role` credentials will be used to perform a `cdk diff`.\n   *\n   * Requires bootstrap stack version 8.\n   *\n   * @default true\n   */\n  readonly useLookupRoleForStackOperations?: boolean;\n\n  /**\n   * External ID to use when assuming role for image asset publishing\n   *\n   * @default - No external ID\n   */\n  readonly imageAssetPublishingExternalId?: string;\n\n  /**\n   * External ID to use when assuming role for cloudformation deployments\n   *\n   * @default - No external ID\n   */\n  readonly deployRoleExternalId?: string;\n\n  /**\n   * The role to assume to initiate a deployment in this environment\n   *\n   * You must supply this if you have given a non-standard name to the publishing role.\n   *\n   * The placeholders `${Qualifier}`, `${AWS::AccountId}` and `${AWS::Region}` will\n   * be replaced with the values of qualifier and the stack's account and region,\n   * respectively.\n   *\n   * @default DefaultStackSynthesizer.DEFAULT_DEPLOY_ROLE_ARN\n   */\n  readonly deployRoleArn?: string;\n\n  /**\n   * The role CloudFormation will assume when deploying the Stack\n   *\n   * You must supply this if you have given a non-standard name to the execution role.\n   *\n   * The placeholders `${Qualifier}`, `${AWS::AccountId}` and `${AWS::Region}` will\n   * be replaced with the values of qualifier and the stack's account and region,\n   * respectively.\n   *\n   * @default DefaultStackSynthesizer.DEFAULT_CLOUDFORMATION_ROLE_ARN\n   */\n  readonly cloudFormationExecutionRole?: string;\n\n  /**\n   * Name of the CloudFormation Export with the asset key name\n   *\n   * You must supply this if you have given a non-standard name to the KMS key export\n   *\n   * The placeholders `${Qualifier}`, `${AWS::AccountId}` and `${AWS::Region}` will\n   * be replaced with the values of qualifier and the stack's account and region,\n   * respectively.\n   *\n   * @default DefaultStackSynthesizer.DEFAULT_FILE_ASSET_KEY_ARN_EXPORT_NAME\n   * @deprecated This property is not used anymore\n   */\n  readonly fileAssetKeyArnExportName?: string;\n\n  /**\n   * Qualifier to disambiguate multiple environments in the same account\n   *\n   * You can use this and leave the other naming properties empty if you have deployed\n   * the bootstrap environment with standard names but only differnet qualifiers.\n   *\n   * @default - Value of context key '@aws-cdk/core:bootstrapQualifier' if set, otherwise `DefaultStackSynthesizer.DEFAULT_QUALIFIER`\n   */\n  readonly qualifier?: string;\n\n  /**\n   * Whether to add a Rule to the stack template verifying the bootstrap stack version\n   *\n   * This generally should be left set to `true`, unless you explicitly\n   * want to be able to deploy to an unbootstrapped environment.\n   *\n   * @default true\n   */\n  readonly generateBootstrapVersionRule?: boolean;\n\n  /**\n   * bucketPrefix to use while storing S3 Assets\n   *\n   * @default - DefaultStackSynthesizer.DEFAULT_FILE_ASSET_PREFIX\n   */\n  readonly bucketPrefix?: string;\n\n  /**\n   * A prefix to use while tagging and uploading Docker images to ECR.\n   *\n   * This does not add any separators - the source hash will be appended to\n   * this string directly.\n   *\n   * @default - DefaultStackSynthesizer.DEFAULT_DOCKER_ASSET_PREFIX\n   */\n  readonly dockerTagPrefix?: string;\n\n  /**\n   * Bootstrap stack version SSM parameter.\n   *\n   * The placeholder `${Qualifier}` will be replaced with the value of qualifier.\n   *\n   * @default DefaultStackSynthesizer.DEFAULT_BOOTSTRAP_STACK_VERSION_SSM_PARAMETER\n   */\n  readonly bootstrapStackVersionSsmParameter?: string;\n}\n\n/**\n * Uses conventionally named roles and asset storage locations\n *\n * This synthesizer:\n *\n * - Supports cross-account deployments (the CLI can have credentials to one\n *   account, and you can still deploy to another account by assuming roles with\n *   well-known names in the other account).\n * - Supports the **CDK Pipelines** library.\n *\n * Requires the environment to have been bootstrapped with Bootstrap Stack V2\n * (also known as \"modern bootstrap stack\"). The synthesizer adds a version\n * check to the template, to make sure the bootstrap stack is recent enough\n * to support all features expected by this synthesizer.\n */\nexport class DefaultStackSynthesizer extends StackSynthesizer {\n  /**\n   * Default ARN qualifier\n   */\n  public static readonly DEFAULT_QUALIFIER = 'hnb659fds';\n\n  /**\n   * Default CloudFormation role ARN.\n   */\n  public static readonly DEFAULT_CLOUDFORMATION_ROLE_ARN = 'arn:${AWS::Partition}:iam::${AWS::AccountId}:role/cdk-${Qualifier}-cfn-exec-role-${AWS::AccountId}-${AWS::Region}';\n\n  /**\n   * Default deploy role ARN.\n   */\n  public static readonly DEFAULT_DEPLOY_ROLE_ARN = 'arn:${AWS::Partition}:iam::${AWS::AccountId}:role/cdk-${Qualifier}-deploy-role-${AWS::AccountId}-${AWS::Region}';\n\n  /**\n   * Default asset publishing role ARN for file (S3) assets.\n   */\n  public static readonly DEFAULT_FILE_ASSET_PUBLISHING_ROLE_ARN = 'arn:${AWS::Partition}:iam::${AWS::AccountId}:role/cdk-${Qualifier}-file-publishing-role-${AWS::AccountId}-${AWS::Region}';\n\n  /**\n   * Default asset publishing role ARN for image (ECR) assets.\n   */\n  public static readonly DEFAULT_IMAGE_ASSET_PUBLISHING_ROLE_ARN = 'arn:${AWS::Partition}:iam::${AWS::AccountId}:role/cdk-${Qualifier}-image-publishing-role-${AWS::AccountId}-${AWS::Region}';\n\n  /**\n   * Default lookup role ARN for missing values.\n   */\n  public static readonly DEFAULT_LOOKUP_ROLE_ARN = 'arn:${AWS::Partition}:iam::${AWS::AccountId}:role/cdk-${Qualifier}-lookup-role-${AWS::AccountId}-${AWS::Region}';\n\n  /**\n   * Default image assets repository name\n   */\n  public static readonly DEFAULT_IMAGE_ASSETS_REPOSITORY_NAME = 'cdk-${Qualifier}-container-assets-${AWS::AccountId}-${AWS::Region}';\n\n  /**\n   * Default file assets bucket name\n   */\n  public static readonly DEFAULT_FILE_ASSETS_BUCKET_NAME = 'cdk-${Qualifier}-assets-${AWS::AccountId}-${AWS::Region}';\n\n  /**\n   * Name of the CloudFormation Export with the asset key name\n   */\n  public static readonly DEFAULT_FILE_ASSET_KEY_ARN_EXPORT_NAME = 'CdkBootstrap-${Qualifier}-FileAssetKeyArn';\n\n  /**\n   * Default file asset prefix\n   */\n  public static readonly DEFAULT_FILE_ASSET_PREFIX = '';\n  /**\n   * Default Docker asset prefix\n   */\n  public static readonly DEFAULT_DOCKER_ASSET_PREFIX = '';\n\n  /**\n   * Default bootstrap stack version SSM parameter.\n   */\n  public static readonly DEFAULT_BOOTSTRAP_STACK_VERSION_SSM_PARAMETER = '/cdk-bootstrap/${Qualifier}/version';\n\n  private _stack?: Stack;\n  private bucketName?: string;\n  private repositoryName?: string;\n  private _deployRoleArn?: string;\n  private _cloudFormationExecutionRoleArn?: string;\n  private fileAssetPublishingRoleArn?: string;\n  private imageAssetPublishingRoleArn?: string;\n  private lookupRoleArn?: string;\n  private useLookupRoleForStackOperations: boolean;\n  private qualifier?: string;\n  private bucketPrefix?: string;\n  private dockerTagPrefix?: string;\n  private bootstrapStackVersionSsmParameter?: string;\n\n  private assetManifest = new AssetManifestBuilder();\n\n  constructor(private readonly props: DefaultStackSynthesizerProps = {}) {\n    super();\n    this.useLookupRoleForStackOperations = props.useLookupRoleForStackOperations ?? true;\n\n    for (const key in props) {\n      if (props.hasOwnProperty(key)) {\n        validateNoToken(key as keyof DefaultStackSynthesizerProps);\n      }\n    }\n\n    function validateNoToken<A extends keyof DefaultStackSynthesizerProps>(key: A) {\n      const prop = props[key];\n      if (typeof prop === 'string' && Token.isUnresolved(prop)) {\n        throw new Error(`DefaultSynthesizer property '${key}' cannot contain tokens; only the following placeholder strings are allowed: ` + [\n          '${Qualifier}',\n          cxapi.EnvironmentPlaceholders.CURRENT_REGION,\n          cxapi.EnvironmentPlaceholders.CURRENT_ACCOUNT,\n          cxapi.EnvironmentPlaceholders.CURRENT_PARTITION,\n        ].join(', '));\n      }\n    }\n  }\n\n  public bind(stack: Stack): void {\n    if (this._stack !== undefined) {\n      throw new Error('A StackSynthesizer can only be used for one Stack: create a new instance to use with a different Stack');\n    }\n\n    this._stack = stack;\n\n    const qualifier = this.props.qualifier ?? stack.node.tryGetContext(BOOTSTRAP_QUALIFIER_CONTEXT) ?? DefaultStackSynthesizer.DEFAULT_QUALIFIER;\n    this.qualifier = qualifier;\n\n    const spec = new StringSpecializer(stack, qualifier);\n\n    /* eslint-disable max-len */\n    this.bucketName = spec.specialize(this.props.fileAssetsBucketName ?? DefaultStackSynthesizer.DEFAULT_FILE_ASSETS_BUCKET_NAME);\n    this.repositoryName = spec.specialize(this.props.imageAssetsRepositoryName ?? DefaultStackSynthesizer.DEFAULT_IMAGE_ASSETS_REPOSITORY_NAME);\n    this._deployRoleArn = spec.specialize(this.props.deployRoleArn ?? DefaultStackSynthesizer.DEFAULT_DEPLOY_ROLE_ARN);\n    this._cloudFormationExecutionRoleArn = spec.specialize(this.props.cloudFormationExecutionRole ?? DefaultStackSynthesizer.DEFAULT_CLOUDFORMATION_ROLE_ARN);\n    this.fileAssetPublishingRoleArn = spec.specialize(this.props.fileAssetPublishingRoleArn ?? DefaultStackSynthesizer.DEFAULT_FILE_ASSET_PUBLISHING_ROLE_ARN);\n    this.imageAssetPublishingRoleArn = spec.specialize(this.props.imageAssetPublishingRoleArn ?? DefaultStackSynthesizer.DEFAULT_IMAGE_ASSET_PUBLISHING_ROLE_ARN);\n    this.lookupRoleArn = spec.specialize(this.props.lookupRoleArn ?? DefaultStackSynthesizer.DEFAULT_LOOKUP_ROLE_ARN);\n    this.bucketPrefix = spec.specialize(this.props.bucketPrefix ?? DefaultStackSynthesizer.DEFAULT_FILE_ASSET_PREFIX);\n    this.dockerTagPrefix = spec.specialize(this.props.dockerTagPrefix ?? DefaultStackSynthesizer.DEFAULT_DOCKER_ASSET_PREFIX);\n    this.bootstrapStackVersionSsmParameter = spec.qualifierOnly(this.props.bootstrapStackVersionSsmParameter ?? DefaultStackSynthesizer.DEFAULT_BOOTSTRAP_STACK_VERSION_SSM_PARAMETER);\n    /* eslint-enable max-len */\n  }\n\n  public addFileAsset(asset: FileAssetSource): FileAssetLocation {\n    assertBound(this.stack);\n    assertBound(this.bucketName);\n    assertBound(this.bucketPrefix);\n\n    return this.assetManifest.addFileAssetDefault(asset, this.stack, this.bucketName, this.bucketPrefix, {\n      assumeRoleArn: this.fileAssetPublishingRoleArn,\n      assumeRoleExternalId: this.props.fileAssetPublishingExternalId,\n    });\n  }\n\n  public addDockerImageAsset(asset: DockerImageAssetSource): DockerImageAssetLocation {\n    assertBound(this.stack);\n    assertBound(this.repositoryName);\n    assertBound(this.dockerTagPrefix);\n\n    return this.assetManifest.addDockerImageAssetDefault(asset, this.stack, this.repositoryName, this.dockerTagPrefix, {\n      assumeRoleArn: this.imageAssetPublishingRoleArn,\n      assumeRoleExternalId: this.props.imageAssetPublishingExternalId,\n    });\n  }\n\n  protected synthesizeStackTemplate(stack: Stack, session: ISynthesisSession) {\n    stack._synthesizeTemplate(session, this.lookupRoleArn);\n  }\n\n  /**\n   * Synthesize the associated stack to the session\n   */\n  public synthesize(session: ISynthesisSession): void {\n    assertBound(this.stack);\n    assertBound(this.qualifier);\n\n    // Must be done here -- if it's done in bind() (called in the Stack's constructor)\n    // then it will become impossible to set context after that.\n    //\n    // If it's done AFTER _synthesizeTemplate(), then the template won't contain the\n    // right constructs.\n    if (this.props.generateBootstrapVersionRule ?? true) {\n      addBootstrapVersionRule(this.stack, MIN_BOOTSTRAP_STACK_VERSION, <string> this.bootstrapStackVersionSsmParameter);\n    }\n\n    this.synthesizeStackTemplate(this.stack, session);\n\n    const templateAsset = this.addFileAsset(stackTemplateFileAsset(this.stack, session));\n\n    const assetManifestId = this.assetManifest.writeManifest(this.stack, session, {\n      requiresBootstrapStackVersion: MIN_BOOTSTRAP_STACK_VERSION,\n      bootstrapStackVersionSsmParameter: this.bootstrapStackVersionSsmParameter,\n    });\n\n    this.emitStackArtifact(this.stack, session, {\n      assumeRoleExternalId: this.props.deployRoleExternalId,\n      assumeRoleArn: this._deployRoleArn,\n      cloudFormationExecutionRoleArn: this._cloudFormationExecutionRoleArn,\n      stackTemplateAssetObjectUrl: templateAsset.s3ObjectUrlWithPlaceholders,\n      requiresBootstrapStackVersion: MIN_BOOTSTRAP_STACK_VERSION,\n      bootstrapStackVersionSsmParameter: this.bootstrapStackVersionSsmParameter,\n      additionalDependencies: [assetManifestId],\n      lookupRole: this.useLookupRoleForStackOperations && this.lookupRoleArn ? {\n        arn: this.lookupRoleArn,\n        assumeRoleExternalId: this.props.lookupRoleExternalId,\n        requiresBootstrapStackVersion: MIN_LOOKUP_ROLE_BOOTSTRAP_STACK_VERSION,\n        bootstrapStackVersionSsmParameter: this.bootstrapStackVersionSsmParameter,\n      } : undefined,\n    });\n  }\n\n  /**\n   * Returns the ARN of the deploy Role.\n   */\n  public get deployRoleArn(): string {\n    if (!this._deployRoleArn) {\n      throw new Error('deployRoleArn getter can only be called after the synthesizer has been bound to a Stack');\n    }\n    return this._deployRoleArn;\n  }\n\n  /**\n   * Returns the ARN of the CFN execution Role.\n   */\n  public get cloudFormationExecutionRoleArn(): string {\n    if (!this._cloudFormationExecutionRoleArn) {\n      throw new Error('cloudFormationExecutionRoleArn getter can only be called after the synthesizer has been bound to a Stack');\n    }\n    return this._cloudFormationExecutionRoleArn;\n  }\n\n  protected get stack(): Stack | undefined {\n    return this._stack;\n  }\n}\n\n/**\n * Add a CfnRule to the Stack which checks the current version of the bootstrap stack this template is targeting\n *\n * The CLI normally checks this, but in a pipeline the CLI is not involved\n * so we encode this rule into the template in a way that CloudFormation will check it.\n */\nfunction addBootstrapVersionRule(stack: Stack, requiredVersion: number, bootstrapStackVersionSsmParameter: string) {\n  // Because of https://github.com/aws/aws-cdk/blob/master/packages/assert-internal/lib/synth-utils.ts#L74\n  // synthesize() may be called more than once on a stack in unit tests, and the below would break\n  // if we execute it a second time. Guard against the constructs already existing.\n  if (stack.node.tryFindChild('BootstrapVersion')) { return; }\n\n  const param = new CfnParameter(stack, 'BootstrapVersion', {\n    type: 'AWS::SSM::Parameter::Value<String>',\n    description: `Version of the CDK Bootstrap resources in this environment, automatically retrieved from SSM Parameter Store. ${cxapi.SSMPARAM_NO_INVALIDATE}`,\n    default: bootstrapStackVersionSsmParameter,\n  });\n\n  // There is no >= check in CloudFormation, so we have to check the number\n  // is NOT in [1, 2, 3, ... <required> - 1]\n  const oldVersions = range(1, requiredVersion).map(n => `${n}`);\n\n  new CfnRule(stack, 'CheckBootstrapVersion', {\n    assertions: [\n      {\n        assert: Fn.conditionNot(Fn.conditionContains(oldVersions, param.valueAsString)),\n        assertDescription: `CDK bootstrap stack version ${requiredVersion} required. Please run 'cdk bootstrap' with a recent version of the CDK CLI.`,\n      },\n    ],\n  });\n}\n\nfunction range(startIncl: number, endExcl: number) {\n  const ret = new Array<number>();\n  for (let i = startIncl; i < endExcl; i++) {\n    ret.push(i);\n  }\n  return ret;\n}\n\n"]}