@monorepolint/rules 0.5.0-alpha.10 → 0.5.0-alpha.103

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 (155) hide show
  1. package/.turbo/turbo-clean.log +4 -0
  2. package/.turbo/turbo-compile-typescript.log +4 -0
  3. package/.turbo/turbo-lint.log +107 -0
  4. package/.turbo/turbo-test.log +631 -0
  5. package/.turbo/turbo-transpile-typescript.log +18 -0
  6. package/build/js/index.cjs +1491 -0
  7. package/build/js/index.cjs.map +1 -0
  8. package/build/js/index.js +1433 -0
  9. package/build/js/index.js.map +1 -0
  10. package/build/tsconfig.tsbuildinfo +1 -0
  11. package/{lib/__tests__/utils.d.ts → build/types/__tests__/alphabeticalScripts.spec.d.ts} +2 -2
  12. package/build/types/__tests__/alphabeticalScripts.spec.d.ts.map +1 -0
  13. package/build/types/__tests__/bannedDependencies.spec.d.ts +2 -0
  14. package/build/types/__tests__/bannedDependencies.spec.d.ts.map +1 -0
  15. package/{lib → build/types}/__tests__/consistentDependencies.spec.d.ts +0 -0
  16. package/build/types/__tests__/consistentDependencies.spec.d.ts.map +1 -0
  17. package/build/types/__tests__/consistentVersions.spec.d.ts +8 -0
  18. package/build/types/__tests__/consistentVersions.spec.d.ts.map +1 -0
  19. package/build/types/__tests__/fileContents.spec.d.ts +8 -0
  20. package/build/types/__tests__/fileContents.spec.d.ts.map +1 -0
  21. package/build/types/__tests__/mustSatisfyPeerDependencies.spec.d.ts +8 -0
  22. package/build/types/__tests__/mustSatisfyPeerDependencies.spec.d.ts.map +1 -0
  23. package/build/types/__tests__/nestedWorkspaces.spec.d.ts +2 -0
  24. package/build/types/__tests__/nestedWorkspaces.spec.d.ts.map +1 -0
  25. package/{lib → build/types}/__tests__/packageEntry.spec.d.ts +0 -0
  26. package/build/types/__tests__/packageEntry.spec.d.ts.map +1 -0
  27. package/{lib → build/types}/__tests__/packageOrder.spec.d.ts +0 -0
  28. package/build/types/__tests__/packageOrder.spec.d.ts.map +1 -0
  29. package/{lib → build/types}/__tests__/packageScript.spec.d.ts +0 -0
  30. package/build/types/__tests__/packageScript.spec.d.ts.map +1 -0
  31. package/build/types/__tests__/requireDependency.spec.d.ts +2 -0
  32. package/build/types/__tests__/requireDependency.spec.d.ts.map +1 -0
  33. package/build/types/__tests__/utils.d.ts +81 -0
  34. package/build/types/__tests__/utils.d.ts.map +1 -0
  35. package/build/types/alphabeticalDependencies.d.ts +23 -0
  36. package/build/types/alphabeticalDependencies.d.ts.map +1 -0
  37. package/build/types/alphabeticalScripts.d.ts +23 -0
  38. package/build/types/alphabeticalScripts.d.ts.map +1 -0
  39. package/build/types/bannedDependencies.d.ts +134 -0
  40. package/build/types/bannedDependencies.d.ts.map +1 -0
  41. package/build/types/consistentDependencies.d.ts +38 -0
  42. package/build/types/consistentDependencies.d.ts.map +1 -0
  43. package/build/types/consistentVersions.d.ts +47 -0
  44. package/build/types/consistentVersions.d.ts.map +1 -0
  45. package/build/types/fileContents.d.ts +111 -0
  46. package/build/types/fileContents.d.ts.map +1 -0
  47. package/build/types/index.d.ts +20 -0
  48. package/build/types/index.d.ts.map +1 -0
  49. package/build/types/mustSatisfyPeerDependencies.d.ts +721 -0
  50. package/build/types/mustSatisfyPeerDependencies.d.ts.map +1 -0
  51. package/build/types/nestedWorkspaces.d.ts +24 -0
  52. package/build/types/nestedWorkspaces.d.ts.map +1 -0
  53. package/build/types/packageEntry.d.ts +115 -0
  54. package/build/types/packageEntry.d.ts.map +1 -0
  55. package/build/types/packageOrder.d.ts +33 -0
  56. package/build/types/packageOrder.d.ts.map +1 -0
  57. package/build/types/packageScript.d.ts +89 -0
  58. package/build/types/packageScript.d.ts.map +1 -0
  59. package/build/types/requireDependency.d.ts +99 -0
  60. package/build/types/requireDependency.d.ts.map +1 -0
  61. package/build/types/standardTsconfig.d.ts +113 -0
  62. package/build/types/standardTsconfig.d.ts.map +1 -0
  63. package/build/types/util/checkAlpha.d.ts +10 -0
  64. package/build/types/util/checkAlpha.d.ts.map +1 -0
  65. package/build/types/util/createNewRuleConversion.d.ts +30 -0
  66. package/build/types/util/createNewRuleConversion.d.ts.map +1 -0
  67. package/build/types/util/makeDirectory.d.ts +8 -0
  68. package/build/types/util/makeDirectory.d.ts.map +1 -0
  69. package/build/types/util/packageDependencyGraphService.d.ts +37 -0
  70. package/build/types/util/packageDependencyGraphService.d.ts.map +1 -0
  71. package/{jest.config.js → jest.config.cjs} +0 -0
  72. package/package.json +39 -20
  73. package/src/__tests__/alphabeticalScripts.spec.ts +76 -0
  74. package/src/__tests__/bannedDependencies.spec.ts +191 -0
  75. package/src/__tests__/consistentDependencies.spec.ts +41 -27
  76. package/src/__tests__/consistentVersions.spec.ts +224 -0
  77. package/src/__tests__/fileContents.spec.ts +75 -0
  78. package/src/__tests__/mustSatisfyPeerDependencies.spec.ts +1189 -0
  79. package/src/__tests__/nestedWorkspaces.spec.ts +153 -0
  80. package/src/__tests__/packageEntry.spec.ts +99 -31
  81. package/src/__tests__/packageOrder.spec.ts +48 -41
  82. package/src/__tests__/packageScript.spec.ts +66 -56
  83. package/src/__tests__/requireDependency.spec.ts +152 -0
  84. package/src/__tests__/utils.ts +115 -11
  85. package/src/alphabeticalDependencies.ts +6 -48
  86. package/src/alphabeticalScripts.ts +21 -0
  87. package/src/bannedDependencies.ts +135 -44
  88. package/src/consistentDependencies.ts +38 -14
  89. package/src/consistentVersions.ts +142 -0
  90. package/src/fileContents.ts +35 -30
  91. package/src/index.ts +13 -8
  92. package/src/mustSatisfyPeerDependencies.ts +748 -0
  93. package/src/nestedWorkspaces.ts +61 -0
  94. package/src/packageEntry.ts +72 -27
  95. package/src/packageOrder.ts +13 -9
  96. package/src/packageScript.ts +13 -15
  97. package/src/requireDependency.ts +71 -0
  98. package/src/standardTsconfig.ts +49 -24
  99. package/src/util/checkAlpha.ts +59 -0
  100. package/src/util/createNewRuleConversion.ts +38 -0
  101. package/src/util/makeDirectory.ts +24 -0
  102. package/src/util/packageDependencyGraphService.ts +114 -0
  103. package/tsconfig.json +10 -2
  104. package/lib/__tests__/consistentDependencies.spec.d.ts.map +0 -1
  105. package/lib/__tests__/consistentDependencies.spec.js +0 -108
  106. package/lib/__tests__/consistentDependencies.spec.js.map +0 -1
  107. package/lib/__tests__/packageEntry.spec.d.ts.map +0 -1
  108. package/lib/__tests__/packageEntry.spec.js +0 -99
  109. package/lib/__tests__/packageEntry.spec.js.map +0 -1
  110. package/lib/__tests__/packageOrder.spec.d.ts.map +0 -1
  111. package/lib/__tests__/packageOrder.spec.js +0 -115
  112. package/lib/__tests__/packageOrder.spec.js.map +0 -1
  113. package/lib/__tests__/packageScript.spec.d.ts.map +0 -1
  114. package/lib/__tests__/packageScript.spec.js +0 -172
  115. package/lib/__tests__/packageScript.spec.js.map +0 -1
  116. package/lib/__tests__/utils.d.ts.map +0 -1
  117. package/lib/__tests__/utils.js +0 -23
  118. package/lib/__tests__/utils.js.map +0 -1
  119. package/lib/alphabeticalDependencies.d.ts +0 -10
  120. package/lib/alphabeticalDependencies.d.ts.map +0 -1
  121. package/lib/alphabeticalDependencies.js +0 -56
  122. package/lib/alphabeticalDependencies.js.map +0 -1
  123. package/lib/bannedDependencies.d.ts +0 -15
  124. package/lib/bannedDependencies.d.ts.map +0 -1
  125. package/lib/bannedDependencies.js +0 -57
  126. package/lib/bannedDependencies.js.map +0 -1
  127. package/lib/consistentDependencies.d.ts +0 -10
  128. package/lib/consistentDependencies.d.ts.map +0 -1
  129. package/lib/consistentDependencies.js +0 -57
  130. package/lib/consistentDependencies.js.map +0 -1
  131. package/lib/fileContents.d.ts +0 -25
  132. package/lib/fileContents.d.ts.map +0 -1
  133. package/lib/fileContents.js +0 -77
  134. package/lib/fileContents.js.map +0 -1
  135. package/lib/index.d.ts +0 -15
  136. package/lib/index.d.ts.map +0 -1
  137. package/lib/index.js +0 -25
  138. package/lib/index.js.map +0 -1
  139. package/lib/packageEntry.d.ts +0 -16
  140. package/lib/packageEntry.d.ts.map +0 -1
  141. package/lib/packageEntry.js +0 -40
  142. package/lib/packageEntry.js.map +0 -1
  143. package/lib/packageOrder.d.ts +0 -12
  144. package/lib/packageOrder.d.ts.map +0 -1
  145. package/lib/packageOrder.js +0 -103
  146. package/lib/packageOrder.js.map +0 -1
  147. package/lib/packageScript.d.ts +0 -25
  148. package/lib/packageScript.d.ts.map +0 -1
  149. package/lib/packageScript.js +0 -89
  150. package/lib/packageScript.js.map +0 -1
  151. package/lib/standardTsconfig.d.ts +0 -33
  152. package/lib/standardTsconfig.d.ts.map +0 -1
  153. package/lib/standardTsconfig.js +0 -98
  154. package/lib/standardTsconfig.js.map +0 -1
  155. package/tsconfig.tsbuildinfo +0 -2439
