@monorepolint/rules 0.5.0-alpha.7 → 0.5.0-alpha.70

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 (131) hide show
  1. package/lib/__tests__/alphabeticalScripts.spec.d.ts +8 -0
  2. package/lib/__tests__/alphabeticalScripts.spec.d.ts.map +1 -0
  3. package/lib/__tests__/alphabeticalScripts.spec.js +63 -0
  4. package/lib/__tests__/alphabeticalScripts.spec.js.map +1 -0
  5. package/lib/__tests__/bannedDependencies.spec.d.ts +2 -0
  6. package/lib/__tests__/bannedDependencies.spec.d.ts.map +1 -0
  7. package/lib/__tests__/bannedDependencies.spec.js +103 -0
  8. package/lib/__tests__/bannedDependencies.spec.js.map +1 -0
  9. package/lib/__tests__/consistentDependencies.spec.js +26 -11
  10. package/lib/__tests__/consistentDependencies.spec.js.map +1 -1
  11. package/lib/__tests__/consistentVersions.spec.d.ts +8 -0
  12. package/lib/__tests__/consistentVersions.spec.d.ts.map +1 -0
  13. package/lib/__tests__/consistentVersions.spec.js +182 -0
  14. package/lib/__tests__/consistentVersions.spec.js.map +1 -0
  15. package/lib/__tests__/fileContents.spec.d.ts +8 -0
  16. package/lib/__tests__/fileContents.spec.d.ts.map +1 -0
  17. package/lib/__tests__/fileContents.spec.js +66 -0
  18. package/lib/__tests__/fileContents.spec.js.map +1 -0
  19. package/lib/__tests__/mustSatisfyPeerDependencies.spec.d.ts +8 -0
  20. package/lib/__tests__/mustSatisfyPeerDependencies.spec.d.ts.map +1 -0
  21. package/lib/__tests__/mustSatisfyPeerDependencies.spec.js +1062 -0
  22. package/lib/__tests__/mustSatisfyPeerDependencies.spec.js.map +1 -0
  23. package/lib/__tests__/nestedWorkspaces.spec.d.ts +2 -0
  24. package/lib/__tests__/nestedWorkspaces.spec.d.ts.map +1 -0
  25. package/lib/__tests__/nestedWorkspaces.spec.js +122 -0
  26. package/lib/__tests__/nestedWorkspaces.spec.js.map +1 -0
  27. package/lib/__tests__/packageEntry.spec.d.ts +8 -0
  28. package/lib/__tests__/packageEntry.spec.d.ts.map +1 -0
  29. package/lib/__tests__/packageEntry.spec.js +137 -0
  30. package/lib/__tests__/packageEntry.spec.js.map +1 -0
  31. package/lib/__tests__/requireDependency.spec.d.ts +2 -0
  32. package/lib/__tests__/requireDependency.spec.d.ts.map +1 -0
  33. package/lib/__tests__/requireDependency.spec.js +122 -0
  34. package/lib/__tests__/requireDependency.spec.js.map +1 -0
  35. package/lib/__tests__/utils.d.ts +1 -0
  36. package/lib/__tests__/utils.d.ts.map +1 -1
  37. package/lib/__tests__/utils.js +11 -0
  38. package/lib/__tests__/utils.js.map +1 -1
  39. package/lib/alphabeticalDependencies.d.ts +3 -1
  40. package/lib/alphabeticalDependencies.d.ts.map +1 -1
  41. package/lib/alphabeticalDependencies.js +5 -39
  42. package/lib/alphabeticalDependencies.js.map +1 -1
  43. package/lib/alphabeticalScripts.d.ts +12 -0
  44. package/lib/alphabeticalScripts.d.ts.map +1 -0
  45. package/lib/alphabeticalScripts.js +20 -0
  46. package/lib/alphabeticalScripts.js.map +1 -0
  47. package/lib/bannedDependencies.d.ts +12 -3
  48. package/lib/bannedDependencies.d.ts.map +1 -1
  49. package/lib/bannedDependencies.js +59 -14
  50. package/lib/bannedDependencies.js.map +1 -1
  51. package/lib/consistentDependencies.d.ts +8 -1
  52. package/lib/consistentDependencies.d.ts.map +1 -1
  53. package/lib/consistentDependencies.js +28 -9
  54. package/lib/consistentDependencies.js.map +1 -1
  55. package/lib/consistentVersions.d.ts +14 -0
  56. package/lib/consistentVersions.d.ts.map +1 -0
  57. package/lib/consistentVersions.js +94 -0
  58. package/lib/consistentVersions.js.map +1 -0
  59. package/lib/fileContents.d.ts.map +1 -1
  60. package/lib/fileContents.js +6 -2
  61. package/lib/fileContents.js.map +1 -1
  62. package/lib/index.d.ts +5 -0
  63. package/lib/index.d.ts.map +1 -1
  64. package/lib/index.js +18 -8
  65. package/lib/index.js.map +1 -1
  66. package/lib/mustSatisfyPeerDependencies.d.ts +240 -0
  67. package/lib/mustSatisfyPeerDependencies.d.ts.map +1 -0
  68. package/lib/mustSatisfyPeerDependencies.js +634 -0
  69. package/lib/mustSatisfyPeerDependencies.js.map +1 -0
  70. package/lib/nestedWorkspaces.d.ts +13 -0
  71. package/lib/nestedWorkspaces.d.ts.map +1 -0
  72. package/lib/nestedWorkspaces.js +50 -0
  73. package/lib/nestedWorkspaces.js.map +1 -0
  74. package/lib/packageEntry.d.ts +22 -4
  75. package/lib/packageEntry.d.ts.map +1 -1
  76. package/lib/packageEntry.js +45 -16
  77. package/lib/packageEntry.js.map +1 -1
  78. package/lib/packageOrder.d.ts.map +1 -1
  79. package/lib/packageOrder.js +6 -4
  80. package/lib/packageOrder.js.map +1 -1
  81. package/lib/packageScript.js +4 -3
  82. package/lib/packageScript.js.map +1 -1
  83. package/lib/requireDependency.d.ts +15 -0
  84. package/lib/requireDependency.d.ts.map +1 -0
  85. package/lib/requireDependency.js +65 -0
  86. package/lib/requireDependency.js.map +1 -0
  87. package/lib/standardTsconfig.d.ts +12 -0
  88. package/lib/standardTsconfig.d.ts.map +1 -1
  89. package/lib/standardTsconfig.js +17 -7
  90. package/lib/standardTsconfig.js.map +1 -1
  91. package/lib/util/checkAlpha.d.ts +9 -0
  92. package/lib/util/checkAlpha.d.ts.map +1 -0
  93. package/lib/util/checkAlpha.js +48 -0
  94. package/lib/util/checkAlpha.js.map +1 -0
  95. package/lib/util/makeDirectory.d.ts +8 -0
  96. package/lib/util/makeDirectory.d.ts.map +1 -0
  97. package/lib/util/makeDirectory.js +28 -0
  98. package/lib/util/makeDirectory.js.map +1 -0
  99. package/lib/util/packageDependencyGraphService.d.ts +37 -0
  100. package/lib/util/packageDependencyGraphService.d.ts.map +1 -0
  101. package/lib/util/packageDependencyGraphService.js +70 -0
  102. package/lib/util/packageDependencyGraphService.js.map +1 -0
  103. package/package.json +17 -13
  104. package/src/__tests__/alphabeticalScripts.spec.ts +76 -0
  105. package/src/__tests__/bannedDependencies.spec.ts +117 -0
  106. package/src/__tests__/consistentDependencies.spec.ts +29 -10
  107. package/src/__tests__/consistentVersions.spec.ts +218 -0
  108. package/src/__tests__/fileContents.spec.ts +82 -0
  109. package/src/__tests__/mustSatisfyPeerDependencies.spec.ts +1183 -0
  110. package/src/__tests__/nestedWorkspaces.spec.ts +146 -0
  111. package/src/__tests__/packageEntry.spec.ts +181 -0
  112. package/src/__tests__/requireDependency.spec.ts +146 -0
  113. package/src/__tests__/utils.ts +12 -0
  114. package/src/alphabeticalDependencies.ts +3 -47
  115. package/src/alphabeticalScripts.ts +19 -0
  116. package/src/bannedDependencies.ts +74 -16
  117. package/src/consistentDependencies.ts +33 -10
  118. package/src/consistentVersions.ts +140 -0
  119. package/src/fileContents.ts +5 -3
  120. package/src/index.ts +5 -0
  121. package/src/mustSatisfyPeerDependencies.ts +735 -0
  122. package/src/nestedWorkspaces.ts +59 -0
  123. package/src/packageEntry.ts +56 -17
  124. package/src/packageOrder.ts +5 -4
  125. package/src/packageScript.ts +3 -3
  126. package/src/requireDependency.ts +69 -0
  127. package/src/standardTsconfig.ts +19 -7
  128. package/src/util/checkAlpha.ts +56 -0
  129. package/src/util/makeDirectory.ts +24 -0
  130. package/src/util/packageDependencyGraphService.ts +111 -0
  131. package/tsconfig.tsbuildinfo +5399 -1591
