UNPKG

@cloudcamp/aws-runtime

Version:

CloudCamp - Launch faster by building scalable infrastructure in few lines of code.

322 lines 32.5 kB
"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"]}