@angular/cli
Version:
CLI tool for Angular
278 lines • 35.7 kB
JavaScript
;
/**
* @license
* Copyright Google LLC All Rights Reserved.
*
* Use of this source code is governed by an MIT-style license that can be
* found in the LICENSE file at https://angular.io/license
*/
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
if (k2 === undefined) k2 = k;
var desc = Object.getOwnPropertyDescriptor(m, k);
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
desc = { enumerable: true, get: function() { return m[k]; } };
}
Object.defineProperty(o, k2, desc);
}) : (function(o, m, k, k2) {
if (k2 === undefined) k2 = k;
o[k2] = m[k];
}));
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
Object.defineProperty(o, "default", { enumerable: true, value: v });
}) : function(o, v) {
o["default"] = v;
});
var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {
var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
return c > 3 && r && Object.defineProperty(target, key, r), r;
};
var __importStar = (this && this.__importStar) || function (mod) {
if (mod && mod.__esModule) return mod;
var result = {};
if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
__setModuleDefault(result, mod);
return result;
};
var __metadata = (this && this.__metadata) || function (k, v) {
if (typeof Reflect === "object" && typeof Reflect.metadata === "function") return Reflect.metadata(k, v);
};
var __importDefault = (this && this.__importDefault) || function (mod) {
return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.CommandModuleError = exports.CommandModule = exports.CommandScope = void 0;
const core_1 = require("@angular-devkit/core");
const fs_1 = require("fs");
const path = __importStar(require("path"));
const yargs_1 = __importDefault(require("yargs"));
const helpers_1 = require("yargs/helpers");
const analytics_1 = require("../analytics/analytics");
const analytics_collector_1 = require("../analytics/analytics-collector");
const analytics_parameters_1 = require("../analytics/analytics-parameters");
const completion_1 = require("../utilities/completion");
const memoize_1 = require("../utilities/memoize");
var CommandScope;
(function (CommandScope) {
/** Command can only run inside an Angular workspace. */
CommandScope[CommandScope["In"] = 0] = "In";
/** Command can only run outside an Angular workspace. */
CommandScope[CommandScope["Out"] = 1] = "Out";
/** Command can run inside and outside an Angular workspace. */
CommandScope[CommandScope["Both"] = 2] = "Both";
})(CommandScope || (exports.CommandScope = CommandScope = {}));
class CommandModule {
constructor(context) {
this.context = context;
this.shouldReportAnalytics = true;
this.scope = CommandScope.Both;
this.optionsWithAnalytics = new Map();
}
/**
* Description object which contains the long command descroption.
* This is used to generate JSON help wich is used in AIO.
*
* `false` will result in a hidden command.
*/
get fullDescribe() {
return this.describe === false
? false
: {
describe: this.describe,
...(this.longDescriptionPath
? {
longDescriptionRelativePath: path
.relative(path.join(__dirname, '../../../../'), this.longDescriptionPath)
.replace(/\\/g, path.posix.sep),
longDescription: (0, fs_1.readFileSync)(this.longDescriptionPath, 'utf8').replace(/\r\n/g, '\n'),
}
: {}),
};
}
get commandName() {
return this.command.split(' ', 1)[0];
}
async handler(args) {
const { _, $0, ...options } = args;
// Camelize options as yargs will return the object in kebab-case when camel casing is disabled.
const camelCasedOptions = {};
for (const [key, value] of Object.entries(options)) {
camelCasedOptions[helpers_1.Parser.camelCase(key)] = value;
}
// Set up autocompletion if appropriate.
const autocompletionExitCode = await (0, completion_1.considerSettingUpAutocompletion)(this.commandName, this.context.logger);
if (autocompletionExitCode !== undefined) {
process.exitCode = autocompletionExitCode;
return;
}
// Gather and report analytics.
const analytics = await this.getAnalytics();
const stopPeriodicFlushes = analytics && analytics.periodFlush();
let exitCode;
try {
if (analytics) {
this.reportCommandRunAnalytics(analytics);
this.reportWorkspaceInfoAnalytics(analytics);
}
exitCode = await this.run(camelCasedOptions);
}
catch (e) {
if (e instanceof core_1.schema.SchemaValidationException) {
this.context.logger.fatal(`Error: ${e.message}`);
exitCode = 1;
}
else {
throw e;
}
}
finally {
await stopPeriodicFlushes?.();
if (typeof exitCode === 'number' && exitCode > 0) {
process.exitCode = exitCode;
}
}
}
async getAnalytics() {
if (!this.shouldReportAnalytics) {
return undefined;
}
const userId = await (0, analytics_1.getAnalyticsUserId)(this.context,
// Don't prompt for `ng update` and `ng analytics` commands.
['update', 'analytics'].includes(this.commandName));
return userId ? new analytics_collector_1.AnalyticsCollector(this.context, userId) : undefined;
}
/**
* Adds schema options to a command also this keeps track of options that are required for analytics.
* **Note:** This method should be called from the command bundler method.
*/
addSchemaOptionsToCommand(localYargs, options) {
const booleanOptionsWithNoPrefix = new Set();
for (const option of options) {
const { default: defaultVal, positional, deprecated, description, alias, userAnalytics, type, hidden, name, choices, } = option;
const sharedOptions = {
alias,
hidden,
description,
deprecated,
choices,
// This should only be done when `--help` is used otherwise default will override options set in angular.json.
...(this.context.args.options.help ? { default: defaultVal } : {}),
};
let dashedName = core_1.strings.dasherize(name);
// Handle options which have been defined in the schema with `no` prefix.
if (type === 'boolean' && dashedName.startsWith('no-')) {
dashedName = dashedName.slice(3);
booleanOptionsWithNoPrefix.add(dashedName);
}
if (positional === undefined) {
localYargs = localYargs.option(dashedName, {
type,
...sharedOptions,
});
}
else {
localYargs = localYargs.positional(dashedName, {
type: type === 'array' || type === 'count' ? 'string' : type,
...sharedOptions,
});
}
// Record option of analytics.
if (userAnalytics !== undefined) {
this.optionsWithAnalytics.set(name, userAnalytics);
}
}
// Handle options which have been defined in the schema with `no` prefix.
if (booleanOptionsWithNoPrefix.size) {
localYargs.middleware((options) => {
for (const key of booleanOptionsWithNoPrefix) {
if (key in options) {
options[`no-${key}`] = !options[key];
delete options[key];
}
}
}, false);
}
return localYargs;
}
getWorkspaceOrThrow() {
const { workspace } = this.context;
if (!workspace) {
throw new CommandModuleError('A workspace is required for this command.');
}
return workspace;
}
/**
* Flush on an interval (if the event loop is waiting).
*
* @returns a method that when called will terminate the periodic
* flush and call flush one last time.
*/
getAnalyticsParameters(options) {
const parameters = {};
const validEventCustomDimensionAndMetrics = new Set([
...Object.values(analytics_parameters_1.EventCustomDimension),
...Object.values(analytics_parameters_1.EventCustomMetric),
]);
for (const [name, ua] of this.optionsWithAnalytics) {
const value = options[name];
if ((typeof value === 'string' || typeof value === 'number' || typeof value === 'boolean') &&
validEventCustomDimensionAndMetrics.has(ua)) {
parameters[ua] = value;
}
}
return parameters;
}
reportCommandRunAnalytics(analytics) {
// eslint-disable-next-line @typescript-eslint/no-explicit-any
const internalMethods = yargs_1.default.getInternalMethods();
// $0 generate component [name] -> generate_component
// $0 add <collection> -> add
const fullCommand = internalMethods.getUsageInstance().getUsage()[0][0]
.split(' ')
.filter((x) => {
const code = x.charCodeAt(0);
return code >= 97 && code <= 122;
})
.join('_');
analytics.reportCommandRunEvent(fullCommand);
}
reportWorkspaceInfoAnalytics(analytics) {
const { workspace } = this.context;
if (!workspace) {
return;
}
let applicationProjectsCount = 0;
let librariesProjectsCount = 0;
for (const project of workspace.projects.values()) {
switch (project.extensions['projectType']) {
case 'application':
applicationProjectsCount++;
break;
case 'library':
librariesProjectsCount++;
break;
}
}
analytics.reportWorkspaceInfoEvent({
[analytics_parameters_1.EventCustomMetric.AllProjectsCount]: librariesProjectsCount + applicationProjectsCount,
[analytics_parameters_1.EventCustomMetric.ApplicationProjectsCount]: applicationProjectsCount,
[analytics_parameters_1.EventCustomMetric.LibraryProjectsCount]: librariesProjectsCount,
});
}
}
exports.CommandModule = CommandModule;
__decorate([
memoize_1.memoize,
__metadata("design:type", Function),
__metadata("design:paramtypes", []),
__metadata("design:returntype", Promise)
], CommandModule.prototype, "getAnalytics", null);
/**
* Creates an known command module error.
* This is used so during executation we can filter between known validation error and real non handled errors.
*/
class CommandModuleError extends Error {
}
exports.CommandModuleError = CommandModuleError;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"command-module.js","sourceRoot":"","sources":["../../../../../../../../packages/angular/cli/src/command-builder/command-module.ts"],"names":[],"mappings":"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","sourcesContent":["/**\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\nimport { logging, schema, strings } from '@angular-devkit/core';\nimport { readFileSync } from 'fs';\nimport * as path from 'path';\nimport yargs, {\n  Arguments,\n  ArgumentsCamelCase,\n  Argv,\n  CamelCaseKey,\n  PositionalOptions,\n  CommandModule as YargsCommandModule,\n  Options as YargsOptions,\n} from 'yargs';\nimport { Parser as yargsParser } from 'yargs/helpers';\nimport { getAnalyticsUserId } from '../analytics/analytics';\nimport { AnalyticsCollector } from '../analytics/analytics-collector';\nimport { EventCustomDimension, EventCustomMetric } from '../analytics/analytics-parameters';\nimport { considerSettingUpAutocompletion } from '../utilities/completion';\nimport { AngularWorkspace } from '../utilities/config';\nimport { memoize } from '../utilities/memoize';\nimport { PackageManagerUtils } from '../utilities/package-manager';\nimport { Option } from './utilities/json-schema';\n\nexport type Options<T> = { [key in keyof T as CamelCaseKey<key>]: T[key] };\n\nexport enum CommandScope {\n  /** Command can only run inside an Angular workspace. */\n  In,\n  /** Command can only run outside an Angular workspace. */\n  Out,\n  /** Command can run inside and outside an Angular workspace. */\n  Both,\n}\n\nexport interface CommandContext {\n  currentDirectory: string;\n  root: string;\n  workspace?: AngularWorkspace;\n  globalConfiguration: AngularWorkspace;\n  logger: logging.Logger;\n  packageManager: PackageManagerUtils;\n  /** Arguments parsed in free-from without parser configuration. */\n  args: {\n    positional: string[];\n    options: {\n      help: boolean;\n      jsonHelp: boolean;\n      getYargsCompletions: boolean;\n    } & Record<string, unknown>;\n  };\n}\n\nexport type OtherOptions = Record<string, unknown>;\n\nexport interface CommandModuleImplementation<T extends {} = {}>\n  extends Omit<YargsCommandModule<{}, T>, 'builder' | 'handler'> {\n  /** Scope in which the command can be executed in. */\n  scope: CommandScope;\n  /** Path used to load the long description for the command in JSON help text. */\n  longDescriptionPath?: string;\n  /** Object declaring the options the command accepts, or a function accepting and returning a yargs instance. */\n  builder(argv: Argv): Promise<Argv<T>> | Argv<T>;\n  /** A function which will be passed the parsed argv. */\n  run(options: Options<T> & OtherOptions): Promise<number | void> | number | void;\n}\n\nexport interface FullDescribe {\n  describe?: string;\n  longDescription?: string;\n  longDescriptionRelativePath?: string;\n}\n\nexport abstract class CommandModule<T extends {} = {}> implements CommandModuleImplementation<T> {\n  abstract readonly command: string;\n  abstract readonly describe: string | false;\n  abstract readonly longDescriptionPath?: string;\n  protected readonly shouldReportAnalytics: boolean = true;\n  readonly scope: CommandScope = CommandScope.Both;\n\n  private readonly optionsWithAnalytics = new Map<string, string>();\n\n  constructor(protected readonly context: CommandContext) {}\n\n  /**\n   * Description object which contains the long command descroption.\n   * This is used to generate JSON help wich is used in AIO.\n   *\n   * `false` will result in a hidden command.\n   */\n  public get fullDescribe(): FullDescribe | false {\n    return this.describe === false\n      ? false\n      : {\n          describe: this.describe,\n          ...(this.longDescriptionPath\n            ? {\n                longDescriptionRelativePath: path\n                  .relative(path.join(__dirname, '../../../../'), this.longDescriptionPath)\n                  .replace(/\\\\/g, path.posix.sep),\n                longDescription: readFileSync(this.longDescriptionPath, 'utf8').replace(\n                  /\\r\\n/g,\n                  '\\n',\n                ),\n              }\n            : {}),\n        };\n  }\n\n  protected get commandName(): string {\n    return this.command.split(' ', 1)[0];\n  }\n\n  abstract builder(argv: Argv): Promise<Argv<T>> | Argv<T>;\n  abstract run(options: Options<T> & OtherOptions): Promise<number | void> | number | void;\n\n  async handler(args: ArgumentsCamelCase<T> & OtherOptions): Promise<void> {\n    const { _, $0, ...options } = args;\n\n    // Camelize options as yargs will return the object in kebab-case when camel casing is disabled.\n    const camelCasedOptions: Record<string, unknown> = {};\n    for (const [key, value] of Object.entries(options)) {\n      camelCasedOptions[yargsParser.camelCase(key)] = value;\n    }\n\n    // Set up autocompletion if appropriate.\n    const autocompletionExitCode = await considerSettingUpAutocompletion(\n      this.commandName,\n      this.context.logger,\n    );\n    if (autocompletionExitCode !== undefined) {\n      process.exitCode = autocompletionExitCode;\n\n      return;\n    }\n\n    // Gather and report analytics.\n    const analytics = await this.getAnalytics();\n    const stopPeriodicFlushes = analytics && analytics.periodFlush();\n\n    let exitCode: number | void | undefined;\n    try {\n      if (analytics) {\n        this.reportCommandRunAnalytics(analytics);\n        this.reportWorkspaceInfoAnalytics(analytics);\n      }\n\n      exitCode = await this.run(camelCasedOptions as Options<T> & OtherOptions);\n    } catch (e) {\n      if (e instanceof schema.SchemaValidationException) {\n        this.context.logger.fatal(`Error: ${e.message}`);\n        exitCode = 1;\n      } else {\n        throw e;\n      }\n    } finally {\n      await stopPeriodicFlushes?.();\n\n      if (typeof exitCode === 'number' && exitCode > 0) {\n        process.exitCode = exitCode;\n      }\n    }\n  }\n\n  @memoize\n  protected async getAnalytics(): Promise<AnalyticsCollector | undefined> {\n    if (!this.shouldReportAnalytics) {\n      return undefined;\n    }\n\n    const userId = await getAnalyticsUserId(\n      this.context,\n      // Don't prompt for `ng update` and `ng analytics` commands.\n      ['update', 'analytics'].includes(this.commandName),\n    );\n\n    return userId ? new AnalyticsCollector(this.context, userId) : undefined;\n  }\n\n  /**\n   * Adds schema options to a command also this keeps track of options that are required for analytics.\n   * **Note:** This method should be called from the command bundler method.\n   */\n  protected addSchemaOptionsToCommand<T>(localYargs: Argv<T>, options: Option[]): Argv<T> {\n    const booleanOptionsWithNoPrefix = new Set<string>();\n\n    for (const option of options) {\n      const {\n        default: defaultVal,\n        positional,\n        deprecated,\n        description,\n        alias,\n        userAnalytics,\n        type,\n        hidden,\n        name,\n        choices,\n      } = option;\n\n      const sharedOptions: YargsOptions & PositionalOptions = {\n        alias,\n        hidden,\n        description,\n        deprecated,\n        choices,\n        // This should only be done when `--help` is used otherwise default will override options set in angular.json.\n        ...(this.context.args.options.help ? { default: defaultVal } : {}),\n      };\n\n      let dashedName = strings.dasherize(name);\n\n      // Handle options which have been defined in the schema with `no` prefix.\n      if (type === 'boolean' && dashedName.startsWith('no-')) {\n        dashedName = dashedName.slice(3);\n        booleanOptionsWithNoPrefix.add(dashedName);\n      }\n\n      if (positional === undefined) {\n        localYargs = localYargs.option(dashedName, {\n          type,\n          ...sharedOptions,\n        });\n      } else {\n        localYargs = localYargs.positional(dashedName, {\n          type: type === 'array' || type === 'count' ? 'string' : type,\n          ...sharedOptions,\n        });\n      }\n\n      // Record option of analytics.\n      if (userAnalytics !== undefined) {\n        this.optionsWithAnalytics.set(name, userAnalytics);\n      }\n    }\n\n    // Handle options which have been defined in the schema with `no` prefix.\n    if (booleanOptionsWithNoPrefix.size) {\n      localYargs.middleware((options: Arguments) => {\n        for (const key of booleanOptionsWithNoPrefix) {\n          if (key in options) {\n            options[`no-${key}`] = !options[key];\n            delete options[key];\n          }\n        }\n      }, false);\n    }\n\n    return localYargs;\n  }\n\n  protected getWorkspaceOrThrow(): AngularWorkspace {\n    const { workspace } = this.context;\n    if (!workspace) {\n      throw new CommandModuleError('A workspace is required for this command.');\n    }\n\n    return workspace;\n  }\n\n  /**\n   * Flush on an interval (if the event loop is waiting).\n   *\n   * @returns a method that when called will terminate the periodic\n   * flush and call flush one last time.\n   */\n  protected getAnalyticsParameters(\n    options: (Options<T> & OtherOptions) | OtherOptions,\n  ): Partial<Record<EventCustomDimension | EventCustomMetric, string | boolean | number>> {\n    const parameters: Partial<\n      Record<EventCustomDimension | EventCustomMetric, string | boolean | number>\n    > = {};\n\n    const validEventCustomDimensionAndMetrics = new Set([\n      ...Object.values(EventCustomDimension),\n      ...Object.values(EventCustomMetric),\n    ]);\n\n    for (const [name, ua] of this.optionsWithAnalytics) {\n      const value = options[name];\n      if (\n        (typeof value === 'string' || typeof value === 'number' || typeof value === 'boolean') &&\n        validEventCustomDimensionAndMetrics.has(ua as EventCustomDimension | EventCustomMetric)\n      ) {\n        parameters[ua as EventCustomDimension | EventCustomMetric] = value;\n      }\n    }\n\n    return parameters;\n  }\n\n  private reportCommandRunAnalytics(analytics: AnalyticsCollector): void {\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    const internalMethods = (yargs as any).getInternalMethods();\n    // $0 generate component [name] -> generate_component\n    // $0 add <collection> -> add\n    const fullCommand = (internalMethods.getUsageInstance().getUsage()[0][0] as string)\n      .split(' ')\n      .filter((x) => {\n        const code = x.charCodeAt(0);\n\n        return code >= 97 && code <= 122;\n      })\n      .join('_');\n\n    analytics.reportCommandRunEvent(fullCommand);\n  }\n\n  private reportWorkspaceInfoAnalytics(analytics: AnalyticsCollector): void {\n    const { workspace } = this.context;\n    if (!workspace) {\n      return;\n    }\n\n    let applicationProjectsCount = 0;\n    let librariesProjectsCount = 0;\n    for (const project of workspace.projects.values()) {\n      switch (project.extensions['projectType']) {\n        case 'application':\n          applicationProjectsCount++;\n          break;\n        case 'library':\n          librariesProjectsCount++;\n          break;\n      }\n    }\n\n    analytics.reportWorkspaceInfoEvent({\n      [EventCustomMetric.AllProjectsCount]: librariesProjectsCount + applicationProjectsCount,\n      [EventCustomMetric.ApplicationProjectsCount]: applicationProjectsCount,\n      [EventCustomMetric.LibraryProjectsCount]: librariesProjectsCount,\n    });\n  }\n}\n\n/**\n * Creates an known command module error.\n * This is used so during executation we can filter between known validation error and real non handled errors.\n */\nexport class CommandModuleError extends Error {}\n"]}