projen
Version:
CDK for software projects
118 lines • 13.9 kB
JavaScript
"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"]}