@aws-cdk/core
Version:
AWS Cloud Development Kit Core Library
127 lines • 17.8 kB
JavaScript
"use strict";
var _a;
Object.defineProperty(exports, "__esModule", { value: true });
exports.Stage = 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 constructs_1 = require("constructs");
const synthesis_1 = require("./private/synthesis");
// v2 - keep this import as a separate section to reduce merge conflict when forward merging with the v2 branch.
// eslint-disable-next-line
const construct_compat_1 = require("./construct-compat");
const STAGE_SYMBOL = Symbol.for('@aws-cdk/core.Stage');
/**
* An abstract application modeling unit consisting of Stacks that should be
* deployed together.
*
* Derive a subclass of `Stage` and use it to model a single instance of your
* application.
*
* You can then instantiate your subclass multiple times to model multiple
* copies of your application which should be be deployed to different
* environments.
*/
class Stage extends construct_compat_1.Construct {
constructor(scope, id, props = {}) {
super(scope, id);
try {
jsiiDeprecationWarnings._aws_cdk_core_StageProps(props);
}
catch (error) {
if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
Error.captureStackTrace(error, Stage);
}
throw error;
}
if (id !== '' && !/^[a-z][a-z0-9\-\_\.]+$/i.test(id)) {
throw new Error(`invalid stage name "${id}". Stage name must start with a letter and contain only alphanumeric characters, hypens ('-'), underscores ('_') and periods ('.')`);
}
Object.defineProperty(this, STAGE_SYMBOL, { value: true });
this.parentStage = Stage.of(this);
this.region = props.env?.region ?? this.parentStage?.region;
this.account = props.env?.account ?? this.parentStage?.account;
this._assemblyBuilder = this.createBuilder(props.outdir);
this.stageName = [this.parentStage?.stageName, id].filter(x => x).join('-');
}
/**
* Return the stage this construct is contained with, if available. If called
* on a nested stage, returns its parent.
*
*/
static of(construct) {
return constructs_1.Node.of(construct).scopes.reverse().slice(1).find(Stage.isStage);
}
/**
* Test whether the given construct is a stage.
*
*/
static isStage(x) {
return x !== null && typeof (x) === 'object' && STAGE_SYMBOL in x;
}
/**
* The cloud assembly output directory.
*/
get outdir() {
return this._assemblyBuilder.outdir;
}
/**
* The cloud assembly asset output directory.
*/
get assetOutdir() {
return this._assemblyBuilder.assetOutdir;
}
/**
* Artifact ID of the assembly if it is a nested stage. The root stage (app)
* will return an empty string.
*
* Derived from the construct path.
*
*/
get artifactId() {
if (!this.node.path) {
return '';
}
return `assembly-${this.node.path.replace(/\//g, '-').replace(/^-+|-+$/g, '')}`;
}
/**
* Synthesize this stage into a cloud assembly.
*
* Once an assembly has been synthesized, it cannot be modified. Subsequent
* calls will return the same assembly.
*/
synth(options = {}) {
try {
jsiiDeprecationWarnings._aws_cdk_core_StageSynthesisOptions(options);
}
catch (error) {
if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
Error.captureStackTrace(error, this.synth);
}
throw error;
}
if (!this.assembly || options.force) {
this.assembly = synthesis_1.synthesize(this, {
skipValidation: options.skipValidation,
validateOnSynthesis: options.validateOnSynthesis,
});
}
return this.assembly;
}
createBuilder(outdir) {
// cannot specify "outdir" if we are a nested stage
if (this.parentStage && outdir) {
throw new Error('"outdir" cannot be specified for nested stages');
}
// Need to determine fixed output directory already, because we must know where
// to write sub-assemblies (which must happen before we actually get to this app's
// synthesize() phase).
return this.parentStage
? this.parentStage._assemblyBuilder.createNestedAssembly(this.artifactId, this.node.path)
: new cxapi.CloudAssemblyBuilder(outdir);
}
}
exports.Stage = Stage;
_a = JSII_RTTI_SYMBOL_1;
Stage[_a] = { fqn: "@aws-cdk/core.Stage", version: "1.204.0" };
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"stage.js","sourceRoot":"","sources":["stage.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cxapi from '@aws-cdk/cx-api';\nimport { IConstruct, Construct, Node } from 'constructs';\nimport { Environment } from './environment';\nimport { synthesize } from './private/synthesis';\n\n// v2 - keep this import as a separate section to reduce merge conflict when forward merging with the v2 branch.\n// eslint-disable-next-line\nimport { Construct as CoreConstruct } from './construct-compat';\n\nconst STAGE_SYMBOL = Symbol.for('@aws-cdk/core.Stage');\n\n/**\n * Initialization props for a stage.\n */\nexport interface StageProps {\n  /**\n   * Default AWS environment (account/region) for `Stack`s in this `Stage`.\n   *\n   * Stacks defined inside this `Stage` with either `region` or `account` missing\n   * from its env will use the corresponding field given here.\n   *\n   * If either `region` or `account`is is not configured for `Stack` (either on\n   * the `Stack` itself or on the containing `Stage`), the Stack will be\n   * *environment-agnostic*.\n   *\n   * Environment-agnostic stacks can be deployed to any environment, may not be\n   * able to take advantage of all features of the CDK. For example, they will\n   * not be able to use environmental context lookups, will not automatically\n   * translate Service Principals to the right format based on the environment's\n   * AWS partition, and other such enhancements.\n   *\n   * @example\n   *\n   * // Use a concrete account and region to deploy this Stage to\n   * new Stage(app, 'Stage1', {\n   *   env: { account: '123456789012', region: 'us-east-1' },\n   * });\n   *\n   * // Use the CLI's current credentials to determine the target environment\n   * new Stage(app, 'Stage2', {\n   *   env: { account: process.env.CDK_DEFAULT_ACCOUNT, region: process.env.CDK_DEFAULT_REGION },\n   * });\n   *\n   * @default - The environments should be configured on the `Stack`s.\n   */\n  readonly env?: Environment;\n\n  /**\n   * The output directory into which to emit synthesized artifacts.\n   *\n   * Can only be specified if this stage is the root stage (the app). If this is\n   * specified and this stage is nested within another stage, an error will be\n   * thrown.\n   *\n   * @default - for nested stages, outdir will be determined as a relative\n   * directory to the outdir of the app. For apps, if outdir is not specified, a\n   * temporary directory will be created.\n   */\n  readonly outdir?: string;\n}\n\n/**\n * An abstract application modeling unit consisting of Stacks that should be\n * deployed together.\n *\n * Derive a subclass of `Stage` and use it to model a single instance of your\n * application.\n *\n * You can then instantiate your subclass multiple times to model multiple\n * copies of your application which should be be deployed to different\n * environments.\n */\nexport class Stage extends CoreConstruct {\n  /**\n   * Return the stage this construct is contained with, if available. If called\n   * on a nested stage, returns its parent.\n   *\n   */\n  public static of(construct: IConstruct): Stage | undefined {\n    return Node.of(construct).scopes.reverse().slice(1).find(Stage.isStage);\n  }\n\n  /**\n   * Test whether the given construct is a stage.\n   *\n   */\n  public static isStage(x: any ): x is Stage {\n    return x !== null && typeof(x) === 'object' && STAGE_SYMBOL in x;\n  }\n\n  /**\n   * The default region for all resources defined within this stage.\n   *\n   */\n  public readonly region?: string;\n\n  /**\n   * The default account for all resources defined within this stage.\n   *\n   */\n  public readonly account?: string;\n\n  /**\n   * The cloud assembly builder that is being used for this App\n   *\n   * @internal\n   */\n  public readonly _assemblyBuilder: cxapi.CloudAssemblyBuilder;\n\n  /**\n   * The name of the stage. Based on names of the parent stages separated by\n   * hypens.\n   *\n   */\n  public readonly stageName: string;\n\n  /**\n   * The parent stage or `undefined` if this is the app.\n   * *\n   */\n  public readonly parentStage?: Stage;\n\n  /**\n   * The cached assembly if it was already built\n   */\n  private assembly?: cxapi.CloudAssembly;\n\n  constructor(scope: Construct, id: string, props: StageProps = {}) {\n    super(scope, id);\n\n    if (id !== '' && !/^[a-z][a-z0-9\\-\\_\\.]+$/i.test(id)) {\n      throw new Error(`invalid stage name \"${id}\". Stage name must start with a letter and contain only alphanumeric characters, hypens ('-'), underscores ('_') and periods ('.')`);\n    }\n\n    Object.defineProperty(this, STAGE_SYMBOL, { value: true });\n\n    this.parentStage = Stage.of(this);\n\n    this.region = props.env?.region ?? this.parentStage?.region;\n    this.account = props.env?.account ?? this.parentStage?.account;\n\n    this._assemblyBuilder = this.createBuilder(props.outdir);\n    this.stageName = [this.parentStage?.stageName, id].filter(x => x).join('-');\n  }\n\n  /**\n   * The cloud assembly output directory.\n   */\n  public get outdir() {\n    return this._assemblyBuilder.outdir;\n  }\n\n  /**\n   * The cloud assembly asset output directory.\n   */\n  public get assetOutdir() {\n    return this._assemblyBuilder.assetOutdir;\n  }\n\n  /**\n   * Artifact ID of the assembly if it is a nested stage. The root stage (app)\n   * will return an empty string.\n   *\n   * Derived from the construct path.\n   *\n   */\n  public get artifactId() {\n    if (!this.node.path) { return ''; }\n    return `assembly-${this.node.path.replace(/\\//g, '-').replace(/^-+|-+$/g, '')}`;\n  }\n\n  /**\n   * Synthesize this stage into a cloud assembly.\n   *\n   * Once an assembly has been synthesized, it cannot be modified. Subsequent\n   * calls will return the same assembly.\n   */\n  public synth(options: StageSynthesisOptions = { }): cxapi.CloudAssembly {\n    if (!this.assembly || options.force) {\n      this.assembly = synthesize(this, {\n        skipValidation: options.skipValidation,\n        validateOnSynthesis: options.validateOnSynthesis,\n      });\n    }\n\n    return this.assembly;\n  }\n\n  private createBuilder(outdir?: string) {\n    // cannot specify \"outdir\" if we are a nested stage\n    if (this.parentStage && outdir) {\n      throw new Error('\"outdir\" cannot be specified for nested stages');\n    }\n\n    // Need to determine fixed output directory already, because we must know where\n    // to write sub-assemblies (which must happen before we actually get to this app's\n    // synthesize() phase).\n    return this.parentStage\n      ? this.parentStage._assemblyBuilder.createNestedAssembly(this.artifactId, this.node.path)\n      : new cxapi.CloudAssemblyBuilder(outdir);\n  }\n}\n\n/**\n * Options for assembly synthesis.\n */\nexport interface StageSynthesisOptions {\n  /**\n   * Should we skip construct validation.\n   * @default - false\n   */\n  readonly skipValidation?: boolean;\n\n  /**\n   * Whether the stack should be validated after synthesis to check for error metadata\n   *\n   * @default - false\n   */\n  readonly validateOnSynthesis?: boolean;\n\n  /**\n   * Force a re-synth, even if the stage has already been synthesized.\n   * This is used by tests to allow for incremental verification of the output.\n   * Do not use in production.\n   * @default false\n   */\n  readonly force?: boolean;\n}\n"]}