@monorepolint/rules 0.5.0-alpha.9 → 0.5.0-alpha.95

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (138) hide show
  1. package/lib/__tests__/alphabeticalScripts.spec.d.ts +8 -0
  2. package/lib/__tests__/alphabeticalScripts.spec.d.ts.map +1 -0
  3. package/lib/__tests__/alphabeticalScripts.spec.js +61 -0
  4. package/lib/__tests__/alphabeticalScripts.spec.js.map +1 -0
  5. package/lib/__tests__/bannedDependencies.spec.d.ts +2 -0
  6. package/lib/__tests__/bannedDependencies.spec.d.ts.map +1 -0
  7. package/lib/__tests__/bannedDependencies.spec.js +161 -0
  8. package/lib/__tests__/bannedDependencies.spec.js.map +1 -0
  9. package/lib/__tests__/consistentDependencies.spec.js +30 -23
  10. package/lib/__tests__/consistentDependencies.spec.js.map +1 -1
  11. package/lib/__tests__/consistentVersions.spec.d.ts +8 -0
  12. package/lib/__tests__/consistentVersions.spec.d.ts.map +1 -0
  13. package/lib/__tests__/consistentVersions.spec.js +183 -0
  14. package/lib/__tests__/consistentVersions.spec.js.map +1 -0
  15. package/lib/__tests__/fileContents.spec.d.ts +8 -0
  16. package/lib/__tests__/fileContents.spec.d.ts.map +1 -0
  17. package/lib/__tests__/fileContents.spec.js +59 -0
  18. package/lib/__tests__/fileContents.spec.js.map +1 -0
  19. package/lib/__tests__/mustSatisfyPeerDependencies.spec.d.ts +8 -0
  20. package/lib/__tests__/mustSatisfyPeerDependencies.spec.d.ts.map +1 -0
  21. package/lib/__tests__/mustSatisfyPeerDependencies.spec.js +1063 -0
  22. package/lib/__tests__/mustSatisfyPeerDependencies.spec.js.map +1 -0
  23. package/lib/__tests__/nestedWorkspaces.spec.d.ts +2 -0
  24. package/lib/__tests__/nestedWorkspaces.spec.d.ts.map +1 -0
  25. package/lib/__tests__/nestedWorkspaces.spec.js +124 -0
  26. package/lib/__tests__/nestedWorkspaces.spec.js.map +1 -0
  27. package/lib/__tests__/packageEntry.spec.js +72 -27
  28. package/lib/__tests__/packageEntry.spec.js.map +1 -1
  29. package/lib/__tests__/packageOrder.spec.js +34 -33
  30. package/lib/__tests__/packageOrder.spec.js.map +1 -1
  31. package/lib/__tests__/packageScript.spec.js +50 -51
  32. package/lib/__tests__/packageScript.spec.js.map +1 -1
  33. package/lib/__tests__/requireDependency.spec.d.ts +2 -0
  34. package/lib/__tests__/requireDependency.spec.d.ts.map +1 -0
  35. package/lib/__tests__/requireDependency.spec.js +123 -0
  36. package/lib/__tests__/requireDependency.spec.js.map +1 -0
  37. package/lib/__tests__/utils.d.ts +69 -1
  38. package/lib/__tests__/utils.d.ts.map +1 -1
  39. package/lib/__tests__/utils.js +72 -12
  40. package/lib/__tests__/utils.js.map +1 -1
  41. package/lib/alphabeticalDependencies.d.ts +3 -1
  42. package/lib/alphabeticalDependencies.d.ts.map +1 -1
  43. package/lib/alphabeticalDependencies.js +5 -39
  44. package/lib/alphabeticalDependencies.js.map +1 -1
  45. package/lib/alphabeticalScripts.d.ts +12 -0
  46. package/lib/alphabeticalScripts.d.ts.map +1 -0
  47. package/lib/alphabeticalScripts.js +20 -0
  48. package/lib/alphabeticalScripts.js.map +1 -0
  49. package/lib/bannedDependencies.d.ts +23 -5
  50. package/lib/bannedDependencies.d.ts.map +1 -1
  51. package/lib/bannedDependencies.js +110 -36
  52. package/lib/bannedDependencies.js.map +1 -1
  53. package/lib/consistentDependencies.d.ts +8 -1
  54. package/lib/consistentDependencies.d.ts.map +1 -1
  55. package/lib/consistentDependencies.js +30 -12
  56. package/lib/consistentDependencies.js.map +1 -1
  57. package/lib/consistentVersions.d.ts +14 -0
  58. package/lib/consistentVersions.d.ts.map +1 -0
  59. package/lib/consistentVersions.js +94 -0
  60. package/lib/consistentVersions.js.map +1 -0
  61. package/lib/fileContents.d.ts +8 -8
  62. package/lib/fileContents.d.ts.map +1 -1
  63. package/lib/fileContents.js +32 -27
  64. package/lib/fileContents.js.map +1 -1
  65. package/lib/index.d.ts +5 -0
  66. package/lib/index.d.ts.map +1 -1
  67. package/lib/index.js +19 -8
  68. package/lib/index.js.map +1 -1
  69. package/lib/mustSatisfyPeerDependencies.d.ts +240 -0
  70. package/lib/mustSatisfyPeerDependencies.d.ts.map +1 -0
  71. package/lib/mustSatisfyPeerDependencies.js +636 -0
  72. package/lib/mustSatisfyPeerDependencies.js.map +1 -0
  73. package/lib/nestedWorkspaces.d.ts +13 -0
  74. package/lib/nestedWorkspaces.d.ts.map +1 -0
  75. package/lib/nestedWorkspaces.js +50 -0
  76. package/lib/nestedWorkspaces.js.map +1 -0
  77. package/lib/packageEntry.d.ts +15 -6
  78. package/lib/packageEntry.d.ts.map +1 -1
  79. package/lib/packageEntry.js +53 -18
  80. package/lib/packageEntry.js.map +1 -1
  81. package/lib/packageOrder.d.ts +3 -3
  82. package/lib/packageOrder.d.ts.map +1 -1
  83. package/lib/packageOrder.js +8 -7
  84. package/lib/packageOrder.js.map +1 -1
  85. package/lib/packageScript.d.ts +10 -10
  86. package/lib/packageScript.js +6 -5
  87. package/lib/packageScript.js.map +1 -1
  88. package/lib/requireDependency.d.ts +15 -0
  89. package/lib/requireDependency.d.ts.map +1 -0
  90. package/lib/requireDependency.js +65 -0
  91. package/lib/requireDependency.js.map +1 -0
  92. package/lib/standardTsconfig.d.ts +20 -8
  93. package/lib/standardTsconfig.d.ts.map +1 -1
  94. package/lib/standardTsconfig.js +36 -18
  95. package/lib/standardTsconfig.js.map +1 -1
  96. package/lib/util/checkAlpha.d.ts +10 -0
  97. package/lib/util/checkAlpha.d.ts.map +1 -0
  98. package/lib/util/checkAlpha.js +51 -0
  99. package/lib/util/checkAlpha.js.map +1 -0
  100. package/lib/util/makeDirectory.d.ts +8 -0
  101. package/lib/util/makeDirectory.d.ts.map +1 -0
  102. package/lib/util/makeDirectory.js +28 -0
  103. package/lib/util/makeDirectory.js.map +1 -0
  104. package/lib/util/packageDependencyGraphService.d.ts +37 -0
  105. package/lib/util/packageDependencyGraphService.d.ts.map +1 -0
  106. package/lib/util/packageDependencyGraphService.js +70 -0
  107. package/lib/util/packageDependencyGraphService.js.map +1 -0
  108. package/package.json +18 -15
  109. package/src/__tests__/alphabeticalScripts.spec.ts +75 -0
  110. package/src/__tests__/bannedDependencies.spec.ts +191 -0
  111. package/src/__tests__/consistentDependencies.spec.ts +39 -26
  112. package/src/__tests__/consistentVersions.spec.ts +223 -0
  113. package/src/__tests__/fileContents.spec.ts +74 -0
  114. package/src/__tests__/mustSatisfyPeerDependencies.spec.ts +1188 -0
  115. package/src/__tests__/nestedWorkspaces.spec.ts +152 -0
  116. package/src/__tests__/packageEntry.spec.ts +98 -31
  117. package/src/__tests__/packageOrder.spec.ts +46 -40
  118. package/src/__tests__/packageScript.spec.ts +62 -54
  119. package/src/__tests__/requireDependency.spec.ts +151 -0
  120. package/src/__tests__/utils.ts +113 -11
  121. package/src/alphabeticalDependencies.ts +3 -47
  122. package/src/alphabeticalScripts.ts +19 -0
  123. package/src/bannedDependencies.ts +133 -43
  124. package/src/consistentDependencies.ts +34 -12
  125. package/src/consistentVersions.ts +140 -0
  126. package/src/fileContents.ts +31 -27
  127. package/src/index.ts +5 -0
  128. package/src/mustSatisfyPeerDependencies.ts +739 -0
  129. package/src/nestedWorkspaces.ts +59 -0
  130. package/src/packageEntry.ts +67 -24
  131. package/src/packageOrder.ts +6 -6
  132. package/src/packageScript.ts +3 -3
  133. package/src/requireDependency.ts +69 -0
  134. package/src/standardTsconfig.ts +40 -18
  135. package/src/util/checkAlpha.ts +59 -0
  136. package/src/util/makeDirectory.ts +24 -0
  137. package/src/util/packageDependencyGraphService.ts +114 -0
  138. package/tsconfig.tsbuildinfo +1 -2439
