projen
Version:
CDK for software projects
224 lines • 32.4 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", { value: true });
exports.ModuleImports = void 0;
exports.renderProjenInitOptions = renderProjenInitOptions;
exports.resolveInitProject = resolveInitProject;
exports.renderJavaScriptOptions = renderJavaScriptOptions;
const inventory = require("../inventory");
const option_hints_1 = require("../option-hints");
const PROJEN_NEW = "__new__";
const TAB = makePadding(2);
/**
* Renders options as if the project was created via `projen new` (embeds the __new__ field).
*/
function renderProjenInitOptions(fqn, args, comments = option_hints_1.InitProjectOptionHints.NONE) {
return {
...args,
[PROJEN_NEW]: { fqn, args, comments },
};
}
function resolveInitProject(opts) {
const f = opts[PROJEN_NEW];
if (!f) {
return undefined;
}
const type = inventory.resolveProjectType(f.fqn);
if (!type) {
throw new Error(`unable to resolve project type for ${f.fqn}`);
}
return {
args: f.args,
fqn: f.fqn,
type: type,
comments: f.comments,
};
}
class ModuleImports {
constructor() {
this.imports = new Map();
}
/**
* Add a named import from a module
*/
add(moduleName, importName) {
const moduleImports = this.imports.get(moduleName) ?? new Set();
moduleImports.add(importName);
this.imports.set(moduleName, moduleImports);
}
/**
* Get all named imports for a module
*/
get(moduleName) {
const moduleImports = this.imports.get(moduleName) ?? new Set();
return Array.from(moduleImports);
}
/**
* Get a list of all used modules
*/
get modules() {
return Array.from(this.imports.keys());
}
/**
* Return all imports as ESM import statements
*/
asEsmImports() {
return this.all().map(([moduleName, namedImports]) => `import { ${[...namedImports].join(", ")} } from "${moduleName}";`);
}
/**
* Return all imports as CJS require statements
*/
asCjsRequire() {
return this.all().map(([moduleName, namedImports]) => `const { ${[...namedImports].join(", ")} } = require("${moduleName}");`);
}
all() {
const allImports = Object.fromEntries(this.imports);
return Object.entries(allImports).map(([key, value]) => [
key,
Array.from(value).sort(),
]);
}
}
exports.ModuleImports = ModuleImports;
/**
* Prints all parameters that can be used in a project type, alongside their descriptions.
*
* Parameters in `params` that aren't undefined are rendered as defaults,
* while all other parameters are rendered as commented out.
*
* Returns the printed output and a set of required imports as an object
* in the form { options, imports }.
*/
function renderJavaScriptOptions(opts) {
const renders = {};
const optionsWithDefaults = [];
const allImports = new ModuleImports();
for (const option of opts.type.options) {
if (option.deprecated) {
continue;
}
const optionName = option.name;
if (opts.args[optionName] !== undefined) {
const arg = opts.args[optionName];
const { js, moduleName, importName } = renderArgAsJavaScript(arg, option);
renders[optionName] = `${optionName}: ${js},`;
optionsWithDefaults.push(optionName);
if (moduleName && importName) {
allImports.add(moduleName, importName);
if (opts.prefixImports) {
const prefix = `${opts.prefixImports}["${moduleName}"].`;
renders[optionName] = `${optionName}: ${prefix}${js},`;
}
}
}
else {
const defaultValue = option.default?.startsWith("-")
? undefined
: option.default ?? undefined;
renders[optionName] = `// ${optionName}: ${defaultValue},`;
}
}
const bootstrap = opts.bootstrap ?? false;
if (bootstrap) {
for (const arg of opts.omitFromBootstrap ?? []) {
delete opts.args[arg];
}
renders[PROJEN_NEW] = `${PROJEN_NEW}: ${JSON.stringify({
args: opts.args,
fqn: opts.type.fqn,
comments: opts.comments,
})},`;
optionsWithDefaults.push(PROJEN_NEW);
}
// generate rendering
const result = [];
result.push("{");
// render options with defaults
optionsWithDefaults.sort();
for (const optionName of optionsWithDefaults) {
result.push(`${TAB}${renders[optionName]}`);
}
if (result.length > 1) {
result.push("");
}
// render options without defaults as comments
if (opts.comments === option_hints_1.InitProjectOptionHints.ALL) {
const options = opts.type.options.filter((opt) => !opt.deprecated && opts.args[opt.name] === undefined);
result.push(...renderCommentedOptionsByModule(renders, options));
}
else if (opts.comments === option_hints_1.InitProjectOptionHints.FEATURED) {
const options = opts.type.options.filter((opt) => !opt.deprecated && opts.args[opt.name] === undefined && opt.featured);
result.push(...renderCommentedOptionsInOrder(renders, options));
}
else if (opts.comments === option_hints_1.InitProjectOptionHints.NONE) {
// don't render any extra options
}
if (result[result.length - 1] === "") {
result.pop();
}
result.push("}");
return { renderedOptions: result.join("\n"), imports: allImports };
}
function renderCommentedOptionsByModule(renders, options) {
const optionsByModule = {};
for (const option of options) {
const parentModule = option.parent;
optionsByModule[parentModule] = optionsByModule[parentModule] ?? [];
optionsByModule[parentModule].push(option);
}
for (const parentModule in optionsByModule) {
optionsByModule[parentModule].sort((o1, o2) => o1.name.localeCompare(o2.name));
}
const result = [];
const marginSize = Math.max(...options.map((opt) => renders[opt.name].length));
for (const [moduleName, optionGroup] of Object.entries(optionsByModule).sort()) {
result.push(`${TAB}/* ${moduleName} */`);
for (const option of optionGroup) {
const paramRender = renders[option.name];
const docstring = option.docs || "No documentation found.";
result.push(`${TAB}${paramRender}${makePadding(marginSize - paramRender.length + 2)}/* ${docstring} */`);
}
result.push("");
}
return result;
}
function renderCommentedOptionsInOrder(renders, options) {
const result = [];
const marginSize = Math.max(...options.map((opt) => renders[opt.name].length));
for (const option of options) {
const paramRender = renders[option.name];
const docstring = option.docs || "No documentation found.";
result.push(`${TAB}${paramRender}${makePadding(marginSize - paramRender.length + 2)}/* ${docstring} */`);
}
return result;
}
/**
* Renders a value as a JavaScript value, converting strings to enums where
* appropriate. The type must be JSON-like (string, number, boolean, array,
* enum, or JSON object).
*
* Returns a JavaScript expression as a string, and the names of any
* necessary imports.
*/
function renderArgAsJavaScript(arg, option) {
if (option.kind === "enum") {
if (!option.fqn) {
throw new Error(`fqn field is missing from enum option ${option.name}`);
}
const parts = option.fqn.split("."); // -> ['projen', 'web', 'MyEnum']
const enumChoice = String(arg).toUpperCase().replace(/-/g, "_"); // custom-value -> CUSTOM_VALUE
const js = `${parts.slice(1).join(".")}.${enumChoice}`; // -> web.MyEnum.CUSTOM_VALUE
const moduleName = parts[0]; // -> projen
const importName = parts[1]; // -> web
return { js, moduleName, importName };
}
else if (option.jsonLike) {
return { js: JSON.stringify(arg) };
}
else {
throw new Error(`Unexpected option ${option.name} - cannot render a value for this option because it does not have a JSON-like type.`);
}
}
function makePadding(paddingLength) {
return " ".repeat(paddingLength);
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"render-options.js","sourceRoot":"","sources":["../../src/javascript/render-options.ts"],"names":[],"mappings":"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","sourcesContent":["import * as inventory from \"../inventory\";\nimport { InitProjectOptionHints } from \"../option-hints\";\n\nconst PROJEN_NEW = \"__new__\";\nconst TAB = makePadding(2);\n\n/**\n * Options for `renderProjectOptions`.\n */\nexport interface RenderProjectOptions {\n  /**\n   * The project type to render.\n   */\n  readonly type: inventory.ProjectType;\n\n  /**\n   * Project arguments as passed to `projen new`.\n   */\n  readonly args: Record<string, any>;\n\n  /**\n   * Include commented out options.\n   * @default InitProjectOptionHints.FEATURED\n   */\n  readonly comments?: InitProjectOptionHints;\n\n  /**\n   * Inject a `__new__` attribute to the project constructor with a stringified\n   * version of the project parameters and a `jsiiFqn` attribute that includes\n   * the FQN of the project type. This is needed in order to generate initial\n   * projenrc files.\n   *\n   * @default false\n   */\n  readonly bootstrap?: boolean;\n\n  /**\n   * A list of fields to omit from the initial projenrc file.\n   * @default - none\n   */\n  readonly omitFromBootstrap?: string[];\n\n  /**\n   * Prefix all imports with this string and the full module name\n   * This is required when executing options code in a vm\n   *\n   * @default - only use submodule as prefix\n   */\n  readonly prefixImports?: string;\n}\n\n/**\n * Information passed from `projen new` to the project object when the project\n * is first created. It is used to generate projenrc files in various languages.\n */\ninterface ProjenInit {\n  /**\n   * The JSII FQN of the project type.\n   */\n  readonly fqn: string;\n\n  /**\n   * Initial arguments passed to `projen new`.\n   */\n  readonly args: Record<string, any>;\n\n  /**\n   * Include commented out options. Does not apply to projenrc.json files.\n   */\n  readonly comments: InitProjectOptionHints;\n}\n\n/**\n * Renders options as if the project was created via `projen new` (embeds the __new__ field).\n */\nexport function renderProjenInitOptions(\n  fqn: string,\n  args: Record<string, any>,\n  comments: InitProjectOptionHints = InitProjectOptionHints.NONE\n): any {\n  return {\n    ...args,\n    [PROJEN_NEW]: { fqn, args, comments } as ProjenInit,\n  };\n}\n\nexport function resolveInitProject(opts: any) {\n  const f = opts[PROJEN_NEW] as ProjenInit;\n  if (!f) {\n    return undefined;\n  }\n\n  const type = inventory.resolveProjectType(f.fqn);\n  if (!type) {\n    throw new Error(`unable to resolve project type for ${f.fqn}`);\n  }\n  return {\n    args: f.args,\n    fqn: f.fqn,\n    type: type,\n    comments: f.comments,\n  };\n}\n\nexport class ModuleImports {\n  private imports: Map<string, Set<string>> = new Map();\n\n  /**\n   * Add a named import from a module\n   */\n  public add(moduleName: string, importName: string) {\n    const moduleImports = this.imports.get(moduleName) ?? new Set();\n    moduleImports.add(importName);\n    this.imports.set(moduleName, moduleImports);\n  }\n\n  /**\n   * Get all named imports for a module\n   */\n  public get(moduleName: string): string[] {\n    const moduleImports = this.imports.get(moduleName) ?? new Set();\n    return Array.from(moduleImports);\n  }\n\n  /**\n   * Get a list of all used modules\n   */\n  public get modules(): string[] {\n    return Array.from(this.imports.keys());\n  }\n\n  /**\n   * Return all imports as ESM import statements\n   */\n  public asEsmImports(): string[] {\n    return this.all().map(\n      ([moduleName, namedImports]) =>\n        `import { ${[...namedImports].join(\", \")} } from \"${moduleName}\";`\n    );\n  }\n\n  /**\n   * Return all imports as CJS require statements\n   */\n  public asCjsRequire(): string[] {\n    return this.all().map(\n      ([moduleName, namedImports]) =>\n        `const { ${[...namedImports].join(\", \")} } = require(\"${moduleName}\");`\n    );\n  }\n\n  private all(): Array<[string, string[]]> {\n    const allImports = Object.fromEntries(this.imports);\n    return Object.entries(allImports).map(([key, value]) => [\n      key,\n      Array.from(value).sort(),\n    ]);\n  }\n}\n\n/**\n * Prints all parameters that can be used in a project type, alongside their descriptions.\n *\n * Parameters in `params` that aren't undefined are rendered as defaults,\n * while all other parameters are rendered as commented out.\n *\n * Returns the printed output and a set of required imports as an object\n * in the form { options, imports }.\n */\nexport function renderJavaScriptOptions(opts: RenderProjectOptions): {\n  renderedOptions: string;\n  imports: ModuleImports;\n} {\n  const renders: Record<string, string> = {};\n  const optionsWithDefaults: string[] = [];\n  const allImports = new ModuleImports();\n\n  for (const option of opts.type.options) {\n    if (option.deprecated) {\n      continue;\n    }\n\n    const optionName = option.name;\n\n    if (opts.args[optionName] !== undefined) {\n      const arg = opts.args[optionName];\n      const { js, moduleName, importName } = renderArgAsJavaScript(arg, option);\n\n      renders[optionName] = `${optionName}: ${js},`;\n      optionsWithDefaults.push(optionName);\n\n      if (moduleName && importName) {\n        allImports.add(moduleName, importName);\n        if (opts.prefixImports) {\n          const prefix = `${opts.prefixImports}[\"${moduleName}\"].`;\n          renders[optionName] = `${optionName}: ${prefix}${js},`;\n        }\n      }\n    } else {\n      const defaultValue = option.default?.startsWith(\"-\")\n        ? undefined\n        : option.default ?? undefined;\n      renders[optionName] = `// ${optionName}: ${defaultValue},`;\n    }\n  }\n\n  const bootstrap = opts.bootstrap ?? false;\n  if (bootstrap) {\n    for (const arg of opts.omitFromBootstrap ?? []) {\n      delete opts.args[arg];\n    }\n    renders[PROJEN_NEW] = `${PROJEN_NEW}: ${JSON.stringify({\n      args: opts.args,\n      fqn: opts.type.fqn,\n      comments: opts.comments,\n    } as ProjenInit)},`;\n    optionsWithDefaults.push(PROJEN_NEW);\n  }\n\n  // generate rendering\n  const result: string[] = [];\n  result.push(\"{\");\n\n  // render options with defaults\n  optionsWithDefaults.sort();\n  for (const optionName of optionsWithDefaults) {\n    result.push(`${TAB}${renders[optionName]}`);\n  }\n  if (result.length > 1) {\n    result.push(\"\");\n  }\n\n  // render options without defaults as comments\n  if (opts.comments === InitProjectOptionHints.ALL) {\n    const options = opts.type.options.filter(\n      (opt) => !opt.deprecated && opts.args[opt.name] === undefined\n    );\n    result.push(...renderCommentedOptionsByModule(renders, options));\n  } else if (opts.comments === InitProjectOptionHints.FEATURED) {\n    const options = opts.type.options.filter(\n      (opt) =>\n        !opt.deprecated && opts.args[opt.name] === undefined && opt.featured\n    );\n    result.push(...renderCommentedOptionsInOrder(renders, options));\n  } else if (opts.comments === InitProjectOptionHints.NONE) {\n    // don't render any extra options\n  }\n\n  if (result[result.length - 1] === \"\") {\n    result.pop();\n  }\n  result.push(\"}\");\n  return { renderedOptions: result.join(\"\\n\"), imports: allImports };\n}\n\nfunction renderCommentedOptionsByModule(\n  renders: Record<string, string>,\n  options: inventory.ProjectOption[]\n) {\n  const optionsByModule: Record<string, inventory.ProjectOption[]> = {};\n\n  for (const option of options) {\n    const parentModule = option.parent;\n    optionsByModule[parentModule] = optionsByModule[parentModule] ?? [];\n    optionsByModule[parentModule].push(option);\n  }\n\n  for (const parentModule in optionsByModule) {\n    optionsByModule[parentModule].sort((o1, o2) =>\n      o1.name.localeCompare(o2.name)\n    );\n  }\n\n  const result = [];\n  const marginSize = Math.max(\n    ...options.map((opt) => renders[opt.name].length)\n  );\n  for (const [moduleName, optionGroup] of Object.entries(\n    optionsByModule\n  ).sort()) {\n    result.push(`${TAB}/* ${moduleName} */`);\n    for (const option of optionGroup) {\n      const paramRender = renders[option.name];\n      const docstring = option.docs || \"No documentation found.\";\n      result.push(\n        `${TAB}${paramRender}${makePadding(\n          marginSize - paramRender.length + 2\n        )}/* ${docstring} */`\n      );\n    }\n    result.push(\"\");\n  }\n  return result;\n}\n\nfunction renderCommentedOptionsInOrder(\n  renders: Record<string, string>,\n  options: inventory.ProjectOption[]\n) {\n  const result = [];\n  const marginSize = Math.max(\n    ...options.map((opt) => renders[opt.name].length)\n  );\n  for (const option of options) {\n    const paramRender = renders[option.name];\n    const docstring = option.docs || \"No documentation found.\";\n    result.push(\n      `${TAB}${paramRender}${makePadding(\n        marginSize - paramRender.length + 2\n      )}/* ${docstring} */`\n    );\n  }\n  return result;\n}\n\n/**\n * Renders a value as a JavaScript value, converting strings to enums where\n * appropriate. The type must be JSON-like (string, number, boolean, array,\n * enum, or JSON object).\n *\n * Returns a JavaScript expression as a string, and the names of any\n * necessary imports.\n */\nfunction renderArgAsJavaScript(arg: any, option: inventory.ProjectOption) {\n  if (option.kind === \"enum\") {\n    if (!option.fqn) {\n      throw new Error(`fqn field is missing from enum option ${option.name}`);\n    }\n    const parts = option.fqn.split(\".\"); // -> ['projen', 'web', 'MyEnum']\n    const enumChoice = String(arg).toUpperCase().replace(/-/g, \"_\"); // custom-value -> CUSTOM_VALUE\n    const js = `${parts.slice(1).join(\".\")}.${enumChoice}`; // -> web.MyEnum.CUSTOM_VALUE\n    const moduleName = parts[0]; // -> projen\n    const importName = parts[1]; // -> web\n    return { js, moduleName, importName };\n  } else if (option.jsonLike) {\n    return { js: JSON.stringify(arg) };\n  } else {\n    throw new Error(\n      `Unexpected option ${option.name} - cannot render a value for this option because it does not have a JSON-like type.`\n    );\n  }\n}\n\nfunction makePadding(paddingLength: number): string {\n  return \" \".repeat(paddingLength);\n}\n"]}