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 __importStar = (this && this.__importStar) || function (mod) {
|
26 | if (mod && mod.__esModule) return mod;
|
27 | var result = {};
|
28 | if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
|
29 | __setModuleDefault(result, mod);
|
30 | return result;
|
31 | };
|
32 | Object.defineProperty(exports, "__esModule", { value: true });
|
33 | exports.TypeScriptPathsPlugin = void 0;
|
34 | const path = __importStar(require("path"));
|
35 | class TypeScriptPathsPlugin {
|
36 | constructor(options) {
|
37 | if (options) {
|
38 | this.update(options);
|
39 | }
|
40 | }
|
41 | /**
|
42 | * Update the plugin with new path mapping option values.
|
43 | * The options will also be preprocessed to reduce the overhead of individual resolve actions
|
44 | * during a build.
|
45 | *
|
46 | * @param options The `paths` and `baseUrl` options from TypeScript's `CompilerOptions`.
|
47 | */
|
48 | update(options) {
|
49 | var _a, _b;
|
50 | this.baseUrl = options.baseUrl;
|
51 | this.patterns = undefined;
|
52 | if (options.paths) {
|
53 | for (const [pattern, potentials] of Object.entries(options.paths)) {
|
54 | // Ignore any entries that would not result in a new mapping
|
55 | if (potentials.length === 0 || potentials.every((potential) => potential === '*')) {
|
56 | continue;
|
57 | }
|
58 | const starIndex = pattern.indexOf('*');
|
59 | let prefix = pattern;
|
60 | let suffix;
|
61 | if (starIndex > -1) {
|
62 | prefix = pattern.slice(0, starIndex);
|
63 | if (starIndex < pattern.length - 1) {
|
64 | suffix = pattern.slice(starIndex + 1);
|
65 | }
|
66 | }
|
67 | (_a = this.patterns) !== null && _a !== void 0 ? _a : (this.patterns = []);
|
68 | this.patterns.push({
|
69 | starIndex,
|
70 | prefix,
|
71 | suffix,
|
72 | potentials: potentials.map((potential) => {
|
73 | const potentialStarIndex = potential.indexOf('*');
|
74 | if (potentialStarIndex === -1) {
|
75 | return { hasStar: false, prefix: potential };
|
76 | }
|
77 | return {
|
78 | hasStar: true,
|
79 | prefix: potential.slice(0, potentialStarIndex),
|
80 | suffix: potentialStarIndex < potential.length - 1
|
81 | ? potential.slice(potentialStarIndex + 1)
|
82 | : undefined,
|
83 | };
|
84 | }),
|
85 | });
|
86 | }
|
87 | // Sort patterns so that exact matches take priority then largest prefix match
|
88 | (_b = this.patterns) === null || _b === void 0 ? void 0 : _b.sort((a, b) => {
|
89 | if (a.starIndex === -1) {
|
90 | return -1;
|
91 | }
|
92 | else if (b.starIndex === -1) {
|
93 | return 1;
|
94 | }
|
95 | else {
|
96 | return b.starIndex - a.starIndex;
|
97 | }
|
98 | });
|
99 | }
|
100 | }
|
101 | apply(resolver) {
|
102 | const target = resolver.ensureHook('resolve');
|
103 | // To support synchronous resolvers this hook cannot be promise based.
|
104 | // Webpack supports synchronous resolution with `tap` and `tapAsync` hooks.
|
105 | resolver
|
106 | .getHook('described-resolve')
|
107 | .tapAsync('TypeScriptPathsPlugin', (request, resolveContext, callback) => {
|
108 | var _a, _b;
|
109 | // Preprocessing of the options will ensure that `patterns` is either undefined or has elements to check
|
110 | if (!this.patterns) {
|
111 | callback();
|
112 | return;
|
113 | }
|
114 | if (!request || request.typescriptPathMapped) {
|
115 | callback();
|
116 | return;
|
117 | }
|
118 | const originalRequest = request.request || request.path;
|
119 | if (!originalRequest) {
|
120 | callback();
|
121 | return;
|
122 | }
|
123 | // Only work on Javascript/TypeScript issuers.
|
124 | if (!((_b = (_a = request === null || request === void 0 ? void 0 : request.context) === null || _a === void 0 ? void 0 : _a.issuer) === null || _b === void 0 ? void 0 : _b.match(/\.[cm]?[jt]sx?$/))) {
|
125 | callback();
|
126 | return;
|
127 | }
|
128 | switch (originalRequest[0]) {
|
129 | case '.':
|
130 | case '/':
|
131 | // Relative or absolute requests are not mapped
|
132 | callback();
|
133 | return;
|
134 | case '!':
|
135 | // Ignore all webpack special requests
|
136 | if (originalRequest.length > 1 && originalRequest[1] === '!') {
|
137 | callback();
|
138 | return;
|
139 | }
|
140 | break;
|
141 | }
|
142 | // A generator is used to limit the amount of replacements requests that need to be created.
|
143 | // For example, if the first one resolves, any others are not needed and do not need
|
144 | // to be created.
|
145 | const requests = this.createReplacementRequests(request, originalRequest);
|
146 | const tryResolve = () => {
|
147 | const next = requests.next();
|
148 | if (next.done) {
|
149 | callback();
|
150 | return;
|
151 | }
|
152 | resolver.doResolve(target, next.value, '', resolveContext, (error, result) => {
|
153 | if (error) {
|
154 | callback(error);
|
155 | }
|
156 | else if (result) {
|
157 | callback(undefined, result);
|
158 | }
|
159 | else {
|
160 | tryResolve();
|
161 | }
|
162 | });
|
163 | };
|
164 | tryResolve();
|
165 | });
|
166 | }
|
167 | *findReplacements(originalRequest) {
|
168 | if (!this.patterns) {
|
169 | return;
|
170 | }
|
171 | // check if any path mapping rules are relevant
|
172 | for (const { starIndex, prefix, suffix, potentials } of this.patterns) {
|
173 | let partial;
|
174 | if (starIndex === -1) {
|
175 | // No star means an exact match is required
|
176 | if (prefix === originalRequest) {
|
177 | partial = '';
|
178 | }
|
179 | }
|
180 | else if (starIndex === 0 && !suffix) {
|
181 | // Everything matches a single wildcard pattern ("*")
|
182 | partial = originalRequest;
|
183 | }
|
184 | else if (!suffix) {
|
185 | // No suffix means the star is at the end of the pattern
|
186 | if (originalRequest.startsWith(prefix)) {
|
187 | partial = originalRequest.slice(prefix.length);
|
188 | }
|
189 | }
|
190 | else {
|
191 | // Star was in the middle of the pattern
|
192 | if (originalRequest.startsWith(prefix) && originalRequest.endsWith(suffix)) {
|
193 | partial = originalRequest.substring(prefix.length, originalRequest.length - suffix.length);
|
194 | }
|
195 | }
|
196 | // If request was not matched, move on to the next pattern
|
197 | if (partial === undefined) {
|
198 | continue;
|
199 | }
|
200 | // Create the full replacement values based on the original request and the potentials
|
201 | // for the successfully matched pattern.
|
202 | for (const { hasStar, prefix, suffix } of potentials) {
|
203 | let replacement = prefix;
|
204 | if (hasStar) {
|
205 | replacement += partial;
|
206 | if (suffix) {
|
207 | replacement += suffix;
|
208 | }
|
209 | }
|
210 | yield replacement;
|
211 | }
|
212 | }
|
213 | }
|
214 | *createReplacementRequests(request, originalRequest) {
|
215 | var _a;
|
216 | for (const replacement of this.findReplacements(originalRequest)) {
|
217 | const targetPath = path.resolve((_a = this.baseUrl) !== null && _a !== void 0 ? _a : '', replacement);
|
218 | // Resolution in the original callee location, but with the updated request
|
219 | // to point to the mapped target location.
|
220 | yield {
|
221 | ...request,
|
222 | request: targetPath,
|
223 | typescriptPathMapped: true,
|
224 | };
|
225 | // If there is no extension. i.e. the target does not refer to an explicit
|
226 | // file, then this is a candidate for module/package resolution.
|
227 | const canBeModule = path.extname(targetPath) === '';
|
228 | if (canBeModule) {
|
229 | // Resolution in the target location, preserving the original request.
|
230 | // This will work with the `resolve-in-package` resolution hook, supporting
|
231 | // package exports for e.g. locally-built APF libraries.
|
232 | yield {
|
233 | ...request,
|
234 | path: targetPath,
|
235 | typescriptPathMapped: true,
|
236 | };
|
237 | }
|
238 | }
|
239 | }
|
240 | }
|
241 | exports.TypeScriptPathsPlugin = TypeScriptPathsPlugin;
|
242 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"paths-plugin.js","sourceRoot":"","sources":["../../../../../../../packages/ngtools/webpack/src/paths-plugin.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 * as path from 'path';\nimport { CompilerOptions } from 'typescript';\nimport type { Resolver } from 'webpack';\n\n// eslint-disable-next-line @typescript-eslint/no-empty-interface\nexport interface TypeScriptPathsPluginOptions extends Pick<CompilerOptions, 'paths' | 'baseUrl'> {}\n\n// Extract ResolverRequest type from Webpack types since it is not directly exported\ntype ResolverRequest = NonNullable<Parameters<Parameters<Resolver['resolve']>[4]>[2]>;\n\ninterface PathPluginResolverRequest extends ResolverRequest {\n  context?: {\n    issuer?: string;\n  };\n  typescriptPathMapped?: boolean;\n}\n\ninterface PathPattern {\n  starIndex: number;\n  prefix: string;\n  suffix?: string;\n  potentials: { hasStar: boolean; prefix: string; suffix?: string }[];\n}\n\nexport class TypeScriptPathsPlugin {\n  private baseUrl?: string;\n  private patterns?: PathPattern[];\n\n  constructor(options?: TypeScriptPathsPluginOptions) {\n    if (options) {\n      this.update(options);\n    }\n  }\n\n  /**\n   * Update the plugin with new path mapping option values.\n   * The options will also be preprocessed to reduce the overhead of individual resolve actions\n   * during a build.\n   *\n   * @param options The `paths` and `baseUrl` options from TypeScript's `CompilerOptions`.\n   */\n  update(options: TypeScriptPathsPluginOptions): void {\n    this.baseUrl = options.baseUrl;\n    this.patterns = undefined;\n\n    if (options.paths) {\n      for (const [pattern, potentials] of Object.entries(options.paths)) {\n        // Ignore any entries that would not result in a new mapping\n        if (potentials.length === 0 || potentials.every((potential) => potential === '*')) {\n          continue;\n        }\n\n        const starIndex = pattern.indexOf('*');\n        let prefix = pattern;\n        let suffix;\n        if (starIndex > -1) {\n          prefix = pattern.slice(0, starIndex);\n          if (starIndex < pattern.length - 1) {\n            suffix = pattern.slice(starIndex + 1);\n          }\n        }\n\n        this.patterns ??= [];\n        this.patterns.push({\n          starIndex,\n          prefix,\n          suffix,\n          potentials: potentials.map((potential) => {\n            const potentialStarIndex = potential.indexOf('*');\n            if (potentialStarIndex === -1) {\n              return { hasStar: false, prefix: potential };\n            }\n\n            return {\n              hasStar: true,\n              prefix: potential.slice(0, potentialStarIndex),\n              suffix:\n                potentialStarIndex < potential.length - 1\n                  ? potential.slice(potentialStarIndex + 1)\n                  : undefined,\n            };\n          }),\n        });\n      }\n\n      // Sort patterns so that exact matches take priority then largest prefix match\n      this.patterns?.sort((a, b) => {\n        if (a.starIndex === -1) {\n          return -1;\n        } else if (b.starIndex === -1) {\n          return 1;\n        } else {\n          return b.starIndex - a.starIndex;\n        }\n      });\n    }\n  }\n\n  apply(resolver: Resolver): void {\n    const target = resolver.ensureHook('resolve');\n\n    // To support synchronous resolvers this hook cannot be promise based.\n    // Webpack supports synchronous resolution with `tap` and `tapAsync` hooks.\n    resolver\n      .getHook('described-resolve')\n      .tapAsync(\n        'TypeScriptPathsPlugin',\n        (request: PathPluginResolverRequest, resolveContext, callback) => {\n          // Preprocessing of the options will ensure that `patterns` is either undefined or has elements to check\n          if (!this.patterns) {\n            callback();\n\n            return;\n          }\n\n          if (!request || request.typescriptPathMapped) {\n            callback();\n\n            return;\n          }\n\n          const originalRequest = request.request || request.path;\n          if (!originalRequest) {\n            callback();\n\n            return;\n          }\n\n          // Only work on Javascript/TypeScript issuers.\n          if (!request?.context?.issuer?.match(/\\.[cm]?[jt]sx?$/)) {\n            callback();\n\n            return;\n          }\n\n          switch (originalRequest[0]) {\n            case '.':\n            case '/':\n              // Relative or absolute requests are not mapped\n              callback();\n\n              return;\n            case '!':\n              // Ignore all webpack special requests\n              if (originalRequest.length > 1 && originalRequest[1] === '!') {\n                callback();\n\n                return;\n              }\n              break;\n          }\n\n          // A generator is used to limit the amount of replacements requests that need to be created.\n          // For example, if the first one resolves, any others are not needed and do not need\n          // to be created.\n          const requests = this.createReplacementRequests(request, originalRequest);\n\n          const tryResolve = () => {\n            const next = requests.next();\n            if (next.done) {\n              callback();\n\n              return;\n            }\n\n            resolver.doResolve(\n              target,\n              next.value,\n              '',\n              resolveContext,\n              (error: Error | null | undefined, result: ResolverRequest | null | undefined) => {\n                if (error) {\n                  callback(error);\n                } else if (result) {\n                  callback(undefined, result);\n                } else {\n                  tryResolve();\n                }\n              },\n            );\n          };\n\n          tryResolve();\n        },\n      );\n  }\n\n  *findReplacements(originalRequest: string): IterableIterator<string> {\n    if (!this.patterns) {\n      return;\n    }\n\n    // check if any path mapping rules are relevant\n    for (const { starIndex, prefix, suffix, potentials } of this.patterns) {\n      let partial;\n\n      if (starIndex === -1) {\n        // No star means an exact match is required\n        if (prefix === originalRequest) {\n          partial = '';\n        }\n      } else if (starIndex === 0 && !suffix) {\n        // Everything matches a single wildcard pattern (\"*\")\n        partial = originalRequest;\n      } else if (!suffix) {\n        // No suffix means the star is at the end of the pattern\n        if (originalRequest.startsWith(prefix)) {\n          partial = originalRequest.slice(prefix.length);\n        }\n      } else {\n        // Star was in the middle of the pattern\n        if (originalRequest.startsWith(prefix) && originalRequest.endsWith(suffix)) {\n          partial = originalRequest.substring(\n            prefix.length,\n            originalRequest.length - suffix.length,\n          );\n        }\n      }\n\n      // If request was not matched, move on to the next pattern\n      if (partial === undefined) {\n        continue;\n      }\n\n      // Create the full replacement values based on the original request and the potentials\n      // for the successfully matched pattern.\n      for (const { hasStar, prefix, suffix } of potentials) {\n        let replacement = prefix;\n\n        if (hasStar) {\n          replacement += partial;\n          if (suffix) {\n            replacement += suffix;\n          }\n        }\n\n        yield replacement;\n      }\n    }\n  }\n\n  *createReplacementRequests(\n    request: PathPluginResolverRequest,\n    originalRequest: string,\n  ): IterableIterator<PathPluginResolverRequest> {\n    for (const replacement of this.findReplacements(originalRequest)) {\n      const targetPath = path.resolve(this.baseUrl ?? '', replacement);\n      // Resolution in the original callee location, but with the updated request\n      // to point to the mapped target location.\n      yield {\n        ...request,\n        request: targetPath,\n        typescriptPathMapped: true,\n      };\n\n      // If there is no extension. i.e. the target does not refer to an explicit\n      // file, then this is a candidate for module/package resolution.\n      const canBeModule = path.extname(targetPath) === '';\n      if (canBeModule) {\n        // Resolution in the target location, preserving the original request.\n        // This will work with the `resolve-in-package` resolution hook, supporting\n        // package exports for e.g. locally-built APF libraries.\n        yield {\n          ...request,\n          path: targetPath,\n          typescriptPathMapped: true,\n        };\n      }\n    }\n  }\n}\n"]} |
\ | No newline at end of file |