UNPKG

35.9 kBJavaScriptView Raw
1"use strict";
2/**
3 * @license
4 * Copyright Google LLC All Rights Reserved.
5 *
6 * Use of this source code is governed by an MIT-style license that can be
7 * found in the LICENSE file at https://angular.io/license
8 */
9var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
10 if (k2 === undefined) k2 = k;
11 var desc = Object.getOwnPropertyDescriptor(m, k);
12 if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
13 desc = { enumerable: true, get: function() { return m[k]; } };
14 }
15 Object.defineProperty(o, k2, desc);
16}) : (function(o, m, k, k2) {
17 if (k2 === undefined) k2 = k;
18 o[k2] = m[k];
19}));
20var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
21 Object.defineProperty(o, "default", { enumerable: true, value: v });
22}) : function(o, v) {
23 o["default"] = v;
24});
25var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {
26 var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
27 if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
28 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;
29 return c > 3 && r && Object.defineProperty(target, key, r), r;
30};
31var __importStar = (this && this.__importStar) || function (mod) {
32 if (mod && mod.__esModule) return mod;
33 var result = {};
34 if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
35 __setModuleDefault(result, mod);
36 return result;
37};
38var __metadata = (this && this.__metadata) || function (k, v) {
39 if (typeof Reflect === "object" && typeof Reflect.metadata === "function") return Reflect.metadata(k, v);
40};
41var __importDefault = (this && this.__importDefault) || function (mod) {
42 return (mod && mod.__esModule) ? mod : { "default": mod };
43};
44Object.defineProperty(exports, "__esModule", { value: true });
45exports.CommandModuleError = exports.CommandModule = exports.CommandScope = void 0;
46const core_1 = require("@angular-devkit/core");
47const fs_1 = require("fs");
48const path = __importStar(require("path"));
49const yargs_1 = __importDefault(require("yargs"));
50const helpers_1 = require("yargs/helpers");
51const analytics_1 = require("../analytics/analytics");
52const analytics_collector_1 = require("../analytics/analytics-collector");
53const analytics_parameters_1 = require("../analytics/analytics-parameters");
54const completion_1 = require("../utilities/completion");
55const memoize_1 = require("../utilities/memoize");
56var CommandScope;
57(function (CommandScope) {
58 /** Command can only run inside an Angular workspace. */
59 CommandScope[CommandScope["In"] = 0] = "In";
60 /** Command can only run outside an Angular workspace. */
61 CommandScope[CommandScope["Out"] = 1] = "Out";
62 /** Command can run inside and outside an Angular workspace. */
63 CommandScope[CommandScope["Both"] = 2] = "Both";
64})(CommandScope = exports.CommandScope || (exports.CommandScope = {}));
65class CommandModule {
66 constructor(context) {
67 this.context = context;
68 this.shouldReportAnalytics = true;
69 this.scope = CommandScope.Both;
70 this.optionsWithAnalytics = new Map();
71 }
72 /**
73 * Description object which contains the long command descroption.
74 * This is used to generate JSON help wich is used in AIO.
75 *
76 * `false` will result in a hidden command.
77 */
78 get fullDescribe() {
79 return this.describe === false
80 ? false
81 : {
82 describe: this.describe,
83 ...(this.longDescriptionPath
84 ? {
85 longDescriptionRelativePath: path
86 .relative(path.join(__dirname, '../../../../'), this.longDescriptionPath)
87 .replace(/\\/g, path.posix.sep),
88 longDescription: (0, fs_1.readFileSync)(this.longDescriptionPath, 'utf8').replace(/\r\n/g, '\n'),
89 }
90 : {}),
91 };
92 }
93 get commandName() {
94 return this.command.split(' ', 1)[0];
95 }
96 async handler(args) {
97 const { _, $0, ...options } = args;
98 // Camelize options as yargs will return the object in kebab-case when camel casing is disabled.
99 const camelCasedOptions = {};
100 for (const [key, value] of Object.entries(options)) {
101 camelCasedOptions[helpers_1.Parser.camelCase(key)] = value;
102 }
103 // Set up autocompletion if appropriate.
104 const autocompletionExitCode = await (0, completion_1.considerSettingUpAutocompletion)(this.commandName, this.context.logger);
105 if (autocompletionExitCode !== undefined) {
106 process.exitCode = autocompletionExitCode;
107 return;
108 }
109 // Gather and report analytics.
110 const analytics = await this.getAnalytics();
111 const stopPeriodicFlushes = analytics && analytics.periodFlush();
112 let exitCode;
113 try {
114 if (analytics) {
115 this.reportCommandRunAnalytics(analytics);
116 this.reportWorkspaceInfoAnalytics(analytics);
117 }
118 exitCode = await this.run(camelCasedOptions);
119 }
120 catch (e) {
121 if (e instanceof core_1.schema.SchemaValidationException) {
122 this.context.logger.fatal(`Error: ${e.message}`);
123 exitCode = 1;
124 }
125 else {
126 throw e;
127 }
128 }
129 finally {
130 await (stopPeriodicFlushes === null || stopPeriodicFlushes === void 0 ? void 0 : stopPeriodicFlushes());
131 if (typeof exitCode === 'number' && exitCode > 0) {
132 process.exitCode = exitCode;
133 }
134 }
135 }
136 async getAnalytics() {
137 if (!this.shouldReportAnalytics) {
138 return undefined;
139 }
140 const userId = await (0, analytics_1.getAnalyticsUserId)(this.context,
141 // Don't prompt for `ng update` and `ng analytics` commands.
142 ['update', 'analytics'].includes(this.commandName));
143 return userId ? new analytics_collector_1.AnalyticsCollector(this.context, userId) : undefined;
144 }
145 /**
146 * Adds schema options to a command also this keeps track of options that are required for analytics.
147 * **Note:** This method should be called from the command bundler method.
148 */
149 addSchemaOptionsToCommand(localYargs, options) {
150 const booleanOptionsWithNoPrefix = new Set();
151 for (const option of options) {
152 const { default: defaultVal, positional, deprecated, description, alias, userAnalytics, type, hidden, name, choices, } = option;
153 const sharedOptions = {
154 alias,
155 hidden,
156 description,
157 deprecated,
158 choices,
159 // This should only be done when `--help` is used otherwise default will override options set in angular.json.
160 ...(this.context.args.options.help ? { default: defaultVal } : {}),
161 };
162 let dashedName = core_1.strings.dasherize(name);
163 // Handle options which have been defined in the schema with `no` prefix.
164 if (type === 'boolean' && dashedName.startsWith('no-')) {
165 dashedName = dashedName.slice(3);
166 booleanOptionsWithNoPrefix.add(dashedName);
167 }
168 if (positional === undefined) {
169 localYargs = localYargs.option(dashedName, {
170 type,
171 ...sharedOptions,
172 });
173 }
174 else {
175 localYargs = localYargs.positional(dashedName, {
176 type: type === 'array' || type === 'count' ? 'string' : type,
177 ...sharedOptions,
178 });
179 }
180 // Record option of analytics.
181 if (userAnalytics !== undefined) {
182 this.optionsWithAnalytics.set(name, userAnalytics);
183 }
184 }
185 // Handle options which have been defined in the schema with `no` prefix.
186 if (booleanOptionsWithNoPrefix.size) {
187 localYargs.middleware((options) => {
188 for (const key of booleanOptionsWithNoPrefix) {
189 if (key in options) {
190 options[`no-${key}`] = !options[key];
191 delete options[key];
192 }
193 }
194 }, false);
195 }
196 return localYargs;
197 }
198 getWorkspaceOrThrow() {
199 const { workspace } = this.context;
200 if (!workspace) {
201 throw new CommandModuleError('A workspace is required for this command.');
202 }
203 return workspace;
204 }
205 /**
206 * Flush on an interval (if the event loop is waiting).
207 *
208 * @returns a method that when called will terminate the periodic
209 * flush and call flush one last time.
210 */
211 getAnalyticsParameters(options) {
212 const parameters = {};
213 const validEventCustomDimensionAndMetrics = new Set([
214 ...Object.values(analytics_parameters_1.EventCustomDimension),
215 ...Object.values(analytics_parameters_1.EventCustomMetric),
216 ]);
217 for (const [name, ua] of this.optionsWithAnalytics) {
218 const value = options[name];
219 if ((typeof value === 'string' || typeof value === 'number' || typeof value === 'boolean') &&
220 validEventCustomDimensionAndMetrics.has(ua)) {
221 parameters[ua] = value;
222 }
223 }
224 return parameters;
225 }
226 reportCommandRunAnalytics(analytics) {
227 // eslint-disable-next-line @typescript-eslint/no-explicit-any
228 const internalMethods = yargs_1.default.getInternalMethods();
229 // $0 generate component [name] -> generate_component
230 // $0 add <collection> -> add
231 const fullCommand = internalMethods.getUsageInstance().getUsage()[0][0]
232 .split(' ')
233 .filter((x) => {
234 const code = x.charCodeAt(0);
235 return code >= 97 && code <= 122;
236 })
237 .join('_');
238 analytics.reportCommandRunEvent(fullCommand);
239 }
240 reportWorkspaceInfoAnalytics(analytics) {
241 const { workspace } = this.context;
242 if (!workspace) {
243 return;
244 }
245 let applicationProjectsCount = 0;
246 let librariesProjectsCount = 0;
247 for (const project of workspace.projects.values()) {
248 switch (project.extensions['projectType']) {
249 case 'application':
250 applicationProjectsCount++;
251 break;
252 case 'library':
253 librariesProjectsCount++;
254 break;
255 }
256 }
257 analytics.reportWorkspaceInfoEvent({
258 [analytics_parameters_1.EventCustomMetric.AllProjectsCount]: librariesProjectsCount + applicationProjectsCount,
259 [analytics_parameters_1.EventCustomMetric.ApplicationProjectsCount]: applicationProjectsCount,
260 [analytics_parameters_1.EventCustomMetric.LibraryProjectsCount]: librariesProjectsCount,
261 });
262 }
263}
264__decorate([
265 memoize_1.memoize,
266 __metadata("design:type", Function),
267 __metadata("design:paramtypes", []),
268 __metadata("design:returntype", Promise)
269], CommandModule.prototype, "getAnalytics", null);
270exports.CommandModule = CommandModule;
271/**
272 * Creates an known command module error.
273 * This is used so during executation we can filter between known validation error and real non handled errors.
274 */
275class CommandModuleError extends Error {
276}
277exports.CommandModuleError = CommandModuleError;
278//# 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"]}
\No newline at end of file