UNPKG

projen

Version:

CDK for software projects

118 lines 13.9 kB
"use strict"; var _a; Object.defineProperty(exports, "__esModule", { value: true }); exports.DevContainer = void 0; const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti"); const component_1 = require("../component"); const json_1 = require("../json"); /** * Relative path of devcontainer file to generate */ const DEVCONTAINER_FILE = ".devcontainer.json"; /** * A development environment running VSCode in a container; used by GitHub * codespaces. */ class DevContainer extends component_1.Component { constructor(project, options = {}) { super(project); this.postCreateTasks = new Array(); this.ports = new Array(); this.vscodeExtensions = new Array(); this.features = new Array(); this.dockerImage = options?.dockerImage; if (options?.tasks) { for (const task of options.tasks) { this.addTasks(task); } } if (options?.ports) { this.addPorts(...options.ports); } if (options?.vscodeExtensions) { this.addVscodeExtensions(...options.vscodeExtensions); } if (options.features) { this.addFeatures(...options.features); } this.config = { image: () => this.dockerImage?.image, build: { dockerfile: () => this.dockerImage?.dockerFile, }, postCreateCommand: () => this.renderTasks(), forwardPorts: this.ports, extensions: this.vscodeExtensions, features: () => this.renderFeatures(), }; new json_1.JsonFile(this.project, DEVCONTAINER_FILE, { obj: this.config, omitEmpty: true, }); } addDockerImage(image) { if (this.dockerImage) { throw new Error("dockerImage cannot be redefined."); } this.dockerImage = image; } /** * Adds tasks to run when the container starts. Tasks will be run in sequence. * * @param tasks The new tasks */ addTasks(...tasks) { this.postCreateTasks.push(...tasks); } /** * Adds ports that should be exposed (forwarded) from the container. * * @param ports The new ports */ addPorts(...ports) { this.ports.push(...ports); } /** * Adds a list of VSCode extensions that should be automatically installed * in the container. * * @param extensions The extension IDs */ addVscodeExtensions(...extensions) { this.vscodeExtensions.push(...extensions); } /** * Adds a list of VSCode features that should be automatically installed * in the container. * * @param features featureName and version(optional default: latest) */ addFeatures(...features) { this.features.push(...features); } renderTasks() { if (this.postCreateTasks.length !== 0) { return this.postCreateTasks .map((task) => `( npx projen ${task.name} )`) .join(" && "); } else { return undefined; } } renderFeatures() { if (this.features.length === 0) { return undefined; } return this.features.reduce((pv, feature) => { pv[feature.name] = { version: feature.version ?? "latest", }; return pv; }, {}); } } exports.DevContainer = DevContainer; _a = JSII_RTTI_SYMBOL_1; DevContainer[_a] = { fqn: "projen.vscode.DevContainer", version: "0.95.2" }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"devcontainer.js","sourceRoot":"","sources":["../../src/vscode/devcontainer.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component } from \"../component\";\nimport {\n  DevEnvironmentDockerImage,\n  DevEnvironmentOptions,\n  IDevEnvironment,\n} from \"../dev-env\";\nimport { JsonFile } from \"../json\";\nimport { Project } from \"../project\";\nimport { Task } from \"../task\";\n\n/**\n * Relative path of devcontainer file to generate\n */\nconst DEVCONTAINER_FILE = \".devcontainer.json\";\n\n/**\n * devcontainer features options\n * @see https://containers.dev/implementors/features/#devcontainer-json-properties\n */\nexport interface DevContainerFeature {\n  /**\n   * feature name\n   */\n  readonly name: string;\n  /**\n   * feature version\n   * @default latest\n   */\n  readonly version?: string;\n}\n\n/**\n * Constructor options for the DevContainer component.\n *\n * The default docker image used for GitHub Codespaces is defined here:\n * @see https://github.com/microsoft/vscode-dev-containers/tree/master/containers/codespaces-linux\n */\nexport interface DevContainerOptions extends DevEnvironmentOptions {\n  /**\n   * An array of VSCode features that specify the features that should be\n   * installed inside the container when it is created.\n   */\n  readonly features?: DevContainerFeature[];\n}\n\nexport interface IDevContainerEnvironment extends IDevEnvironment {\n  /**\n   * Adds a list of VSCode features that should be automatically installed\n   * in the container.\n   *\n   * @param features featureName and version(optional default: latest)\n   */\n  addFeatures(...features: DevContainerFeature[]): void;\n}\n\n/**\n * A development environment running VSCode in a container; used by GitHub\n * codespaces.\n */\nexport class DevContainer\n  extends Component\n  implements IDevContainerEnvironment\n{\n  private dockerImage: DevEnvironmentDockerImage | undefined;\n  private readonly postCreateTasks: Task[];\n  private readonly ports: string[];\n  private readonly vscodeExtensions: string[];\n  private readonly features: DevContainerFeature[];\n\n  /**\n   * Direct access to the devcontainer configuration (escape hatch)\n   */\n  public readonly config: any;\n\n  constructor(project: Project, options: DevContainerOptions = {}) {\n    super(project);\n\n    this.postCreateTasks = new Array<Task>();\n    this.ports = new Array<string>();\n    this.vscodeExtensions = new Array<string>();\n    this.features = new Array<DevContainerFeature>();\n\n    this.dockerImage = options?.dockerImage;\n\n    if (options?.tasks) {\n      for (const task of options.tasks) {\n        this.addTasks(task);\n      }\n    }\n    if (options?.ports) {\n      this.addPorts(...options.ports);\n    }\n    if (options?.vscodeExtensions) {\n      this.addVscodeExtensions(...options.vscodeExtensions);\n    }\n    if (options.features) {\n      this.addFeatures(...options.features);\n    }\n\n    this.config = {\n      image: () => this.dockerImage?.image,\n      build: {\n        dockerfile: () => this.dockerImage?.dockerFile,\n      },\n      postCreateCommand: () => this.renderTasks(),\n      forwardPorts: this.ports,\n      extensions: this.vscodeExtensions,\n      features: () => this.renderFeatures(),\n    };\n\n    new JsonFile(this.project, DEVCONTAINER_FILE, {\n      obj: this.config,\n      omitEmpty: true,\n    });\n  }\n\n  public addDockerImage(image: DevEnvironmentDockerImage) {\n    if (this.dockerImage) {\n      throw new Error(\"dockerImage cannot be redefined.\");\n    }\n    this.dockerImage = image;\n  }\n\n  /**\n   * Adds tasks to run when the container starts. Tasks will be run in sequence.\n   *\n   * @param tasks The new tasks\n   */\n  public addTasks(...tasks: Task[]) {\n    this.postCreateTasks.push(...tasks);\n  }\n\n  /**\n   * Adds ports that should be exposed (forwarded) from the container.\n   *\n   * @param ports The new ports\n   */\n  public addPorts(...ports: string[]) {\n    this.ports.push(...ports);\n  }\n\n  /**\n   * Adds a list of VSCode extensions that should be automatically installed\n   * in the container.\n   *\n   * @param extensions The extension IDs\n   */\n  public addVscodeExtensions(...extensions: string[]) {\n    this.vscodeExtensions.push(...extensions);\n  }\n\n  /**\n   * Adds a list of VSCode features that should be automatically installed\n   * in the container.\n   *\n   * @param features featureName and version(optional default: latest)\n   */\n  public addFeatures(...features: DevContainerFeature[]): void {\n    this.features.push(...features);\n  }\n\n  private renderTasks() {\n    if (this.postCreateTasks.length !== 0) {\n      return this.postCreateTasks\n        .map((task) => `( npx projen ${task.name} )`)\n        .join(\" && \");\n    } else {\n      return undefined;\n    }\n  }\n\n  private renderFeatures() {\n    if (this.features.length === 0) {\n      return undefined;\n    }\n\n    return this.features.reduce<{\n      [key: string]: { version: string };\n    }>((pv, feature) => {\n      pv[feature.name] = {\n        version: feature.version ?? \"latest\",\n      };\n      return pv;\n    }, {});\n  }\n}\n"]}