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/execution/tasks/utils", ["require", "exports", "tslib", "@angular/compiler-cli/ngcc/src/execution/tasks/api"], factory);
|
---|
8 | }
|
---|
9 | })(function (require, exports) {
|
---|
10 | "use strict";
|
---|
11 | Object.defineProperty(exports, "__esModule", { value: true });
|
---|
12 | exports.sortTasksByPriority = exports.getBlockedTasks = exports.getDependentsSet = exports.computeTaskDependencies = exports.stringifyTask = void 0;
|
---|
13 | var tslib_1 = require("tslib");
|
---|
14 | var api_1 = require("@angular/compiler-cli/ngcc/src/execution/tasks/api");
|
---|
15 | /** Stringify a task for debugging purposes. */
|
---|
16 | var stringifyTask = function (task) {
|
---|
17 | return "{entryPoint: " + task.entryPoint.name + ", formatProperty: " + task.formatProperty + ", " +
|
---|
18 | ("processDts: " + api_1.DtsProcessing[task.processDts] + "}");
|
---|
19 | };
|
---|
20 | exports.stringifyTask = stringifyTask;
|
---|
21 | /**
|
---|
22 | * Compute a mapping of tasks to the tasks that are dependent on them (if any).
|
---|
23 | *
|
---|
24 | * Task A can depend upon task B, if either:
|
---|
25 | *
|
---|
26 | * * A and B have the same entry-point _and_ B is generating the typings for that entry-point
|
---|
27 | * (i.e. has `processDts: true`).
|
---|
28 | * * A's entry-point depends on B's entry-point _and_ B is also generating typings.
|
---|
29 | *
|
---|
30 | * NOTE: If a task is not generating typings, then it cannot affect anything which depends on its
|
---|
31 | * entry-point, regardless of the dependency graph. To put this another way, only the task
|
---|
32 | * which produces the typings for a dependency needs to have been completed.
|
---|
33 | *
|
---|
34 | * As a performance optimization, we take into account the fact that `tasks` are sorted in such a
|
---|
35 | * way that a task can only depend on earlier tasks (i.e. dependencies always come before
|
---|
36 | * dependents in the list of tasks).
|
---|
37 | *
|
---|
38 | * @param tasks A (partially ordered) list of tasks.
|
---|
39 | * @param graph The dependency graph between entry-points.
|
---|
40 | * @return A map from each task to those tasks directly dependent upon it.
|
---|
41 | */
|
---|
42 | function computeTaskDependencies(tasks, graph) {
|
---|
43 | var dependencies = new api_1.TaskDependencies();
|
---|
44 | var candidateDependencies = new Map();
|
---|
45 | tasks.forEach(function (task) {
|
---|
46 | var e_1, _a;
|
---|
47 | var entryPointPath = task.entryPoint.path;
|
---|
48 | // Find the earlier tasks (`candidateDependencies`) that this task depends upon.
|
---|
49 | var deps = graph.dependenciesOf(entryPointPath);
|
---|
50 | var taskDependencies = deps.filter(function (dep) { return candidateDependencies.has(dep); })
|
---|
51 | .map(function (dep) { return candidateDependencies.get(dep); });
|
---|
52 | // If this task has dependencies, add it to the dependencies and dependents maps.
|
---|
53 | if (taskDependencies.length > 0) {
|
---|
54 | try {
|
---|
55 | for (var taskDependencies_1 = tslib_1.__values(taskDependencies), taskDependencies_1_1 = taskDependencies_1.next(); !taskDependencies_1_1.done; taskDependencies_1_1 = taskDependencies_1.next()) {
|
---|
56 | var dependency = taskDependencies_1_1.value;
|
---|
57 | var taskDependents = getDependentsSet(dependencies, dependency);
|
---|
58 | taskDependents.add(task);
|
---|
59 | }
|
---|
60 | }
|
---|
61 | catch (e_1_1) { e_1 = { error: e_1_1 }; }
|
---|
62 | finally {
|
---|
63 | try {
|
---|
64 | if (taskDependencies_1_1 && !taskDependencies_1_1.done && (_a = taskDependencies_1.return)) _a.call(taskDependencies_1);
|
---|
65 | }
|
---|
66 | finally { if (e_1) throw e_1.error; }
|
---|
67 | }
|
---|
68 | }
|
---|
69 | if (task.processDts !== api_1.DtsProcessing.No) {
|
---|
70 | // SANITY CHECK:
|
---|
71 | // There should only be one task per entry-point that generates typings (and thus can be a
|
---|
72 | // dependency of other tasks), so the following should theoretically never happen, but check
|
---|
73 | // just in case.
|
---|
74 | if (candidateDependencies.has(entryPointPath)) {
|
---|
75 | var otherTask = candidateDependencies.get(entryPointPath);
|
---|
76 | throw new Error('Invariant violated: Multiple tasks are assigned generating typings for ' +
|
---|
77 | ("'" + entryPointPath + "':\n - " + exports.stringifyTask(otherTask) + "\n - " + exports.stringifyTask(task)));
|
---|
78 | }
|
---|
79 | // This task can potentially be a dependency (i.e. it generates typings), so add it to the
|
---|
80 | // list of candidate dependencies for subsequent tasks.
|
---|
81 | candidateDependencies.set(entryPointPath, task);
|
---|
82 | }
|
---|
83 | else {
|
---|
84 | // This task is not generating typings so we need to add it to the dependents of the task that
|
---|
85 | // does generate typings, if that exists
|
---|
86 | if (candidateDependencies.has(entryPointPath)) {
|
---|
87 | var typingsTask = candidateDependencies.get(entryPointPath);
|
---|
88 | var typingsTaskDependents = getDependentsSet(dependencies, typingsTask);
|
---|
89 | typingsTaskDependents.add(task);
|
---|
90 | }
|
---|
91 | }
|
---|
92 | });
|
---|
93 | return dependencies;
|
---|
94 | }
|
---|
95 | exports.computeTaskDependencies = computeTaskDependencies;
|
---|
96 | function getDependentsSet(map, task) {
|
---|
97 | if (!map.has(task)) {
|
---|
98 | map.set(task, new Set());
|
---|
99 | }
|
---|
100 | return map.get(task);
|
---|
101 | }
|
---|
102 | exports.getDependentsSet = getDependentsSet;
|
---|
103 | /**
|
---|
104 | * Invert the given mapping of Task dependencies.
|
---|
105 | *
|
---|
106 | * @param dependencies The mapping of tasks to the tasks that depend upon them.
|
---|
107 | * @returns A mapping of tasks to the tasks that they depend upon.
|
---|
108 | */
|
---|
109 | function getBlockedTasks(dependencies) {
|
---|
110 | var e_2, _a, e_3, _b;
|
---|
111 | var blockedTasks = new Map();
|
---|
112 | try {
|
---|
113 | for (var dependencies_1 = tslib_1.__values(dependencies), dependencies_1_1 = dependencies_1.next(); !dependencies_1_1.done; dependencies_1_1 = dependencies_1.next()) {
|
---|
114 | var _c = tslib_1.__read(dependencies_1_1.value, 2), dependency = _c[0], dependents = _c[1];
|
---|
115 | try {
|
---|
116 | for (var dependents_1 = (e_3 = void 0, tslib_1.__values(dependents)), dependents_1_1 = dependents_1.next(); !dependents_1_1.done; dependents_1_1 = dependents_1.next()) {
|
---|
117 | var dependent = dependents_1_1.value;
|
---|
118 | var dependentSet = getDependentsSet(blockedTasks, dependent);
|
---|
119 | dependentSet.add(dependency);
|
---|
120 | }
|
---|
121 | }
|
---|
122 | catch (e_3_1) { e_3 = { error: e_3_1 }; }
|
---|
123 | finally {
|
---|
124 | try {
|
---|
125 | if (dependents_1_1 && !dependents_1_1.done && (_b = dependents_1.return)) _b.call(dependents_1);
|
---|
126 | }
|
---|
127 | finally { if (e_3) throw e_3.error; }
|
---|
128 | }
|
---|
129 | }
|
---|
130 | }
|
---|
131 | catch (e_2_1) { e_2 = { error: e_2_1 }; }
|
---|
132 | finally {
|
---|
133 | try {
|
---|
134 | if (dependencies_1_1 && !dependencies_1_1.done && (_a = dependencies_1.return)) _a.call(dependencies_1);
|
---|
135 | }
|
---|
136 | finally { if (e_2) throw e_2.error; }
|
---|
137 | }
|
---|
138 | return blockedTasks;
|
---|
139 | }
|
---|
140 | exports.getBlockedTasks = getBlockedTasks;
|
---|
141 | /**
|
---|
142 | * Sort a list of tasks by priority.
|
---|
143 | *
|
---|
144 | * Priority is determined by the number of other tasks that a task is (transitively) blocking:
|
---|
145 | * The more tasks a task is blocking the higher its priority is, because processing it will
|
---|
146 | * potentially unblock more tasks.
|
---|
147 | *
|
---|
148 | * To keep the behavior predictable, if two tasks block the same number of other tasks, their
|
---|
149 | * relative order in the original `tasks` lists is preserved.
|
---|
150 | *
|
---|
151 | * @param tasks A (partially ordered) list of tasks.
|
---|
152 | * @param dependencies The mapping of tasks to the tasks that depend upon them.
|
---|
153 | * @return The list of tasks sorted by priority.
|
---|
154 | */
|
---|
155 | function sortTasksByPriority(tasks, dependencies) {
|
---|
156 | var priorityPerTask = new Map();
|
---|
157 | var computePriority = function (task, idx) { return [dependencies.has(task) ? dependencies.get(task).size : 0, idx]; };
|
---|
158 | tasks.forEach(function (task, i) { return priorityPerTask.set(task, computePriority(task, i)); });
|
---|
159 | return tasks.slice().sort(function (task1, task2) {
|
---|
160 | var _a = tslib_1.__read(priorityPerTask.get(task1), 2), p1 = _a[0], idx1 = _a[1];
|
---|
161 | var _b = tslib_1.__read(priorityPerTask.get(task2), 2), p2 = _b[0], idx2 = _b[1];
|
---|
162 | return (p2 - p1) || (idx1 - idx2);
|
---|
163 | });
|
---|
164 | }
|
---|
165 | exports.sortTasksByPriority = sortTasksByPriority;
|
---|
166 | });
|
---|
167 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"utils.js","sourceRoot":"","sources":["../../../../../../../../../packages/compiler-cli/ngcc/src/execution/tasks/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 {DepGraph} from 'dependency-graph';\nimport {EntryPoint} from '../../packages/entry_point';\nimport {DtsProcessing, PartiallyOrderedTasks, Task, TaskDependencies} from './api';\n\n/** Stringify a task for debugging purposes. */\nexport const stringifyTask = (task: Task): string =>\n    `{entryPoint: ${task.entryPoint.name}, formatProperty: ${task.formatProperty}, ` +\n    `processDts: ${DtsProcessing[task.processDts]}}`;\n\n/**\n * Compute a mapping of tasks to the tasks that are dependent on them (if any).\n *\n * Task A can depend upon task B, if either:\n *\n * * A and B have the same entry-point _and_ B is generating the typings for that entry-point\n *   (i.e. has `processDts: true`).\n * * A's entry-point depends on B's entry-point _and_ B is also generating typings.\n *\n * NOTE: If a task is not generating typings, then it cannot affect anything which depends on its\n *       entry-point, regardless of the dependency graph. To put this another way, only the task\n *       which produces the typings for a dependency needs to have been completed.\n *\n * As a performance optimization, we take into account the fact that `tasks` are sorted in such a\n * way that a task can only depend on earlier tasks (i.e. dependencies always come before\n * dependents in the list of tasks).\n *\n * @param tasks A (partially ordered) list of tasks.\n * @param graph The dependency graph between entry-points.\n * @return A map from each task to those tasks directly dependent upon it.\n */\nexport function computeTaskDependencies(\n    tasks: PartiallyOrderedTasks, graph: DepGraph<EntryPoint>): TaskDependencies {\n  const dependencies = new TaskDependencies();\n  const candidateDependencies = new Map<string, Task>();\n\n  tasks.forEach(task => {\n    const entryPointPath = task.entryPoint.path;\n\n    // Find the earlier tasks (`candidateDependencies`) that this task depends upon.\n    const deps = graph.dependenciesOf(entryPointPath);\n    const taskDependencies = deps.filter(dep => candidateDependencies.has(dep))\n                                 .map(dep => candidateDependencies.get(dep)!);\n\n    // If this task has dependencies, add it to the dependencies and dependents maps.\n    if (taskDependencies.length > 0) {\n      for (const dependency of taskDependencies) {\n        const taskDependents = getDependentsSet(dependencies, dependency);\n        taskDependents.add(task);\n      }\n    }\n\n    if (task.processDts !== DtsProcessing.No) {\n      // SANITY CHECK:\n      // There should only be one task per entry-point that generates typings (and thus can be a\n      // dependency of other tasks), so the following should theoretically never happen, but check\n      // just in case.\n      if (candidateDependencies.has(entryPointPath)) {\n        const otherTask = candidateDependencies.get(entryPointPath)!;\n        throw new Error(\n            'Invariant violated: Multiple tasks are assigned generating typings for ' +\n            `'${entryPointPath}':\\n  - ${stringifyTask(otherTask)}\\n  - ${stringifyTask(task)}`);\n      }\n      // This task can potentially be a dependency (i.e. it generates typings), so add it to the\n      // list of candidate dependencies for subsequent tasks.\n      candidateDependencies.set(entryPointPath, task);\n    } else {\n      // This task is not generating typings so we need to add it to the dependents of the task that\n      // does generate typings, if that exists\n      if (candidateDependencies.has(entryPointPath)) {\n        const typingsTask = candidateDependencies.get(entryPointPath)!;\n        const typingsTaskDependents = getDependentsSet(dependencies, typingsTask);\n        typingsTaskDependents.add(task);\n      }\n    }\n  });\n\n  return dependencies;\n}\n\nexport function getDependentsSet(map: TaskDependencies, task: Task): Set<Task> {\n  if (!map.has(task)) {\n    map.set(task, new Set());\n  }\n  return map.get(task)!;\n}\n\n/**\n * Invert the given mapping of Task dependencies.\n *\n * @param dependencies The mapping of tasks to the tasks that depend upon them.\n * @returns A mapping of tasks to the tasks that they depend upon.\n */\nexport function getBlockedTasks(dependencies: TaskDependencies): Map<Task, Set<Task>> {\n  const blockedTasks = new Map<Task, Set<Task>>();\n  for (const [dependency, dependents] of dependencies) {\n    for (const dependent of dependents) {\n      const dependentSet = getDependentsSet(blockedTasks, dependent);\n      dependentSet.add(dependency);\n    }\n  }\n  return blockedTasks;\n}\n\n/**\n * Sort a list of tasks by priority.\n *\n * Priority is determined by the number of other tasks that a task is (transitively) blocking:\n * The more tasks a task is blocking the higher its priority is, because processing it will\n * potentially unblock more tasks.\n *\n * To keep the behavior predictable, if two tasks block the same number of other tasks, their\n * relative order in the original `tasks` lists is preserved.\n *\n * @param tasks A (partially ordered) list of tasks.\n * @param dependencies The mapping of tasks to the tasks that depend upon them.\n * @return The list of tasks sorted by priority.\n */\nexport function sortTasksByPriority(\n    tasks: PartiallyOrderedTasks, dependencies: TaskDependencies): PartiallyOrderedTasks {\n  const priorityPerTask = new Map<Task, [number, number]>();\n  const computePriority = (task: Task, idx: number):\n      [number, number] => [dependencies.has(task) ? dependencies.get(task)!.size : 0, idx];\n\n  tasks.forEach((task, i) => priorityPerTask.set(task, computePriority(task, i)));\n\n  return tasks.slice().sort((task1, task2) => {\n    const [p1, idx1] = priorityPerTask.get(task1)!;\n    const [p2, idx2] = priorityPerTask.get(task2)!;\n\n    return (p2 - p1) || (idx1 - idx2);\n  });\n}\n"]} |
---|