UNPKG

projen

Version:

CDK for software projects

224 lines • 32.4 kB
"use strict"; 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"]}