UNPKG

44.9 kBJavaScriptView Raw
1#!/usr/bin/env node
2"use strict";
3/**
4 * @license
5 * Copyright Google LLC All Rights Reserved.
6 *
7 * Use of this source code is governed by an MIT-style license that can be
8 * found in the LICENSE file at https://angular.io/license
9 */
10var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
11 if (k2 === undefined) k2 = k;
12 var desc = Object.getOwnPropertyDescriptor(m, k);
13 if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
14 desc = { enumerable: true, get: function() { return m[k]; } };
15 }
16 Object.defineProperty(o, k2, desc);
17}) : (function(o, m, k, k2) {
18 if (k2 === undefined) k2 = k;
19 o[k2] = m[k];
20}));
21var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
22 Object.defineProperty(o, "default", { enumerable: true, value: v });
23}) : function(o, v) {
24 o["default"] = v;
25});
26var __importStar = (this && this.__importStar) || function (mod) {
27 if (mod && mod.__esModule) return mod;
28 var result = {};
29 if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
30 __setModuleDefault(result, mod);
31 return result;
32};
33Object.defineProperty(exports, "__esModule", { value: true });
34exports.main = void 0;
35// symbol polyfill must go first
36require("symbol-observable");
37const core_1 = require("@angular-devkit/core");
38const node_1 = require("@angular-devkit/core/node");
39const schematics_1 = require("@angular-devkit/schematics");
40const tools_1 = require("@angular-devkit/schematics/tools");
41const ansiColors = __importStar(require("ansi-colors"));
42const inquirer = __importStar(require("inquirer"));
43const yargs_parser_1 = __importStar(require("yargs-parser"));
44/**
45 * Parse the name of schematic passed in argument, and return a {collection, schematic} named
46 * tuple. The user can pass in `collection-name:schematic-name`, and this function will either
47 * return `{collection: 'collection-name', schematic: 'schematic-name'}`, or it will error out
48 * and show usage.
49 *
50 * In the case where a collection name isn't part of the argument, the default is to use the
51 * schematics package (@angular-devkit/schematics-cli) as the collection.
52 *
53 * This logic is entirely up to the tooling.
54 *
55 * @param str The argument to parse.
56 * @return {{collection: string, schematic: (string)}}
57 */
58function parseSchematicName(str) {
59 let collection = '@angular-devkit/schematics-cli';
60 let schematic = str;
61 if (schematic === null || schematic === void 0 ? void 0 : schematic.includes(':')) {
62 const lastIndexOfColon = schematic.lastIndexOf(':');
63 [collection, schematic] = [
64 schematic.slice(0, lastIndexOfColon),
65 schematic.substring(lastIndexOfColon + 1),
66 ];
67 }
68 return { collection, schematic };
69}
70function _listSchematics(workflow, collectionName, logger) {
71 try {
72 const collection = workflow.engine.createCollection(collectionName);
73 logger.info(collection.listSchematicNames().join('\n'));
74 }
75 catch (error) {
76 logger.fatal(error instanceof Error ? error.message : `${error}`);
77 return 1;
78 }
79 return 0;
80}
81function _createPromptProvider() {
82 return (definitions) => {
83 const questions = definitions.map((definition) => {
84 const question = {
85 name: definition.id,
86 message: definition.message,
87 default: definition.default,
88 };
89 const validator = definition.validator;
90 if (validator) {
91 question.validate = (input) => validator(input);
92 }
93 switch (definition.type) {
94 case 'confirmation':
95 return { ...question, type: 'confirm' };
96 case 'list':
97 return {
98 ...question,
99 type: definition.multiselect ? 'checkbox' : 'list',
100 choices: definition.items &&
101 definition.items.map((item) => {
102 if (typeof item == 'string') {
103 return item;
104 }
105 else {
106 return {
107 name: item.label,
108 value: item.value,
109 };
110 }
111 }),
112 };
113 default:
114 return { ...question, type: definition.type };
115 }
116 });
117 return inquirer.prompt(questions);
118 };
119}
120// eslint-disable-next-line max-lines-per-function
121async function main({ args, stdout = process.stdout, stderr = process.stderr, }) {
122 const { cliOptions, schematicOptions, _ } = parseArgs(args);
123 // Create a separate instance to prevent unintended global changes to the color configuration
124 const colors = ansiColors.create();
125 /** Create the DevKit Logger used through the CLI. */
126 const logger = (0, node_1.createConsoleLogger)(!!cliOptions.verbose, stdout, stderr, {
127 info: (s) => s,
128 debug: (s) => s,
129 warn: (s) => colors.bold.yellow(s),
130 error: (s) => colors.bold.red(s),
131 fatal: (s) => colors.bold.red(s),
132 });
133 if (cliOptions.help) {
134 logger.info(getUsage());
135 return 0;
136 }
137 /** Get the collection an schematic name from the first argument. */
138 const { collection: collectionName, schematic: schematicName } = parseSchematicName(_.shift() || null);
139 const isLocalCollection = collectionName.startsWith('.') || collectionName.startsWith('/');
140 /** Gather the arguments for later use. */
141 const debugPresent = cliOptions.debug !== null;
142 const debug = debugPresent ? !!cliOptions.debug : isLocalCollection;
143 const dryRunPresent = cliOptions['dry-run'] !== null;
144 const dryRun = dryRunPresent ? !!cliOptions['dry-run'] : debug;
145 const force = !!cliOptions.force;
146 const allowPrivate = !!cliOptions['allow-private'];
147 /** Create the workflow scoped to the working directory that will be executed with this run. */
148 const workflow = new tools_1.NodeWorkflow(process.cwd(), {
149 force,
150 dryRun,
151 resolvePaths: [process.cwd(), __dirname],
152 schemaValidation: true,
153 });
154 /** If the user wants to list schematics, we simply show all the schematic names. */
155 if (cliOptions['list-schematics']) {
156 return _listSchematics(workflow, collectionName, logger);
157 }
158 if (!schematicName) {
159 logger.info(getUsage());
160 return 1;
161 }
162 if (debug) {
163 logger.info(`Debug mode enabled${isLocalCollection ? ' by default for local collections' : ''}.`);
164 }
165 // Indicate to the user when nothing has been done. This is automatically set to off when there's
166 // a new DryRunEvent.
167 let nothingDone = true;
168 // Logging queue that receives all the messages to show the users. This only get shown when no
169 // errors happened.
170 let loggingQueue = [];
171 let error = false;
172 /**
173 * Logs out dry run events.
174 *
175 * All events will always be executed here, in order of discovery. That means that an error would
176 * be shown along other events when it happens. Since errors in workflows will stop the Observable
177 * from completing successfully, we record any events other than errors, then on completion we
178 * show them.
179 *
180 * This is a simple way to only show errors when an error occur.
181 */
182 workflow.reporter.subscribe((event) => {
183 nothingDone = false;
184 // Strip leading slash to prevent confusion.
185 const eventPath = event.path.startsWith('/') ? event.path.slice(1) : event.path;
186 switch (event.kind) {
187 case 'error':
188 error = true;
189 const desc = event.description == 'alreadyExist' ? 'already exists' : 'does not exist';
190 logger.error(`ERROR! ${eventPath} ${desc}.`);
191 break;
192 case 'update':
193 loggingQueue.push(`${colors.cyan('UPDATE')} ${eventPath} (${event.content.length} bytes)`);
194 break;
195 case 'create':
196 loggingQueue.push(`${colors.green('CREATE')} ${eventPath} (${event.content.length} bytes)`);
197 break;
198 case 'delete':
199 loggingQueue.push(`${colors.yellow('DELETE')} ${eventPath}`);
200 break;
201 case 'rename':
202 const eventToPath = event.to.startsWith('/') ? event.to.slice(1) : event.to;
203 loggingQueue.push(`${colors.blue('RENAME')} ${eventPath} => ${eventToPath}`);
204 break;
205 }
206 });
207 /**
208 * Listen to lifecycle events of the workflow to flush the logs between each phases.
209 */
210 workflow.lifeCycle.subscribe((event) => {
211 if (event.kind == 'workflow-end' || event.kind == 'post-tasks-start') {
212 if (!error) {
213 // Flush the log queue and clean the error state.
214 loggingQueue.forEach((log) => logger.info(log));
215 }
216 loggingQueue = [];
217 error = false;
218 }
219 });
220 // Show usage of deprecated options
221 workflow.registry.useXDeprecatedProvider((msg) => logger.warn(msg));
222 // Pass the rest of the arguments as the smart default "argv". Then delete it.
223 workflow.registry.addSmartDefaultProvider('argv', (schema) => 'index' in schema ? _[Number(schema['index'])] : _);
224 // Add prompts.
225 if (cliOptions.interactive && isTTY()) {
226 workflow.registry.usePromptProvider(_createPromptProvider());
227 }
228 /**
229 * Execute the workflow, which will report the dry run events, run the tasks, and complete
230 * after all is done.
231 *
232 * The Observable returned will properly cancel the workflow if unsubscribed, error out if ANY
233 * step of the workflow failed (sink or task), with details included, and will only complete
234 * when everything is done.
235 */
236 try {
237 await workflow
238 .execute({
239 collection: collectionName,
240 schematic: schematicName,
241 options: schematicOptions,
242 allowPrivate: allowPrivate,
243 debug: debug,
244 logger: logger,
245 })
246 .toPromise();
247 if (nothingDone) {
248 logger.info('Nothing to be done.');
249 }
250 else if (dryRun) {
251 logger.info(`Dry run enabled${dryRunPresent ? '' : ' by default in debug mode'}. No files written to disk.`);
252 }
253 return 0;
254 }
255 catch (err) {
256 if (err instanceof schematics_1.UnsuccessfulWorkflowExecution) {
257 // "See above" because we already printed the error.
258 logger.fatal('The Schematic workflow failed. See above.');
259 }
260 else if (debug && err instanceof Error) {
261 logger.fatal(`An error occured:\n${err.stack}`);
262 }
263 else {
264 logger.fatal(`Error: ${err instanceof Error ? err.message : err}`);
265 }
266 return 1;
267 }
268}
269exports.main = main;
270/**
271 * Get usage of the CLI tool.
272 */
273function getUsage() {
274 return core_1.tags.stripIndent `
275 schematics [collection-name:]schematic-name [options, ...]
276
277 By default, if the collection name is not specified, use the internal collection provided
278 by the Schematics CLI.
279
280 Options:
281 --debug Debug mode. This is true by default if the collection is a relative
282 path (in that case, turn off with --debug=false).
283
284 --allow-private Allow private schematics to be run from the command line. Default to
285 false.
286
287 --dry-run Do not output anything, but instead just show what actions would be
288 performed. Default to true if debug is also true.
289
290 --force Force overwriting files that would otherwise be an error.
291
292 --list-schematics List all schematics from the collection, by name. A collection name
293 should be suffixed by a colon. Example: '@angular-devkit/schematics-cli:'.
294
295 --no-interactive Disables interactive input prompts.
296
297 --verbose Show more information.
298
299 --help Show this message.
300
301 Any additional option is passed to the Schematics depending on its schema.
302 `;
303}
304/** Parse the command line. */
305const booleanArgs = [
306 'allow-private',
307 'debug',
308 'dry-run',
309 'force',
310 'help',
311 'list-schematics',
312 'verbose',
313 'interactive',
314];
315/** Parse the command line. */
316function parseArgs(args) {
317 const { _, ...options } = (0, yargs_parser_1.default)(args, {
318 boolean: booleanArgs,
319 default: {
320 'interactive': true,
321 'debug': null,
322 'dry-run': null,
323 },
324 configuration: {
325 'dot-notation': false,
326 'boolean-negation': true,
327 'strip-aliased': true,
328 'camel-case-expansion': false,
329 },
330 });
331 // Camelize options as yargs will return the object in kebab-case when camel casing is disabled.
332 const schematicOptions = {};
333 const cliOptions = {};
334 const isCliOptions = (key) => booleanArgs.includes(key);
335 for (const [key, value] of Object.entries(options)) {
336 if (/[A-Z]/.test(key)) {
337 throw new Error(`Unknown argument ${key}. Did you mean ${(0, yargs_parser_1.decamelize)(key)}?`);
338 }
339 if (isCliOptions(key)) {
340 cliOptions[key] = value;
341 }
342 else {
343 schematicOptions[(0, yargs_parser_1.camelCase)(key)] = value;
344 }
345 }
346 return {
347 _: _.map((v) => v.toString()),
348 schematicOptions,
349 cliOptions,
350 };
351}
352function isTTY() {
353 const isTruthy = (value) => {
354 // Returns true if value is a string that is anything but 0 or false.
355 return value !== undefined && value !== '0' && value.toUpperCase() !== 'FALSE';
356 };
357 // If we force TTY, we always return true.
358 const force = process.env['NG_FORCE_TTY'];
359 if (force !== undefined) {
360 return isTruthy(force);
361 }
362 return !!process.stdout.isTTY && !isTruthy(process.env['CI']);
363}
364if (require.main === module) {
365 const args = process.argv.slice(2);
366 main({ args })
367 .then((exitCode) => (process.exitCode = exitCode))
368 .catch((e) => {
369 throw e;
370 });
371}
372//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"schematics.js","sourceRoot":"","sources":["../../../../../../../packages/angular_devkit/schematics_cli/bin/schematics.ts"],"names":[],"mappings":"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","sourcesContent":["#!/usr/bin/env node\n/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\n// symbol polyfill must go first\nimport 'symbol-observable';\nimport { logging, schema, tags } from '@angular-devkit/core';\nimport { ProcessOutput, createConsoleLogger } from '@angular-devkit/core/node';\nimport { UnsuccessfulWorkflowExecution } from '@angular-devkit/schematics';\nimport { NodeWorkflow } from '@angular-devkit/schematics/tools';\nimport * as ansiColors from 'ansi-colors';\nimport * as inquirer from 'inquirer';\nimport yargsParser, { camelCase, decamelize } from 'yargs-parser';\n\n/**\n * Parse the name of schematic passed in argument, and return a {collection, schematic} named\n * tuple. The user can pass in `collection-name:schematic-name`, and this function will either\n * return `{collection: 'collection-name', schematic: 'schematic-name'}`, or it will error out\n * and show usage.\n *\n * In the case where a collection name isn't part of the argument, the default is to use the\n * schematics package (@angular-devkit/schematics-cli) as the collection.\n *\n * This logic is entirely up to the tooling.\n *\n * @param str The argument to parse.\n * @return {{collection: string, schematic: (string)}}\n */\nfunction parseSchematicName(str: string | null): { collection: string; schematic: string | null } {\n  let collection = '@angular-devkit/schematics-cli';\n\n  let schematic = str;\n  if (schematic?.includes(':')) {\n    const lastIndexOfColon = schematic.lastIndexOf(':');\n    [collection, schematic] = [\n      schematic.slice(0, lastIndexOfColon),\n      schematic.substring(lastIndexOfColon + 1),\n    ];\n  }\n\n  return { collection, schematic };\n}\n\nexport interface MainOptions {\n  args: string[];\n  stdout?: ProcessOutput;\n  stderr?: ProcessOutput;\n}\n\nfunction _listSchematics(workflow: NodeWorkflow, collectionName: string, logger: logging.Logger) {\n  try {\n    const collection = workflow.engine.createCollection(collectionName);\n    logger.info(collection.listSchematicNames().join('\\n'));\n  } catch (error) {\n    logger.fatal(error instanceof Error ? error.message : `${error}`);\n\n    return 1;\n  }\n\n  return 0;\n}\n\nfunction _createPromptProvider(): schema.PromptProvider {\n  return (definitions) => {\n    const questions: inquirer.QuestionCollection = definitions.map((definition) => {\n      const question: inquirer.Question = {\n        name: definition.id,\n        message: definition.message,\n        default: definition.default,\n      };\n\n      const validator = definition.validator;\n      if (validator) {\n        question.validate = (input) => validator(input);\n      }\n\n      switch (definition.type) {\n        case 'confirmation':\n          return { ...question, type: 'confirm' };\n        case 'list':\n          return {\n            ...question,\n            type: definition.multiselect ? 'checkbox' : 'list',\n            choices:\n              definition.items &&\n              definition.items.map((item) => {\n                if (typeof item == 'string') {\n                  return item;\n                } else {\n                  return {\n                    name: item.label,\n                    value: item.value,\n                  };\n                }\n              }),\n          };\n        default:\n          return { ...question, type: definition.type };\n      }\n    });\n\n    return inquirer.prompt(questions);\n  };\n}\n\n// eslint-disable-next-line max-lines-per-function\nexport async function main({\n  args,\n  stdout = process.stdout,\n  stderr = process.stderr,\n}: MainOptions): Promise<0 | 1> {\n  const { cliOptions, schematicOptions, _ } = parseArgs(args);\n\n  // Create a separate instance to prevent unintended global changes to the color configuration\n  const colors = ansiColors.create();\n\n  /** Create the DevKit Logger used through the CLI. */\n  const logger = createConsoleLogger(!!cliOptions.verbose, stdout, stderr, {\n    info: (s) => s,\n    debug: (s) => s,\n    warn: (s) => colors.bold.yellow(s),\n    error: (s) => colors.bold.red(s),\n    fatal: (s) => colors.bold.red(s),\n  });\n\n  if (cliOptions.help) {\n    logger.info(getUsage());\n\n    return 0;\n  }\n\n  /** Get the collection an schematic name from the first argument. */\n  const { collection: collectionName, schematic: schematicName } = parseSchematicName(\n    _.shift() || null,\n  );\n\n  const isLocalCollection = collectionName.startsWith('.') || collectionName.startsWith('/');\n\n  /** Gather the arguments for later use. */\n  const debugPresent = cliOptions.debug !== null;\n  const debug = debugPresent ? !!cliOptions.debug : isLocalCollection;\n  const dryRunPresent = cliOptions['dry-run'] !== null;\n  const dryRun = dryRunPresent ? !!cliOptions['dry-run'] : debug;\n  const force = !!cliOptions.force;\n  const allowPrivate = !!cliOptions['allow-private'];\n\n  /** Create the workflow scoped to the working directory that will be executed with this run. */\n  const workflow = new NodeWorkflow(process.cwd(), {\n    force,\n    dryRun,\n    resolvePaths: [process.cwd(), __dirname],\n    schemaValidation: true,\n  });\n\n  /** If the user wants to list schematics, we simply show all the schematic names. */\n  if (cliOptions['list-schematics']) {\n    return _listSchematics(workflow, collectionName, logger);\n  }\n\n  if (!schematicName) {\n    logger.info(getUsage());\n\n    return 1;\n  }\n\n  if (debug) {\n    logger.info(\n      `Debug mode enabled${isLocalCollection ? ' by default for local collections' : ''}.`,\n    );\n  }\n\n  // Indicate to the user when nothing has been done. This is automatically set to off when there's\n  // a new DryRunEvent.\n  let nothingDone = true;\n\n  // Logging queue that receives all the messages to show the users. This only get shown when no\n  // errors happened.\n  let loggingQueue: string[] = [];\n  let error = false;\n\n  /**\n   * Logs out dry run events.\n   *\n   * All events will always be executed here, in order of discovery. That means that an error would\n   * be shown along other events when it happens. Since errors in workflows will stop the Observable\n   * from completing successfully, we record any events other than errors, then on completion we\n   * show them.\n   *\n   * This is a simple way to only show errors when an error occur.\n   */\n  workflow.reporter.subscribe((event) => {\n    nothingDone = false;\n    // Strip leading slash to prevent confusion.\n    const eventPath = event.path.startsWith('/') ? event.path.slice(1) : event.path;\n\n    switch (event.kind) {\n      case 'error':\n        error = true;\n\n        const desc = event.description == 'alreadyExist' ? 'already exists' : 'does not exist';\n        logger.error(`ERROR! ${eventPath} ${desc}.`);\n        break;\n      case 'update':\n        loggingQueue.push(`${colors.cyan('UPDATE')} ${eventPath} (${event.content.length} bytes)`);\n        break;\n      case 'create':\n        loggingQueue.push(`${colors.green('CREATE')} ${eventPath} (${event.content.length} bytes)`);\n        break;\n      case 'delete':\n        loggingQueue.push(`${colors.yellow('DELETE')} ${eventPath}`);\n        break;\n      case 'rename':\n        const eventToPath = event.to.startsWith('/') ? event.to.slice(1) : event.to;\n        loggingQueue.push(`${colors.blue('RENAME')} ${eventPath} => ${eventToPath}`);\n        break;\n    }\n  });\n\n  /**\n   * Listen to lifecycle events of the workflow to flush the logs between each phases.\n   */\n  workflow.lifeCycle.subscribe((event) => {\n    if (event.kind == 'workflow-end' || event.kind == 'post-tasks-start') {\n      if (!error) {\n        // Flush the log queue and clean the error state.\n        loggingQueue.forEach((log) => logger.info(log));\n      }\n\n      loggingQueue = [];\n      error = false;\n    }\n  });\n\n  // Show usage of deprecated options\n  workflow.registry.useXDeprecatedProvider((msg) => logger.warn(msg));\n\n  // Pass the rest of the arguments as the smart default \"argv\". Then delete it.\n  workflow.registry.addSmartDefaultProvider('argv', (schema) =>\n    'index' in schema ? _[Number(schema['index'])] : _,\n  );\n\n  // Add prompts.\n  if (cliOptions.interactive && isTTY()) {\n    workflow.registry.usePromptProvider(_createPromptProvider());\n  }\n\n  /**\n   *  Execute the workflow, which will report the dry run events, run the tasks, and complete\n   *  after all is done.\n   *\n   *  The Observable returned will properly cancel the workflow if unsubscribed, error out if ANY\n   *  step of the workflow failed (sink or task), with details included, and will only complete\n   *  when everything is done.\n   */\n  try {\n    await workflow\n      .execute({\n        collection: collectionName,\n        schematic: schematicName,\n        options: schematicOptions,\n        allowPrivate: allowPrivate,\n        debug: debug,\n        logger: logger,\n      })\n      .toPromise();\n\n    if (nothingDone) {\n      logger.info('Nothing to be done.');\n    } else if (dryRun) {\n      logger.info(\n        `Dry run enabled${\n          dryRunPresent ? '' : ' by default in debug mode'\n        }. No files written to disk.`,\n      );\n    }\n\n    return 0;\n  } catch (err) {\n    if (err instanceof UnsuccessfulWorkflowExecution) {\n      // \"See above\" because we already printed the error.\n      logger.fatal('The Schematic workflow failed. See above.');\n    } else if (debug && err instanceof Error) {\n      logger.fatal(`An error occured:\\n${err.stack}`);\n    } else {\n      logger.fatal(`Error: ${err instanceof Error ? err.message : err}`);\n    }\n\n    return 1;\n  }\n}\n\n/**\n * Get usage of the CLI tool.\n */\nfunction getUsage(): string {\n  return tags.stripIndent`\n  schematics [collection-name:]schematic-name [options, ...]\n\n  By default, if the collection name is not specified, use the internal collection provided\n  by the Schematics CLI.\n\n  Options:\n      --debug             Debug mode. This is true by default if the collection is a relative\n                          path (in that case, turn off with --debug=false).\n\n      --allow-private     Allow private schematics to be run from the command line. Default to\n                          false.\n\n      --dry-run           Do not output anything, but instead just show what actions would be\n                          performed. Default to true if debug is also true.\n\n      --force             Force overwriting files that would otherwise be an error.\n\n      --list-schematics   List all schematics from the collection, by name. A collection name\n                          should be suffixed by a colon. Example: '@angular-devkit/schematics-cli:'.\n\n      --no-interactive    Disables interactive input prompts.\n\n      --verbose           Show more information.\n\n      --help              Show this message.\n\n  Any additional option is passed to the Schematics depending on its schema.\n  `;\n}\n\n/** Parse the command line. */\nconst booleanArgs = [\n  'allow-private',\n  'debug',\n  'dry-run',\n  'force',\n  'help',\n  'list-schematics',\n  'verbose',\n  'interactive',\n] as const;\n\ntype ElementType<T extends ReadonlyArray<unknown>> = T extends ReadonlyArray<infer ElementType>\n  ? ElementType\n  : never;\n\ninterface Options {\n  _: string[];\n  schematicOptions: Record<string, unknown>;\n  cliOptions: Partial<Record<ElementType<typeof booleanArgs>, boolean | null>>;\n}\n\n/** Parse the command line. */\nfunction parseArgs(args: string[]): Options {\n  const { _, ...options } = yargsParser(args, {\n    boolean: booleanArgs as unknown as string[],\n    default: {\n      'interactive': true,\n      'debug': null,\n      'dry-run': null,\n    },\n    configuration: {\n      'dot-notation': false,\n      'boolean-negation': true,\n      'strip-aliased': true,\n      'camel-case-expansion': false,\n    },\n  });\n\n  // Camelize options as yargs will return the object in kebab-case when camel casing is disabled.\n  const schematicOptions: Options['schematicOptions'] = {};\n  const cliOptions: Options['cliOptions'] = {};\n\n  const isCliOptions = (\n    key: ElementType<typeof booleanArgs> | string,\n  ): key is ElementType<typeof booleanArgs> =>\n    booleanArgs.includes(key as ElementType<typeof booleanArgs>);\n\n  for (const [key, value] of Object.entries(options)) {\n    if (/[A-Z]/.test(key)) {\n      throw new Error(`Unknown argument ${key}. Did you mean ${decamelize(key)}?`);\n    }\n\n    if (isCliOptions(key)) {\n      cliOptions[key] = value;\n    } else {\n      schematicOptions[camelCase(key)] = value;\n    }\n  }\n\n  return {\n    _: _.map((v) => v.toString()),\n    schematicOptions,\n    cliOptions,\n  };\n}\n\nfunction isTTY(): boolean {\n  const isTruthy = (value: undefined | string) => {\n    // Returns true if value is a string that is anything but 0 or false.\n    return value !== undefined && value !== '0' && value.toUpperCase() !== 'FALSE';\n  };\n\n  // If we force TTY, we always return true.\n  const force = process.env['NG_FORCE_TTY'];\n  if (force !== undefined) {\n    return isTruthy(force);\n  }\n\n  return !!process.stdout.isTTY && !isTruthy(process.env['CI']);\n}\n\nif (require.main === module) {\n  const args = process.argv.slice(2);\n  main({ args })\n    .then((exitCode) => (process.exitCode = exitCode))\n    .catch((e) => {\n      throw e;\n    });\n}\n"]}
\No newline at end of file