@@ -0,0 +1,739 @@
1
+ /*!
2
+ * Copyright 2020 Palantir Technologies, Inc.
3
+ *
4
+ * Licensed under the MIT license. See LICENSE file in the project root for details.
5
+ *
6
+ */
7
+
8
+ import { Context, RuleModule } from "@monorepolint/core";
9
+ import { Host, mutateJson, PackageJson } from "@monorepolint/utils";
10
+ import path from "path";
11
+ import resolvePackagePath from "resolve-package-path";
12
+ import * as r from "runtypes";
13
+ import { coerce } from "semver";
14
+
15
+ const Options = r.Union(
16
+ r.Partial({
17
+ skipUnparseableRanges: r.Undefined,
18
+ dependencyWhitelist: r.Undefined,
19
+ dependencyBlacklist: r.Undefined,
20
+ enforceForDevDependencies: r.Undefined,
21
+ }),
22
+ r
23
+ .Record({
24
+ skipUnparseableRanges: r.Boolean,
25
+ })
26
+ .And(
27
+ r.Partial({
28
+ dependencyWhitelist: r.Undefined,
29
+ dependencyBlacklist: r.Undefined,
30
+ enforceForDevDependencies: r.Undefined,
31
+ })
32
+ ),
33
+ r
34
+ .Record({
35
+ dependencyWhitelist: r.Array(r.String),
36
+ })
37
+ .And(
38
+ r.Partial({
39
+ skipUnparseableRanges: r.Undefined,
40
+ dependencyBlacklist: r.Undefined,
41
+ enforceForDevDependencies: r.Undefined,
42
+ })
43
+ ),
44
+ r
45
+ .Record({
46
+ dependencyBlacklist: r.Array(r.String),
47
+ })
48
+ .And(
49
+ r.Partial({
50
+ skipUnparseableRanges: r.Undefined,
51
+ dependencyWhitelist: r.Undefined,
52
+ enforceForDevDependencies: r.Undefined,
53
+ })
54
+ ),
55
+ r
56
+ .Record({
57
+ enforceForDevDependencies: r.Boolean,
58
+ })
59
+ .And(
60
+ r.Partial({
61
+ skipUnparseableRanges: r.Undefined,
62
+ dependencyWhitelist: r.Undefined,
63
+ dependencyBlacklist: r.Undefined,
64
+ })
65
+ ),
66
+ r
67
+ .Record({
68
+ skipUnparseableRanges: r.Boolean,
69
+ dependencyWhitelist: r.Array(r.String),
70
+ })
71
+ .And(
72
+ r.Partial({
73
+ dependencyBlacklist: r.Undefined,
74
+ enforceForDevDependencies: r.Undefined,
75
+ })
76
+ ),
77
+ r
78
+ .Record({
79
+ skipUnparseableRanges: r.Boolean,
80
+ dependencyBlacklist: r.Array(r.String),
81
+ })
82
+ .And(
83
+ r.Partial({
84
+ dependencyWhitelist: r.Undefined,
85
+ enforceForDevDependencies: r.Undefined,
86
+ })
87
+ ),
88
+ r
89
+ .Record({
90
+ skipUnparseableRanges: r.Boolean,
91
+ enforceForDevDependencies: r.Boolean,
92
+ })
93
+ .And(
94
+ r.Partial({
95
+ dependencyWhitelist: r.Undefined,
96
+ dependencyBlacklist: r.Undefined,
97
+ })
98
+ ),
99
+ r
100
+ .Record({
101
+ dependencyWhitelist: r.Array(r.String),
102
+ dependencyBlacklist: r.Array(r.String),
103
+ })
104
+ .And(
105
+ r.Partial({
106
+ skipUnparseableRanges: r.Undefined,
107
+ enforceForDevDependencies: r.Undefined,
108
+ })
109
+ ),
110
+ r
111
+ .Record({
112
+ dependencyWhitelist: r.Array(r.String),
113
+ enforceForDevDependencies: r.Boolean,
114
+ })
115
+ .And(
116
+ r.Partial({
117
+ skipUnparseableRanges: r.Undefined,
118
+ dependencyBlacklist: r.Undefined,
119
+ })
120
+ ),
121
+ r
122
+ .Record({
123
+ dependencyBlacklist: r.Array(r.String),
124
+ enforceForDevDependencies: r.Boolean,
125
+ })
126
+ .And(
127
+ r.Partial({
128
+ skipUnparseableRanges: r.Undefined,
129
+ dependencyWhitelist: r.Undefined,
130
+ })
131
+ ),
132
+ r
133
+ .Record({
134
+ skipUnparseableRanges: r.Boolean,
135
+ dependencyWhitelist: r.Array(r.String),
136
+ dependencyBlacklist: r.Array(r.String),
137
+ })
138
+ .And(
139
+ r.Partial({
140
+ enforceForDevDependencies: r.Undefined,
141
+ })
142
+ ),
143
+ r
144
+ .Record({
145
+ skipUnparseableRanges: r.Boolean,
146
+ dependencyWhitelist: r.Array(r.String),
147
+ enforceForDevDependencies: r.Boolean,
148
+ })
149
+ .And(
150
+ r.Partial({
151
+ dependencyBlacklist: r.Undefined,
152
+ })
153
+ ),
154
+ r
155
+ .Record({
156
+ skipUnparseableRanges: r.Boolean,
157
+ dependencyBlacklist: r.Array(r.String),
158
+ enforceForDevDependencies: r.Boolean,
159
+ })
160
+ .And(
161
+ r.Partial({
162
+ dependencyWhitelist: r.Undefined,
163
+ })
164
+ ),
165
+ r
166
+ .Record({
167
+ dependencyWhitelist: r.Array(r.String),
168
+ dependencyBlacklist: r.Array(r.String),
169
+ enforceForDevDependencies: r.Boolean,
170
+ })
171
+ .And(
172
+ r.Partial({
173
+ skipUnparseableRanges: r.Undefined,
174
+ })
175
+ ),
176
+ r.Record({
177
+ skipUnparseableRanges: r.Boolean,
178
+ dependencyWhitelist: r.Array(r.String),
179
+ dependencyBlacklist: r.Array(r.String),
180
+ enforceForDevDependencies: r.Boolean,
181
+ })
182
+ );
183
+
184
+ export type Options = r.Static<typeof Options>;
185
+
186
+ export const mustSatisfyPeerDependencies: RuleModule<typeof Options> = {
187
+ check: checkSatisfyPeerDependencies,
188
+ optionsRuntype: Options,
189
+ };
190
+
191
+ /**
192
+ * separating on `|`, this regex allows any of the following formats:
193
+ * - `*`
194
+ * - `x`
195
+ *
196
+ * More info: https://docs.npmjs.com/about-semantic-versioning
197
+ */
198
+ export const MATCH_ANY_VERSION_RANGE = /^(\*|x)$/;
199
+
200
+ /**
201
+ * This regex allows any of the following formats:
202
+ * - `>=15`
203
+ * - `>=15.2`
204
+ * - `>=15.2.1`
205
+ * - `>=15.2.1-rc.0`
206
+ * - `>=15.2.1+sha`
207
+ * - `>=15.2.1-rc.0+sha`
208
+ *
209
+ * See https://semver.org/#spec-item-9 for details about semver formatting, and
210
+ * https://regex101.com/r/vkijKf/1/ for a sample Regex.
211
+ *
212
+ * Note that the semver spec does _not_ specify npm range syntax. (`^`, `||`, `~`, `>`, etc.)
213
+ *
214
+ * More info: https://docs.npmjs.com/about-semantic-versioning
215
+ */
216
+ export const MATCH_GREATER_OR_EQUAL_VERSION_RANGE = /^>= ?\d+(?:\.\d+|\.\d+\.\d+(?:-((?:0|[1-9]\d*|\d*[a-zA-Z-][0-9a-zA-Z-]*)(?:\.(?:0|[1-9]\d*|\d*[a-zA-Z-][0-9a-zA-Z-]*))*))?(?:\+([0-9a-zA-Z-]+(?:\.[0-9a-zA-Z-]+)*))?)?$/;
217
+
218
+ /**
219
+ * This regex allows any of the following formats:
220
+ * - `15`
221
+ * - `^15`
222
+ * - `15.x`
223
+ * - `^15.x`
224
+ * - `15.x.x`
225
+ * - `^15.x.x`
226
+ * - `^15.2`
227
+ * - `^15.2.x`
228
+ * - `^15.2.1`
229
+ * - `^15.2.1-rc.0`
230
+ * - `^15.2.1+sha`
231
+ * - `^15.2.1-rc.0+sha`
232
+ *
233
+ * See https://semver.org/#spec-item-9 for details about semver formatting, and
234
+ * https://regex101.com/r/vkijKf/1/ for a sample Regex.
235
+ *
236
+ * Note that the semver spec does _not_ specify npm range syntax. (`^`, `||`, `~`, `>`, etc.)
237
+ *
238
+ * More info: https://docs.npmjs.com/about-semantic-versioning
239
+ */
240
+ export const MATCH_MAJOR_VERSION_RANGE = /^(?:\^?\d+|\^?\d+\.x|\^?\d+\.x\.x|\^\d+\.\d+|\^\d+\.\d+\.x|\^\d+\.\d+\.\d+(?:-((?:0|[1-9]\d*|\d*[a-zA-Z-][0-9a-zA-Z-]*)(?:\.(?:0|[1-9]\d*|\d*[a-zA-Z-][0-9a-zA-Z-]*))*))?(?:\+([0-9a-zA-Z-]+(?:\.[0-9a-zA-Z-]+)*))?)$/;
241
+
242
+ /**
243
+ * Does not currently accept `<`, `<=`, `>`, `=` or `-` for ranges (e.g. `> 2.5.1 < 3` or `1.0.0 - 1.2.0`),
244
+ * though it will accept isolated `>=` ranges (e.g. `>=2.5.1`, but not `^1 || >=2.5.1`)
245
+ *
246
+ * See https://semver.org/#spec-item-9 for details about semver formatting, and
247
+ * https://regex101.com/r/vkijKf/1/ for a sample Regex.
248
+ *
249
+ * Note that the semver spec does _not_ specify npm range syntax. (`^`, `||`, `~`, `>`, etc.)
250
+ *
251
+ * More info: https://docs.npmjs.com/about-semantic-versioning
252
+ *
253
+ * TODO: accept minor pins `~4.2.1`
254
+ */
255
+ export const RANGE_REGEX = /^(\*|x|>= ?\d+(?:\.\d+|\.\d+\.\d+(?:-((?:0|[1-9]\d*|\d*[a-zA-Z-][0-9a-zA-Z-]*)(?:\.(?:0|[1-9]\d*|\d*[a-zA-Z-][0-9a-zA-Z-]*))*))?(?:\+([0-9a-zA-Z-]+(?:\.[0-9a-zA-Z-]+)*))?)?|\^?\d+(\.x|\.x\.x|\.\d+|\.\d+\.x|\.\d+\.\d+(?:-((?:0|[1-9]\d*|\d*[a-zA-Z-][0-9a-zA-Z-]*)(?:\.(?:0|[1-9]\d*|\d*[a-zA-Z-][0-9a-zA-Z-]*))*))?(?:\+([0-9a-zA-Z-]+(?:\.[0-9a-zA-Z-]+)*))?)?( \|\| \^?\d+(\.x|\.x\.x|\.\d+|\.\d+\.x|\.\d+\.\d+(?:-((?:0|[1-9]\d*|\d*[a-zA-Z-][0-9a-zA-Z-]*)(?:\.(?:0|[1-9]\d*|\d*[a-zA-Z-][0-9a-zA-Z-]*))*))?(?:\+([0-9a-zA-Z-]+(?:\.[0-9a-zA-Z-]+)*))?)?)*)$/;
256
+
257
+ interface IPeerDependencyRequirement {
258
+ fromPackageName: string;
259
+ range: ValidRange;
260
+ }
261
+
262
+ interface IResolvedPeerDependencyRequirement {
263
+ fromPeerDependencyRequirements: IPeerDependencyRequirement[];
264
+ range: ValidRange;
265
+ }
266
+
267
+ function checkSatisfyPeerDependencies(context: Context, opts: Options) {
268
+ const { dependencyBlacklist, dependencyWhitelist, enforceForDevDependencies, skipUnparseableRanges } = opts;
269
+ const packageJsonPath = path.resolve(context.getPackageJsonPath());
270
+ const packageJson: PackageJson = require(packageJsonPath);
271
+ const packageDependencies = packageJson.dependencies || {};
272
+ const packageDevDependencies = packageJson.devDependencies || {};
273
+ const packagePeerDependencies = packageJson.peerDependencies || {};
274
+ const packageName = packageJson.name || packageJsonPath;
275
+
276
+ // check that no peer dependencies are also declared as regular dependencies
277
+ for (const [peerDependencyName, peerDependencyRange] of Object.entries(packagePeerDependencies)) {
278
+ if (shouldSkipPackage({ dependencyBlacklist, dependencyWhitelist, packageName: peerDependencyName })) {
279
+ continue;
280
+ }
281
+
282
+ const dependencyRange = packageDependencies[peerDependencyName];
283
+ if (dependencyRange != null) {
284
+ context.addError({
285
+ file: packageJsonPath,
286
+ message:
287
+ `[0] Package ${packageName} has overloaded ${peerDependencyName} dependencies.\n\t` +
288
+ `Peer dependency '${peerDependencyRange}' and regular dependency '${dependencyRange}'.`,
289
+ });
290
+ }
291
+ }
292
+
293
+ // map of all inherited peer dependency requirements
294
+ const allRequiredPeerDependencies: { [peerDependencyName: string]: IPeerDependencyRequirement[] } = {};
295
+
296
+ // for each of this package's dependencies, add the dependency's peer requirements into `allRequiredPeerDependencies`
297
+ const allDependencies = enforceForDevDependencies
298
+ ? [...Object.keys(packageDependencies), ...Object.keys(packageDevDependencies)]
299
+ : Object.keys(packageDependencies);
300
+ for (const dependency of allDependencies) {
301
+ const dependencyPackageJsonPath = resolvePackagePath(dependency, path.dirname(packageJsonPath));
302
+ if (dependencyPackageJsonPath == null) {
303
+ throw new Error(`Could not resolve ${dependency} from ${path.dirname(packageJsonPath)}`);
304
+ }
305
+ const dependencyPackageJson: PackageJson = require(dependencyPackageJsonPath);
306
+ const requiredPeerDependencies = dependencyPackageJson.peerDependencies;
307
+ if (requiredPeerDependencies == null) {
308
+ continue;
309
+ }
310
+ for (const [peerDependencyName, range] of Object.entries(requiredPeerDependencies)) {
311
+ if (shouldSkipPackage({ dependencyBlacklist, dependencyWhitelist, packageName: peerDependencyName })) {
312
+ continue;
313
+ }
314
+
315
+ if (!isValidRange(range)) {
316
+ const message = `Unable to parse ${dependencyPackageJson.name}'s ${peerDependencyName} peer dependency range '${range}'.`;
317
+ if (skipUnparseableRanges) {
318
+ context.addWarning({ file: dependencyPackageJsonPath, message });
319
+ continue;
320
+ }
321
+ throw new Error(message);
322
+ }
323
+ if (allRequiredPeerDependencies[peerDependencyName] == null) {
324
+ allRequiredPeerDependencies[peerDependencyName] = [];
325
+ }
326
+ allRequiredPeerDependencies[peerDependencyName].push({ fromPackageName: dependencyPackageJson.name!, range });
327
+ }
328
+ }
329
+
330
+ for (const [peerDependencyName, peerDependencyRequirements] of Object.entries(allRequiredPeerDependencies)) {
331
+ // for each inherited peer dependency, determine the strictest range
332
+ let mostStrictPeerRequirement: IResolvedPeerDependencyRequirement = {
333
+ fromPeerDependencyRequirements: [peerDependencyRequirements[0]],
334
+ range: peerDependencyRequirements[0].range,
335
+ };
336
+ for (const peerRequirement of peerDependencyRequirements) {
337
+ if (doesASatisfyB(mostStrictPeerRequirement.range, peerRequirement.range)) {
338
+ continue;
339
+ } else if (doesASatisfyB(peerRequirement.range, mostStrictPeerRequirement.range)) {
340
+ mostStrictPeerRequirement = {
341
+ fromPeerDependencyRequirements: [peerRequirement],
342
+ range: peerRequirement.range,
343
+ };
344
+ } else {
345
+ const maybeIntersection = findIntersection(peerRequirement.range, mostStrictPeerRequirement.range);
346
+ if (maybeIntersection !== undefined) {
347
+ mostStrictPeerRequirement = {
348
+ fromPeerDependencyRequirements: [
349
+ ...mostStrictPeerRequirement.fromPeerDependencyRequirements,
350
+ peerRequirement,
351
+ ],
352
+ range: maybeIntersection,
353
+ };
354
+ } else {
355
+ context.addError({
356
+ file: packageJsonPath,
357
+ message:
358
+ `[1] Package ${packageName} has conflicting inherited ${peerDependencyName} peer dependencies.\n\t` +
359
+ `Dependency ${peerRequirement.fromPackageName} requires '${peerRequirement.range}' but\n\t` +
360
+ getMostStrictStatement(mostStrictPeerRequirement),
361
+ });
362
+ }
363
+ }
364
+ }
365
+
366
+ // if this package has a dependency on an inherited peer dependency,
367
+ // the range must be equal to or stricter than `mostStrictPeerRequirement`
368
+ const packageDependencyRange = packageDependencies[peerDependencyName];
369
+ if (packageDependencyRange != null) {
370
+ if (!isValidRange(packageDependencyRange)) {
371
+ const message = `Unable to parse ${packageName}'s ${peerDependencyName} dependency range '${packageDependencyRange}'.`;
372
+ if (skipUnparseableRanges) {
373
+ context.addWarning({ file: packageJsonPath, message });
374
+ } else {
375
+ throw new Error(message);
376
+ }
377
+ } else if (!doesASatisfyB(packageDependencyRange, mostStrictPeerRequirement.range)) {
378
+ context.addError({
379
+ file: packageJsonPath,
380
+ message:
381
+ `[2] Package ${packageName} dependency on ${peerDependencyName} '${packageDependencyRange}' does not satisfy inherited peer dependencies.\n\t` +
382
+ getMostStrictStatement(mostStrictPeerRequirement),
383
+ });
384
+ }
385
+ }
386
+
387
+ // for every inherited peer dependency, this package must declare a dependency or peer dependency
388
+ // equal to or stricter than `mostStrictPeerRequirement`
389
+ const packagePeerDependencyRange = packagePeerDependencies[peerDependencyName];
390
+ if (packageDependencyRange == null && packagePeerDependencyRange == null) {
391
+ context.addError({
392
+ file: packageJsonPath,
393
+ message:
394
+ `[3] Package ${packageName} is missing required ${peerDependencyName} dependency.\n\t` +
395
+ getMostStrictStatement(mostStrictPeerRequirement),
396
+ fixer: getAddDependencyTypeFixer({
397
+ packageJsonPath,
398
+ dependencyType: "peerDependencies",
399
+ dependencyName: peerDependencyName,
400
+ version: mostStrictPeerRequirement.range,
401
+ host: context.host,
402
+ }),
403
+ });
404
+ }
405
+
406
+ // if this package has a peer dependency on an inherited peer dependency,
407
+ // the range must be equal to or stricter than `mostStrictPeerRequirement`
408
+ if (packagePeerDependencyRange != null) {
409
+ if (!isValidRange(packagePeerDependencyRange)) {
410
+ const message = `Unable to parse ${packageName}'s ${peerDependencyName} peer dependency range '${packagePeerDependencyRange}'.`;
411
+ if (skipUnparseableRanges) {
412
+ context.addWarning({ file: packageJsonPath, message });
413
+ } else {
414
+ throw new Error(message);
415
+ }
416
+ } else if (!doesASatisfyB(packagePeerDependencyRange, mostStrictPeerRequirement.range)) {
417
+ context.addError({
418
+ file: packageJsonPath,
419
+ message:
420
+ `[4] Package ${packageName} peer dependency on ${peerDependencyName} '${packagePeerDependencyRange}' is not strict enough.\n\t` +
421
+ getMostStrictStatement(mostStrictPeerRequirement),
422
+ fixer: getAddDependencyTypeFixer({
423
+ packageJsonPath,
424
+ dependencyType: "peerDependencies",
425
+ dependencyName: peerDependencyName,
426
+ version: mostStrictPeerRequirement.range,
427
+ host: context.host,
428
+ }),
429
+ });
430
+ }
431
+ }
432
+ }
433
+ }
434
+
435
+ function shouldSkipPackage({
436
+ dependencyBlacklist,
437
+ dependencyWhitelist,
438
+ packageName,
439
+ }: {
440
+ dependencyBlacklist?: string[];
441
+ dependencyWhitelist?: string[];
442
+ packageName: string;
443
+ }) {
444
+ // blacklist should take precedance
445
+ if (
446
+ (dependencyBlacklist != null && dependencyBlacklist.includes(packageName)) ||
447
+ (dependencyWhitelist != null && !dependencyWhitelist.includes(packageName))
448
+ ) {
449
+ return true;
450
+ }
451
+ return false;
452
+ }
453
+
454
+ function getMostStrictStatement(mostStrictPeerRequirement: IResolvedPeerDependencyRequirement) {
455
+ if (mostStrictPeerRequirement.fromPeerDependencyRequirements.length === 1) {
456
+ const dependencyName = mostStrictPeerRequirement.fromPeerDependencyRequirements[0].fromPackageName;
457
+ return `Dependency ${dependencyName} requires '${mostStrictPeerRequirement.range}'.`;
458
+ } else {
459
+ const dependencyNames = mostStrictPeerRequirement.fromPeerDependencyRequirements
460
+ .map((peerDependencyRequirement) => peerDependencyRequirement.fromPackageName)
461
+ .join(", ");
462
+ const dependencyRequirements = mostStrictPeerRequirement.fromPeerDependencyRequirements
463
+ .map((peerDependencyRequirement) => `'${peerDependencyRequirement.range}'`)
464
+ .join(", ");
465
+ return (
466
+ `Dependencies [${dependencyNames}] require [${dependencyRequirements}] respectively, ` +
467
+ `resolving to '${mostStrictPeerRequirement.range}'.`
468
+ );
469
+ }
470
+ }
471
+
472
+ /**
473
+ * Given two version ranges, find the maximum intersecting range
474
+ * of `a` and `b`. `findIntersection(a,b)` should return the same
475
+ * result as `findIntersection(b,a)`.
476
+ *
477
+ * NOTE: This code assumes that input version ranges match `RANGE_REGEX`.
478
+ * Additionally, major version ranges must not be repeated in union ranges.
479
+ * e.g. `^15.0.5 || ^16.0.0` is permitted, but `15.0.5 || 15.0.999` is not.
480
+ *
481
+ * EXAMPLES:
482
+ * findIntersection("15.1.0", "*") => "15.1.0"
483
+ * findIntersection("^15", "*") => "^15"
484
+ * findIntersection(">=15", "*") => ">=15"
485
+ * findIntersection("*", "*") => "*"
486
+ * findIntersection("15.1.0", ">=1") => "15.1.0"
487
+ * findIntersection("^15", ">=1") => "^15"
488
+ * findIntersection(">=15", ">=1") => ">=15"
489
+ * findIntersection("15.1.0", "^15") => "15.1.0"
490
+ * findIntersection("^15.2", "^15") => "^15.2"
491
+ * findIntersection("14", "^15") => undefined
492
+ * findIntersection("15.1.0", "^15 || ^16") => "15.1.0"
493
+ * findIntersection("^15.2", "^15 || ^16") => "^15.2"
494
+ * findIntersection("14", "^15 || ^16") => undefined
495
+ * findIntersection("^15.2 || ^16", "^15 || ^16.4") => "^15.2 || ^16.4"
496
+ *
497
+ * @param a version range that matches `RANGE_REGEX`
498
+ * @param b version range that matches `RANGE_REGEX`
499
+ * @returns the maximum intersecting `ValidRange`, or `undefined` if there is no intersection
500
+ */
501
+ export function findIntersection(a: ValidRange, b: ValidRange): ValidRange | undefined {
502
+ if (doesASatisfyB(a, b)) {
503
+ return a;
504
+ } else if (doesASatisfyB(b, a)) {
505
+ return b;
506
+ }
507
+
508
+ // It's safe to assume that `isAnyVersionRange(a)` and `isAnyVersionRange(b)` are false,
509
+ // since a `MATCH_ANY_VERSION_RANGE` would have been satisfied by anything.
510
+ if (isAnyVersionRange(a) || isAnyVersionRange(b)) {
511
+ throw new Error();
512
+ }
513
+
514
+ const aVersions = a.includes("||") ? a.split("||").map((s) => s.trim()) : [a];
515
+ const bVersions = b.includes("||") ? b.split("||").map((s) => s.trim()) : [b];
516
+
517
+ const aIsGreaterOrEqualVersionRange = isGreaterOrEqualVersionRange(a);
518
+ const bIsGreaterOrEqualVersionRange = isGreaterOrEqualVersionRange(b);
519
+ if (aIsGreaterOrEqualVersionRange && bIsGreaterOrEqualVersionRange) {
520
+ // If the ranges were equal, they'd both satisfy each other.
521
+ // Otherwise, the higher range should have satisfied the lower range.
522
+ throw new Error();
523
+ }
524
+
525
+ if (aIsGreaterOrEqualVersionRange) {
526
+ const aSemVer = coerce(a)!;
527
+ // keep every version where `bSemVer` is >= `aSemVer`
528
+ // where bVersion has an operator, we need to adjust the version to take into
529
+ // account aVersion, e.g. if a is >=15.2.3 and b is ^15, then the intersection is actually ^15.2.3
530
+ const compatibleBVersions = bVersions
531
+ .map((bVersion) => {
532
+ const bSemVer = coerce(bVersion)!;
533
+ if (bVersion.startsWith("^") && bSemVer.major >= aSemVer.major) {
534
+ return `^${bSemVer.compare(aSemVer) >= 0 ? bSemVer.raw : aSemVer.raw}`;
535
+ }
536
+ return bSemVer.compare(aSemVer) !== -1 ? bVersion : undefined;
537
+ })
538
+ .filter((bVersion): bVersion is string => bVersion != null);
539
+ if (compatibleBVersions.length === 0) {
540
+ return undefined;
541
+ }
542
+ return compatibleBVersions.join(" || ") as ValidRange;
543
+ }
544
+ if (bIsGreaterOrEqualVersionRange) {
545
+ const bSemVer = coerce(b)!;
546
+ // keep every version where `aSemVer` is >= `bSemVer`
547
+ // where aVersion has an operator, we need to adjust the version to take into
548
+ // account bVersion, e.g. if b is >=15.2.3 and a is ^15, then the intersection is actually ^15.2.3
549
+ const compatibleAVersions = aVersions
550
+ .map((aVersion) => {
551
+ const aSemVer = coerce(aVersion)!;
552
+ if (aVersion.startsWith("^") && aSemVer.major >= bSemVer.major) {
553
+ return `^${aSemVer.compare(bSemVer) >= 0 ? aSemVer.raw : bSemVer.raw}`;
554
+ }
555
+ return aSemVer.compare(bSemVer) !== -1 ? aVersion : undefined;
556
+ })
557
+ .filter((aVersion): aVersion is string => aVersion != null);
558
+ if (compatibleAVersions.length === 0) {
559
+ return undefined;
560
+ }
561
+ return compatibleAVersions.join(" || ") as ValidRange;
562
+ }
563
+
564
+ const compatibleVersions = aVersions
565
+ .map((aVersion) => {
566
+ const aSemVer = coerce(aVersion)!;
567
+ const majorMatchingBVersion = bVersions.find((m) => coerce(m)!.major === aSemVer.major);
568
+ if (majorMatchingBVersion === undefined) {
569
+ // there is no intersecting `b` major version for this `a` major version
570
+ return undefined;
571
+ }
572
+ if (doesASatisfyB(aVersion as ValidRange, majorMatchingBVersion as ValidRange)) {
573
+ return aVersion;
574
+ } else if (doesASatisfyB(majorMatchingBVersion as ValidRange, aVersion as ValidRange)) {
575
+ return majorMatchingBVersion;
576
+ } else {
577
+ return undefined;
578
+ }
579
+ })
580
+ .filter((aVersion) => aVersion !== undefined);
581
+ if (compatibleVersions.length === 0) {
582
+ return undefined;
583
+ }
584
+ return compatibleVersions.join(" || ") as ValidRange;
585
+ }
586
+
587
+ /**
588
+ * Given two version ranges, determine whether `a` satisfies `b`.
589
+ * `a` satisfies `b` iff `a` is a "more strict than or equal to" subset of `b`.
590
+ * For example, both `^15` and `^15.2.0` satisfy `^15`, but `^15 || ^16` does not.
591
+ *
592
+ * NOTE: This code assumes that input version ranges match `RANGE_REGEX`.
593
+ * Additionally, major version ranges must not be repeated in union ranges.
594
+ * e.g. `^15.0.5 || ^16.0.0` is permitted, but `15.0.5 || 15.0.999` is not.
595
+ *
596
+ * To determine that `a` is "more strict than or equal to" `b`, we first
597
+ * split the set of all versions or ranges that are potentially unioned in `a` and `b`.
598
+ * For example, if `a` is `15.0.5`, we produce the set `[ "15.0.5" ]`,
599
+ * and if `b` is `^15 || ^16`, we produce the set `[ "^15", "^16" ]`.
600
+ * `a` is "more strict than or equal to" `b` iff each entry in `a`'s set
601
+ * satisfies (equal to or greater than) some entry in `b`.
602
+ *
603
+ * The following version ranges satisfy `^15.0.5 || ^16.0.0`:
604
+ * - `^15.0.5 || ^16.0.0`
605
+ * - `^15.0.5 || ^16.x.x`
606
+ * - `15.0.5 || 16.0.0`
607
+ * - `^15.0.999 || ^16.0.0`
608
+ * - `^15.0.5 || ^16.0.999`
609
+ * - `^15.0.999`
610
+ * - `^16.0.0`
611
+ * - `16.0.0`
612
+ * The following version ranges do not satisfy `^15.0.5 || ^16.0.0`:
613
+ * - `^15.0.0 || ^16.0.0`
614
+ * - `^15.0.5 || ^17.0.0`
615
+ * - `^14.0.0 || ^15.0.5 || ^16.0.0`
616
+ * - `^17.0.0`
617
+ * - `17.0.0`
618
+ *
619
+ * @param a version range that matches `RANGE_REGEX`
620
+ * @param b version range that matches `RANGE_REGEX`
621
+ * @returns `true` if `a` is more strict than or equal to `b`, `false` otherwise
622
+ */
623
+ export function doesASatisfyB(a: ValidRange, b: ValidRange): boolean {
624
+ if (a === b) {
625
+ return true;
626
+ }
627
+
628
+ const aIsAnyVersionRange = isAnyVersionRange(a);
629
+ const bIsAnyVersionRange = isAnyVersionRange(b);
630
+ if (bIsAnyVersionRange) {
631
+ return true;
632
+ } else if (aIsAnyVersionRange) {
633
+ // `bIsAnyVersionRange` is `false`
634
+ // `a` permits more values than `b`, therefore `a` is "less strict"
635
+ return false;
636
+ }
637
+
638
+ const aVersions = a.includes("||") ? a.split("||").map((s) => s.trim()) : [a];
639
+ const bVersions = b.includes("||") ? b.split("||").map((s) => s.trim()) : [b];
640
+
641
+ const aIsGreaterOrEqualVersionRange = isGreaterOrEqualVersionRange(a);
642
+ const bIsGreaterOrEqualVersionRange = isGreaterOrEqualVersionRange(b);
643
+ if (aIsGreaterOrEqualVersionRange && bIsGreaterOrEqualVersionRange) {
644
+ const aSemVer = coerce(a)!;
645
+ const bSemVer = coerce(b)!;
646
+ // `a` satisfies `b` so long as `aSemVer` is greater than or equal to `bSemVer`
647
+ return aSemVer.compare(bSemVer) !== -1;
648
+ } else if (bIsGreaterOrEqualVersionRange) {
649
+ const bSemVer = coerce(b)!;
650
+ return aVersions.every((aVersion) => {
651
+ const aSemVer = coerce(aVersion)!;
652
+ // `a` satisfies `b` so long as `aSemVer` is greater than or equal to `bSemVer`
653
+ return aSemVer.compare(bSemVer) !== -1;
654
+ });
655
+ } else if (aIsGreaterOrEqualVersionRange) {
656
+ // `bIsGreaterOrEqualVersionRange` is `false` (and `bIsAnyVersionRange` is `false`)
657
+ // `a` permits more values than `b`, therefore `a` is "less strict"
658
+ return false;
659
+ }
660
+
661
+ return aVersions.every((aVersion) => {
662
+ const aSemVer = coerce(aVersion)!;
663
+ const majorMatchingBVersion = bVersions.find((m) => coerce(m)!.major === aSemVer.major);
664
+ if (majorMatchingBVersion === undefined) {
665
+ // `a` permits a major version that is not permitted by `b`, therefore `a` is "less strict"
666
+ return false;
667
+ }
668
+
669
+ const aVersionIsRange = isMajorVersionRange(aVersion);
670
+ const majorMatchingBSemVer = coerce(majorMatchingBVersion)!;
671
+ const majorMatchingBVersionIsRange = isMajorVersionRange(majorMatchingBVersion);
672
+
673
+ if (majorMatchingBVersionIsRange) {
674
+ // `a` satisfies `b` so long as `aSemVer` is greater than or equal to `majorMatchingBSemVer`
675
+ // this is true whether or not `aVersionIsRange`
676
+ return aSemVer.compare(majorMatchingBSemVer) !== -1;
677
+ } else {
678
+ // `majorMatchingBVersionIsRange` is `false`
679
+ if (aVersionIsRange) {
680
+ // `a` permits more values than `b`, therefore `a` is "less strict"
681
+ // e.g if `b` is `15.5.5`, this is true whether `a` is `^15.0.0`, `^15.5.5`, or `^15.9.9`
682
+ return false;
683
+ } else {
684
+ // `aVersionIsRange` is `false`
685
+ // `a` satisfies `b` if and only if `aSemVer` is equal to `majorMatchingBSemVer`
686
+ return aSemVer.compare(majorMatchingBSemVer) === 0;
687
+ }
688
+ }
689
+ });
690
+ }
691
+
692
+ /**
693
+ * Returns true if the version evaluates to 'any', e.g. * or x
694
+ */
695
+ function isAnyVersionRange(version: string): boolean {
696
+ return MATCH_ANY_VERSION_RANGE.test(version);
697
+ }
698
+
699
+ /**
700
+ * Retruns true if the version begins with '>='
701
+ */
702
+ function isGreaterOrEqualVersionRange(version: string): boolean {
703
+ return MATCH_GREATER_OR_EQUAL_VERSION_RANGE.test(version);
704
+ }
705
+
706
+ function isMajorVersionRange(version: string): boolean {
707
+ return MATCH_MAJOR_VERSION_RANGE.test(version);
708
+ }
709
+
710
+ export type ValidRange = string & { _type: "valid range" };
711
+ export function isValidRange(version: string): version is ValidRange {
712
+ return RANGE_REGEX.test(version);
713
+ }
714
+
715
+ type IDependencyType = "dependencies" | "devDependencies" | "peerDependencies";
716
+
717
+ function getAddDependencyTypeFixer({
718
+ packageJsonPath,
719
+ dependencyType,
720
+ dependencyName,
721
+ version,
722
+ host,
723
+ }: {
724
+ packageJsonPath: string;
725
+ dependencyType: IDependencyType;
726
+ dependencyName: string;
727
+ version: string;
728
+ host: Host;
729
+ }) {
730
+ return () => {
731
+ mutateJson<PackageJson>(packageJsonPath, host, (packageJson) => {
732
+ if (packageJson[dependencyType] == null) {
733
+ packageJson[dependencyType] = {};
734
+ }
735
+ packageJson[dependencyType]![dependencyName] = version;
736
+ return packageJson;
737
+ });
738
+ };
739
+ }