@cloudcamp/aws-runtime
Version:
CloudCamp - Launch faster by building scalable infrastructure in few lines of code.
322 lines • 32.5 kB
JavaScript
"use strict";
var _a;
Object.defineProperty(exports, "__esModule", { value: true });
exports.App = void 0;
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
const cdk = require("aws-cdk-lib/core");
const _ = require("lodash");
const pipeline_1 = require("./pipeline");
const stage_1 = require("./stage");
const utils_1 = require("./utils");
const constants_1 = require("./constants");
const ssm = require("aws-cdk-lib/aws-ssm");
const pipelines = require("aws-cdk-lib/pipelines");
const stack_1 = require("./stack");
/**
* (experimental) App represents your application running in the cloud․ Every app contains one or more ``{@link Stack | Stacks}``, which you can use to add your own resources, like a ``{@link WebService | WebService}`` or a ``{@link Database | Database}``.
*
* An app can be as big or small as you like - from a single webserver to
* dozens of load-balanced instances serving different parts of your
* application.
*
* This example adds a ``{@link WebService | WebService}`` to the
* ``{@link App.production | production}`` stack:
*
* ```ts
* import { App, WebService } from "@cloudcamp/aws-runtime";
*
* let app = new App();
*
* new WebService(app.production, "prod-web", {
* dockerfile: "../Dockerfile"
* });
* ```
*
* @experimental
* @order 1
*/
class App extends cdk.App {
/**
* (experimental) Initialize your cloudcamp application.
*
* @experimental
* @remarks App is a singleton class - it is instantiated exactly once, before
* any other resources are created.
* @topic Initialization
*/
constructor() {
super({ autoSynth: true });
this.stages = new Map();
if (App.INSTANCE) {
throw new Error("App has already been instantiated.");
}
App.INSTANCE = this;
this.configuration = {
account: this.getContextOrThrow(constants_1.CONTEXT_KEY_ACCOUNT),
region: this.getContextOrThrow(constants_1.CONTEXT_KEY_REGION),
repository: this.getContextOrThrow(constants_1.CONTEXT_KEY_REPOSITORY),
name: this.getContextOrThrow(constants_1.CONTEXT_KEY_NAME),
branch: this.getContextOrThrow(constants_1.CONTEXT_KEY_BRANCH),
vpcId: this.getContextOrThrow(constants_1.CONTEXT_KEY_VPC),
repositoryTokenSecret: this.getContextOrThrow(constants_1.CONTEXT_REPOSITORY_TOKEN_SECRET),
};
cdk.Tags.of(this).add(constants_1.TAG_APP_NAME, this.configuration.name);
this.pipelineStack = this.setupCodePipeline();
//add pipelineStack to our global list of stacks
new ssm.StringParameter(this.pipelineStack, "ssm-stack", {
parameterName: `/cloudcamp/${App.instance.configuration.name}/_/stack/${_.kebabCase(this.pipelineStack.stackName)}`,
stringValue: this.pipelineStack.stackName,
});
// We need the name of the codepipeline for later use
new ssm.StringParameter(this.pipelineStack, "ssm-codepipeline", {
parameterName: `/cloudcamp/${this.configuration.name}/_/codepipeline`,
stringValue: this.pipelineStack.pipelineName,
});
// Also, we need to identify the pipeline stack
new ssm.StringParameter(this.pipelineStack, "ssm-pipeline-stack", {
parameterName: `/cloudcamp/${this.configuration.name}/_/pipeline-stack`,
stringValue: this.pipelineStack.stackName,
});
}
getContextOrThrow(key) {
let value = this.node.tryGetContext(key);
if (value == null) {
throw new Error("Missing config in cdk.json: " + key);
}
return value;
}
/**
* (experimental) Returns the global App singleton instance.
*
* Throws an exception if App has not been instantiated yet.
*
* @experimental
*/
static get instance() {
if (!App.INSTANCE) {
throw new Error("instance() called but App has not been instantiated yet.");
}
return App.INSTANCE;
}
/**
* (experimental) Use this stack for anything related to the network, for example DNS entries.
*
* @experimental
* @topic Stacks
* @remarks To deploy resources to the cloud, they are added to a ``Stack``.
* CloudCamp comes with three default stacks:
*/
get network() {
return this.getOrAddStage("network").stack;
}
/**
* (experimental) This stack can be used to create an environment for testing changes before deploying to production.
*
* @experimental
*/
get staging() {
return this.getOrAddStage("staging").stack;
}
/**
* (experimental) The production stack.
*
* @experimental
*/
get production() {
return this.getOrAddStage("production").stack;
}
/**
* (experimental) Get an existing stack by name.
*
* @param name The name of the stack.
* @experimental
* @topic Adding custom stacks
* @remarks In addition to the default stacks provided by cloudcamp, you can
* create your own stacks.
*/
getStack(name) {
try {
return this.getStage(name).stack;
}
catch (e) {
if (e.message.startsWith("Stage doesn't exist")) {
throw new Error("Stack doesn't exist: " + name);
}
else {
throw e;
}
}
}
/**
* (experimental) Add a new stack to your application.
*
* Pass a stack name to create a new, empty stack:
*
* ```ts
* void 0;
* import { App, WebService, Stack} from "@cloudcamp/aws-runtime";
* const app = new App();
* void 'show';
* const devStack = app.stack("development");
* ```
*
* @param name The name of the stack.
* @experimental
*/
stack(name) {
if (this.stages.get(name)) {
throw new Error("Stack already exists: " + name);
}
return this.getOrAddStage(name).stack;
}
getOrAddStage(id) {
let existingStage = this.stages.get(id);
if (existingStage) {
return existingStage;
}
let stage = new stage_1.Stage(this, _.upperFirst(_.camelCase(id)));
let stack = new stack_1.Stack(stage, id);
stage.stack = stack;
this.stages.set(id, stage);
return stage;
}
/**
* (experimental) Get an existing stage by name.
*
* Stages can be obtained by their name to modify their attributes. A common
* use case is to require manual approval to deploy to the production stage:
*
* ```ts
* const stage = app.getStage("production");
* stage.needsManualApproval = true;
* ```
*
* @param name The name of the stage.
* @experimental
* @topic Stages
* @remarks Stages are responsible for building your stacks. By default,
* CloudCamp creates a stage for each stack and gives it the same name. You
* can customize this behaviour by adding your own stages.
*/
getStage(name) {
if (name in ["network", "staging", "production"]) {
return this.getOrAddStage(name);
}
if (!this.stages.has(name)) {
throw new Error("Stage doesn't exist: " + name);
}
return this.stages.get(name);
}
/**
* (experimental) Add a new stage.
*
* This method can be used to add a stage with a custom ``Stack`` subclass.
* ```ts
* import { App, WebService, Stack } from "@cloudcamp/aws-runtime";
*
* class CustomStack extends Stack {
* constructor(scope: cdk.Construct, id: string, props: cdk.StackProps) {
* super(scope, id, props);
* new WebService(this, "web", {
* dockerfile: "../Dockerfile"
* });
* }
* }
*
* // later in the code...
* const app = new App();
* const stage = app.stage("dev");
* const stack = new CustomStack(stage, "dev");
*
* // stack is automatically set to the new stack we created
* if (stage.stack == stack) {
* // outputs "True!"
* console.log("True!");
* }
* ```
*
* @param name The name of the stage.
* @param stage An optional stage object.
* @experimental
*/
stage(name, stage) {
if (this.stages.has(name)) {
throw new Error("Stage already exists: " + name);
}
if (!stage) {
stage = new stage_1.Stage(this, _.upperFirst(_.camelCase(name)));
}
this.stages.set(name, stage);
return stage;
}
/**
* @experimental
* @ignore true
*/
get pipeline() {
return this.pipelineStack.pipeline;
}
setupCodePipeline() {
const repositoryUrl = this.configuration.repository;
const branch = this.configuration.branch;
let parsed = utils_1.parseRepositoryUrl(repositoryUrl);
let repositoryTokenSecretName = this.configuration.repositoryTokenSecret;
return new pipeline_1.PipelineStack(this, _.upperFirst(_.camelCase(this.configuration.name + "-pipeline")), {
appName: this.configuration.name,
repositoryTokenSecretName: repositoryTokenSecretName,
host: parsed.host,
owner: parsed.owner,
repo: parsed.repo,
branch: branch,
});
}
/**
* (experimental) Synthesize this stage into a cloud assembly.
*
* Once an assembly has been synthesized, it cannot be modified. Subsequent
* calls will return the same assembly.
*
* @experimental
* @ignore true
*/
synth(options) {
let names = Array.from(this.stages.keys());
names.sort((a, b) => {
switch (a) {
case "network":
if (b == "staging" || b == "production") {
return -1;
}
return 0;
case "staging":
if (b == "network") {
return 1;
}
if (b == "production") {
return -1;
}
return 0;
case "production":
if (b == "network" || b == "staging") {
return 1;
}
return 0;
default:
return 0;
}
});
for (let name of names) {
let stage = this.stages.get(name);
let pre = undefined;
if (stage.needsManualApproval) {
pre = [new pipelines.ManualApprovalStep("approve-" + name)];
}
this.pipeline.addStage(stage, { pre: pre });
}
return super.synth(options);
}
}
exports.App = App;
_a = JSII_RTTI_SYMBOL_1;
App[_a] = { fqn: "@cloudcamp/aws-runtime.App", version: "0.0.1" };
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"app.js","sourceRoot":"","sources":["../src/app.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cdk from \"aws-cdk-lib/core\";\nimport * as _ from \"lodash\";\nimport { PipelineStack } from \"./pipeline\";\nimport { Stage } from \"./stage\";\nimport { parseRepositoryUrl } from \"./utils\";\nimport {\n  CONTEXT_KEY_ACCOUNT,\n  CONTEXT_KEY_BRANCH,\n  CONTEXT_KEY_NAME,\n  CONTEXT_KEY_REGION,\n  CONTEXT_KEY_REPOSITORY,\n  CONTEXT_KEY_VPC,\n  CONTEXT_REPOSITORY_TOKEN_SECRET,\n  TAG_APP_NAME,\n} from \"./constants\";\nimport * as ssm from \"aws-cdk-lib/aws-ssm\";\nimport * as cxapi from \"aws-cdk-lib/cx-api\";\nimport * as pipelines from \"aws-cdk-lib/pipelines\";\nimport { Stack } from \"./stack\";\n\nexport interface Configuration {\n  readonly name: string;\n  readonly repository: string;\n  readonly account: string;\n  readonly region: string;\n  readonly branch: string;\n  readonly vpcId: string;\n  readonly repositoryTokenSecret: string;\n}\n\n                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                            \n\nexport class App extends cdk.App {\n  private static INSTANCE?: App;\n\n                                                                                                                                                                                                                        \n  constructor() {\n    super({ autoSynth: true });\n\n    if (App.INSTANCE) {\n      throw new Error(\"App has already been instantiated.\");\n    }\n\n    App.INSTANCE = this;\n\n    this.configuration = {\n      account: this.getContextOrThrow(CONTEXT_KEY_ACCOUNT),\n      region: this.getContextOrThrow(CONTEXT_KEY_REGION),\n      repository: this.getContextOrThrow(CONTEXT_KEY_REPOSITORY),\n      name: this.getContextOrThrow(CONTEXT_KEY_NAME),\n      branch: this.getContextOrThrow(CONTEXT_KEY_BRANCH),\n      vpcId: this.getContextOrThrow(CONTEXT_KEY_VPC),\n      repositoryTokenSecret: this.getContextOrThrow(\n        CONTEXT_REPOSITORY_TOKEN_SECRET\n      ),\n    };\n\n    cdk.Tags.of(this).add(TAG_APP_NAME, this.configuration.name);\n    this.pipelineStack = this.setupCodePipeline();\n\n    //add pipelineStack to our global list of stacks\n    new ssm.StringParameter(this.pipelineStack, \"ssm-stack\", {\n      parameterName: `/cloudcamp/${\n        App.instance.configuration.name\n      }/_/stack/${_.kebabCase(this.pipelineStack.stackName)}`,\n      stringValue: this.pipelineStack.stackName,\n    });\n\n    // We need the name of the codepipeline for later use\n    new ssm.StringParameter(this.pipelineStack, \"ssm-codepipeline\", {\n      parameterName: `/cloudcamp/${this.configuration.name}/_/codepipeline`,\n      stringValue: this.pipelineStack.pipelineName,\n    });\n\n    // Also, we need to identify the pipeline stack\n    new ssm.StringParameter(this.pipelineStack, \"ssm-pipeline-stack\", {\n      parameterName: `/cloudcamp/${this.configuration.name}/_/pipeline-stack`,\n      stringValue: this.pipelineStack.stackName,\n    });\n  }\n\n  private getContextOrThrow(key: string): string {\n    let value: string = this.node.tryGetContext(key);\n    if (value == null) {\n      throw new Error(\"Missing config in cdk.json: \" + key);\n    }\n    return value;\n  }\n\n                                                                                                                               \n  public static get instance(): App {\n    if (!App.INSTANCE) {\n      throw new Error(\n        \"instance() called but App has not been instantiated yet.\"\n      );\n    }\n    return App.INSTANCE;\n  }\n\n                                                                                                                                                                                                                                                         \n  get network(): Stack {\n    return this.getOrAddStage(\"network\").stack;\n  }\n\n                                                                                                                         \n  get staging(): Stack {\n    return this.getOrAddStage(\"staging\").stack;\n  }\n\n                                      \n  get production(): Stack {\n    return this.getOrAddStage(\"production\").stack;\n  }\n\n                                                                                                                                                                                                                                                   \n  public getStack(name: string): Stack {\n    try {\n      return this.getStage(name).stack;\n    } catch (e) {\n      if ((e as Error).message.startsWith(\"Stage doesn't exist\")) {\n        throw new Error(\"Stack doesn't exist: \" + name);\n      } else {\n        throw e;\n      }\n    }\n  }\n\n                                                                                                                                                                                                                                                                                                                                                                               \n  public stack(name: string): Stack {\n    if (this.stages.get(name)) {\n      throw new Error(\"Stack already exists: \" + name);\n    }\n    return this.getOrAddStage(name).stack;\n  }\n\n  private stages: Map<string, Stage> = new Map();\n\n  private getOrAddStage(id: string) {\n    let existingStage = this.stages.get(id);\n    if (existingStage) {\n      return existingStage;\n    }\n    let stage = new Stage(this, _.upperFirst(_.camelCase(id)));\n    let stack = new Stack(stage, id);\n    stage.stack = stack;\n    this.stages.set(id, stage);\n\n    return stage;\n  }\n\n                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                 \n\n  public getStage(name: string): Stage {\n    if (name in [\"network\", \"staging\", \"production\"]) {\n      return this.getOrAddStage(name);\n    }\n    if (!this.stages.has(name)) {\n      throw new Error(\"Stage doesn't exist: \" + name);\n    }\n    return this.stages.get(name)!;\n  }\n\n                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                      \n  public stage(name: string, stage?: Stage): Stage {\n    if (this.stages.has(name)) {\n      throw new Error(\"Stage already exists: \" + name);\n    }\n    if (!stage) {\n      stage = new Stage(this, _.upperFirst(_.camelCase(name)));\n    }\n    this.stages.set(name, stage);\n    return stage;\n  }\n\n                        \n  configuration: Configuration;\n\n  private pipelineStack: PipelineStack;\n                        \n  public get pipeline() {\n    return this.pipelineStack.pipeline;\n  }\n\n  private setupCodePipeline(): PipelineStack {\n    const repositoryUrl = this.configuration.repository;\n    const branch = this.configuration.branch;\n    let parsed = parseRepositoryUrl(repositoryUrl);\n    let repositoryTokenSecretName = this.configuration.repositoryTokenSecret;\n    return new PipelineStack(\n      this,\n      _.upperFirst(_.camelCase(this.configuration.name + \"-pipeline\")),\n      {\n        appName: this.configuration.name,\n        repositoryTokenSecretName: repositoryTokenSecretName,\n        host: parsed.host,\n        owner: parsed.owner,\n        repo: parsed.repo,\n        branch: branch,\n      }\n    );\n  }\n\n                        \n  synth(options?: cdk.StageSynthesisOptions): cxapi.CloudAssembly {\n    let names = Array.from(this.stages.keys());\n    names.sort((a, b) => {\n      switch (a) {\n        case \"network\":\n          if (b == \"staging\" || b == \"production\") {\n            return -1;\n          }\n          return 0;\n        case \"staging\":\n          if (b == \"network\") {\n            return 1;\n          }\n          if (b == \"production\") {\n            return -1;\n          }\n          return 0;\n        case \"production\":\n          if (b == \"network\" || b == \"staging\") {\n            return 1;\n          }\n          return 0;\n        default:\n          return 0;\n      }\n    });\n    for (let name of names) {\n      let stage = this.stages.get(name)!;\n      let pre = undefined;\n      if (stage.needsManualApproval) {\n        pre = [new pipelines.ManualApprovalStep(\"approve-\" + name)];\n      }\n      this.pipeline.addStage(stage, { pre: pre });\n    }\n    return super.synth(options);\n  }\n}\n"]}