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 | Object.defineProperty(exports, "__esModule", { value: true });
|
10 | exports.ArchitectBaseCommandModule = void 0;
|
11 | const architect_1 = require("@angular-devkit/architect");
|
12 | const node_1 = require("@angular-devkit/architect/node");
|
13 | const core_1 = require("@angular-devkit/core");
|
14 | const child_process_1 = require("child_process");
|
15 | const fs_1 = require("fs");
|
16 | const path_1 = require("path");
|
17 | const analytics_1 = require("../analytics/analytics");
|
18 | const analytics_parameters_1 = require("../analytics/analytics-parameters");
|
19 | const error_1 = require("../utilities/error");
|
20 | const prompt_1 = require("../utilities/prompt");
|
21 | const tty_1 = require("../utilities/tty");
|
22 | const command_module_1 = require("./command-module");
|
23 | const json_schema_1 = require("./utilities/json-schema");
|
24 | class ArchitectBaseCommandModule extends command_module_1.CommandModule {
|
25 | constructor() {
|
26 | super(...arguments);
|
27 | this.scope = command_module_1.CommandScope.In;
|
28 | }
|
29 | async runSingleTarget(target, options) {
|
30 | const architectHost = await this.getArchitectHost();
|
31 | let builderName;
|
32 | try {
|
33 | builderName = await architectHost.getBuilderNameForTarget(target);
|
34 | }
|
35 | catch (e) {
|
36 | (0, error_1.assertIsError)(e);
|
37 | return this.onMissingTarget(e.message);
|
38 | }
|
39 | const { logger } = this.context;
|
40 | const run = await this.getArchitect().scheduleTarget(target, options, {
|
41 | logger,
|
42 | });
|
43 | const analytics = (0, analytics_1.isPackageNameSafeForAnalytics)(builderName)
|
44 | ? await this.getAnalytics()
|
45 | : undefined;
|
46 | let outputSubscription;
|
47 | if (analytics) {
|
48 | analytics.reportArchitectRunEvent({
|
49 | [analytics_parameters_1.EventCustomDimension.BuilderTarget]: builderName,
|
50 | });
|
51 | let firstRun = true;
|
52 | outputSubscription = run.output.subscribe(({ stats }) => {
|
53 | const parameters = this.builderStatsToAnalyticsParameters(stats, builderName);
|
54 | if (!parameters) {
|
55 | return;
|
56 | }
|
57 | if (firstRun) {
|
58 | firstRun = false;
|
59 | analytics.reportBuildRunEvent(parameters);
|
60 | }
|
61 | else {
|
62 | analytics.reportRebuildRunEvent(parameters);
|
63 | }
|
64 | });
|
65 | }
|
66 | try {
|
67 | const { error, success } = await run.output.toPromise();
|
68 | if (error) {
|
69 | logger.error(error);
|
70 | }
|
71 | return success ? 0 : 1;
|
72 | }
|
73 | finally {
|
74 | await run.stop();
|
75 | outputSubscription === null || outputSubscription === void 0 ? void 0 : outputSubscription.unsubscribe();
|
76 | }
|
77 | }
|
78 | builderStatsToAnalyticsParameters(stats, builderName) {
|
79 | if (!stats || typeof stats !== 'object' || !('durationInMs' in stats)) {
|
80 | return undefined;
|
81 | }
|
82 | const { optimization, allChunksCount, aot, lazyChunksCount, initialChunksCount, durationInMs, changedChunksCount, cssSizeInBytes, jsSizeInBytes, ngComponentCount, } = stats;
|
83 | return {
|
84 | [analytics_parameters_1.EventCustomDimension.BuilderTarget]: builderName,
|
85 | [analytics_parameters_1.EventCustomDimension.Aot]: aot,
|
86 | [analytics_parameters_1.EventCustomDimension.Optimization]: optimization,
|
87 | [analytics_parameters_1.EventCustomMetric.AllChunksCount]: allChunksCount,
|
88 | [analytics_parameters_1.EventCustomMetric.LazyChunksCount]: lazyChunksCount,
|
89 | [analytics_parameters_1.EventCustomMetric.InitialChunksCount]: initialChunksCount,
|
90 | [analytics_parameters_1.EventCustomMetric.ChangedChunksCount]: changedChunksCount,
|
91 | [analytics_parameters_1.EventCustomMetric.DurationInMs]: durationInMs,
|
92 | [analytics_parameters_1.EventCustomMetric.JsSizeInBytes]: jsSizeInBytes,
|
93 | [analytics_parameters_1.EventCustomMetric.CssSizeInBytes]: cssSizeInBytes,
|
94 | [analytics_parameters_1.EventCustomMetric.NgComponentCount]: ngComponentCount,
|
95 | };
|
96 | }
|
97 | getArchitectHost() {
|
98 | if (this._architectHost) {
|
99 | return this._architectHost;
|
100 | }
|
101 | const workspace = this.getWorkspaceOrThrow();
|
102 | return (this._architectHost = new node_1.WorkspaceNodeModulesArchitectHost(workspace, workspace.basePath));
|
103 | }
|
104 | getArchitect() {
|
105 | if (this._architect) {
|
106 | return this._architect;
|
107 | }
|
108 | const registry = new core_1.json.schema.CoreSchemaRegistry();
|
109 | registry.addPostTransform(core_1.json.schema.transforms.addUndefinedDefaults);
|
110 | registry.useXDeprecatedProvider((msg) => this.context.logger.warn(msg));
|
111 | const architectHost = this.getArchitectHost();
|
112 | return (this._architect = new architect_1.Architect(architectHost, registry));
|
113 | }
|
114 | async getArchitectTargetOptions(target) {
|
115 | const architectHost = this.getArchitectHost();
|
116 | let builderConf;
|
117 | try {
|
118 | builderConf = await architectHost.getBuilderNameForTarget(target);
|
119 | }
|
120 | catch (_a) {
|
121 | return [];
|
122 | }
|
123 | let builderDesc;
|
124 | try {
|
125 | builderDesc = await architectHost.resolveBuilder(builderConf);
|
126 | }
|
127 | catch (e) {
|
128 | (0, error_1.assertIsError)(e);
|
129 | if (e.code === 'MODULE_NOT_FOUND') {
|
130 | this.warnOnMissingNodeModules();
|
131 | throw new command_module_1.CommandModuleError(`Could not find the '${builderConf}' builder's node package.`);
|
132 | }
|
133 | throw e;
|
134 | }
|
135 | return (0, json_schema_1.parseJsonSchemaToOptions)(new core_1.json.schema.CoreSchemaRegistry(), builderDesc.optionSchema, true);
|
136 | }
|
137 | warnOnMissingNodeModules() {
|
138 | var _a;
|
139 | const basePath = (_a = this.context.workspace) === null || _a === void 0 ? void 0 : _a.basePath;
|
140 | if (!basePath) {
|
141 | return;
|
142 | }
|
143 | // Check for a `node_modules` directory (npm, yarn non-PnP, etc.)
|
144 | if ((0, fs_1.existsSync)((0, path_1.resolve)(basePath, 'node_modules'))) {
|
145 | return;
|
146 | }
|
147 | // Check for yarn PnP files
|
148 | if ((0, fs_1.existsSync)((0, path_1.resolve)(basePath, '.pnp.js')) ||
|
149 | (0, fs_1.existsSync)((0, path_1.resolve)(basePath, '.pnp.cjs')) ||
|
150 | (0, fs_1.existsSync)((0, path_1.resolve)(basePath, '.pnp.mjs'))) {
|
151 | return;
|
152 | }
|
153 | this.context.logger.warn(`Node packages may not be installed. Try installing with '${this.context.packageManager.name} install'.`);
|
154 | }
|
155 | getArchitectTarget() {
|
156 | return this.commandName;
|
157 | }
|
158 | async onMissingTarget(defaultMessage) {
|
159 | const { logger } = this.context;
|
160 | const choices = this.missingTargetChoices;
|
161 | if (!(choices === null || choices === void 0 ? void 0 : choices.length)) {
|
162 | logger.error(defaultMessage);
|
163 | return 1;
|
164 | }
|
165 | const missingTargetMessage = `Cannot find "${this.getArchitectTarget()}" target for the specified project.\n` +
|
166 | `You can add a package that implements these capabilities.\n\n` +
|
167 | `For example:\n` +
|
168 | choices.map(({ name, value }) => ` ${name}: ng add ${value}`).join('\n') +
|
169 | '\n';
|
170 | if ((0, tty_1.isTTY)()) {
|
171 | // Use prompts to ask the user if they'd like to install a package.
|
172 | logger.warn(missingTargetMessage);
|
173 | const packageToInstall = await this.getMissingTargetPackageToInstall(choices);
|
174 | if (packageToInstall) {
|
175 | // Example run: `ng add @angular-eslint/schematics`.
|
176 | const binPath = (0, path_1.resolve)(__dirname, '../../bin/ng.js');
|
177 | const { error } = (0, child_process_1.spawnSync)(process.execPath, [binPath, 'add', packageToInstall], {
|
178 | stdio: 'inherit',
|
179 | });
|
180 | if (error) {
|
181 | throw error;
|
182 | }
|
183 | }
|
184 | }
|
185 | else {
|
186 | // Non TTY display error message.
|
187 | logger.error(missingTargetMessage);
|
188 | }
|
189 | return 1;
|
190 | }
|
191 | async getMissingTargetPackageToInstall(choices) {
|
192 | if (choices.length === 1) {
|
193 | // Single choice
|
194 | const { name, value } = choices[0];
|
195 | if (await (0, prompt_1.askConfirmation)(`Would you like to add ${name} now?`, true, false)) {
|
196 | return value;
|
197 | }
|
198 | return null;
|
199 | }
|
200 | // Multiple choice
|
201 | return (0, prompt_1.askQuestion)(`Would you like to add a package with "${this.getArchitectTarget()}" capabilities now?`, [
|
202 | {
|
203 | name: 'No',
|
204 | value: null,
|
205 | },
|
206 | ...choices,
|
207 | ], 0, null);
|
208 | }
|
209 | }
|
210 | exports.ArchitectBaseCommandModule = ArchitectBaseCommandModule;
|
211 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"architect-base-command-module.js","sourceRoot":"","sources":["../../../../../../../../packages/angular/cli/src/command-builder/architect-base-command-module.ts"],"names":[],"mappings":";AAAA;;;;;;GAMG;;;AAEH,yDAA8D;AAC9D,yDAGwC;AACxC,+CAA4C;AAC5C,iDAA0C;AAC1C,2BAAgC;AAChC,+BAA+B;AAC/B,sDAAuE;AACvE,4EAA4F;AAC5F,8CAAmD;AACnD,gDAAmE;AACnE,0CAAyC;AACzC,qDAM0B;AAC1B,yDAA2E;AAO3E,MAAsB,0BACpB,SAAQ,8BAAgB;IAD1B;;QAIW,UAAK,GAAG,6BAAY,CAAC,EAAE,CAAC;IA+PnC,CAAC;IA5PW,KAAK,CAAC,eAAe,CAAC,MAAc,EAAE,OAAqB;QACnE,MAAM,aAAa,GAAG,MAAM,IAAI,CAAC,gBAAgB,EAAE,CAAC;QAEpD,IAAI,WAAmB,CAAC;QACxB,IAAI;YACF,WAAW,GAAG,MAAM,aAAa,CAAC,uBAAuB,CAAC,MAAM,CAAC,CAAC;SACnE;QAAC,OAAO,CAAC,EAAE;YACV,IAAA,qBAAa,EAAC,CAAC,CAAC,CAAC;YAEjB,OAAO,IAAI,CAAC,eAAe,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC;SACxC;QAED,MAAM,EAAE,MAAM,EAAE,GAAG,IAAI,CAAC,OAAO,CAAC;QAChC,MAAM,GAAG,GAAG,MAAM,IAAI,CAAC,YAAY,EAAE,CAAC,cAAc,CAAC,MAAM,EAAE,OAA0B,EAAE;YACvF,MAAM;SACP,CAAC,CAAC;QAEH,MAAM,SAAS,GAAG,IAAA,yCAA6B,EAAC,WAAW,CAAC;YAC1D,CAAC,CAAC,MAAM,IAAI,CAAC,YAAY,EAAE;YAC3B,CAAC,CAAC,SAAS,CAAC;QAEd,IAAI,kBAAkB,CAAC;QACvB,IAAI,SAAS,EAAE;YACb,SAAS,CAAC,uBAAuB,CAAC;gBAChC,CAAC,2CAAoB,CAAC,aAAa,CAAC,EAAE,WAAW;aAClD,CAAC,CAAC;YAEH,IAAI,QAAQ,GAAG,IAAI,CAAC;YACpB,kBAAkB,GAAG,GAAG,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC,EAAE,KAAK,EAAE,EAAE,EAAE;gBACtD,MAAM,UAAU,GAAG,IAAI,CAAC,iCAAiC,CAAC,KAAK,EAAE,WAAW,CAAC,CAAC;gBAC9E,IAAI,CAAC,UAAU,EAAE;oBACf,OAAO;iBACR;gBAED,IAAI,QAAQ,EAAE;oBACZ,QAAQ,GAAG,KAAK,CAAC;oBACjB,SAAS,CAAC,mBAAmB,CAAC,UAAU,CAAC,CAAC;iBAC3C;qBAAM;oBACL,SAAS,CAAC,qBAAqB,CAAC,UAAU,CAAC,CAAC;iBAC7C;YACH,CAAC,CAAC,CAAC;SACJ;QAED,IAAI;YACF,MAAM,EAAE,KAAK,EAAE,OAAO,EAAE,GAAG,MAAM,GAAG,CAAC,MAAM,CAAC,SAAS,EAAE,CAAC;YAExD,IAAI,KAAK,EAAE;gBACT,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;aACrB;YAED,OAAO,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;SACxB;gBAAS;YACR,MAAM,GAAG,CAAC,IAAI,EAAE,CAAC;YACjB,kBAAkB,aAAlB,kBAAkB,uBAAlB,kBAAkB,CAAE,WAAW,EAAE,CAAC;SACnC;IACH,CAAC;IAEO,iCAAiC,CACvC,KAAqB,EACrB,WAAmB;QAKnB,IAAI,CAAC,KAAK,IAAI,OAAO,KAAK,KAAK,QAAQ,IAAI,CAAC,CAAC,cAAc,IAAI,KAAK,CAAC,EAAE;YACrE,OAAO,SAAS,CAAC;SAClB;QAED,MAAM,EACJ,YAAY,EACZ,cAAc,EACd,GAAG,EACH,eAAe,EACf,kBAAkB,EAClB,YAAY,EACZ,kBAAkB,EAClB,cAAc,EACd,aAAa,EACb,gBAAgB,GACjB,GAAG,KAAK,CAAC;QAEV,OAAO;YACL,CAAC,2CAAoB,CAAC,aAAa,CAAC,EAAE,WAAW;YACjD,CAAC,2CAAoB,CAAC,GAAG,CAAC,EAAE,GAAG;YAC/B,CAAC,2CAAoB,CAAC,YAAY,CAAC,EAAE,YAAY;YACjD,CAAC,wCAAiB,CAAC,cAAc,CAAC,EAAE,cAAc;YAClD,CAAC,wCAAiB,CAAC,eAAe,CAAC,EAAE,eAAe;YACpD,CAAC,wCAAiB,CAAC,kBAAkB,CAAC,EAAE,kBAAkB;YAC1D,CAAC,wCAAiB,CAAC,kBAAkB,CAAC,EAAE,kBAAkB;YAC1D,CAAC,wCAAiB,CAAC,YAAY,CAAC,EAAE,YAAY;YAC9C,CAAC,wCAAiB,CAAC,aAAa,CAAC,EAAE,aAAa;YAChD,CAAC,wCAAiB,CAAC,cAAc,CAAC,EAAE,cAAc;YAClD,CAAC,wCAAiB,CAAC,gBAAgB,CAAC,EAAE,gBAAgB;SACvD,CAAC;IACJ,CAAC;IAGS,gBAAgB;QACxB,IAAI,IAAI,CAAC,cAAc,EAAE;YACvB,OAAO,IAAI,CAAC,cAAc,CAAC;SAC5B;QAED,MAAM,SAAS,GAAG,IAAI,CAAC,mBAAmB,EAAE,CAAC;QAE7C,OAAO,CAAC,IAAI,CAAC,cAAc,GAAG,IAAI,wCAAiC,CACjE,SAAS,EACT,SAAS,CAAC,QAAQ,CACnB,CAAC,CAAC;IACL,CAAC;IAGS,YAAY;QACpB,IAAI,IAAI,CAAC,UAAU,EAAE;YACnB,OAAO,IAAI,CAAC,UAAU,CAAC;SACxB;QAED,MAAM,QAAQ,GAAG,IAAI,WAAI,CAAC,MAAM,CAAC,kBAAkB,EAAE,CAAC;QACtD,QAAQ,CAAC,gBAAgB,CAAC,WAAI,CAAC,MAAM,CAAC,UAAU,CAAC,oBAAoB,CAAC,CAAC;QACvE,QAAQ,CAAC,sBAAsB,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC;QAExE,MAAM,aAAa,GAAG,IAAI,CAAC,gBAAgB,EAAE,CAAC;QAE9C,OAAO,CAAC,IAAI,CAAC,UAAU,GAAG,IAAI,qBAAS,CAAC,aAAa,EAAE,QAAQ,CAAC,CAAC,CAAC;IACpE,CAAC;IAES,KAAK,CAAC,yBAAyB,CAAC,MAAc;QACtD,MAAM,aAAa,GAAG,IAAI,CAAC,gBAAgB,EAAE,CAAC;QAC9C,IAAI,WAAmB,CAAC;QAExB,IAAI;YACF,WAAW,GAAG,MAAM,aAAa,CAAC,uBAAuB,CAAC,MAAM,CAAC,CAAC;SACnE;QAAC,WAAM;YACN,OAAO,EAAE,CAAC;SACX;QAED,IAAI,WAAmC,CAAC;QACxC,IAAI;YACF,WAAW,GAAG,MAAM,aAAa,CAAC,cAAc,CAAC,WAAW,CAAC,CAAC;SAC/D;QAAC,OAAO,CAAC,EAAE;YACV,IAAA,qBAAa,EAAC,CAAC,CAAC,CAAC;YACjB,IAAI,CAAC,CAAC,IAAI,KAAK,kBAAkB,EAAE;gBACjC,IAAI,CAAC,wBAAwB,EAAE,CAAC;gBAChC,MAAM,IAAI,mCAAkB,CAAC,uBAAuB,WAAW,2BAA2B,CAAC,CAAC;aAC7F;YAED,MAAM,CAAC,CAAC;SACT;QAED,OAAO,IAAA,sCAAwB,EAC7B,IAAI,WAAI,CAAC,MAAM,CAAC,kBAAkB,EAAE,EACpC,WAAW,CAAC,YAA+B,EAC3C,IAAI,CACL,CAAC;IACJ,CAAC;IAEO,wBAAwB;;QAC9B,MAAM,QAAQ,GAAG,MAAA,IAAI,CAAC,OAAO,CAAC,SAAS,0CAAE,QAAQ,CAAC;QAClD,IAAI,CAAC,QAAQ,EAAE;YACb,OAAO;SACR;QAED,iEAAiE;QACjE,IAAI,IAAA,eAAU,EAAC,IAAA,cAAO,EAAC,QAAQ,EAAE,cAAc,CAAC,CAAC,EAAE;YACjD,OAAO;SACR;QAED,2BAA2B;QAC3B,IACE,IAAA,eAAU,EAAC,IAAA,cAAO,EAAC,QAAQ,EAAE,SAAS,CAAC,CAAC;YACxC,IAAA,eAAU,EAAC,IAAA,cAAO,EAAC,QAAQ,EAAE,UAAU,CAAC,CAAC;YACzC,IAAA,eAAU,EAAC,IAAA,cAAO,EAAC,QAAQ,EAAE,UAAU,CAAC,CAAC,EACzC;YACA,OAAO;SACR;QAED,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,IAAI,CACtB,4DAA4D,IAAI,CAAC,OAAO,CAAC,cAAc,CAAC,IAAI,YAAY,CACzG,CAAC;IACJ,CAAC;IAES,kBAAkB;QAC1B,OAAO,IAAI,CAAC,WAAW,CAAC;IAC1B,CAAC;IAES,KAAK,CAAC,eAAe,CAAC,cAAsB;QACpD,MAAM,EAAE,MAAM,EAAE,GAAG,IAAI,CAAC,OAAO,CAAC;QAChC,MAAM,OAAO,GAAG,IAAI,CAAC,oBAAoB,CAAC;QAE1C,IAAI,CAAC,CAAA,OAAO,aAAP,OAAO,uBAAP,OAAO,CAAE,MAAM,CAAA,EAAE;YACpB,MAAM,CAAC,KAAK,CAAC,cAAc,CAAC,CAAC;YAE7B,OAAO,CAAC,CAAC;SACV;QAED,MAAM,oBAAoB,GACxB,gBAAgB,IAAI,CAAC,kBAAkB,EAAE,uCAAuC;YAChF,+DAA+D;YAC/D,gBAAgB;YAChB,OAAO,CAAC,GAAG,CAAC,CAAC,EAAE,IAAI,EAAE,KAAK,EAAE,EAAE,EAAE,CAAC,KAAK,IAAI,YAAY,KAAK,EAAE,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC;YACzE,IAAI,CAAC;QAEP,IAAI,IAAA,WAAK,GAAE,EAAE;YACX,mEAAmE;YACnE,MAAM,CAAC,IAAI,CAAC,oBAAoB,CAAC,CAAC;YAElC,MAAM,gBAAgB,GAAG,MAAM,IAAI,CAAC,gCAAgC,CAAC,OAAO,CAAC,CAAC;YAC9E,IAAI,gBAAgB,EAAE;gBACpB,oDAAoD;gBACpD,MAAM,OAAO,GAAG,IAAA,cAAO,EAAC,SAAS,EAAE,iBAAiB,CAAC,CAAC;gBACtD,MAAM,EAAE,KAAK,EAAE,GAAG,IAAA,yBAAS,EAAC,OAAO,CAAC,QAAQ,EAAE,CAAC,OAAO,EAAE,KAAK,EAAE,gBAAgB,CAAC,EAAE;oBAChF,KAAK,EAAE,SAAS;iBACjB,CAAC,CAAC;gBAEH,IAAI,KAAK,EAAE;oBACT,MAAM,KAAK,CAAC;iBACb;aACF;SACF;aAAM;YACL,iCAAiC;YACjC,MAAM,CAAC,KAAK,CAAC,oBAAoB,CAAC,CAAC;SACpC;QAED,OAAO,CAAC,CAAC;IACX,CAAC;IAEO,KAAK,CAAC,gCAAgC,CAC5C,OAA8B;QAE9B,IAAI,OAAO,CAAC,MAAM,KAAK,CAAC,EAAE;YACxB,gBAAgB;YAChB,MAAM,EAAE,IAAI,EAAE,KAAK,EAAE,GAAG,OAAO,CAAC,CAAC,CAAC,CAAC;YACnC,IAAI,MAAM,IAAA,wBAAe,EAAC,yBAAyB,IAAI,OAAO,EAAE,IAAI,EAAE,KAAK,CAAC,EAAE;gBAC5E,OAAO,KAAK,CAAC;aACd;YAED,OAAO,IAAI,CAAC;SACb;QAED,kBAAkB;QAClB,OAAO,IAAA,oBAAW,EAChB,yCAAyC,IAAI,CAAC,kBAAkB,EAAE,qBAAqB,EACvF;YACE;gBACE,IAAI,EAAE,IAAI;gBACV,KAAK,EAAE,IAAI;aACZ;YACD,GAAG,OAAO;SACX,EACD,CAAC,EACD,IAAI,CACL,CAAC;IACJ,CAAC;CACF;AAnQD,gEAmQC","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 { Architect, Target } from '@angular-devkit/architect';\nimport {\n  NodeModulesBuilderInfo,\n  WorkspaceNodeModulesArchitectHost,\n} from '@angular-devkit/architect/node';\nimport { json } from '@angular-devkit/core';\nimport { spawnSync } from 'child_process';\nimport { existsSync } from 'fs';\nimport { resolve } from 'path';\nimport { isPackageNameSafeForAnalytics } from '../analytics/analytics';\nimport { EventCustomDimension, EventCustomMetric } from '../analytics/analytics-parameters';\nimport { assertIsError } from '../utilities/error';\nimport { askConfirmation, askQuestion } from '../utilities/prompt';\nimport { isTTY } from '../utilities/tty';\nimport {\n  CommandModule,\n  CommandModuleError,\n  CommandModuleImplementation,\n  CommandScope,\n  OtherOptions,\n} from './command-module';\nimport { Option, parseJsonSchemaToOptions } from './utilities/json-schema';\n\nexport interface MissingTargetChoice {\n  name: string;\n  value: string;\n}\n\nexport abstract class ArchitectBaseCommandModule<T extends object>\n  extends CommandModule<T>\n  implements CommandModuleImplementation<T>\n{\n  override scope = CommandScope.In;\n  protected readonly missingTargetChoices: MissingTargetChoice[] | undefined;\n\n  protected async runSingleTarget(target: Target, options: OtherOptions): Promise<number> {\n    const architectHost = await this.getArchitectHost();\n\n    let builderName: string;\n    try {\n      builderName = await architectHost.getBuilderNameForTarget(target);\n    } catch (e) {\n      assertIsError(e);\n\n      return this.onMissingTarget(e.message);\n    }\n\n    const { logger } = this.context;\n    const run = await this.getArchitect().scheduleTarget(target, options as json.JsonObject, {\n      logger,\n    });\n\n    const analytics = isPackageNameSafeForAnalytics(builderName)\n      ? await this.getAnalytics()\n      : undefined;\n\n    let outputSubscription;\n    if (analytics) {\n      analytics.reportArchitectRunEvent({\n        [EventCustomDimension.BuilderTarget]: builderName,\n      });\n\n      let firstRun = true;\n      outputSubscription = run.output.subscribe(({ stats }) => {\n        const parameters = this.builderStatsToAnalyticsParameters(stats, builderName);\n        if (!parameters) {\n          return;\n        }\n\n        if (firstRun) {\n          firstRun = false;\n          analytics.reportBuildRunEvent(parameters);\n        } else {\n          analytics.reportRebuildRunEvent(parameters);\n        }\n      });\n    }\n\n    try {\n      const { error, success } = await run.output.toPromise();\n\n      if (error) {\n        logger.error(error);\n      }\n\n      return success ? 0 : 1;\n    } finally {\n      await run.stop();\n      outputSubscription?.unsubscribe();\n    }\n  }\n\n  private builderStatsToAnalyticsParameters(\n    stats: json.JsonValue,\n    builderName: string,\n  ): Partial<\n    | Record<EventCustomDimension & EventCustomMetric, string | number | undefined | boolean>\n    | undefined\n  > {\n    if (!stats || typeof stats !== 'object' || !('durationInMs' in stats)) {\n      return undefined;\n    }\n\n    const {\n      optimization,\n      allChunksCount,\n      aot,\n      lazyChunksCount,\n      initialChunksCount,\n      durationInMs,\n      changedChunksCount,\n      cssSizeInBytes,\n      jsSizeInBytes,\n      ngComponentCount,\n    } = stats;\n\n    return {\n      [EventCustomDimension.BuilderTarget]: builderName,\n      [EventCustomDimension.Aot]: aot,\n      [EventCustomDimension.Optimization]: optimization,\n      [EventCustomMetric.AllChunksCount]: allChunksCount,\n      [EventCustomMetric.LazyChunksCount]: lazyChunksCount,\n      [EventCustomMetric.InitialChunksCount]: initialChunksCount,\n      [EventCustomMetric.ChangedChunksCount]: changedChunksCount,\n      [EventCustomMetric.DurationInMs]: durationInMs,\n      [EventCustomMetric.JsSizeInBytes]: jsSizeInBytes,\n      [EventCustomMetric.CssSizeInBytes]: cssSizeInBytes,\n      [EventCustomMetric.NgComponentCount]: ngComponentCount,\n    };\n  }\n\n  private _architectHost: WorkspaceNodeModulesArchitectHost | undefined;\n  protected getArchitectHost(): WorkspaceNodeModulesArchitectHost {\n    if (this._architectHost) {\n      return this._architectHost;\n    }\n\n    const workspace = this.getWorkspaceOrThrow();\n\n    return (this._architectHost = new WorkspaceNodeModulesArchitectHost(\n      workspace,\n      workspace.basePath,\n    ));\n  }\n\n  private _architect: Architect | undefined;\n  protected getArchitect(): Architect {\n    if (this._architect) {\n      return this._architect;\n    }\n\n    const registry = new json.schema.CoreSchemaRegistry();\n    registry.addPostTransform(json.schema.transforms.addUndefinedDefaults);\n    registry.useXDeprecatedProvider((msg) => this.context.logger.warn(msg));\n\n    const architectHost = this.getArchitectHost();\n\n    return (this._architect = new Architect(architectHost, registry));\n  }\n\n  protected async getArchitectTargetOptions(target: Target): Promise<Option[]> {\n    const architectHost = this.getArchitectHost();\n    let builderConf: string;\n\n    try {\n      builderConf = await architectHost.getBuilderNameForTarget(target);\n    } catch {\n      return [];\n    }\n\n    let builderDesc: NodeModulesBuilderInfo;\n    try {\n      builderDesc = await architectHost.resolveBuilder(builderConf);\n    } catch (e) {\n      assertIsError(e);\n      if (e.code === 'MODULE_NOT_FOUND') {\n        this.warnOnMissingNodeModules();\n        throw new CommandModuleError(`Could not find the '${builderConf}' builder's node package.`);\n      }\n\n      throw e;\n    }\n\n    return parseJsonSchemaToOptions(\n      new json.schema.CoreSchemaRegistry(),\n      builderDesc.optionSchema as json.JsonObject,\n      true,\n    );\n  }\n\n  private warnOnMissingNodeModules(): void {\n    const basePath = this.context.workspace?.basePath;\n    if (!basePath) {\n      return;\n    }\n\n    // Check for a `node_modules` directory (npm, yarn non-PnP, etc.)\n    if (existsSync(resolve(basePath, 'node_modules'))) {\n      return;\n    }\n\n    // Check for yarn PnP files\n    if (\n      existsSync(resolve(basePath, '.pnp.js')) ||\n      existsSync(resolve(basePath, '.pnp.cjs')) ||\n      existsSync(resolve(basePath, '.pnp.mjs'))\n    ) {\n      return;\n    }\n\n    this.context.logger.warn(\n      `Node packages may not be installed. Try installing with '${this.context.packageManager.name} install'.`,\n    );\n  }\n\n  protected getArchitectTarget(): string {\n    return this.commandName;\n  }\n\n  protected async onMissingTarget(defaultMessage: string): Promise<1> {\n    const { logger } = this.context;\n    const choices = this.missingTargetChoices;\n\n    if (!choices?.length) {\n      logger.error(defaultMessage);\n\n      return 1;\n    }\n\n    const missingTargetMessage =\n      `Cannot find \"${this.getArchitectTarget()}\" target for the specified project.\\n` +\n      `You can add a package that implements these capabilities.\\n\\n` +\n      `For example:\\n` +\n      choices.map(({ name, value }) => `  ${name}: ng add ${value}`).join('\\n') +\n      '\\n';\n\n    if (isTTY()) {\n      // Use prompts to ask the user if they'd like to install a package.\n      logger.warn(missingTargetMessage);\n\n      const packageToInstall = await this.getMissingTargetPackageToInstall(choices);\n      if (packageToInstall) {\n        // Example run: `ng add @angular-eslint/schematics`.\n        const binPath = resolve(__dirname, '../../bin/ng.js');\n        const { error } = spawnSync(process.execPath, [binPath, 'add', packageToInstall], {\n          stdio: 'inherit',\n        });\n\n        if (error) {\n          throw error;\n        }\n      }\n    } else {\n      // Non TTY display error message.\n      logger.error(missingTargetMessage);\n    }\n\n    return 1;\n  }\n\n  private async getMissingTargetPackageToInstall(\n    choices: MissingTargetChoice[],\n  ): Promise<string | null> {\n    if (choices.length === 1) {\n      // Single choice\n      const { name, value } = choices[0];\n      if (await askConfirmation(`Would you like to add ${name} now?`, true, false)) {\n        return value;\n      }\n\n      return null;\n    }\n\n    // Multiple choice\n    return askQuestion(\n      `Would you like to add a package with \"${this.getArchitectTarget()}\" capabilities now?`,\n      [\n        {\n          name: 'No',\n          value: null,\n        },\n        ...choices,\n      ],\n      0,\n      null,\n    );\n  }\n}\n"]} |
\ | No newline at end of file |