UNPKG

17.8 kBJavaScriptView Raw
1"use strict";
2var _a;
3Object.defineProperty(exports, "__esModule", { value: true });
4exports.Stage = void 0;
5const jsiiDeprecationWarnings = require("../.warnings.jsii.js");
6const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
7const cxapi = require("@aws-cdk/cx-api");
8const constructs_1 = require("constructs");
9const synthesis_1 = require("./private/synthesis");
10// v2 - keep this import as a separate section to reduce merge conflict when forward merging with the v2 branch.
11// eslint-disable-next-line
12const construct_compat_1 = require("./construct-compat");
13const STAGE_SYMBOL = Symbol.for('@aws-cdk/core.Stage');
14/**
15 * An abstract application modeling unit consisting of Stacks that should be
16 * deployed together.
17 *
18 * Derive a subclass of `Stage` and use it to model a single instance of your
19 * application.
20 *
21 * You can then instantiate your subclass multiple times to model multiple
22 * copies of your application which should be be deployed to different
23 * environments.
24 */
25class Stage extends construct_compat_1.Construct {
26 constructor(scope, id, props = {}) {
27 super(scope, id);
28 try {
29 jsiiDeprecationWarnings._aws_cdk_core_StageProps(props);
30 }
31 catch (error) {
32 if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
33 Error.captureStackTrace(error, Stage);
34 }
35 throw error;
36 }
37 if (id !== '' && !/^[a-z][a-z0-9\-\_\.]+$/i.test(id)) {
38 throw new Error(`invalid stage name "${id}". Stage name must start with a letter and contain only alphanumeric characters, hypens ('-'), underscores ('_') and periods ('.')`);
39 }
40 Object.defineProperty(this, STAGE_SYMBOL, { value: true });
41 this.parentStage = Stage.of(this);
42 this.region = props.env?.region ?? this.parentStage?.region;
43 this.account = props.env?.account ?? this.parentStage?.account;
44 this._assemblyBuilder = this.createBuilder(props.outdir);
45 this.stageName = [this.parentStage?.stageName, id].filter(x => x).join('-');
46 }
47 /**
48 * Return the stage this construct is contained with, if available. If called
49 * on a nested stage, returns its parent.
50 *
51 */
52 static of(construct) {
53 return constructs_1.Node.of(construct).scopes.reverse().slice(1).find(Stage.isStage);
54 }
55 /**
56 * Test whether the given construct is a stage.
57 *
58 */
59 static isStage(x) {
60 return x !== null && typeof (x) === 'object' && STAGE_SYMBOL in x;
61 }
62 /**
63 * The cloud assembly output directory.
64 */
65 get outdir() {
66 return this._assemblyBuilder.outdir;
67 }
68 /**
69 * The cloud assembly asset output directory.
70 */
71 get assetOutdir() {
72 return this._assemblyBuilder.assetOutdir;
73 }
74 /**
75 * Artifact ID of the assembly if it is a nested stage. The root stage (app)
76 * will return an empty string.
77 *
78 * Derived from the construct path.
79 *
80 */
81 get artifactId() {
82 if (!this.node.path) {
83 return '';
84 }
85 return `assembly-${this.node.path.replace(/\//g, '-').replace(/^-+|-+$/g, '')}`;
86 }
87 /**
88 * Synthesize this stage into a cloud assembly.
89 *
90 * Once an assembly has been synthesized, it cannot be modified. Subsequent
91 * calls will return the same assembly.
92 */
93 synth(options = {}) {
94 try {
95 jsiiDeprecationWarnings._aws_cdk_core_StageSynthesisOptions(options);
96 }
97 catch (error) {
98 if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
99 Error.captureStackTrace(error, this.synth);
100 }
101 throw error;
102 }
103 if (!this.assembly || options.force) {
104 this.assembly = synthesis_1.synthesize(this, {
105 skipValidation: options.skipValidation,
106 validateOnSynthesis: options.validateOnSynthesis,
107 });
108 }
109 return this.assembly;
110 }
111 createBuilder(outdir) {
112 // cannot specify "outdir" if we are a nested stage
113 if (this.parentStage && outdir) {
114 throw new Error('"outdir" cannot be specified for nested stages');
115 }
116 // Need to determine fixed output directory already, because we must know where
117 // to write sub-assemblies (which must happen before we actually get to this app's
118 // synthesize() phase).
119 return this.parentStage
120 ? this.parentStage._assemblyBuilder.createNestedAssembly(this.artifactId, this.node.path)
121 : new cxapi.CloudAssemblyBuilder(outdir);
122 }
123}
124exports.Stage = Stage;
125_a = JSII_RTTI_SYMBOL_1;
126Stage[_a] = { fqn: "@aws-cdk/core.Stage", version: "1.204.0" };
127//# 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"]}
\No newline at end of file