@@ -0,0 +1,1433 @@
1
+ // src/alphabeticalDependencies.ts
2
+ import * as r from "runtypes";
3
+
4
+ // src/util/checkAlpha.ts
5
+ import { diff } from "jest-diff";
6
+ function checkAlpha(context, block) {
7
+ const packageJson = context.getPackageJson();
8
+ const packagePath = context.getPackageJsonPath();
9
+ const blockToSort = packageJson[block];
10
+ if (blockToSort === void 0) {
11
+ return;
12
+ }
13
+ const actualOrder = Object.keys(blockToSort);
14
+ const expectedOrder = actualOrder.slice().sort();
15
+ if (!arrayOrderCompare(actualOrder, expectedOrder)) {
16
+ context.addError({
17
+ file: packagePath,
18
+ message: createIncorrectOrderErrorMessage(block, packageJson.name),
19
+ longMessage: diff(expectedOrder, actualOrder, { expand: true }),
20
+ fixer: () => {
21
+ const expectedDependencies = {};
22
+ expectedOrder.forEach((dep) => {
23
+ expectedDependencies[dep] = blockToSort[dep];
24
+ });
25
+ const newPackageJson = { ...packageJson };
26
+ newPackageJson[block] = expectedDependencies;
27
+ context.host.writeJson(packagePath, newPackageJson);
28
+ }
29
+ });
30
+ }
31
+ }
32
+ function arrayOrderCompare(a, b) {
33
+ for (let index = 0; index < a.length; index++) {
34
+ if (a[index] !== b[index]) {
35
+ return false;
36
+ }
37
+ }
38
+ return true;
39
+ }
40
+ function createIncorrectOrderErrorMessage(block, packageName) {
41
+ return `Incorrect order of ${block} in ${packageName}'s package.json`;
42
+ }
43
+
44
+ // src/util/createNewRuleConversion.ts
45
+ var id = 0;
46
+ var NewRuleConverterBase = class {
47
+ constructor(name, checkFunc, optionsRuntype, ruleEntry) {
48
+ this.name = name;
49
+ this.checkFunc = checkFunc;
50
+ this.optionsRuntype = optionsRuntype;
51
+ this.ruleEntry = ruleEntry;
52
+ this.check = (context) => this.checkFunc(context, this.ruleEntry.options, { id: this.id });
53
+ this.options = ruleEntry.options;
54
+ this.id = `${this.name} :: ${id++}`;
55
+ }
56
+ };
57
+ function createNewRuleConversion(name, old) {
58
+ return class NewRule extends NewRuleConverterBase {
59
+ constructor(ruleEntry) {
60
+ super(name, old.check, old.optionsRuntype, ruleEntry);
61
+ }
62
+ };
63
+ }
64
+
65
+ // src/alphabeticalDependencies.ts
66
+ var Options = r.Undefined;
67
+ var alphabeticalDependencies = {
68
+ check: function expectAlphabeticalDependencies(context) {
69
+ checkAlpha(context, "dependencies");
70
+ checkAlpha(context, "devDependencies");
71
+ checkAlpha(context, "peerDependencies");
72
+ },
73
+ optionsRuntype: Options
74
+ };
75
+ var AlphabeticalDependencies = createNewRuleConversion("AlphabetialDependencies", alphabeticalDependencies);
76
+
77
+ // src/alphabeticalScripts.ts
78
+ import * as r2 from "runtypes";
79
+ var Options2 = r2.Undefined;
80
+ var alphabeticalScripts = {
81
+ check: function expectAlphabeticalScripts(context) {
82
+ checkAlpha(context, "scripts");
83
+ },
84
+ optionsRuntype: Options2
85
+ };
86
+ var AlphabeticalScripts = createNewRuleConversion("AlphabeticalScripts", alphabeticalScripts);
87
+
88
+ // src/bannedDependencies.ts
89
+ import { matchesAnyGlob } from "@monorepolint/utils";
90
+ import { AggregateTiming } from "@monorepolint/utils";
91
+ import * as path2 from "path";
92
+ import * as r3 from "runtypes";
93
+
94
+ // src/util/packageDependencyGraphService.ts
95
+ import * as path from "path";
96
+ import resolvePackagePath from "resolve-package-path";
97
+ var PackageDependencyGraphService = class {
98
+ buildDependencyGraph(startPackageJsonPath, host, maxDepth) {
99
+ const nodes = /* @__PURE__ */ new Map();
100
+ const visit = (packageJsonPath, currentDepth) => {
101
+ if (nodes.has(packageJsonPath)) {
102
+ return nodes.get(packageJsonPath);
103
+ }
104
+ const packageJson = host.readJson(packageJsonPath);
105
+ const node = {
106
+ packageJson,
107
+ dependencies: /* @__PURE__ */ new Map(),
108
+ paths: {
109
+ packageJsonPath,
110
+ rootDirectory: path.dirname(packageJsonPath)
111
+ }
112
+ };
113
+ nodes.set(packageJsonPath, node);
114
+ const nextDepth = currentDepth + 1;
115
+ if (maxDepth == null || nextDepth <= maxDepth) {
116
+ const dependencies = packageJson.dependencies != null ? Object.keys(packageJson.dependencies) : [];
117
+ for (const dependency of dependencies) {
118
+ const dependencyPackageJsonPath = resolvePackagePath(dependency, node.paths.rootDirectory);
119
+ if (dependencyPackageJsonPath == null) {
120
+ throw new Error(`Could not resolve ${dependency} from ${node.paths.rootDirectory}`);
121
+ }
122
+ node.dependencies.set(dependency, visit(dependencyPackageJsonPath, nextDepth));
123
+ }
124
+ }
125
+ return node;
126
+ };
127
+ return visit(startPackageJsonPath, 0);
128
+ }
129
+ *traverse(root, opts = { traverseAllPaths: false }) {
130
+ const visited = /* @__PURE__ */ new Set();
131
+ function* visit(node, importPath = []) {
132
+ if (!opts.traverseAllPaths && visited.has(node)) {
133
+ return;
134
+ }
135
+ if (importPath.indexOf(node) !== -1) {
136
+ return;
137
+ }
138
+ visited.add(node);
139
+ importPath = [...importPath, node];
140
+ yield { ...node, importPath };
141
+ for (const dependency of node.dependencies.values()) {
142
+ yield* visit(dependency, importPath);
143
+ }
144
+ }
145
+ yield* visit(root);
146
+ }
147
+ };
148
+
149
+ // src/bannedDependencies.ts
150
+ var bannedDepGlobsField = r3.Union(
151
+ r3.Array(r3.String),
152
+ r3.Record({
153
+ glob: r3.Array(r3.String).optional(),
154
+ exact: r3.Array(r3.String).optional()
155
+ })
156
+ );
157
+ var Options3 = r3.Union(
158
+ r3.Record({
159
+ bannedDependencies: bannedDepGlobsField,
160
+ bannedTransitiveDependencies: r3.Undefined.optional()
161
+ }),
162
+ r3.Record({
163
+ bannedDependencies: bannedDepGlobsField.optional(),
164
+ bannedTransitiveDependencies: r3.Array(r3.String)
165
+ }),
166
+ r3.Record({
167
+ bannedDependencies: bannedDepGlobsField.optional(),
168
+ bannedTransitiveDependencies: r3.Array(r3.String).optional()
169
+ })
170
+ );
171
+ var setCache = /* @__PURE__ */ new Map();
172
+ var aggregateTiming = new AggregateTiming(":bannedDependencies stats");
173
+ var bannedDependencies = {
174
+ check: function expectAllowedDependencies(context, opts, extra) {
175
+ aggregateTiming.start(extra?.id ?? "unknown id");
176
+ const packageJson = context.getPackageJson();
177
+ const packagePath = context.getPackageJsonPath();
178
+ const curDeps = packageJson.dependencies && Object.keys(packageJson.dependencies);
179
+ const curDevDeps = packageJson.devDependencies && Object.keys(packageJson.devDependencies);
180
+ const curPeerDeps = packageJson.peerDependencies && Object.keys(packageJson.peerDependencies);
181
+ const { bannedDependencies: banned, bannedTransitiveDependencies: transitives } = opts;
182
+ const globs = banned && (Array.isArray(banned) ? banned : banned.glob);
183
+ const exacts = banned && (Array.isArray(banned) ? void 0 : banned.exact);
184
+ const violations = /* @__PURE__ */ new Set();
185
+ if (globs) {
186
+ if (curDeps)
187
+ populateProblemsGlobs(globs, curDeps, violations);
188
+ if (curDevDeps)
189
+ populateProblemsGlobs(globs, curDevDeps, violations);
190
+ if (curPeerDeps)
191
+ populateProblemsGlobs(globs, curPeerDeps, violations);
192
+ }
193
+ if (exacts) {
194
+ let set = setCache.get(exacts);
195
+ if (set === void 0) {
196
+ set = new Set(exacts);
197
+ setCache.set(exacts, set);
198
+ }
199
+ if (curDeps)
200
+ populateProblemsExact(set, curDeps, violations);
201
+ if (curDevDeps)
202
+ populateProblemsExact(set, curDevDeps, violations);
203
+ if (curPeerDeps)
204
+ populateProblemsExact(set, curPeerDeps, violations);
205
+ }
206
+ if (violations.size > 0) {
207
+ context.addError({
208
+ file: packagePath,
209
+ message: `Found ${violations.size} banned dependencies of package.json:
210
+ ` + Array.from(violations).map((v) => `'${v}'`).join(", ")
211
+ });
212
+ }
213
+ if (transitives) {
214
+ let set = setCache.get(transitives);
215
+ if (set === void 0) {
216
+ set = new Set(transitives);
217
+ setCache.set(transitives, set);
218
+ }
219
+ checkTransitives(context, set);
220
+ }
221
+ aggregateTiming.stop();
222
+ },
223
+ optionsRuntype: Options3,
224
+ printStats: () => {
225
+ aggregateTiming.printResults();
226
+ }
227
+ };
228
+ var BannedDependencies = createNewRuleConversion("BannedDependencies", bannedDependencies);
229
+ function populateProblemsExact(banned, dependencies, violations) {
230
+ for (const dependency of dependencies) {
231
+ if (banned.has(dependency)) {
232
+ violations.add(dependency);
233
+ }
234
+ }
235
+ }
236
+ function populateProblemsGlobs(bannedDependencyGlobs, dependencies, violations) {
237
+ for (const dependency of dependencies) {
238
+ if (matchesAnyGlob(dependency, bannedDependencyGlobs)) {
239
+ violations.add(dependency);
240
+ }
241
+ }
242
+ }
243
+ function checkTransitives(context, banned) {
244
+ const graphService = new PackageDependencyGraphService();
245
+ const root = graphService.buildDependencyGraph(path2.resolve(context.getPackageJsonPath()), context.host);
246
+ for (const { dependencies, importPath } of graphService.traverse(root)) {
247
+ for (const [dependency] of dependencies) {
248
+ if (banned.has(dependency)) {
249
+ const [, ...importPathWithoutRoot] = importPath;
250
+ const pathing = [...importPathWithoutRoot.map(nameOrPackageJsonPath), dependency].join(" -> ");
251
+ context.addError({
252
+ file: root.paths.packageJsonPath,
253
+ message: `Banned transitive dependencies in repo: ${pathing}`
254
+ });
255
+ }
256
+ }
257
+ }
258
+ }
259
+ function nameOrPackageJsonPath(node) {
260
+ return node.packageJson.name ?? node.paths.packageJsonPath;
261
+ }
262
+
263
+ // src/consistentDependencies.ts
264
+ import { diff as diff2 } from "jest-diff";
265
+ import * as r4 from "runtypes";
266
+ var Options4 = r4.Record({
267
+ ignoredDependencies: r4.Array(r4.String).Or(r4.Undefined)
268
+ }).Or(r4.Undefined);
269
+ var skippedVersions = ["*", "latest"];
270
+ var consistentDependencies = {
271
+ check: function expectConsistentDependencies(context, args) {
272
+ checkDeps(context, args, "dependencies");
273
+ checkDeps(context, args, "devDependencies");
274
+ },
275
+ optionsRuntype: Options4
276
+ };
277
+ var ConsistentDependencies = createNewRuleConversion("ConsistentDependencies", consistentDependencies);
278
+ function checkDeps(context, args, block) {
279
+ const packageJson = context.getPackageJson();
280
+ const packagePath = context.getPackageJsonPath();
281
+ const dependencies = packageJson[block];
282
+ const workspacePackageJson = context.getWorkspaceContext().getPackageJson();
283
+ const workspaceDependencies = workspacePackageJson[block];
284
+ const ignoredDeps = args?.ignoredDependencies ?? [];
285
+ const depsToCheck = workspaceDependencies == null || ignoredDeps.length === 0 ? workspaceDependencies : omit(workspaceDependencies, ignoredDeps);
286
+ if (dependencies === void 0 || depsToCheck === void 0) {
287
+ return;
288
+ }
289
+ const expectedDependencies = {
290
+ ...dependencies,
291
+ ...filterKeys(depsToCheck, dependencies)
292
+ };
293
+ if (JSON.stringify(dependencies) !== JSON.stringify(expectedDependencies)) {
294
+ context.addError({
295
+ file: packagePath,
296
+ message: `Inconsistent ${block} with root in package.json`,
297
+ longMessage: diff2(expectedDependencies, dependencies, { expand: true }),
298
+ fixer: () => {
299
+ const newPackageJson = { ...packageJson };
300
+ newPackageJson[block] = expectedDependencies;
301
+ context.host.writeJson(packagePath, newPackageJson);
302
+ }
303
+ });
304
+ }
305
+ }
306
+ function filterKeys(ob, filterOb) {
307
+ const newOb = {};
308
+ for (const key of Object.keys(filterOb)) {
309
+ if (ob[key] !== void 0 && skippedVersions.indexOf(filterOb[key]) === -1) {
310
+ newOb[key] = ob[key];
311
+ }
312
+ }
313
+ return newOb;
314
+ }
315
+ function omit(obj, keysToOmit) {
316
+ const newObj = {};
317
+ const filtered = Object.entries(obj).filter(([key]) => !keysToOmit.includes(key));
318
+ for (const [key, value] of filtered) {
319
+ newObj[key] = value;
320
+ }
321
+ return newObj;
322
+ }
323
+
324
+ // src/consistentVersions.ts
325
+ import { mutateJson } from "@monorepolint/utils";
326
+ import * as r5 from "runtypes";
327
+ import { coerce } from "semver";
328
+ var Options5 = r5.Record({
329
+ matchDependencyVersions: r5.Dictionary(r5.Union(r5.String, r5.Array(r5.String)))
330
+ });
331
+ var consistentVersions = {
332
+ check: checkConsistentVersions,
333
+ optionsRuntype: Options5
334
+ };
335
+ function checkConsistentVersions(context, options) {
336
+ for (const [dependencyPackageName, expectedPackageDependencyValue] of Object.entries(
337
+ options.matchDependencyVersions
338
+ )) {
339
+ if (Array.isArray(expectedPackageDependencyValue)) {
340
+ ensurePackageMatchesSomeVersion(context, dependencyPackageName, expectedPackageDependencyValue);
341
+ } else {
342
+ ensurePackageIsCorrectVersion(context, dependencyPackageName, expectedPackageDependencyValue);
343
+ }
344
+ }
345
+ }
346
+ var ensurePackageIsCorrectVersion = (context, dependencyPackageName, expectedPackageDependencyValue) => {
347
+ const packageJson = context.getPackageJson();
348
+ const packageJsonPath = context.getPackageJsonPath();
349
+ const expectedPackageDependencyVersion = coerce(expectedPackageDependencyValue);
350
+ if (expectedPackageDependencyVersion == null) {
351
+ throw new Error(
352
+ `Malformed expected package dependency version defined in monorepolint configuration: ${dependencyPackageName} @ '${expectedPackageDependencyValue}'`
353
+ );
354
+ }
355
+ const actualPackageDependencyValue = packageJson.dependencies && packageJson.dependencies[dependencyPackageName];
356
+ const actualPackageDependencyVersion = coerce(actualPackageDependencyValue);
357
+ if (actualPackageDependencyVersion != null && actualPackageDependencyVersion.raw !== expectedPackageDependencyVersion.raw) {
358
+ context.addError({
359
+ file: packageJsonPath,
360
+ message: `Expected dependency on ${dependencyPackageName} to match version defined in monorepolint configuration '${expectedPackageDependencyValue}', got '${actualPackageDependencyValue}' instead.`,
361
+ fixer: () => mutateJson(packageJsonPath, context.host, (input) => {
362
+ input.dependencies[dependencyPackageName] = expectedPackageDependencyValue;
363
+ return input;
364
+ })
365
+ });
366
+ }
367
+ const actualPackageDevDependencyValue = packageJson.devDependencies && packageJson.devDependencies[dependencyPackageName];
368
+ const actualPackageDevDependencyVersion = coerce(actualPackageDevDependencyValue);
369
+ if (actualPackageDevDependencyVersion != null && actualPackageDevDependencyVersion.raw !== expectedPackageDependencyVersion.raw) {
370
+ context.addError({
371
+ file: packageJsonPath,
372
+ message: `Expected devDependency on ${dependencyPackageName} to match version defined in monorepolint configuration '${expectedPackageDependencyValue}', got '${actualPackageDevDependencyValue}' instead`,
373
+ fixer: () => mutateJson(packageJsonPath, context.host, (input) => {
374
+ input.devDependencies[dependencyPackageName] = expectedPackageDependencyValue;
375
+ return input;
376
+ })
377
+ });
378
+ }
379
+ };
380
+ var ensurePackageMatchesSomeVersion = (context, dependencyPackageName, acceptedPackageDependencyValues) => {
381
+ const packageJson = context.getPackageJson();
382
+ const packageJsonPath = context.getPackageJsonPath();
383
+ const acceptedPackageDependencyVersions = acceptedPackageDependencyValues.map(
384
+ (acceptedPackageDependencyValue) => {
385
+ const acceptedPackageDependencyVersion = coerce(acceptedPackageDependencyValue);
386
+ if (acceptedPackageDependencyVersion == null) {
387
+ throw new Error(
388
+ `Malformed accepted package dependency version defined in monorepolint configuration: ${dependencyPackageName} @ '${acceptedPackageDependencyValue}'`
389
+ );
390
+ }
391
+ return acceptedPackageDependencyVersion;
392
+ }
393
+ );
394
+ const actualPackageDependencyValue = packageJson.dependencies && packageJson.dependencies[dependencyPackageName];
395
+ const actualPackageDependencyVersion = coerce(actualPackageDependencyValue);
396
+ if (actualPackageDependencyVersion != null && acceptedPackageDependencyVersions.every(
397
+ (acceptedPackageDependencyVersion) => actualPackageDependencyVersion.raw !== acceptedPackageDependencyVersion.raw
398
+ )) {
399
+ context.addError({
400
+ file: packageJsonPath,
401
+ message: `Expected dependency on ${dependencyPackageName} to match one of '${JSON.stringify(
402
+ acceptedPackageDependencyValues
403
+ )}', got '${actualPackageDependencyValue}' instead.`
404
+ });
405
+ }
406
+ const actualPackageDevDependencyValue = packageJson.devDependencies && packageJson.devDependencies[dependencyPackageName];
407
+ const actualPackageDevDependencyVersion = coerce(actualPackageDevDependencyValue);
408
+ if (actualPackageDevDependencyVersion != null && acceptedPackageDependencyVersions.every(
409
+ (acceptedPackageDependencyVersion) => actualPackageDevDependencyVersion.raw !== acceptedPackageDependencyVersion.raw
410
+ )) {
411
+ context.addError({
412
+ file: packageJsonPath,
413
+ message: `Expected devDependency on ${dependencyPackageName} to match one of '${JSON.stringify(
414
+ acceptedPackageDependencyValues
415
+ )}', got '${actualPackageDevDependencyValue}' instead.`
416
+ });
417
+ }
418
+ };
419
+ var ConsistentVersions = createNewRuleConversion("ConsistentVersions", consistentVersions);
420
+
421
+ // src/fileContents.ts
422
+ import { diff as diff3 } from "jest-diff";
423
+ import * as path3 from "path";
424
+ import * as r6 from "runtypes";
425
+ var Options6 = r6.Union(
426
+ r6.Record({
427
+ file: r6.String,
428
+ generator: r6.Function,
429
+ template: r6.Undefined.optional(),
430
+ templateFile: r6.Undefined.optional()
431
+ }),
432
+ r6.Record({
433
+ file: r6.String,
434
+ generator: r6.Undefined.optional(),
435
+ template: r6.String,
436
+ templateFile: r6.Undefined.optional()
437
+ }),
438
+ r6.Record({
439
+ file: r6.String,
440
+ generator: r6.Undefined.optional(),
441
+ template: r6.Undefined.optional(),
442
+ templateFile: r6.String
443
+ })
444
+ );
445
+ var fileContents = {
446
+ check: function expectFileContents(context, opts) {
447
+ const fullPath = path3.join(context.packageDir, opts.file);
448
+ const expectedContent = getExpectedContents(context, opts);
449
+ const pathExists = context.host.exists(fullPath);
450
+ const actualContent = pathExists ? context.host.readFile(fullPath, { encoding: "utf-8" }) : void 0;
451
+ if (actualContent !== expectedContent) {
452
+ context.addError({
453
+ file: fullPath,
454
+ message: "Expect file contents to match",
455
+ longMessage: diff3(expectedContent, actualContent, { expand: true }),
456
+ fixer: () => {
457
+ if (expectedContent === void 0) {
458
+ if (pathExists)
459
+ context.host.deleteFile(fullPath);
460
+ } else {
461
+ context.host.mkdir(path3.dirname(fullPath), { recursive: true });
462
+ context.host.writeFile(fullPath, expectedContent, { encoding: "utf-8" });
463
+ }
464
+ }
465
+ });
466
+ }
467
+ },
468
+ optionsRuntype: Options6
469
+ };
470
+ var optionsCache = /* @__PURE__ */ new Map();
471
+ function getExpectedContents(context, opts) {
472
+ if (optionsCache.has(opts)) {
473
+ const cachedEntry = optionsCache.get(opts);
474
+ if (cachedEntry && typeof cachedEntry === "function") {
475
+ return cachedEntry(context);
476
+ }
477
+ return cachedEntry;
478
+ }
479
+ if (opts.generator) {
480
+ optionsCache.set(opts, opts.generator);
481
+ return opts.generator(context);
482
+ } else if (opts.templateFile) {
483
+ const { packageDir: workspacePackageDir } = context.getWorkspaceContext();
484
+ const fullPath = path3.resolve(workspacePackageDir, opts.templateFile);
485
+ const template = context.host.readFile(fullPath, { encoding: "utf-8" });
486
+ optionsCache.set(opts, template);
487
+ return template;
488
+ } else {
489
+ optionsCache.set(opts, opts.template);
490
+ return opts.template;
491
+ }
492
+ }
493
+ var FileContents = createNewRuleConversion("FileContents", fileContents);
494
+
495
+ // src/mustSatisfyPeerDependencies.ts
496
+ import { mutateJson as mutateJson2 } from "@monorepolint/utils";
497
+ import * as path4 from "path";
498
+ import * as r7 from "runtypes";
499
+ import { coerce as coerce2 } from "semver";
500
+ import resolvePackagePath2 from "resolve-package-path";
501
+ var Options7 = r7.Union(
502
+ r7.Partial({
503
+ skipUnparseableRanges: r7.Undefined,
504
+ dependencyWhitelist: r7.Undefined,
505
+ dependencyBlacklist: r7.Undefined,
506
+ enforceForDevDependencies: r7.Undefined
507
+ }),
508
+ r7.Record({
509
+ skipUnparseableRanges: r7.Boolean
510
+ }).And(
511
+ r7.Partial({
512
+ dependencyWhitelist: r7.Undefined,
513
+ dependencyBlacklist: r7.Undefined,
514
+ enforceForDevDependencies: r7.Undefined
515
+ })
516
+ ),
517
+ r7.Record({
518
+ dependencyWhitelist: r7.Array(r7.String)
519
+ }).And(
520
+ r7.Partial({
521
+ skipUnparseableRanges: r7.Undefined,
522
+ dependencyBlacklist: r7.Undefined,
523
+ enforceForDevDependencies: r7.Undefined
524
+ })
525
+ ),
526
+ r7.Record({
527
+ dependencyBlacklist: r7.Array(r7.String)
528
+ }).And(
529
+ r7.Partial({
530
+ skipUnparseableRanges: r7.Undefined,
531
+ dependencyWhitelist: r7.Undefined,
532
+ enforceForDevDependencies: r7.Undefined
533
+ })
534
+ ),
535
+ r7.Record({
536
+ enforceForDevDependencies: r7.Boolean
537
+ }).And(
538
+ r7.Partial({
539
+ skipUnparseableRanges: r7.Undefined,
540
+ dependencyWhitelist: r7.Undefined,
541
+ dependencyBlacklist: r7.Undefined
542
+ })
543
+ ),
544
+ r7.Record({
545
+ skipUnparseableRanges: r7.Boolean,
546
+ dependencyWhitelist: r7.Array(r7.String)
547
+ }).And(
548
+ r7.Partial({
549
+ dependencyBlacklist: r7.Undefined,
550
+ enforceForDevDependencies: r7.Undefined
551
+ })
552
+ ),
553
+ r7.Record({
554
+ skipUnparseableRanges: r7.Boolean,
555
+ dependencyBlacklist: r7.Array(r7.String)
556
+ }).And(
557
+ r7.Partial({
558
+ dependencyWhitelist: r7.Undefined,
559
+ enforceForDevDependencies: r7.Undefined
560
+ })
561
+ ),
562
+ r7.Record({
563
+ skipUnparseableRanges: r7.Boolean,
564
+ enforceForDevDependencies: r7.Boolean
565
+ }).And(
566
+ r7.Partial({
567
+ dependencyWhitelist: r7.Undefined,
568
+ dependencyBlacklist: r7.Undefined
569
+ })
570
+ ),
571
+ r7.Record({
572
+ dependencyWhitelist: r7.Array(r7.String),
573
+ dependencyBlacklist: r7.Array(r7.String)
574
+ }).And(
575
+ r7.Partial({
576
+ skipUnparseableRanges: r7.Undefined,
577
+ enforceForDevDependencies: r7.Undefined
578
+ })
579
+ ),
580
+ r7.Record({
581
+ dependencyWhitelist: r7.Array(r7.String),
582
+ enforceForDevDependencies: r7.Boolean
583
+ }).And(
584
+ r7.Partial({
585
+ skipUnparseableRanges: r7.Undefined,
586
+ dependencyBlacklist: r7.Undefined
587
+ })
588
+ ),
589
+ r7.Record({
590
+ dependencyBlacklist: r7.Array(r7.String),
591
+ enforceForDevDependencies: r7.Boolean
592
+ }).And(
593
+ r7.Partial({
594
+ skipUnparseableRanges: r7.Undefined,
595
+ dependencyWhitelist: r7.Undefined
596
+ })
597
+ ),
598
+ r7.Record({
599
+ skipUnparseableRanges: r7.Boolean,
600
+ dependencyWhitelist: r7.Array(r7.String),
601
+ dependencyBlacklist: r7.Array(r7.String)
602
+ }).And(
603
+ r7.Partial({
604
+ enforceForDevDependencies: r7.Undefined
605
+ })
606
+ ),
607
+ r7.Record({
608
+ skipUnparseableRanges: r7.Boolean,
609
+ dependencyWhitelist: r7.Array(r7.String),
610
+ enforceForDevDependencies: r7.Boolean
611
+ }).And(
612
+ r7.Partial({
613
+ dependencyBlacklist: r7.Undefined
614
+ })
615
+ ),
616
+ r7.Record({
617
+ skipUnparseableRanges: r7.Boolean,
618
+ dependencyBlacklist: r7.Array(r7.String),
619
+ enforceForDevDependencies: r7.Boolean
620
+ }).And(
621
+ r7.Partial({
622
+ dependencyWhitelist: r7.Undefined
623
+ })
624
+ ),
625
+ r7.Record({
626
+ dependencyWhitelist: r7.Array(r7.String),
627
+ dependencyBlacklist: r7.Array(r7.String),
628
+ enforceForDevDependencies: r7.Boolean
629
+ }).And(
630
+ r7.Partial({
631
+ skipUnparseableRanges: r7.Undefined
632
+ })
633
+ ),
634
+ r7.Record({
635
+ skipUnparseableRanges: r7.Boolean,
636
+ dependencyWhitelist: r7.Array(r7.String),
637
+ dependencyBlacklist: r7.Array(r7.String),
638
+ enforceForDevDependencies: r7.Boolean
639
+ })
640
+ );
641
+ var mustSatisfyPeerDependencies = {
642
+ check: checkSatisfyPeerDependencies,
643
+ optionsRuntype: Options7
644
+ };
645
+ var MustSatisfyPeerDependencies = createNewRuleConversion(
646
+ "MustSatisfyPeerDependencies",
647
+ mustSatisfyPeerDependencies
648
+ );
649
+ var MATCH_ANY_VERSION_RANGE = /^(\*|x)$/;
650
+ var 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-]+)*))?)?$/;
651
+ var 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-]+)*))?)$/;
652
+ var 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-]+)*))?)?)*)$/;
653
+ function checkSatisfyPeerDependencies(context, opts) {
654
+ const { dependencyBlacklist, dependencyWhitelist, enforceForDevDependencies, skipUnparseableRanges } = opts;
655
+ const packageJsonPath = path4.resolve(context.getPackageJsonPath());
656
+ const packageJson = context.host.readJson(packageJsonPath);
657
+ const packageDependencies = packageJson.dependencies || {};
658
+ const packageDevDependencies = packageJson.devDependencies || {};
659
+ const packagePeerDependencies = packageJson.peerDependencies || {};
660
+ const packageName = packageJson.name || packageJsonPath;
661
+ for (const [peerDependencyName, peerDependencyRange] of Object.entries(packagePeerDependencies)) {
662
+ if (shouldSkipPackage({ dependencyBlacklist, dependencyWhitelist, packageName: peerDependencyName })) {
663
+ continue;
664
+ }
665
+ const dependencyRange = packageDependencies[peerDependencyName];
666
+ if (dependencyRange != null) {
667
+ context.addError({
668
+ file: packageJsonPath,
669
+ message: `[0] Package ${packageName} has overloaded ${peerDependencyName} dependencies.
670
+ Peer dependency '${peerDependencyRange}' and regular dependency '${dependencyRange}'.`
671
+ });
672
+ }
673
+ }
674
+ const allRequiredPeerDependencies = {};
675
+ const allDependencies = enforceForDevDependencies ? [...Object.keys(packageDependencies), ...Object.keys(packageDevDependencies)] : Object.keys(packageDependencies);
676
+ for (const dependency of allDependencies) {
677
+ const dependencyPackageJsonPath = resolvePackagePath2(dependency, path4.dirname(packageJsonPath));
678
+ if (dependencyPackageJsonPath == null) {
679
+ throw new Error(`Could not resolve ${dependency} from ${path4.dirname(packageJsonPath)}`);
680
+ }
681
+ const dependencyPackageJson = context.host.readJson(dependencyPackageJsonPath);
682
+ const requiredPeerDependencies = dependencyPackageJson.peerDependencies;
683
+ if (requiredPeerDependencies == null) {
684
+ continue;
685
+ }
686
+ for (const [peerDependencyName, range] of Object.entries(requiredPeerDependencies)) {
687
+ if (shouldSkipPackage({ dependencyBlacklist, dependencyWhitelist, packageName: peerDependencyName })) {
688
+ continue;
689
+ }
690
+ if (!isValidRange(range)) {
691
+ const message = `Unable to parse ${dependencyPackageJson.name}'s ${peerDependencyName} peer dependency range '${range}'.`;
692
+ if (skipUnparseableRanges) {
693
+ context.addWarning({ file: dependencyPackageJsonPath, message });
694
+ continue;
695
+ }
696
+ throw new Error(message);
697
+ }
698
+ if (allRequiredPeerDependencies[peerDependencyName] == null) {
699
+ allRequiredPeerDependencies[peerDependencyName] = [];
700
+ }
701
+ allRequiredPeerDependencies[peerDependencyName].push({ fromPackageName: dependencyPackageJson.name, range });
702
+ }
703
+ }
704
+ for (const [peerDependencyName, peerDependencyRequirements] of Object.entries(allRequiredPeerDependencies)) {
705
+ let mostStrictPeerRequirement = {
706
+ fromPeerDependencyRequirements: [peerDependencyRequirements[0]],
707
+ range: peerDependencyRequirements[0].range
708
+ };
709
+ for (const peerRequirement of peerDependencyRequirements) {
710
+ if (doesASatisfyB(mostStrictPeerRequirement.range, peerRequirement.range)) {
711
+ continue;
712
+ } else if (doesASatisfyB(peerRequirement.range, mostStrictPeerRequirement.range)) {
713
+ mostStrictPeerRequirement = {
714
+ fromPeerDependencyRequirements: [peerRequirement],
715
+ range: peerRequirement.range
716
+ };
717
+ } else {
718
+ const maybeIntersection = findIntersection(peerRequirement.range, mostStrictPeerRequirement.range);
719
+ if (maybeIntersection !== void 0) {
720
+ mostStrictPeerRequirement = {
721
+ fromPeerDependencyRequirements: [
722
+ ...mostStrictPeerRequirement.fromPeerDependencyRequirements,
723
+ peerRequirement
724
+ ],
725
+ range: maybeIntersection
726
+ };
727
+ } else {
728
+ context.addError({
729
+ file: packageJsonPath,
730
+ message: `[1] Package ${packageName} has conflicting inherited ${peerDependencyName} peer dependencies.
731
+ Dependency ${peerRequirement.fromPackageName} requires '${peerRequirement.range}' but
732
+ ` + getMostStrictStatement(mostStrictPeerRequirement)
733
+ });
734
+ }
735
+ }
736
+ }
737
+ const packageDependencyRange = packageDependencies[peerDependencyName];
738
+ if (packageDependencyRange != null) {
739
+ if (!isValidRange(packageDependencyRange)) {
740
+ const message = `Unable to parse ${packageName}'s ${peerDependencyName} dependency range '${packageDependencyRange}'.`;
741
+ if (skipUnparseableRanges) {
742
+ context.addWarning({ file: packageJsonPath, message });
743
+ } else {
744
+ throw new Error(message);
745
+ }
746
+ } else if (!doesASatisfyB(packageDependencyRange, mostStrictPeerRequirement.range)) {
747
+ context.addError({
748
+ file: packageJsonPath,
749
+ message: `[2] Package ${packageName} dependency on ${peerDependencyName} '${packageDependencyRange}' does not satisfy inherited peer dependencies.
750
+ ` + getMostStrictStatement(mostStrictPeerRequirement)
751
+ });
752
+ }
753
+ }
754
+ const packagePeerDependencyRange = packagePeerDependencies[peerDependencyName];
755
+ if (packageDependencyRange == null && packagePeerDependencyRange == null) {
756
+ context.addError({
757
+ file: packageJsonPath,
758
+ message: `[3] Package ${packageName} is missing required ${peerDependencyName} dependency.
759
+ ` + getMostStrictStatement(mostStrictPeerRequirement),
760
+ fixer: getAddDependencyTypeFixer({
761
+ packageJsonPath,
762
+ dependencyType: "peerDependencies",
763
+ dependencyName: peerDependencyName,
764
+ version: mostStrictPeerRequirement.range,
765
+ host: context.host
766
+ })
767
+ });
768
+ }
769
+ if (packagePeerDependencyRange != null) {
770
+ if (!isValidRange(packagePeerDependencyRange)) {
771
+ const message = `Unable to parse ${packageName}'s ${peerDependencyName} peer dependency range '${packagePeerDependencyRange}'.`;
772
+ if (skipUnparseableRanges) {
773
+ context.addWarning({ file: packageJsonPath, message });
774
+ } else {
775
+ throw new Error(message);
776
+ }
777
+ } else if (!doesASatisfyB(packagePeerDependencyRange, mostStrictPeerRequirement.range)) {
778
+ context.addError({
779
+ file: packageJsonPath,
780
+ message: `[4] Package ${packageName} peer dependency on ${peerDependencyName} '${packagePeerDependencyRange}' is not strict enough.
781
+ ` + getMostStrictStatement(mostStrictPeerRequirement),
782
+ fixer: getAddDependencyTypeFixer({
783
+ packageJsonPath,
784
+ dependencyType: "peerDependencies",
785
+ dependencyName: peerDependencyName,
786
+ version: mostStrictPeerRequirement.range,
787
+ host: context.host
788
+ })
789
+ });
790
+ }
791
+ }
792
+ }
793
+ }
794
+ function shouldSkipPackage({
795
+ dependencyBlacklist,
796
+ dependencyWhitelist,
797
+ packageName
798
+ }) {
799
+ if (dependencyBlacklist != null && dependencyBlacklist.includes(packageName) || dependencyWhitelist != null && !dependencyWhitelist.includes(packageName)) {
800
+ return true;
801
+ }
802
+ return false;
803
+ }
804
+ function getMostStrictStatement(mostStrictPeerRequirement) {
805
+ if (mostStrictPeerRequirement.fromPeerDependencyRequirements.length === 1) {
806
+ const dependencyName = mostStrictPeerRequirement.fromPeerDependencyRequirements[0].fromPackageName;
807
+ return `Dependency ${dependencyName} requires '${mostStrictPeerRequirement.range}'.`;
808
+ } else {
809
+ const dependencyNames = mostStrictPeerRequirement.fromPeerDependencyRequirements.map((peerDependencyRequirement) => peerDependencyRequirement.fromPackageName).join(", ");
810
+ const dependencyRequirements = mostStrictPeerRequirement.fromPeerDependencyRequirements.map((peerDependencyRequirement) => `'${peerDependencyRequirement.range}'`).join(", ");
811
+ return `Dependencies [${dependencyNames}] require [${dependencyRequirements}] respectively, resolving to '${mostStrictPeerRequirement.range}'.`;
812
+ }
813
+ }
814
+ function findIntersection(a, b) {
815
+ if (doesASatisfyB(a, b)) {
816
+ return a;
817
+ } else if (doesASatisfyB(b, a)) {
818
+ return b;
819
+ }
820
+ if (isAnyVersionRange(a) || isAnyVersionRange(b)) {
821
+ throw new Error();
822
+ }
823
+ const aVersions = a.includes("||") ? a.split("||").map((s) => s.trim()) : [a];
824
+ const bVersions = b.includes("||") ? b.split("||").map((s) => s.trim()) : [b];
825
+ const aIsGreaterOrEqualVersionRange = isGreaterOrEqualVersionRange(a);
826
+ const bIsGreaterOrEqualVersionRange = isGreaterOrEqualVersionRange(b);
827
+ if (aIsGreaterOrEqualVersionRange && bIsGreaterOrEqualVersionRange) {
828
+ throw new Error();
829
+ }
830
+ if (aIsGreaterOrEqualVersionRange) {
831
+ const aSemVer = coerce2(a);
832
+ const compatibleBVersions = bVersions.map((bVersion) => {
833
+ const bSemVer = coerce2(bVersion);
834
+ if (bVersion.startsWith("^") && bSemVer.major >= aSemVer.major) {
835
+ return `^${bSemVer.compare(aSemVer) >= 0 ? bSemVer.raw : aSemVer.raw}`;
836
+ }
837
+ return bSemVer.compare(aSemVer) !== -1 ? bVersion : void 0;
838
+ }).filter((bVersion) => bVersion != null);
839
+ if (compatibleBVersions.length === 0) {
840
+ return void 0;
841
+ }
842
+ return compatibleBVersions.join(" || ");
843
+ }
844
+ if (bIsGreaterOrEqualVersionRange) {
845
+ const bSemVer = coerce2(b);
846
+ const compatibleAVersions = aVersions.map((aVersion) => {
847
+ const aSemVer = coerce2(aVersion);
848
+ if (aVersion.startsWith("^") && aSemVer.major >= bSemVer.major) {
849
+ return `^${aSemVer.compare(bSemVer) >= 0 ? aSemVer.raw : bSemVer.raw}`;
850
+ }
851
+ return aSemVer.compare(bSemVer) !== -1 ? aVersion : void 0;
852
+ }).filter((aVersion) => aVersion != null);
853
+ if (compatibleAVersions.length === 0) {
854
+ return void 0;
855
+ }
856
+ return compatibleAVersions.join(" || ");
857
+ }
858
+ const compatibleVersions = aVersions.map((aVersion) => {
859
+ const aSemVer = coerce2(aVersion);
860
+ const majorMatchingBVersion = bVersions.find((m) => coerce2(m).major === aSemVer.major);
861
+ if (majorMatchingBVersion === void 0) {
862
+ return void 0;
863
+ }
864
+ if (doesASatisfyB(aVersion, majorMatchingBVersion)) {
865
+ return aVersion;
866
+ } else if (doesASatisfyB(majorMatchingBVersion, aVersion)) {
867
+ return majorMatchingBVersion;
868
+ } else {
869
+ return void 0;
870
+ }
871
+ }).filter((aVersion) => aVersion !== void 0);
872
+ if (compatibleVersions.length === 0) {
873
+ return void 0;
874
+ }
875
+ return compatibleVersions.join(" || ");
876
+ }
877
+ function doesASatisfyB(a, b) {
878
+ if (a === b) {
879
+ return true;
880
+ }
881
+ const aIsAnyVersionRange = isAnyVersionRange(a);
882
+ const bIsAnyVersionRange = isAnyVersionRange(b);
883
+ if (bIsAnyVersionRange) {
884
+ return true;
885
+ } else if (aIsAnyVersionRange) {
886
+ return false;
887
+ }
888
+ const aVersions = a.includes("||") ? a.split("||").map((s) => s.trim()) : [a];
889
+ const bVersions = b.includes("||") ? b.split("||").map((s) => s.trim()) : [b];
890
+ const aIsGreaterOrEqualVersionRange = isGreaterOrEqualVersionRange(a);
891
+ const bIsGreaterOrEqualVersionRange = isGreaterOrEqualVersionRange(b);
892
+ if (aIsGreaterOrEqualVersionRange && bIsGreaterOrEqualVersionRange) {
893
+ const aSemVer = coerce2(a);
894
+ const bSemVer = coerce2(b);
895
+ return aSemVer.compare(bSemVer) !== -1;
896
+ } else if (bIsGreaterOrEqualVersionRange) {
897
+ const bSemVer = coerce2(b);
898
+ return aVersions.every((aVersion) => {
899
+ const aSemVer = coerce2(aVersion);
900
+ return aSemVer.compare(bSemVer) !== -1;
901
+ });
902
+ } else if (aIsGreaterOrEqualVersionRange) {
903
+ return false;
904
+ }
905
+ return aVersions.every((aVersion) => {
906
+ const aSemVer = coerce2(aVersion);
907
+ const majorMatchingBVersion = bVersions.find((m) => coerce2(m).major === aSemVer.major);
908
+ if (majorMatchingBVersion === void 0) {
909
+ return false;
910
+ }
911
+ const aVersionIsRange = isMajorVersionRange(aVersion);
912
+ const majorMatchingBSemVer = coerce2(majorMatchingBVersion);
913
+ const majorMatchingBVersionIsRange = isMajorVersionRange(majorMatchingBVersion);
914
+ if (majorMatchingBVersionIsRange) {
915
+ return aSemVer.compare(majorMatchingBSemVer) !== -1;
916
+ } else {
917
+ if (aVersionIsRange) {
918
+ return false;
919
+ } else {
920
+ return aSemVer.compare(majorMatchingBSemVer) === 0;
921
+ }
922
+ }
923
+ });
924
+ }
925
+ function isAnyVersionRange(version) {
926
+ return MATCH_ANY_VERSION_RANGE.test(version);
927
+ }
928
+ function isGreaterOrEqualVersionRange(version) {
929
+ return MATCH_GREATER_OR_EQUAL_VERSION_RANGE.test(version);
930
+ }
931
+ function isMajorVersionRange(version) {
932
+ return MATCH_MAJOR_VERSION_RANGE.test(version);
933
+ }
934
+ function isValidRange(version) {
935
+ return RANGE_REGEX.test(version);
936
+ }
937
+ function getAddDependencyTypeFixer({
938
+ packageJsonPath,
939
+ dependencyType,
940
+ dependencyName,
941
+ version,
942
+ host
943
+ }) {
944
+ return () => {
945
+ mutateJson2(packageJsonPath, host, (packageJson) => {
946
+ if (packageJson[dependencyType] == null) {
947
+ packageJson[dependencyType] = {};
948
+ }
949
+ packageJson[dependencyType][dependencyName] = version;
950
+ return packageJson;
951
+ });
952
+ };
953
+ }
954
+
955
+ // src/packageOrder.ts
956
+ import { diff as diff4 } from "jest-diff";
957
+ import * as r8 from "runtypes";
958
+ var Options8 = r8.Record({
959
+ order: r8.Union(r8.Array(r8.String), r8.Function)
960
+ }).Or(r8.Undefined);
961
+ var defaultKeyOrder = [
962
+ "name",
963
+ "version",
964
+ "description",
965
+ "author",
966
+ "contributors",
967
+ "url",
968
+ "license",
969
+ "type",
970
+ "exports",
971
+ "private",
972
+ "engines",
973
+ "bin",
974
+ "types",
975
+ "main",
976
+ "module",
977
+ "typings",
978
+ "style",
979
+ "sideEffects",
980
+ "workspaces",
981
+ "husky",
982
+ "lint-staged",
983
+ "files",
984
+ "scripts",
985
+ "resolutions",
986
+ "dependencies",
987
+ "peerDependencies",
988
+ "devDependencies",
989
+ "optionalDependencies",
990
+ "publishConfig"
991
+ ];
992
+ var packageOrder = {
993
+ check: function expectPackageOrder(context, opts) {
994
+ const packageJson = context.getPackageJson();
995
+ const packagePath = context.getPackageJsonPath();
996
+ const order = opts === void 0 ? defaultKeyOrder : opts.order;
997
+ const comparator = isOrderFunction(order) ? order(context) : createComparator(order);
998
+ const actualOrder = Object.keys(packageJson);
999
+ const expectedOrder = actualOrder.slice().sort(comparator);
1000
+ if (!arrayOrderCompare2(actualOrder, expectedOrder)) {
1001
+ context.addError({
1002
+ file: packagePath,
1003
+ message: "Incorrect order of fields in package.json",
1004
+ longMessage: diff4(expectedOrder, actualOrder, { expand: true }),
1005
+ fixer: () => {
1006
+ const expectedPackageJson = {};
1007
+ expectedOrder.forEach((key) => {
1008
+ expectedPackageJson[key] = packageJson[key];
1009
+ });
1010
+ context.host.writeJson(packagePath, expectedPackageJson);
1011
+ }
1012
+ });
1013
+ }
1014
+ },
1015
+ optionsRuntype: Options8
1016
+ };
1017
+ var PackageOrder = createNewRuleConversion("PackageOrder", packageOrder);
1018
+ function arrayOrderCompare2(a, b) {
1019
+ for (let index = 0; index < a.length; index++) {
1020
+ if (a[index] !== b[index]) {
1021
+ return false;
1022
+ }
1023
+ }
1024
+ return true;
1025
+ }
1026
+ function createComparator(order) {
1027
+ return (a, b) => {
1028
+ const aIndex = order.indexOf(a);
1029
+ const bIndex = order.indexOf(b);
1030
+ if (aIndex >= 0 && bIndex < 0) {
1031
+ return -1;
1032
+ } else if (aIndex < 0 && bIndex >= 0) {
1033
+ return 1;
1034
+ }
1035
+ const compared = aIndex - bIndex;
1036
+ if (compared !== 0) {
1037
+ return compared;
1038
+ } else {
1039
+ return a.localeCompare(b);
1040
+ }
1041
+ };
1042
+ }
1043
+ function isOrderFunction(order) {
1044
+ return !Array.isArray(order);
1045
+ }
1046
+
1047
+ // src/packageEntry.ts
1048
+ import { mutateJson as mutateJson3 } from "@monorepolint/utils";
1049
+ import { diff as diff5 } from "jest-diff";
1050
+ import * as r9 from "runtypes";
1051
+ var Options9 = r9.Union(
1052
+ r9.Record({
1053
+ entries: r9.Dictionary(r9.Unknown)
1054
+ }).And(
1055
+ r9.Partial({
1056
+ entriesExist: r9.Undefined
1057
+ })
1058
+ ),
1059
+ r9.Record({
1060
+ entriesExist: r9.Array(r9.String)
1061
+ }).And(
1062
+ r9.Partial({
1063
+ entries: r9.Undefined
1064
+ })
1065
+ ),
1066
+ r9.Record({
1067
+ entries: r9.Dictionary(r9.Unknown),
1068
+ entriesExist: r9.Array(r9.String)
1069
+ })
1070
+ );
1071
+ var packageEntry = {
1072
+ check: function expectPackageEntry(context, options) {
1073
+ const packageJson = context.getPackageJson();
1074
+ if (options.entries) {
1075
+ for (const key of Object.keys(options.entries)) {
1076
+ const value = options.entries[key];
1077
+ const entryDiff = diff5(JSON.stringify(value) + "\n", (JSON.stringify(packageJson[key]) || "") + "\n");
1078
+ if (typeof value !== "object" && value !== packageJson[key] || entryDiff == null || !entryDiff.includes("Compared values have no visual difference")) {
1079
+ context.addError({
1080
+ file: context.getPackageJsonPath(),
1081
+ message: createStandardizedEntryErrorMessage(key),
1082
+ longMessage: entryDiff,
1083
+ fixer: () => {
1084
+ mutateJson3(context.getPackageJsonPath(), context.host, (input) => {
1085
+ input[key] = value;
1086
+ return input;
1087
+ });
1088
+ }
1089
+ });
1090
+ }
1091
+ }
1092
+ }
1093
+ if (options.entriesExist) {
1094
+ for (const key of options.entriesExist) {
1095
+ if (packageJson[key] === void 0) {
1096
+ context.addError({
1097
+ file: context.getPackageJsonPath(),
1098
+ message: createExpectedEntryErrorMessage(key)
1099
+ });
1100
+ }
1101
+ }
1102
+ }
1103
+ },
1104
+ optionsRuntype: Options9
1105
+ };
1106
+ var PackageEntry = createNewRuleConversion("PackageEntry", packageEntry);
1107
+ function createStandardizedEntryErrorMessage(key) {
1108
+ return `Expected standardized entry for '${key}'`;
1109
+ }
1110
+ function createExpectedEntryErrorMessage(key) {
1111
+ return `Expected entry for '${key}' to exist`;
1112
+ }
1113
+
1114
+ // src/packageScript.ts
1115
+ import { mutateJson as mutateJson4 } from "@monorepolint/utils";
1116
+ import { diff as diff6 } from "jest-diff";
1117
+ import * as r10 from "runtypes";
1118
+ var Options10 = r10.Record({
1119
+ scripts: r10.Dictionary(
1120
+ r10.Union(
1121
+ r10.String,
1122
+ r10.Record({
1123
+ options: r10.Array(r10.String.Or(r10.Undefined)),
1124
+ fixValue: r10.Union(r10.String, r10.Undefined, r10.Literal(false)).optional()
1125
+ })
1126
+ )
1127
+ )
1128
+ });
1129
+ var MSG_NO_SCRIPTS_BLOCK = "No scripts block in package.json";
1130
+ var packageScript = {
1131
+ check: function expectPackageScript(context, options) {
1132
+ const packageJson = context.getPackageJson();
1133
+ if (packageJson.scripts === void 0) {
1134
+ context.addError({
1135
+ file: context.getPackageJsonPath(),
1136
+ message: MSG_NO_SCRIPTS_BLOCK,
1137
+ fixer: () => {
1138
+ mutateJson4(context.getPackageJsonPath(), context.host, (input) => {
1139
+ input.scripts = {};
1140
+ return input;
1141
+ });
1142
+ }
1143
+ });
1144
+ return;
1145
+ }
1146
+ for (const [name, value] of Object.entries(options.scripts)) {
1147
+ const allowedValues = /* @__PURE__ */ new Set();
1148
+ let fixValue;
1149
+ let allowEmpty = false;
1150
+ let fixToEmpty = false;
1151
+ if (typeof value === "string") {
1152
+ allowedValues.add(value);
1153
+ fixValue = value;
1154
+ } else {
1155
+ for (const q of value.options) {
1156
+ if (q === void 0) {
1157
+ allowEmpty = true;
1158
+ }
1159
+ allowedValues.add(q);
1160
+ }
1161
+ fixToEmpty = Object.prototype.hasOwnProperty.call(value, "fixValue") && value.fixValue === void 0;
1162
+ fixValue = value.fixValue;
1163
+ }
1164
+ const actualValue = packageJson.scripts[name];
1165
+ if (!allowedValues.has(actualValue) && !(allowEmpty === true && actualValue === void 0)) {
1166
+ let fixer;
1167
+ if (fixValue !== false && (fixValue !== void 0 || fixToEmpty === true)) {
1168
+ const q = fixValue;
1169
+ fixer = () => {
1170
+ mutateJson4(context.getPackageJsonPath(), context.host, (input) => {
1171
+ if (fixToEmpty && q === void 0) {
1172
+ delete input.scripts[name];
1173
+ } else {
1174
+ input.scripts[name] = q;
1175
+ }
1176
+ return input;
1177
+ });
1178
+ };
1179
+ }
1180
+ const validOptionsString = Array.from(allowedValues.values()).map((a) => a === void 0 ? "(empty)" : `'${a}'`).join(", ");
1181
+ context.addError({
1182
+ file: context.getPackageJsonPath(),
1183
+ message: `Expected standardized script entry for '${name}'. Valid options: ${validOptionsString}`,
1184
+ longMessage: diff6(validOptionsString + "\n", (packageJson.scripts[name] || "") + "\n"),
1185
+ fixer
1186
+ });
1187
+ }
1188
+ }
1189
+ },
1190
+ optionsRuntype: Options10
1191
+ };
1192
+ var PackageScript = createNewRuleConversion("PackageScript", packageScript);
1193
+
1194
+ // src/standardTsconfig.ts
1195
+ import { matchesAnyGlob as matchesAnyGlob2 } from "@monorepolint/utils";
1196
+ import { diff as diff7 } from "jest-diff";
1197
+ import * as path5 from "path";
1198
+ import * as r11 from "runtypes";
1199
+ var DEFAULT_TSCONFIG_FILENAME = "tsconfig.json";
1200
+ var Options11 = r11.Partial({
1201
+ file: r11.String,
1202
+ generator: r11.Function,
1203
+ tsconfigReferenceFile: r11.String,
1204
+ template: r11.Record({}).Or(r11.String),
1205
+ templateFile: r11.String,
1206
+ excludedReferences: r11.Array(r11.String).Or(r11.Undefined),
1207
+ additionalReferences: r11.Array(r11.String).Or(r11.Undefined)
1208
+ }).withConstraint(({ generator, template, templateFile }) => {
1209
+ let count = 0;
1210
+ if (generator) {
1211
+ count++;
1212
+ }
1213
+ if (template) {
1214
+ count++;
1215
+ }
1216
+ if (templateFile) {
1217
+ count++;
1218
+ }
1219
+ return count === 1 || "Expect one of { generator, template, templateFile }";
1220
+ });
1221
+ var standardTsconfig = {
1222
+ check: async function expectStandardTsconfig(context, opts) {
1223
+ const tsconfigFileName = opts.file ?? DEFAULT_TSCONFIG_FILENAME;
1224
+ const fullPath = path5.resolve(context.packageDir, tsconfigFileName);
1225
+ const generator = getGenerator(context, opts);
1226
+ const expectedContent = await generator(context);
1227
+ const actualContent = context.host.exists(fullPath) ? context.host.readFile(fullPath, { encoding: "utf-8" }) : void 0;
1228
+ if (expectedContent === void 0) {
1229
+ context.addWarning({
1230
+ file: fullPath,
1231
+ message: "Excluding from expect-standard-tsconfig"
1232
+ });
1233
+ return;
1234
+ }
1235
+ if (actualContent !== expectedContent) {
1236
+ context.addError({
1237
+ file: fullPath,
1238
+ message: "Expect file contents to match",
1239
+ longMessage: diff7(expectedContent, actualContent, { expand: true }),
1240
+ fixer: () => {
1241
+ context.host.writeFile(fullPath, expectedContent, {
1242
+ encoding: "utf-8"
1243
+ });
1244
+ }
1245
+ });
1246
+ }
1247
+ },
1248
+ optionsRuntype: Options11
1249
+ };
1250
+ var StandardTsConfig = createNewRuleConversion("StandardTsconfig", standardTsconfig);
1251
+ function getGenerator(context, opts) {
1252
+ if (opts.generator) {
1253
+ return opts.generator;
1254
+ } else if (opts.templateFile) {
1255
+ const { packageDir: workspacePackageDir } = context.getWorkspaceContext();
1256
+ const fullPath = path5.resolve(workspacePackageDir, opts.templateFile);
1257
+ const template = JSON.parse(context.host.readFile(fullPath, { encoding: "utf-8" }));
1258
+ return makeGenerator(template, opts.excludedReferences, opts.additionalReferences, opts.tsconfigReferenceFile);
1259
+ } else if (opts.template) {
1260
+ return makeGenerator(opts.template, opts.excludedReferences, opts.additionalReferences, opts.tsconfigReferenceFile);
1261
+ } else {
1262
+ throw new Error("Unable to make generator");
1263
+ }
1264
+ }
1265
+ function makeGenerator(template, excludedReferences, additionalReferences, tsconfigReferenceFile) {
1266
+ return async function generator(context) {
1267
+ template = {
1268
+ ...template,
1269
+ references: []
1270
+ };
1271
+ const nameToDirectory = await context.getWorkspaceContext().getPackageNameToDir();
1272
+ const packageJson = context.getPackageJson();
1273
+ const deps = [...Object.keys(packageJson.dependencies || {}), ...Object.keys(packageJson.devDependencies || {})];
1274
+ for (const dep of deps) {
1275
+ const packageDir = nameToDirectory.get(dep);
1276
+ if (packageDir !== void 0 && (excludedReferences === void 0 || matchesAnyGlob2(dep, excludedReferences))) {
1277
+ const absoluteReferencePath = tsconfigReferenceFile !== void 0 ? path5.join(packageDir, tsconfigReferenceFile) : packageDir;
1278
+ template.references.push({
1279
+ path: path5.relative(context.packageDir, absoluteReferencePath)
1280
+ });
1281
+ }
1282
+ }
1283
+ if (additionalReferences) {
1284
+ for (const additionalReference of additionalReferences) {
1285
+ template.references.push({
1286
+ path: additionalReference
1287
+ });
1288
+ }
1289
+ }
1290
+ return JSON.stringify(template, void 0, 2) + "\n";
1291
+ };
1292
+ }
1293
+
1294
+ // src/nestedWorkspaces.ts
1295
+ import * as globby from "globby";
1296
+ import * as path6 from "path";
1297
+ import * as r12 from "runtypes";
1298
+ var Options12 = r12.Undefined;
1299
+ var nestedWorkspaces = {
1300
+ check: (context) => {
1301
+ const rootPackageJson = context.getWorkspaceContext().getPackageJson();
1302
+ const packageJsonPaths = globby.globbySync(["*/**/package.json", "!**/node_modules/**"]);
1303
+ const workspaces = Array.isArray(rootPackageJson.workspaces) ? rootPackageJson.workspaces : rootPackageJson.workspaces !== void 0 ? rootPackageJson.workspaces.packages : void 0;
1304
+ if (workspaces === void 0 && packageJsonPaths.length > 0) {
1305
+ context.addError({
1306
+ file: context.getPackageJsonPath(),
1307
+ message: 'The "workspace" field is missing, even though there are workspaces in the repository.'
1308
+ });
1309
+ return;
1310
+ }
1311
+ const workspacePackageJsons = (workspaces || []).map((item) => `${item}/package.json`);
1312
+ const expandedWorkspacesGlobs = globby.globbySync([...workspacePackageJsons, "!**/node_modules/**"]);
1313
+ const difference = packageJsonPaths.filter((packageJsonPath) => !expandedWorkspacesGlobs.includes(packageJsonPath));
1314
+ if (difference.length !== 0) {
1315
+ const differencesList = difference.map((packageJsonPath) => path6.dirname(packageJsonPath)).join(", ");
1316
+ context.addError({
1317
+ file: context.getPackageJsonPath(),
1318
+ message: `The "workspace" field is missing one or more values: ${differencesList}. You may be able to use a glob to avoid listing each workspace individually, e.g. "packages/nested-workspace/*".`
1319
+ });
1320
+ }
1321
+ },
1322
+ optionsRuntype: Options12
1323
+ };
1324
+ var NestedWorkspaces = createNewRuleConversion("NestedWorkspaces", nestedWorkspaces);
1325
+
1326
+ // src/requireDependency.ts
1327
+ import { mutateJson as mutateJson5 } from "@monorepolint/utils";
1328
+ import { diff as diff8 } from "jest-diff";
1329
+ import * as r13 from "runtypes";
1330
+ var Options13 = r13.Partial({
1331
+ dependencies: r13.Dictionary(r13.String),
1332
+ devDependencies: r13.Dictionary(r13.String),
1333
+ peerDependencies: r13.Dictionary(r13.String),
1334
+ optionalDependencies: r13.Dictionary(r13.String)
1335
+ });
1336
+ var requireDependency = {
1337
+ check: function expectPackageEntry2(context, options) {
1338
+ const packageJson = context.getPackageJson();
1339
+ const packageJsonPath = context.getPackageJsonPath();
1340
+ [
1341
+ "dependencies",
1342
+ "devDependencies",
1343
+ "peerDependencies",
1344
+ "optionalDependencies"
1345
+ ].forEach((type) => {
1346
+ if (!options[type]) {
1347
+ return;
1348
+ }
1349
+ if (packageJson[type] === void 0) {
1350
+ context.addError({
1351
+ file: packageJsonPath,
1352
+ message: `No ${type} block, cannot add required ${type}.`,
1353
+ fixer: () => {
1354
+ mutateJson5(packageJsonPath, context.host, (input) => {
1355
+ input[type] = options[type];
1356
+ return input;
1357
+ });
1358
+ }
1359
+ });
1360
+ return;
1361
+ }
1362
+ for (const [dep, version] of Object.entries(options[type])) {
1363
+ if (packageJson[type][dep] !== version) {
1364
+ context.addError({
1365
+ file: packageJsonPath,
1366
+ message: `Expected dependency ${dep}@${version}`,
1367
+ longMessage: diff8(`${dep}@${version}
1368
+ `, `${dep}@${packageJson[type][dep] || "missing"}
1369
+ `),
1370
+ fixer: () => {
1371
+ mutateJson5(packageJsonPath, context.host, (input) => {
1372
+ input[type] = { ...input[type], [dep]: version };
1373
+ return input;
1374
+ });
1375
+ }
1376
+ });
1377
+ }
1378
+ }
1379
+ });
1380
+ },
1381
+ optionsRuntype: Options13
1382
+ };
1383
+ var RequireDependency = createNewRuleConversion("RequireDependency", requireDependency);
1384
+ export {
1385
+ AlphabeticalDependencies,
1386
+ AlphabeticalScripts,
1387
+ BannedDependencies,
1388
+ ConsistentDependencies,
1389
+ ConsistentVersions,
1390
+ FileContents,
1391
+ MustSatisfyPeerDependencies,
1392
+ NestedWorkspaces,
1393
+ PackageEntry,
1394
+ PackageOrder,
1395
+ PackageScript,
1396
+ RequireDependency,
1397
+ StandardTsConfig,
1398
+ alphabeticalDependencies,
1399
+ alphabeticalScripts,
1400
+ bannedDependencies,
1401
+ consistentDependencies,
1402
+ consistentVersions,
1403
+ fileContents,
1404
+ mustSatisfyPeerDependencies,
1405
+ nestedWorkspaces,
1406
+ packageEntry,
1407
+ packageOrder,
1408
+ packageScript,
1409
+ requireDependency,
1410
+ standardTsconfig
1411
+ };
1412
+ /*!
1413
+ * Copyright 2019 Palantir Technologies, Inc.
1414
+ *
1415
+ * Licensed under the MIT license. See LICENSE file in the project root for details.
1416
+ *
1417
+ */
1418
+ /*!
1419
+ * Copyright 2020 Palantir Technologies, Inc.
1420
+ *
1421
+ * Licensed under the MIT license. See LICENSE file in the project root for details.
1422
+ *
1423
+ */
1424
+ /*!
1425
+ * Copyright 2022 Palantir Technologies, Inc.
1426
+ *
1427
+ * Licensed under the MIT license. See LICENSE file in the project root for details.
1428
+ *
1429
+ */
1430
+ /**
1431
+ * @license Copyright 2019 Palantir Technologies, Inc. All rights reserved.
1432
+ */
1433
+ //# sourceMappingURL=index.js.map