@monorepolint/rules 0.5.0-alpha.1 → 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.
- package/.turbo/turbo-clean.log +4 -0
- package/.turbo/turbo-compile-typescript.log +4 -0
- package/.turbo/turbo-lint.log +107 -0
- package/.turbo/turbo-test.log +631 -0
- package/.turbo/turbo-transpile-typescript.log +18 -0
- package/build/js/index.cjs +1491 -0
- package/build/js/index.cjs.map +1 -0
- package/build/js/index.js +1433 -0
- package/build/js/index.js.map +1 -0
- package/build/tsconfig.tsbuildinfo +1 -0
- package/{lib/__tests__/utils.d.ts → build/types/__tests__/alphabeticalScripts.spec.d.ts} +2 -2
- package/build/types/__tests__/alphabeticalScripts.spec.d.ts.map +1 -0
- package/build/types/__tests__/bannedDependencies.spec.d.ts +2 -0
- package/build/types/__tests__/bannedDependencies.spec.d.ts.map +1 -0
- package/build/types/__tests__/consistentDependencies.spec.d.ts +2 -0
- package/build/types/__tests__/consistentDependencies.spec.d.ts.map +1 -0
- package/build/types/__tests__/consistentVersions.spec.d.ts +8 -0
- package/build/types/__tests__/consistentVersions.spec.d.ts.map +1 -0
- package/build/types/__tests__/fileContents.spec.d.ts +8 -0
- package/build/types/__tests__/fileContents.spec.d.ts.map +1 -0
- package/build/types/__tests__/mustSatisfyPeerDependencies.spec.d.ts +8 -0
- package/build/types/__tests__/mustSatisfyPeerDependencies.spec.d.ts.map +1 -0
- package/build/types/__tests__/nestedWorkspaces.spec.d.ts +2 -0
- package/build/types/__tests__/nestedWorkspaces.spec.d.ts.map +1 -0
- package/build/types/__tests__/packageEntry.spec.d.ts +8 -0
- package/build/types/__tests__/packageEntry.spec.d.ts.map +1 -0
- package/{lib → build/types}/__tests__/packageOrder.spec.d.ts +0 -0
- package/build/types/__tests__/packageOrder.spec.d.ts.map +1 -0
- package/{lib → build/types}/__tests__/packageScript.spec.d.ts +0 -0
- package/build/types/__tests__/packageScript.spec.d.ts.map +1 -0
- package/build/types/__tests__/requireDependency.spec.d.ts +2 -0
- package/build/types/__tests__/requireDependency.spec.d.ts.map +1 -0
- package/build/types/__tests__/utils.d.ts +81 -0
- package/build/types/__tests__/utils.d.ts.map +1 -0
- package/build/types/alphabeticalDependencies.d.ts +23 -0
- package/build/types/alphabeticalDependencies.d.ts.map +1 -0
- package/build/types/alphabeticalScripts.d.ts +23 -0
- package/build/types/alphabeticalScripts.d.ts.map +1 -0
- package/build/types/bannedDependencies.d.ts +134 -0
- package/build/types/bannedDependencies.d.ts.map +1 -0
- package/build/types/consistentDependencies.d.ts +38 -0
- package/build/types/consistentDependencies.d.ts.map +1 -0
- package/build/types/consistentVersions.d.ts +47 -0
- package/build/types/consistentVersions.d.ts.map +1 -0
- package/build/types/fileContents.d.ts +111 -0
- package/build/types/fileContents.d.ts.map +1 -0
- package/build/types/index.d.ts +20 -0
- package/build/types/index.d.ts.map +1 -0
- package/build/types/mustSatisfyPeerDependencies.d.ts +721 -0
- package/build/types/mustSatisfyPeerDependencies.d.ts.map +1 -0
- package/build/types/nestedWorkspaces.d.ts +24 -0
- package/build/types/nestedWorkspaces.d.ts.map +1 -0
- package/build/types/packageEntry.d.ts +115 -0
- package/build/types/packageEntry.d.ts.map +1 -0
- package/build/types/packageOrder.d.ts +33 -0
- package/build/types/packageOrder.d.ts.map +1 -0
- package/build/types/packageScript.d.ts +89 -0
- package/build/types/packageScript.d.ts.map +1 -0
- package/build/types/requireDependency.d.ts +99 -0
- package/build/types/requireDependency.d.ts.map +1 -0
- package/build/types/standardTsconfig.d.ts +113 -0
- package/build/types/standardTsconfig.d.ts.map +1 -0
- package/build/types/util/checkAlpha.d.ts +10 -0
- package/build/types/util/checkAlpha.d.ts.map +1 -0
- package/build/types/util/createNewRuleConversion.d.ts +30 -0
- package/build/types/util/createNewRuleConversion.d.ts.map +1 -0
- package/build/types/util/makeDirectory.d.ts +8 -0
- package/build/types/util/makeDirectory.d.ts.map +1 -0
- package/build/types/util/packageDependencyGraphService.d.ts +37 -0
- package/build/types/util/packageDependencyGraphService.d.ts.map +1 -0
- package/{jest.config.js → jest.config.cjs} +0 -0
- package/package.json +39 -18
- package/src/__tests__/alphabeticalScripts.spec.ts +76 -0
- package/src/__tests__/bannedDependencies.spec.ts +191 -0
- package/src/__tests__/consistentDependencies.spec.ts +142 -0
- package/src/__tests__/consistentVersions.spec.ts +224 -0
- package/src/__tests__/fileContents.spec.ts +75 -0
- package/src/__tests__/mustSatisfyPeerDependencies.spec.ts +1189 -0
- package/src/__tests__/nestedWorkspaces.spec.ts +153 -0
- package/src/__tests__/packageEntry.spec.ts +201 -0
- package/src/__tests__/packageOrder.spec.ts +48 -40
- package/src/__tests__/packageScript.spec.ts +149 -65
- package/src/__tests__/requireDependency.spec.ts +152 -0
- package/src/__tests__/utils.ts +115 -11
- package/src/alphabeticalDependencies.ts +6 -48
- package/src/alphabeticalScripts.ts +21 -0
- package/src/bannedDependencies.ts +135 -44
- package/src/consistentDependencies.ts +38 -14
- package/src/consistentVersions.ts +142 -0
- package/src/fileContents.ts +35 -30
- package/src/index.ts +13 -8
- package/src/mustSatisfyPeerDependencies.ts +748 -0
- package/src/nestedWorkspaces.ts +61 -0
- package/src/packageEntry.ts +72 -23
- package/src/packageOrder.ts +15 -10
- package/src/packageScript.ts +60 -13
- package/src/requireDependency.ts +71 -0
- package/src/standardTsconfig.ts +50 -21
- package/src/util/checkAlpha.ts +59 -0
- package/src/util/createNewRuleConversion.ts +38 -0
- package/src/util/makeDirectory.ts +24 -0
- package/src/util/packageDependencyGraphService.ts +114 -0
- package/tsconfig.json +10 -2
- package/lib/__tests__/packageOrder.spec.d.ts.map +0 -1
- package/lib/__tests__/packageOrder.spec.js +0 -114
- package/lib/__tests__/packageOrder.spec.js.map +0 -1
- package/lib/__tests__/packageScript.spec.d.ts.map +0 -1
- package/lib/__tests__/packageScript.spec.js +0 -108
- package/lib/__tests__/packageScript.spec.js.map +0 -1
- package/lib/__tests__/utils.d.ts.map +0 -1
- package/lib/__tests__/utils.js +0 -23
- package/lib/__tests__/utils.js.map +0 -1
- package/lib/alphabeticalDependencies.d.ts +0 -10
- package/lib/alphabeticalDependencies.d.ts.map +0 -1
- package/lib/alphabeticalDependencies.js +0 -56
- package/lib/alphabeticalDependencies.js.map +0 -1
- package/lib/bannedDependencies.d.ts +0 -15
- package/lib/bannedDependencies.d.ts.map +0 -1
- package/lib/bannedDependencies.js +0 -57
- package/lib/bannedDependencies.js.map +0 -1
- package/lib/consistentDependencies.d.ts +0 -10
- package/lib/consistentDependencies.d.ts.map +0 -1
- package/lib/consistentDependencies.js +0 -57
- package/lib/consistentDependencies.js.map +0 -1
- package/lib/fileContents.d.ts +0 -25
- package/lib/fileContents.d.ts.map +0 -1
- package/lib/fileContents.js +0 -77
- package/lib/fileContents.js.map +0 -1
- package/lib/index.d.ts +0 -15
- package/lib/index.d.ts.map +0 -1
- package/lib/index.js +0 -25
- package/lib/index.js.map +0 -1
- package/lib/packageEntry.d.ts +0 -16
- package/lib/packageEntry.d.ts.map +0 -1
- package/lib/packageEntry.js +0 -38
- package/lib/packageEntry.js.map +0 -1
- package/lib/packageOrder.d.ts +0 -12
- package/lib/packageOrder.d.ts.map +0 -1
- package/lib/packageOrder.js +0 -102
- package/lib/packageOrder.js.map +0 -1
- package/lib/packageScript.d.ts +0 -17
- package/lib/packageScript.d.ts.map +0 -1
- package/lib/packageScript.js +0 -51
- package/lib/packageScript.js.map +0 -1
- package/lib/standardTsconfig.d.ts +0 -29
- package/lib/standardTsconfig.d.ts.map +0 -1
- package/lib/standardTsconfig.js +0 -96
- package/lib/standardTsconfig.js.map +0 -1
- package/tsconfig.tsbuildinfo +0 -2181
|
@@ -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
|