projen
Version:
CDK for software projects
99 lines • 14.2 kB
JavaScript
;
var _a;
Object.defineProperty(exports, "__esModule", { value: true });
exports.Projects = void 0;
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
const path = require("path");
const vm = require("vm");
const inventory_1 = require("./inventory");
const render_options_1 = require("./javascript/render-options");
const option_hints_1 = require("./option-hints");
/**
* Programmatic API for projen.
*/
class Projects {
/**
* Creates a new project with defaults.
*
* This function creates the project type in-process (with in VM) and calls
* `.synth()` on it (if `options.synth` is not `false`).
*
* At the moment, it also generates a `.projenrc.js` file with the same code
* that was just executed. In the future, this will also be done by the project
* type, so we can easily support multiple languages of projenrc.
*
* An environment variable (PROJEN_CREATE_PROJECT=true) is set within the VM
* so that custom project types can detect whether the current synthesis is the
* result of a new project creation (and take additional steps accordingly)
*/
static createProject(options) {
createProject(options);
}
constructor() { }
}
exports.Projects = Projects;
_a = JSII_RTTI_SYMBOL_1;
Projects[_a] = { fqn: "projen.Projects", version: "0.95.2" };
function resolveModulePath(moduleName) {
// Default project resolution location
if (moduleName === "projen") {
return "./index";
}
// External projects need to load the module from the modules directory
try {
return path.dirname(require.resolve(path.join(moduleName, "package.json"), {
paths: [process.cwd()],
}));
}
catch (err) {
throw new Error(`External project module '${moduleName}' could not be resolved.`);
}
}
function createProject(opts) {
const projectType = (0, inventory_1.resolveProjectType)(opts.projectFqn);
const mod = resolveModulePath(projectType.moduleName);
// "dir" is exposed as a top-level option to require users to specify a value for it
opts.projectOptions.outdir = opts.dir;
// Generated a random name space for imports used by options
// This is so we can keep the top-level namespace as clean as possible
const optionsImports = "_options" + Math.random().toString(36).slice(2);
// pass the FQN of the project type to the project initializer so it can
// generate the projenrc file.
const { renderedOptions, imports } = (0, render_options_1.renderJavaScriptOptions)({
bootstrap: true,
comments: opts.optionHints ?? option_hints_1.InitProjectOptionHints.FEATURED,
type: projectType,
args: opts.projectOptions,
omitFromBootstrap: ["outdir"],
prefixImports: optionsImports,
});
const initProjectCode = new Array();
// generate a random variable name because jest tests appear to share
// VM contexts, causing
//
// > SyntaxError: Identifier 'project' has already been declared
//
// errors if this isn't unique
const varName = "project" + Math.random().toString(36).slice(2);
initProjectCode.push(`const ${varName} = new ${projectType.typename}(${renderedOptions});`);
if (opts.synth ?? true) {
initProjectCode.push(`${varName}.synth();`);
}
// eslint-disable-next-line @typescript-eslint/no-require-imports
const mainModule = require(mod);
const ctx = vm.createContext({
...mainModule,
[optionsImports]: {
...imports.modules.reduce((optionsContext, currentModule) => ({
...optionsContext,
// eslint-disable-next-line @typescript-eslint/no-require-imports
[currentModule]: require(resolveModulePath(currentModule)),
}), {}),
},
});
const postSynth = opts.post ?? true;
process.env.PROJEN_DISABLE_POST = (!postSynth).toString();
process.env.PROJEN_CREATE_PROJECT = "true";
vm.runInContext(initProjectCode.join("\n"), ctx);
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"projects.js","sourceRoot":"","sources":["../src/projects.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from \"path\";\nimport * as vm from \"vm\";\nimport { resolveProjectType } from \"./inventory\";\nimport { renderJavaScriptOptions } from \"./javascript/render-options\";\nimport { InitProjectOptionHints } from \"./option-hints\";\n\nexport interface CreateProjectOptions {\n  /**\n   * Directory that the project will be generated in.\n   */\n  readonly dir: string;\n\n  /**\n   * Fully-qualified name of the project type (usually formatted\n   * as `projen.module.ProjectType`).\n   * @example `projen.typescript.TypescriptProject`\n   */\n  readonly projectFqn: string;\n\n  /**\n   * Project options. Only JSON-like values can be passed in (strings,\n   * booleans, numbers, enums, arrays, and objects that are not\n   * derived from classes).\n   *\n   * Consult the API reference of the project type you are generating for\n   * information about what fields and types are available.\n   */\n  readonly projectOptions: Record<string, any>;\n\n  /**\n   * Should we render commented-out default options in the projenrc file?\n   * Does not apply to projenrc.json files.\n   *\n   * @default InitProjectOptionHints.FEATURED\n   */\n  readonly optionHints?: InitProjectOptionHints;\n\n  /**\n   * Should we call `project.synth()` or instantiate the project (could still\n   * have side-effects) and render the .projenrc file.\n   *\n   * @default true\n   */\n  readonly synth?: boolean;\n\n  /**\n   * Should we execute post synthesis hooks? (usually package manager install).\n   *\n   * @default true\n   */\n  readonly post?: boolean;\n}\n\n/**\n * Programmatic API for projen.\n */\nexport class Projects {\n  /**\n   * Creates a new project with defaults.\n   *\n   * This function creates the project type in-process (with in VM) and calls\n   * `.synth()` on it (if `options.synth` is not `false`).\n   *\n   * At the moment, it also generates a `.projenrc.js` file with the same code\n   * that was just executed. In the future, this will also be done by the project\n   * type, so we can easily support multiple languages of projenrc.\n   *\n   * An environment variable (PROJEN_CREATE_PROJECT=true) is set within the VM\n   * so that custom project types can detect whether the current synthesis is the\n   * result of a new project creation (and take additional steps accordingly)\n   */\n  public static createProject(options: CreateProjectOptions) {\n    createProject(options);\n  }\n\n  private constructor() {}\n}\n\nfunction resolveModulePath(moduleName: string) {\n  // Default project resolution location\n  if (moduleName === \"projen\") {\n    return \"./index\";\n  }\n\n  // External projects need to load the module from the modules directory\n  try {\n    return path.dirname(\n      require.resolve(path.join(moduleName, \"package.json\"), {\n        paths: [process.cwd()],\n      })\n    );\n  } catch (err) {\n    throw new Error(\n      `External project module '${moduleName}' could not be resolved.`\n    );\n  }\n}\n\nfunction createProject(opts: CreateProjectOptions) {\n  const projectType = resolveProjectType(opts.projectFqn);\n  const mod = resolveModulePath(projectType.moduleName);\n\n  // \"dir\" is exposed as a top-level option to require users to specify a value for it\n  opts.projectOptions.outdir = opts.dir;\n\n  // Generated a random name space for imports used by options\n  // This is so we can keep the top-level namespace as clean as possible\n  const optionsImports = \"_options\" + Math.random().toString(36).slice(2);\n\n  // pass the FQN of the project type to the project initializer so it can\n  // generate the projenrc file.\n  const { renderedOptions, imports } = renderJavaScriptOptions({\n    bootstrap: true,\n    comments: opts.optionHints ?? InitProjectOptionHints.FEATURED,\n    type: projectType,\n    args: opts.projectOptions,\n    omitFromBootstrap: [\"outdir\"],\n    prefixImports: optionsImports,\n  });\n\n  const initProjectCode = new Array<string>();\n\n  // generate a random variable name because jest tests appear to share\n  // VM contexts, causing\n  //\n  // > SyntaxError: Identifier 'project' has already been declared\n  //\n  // errors if this isn't unique\n  const varName = \"project\" + Math.random().toString(36).slice(2);\n  initProjectCode.push(\n    `const ${varName} = new ${projectType.typename}(${renderedOptions});`\n  );\n\n  if (opts.synth ?? true) {\n    initProjectCode.push(`${varName}.synth();`);\n  }\n\n  // eslint-disable-next-line @typescript-eslint/no-require-imports\n  const mainModule = require(mod);\n  const ctx = vm.createContext({\n    ...mainModule,\n    [optionsImports]: {\n      ...imports.modules.reduce(\n        (optionsContext, currentModule) => ({\n          ...optionsContext,\n          // eslint-disable-next-line @typescript-eslint/no-require-imports\n          [currentModule]: require(resolveModulePath(currentModule)),\n        }),\n        {}\n      ),\n    },\n  });\n\n  const postSynth = opts.post ?? true;\n  process.env.PROJEN_DISABLE_POST = (!postSynth).toString();\n  process.env.PROJEN_CREATE_PROJECT = \"true\";\n  vm.runInContext(initProjectCode.join(\"\\n\"), ctx);\n}\n"]}