@@ -10,19 +10,25 @@ import { writeJson } from "@monorepolint/utils";
10
10
  import diff from "jest-diff";
11
11
  import * as r from "runtypes";
12
12
 
13
- const Options = r.Undefined;
14
- type Options = r.Static<typeof Options>;
13
+ const Options = r
14
+ .Record({
15
+ ignoredDependencies: r.Array(r.String).Or(r.Undefined),
16
+ })
17
+ .Or(r.Undefined);
18
+ export type Options = r.Static<typeof Options>;
19
+
20
+ const skippedVersions = ["*", "latest"];
15
21
 
16
22
  export const consistentDependencies = {
17
- check: function expectConsistentDependencies(context: Context) {
18
- checkDeps(context, "dependencies");
19
- checkDeps(context, "devDependencies");
23
+ check: function expectConsistentDependencies(context: Context, args: Options) {
24
+ checkDeps(context, args, "dependencies");
25
+ checkDeps(context, args, "devDependencies");
20
26
  // we don't check peer deps since they can be more lenient
21
27
  },
22
28
  optionsRuntype: Options,
23
29
  } as RuleModule<typeof Options>;
24
30
 
25
- function checkDeps(context: Context, block: "dependencies" | "devDependencies" | "peerDependencies") {
31
+ function checkDeps(context: Context, args: Options, block: "dependencies" | "devDependencies" | "peerDependencies") {
26
32
  const packageJson = context.getPackageJson();
27
33
  const packagePath = context.getPackageJsonPath();
28
34
  const dependencies = packageJson[block];
@@ -30,19 +36,25 @@ function checkDeps(context: Context, block: "dependencies" | "devDependencies" |
30
36
  const workspacePackageJson = context.getWorkspaceContext().getPackageJson();
31
37
  const workspaceDependencies = workspacePackageJson[block];
32
38
 
33
- if (dependencies === undefined || workspaceDependencies === undefined) {
39
+ const ignoredDeps = args?.ignoredDependencies ?? [];
40
+ const depsToCheck =
41
+ workspaceDependencies == null || ignoredDeps.length === 0
42
+ ? workspaceDependencies
43
+ : omit(workspaceDependencies, ignoredDeps);
44
+
45
+ if (dependencies === undefined || depsToCheck === undefined) {
34
46
  return;
35
47
  }
36
48
 
37
49
  const expectedDependencies = {
38
50
  ...dependencies,
39
- ...filterKeys(workspaceDependencies, dependencies),
51
+ ...filterKeys(depsToCheck, dependencies),
40
52
  };
41
53
 
42
54
  if (JSON.stringify(dependencies) !== JSON.stringify(expectedDependencies)) {
43
55
  context.addError({
44
56
  file: packagePath,
45
- message: `Inconsitent ${block} with root in package.json`,
57
+ message: `Inconsistent ${block} with root in package.json`,
46
58
  longMessage: diff(expectedDependencies, dependencies, { expand: true }),
47
59
  fixer: () => {
48
60
  const newPackageJson = { ...packageJson };
@@ -57,10 +69,21 @@ function filterKeys(ob: Record<string, string>, filterOb: Record<string, string>
57
69
  const newOb: Record<string, any> = {};
58
70
 
59
71
  for (const key of Object.keys(filterOb)) {
60
- if (ob[key] !== undefined && filterOb[key] !== "*") {
72
+ if (ob[key] !== undefined && skippedVersions.indexOf(filterOb[key]) === -1) {
61
73
  newOb[key] = ob[key];
62
74
  }
63
75
  }
64
76
 
65
77
  return newOb;
66
78
  }
79
+
80
+ function omit<T>(obj: Record<string, T>, keysToOmit: readonly string[]): Record<string, T> {
81
+ const newObj: Record<string, T> = {};
82
+
83
+ const filtered = Object.entries(obj).filter(([key]) => !keysToOmit.includes(key));
84
+ for (const [key, value] of filtered) {
85
+ newObj[key] = value;
86
+ }
87
+
88
+ return newObj;
89
+ }
@@ -0,0 +1,140 @@
1
+ /*!
2
+ * Copyright 2020 Palantir Technologies, Inc.
3
+ *
4
+ * Licensed under the MIT license. See LICENSE file in the project root for details.
5
+ *
6
+ */
7
+
8
+ import { Context, RuleModule } from "@monorepolint/core";
9
+ import { mutateJson, PackageJson } from "@monorepolint/utils";
10
+ import * as r from "runtypes";
11
+ import { coerce, SemVer } from "semver";
12
+
13
+ export const Options = r.Record({
14
+ matchDependencyVersions: r.Dictionary(r.Union(r.String, r.Array(r.String))),
15
+ });
16
+
17
+ export type Options = r.Static<typeof Options>;
18
+
19
+ export const consistentVersions: RuleModule<typeof Options> = {
20
+ check: checkConsistentVersions,
21
+ optionsRuntype: Options,
22
+ };
23
+
24
+ function checkConsistentVersions(context: Context, options: Options) {
25
+ for (const [dependencyPackageName, expectedPackageDependencyValue] of Object.entries(
26
+ options.matchDependencyVersions
27
+ )) {
28
+ if (Array.isArray(expectedPackageDependencyValue)) {
29
+ ensurePackageMatchesSomeVersion(context, dependencyPackageName, expectedPackageDependencyValue);
30
+ } else {
31
+ ensurePackageIsCorrectVersion(context, dependencyPackageName, expectedPackageDependencyValue);
32
+ }
33
+ }
34
+ }
35
+
36
+ const ensurePackageIsCorrectVersion = (
37
+ context: Context,
38
+ dependencyPackageName: string,
39
+ expectedPackageDependencyValue: string
40
+ ) => {
41
+ const packageJson = context.getPackageJson();
42
+ const packageJsonPath = context.getPackageJsonPath();
43
+
44
+ const expectedPackageDependencyVersion = coerce(expectedPackageDependencyValue);
45
+ if (expectedPackageDependencyVersion == null) {
46
+ throw new Error(
47
+ `Malformed expected package dependency version defined in monorepolint configuration: ${dependencyPackageName} @ '${expectedPackageDependencyValue}'`
48
+ );
49
+ }
50
+
51
+ const actualPackageDependencyValue = packageJson.dependencies && packageJson.dependencies[dependencyPackageName];
52
+ const actualPackageDependencyVersion = coerce(actualPackageDependencyValue);
53
+ if (
54
+ actualPackageDependencyVersion != null &&
55
+ actualPackageDependencyVersion.raw !== expectedPackageDependencyVersion.raw
56
+ ) {
57
+ context.addError({
58
+ file: packageJsonPath,
59
+ message: `Expected dependency on ${dependencyPackageName} to match version defined in monorepolint configuration '${expectedPackageDependencyValue}', got '${actualPackageDependencyValue}' instead.`,
60
+ fixer: () =>
61
+ mutateJson<PackageJson>(packageJsonPath, (input) => {
62
+ input.dependencies![dependencyPackageName] = expectedPackageDependencyValue;
63
+ return input;
64
+ }),
65
+ });
66
+ }
67
+
68
+ const actualPackageDevDependencyValue =
69
+ packageJson.devDependencies && packageJson.devDependencies[dependencyPackageName];
70
+ const actualPackageDevDependencyVersion = coerce(actualPackageDevDependencyValue);
71
+ if (
72
+ actualPackageDevDependencyVersion != null &&
73
+ actualPackageDevDependencyVersion.raw !== expectedPackageDependencyVersion.raw
74
+ ) {
75
+ context.addError({
76
+ file: packageJsonPath,
77
+ message: `Expected devDependency on ${dependencyPackageName} to match version defined in monorepolint configuration '${expectedPackageDependencyValue}', got '${actualPackageDevDependencyValue}' instead`,
78
+ fixer: () =>
79
+ mutateJson<PackageJson>(packageJsonPath, (input) => {
80
+ input.devDependencies![dependencyPackageName] = expectedPackageDependencyValue;
81
+ return input;
82
+ }),
83
+ });
84
+ }
85
+ };
86
+
87
+ const ensurePackageMatchesSomeVersion = (
88
+ context: Context,
89
+ dependencyPackageName: string,
90
+ acceptedPackageDependencyValues: string[]
91
+ ) => {
92
+ const packageJson = context.getPackageJson();
93
+ const packageJsonPath = context.getPackageJsonPath();
94
+
95
+ const acceptedPackageDependencyVersions: SemVer[] = acceptedPackageDependencyValues.map(
96
+ (acceptedPackageDependencyValue) => {
97
+ const acceptedPackageDependencyVersion = coerce(acceptedPackageDependencyValue);
98
+ if (acceptedPackageDependencyVersion == null) {
99
+ throw new Error(
100
+ `Malformed accepted package dependency version defined in monorepolint configuration: ${dependencyPackageName} @ '${acceptedPackageDependencyValue}'`
101
+ );
102
+ }
103
+ return acceptedPackageDependencyVersion;
104
+ }
105
+ );
106
+
107
+ const actualPackageDependencyValue = packageJson.dependencies && packageJson.dependencies[dependencyPackageName];
108
+ const actualPackageDependencyVersion = coerce(actualPackageDependencyValue);
109
+ if (
110
+ actualPackageDependencyVersion != null &&
111
+ acceptedPackageDependencyVersions.every(
112
+ (acceptedPackageDependencyVersion) => actualPackageDependencyVersion.raw !== acceptedPackageDependencyVersion.raw
113
+ )
114
+ ) {
115
+ context.addError({
116
+ file: packageJsonPath,
117
+ message: `Expected dependency on ${dependencyPackageName} to match one of '${JSON.stringify(
118
+ acceptedPackageDependencyValues
119
+ )}', got '${actualPackageDependencyValue}' instead.`,
120
+ });
121
+ }
122
+
123
+ const actualPackageDevDependencyValue =
124
+ packageJson.devDependencies && packageJson.devDependencies[dependencyPackageName];
125
+ const actualPackageDevDependencyVersion = coerce(actualPackageDevDependencyValue);
126
+ if (
127
+ actualPackageDevDependencyVersion != null &&
128
+ acceptedPackageDependencyVersions.every(
129
+ (acceptedPackageDependencyVersion) =>
130
+ actualPackageDevDependencyVersion.raw !== acceptedPackageDependencyVersion.raw
131
+ )
132
+ ) {
133
+ context.addError({
134
+ file: packageJsonPath,
135
+ message: `Expected devDependency on ${dependencyPackageName} to match one of '${JSON.stringify(
136
+ acceptedPackageDependencyValues
137
+ )}', got '${actualPackageDevDependencyValue}' instead.`,
138
+ });
139
+ }
140
+ };
@@ -11,6 +11,7 @@ import { existsSync, readFileSync, unlinkSync, writeFileSync } from "fs";
11
11
  import diff from "jest-diff";
12
12
  import * as path from "path";
13
13
  import * as r from "runtypes";
14
+ import { makeDirectoryRecursively } from "./util/makeDirectory";
14
15
 
15
16
  const Options = r.Union(
16
17
  r.Record({
@@ -43,17 +44,18 @@ export const fileContents = {
43
44
  const generator = getGenerator(context, opts);
44
45
  const expectedContent = generator(context);
45
46
 
46
- const actualContent = existsSync(fullPath) ? readFileSync(fullPath, "utf-8") : undefined;
47
-
47
+ const pathExists = existsSync(fullPath);
48
+ const actualContent = pathExists ? readFileSync(fullPath, "utf-8") : undefined;
48
49
  if (actualContent !== expectedContent) {
49
50
  context.addError({
50
51
  file: fullPath,
51
52
  message: "Expect file contents to match",
52
53
  longMessage: diff(expectedContent, actualContent, { expand: true }),
53
54
  fixer: () => {
54
- if (expectedContent === undefined) {
55
+ if (expectedContent === undefined && pathExists) {
55
56
  unlinkSync(fullPath);
56
57
  } else {
58
+ makeDirectoryRecursively(path.dirname(fullPath));
57
59
  writeFileSync(fullPath, expectedContent);
58
60
  }
59
61
  },
package/src/index.ts CHANGED
@@ -6,10 +6,15 @@
6
6
  */
7
7
 
8
8
  export { alphabeticalDependencies } from "./alphabeticalDependencies";
9
+ export { alphabeticalScripts } from "./alphabeticalScripts";
9
10
  export { bannedDependencies } from "./bannedDependencies";
10
11
  export { consistentDependencies } from "./consistentDependencies";
12
+ export { consistentVersions } from "./consistentVersions";
11
13
  export { fileContents } from "./fileContents";
14
+ export { mustSatisfyPeerDependencies } from "./mustSatisfyPeerDependencies";
12
15
  export { packageOrder } from "./packageOrder";
13
16
  export { packageEntry } from "./packageEntry";
14
17
  export { packageScript } from "./packageScript";
15
18
  export { standardTsconfig } from "./standardTsconfig";
19
+ export { nestedWorkspaces } from "./nestedWorkspaces";
20
+ export { requireDependency } from "./requireDependency";