source: trip-planner-front/node_modules/@angular/compiler-cli/ngcc/src/entry_point_finder/utils.js

Last change on this file was 6a3a178, checked in by Ema <ema_spirova@…>, 3 years ago

initial commit

  • Property mode set to 100644
File size: 26.7 KB
Line 
1(function (factory) {
2 if (typeof module === "object" && typeof module.exports === "object") {
3 var v = factory(require, exports);
4 if (v !== undefined) module.exports = v;
5 }
6 else if (typeof define === "function" && define.amd) {
7 define("@angular/compiler-cli/ngcc/src/entry_point_finder/utils", ["require", "exports", "tslib", "@angular/compiler-cli/src/ngtsc/file_system"], factory);
8 }
9})(function (require, exports) {
10 "use strict";
11 Object.defineProperty(exports, "__esModule", { value: true });
12 exports.trackDuration = exports.getBasePaths = void 0;
13 var tslib_1 = require("tslib");
14 /**
15 * @license
16 * Copyright Google LLC All Rights Reserved.
17 *
18 * Use of this source code is governed by an MIT-style license that can be
19 * found in the LICENSE file at https://angular.io/license
20 */
21 var file_system_1 = require("@angular/compiler-cli/src/ngtsc/file_system");
22 /**
23 * Extract all the base-paths that we need to search for entry-points.
24 *
25 * This always contains the standard base-path (`sourceDirectory`).
26 * But it also parses the `paths` mappings object to guess additional base-paths.
27 *
28 * For example:
29 *
30 * ```
31 * getBasePaths('/node_modules', {baseUrl: '/dist', paths: {'*': ['lib/*', 'lib/generated/*']}})
32 * > ['/node_modules', '/dist/lib']
33 * ```
34 *
35 * Notice that `'/dist'` is not included as there is no `'*'` path,
36 * and `'/dist/lib/generated'` is not included as it is covered by `'/dist/lib'`.
37 *
38 * @param sourceDirectory The standard base-path (e.g. node_modules).
39 * @param pathMappings Path mapping configuration, from which to extract additional base-paths.
40 */
41 function getBasePaths(logger, sourceDirectory, pathMappings) {
42 var e_1, _a, e_2, _b, e_3, _c;
43 var fs = file_system_1.getFileSystem();
44 var basePaths = [sourceDirectory];
45 if (pathMappings) {
46 var baseUrl = fs.resolve(pathMappings.baseUrl);
47 if (fs.isRoot(baseUrl)) {
48 logger.warn("The provided pathMappings baseUrl is the root path " + baseUrl + ".\n" +
49 "This is likely to mess up how ngcc finds entry-points and is probably not correct.\n" +
50 "Please check your path mappings configuration such as in the tsconfig.json file.");
51 }
52 try {
53 for (var _d = tslib_1.__values(Object.values(pathMappings.paths)), _e = _d.next(); !_e.done; _e = _d.next()) {
54 var paths = _e.value;
55 try {
56 for (var paths_1 = (e_2 = void 0, tslib_1.__values(paths)), paths_1_1 = paths_1.next(); !paths_1_1.done; paths_1_1 = paths_1.next()) {
57 var path = paths_1_1.value;
58 var foundMatch = false;
59 // We only want base paths that exist and are not files
60 var _f = extractPathPrefix(path), prefix = _f.prefix, hasWildcard = _f.hasWildcard;
61 var basePath = fs.resolve(baseUrl, prefix);
62 if (fs.exists(basePath) && fs.stat(basePath).isFile()) {
63 basePath = fs.dirname(basePath);
64 }
65 if (fs.exists(basePath)) {
66 // The `basePath` is itself a directory
67 basePaths.push(basePath);
68 foundMatch = true;
69 }
70 if (hasWildcard) {
71 // The path contains a wildcard (`*`) so also try searching for directories that start
72 // with the wildcard prefix path segment.
73 var wildcardContainer = fs.dirname(basePath);
74 var wildcardPrefix = fs.basename(basePath);
75 if (isExistingDirectory(fs, wildcardContainer)) {
76 var candidates = fs.readdir(wildcardContainer);
77 try {
78 for (var candidates_1 = (e_3 = void 0, tslib_1.__values(candidates)), candidates_1_1 = candidates_1.next(); !candidates_1_1.done; candidates_1_1 = candidates_1.next()) {
79 var candidate = candidates_1_1.value;
80 if (candidate.startsWith(wildcardPrefix)) {
81 var candidatePath = fs.resolve(wildcardContainer, candidate);
82 if (isExistingDirectory(fs, candidatePath)) {
83 foundMatch = true;
84 basePaths.push(candidatePath);
85 }
86 }
87 }
88 }
89 catch (e_3_1) { e_3 = { error: e_3_1 }; }
90 finally {
91 try {
92 if (candidates_1_1 && !candidates_1_1.done && (_c = candidates_1.return)) _c.call(candidates_1);
93 }
94 finally { if (e_3) throw e_3.error; }
95 }
96 }
97 }
98 if (!foundMatch) {
99 // We neither found a direct match (i.e. `basePath` is an existing directory) nor a
100 // directory that starts with a wildcard prefix.
101 logger.debug("The basePath \"" + basePath + "\" computed from baseUrl \"" + baseUrl + "\" and path mapping \"" + path + "\" does not exist in the file-system.\n" +
102 "It will not be scanned for entry-points.");
103 }
104 }
105 }
106 catch (e_2_1) { e_2 = { error: e_2_1 }; }
107 finally {
108 try {
109 if (paths_1_1 && !paths_1_1.done && (_b = paths_1.return)) _b.call(paths_1);
110 }
111 finally { if (e_2) throw e_2.error; }
112 }
113 }
114 }
115 catch (e_1_1) { e_1 = { error: e_1_1 }; }
116 finally {
117 try {
118 if (_e && !_e.done && (_a = _d.return)) _a.call(_d);
119 }
120 finally { if (e_1) throw e_1.error; }
121 }
122 }
123 var dedupedBasePaths = dedupePaths(fs, basePaths);
124 // We want to ensure that the `sourceDirectory` is included when it is a node_modules folder.
125 // Otherwise our entry-point finding algorithm would fail to walk that folder.
126 if (fs.basename(sourceDirectory) === 'node_modules' &&
127 !dedupedBasePaths.includes(sourceDirectory)) {
128 dedupedBasePaths.unshift(sourceDirectory);
129 }
130 return dedupedBasePaths;
131 }
132 exports.getBasePaths = getBasePaths;
133 function isExistingDirectory(fs, path) {
134 return fs.exists(path) && fs.stat(path).isDirectory();
135 }
136 /**
137 * Extract everything in the `path` up to the first `*`.
138 * @param path The path to parse.
139 * @returns The extracted prefix and a flag to indicate whether there was a wildcard `*`.
140 */
141 function extractPathPrefix(path) {
142 var _a = tslib_1.__read(path.split('*', 2), 2), prefix = _a[0], rest = _a[1];
143 return { prefix: prefix, hasWildcard: rest !== undefined };
144 }
145 /**
146 * Run a task and track how long it takes.
147 *
148 * @param task The task whose duration we are tracking.
149 * @param log The function to call with the duration of the task.
150 * @returns The result of calling `task`.
151 */
152 function trackDuration(task, log) {
153 var startTime = Date.now();
154 var result = task();
155 var duration = Math.round((Date.now() - startTime) / 100) / 10;
156 log(duration);
157 return result;
158 }
159 exports.trackDuration = trackDuration;
160 /**
161 * Remove paths that are contained by other paths.
162 *
163 * For example:
164 * Given `['a/b/c', 'a/b/x', 'a/b', 'd/e', 'd/f']` we will end up with `['a/b', 'd/e', 'd/f]`.
165 * (Note that we do not get `d` even though `d/e` and `d/f` share a base directory, since `d` is not
166 * one of the base paths.)
167 */
168 function dedupePaths(fs, paths) {
169 var e_4, _a;
170 var root = { children: new Map() };
171 try {
172 for (var paths_2 = tslib_1.__values(paths), paths_2_1 = paths_2.next(); !paths_2_1.done; paths_2_1 = paths_2.next()) {
173 var path = paths_2_1.value;
174 addPath(fs, root, path);
175 }
176 }
177 catch (e_4_1) { e_4 = { error: e_4_1 }; }
178 finally {
179 try {
180 if (paths_2_1 && !paths_2_1.done && (_a = paths_2.return)) _a.call(paths_2);
181 }
182 finally { if (e_4) throw e_4.error; }
183 }
184 return flattenTree(root);
185 }
186 /**
187 * Add a path (defined by the `segments`) to the current `node` in the tree.
188 */
189 function addPath(fs, root, path) {
190 var node = root;
191 if (!fs.isRoot(path)) {
192 var segments = path.split('/');
193 for (var index = 0; index < segments.length; index++) {
194 if (isLeaf(node)) {
195 // We hit a leaf so don't bother processing any more of the path
196 return;
197 }
198 // This is not the end of the path continue to process the rest of this path.
199 var next = segments[index];
200 if (!node.children.has(next)) {
201 node.children.set(next, { children: new Map() });
202 }
203 node = node.children.get(next);
204 }
205 }
206 // This path has finished so convert this node to a leaf
207 convertToLeaf(node, path);
208 }
209 /**
210 * Flatten the tree of nodes back into an array of absolute paths.
211 */
212 function flattenTree(root) {
213 var paths = [];
214 var nodes = [root];
215 for (var index = 0; index < nodes.length; index++) {
216 var node = nodes[index];
217 if (isLeaf(node)) {
218 // We found a leaf so store the currentPath
219 paths.push(node.path);
220 }
221 else {
222 node.children.forEach(function (value) { return nodes.push(value); });
223 }
224 }
225 return paths;
226 }
227 function isLeaf(node) {
228 return node.path !== undefined;
229 }
230 function convertToLeaf(node, path) {
231 node.path = path;
232 }
233});
234//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"utils.js","sourceRoot":"","sources":["../../../../../../../../packages/compiler-cli/ngcc/src/entry_point_finder/utils.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 */\nimport {AbsoluteFsPath, getFileSystem, PathManipulation, ReadonlyFileSystem} from '../../../src/ngtsc/file_system';\nimport {Logger} from '../../../src/ngtsc/logging';\nimport {PathMappings} from '../path_mappings';\n\n/**\n * Extract all the base-paths that we need to search for entry-points.\n *\n * This always contains the standard base-path (`sourceDirectory`).\n * But it also parses the `paths` mappings object to guess additional base-paths.\n *\n * For example:\n *\n * ```\n * getBasePaths('/node_modules', {baseUrl: '/dist', paths: {'*': ['lib/*', 'lib/generated/*']}})\n * > ['/node_modules', '/dist/lib']\n * ```\n *\n * Notice that `'/dist'` is not included as there is no `'*'` path,\n * and `'/dist/lib/generated'` is not included as it is covered by `'/dist/lib'`.\n *\n * @param sourceDirectory The standard base-path (e.g. node_modules).\n * @param pathMappings Path mapping configuration, from which to extract additional base-paths.\n */\nexport function getBasePaths(\n    logger: Logger, sourceDirectory: AbsoluteFsPath,\n    pathMappings: PathMappings|undefined): AbsoluteFsPath[] {\n  const fs = getFileSystem();\n  const basePaths = [sourceDirectory];\n  if (pathMappings) {\n    const baseUrl = fs.resolve(pathMappings.baseUrl);\n    if (fs.isRoot(baseUrl)) {\n      logger.warn(\n          `The provided pathMappings baseUrl is the root path ${baseUrl}.\\n` +\n          `This is likely to mess up how ngcc finds entry-points and is probably not correct.\\n` +\n          `Please check your path mappings configuration such as in the tsconfig.json file.`);\n    }\n    for (const paths of Object.values(pathMappings.paths)) {\n      for (const path of paths) {\n        let foundMatch = false;\n\n        // We only want base paths that exist and are not files\n        const {prefix, hasWildcard} = extractPathPrefix(path);\n        let basePath = fs.resolve(baseUrl, prefix);\n        if (fs.exists(basePath) && fs.stat(basePath).isFile()) {\n          basePath = fs.dirname(basePath);\n        }\n\n        if (fs.exists(basePath)) {\n          // The `basePath` is itself a directory\n          basePaths.push(basePath);\n          foundMatch = true;\n        }\n\n        if (hasWildcard) {\n          // The path contains a wildcard (`*`) so also try searching for directories that start\n          // with the wildcard prefix path segment.\n          const wildcardContainer = fs.dirname(basePath);\n          const wildcardPrefix = fs.basename(basePath);\n          if (isExistingDirectory(fs, wildcardContainer)) {\n            const candidates = fs.readdir(wildcardContainer);\n            for (const candidate of candidates) {\n              if (candidate.startsWith(wildcardPrefix)) {\n                const candidatePath = fs.resolve(wildcardContainer, candidate);\n                if (isExistingDirectory(fs, candidatePath)) {\n                  foundMatch = true;\n                  basePaths.push(candidatePath);\n                }\n              }\n            }\n          }\n        }\n\n        if (!foundMatch) {\n          // We neither found a direct match (i.e. `basePath` is an existing directory) nor a\n          // directory that starts with a wildcard prefix.\n          logger.debug(\n              `The basePath \"${basePath}\" computed from baseUrl \"${baseUrl}\" and path mapping \"${\n                  path}\" does not exist in the file-system.\\n` +\n              `It will not be scanned for entry-points.`);\n        }\n      }\n    }\n  }\n\n  const dedupedBasePaths = dedupePaths(fs, basePaths);\n\n  // We want to ensure that the `sourceDirectory` is included when it is a node_modules folder.\n  // Otherwise our entry-point finding algorithm would fail to walk that folder.\n  if (fs.basename(sourceDirectory) === 'node_modules' &&\n      !dedupedBasePaths.includes(sourceDirectory)) {\n    dedupedBasePaths.unshift(sourceDirectory);\n  }\n\n  return dedupedBasePaths;\n}\n\nfunction isExistingDirectory(fs: ReadonlyFileSystem, path: AbsoluteFsPath): boolean {\n  return fs.exists(path) && fs.stat(path).isDirectory();\n}\n\n/**\n * Extract everything in the `path` up to the first `*`.\n * @param path The path to parse.\n * @returns The extracted prefix and a flag to indicate whether there was a wildcard `*`.\n */\nfunction extractPathPrefix(path: string): {prefix: string, hasWildcard: boolean} {\n  const [prefix, rest] = path.split('*', 2);\n  return {prefix, hasWildcard: rest !== undefined};\n}\n\n/**\n * Run a task and track how long it takes.\n *\n * @param task The task whose duration we are tracking.\n * @param log The function to call with the duration of the task.\n * @returns The result of calling `task`.\n */\nexport function trackDuration<T = void>(task: () => T extends Promise<unknown>? never : T,\n                                                              log: (duration: number) => void): T {\n  const startTime = Date.now();\n  const result = task();\n  const duration = Math.round((Date.now() - startTime) / 100) / 10;\n  log(duration);\n  return result;\n}\n\n/**\n * Remove paths that are contained by other paths.\n *\n * For example:\n * Given `['a/b/c', 'a/b/x', 'a/b', 'd/e', 'd/f']` we will end up with `['a/b', 'd/e', 'd/f]`.\n * (Note that we do not get `d` even though `d/e` and `d/f` share a base directory, since `d` is not\n * one of the base paths.)\n */\nfunction dedupePaths(fs: PathManipulation, paths: AbsoluteFsPath[]): AbsoluteFsPath[] {\n  const root: Node = {children: new Map()};\n  for (const path of paths) {\n    addPath(fs, root, path);\n  }\n  return flattenTree(root);\n}\n\n/**\n * Add a path (defined by the `segments`) to the current `node` in the tree.\n */\nfunction addPath(fs: PathManipulation, root: Node, path: AbsoluteFsPath): void {\n  let node = root;\n  if (!fs.isRoot(path)) {\n    const segments = path.split('/');\n    for (let index = 0; index < segments.length; index++) {\n      if (isLeaf(node)) {\n        // We hit a leaf so don't bother processing any more of the path\n        return;\n      }\n      // This is not the end of the path continue to process the rest of this path.\n      const next = segments[index];\n      if (!node.children.has(next)) {\n        node.children.set(next, {children: new Map()});\n      }\n      node = node.children.get(next)!;\n    }\n  }\n  // This path has finished so convert this node to a leaf\n  convertToLeaf(node, path);\n}\n\n/**\n * Flatten the tree of nodes back into an array of absolute paths.\n */\nfunction flattenTree(root: Node): AbsoluteFsPath[] {\n  const paths: AbsoluteFsPath[] = [];\n  const nodes: Node[] = [root];\n  for (let index = 0; index < nodes.length; index++) {\n    const node = nodes[index];\n    if (isLeaf(node)) {\n      // We found a leaf so store the currentPath\n      paths.push(node.path);\n    } else {\n      node.children.forEach(value => nodes.push(value));\n    }\n  }\n  return paths;\n}\n\nfunction isLeaf(node: Node): node is Leaf {\n  return node.path !== undefined;\n}\n\nfunction convertToLeaf(node: Node, path: AbsoluteFsPath) {\n  node.path = path;\n}\n\ninterface Node {\n  children: Map<string, Node>;\n  path?: AbsoluteFsPath;\n}\n\ntype Leaf = Required<Node>;\n"]}
Note: See TracBrowser for help on using the repository browser.