1 | ;
|
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 | */
|
9 | var __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 | }));
|
20 | var __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 | });
|
25 | var __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 | };
|
31 | var __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 | };
|
38 | var __metadata = (this && this.__metadata) || function (k, v) {
|
39 | if (typeof Reflect === "object" && typeof Reflect.metadata === "function") return Reflect.metadata(k, v);
|
40 | };
|
41 | var __importDefault = (this && this.__importDefault) || function (mod) {
|
42 | return (mod && mod.__esModule) ? mod : { "default": mod };
|
43 | };
|
44 | Object.defineProperty(exports, "__esModule", { value: true });
|
45 | exports.CommandModuleError = exports.CommandModule = exports.CommandScope = void 0;
|
46 | const core_1 = require("@angular-devkit/core");
|
47 | const fs_1 = require("fs");
|
48 | const path = __importStar(require("path"));
|
49 | const yargs_1 = __importDefault(require("yargs"));
|
50 | const helpers_1 = require("yargs/helpers");
|
51 | const analytics_1 = require("../analytics/analytics");
|
52 | const analytics_collector_1 = require("../analytics/analytics-collector");
|
53 | const analytics_parameters_1 = require("../analytics/analytics-parameters");
|
54 | const completion_1 = require("../utilities/completion");
|
55 | const memoize_1 = require("../utilities/memoize");
|
56 | var 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 = {}));
|
65 | class 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);
|
270 | exports.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 | */
|
275 | class CommandModuleError extends Error {
|
276 | }
|
277 | exports.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 |