-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathgolden.js
executable file
·116 lines (116 loc) · 15.9 KB
/
golden.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
/**
* @license
* Copyright Google LLC
*
* Use of this source code is governed by an MIT-style license that can be
* found in the LICENSE file at https://angular.io/license
*/
import { relative } from 'path';
import { convertPathToForwardSlash } from './file_system.js';
/**
* Converts a list of reference chains to a JSON-compatible golden object. Reference chains
* by default use TypeScript source file objects. In order to make those chains printable,
* the source file objects are mapped to their relative file names.
*/
export function convertReferenceChainToGolden(refs, baseDir) {
return (refs
.map(
// Normalize cycles as the paths can vary based on which node in the cycle is visited
// first in the analyzer. The paths represent cycles. Hence we can shift nodes in a
// deterministic way so that the goldens don't change unnecessarily and cycle comparison
// is simpler.
(chain) => normalizeCircularDependency(chain.map(({ fileName }) => convertPathToForwardSlash(relative(baseDir, fileName)))))
// Sort cycles so that the golden doesn't change unnecessarily when cycles are detected
// in different order (e.g. new imports cause cycles to be detected earlier or later).
.sort(compareCircularDependency));
}
/**
* Compares the specified goldens and returns two lists that describe newly
* added circular dependencies, or fixed circular dependencies.
*/
export function compareGoldens(actual, expected) {
const newCircularDeps = [];
const fixedCircularDeps = [];
actual.forEach((a) => {
if (!expected.find((e) => isSameCircularDependency(a, e))) {
newCircularDeps.push(a);
}
});
expected.forEach((e) => {
if (!actual.find((a) => isSameCircularDependency(e, a))) {
fixedCircularDeps.push(e);
}
});
return { newCircularDeps, fixedCircularDeps };
}
/**
* Normalizes the a circular dependency by ensuring that the path starts with the first
* node in alphabetical order. Since the path array represents a cycle, we can make a
* specific node the first element in the path that represents the cycle.
*
* This method is helpful because the path of circular dependencies changes based on which
* file in the path has been visited first by the analyzer. e.g. Assume we have a circular
* dependency represented as: `A -> B -> C`. The analyzer will detect this cycle when it
* visits `A`. Though when a source file that is analyzed before `A` starts importing `B`,
* the cycle path will detected as `B -> C -> A`. This represents the same cycle, but is just
* different due to a limitation of using a data structure that can be written to a text-based
* golden file.
*
* To account for this non-deterministic behavior in goldens, we shift the circular
* dependency path to the first node based on alphabetical order. e.g. `A` will always
* be the first node in the path that represents the cycle.
*/
function normalizeCircularDependency(path) {
if (path.length <= 1) {
return path;
}
let indexFirstNode = 0;
let valueFirstNode = path[0];
// Find a node in the cycle path that precedes all other elements
// in terms of alphabetical order.
for (let i = 1; i < path.length; i++) {
const value = path[i];
if (value.localeCompare(valueFirstNode, 'en') < 0) {
indexFirstNode = i;
valueFirstNode = value;
}
}
// If the alphabetically first node is already at start of the path, just
// return the actual path as no changes need to be made.
if (indexFirstNode === 0) {
return path;
}
// Move the determined first node (as of alphabetical order) to the start of a new
// path array. The nodes before the first node in the old path are then concatenated
// to the end of the new path. This is possible because the path represents a cycle.
return [...path.slice(indexFirstNode), ...path.slice(0, indexFirstNode)];
}
/** Checks whether the specified circular dependencies are equal. */
function isSameCircularDependency(actual, expected) {
if (actual.length !== expected.length) {
return false;
}
for (let i = 0; i < actual.length; i++) {
if (actual[i] !== expected[i]) {
return false;
}
}
return true;
}
/**
* Compares two circular dependencies by respecting the alphabetic order of nodes in the
* cycle paths. The first nodes which don't match in both paths are decisive on the order.
*/
function compareCircularDependency(a, b) {
// Go through nodes in both cycle paths and determine whether `a` should be ordered
// before `b`. The first nodes which don't match decide on the order.
for (let i = 0; i < Math.min(a.length, b.length); i++) {
const compareValue = a[i].localeCompare(b[i], 'en');
if (compareValue !== 0) {
return compareValue;
}
}
// If all nodes are equal in the cycles, the order is based on the length of both cycles.
return a.length - b.length;
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"golden.js","sourceRoot":"","sources":["../../../../../ng-dev/ts-circular-dependencies/golden.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC\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 {relative} from 'path';\n\nimport {ReferenceChain} from './analyzer.js';\nimport {convertPathToForwardSlash} from './file_system.js';\n\nexport type CircularDependency = ReferenceChain<string>;\nexport type Golden = CircularDependency[];\n\n/**\n * Converts a list of reference chains to a JSON-compatible golden object. Reference chains\n * by default use TypeScript source file objects. In order to make those chains printable,\n * the source file objects are mapped to their relative file names.\n */\nexport function convertReferenceChainToGolden(refs: ReferenceChain[], baseDir: string): Golden {\n  return (\n    refs\n      .map(\n        // Normalize cycles as the paths can vary based on which node in the cycle is visited\n        // first in the analyzer. The paths represent cycles. Hence we can shift nodes in a\n        // deterministic way so that the goldens don't change unnecessarily and cycle comparison\n        // is simpler.\n        (chain) =>\n          normalizeCircularDependency(\n            chain.map(({fileName}) => convertPathToForwardSlash(relative(baseDir, fileName))),\n          ),\n      )\n      // Sort cycles so that the golden doesn't change unnecessarily when cycles are detected\n      // in different order (e.g. new imports cause cycles to be detected earlier or later).\n      .sort(compareCircularDependency)\n  );\n}\n\n/**\n * Compares the specified goldens and returns two lists that describe newly\n * added circular dependencies, or fixed circular dependencies.\n */\nexport function compareGoldens(actual: Golden, expected: Golden) {\n  const newCircularDeps: CircularDependency[] = [];\n  const fixedCircularDeps: CircularDependency[] = [];\n  actual.forEach((a) => {\n    if (!expected.find((e) => isSameCircularDependency(a, e))) {\n      newCircularDeps.push(a);\n    }\n  });\n  expected.forEach((e) => {\n    if (!actual.find((a) => isSameCircularDependency(e, a))) {\n      fixedCircularDeps.push(e);\n    }\n  });\n  return {newCircularDeps, fixedCircularDeps};\n}\n\n/**\n * Normalizes the a circular dependency by ensuring that the path starts with the first\n * node in alphabetical order. Since the path array represents a cycle, we can make a\n * specific node the first element in the path that represents the cycle.\n *\n * This method is helpful because the path of circular dependencies changes based on which\n * file in the path has been visited first by the analyzer. e.g. Assume we have a circular\n * dependency represented as: `A -> B -> C`. The analyzer will detect this cycle when it\n * visits `A`. Though when a source file that is analyzed before `A` starts importing `B`,\n * the cycle path will detected as `B -> C -> A`. This represents the same cycle, but is just\n * different due to a limitation of using a data structure that can be written to a text-based\n * golden file.\n *\n * To account for this non-deterministic behavior in goldens, we shift the circular\n * dependency path to the first node based on alphabetical order. e.g. `A` will always\n * be the first node in the path that represents the cycle.\n */\nfunction normalizeCircularDependency(path: CircularDependency): CircularDependency {\n  if (path.length <= 1) {\n    return path;\n  }\n\n  let indexFirstNode: number = 0;\n  let valueFirstNode: string = path[0];\n\n  // Find a node in the cycle path that precedes all other elements\n  // in terms of alphabetical order.\n  for (let i = 1; i < path.length; i++) {\n    const value = path[i];\n    if (value.localeCompare(valueFirstNode, 'en') < 0) {\n      indexFirstNode = i;\n      valueFirstNode = value;\n    }\n  }\n\n  // If the alphabetically first node is already at start of the path, just\n  // return the actual path as no changes need to be made.\n  if (indexFirstNode === 0) {\n    return path;\n  }\n\n  // Move the determined first node (as of alphabetical order) to the start of a new\n  // path array. The nodes before the first node in the old path are then concatenated\n  // to the end of the new path. This is possible because the path represents a cycle.\n  return [...path.slice(indexFirstNode), ...path.slice(0, indexFirstNode)];\n}\n\n/** Checks whether the specified circular dependencies are equal. */\nfunction isSameCircularDependency(actual: CircularDependency, expected: CircularDependency) {\n  if (actual.length !== expected.length) {\n    return false;\n  }\n  for (let i = 0; i < actual.length; i++) {\n    if (actual[i] !== expected[i]) {\n      return false;\n    }\n  }\n  return true;\n}\n\n/**\n * Compares two circular dependencies by respecting the alphabetic order of nodes in the\n * cycle paths. The first nodes which don't match in both paths are decisive on the order.\n */\nfunction compareCircularDependency(a: CircularDependency, b: CircularDependency): number {\n  // Go through nodes in both cycle paths and determine whether `a` should be ordered\n  // before `b`. The first nodes which don't match decide on the order.\n  for (let i = 0; i < Math.min(a.length, b.length); i++) {\n    const compareValue = a[i].localeCompare(b[i], 'en');\n    if (compareValue !== 0) {\n      return compareValue;\n    }\n  }\n  // If all nodes are equal in the cycles, the order is based on the length of both cycles.\n  return a.length - b.length;\n}\n"]}