projen
Version:
CDK for software projects
382 lines • 53.3 kB
JavaScript
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
const fs = require("fs");
const path = require("path");
const semver = require("semver");
const yargs = require("yargs");
const inventory = require("../../inventory");
const logging = require("../../logging");
const option_hints_1 = require("../../option-hints");
const projects_1 = require("../../projects");
const util_1 = require("../../util");
const macros_1 = require("../macros");
const util_2 = require("../util");
class Command {
constructor() {
this.command = "new [PROJECT-TYPE-NAME] [OPTIONS]";
this.describe = [
"Creates a new projen project",
"",
"For a complete list of the available options for a specific project type, run:",
"projen new [PROJECT-TYPE-NAME] --help",
].join("\n");
}
builder(args) {
args.positional("PROJECT-TYPE-NAME", {
describe: "only optional with --from and the external module has only a single project type",
type: "string",
});
args.option("synth", {
type: "boolean",
default: true,
desc: "Synthesize after creating .projenrc.js",
});
args.option("comments", {
type: "boolean",
default: true,
desc: "Include commented out options in .projenrc.js (use --no-comments to disable)",
});
args.option("from", {
type: "string",
alias: "f",
desc: 'External jsii npm module to create project from. Supports any package spec supported by npm (such as "my-pack@^2.0")',
});
args.option("git", {
type: "boolean",
default: true,
desc: "Run `git init` and create an initial commit (use --no-git to disable)",
});
args.example("projen new awscdk-app-ts", 'Creates a new project of built-in type "awscdk-app-ts"');
args.example("projen new --from projen-vue@^2", 'Creates a new project from an external module "projen-vue" with the specified version');
args.example("projen new python --help", 'Shows all options available for the built-in project type "python"');
for (const type of inventory.discover()) {
args.command(type.pjid, type.docs ?? "", {
builder: (cargs) => {
cargs.showHelpOnFail(false);
for (const option of type.options ?? []) {
// not all types can be represented in the cli
if (!argTypeSupported(option)) {
continue;
}
const defaultValue = argInitialValue(option);
cargs.option(option.switch, {
group: !option.optional ? "Required:" : "Optional:",
type: argType(option),
description: argDesc(option),
required: !option.optional,
// yargs behaves differently for arrays if the defaultValue property is present or not
...(!option.optional && defaultValue
? { default: defaultValue }
: {}),
});
}
return cargs;
},
handler: (argv) => initProject(process.cwd(), type, argv),
});
}
// Disable strict mode, otherwise the catch-all doesn't work
args.strictCommands(false);
args
.command({
command: "*",
describe: false,
handler,
})
.middleware((argv) => {
// manually set the matched command as the project type
argv.projectTypeName = argv._[1];
}, true);
return args;
}
async handler(args) {
return handler(args);
}
}
async function handler(args) {
try {
// handle --from which means we want to first install a jsii module and then
// create a project defined within this module.
if (args.from) {
args.from = (0, util_1.normalizePersistedPath)(args.from);
return await initProjectFromModule(process.cwd(), args.from, args);
}
// project type is defined but was not matched by yargs, so print the list of supported types
if (args.projectTypeName) {
const types = inventory.discover();
throw new util_2.CliError(`Project type "${args.projectTypeName}" not found. Available types:\n`, ...types.map((t) => ` ${t.pjid}`), "", `Please specify a project type.`, `Example: npx projen new ${types[0].pjid}`);
}
// Handles the use case that nothing was specified since PROJECT-TYPE is now an optional positional parameter
yargs.showHelp();
}
catch (error) {
if (error instanceof util_2.CliError) {
logging.error(error.message);
logging.empty();
process.exitCode = 1;
return;
}
// unknown error, likely a node runtime exception in project code
// rethrow so the full stack trace is displayed
throw error;
}
}
/**
* Returns the yargs option type for a given project option
*/
function argType(option) {
if (option.kind === "enum") {
return "string";
}
if (isPrimitiveArrayOption(option)) {
return "array";
}
return option.simpleType;
}
/**
* Returns the description for a given project option
*/
function argDesc(option) {
let desc = [option.docs?.replace(/\ *\.$/, "") ?? ""];
const helpDefault = option.initialValue ?? option.default;
if (option.optional && helpDefault) {
desc.push(`[default: ${helpDefault.replace(/^\ *-/, "").replace(/\.$/, "").trim()}]`);
}
return desc.join(" ");
}
/**
* Compute the initial value for a given project option
*/
function argInitialValue(option, cwd = process.cwd()) {
// if we have determined an initial value for the field
// we can show that value in --help
if (option.initialValue) {
return renderDefault(cwd, option.initialValue);
}
}
/**
* Currently we only support these field types as command line options:
* - primitives (string, number, boolean)
* - lists of primitives
* - enums
*/
function argTypeSupported(option) {
return (option.simpleType === "string" ||
option.simpleType === "number" ||
option.simpleType === "boolean" ||
option.kind === "enum" ||
isPrimitiveArrayOption(option));
}
/**
* Checks if the given option is a primitive array
*/
function isPrimitiveArrayOption(option) {
return Boolean(option.jsonLike &&
option.fullType.collection?.kind === "array" &&
option.fullType.collection.elementtype.primitive &&
["string", "number"].includes(option.fullType.collection.elementtype.primitive));
}
/**
* Given a value from "@default", processes macros and returns a stringified
* (quoted) result.
*
* @returns a javascript primitive (could be a string, number or boolean)
*/
function renderDefault(cwd, value) {
return (0, macros_1.tryProcessMacro)(cwd, value) ?? JSON.parse(value);
}
/**
* Converts yargs command line switches to project type props.
* @param type Project type
* @param argv Command line switches
*/
function commandLineToProps(cwd, type, argv) {
const props = {};
// initialize props with default values
for (const prop of type.options) {
props[prop.name] = argInitialValue(prop, cwd);
}
for (const [arg, value] of Object.entries(argv)) {
for (const prop of type.options) {
if (prop.switch === arg) {
let curr = props;
const queue = [...prop.path];
while (true) {
const p = queue.shift();
if (!p) {
break;
}
if (queue.length === 0) {
curr[p] = value;
}
else {
curr[p] = curr[p] ?? {};
curr = curr[p];
}
}
}
}
}
return props;
}
/**
* Generates a new project from an external module.
*
* @param spec The name of the external module to load
* @param args Command line arguments (incl. project type)
*/
async function initProjectFromModule(baseDir, spec, args) {
const projenVersion = args.projenVersion ?? "latest";
const installCommand = (0, util_2.renderInstallCommand)(baseDir, `projen@${projenVersion}`);
if (args.projenVersion) {
(0, util_1.exec)(installCommand, { cwd: baseDir });
}
else {
// do not overwrite existing installation
(0, util_1.exec)(`npm ls --prefix="${baseDir}" --depth=0 --pattern projen || ${installCommand}`, { cwd: baseDir });
}
const installPackageWithCliError = (b, s) => {
try {
return (0, util_2.installPackage)(b, s);
}
catch (error) {
const stderr = error?.stderr?.toString() ?? "";
const isLocal = stderr.includes("code ENOENT");
const isRegistry = stderr.includes("code E404");
if (isLocal || isRegistry) {
const moduleSource = isLocal ? "path" : "registry";
throw new util_2.CliError(`Could not find '${s}' in this ${moduleSource}. Please ensure that the package exists, you have access it and try again.`);
}
throw error;
}
};
const moduleName = installPackageWithCliError(baseDir, spec);
logging.empty();
// Find the just installed package and discover the rest recursively from this package folder
const moduleDir = (0, util_2.findJsiiFilePath)(baseDir, moduleName);
if (!moduleDir) {
throw new util_2.CliError(`Module '${moduleName}' does not look like it is compatible with projen. Reason: Cannot find '${moduleName}/.jsii'. All projen modules must be jsii modules!`);
}
// Only leave projects from the main (requested) package
const projects = inventory
.discover(moduleDir)
.filter((x) => x.moduleName === moduleName); // Only list project types from the requested 'from' module
if (projects.length < 1) {
throw new util_2.CliError(`No project types found after installing "${spec}". The module must export at least one class which extends "projen.Project".`);
}
const requested = args.projectTypeName;
const types = projects.map((p) => p.pjid);
// if user did not specify a project type but the module has more than one, we need them to tell us which one...
if (!requested && projects.length > 1) {
throw new util_2.CliError(`Multiple project types found after installing "${spec}":\n`, ...types.map((t) => ` ${t}`), "", `Please specify a project type.`, `Example: npx projen new --from ${spec} ${types[0]}`);
}
// if user did not specify a type (and we know we have only one), the select it. otherwise, search by pjid.
const type = !requested
? projects[0]
: projects.find((p) => p.pjid === requested);
if (!type) {
throw new util_2.CliError(`Project type "${requested}" not found in "${spec}". Found:\n`, ...types.map((t) => ` ${t}`), "", `Please specify a valid project type.`, `Example: npx projen new --from ${spec} ${types[0]}`);
}
const missingOptions = [];
for (const option of type.options ?? []) {
// not all types can be represented in the cli
if (!argTypeSupported(option)) {
continue;
}
// parse allowed types
if (args[option.name] !== undefined) {
args[option.name] = parseArg(args[option.name], argType(option), option);
args[option.switch] = args[option.name];
continue;
}
// Required option with a default
if (!option.optional && option.default && option.default !== "undefined") {
const defaultValue = renderDefault(baseDir, option.default);
args[option.name] = defaultValue;
args[option.switch] = defaultValue;
}
// Required option, but we could not find a value
if (!option.optional && !args[option.name]) {
missingOptions.push(`--${option.switch} [${argType(option)}] ${argDesc(option)}`);
}
}
// We are missing some required options
if (missingOptions.length) {
throw new util_2.CliError(`Cannot create "${type.fqn}". Missing required option${missingOptions.length > 1 ? "s" : ""}:`, ...missingOptions.map((m) => ` ${m}`));
}
// include a dev dependency for the external module
args.devDeps = [spec];
args["dev-deps"] = [spec];
await initProject(baseDir, type, args);
}
/**
* Parse command line value as option type
*/
function parseArg(value, type, option) {
switch (type) {
case "number":
return parseInt(value);
case "boolean":
return typeof value === "string" ? (0, util_1.isTruthy)(value) : value;
case "array":
if (!Array.isArray(value)) {
value = [value];
}
return value.map((v) => parseArg(v, option?.fullType.collection?.elementtype.primitive || "string"));
// return value unchanged
case "string":
default:
// if we have an unexpected array, use the first element
if (Array.isArray(value)) {
return value[0];
}
return value;
}
}
/**
* Generates a new project.
* @param type Project type
* @param args Command line arguments
* @param additionalProps Additional parameters to include in .projenrc.js
*/
async function initProject(baseDir, type, args) {
// convert command line arguments to project props using type information
const props = commandLineToProps(baseDir, type, args);
projects_1.Projects.createProject({
dir: props.outdir ?? baseDir,
projectFqn: type.fqn,
projectOptions: props,
optionHints: args.comments
? option_hints_1.InitProjectOptionHints.FEATURED
: option_hints_1.InitProjectOptionHints.NONE,
synth: args.synth,
post: args.post,
});
if (fs.existsSync(path.join(baseDir, "package.json")) && args.post) {
(0, util_1.exec)("npm run eslint --if-present", { cwd: baseDir });
}
if (args.git) {
const git = (cmd) => (0, util_1.exec)(`git ${cmd}`, { cwd: baseDir });
const gitversion = (0, util_1.getGitVersion)((0, util_1.execCapture)("git --version", { cwd: baseDir }).toString());
logging.debug("system using git version ", gitversion);
// `git config init.defaultBranch` and `git init -b` are only available since git 2.28.0
if (gitversion && semver.gte(gitversion, "2.28.0")) {
const defaultGitInitBranch = (0, util_1.execOrUndefined)("git config init.defaultBranch", {
cwd: baseDir,
})?.trim() || "main";
git(`init -b ${defaultGitInitBranch}`);
git("add .");
git('commit --allow-empty -m "chore: project created with projen"');
logging.debug(`default branch name set to ${defaultGitInitBranch}`);
}
else {
git("init");
git("add .");
git('commit --allow-empty -m "chore: project created with projen"');
logging.debug("older version of git detected, changed default branch name to main");
git("branch -M main");
}
}
}
exports.default = new Command();
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"new.js","sourceRoot":"","sources":["../../../src/cli/cmds/new.ts"],"names":[],"mappings":"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","sourcesContent":["import type { SpawnSyncReturns } from \"child_process\";\nimport * as fs from \"fs\";\nimport * as path from \"path\";\nimport * as semver from \"semver\";\nimport * as yargs from \"yargs\";\nimport * as inventory from \"../../inventory\";\nimport * as logging from \"../../logging\";\nimport { InitProjectOptionHints } from \"../../option-hints\";\nimport { Projects } from \"../../projects\";\nimport {\n  exec,\n  execCapture,\n  execOrUndefined,\n  getGitVersion,\n  isTruthy,\n  normalizePersistedPath,\n} from \"../../util\";\nimport { tryProcessMacro } from \"../macros\";\nimport {\n  CliError,\n  findJsiiFilePath,\n  installPackage,\n  renderInstallCommand,\n} from \"../util\";\n\nclass Command implements yargs.CommandModule {\n  public readonly command = \"new [PROJECT-TYPE-NAME] [OPTIONS]\";\n  public readonly describe = [\n    \"Creates a new projen project\",\n    \"\",\n    \"For a complete list of the available options for a specific project type, run:\",\n    \"projen new [PROJECT-TYPE-NAME] --help\",\n  ].join(\"\\n\");\n\n  public builder(args: yargs.Argv) {\n    args.positional(\"PROJECT-TYPE-NAME\", {\n      describe:\n        \"only optional with --from and the external module has only a single project type\",\n      type: \"string\",\n    });\n    args.option(\"synth\", {\n      type: \"boolean\",\n      default: true,\n      desc: \"Synthesize after creating .projenrc.js\",\n    });\n    args.option(\"comments\", {\n      type: \"boolean\",\n      default: true,\n      desc: \"Include commented out options in .projenrc.js (use --no-comments to disable)\",\n    });\n    args.option(\"from\", {\n      type: \"string\",\n      alias: \"f\",\n      desc: 'External jsii npm module to create project from. Supports any package spec supported by npm (such as \"my-pack@^2.0\")',\n    });\n    args.option(\"git\", {\n      type: \"boolean\",\n      default: true,\n      desc: \"Run `git init` and create an initial commit (use --no-git to disable)\",\n    });\n    args.example(\n      \"projen new awscdk-app-ts\",\n      'Creates a new project of built-in type \"awscdk-app-ts\"'\n    );\n    args.example(\n      \"projen new --from projen-vue@^2\",\n      'Creates a new project from an external module \"projen-vue\" with the specified version'\n    );\n    args.example(\n      \"projen new python --help\",\n      'Shows all options available for the built-in project type \"python\"'\n    );\n\n    for (const type of inventory.discover()) {\n      args.command(type.pjid, type.docs ?? \"\", {\n        builder: (cargs) => {\n          cargs.showHelpOnFail(false);\n\n          for (const option of type.options ?? []) {\n            // not all types can be represented in the cli\n            if (!argTypeSupported(option)) {\n              continue;\n            }\n\n            const defaultValue = argInitialValue(option);\n            cargs.option(option.switch, {\n              group: !option.optional ? \"Required:\" : \"Optional:\",\n              type: argType(option),\n              description: argDesc(option),\n              required: !option.optional,\n              // yargs behaves differently for arrays if the defaultValue property is present or not\n              ...(!option.optional && defaultValue\n                ? { default: defaultValue }\n                : {}),\n            });\n          }\n\n          return cargs;\n        },\n        handler: (argv) => initProject(process.cwd(), type, argv),\n      });\n    }\n\n    // Disable strict mode, otherwise the catch-all doesn't work\n    args.strictCommands(false);\n    args\n      .command({\n        command: \"*\",\n        describe: false,\n        handler,\n      })\n      .middleware((argv) => {\n        // manually set the matched command as the project type\n        argv.projectTypeName = argv._[1];\n      }, true);\n\n    return args;\n  }\n\n  public async handler(args: any) {\n    return handler(args);\n  }\n}\n\nasync function handler(args: any) {\n  try {\n    // handle --from which means we want to first install a jsii module and then\n    // create a project defined within this module.\n    if (args.from) {\n      args.from = normalizePersistedPath(args.from);\n\n      return await initProjectFromModule(process.cwd(), args.from, args);\n    }\n\n    // project type is defined but was not matched by yargs, so print the list of supported types\n    if (args.projectTypeName) {\n      const types = inventory.discover();\n      throw new CliError(\n        `Project type \"${args.projectTypeName}\" not found. Available types:\\n`,\n        ...types.map((t) => `    ${t.pjid}`),\n        \"\",\n        `Please specify a project type.`,\n        `Example: npx projen new ${types[0].pjid}`\n      );\n    }\n\n    // Handles the use case that nothing was specified since PROJECT-TYPE is now an optional positional parameter\n    yargs.showHelp();\n  } catch (error: unknown) {\n    if (error instanceof CliError) {\n      logging.error(error.message);\n      logging.empty();\n      process.exitCode = 1;\n      return;\n    }\n\n    // unknown error, likely a node runtime exception in project code\n    // rethrow so the full stack trace is displayed\n    throw error;\n  }\n}\n\n/**\n * Returns the yargs option type for a given project option\n */\nfunction argType(\n  option: inventory.ProjectOption\n): \"string\" | \"boolean\" | \"number\" | \"array\" {\n  if (option.kind === \"enum\") {\n    return \"string\";\n  }\n\n  if (isPrimitiveArrayOption(option)) {\n    return \"array\";\n  }\n\n  return option.simpleType as \"string\" | \"boolean\" | \"number\";\n}\n\n/**\n * Returns the description for a given project option\n */\nfunction argDesc(option: inventory.ProjectOption): string {\n  let desc = [option.docs?.replace(/\\ *\\.$/, \"\") ?? \"\"];\n\n  const helpDefault = option.initialValue ?? option.default;\n  if (option.optional && helpDefault) {\n    desc.push(\n      `[default: ${helpDefault.replace(/^\\ *-/, \"\").replace(/\\.$/, \"\").trim()}]`\n    );\n  }\n\n  return desc.join(\" \");\n}\n\n/**\n * Compute the initial value for a given project option\n */\nfunction argInitialValue(\n  option: inventory.ProjectOption,\n  cwd = process.cwd()\n): any {\n  // if we have determined an initial value for the field\n  // we can show that value in --help\n  if (option.initialValue) {\n    return renderDefault(cwd, option.initialValue);\n  }\n}\n\n/**\n * Currently we only support these field types as command line options:\n * - primitives (string, number, boolean)\n * - lists of primitives\n * - enums\n */\nfunction argTypeSupported(option: inventory.ProjectOption): boolean {\n  return (\n    option.simpleType === \"string\" ||\n    option.simpleType === \"number\" ||\n    option.simpleType === \"boolean\" ||\n    option.kind === \"enum\" ||\n    isPrimitiveArrayOption(option)\n  );\n}\n\n/**\n * Checks if the given option is a primitive array\n */\nfunction isPrimitiveArrayOption(option: inventory.ProjectOption): boolean {\n  return Boolean(\n    option.jsonLike &&\n      option.fullType.collection?.kind === \"array\" &&\n      option.fullType.collection.elementtype.primitive &&\n      [\"string\", \"number\"].includes(\n        option.fullType.collection.elementtype.primitive\n      )\n  );\n}\n\n/**\n * Given a value from \"@default\", processes macros and returns a stringified\n * (quoted) result.\n *\n * @returns a javascript primitive (could be a string, number or boolean)\n */\nfunction renderDefault(cwd: string, value: string) {\n  return tryProcessMacro(cwd, value) ?? JSON.parse(value);\n}\n\n/**\n * Converts yargs command line switches to project type props.\n * @param type Project type\n * @param argv Command line switches\n */\nfunction commandLineToProps(\n  cwd: string,\n  type: inventory.ProjectType,\n  argv: Record<string, unknown>\n): Record<string, any> {\n  const props: Record<string, any> = {};\n\n  // initialize props with default values\n  for (const prop of type.options) {\n    props[prop.name] = argInitialValue(prop, cwd);\n  }\n\n  for (const [arg, value] of Object.entries(argv)) {\n    for (const prop of type.options) {\n      if (prop.switch === arg) {\n        let curr = props;\n        const queue = [...prop.path];\n        while (true) {\n          const p = queue.shift();\n          if (!p) {\n            break;\n          }\n          if (queue.length === 0) {\n            curr[p] = value;\n          } else {\n            curr[p] = curr[p] ?? {};\n            curr = curr[p];\n          }\n        }\n      }\n    }\n  }\n\n  return props;\n}\n\n/**\n * Generates a new project from an external module.\n *\n * @param spec The name of the external module to load\n * @param args Command line arguments (incl. project type)\n */\nasync function initProjectFromModule(baseDir: string, spec: string, args: any) {\n  const projenVersion = args.projenVersion ?? \"latest\";\n  const installCommand = renderInstallCommand(\n    baseDir,\n    `projen@${projenVersion}`\n  );\n  if (args.projenVersion) {\n    exec(installCommand, { cwd: baseDir });\n  } else {\n    // do not overwrite existing installation\n    exec(\n      `npm ls --prefix=\"${baseDir}\" --depth=0 --pattern projen || ${installCommand}`,\n      { cwd: baseDir }\n    );\n  }\n\n  const installPackageWithCliError = (b: string, s: string): string => {\n    try {\n      return installPackage(b, s);\n    } catch (error: unknown) {\n      const stderr =\n        (error as SpawnSyncReturns<Buffer>)?.stderr?.toString() ?? \"\";\n      const isLocal = stderr.includes(\"code ENOENT\");\n      const isRegistry = stderr.includes(\"code E404\");\n      if (isLocal || isRegistry) {\n        const moduleSource = isLocal ? \"path\" : \"registry\";\n        throw new CliError(\n          `Could not find '${s}' in this ${moduleSource}. Please ensure that the package exists, you have access it and try again.`\n        );\n      }\n\n      throw error;\n    }\n  };\n\n  const moduleName = installPackageWithCliError(baseDir, spec);\n  logging.empty();\n\n  // Find the just installed package and discover the rest recursively from this package folder\n  const moduleDir = findJsiiFilePath(baseDir, moduleName);\n\n  if (!moduleDir) {\n    throw new CliError(\n      `Module '${moduleName}' does not look like it is compatible with projen. Reason: Cannot find '${moduleName}/.jsii'. All projen modules must be jsii modules!`\n    );\n  }\n\n  // Only leave projects from the main (requested) package\n  const projects = inventory\n    .discover(moduleDir)\n    .filter((x) => x.moduleName === moduleName); // Only list project types from the requested 'from' module\n\n  if (projects.length < 1) {\n    throw new CliError(\n      `No project types found after installing \"${spec}\". The module must export at least one class which extends \"projen.Project\".`\n    );\n  }\n\n  const requested = args.projectTypeName;\n  const types = projects.map((p) => p.pjid);\n\n  // if user did not specify a project type but the module has more than one, we need them to tell us which one...\n  if (!requested && projects.length > 1) {\n    throw new CliError(\n      `Multiple project types found after installing \"${spec}\":\\n`,\n      ...types.map((t) => `    ${t}`),\n      \"\",\n      `Please specify a project type.`,\n      `Example: npx projen new --from ${spec} ${types[0]}`\n    );\n  }\n\n  // if user did not specify a type (and we know we have only one), the select it. otherwise, search by pjid.\n  const type = !requested\n    ? projects[0]\n    : projects.find((p) => p.pjid === requested);\n  if (!type) {\n    throw new CliError(\n      `Project type \"${requested}\" not found in \"${spec}\". Found:\\n`,\n      ...types.map((t) => `    ${t}`),\n      \"\",\n      `Please specify a valid project type.`,\n      `Example: npx projen new --from ${spec} ${types[0]}`\n    );\n  }\n\n  const missingOptions = [];\n\n  for (const option of type.options ?? []) {\n    // not all types can be represented in the cli\n    if (!argTypeSupported(option)) {\n      continue;\n    }\n\n    // parse allowed types\n    if (args[option.name] !== undefined) {\n      args[option.name] = parseArg(args[option.name], argType(option), option);\n      args[option.switch] = args[option.name];\n      continue;\n    }\n\n    // Required option with a default\n    if (!option.optional && option.default && option.default !== \"undefined\") {\n      const defaultValue = renderDefault(baseDir, option.default);\n      args[option.name] = defaultValue;\n      args[option.switch] = defaultValue;\n    }\n\n    // Required option, but we could not find a value\n    if (!option.optional && !args[option.name]) {\n      missingOptions.push(\n        `--${option.switch} [${argType(option)}] ${argDesc(option)}`\n      );\n    }\n  }\n\n  // We are missing some required options\n  if (missingOptions.length) {\n    throw new CliError(\n      `Cannot create \"${type.fqn}\". Missing required option${\n        missingOptions.length > 1 ? \"s\" : \"\"\n      }:`,\n      ...missingOptions.map((m) => `    ${m}`)\n    );\n  }\n\n  // include a dev dependency for the external module\n  args.devDeps = [spec];\n  args[\"dev-deps\"] = [spec];\n\n  await initProject(baseDir, type, args);\n}\n\n/**\n * Parse command line value as option type\n */\nfunction parseArg(\n  value: any,\n  type: string,\n  option?: inventory.ProjectOption\n): any {\n  switch (type) {\n    case \"number\":\n      return parseInt(value);\n    case \"boolean\":\n      return typeof value === \"string\" ? isTruthy(value) : value;\n    ca