eslint-plugin-package-json 0.81.0 → 0.83.0
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/CHANGELOG.md +14 -2
- package/README.md +2 -0
- package/lib/plugin.mjs +16 -14
- package/lib/rules/require-properties.mjs +1 -0
- package/lib/rules/specify-peers-locally.d.mts +6 -0
- package/lib/rules/specify-peers-locally.mjs +65 -0
- package/package.json +1 -1
package/CHANGELOG.md
CHANGED
|
@@ -1,11 +1,23 @@
|
|
|
1
1
|
# Changelog
|
|
2
2
|
|
|
3
|
-
# [0.
|
|
3
|
+
# [0.83.0](https://github.com/JoshuaKGoldberg/eslint-plugin-package-json/compare/v0.82.0...v0.83.0) (2025-11-17)
|
|
4
|
+
|
|
5
|
+
|
|
6
|
+
### Features
|
|
7
|
+
|
|
8
|
+
* **specify-peers-locally:** add rule to require all peers are declared as dev dependencies ([#1407](https://github.com/JoshuaKGoldberg/eslint-plugin-package-json/issues/1407)) ([5a102ec](https://github.com/JoshuaKGoldberg/eslint-plugin-package-json/commit/5a102ecd0f6fb025f42e838492844e2fa32ba786)), closes [#1053](https://github.com/JoshuaKGoldberg/eslint-plugin-package-json/issues/1053)
|
|
4
9
|
|
|
10
|
+
# [0.82.0](https://github.com/JoshuaKGoldberg/eslint-plugin-package-json/compare/v0.81.0...v0.82.0) (2025-11-17)
|
|
11
|
+
|
|
12
|
+
### Features
|
|
13
|
+
|
|
14
|
+
- **require-sideEffects:** add new rule for requiring `sideEffects` ([#1406](https://github.com/JoshuaKGoldberg/eslint-plugin-package-json/issues/1406)) ([0a8388d](https://github.com/JoshuaKGoldberg/eslint-plugin-package-json/commit/0a8388d6be21e6d4d7c0203fd65be96751ca5f29)), closes [#1405](https://github.com/JoshuaKGoldberg/eslint-plugin-package-json/issues/1405)
|
|
15
|
+
|
|
16
|
+
# [0.81.0](https://github.com/JoshuaKGoldberg/eslint-plugin-package-json/compare/v0.80.0...v0.81.0) (2025-11-17)
|
|
5
17
|
|
|
6
18
|
### Features
|
|
7
19
|
|
|
8
|
-
|
|
20
|
+
- **require-attribution:** add new rule to require proper attribution ([#1408](https://github.com/JoshuaKGoldberg/eslint-plugin-package-json/issues/1408)) ([e646f94](https://github.com/JoshuaKGoldberg/eslint-plugin-package-json/commit/e646f941bfc8ac2e4cb7c28509a8bd800b912442)), closes [#1404](https://github.com/JoshuaKGoldberg/eslint-plugin-package-json/issues/1404)
|
|
9
21
|
|
|
10
22
|
# [0.80.0](https://github.com/JoshuaKGoldberg/eslint-plugin-package-json/compare/v0.79.0...v0.80.0) (2025-11-17)
|
|
11
23
|
|
package/README.md
CHANGED
|
@@ -205,6 +205,7 @@ The default settings don't conflict, and Prettier plugins can quickly fix up ord
|
|
|
205
205
|
| [require-name](docs/rules/require-name.md) | Requires the `name` property to be present. | ✔️ ✅ | | | |
|
|
206
206
|
| [require-optionalDependencies](docs/rules/require-optionalDependencies.md) | Requires the `optionalDependencies` property to be present. | | | | |
|
|
207
207
|
| [require-peerDependencies](docs/rules/require-peerDependencies.md) | Requires the `peerDependencies` property to be present. | | | | |
|
|
208
|
+
| [require-sideEffects](docs/rules/require-sideEffects.md) | Requires the `sideEffects` property to be present. | | | | |
|
|
208
209
|
| [require-type](docs/rules/require-type.md) | Requires the `type` property to be present. | ✔️ ✅ | | | |
|
|
209
210
|
| [require-types](docs/rules/require-types.md) | Requires the `types` property to be present. | | | | |
|
|
210
211
|
| [require-version](docs/rules/require-version.md) | Requires the `version` property to be present. | ✔️ ✅ | | | |
|
|
@@ -212,6 +213,7 @@ The default settings don't conflict, and Prettier plugins can quickly fix up ord
|
|
|
212
213
|
| [restrict-private-properties](docs/rules/restrict-private-properties.md) | Disallows unnecessary properties in private packages. | | 🔧 | 💡 | |
|
|
213
214
|
| [scripts-name-casing](docs/rules/scripts-name-casing.md) | Enforce that names for `scripts` are in kebab case (optionally separated by colons). | 🎨 | | 💡 | |
|
|
214
215
|
| [sort-collections](docs/rules/sort-collections.md) | Selected collections must be in a consistent order (lexicographical for most; lifecycle-aware for scripts). | ✔️ ✅ | 🔧 | | |
|
|
216
|
+
| [specify-peers-locally](docs/rules/specify-peers-locally.md) | Requires that all peer dependencies are also declared as dev dependencies | ✔️ ✅ | | 💡 | |
|
|
215
217
|
| [unique-dependencies](docs/rules/unique-dependencies.md) | Checks a dependency isn't specified more than once (i.e. in `dependencies` and `devDependencies`) | ✔️ ✅ | | 💡 | |
|
|
216
218
|
| [valid-author](docs/rules/valid-author.md) | Enforce that the `author` property is valid. | ✔️ ✅ | | | |
|
|
217
219
|
| [valid-bin](docs/rules/valid-bin.md) | Enforce that the `bin` property is valid. | ✔️ ✅ | | | |
|
package/lib/plugin.mjs
CHANGED
|
@@ -11,13 +11,14 @@ import { rule as rule$8 } from "./rules/restrict-dependency-ranges.mjs";
|
|
|
11
11
|
import { rule as rule$9 } from "./rules/restrict-private-properties.mjs";
|
|
12
12
|
import { rule as rule$10 } from "./rules/scripts-name-casing.mjs";
|
|
13
13
|
import { rule as rule$11 } from "./rules/sort-collections.mjs";
|
|
14
|
-
import { rule as rule$12 } from "./rules/
|
|
15
|
-
import { rule as rule$13 } from "./rules/
|
|
16
|
-
import { rule as rule$14 } from "./rules/valid-
|
|
17
|
-
import { rule as rule$15 } from "./rules/valid-
|
|
14
|
+
import { rule as rule$12 } from "./rules/specify-peers-locally.mjs";
|
|
15
|
+
import { rule as rule$13 } from "./rules/unique-dependencies.mjs";
|
|
16
|
+
import { rule as rule$14 } from "./rules/valid-local-dependency.mjs";
|
|
17
|
+
import { rule as rule$15 } from "./rules/valid-name.mjs";
|
|
18
|
+
import { rule as rule$16 } from "./rules/valid-package-definition.mjs";
|
|
18
19
|
import { rules as rules$1 } from "./rules/valid-properties.mjs";
|
|
19
|
-
import { rule as rule$
|
|
20
|
-
import { rule as rule$
|
|
20
|
+
import { rule as rule$17 } from "./rules/valid-repository-directory.mjs";
|
|
21
|
+
import { rule as rule$18 } from "./rules/valid-version.mjs";
|
|
21
22
|
import { createRequire } from "node:module";
|
|
22
23
|
import * as parserJsonc from "jsonc-eslint-parser";
|
|
23
24
|
|
|
@@ -37,16 +38,17 @@ const rules$2 = {
|
|
|
37
38
|
"restrict-private-properties": rule$9,
|
|
38
39
|
"scripts-name-casing": rule$10,
|
|
39
40
|
"sort-collections": rule$11,
|
|
40
|
-
"
|
|
41
|
+
"specify-peers-locally": rule$12,
|
|
42
|
+
"unique-dependencies": rule$13,
|
|
41
43
|
...rules$1,
|
|
42
|
-
"valid-local-dependency": rule$
|
|
43
|
-
"valid-name": rule$
|
|
44
|
-
"valid-package-definition": rule$
|
|
45
|
-
"valid-repository-directory": rule$
|
|
46
|
-
"valid-version": rule$
|
|
44
|
+
"valid-local-dependency": rule$14,
|
|
45
|
+
"valid-name": rule$15,
|
|
46
|
+
"valid-package-definition": rule$16,
|
|
47
|
+
"valid-repository-directory": rule$17,
|
|
48
|
+
"valid-version": rule$18
|
|
47
49
|
};
|
|
48
|
-
const recommendedRules = { ...Object.fromEntries(Object.entries(rules$2).filter(([, rule$
|
|
49
|
-
const stylisticRules = { ...Object.fromEntries(Object.entries(rules$2).filter(([, rule$
|
|
50
|
+
const recommendedRules = { ...Object.fromEntries(Object.entries(rules$2).filter(([, rule$19]) => rule$19.meta.docs?.recommended).map(([name$1]) => ["package-json/" + name$1, "error"])) };
|
|
51
|
+
const stylisticRules = { ...Object.fromEntries(Object.entries(rules$2).filter(([, rule$19]) => rule$19.meta.docs?.category === "Stylistic").map(([name$1]) => ["package-json/" + name$1, "error"])) };
|
|
50
52
|
const plugin = {
|
|
51
53
|
configs: {
|
|
52
54
|
"legacy-recommended": {
|
|
@@ -0,0 +1,65 @@
|
|
|
1
|
+
import { createRule } from "../createRule.mjs";
|
|
2
|
+
import { isJSONStringLiteral } from "../utils/predicates.mjs";
|
|
3
|
+
|
|
4
|
+
//#region src/rules/specify-peers-locally.ts
|
|
5
|
+
const rule = createRule({
|
|
6
|
+
create(context) {
|
|
7
|
+
const devDependencyNames = /* @__PURE__ */ new Set();
|
|
8
|
+
let devDependenciesObjectNode;
|
|
9
|
+
const peerDependencyMap = {};
|
|
10
|
+
return {
|
|
11
|
+
"Program > JSONExpressionStatement > JSONObjectExpression > JSONProperty[key.type=JSONLiteral][value.type=JSONObjectExpression][key.value=devDependencies]"(node) {
|
|
12
|
+
devDependenciesObjectNode = node.value;
|
|
13
|
+
for (const devDependencyPropertyNode of node.value.properties) {
|
|
14
|
+
const dependencyKey = devDependencyPropertyNode.key;
|
|
15
|
+
if (isJSONStringLiteral(dependencyKey)) devDependencyNames.add(dependencyKey.value);
|
|
16
|
+
}
|
|
17
|
+
},
|
|
18
|
+
"Program > JSONExpressionStatement > JSONObjectExpression > JSONProperty[key.type=JSONLiteral][value.type=JSONObjectExpression][key.value=peerDependencies]"(node) {
|
|
19
|
+
for (const peerDependencyPropertyNode of node.value.properties) {
|
|
20
|
+
const dependencyKey = peerDependencyPropertyNode.key;
|
|
21
|
+
if (isJSONStringLiteral(dependencyKey)) peerDependencyMap[dependencyKey.value] = peerDependencyPropertyNode;
|
|
22
|
+
}
|
|
23
|
+
},
|
|
24
|
+
"Program:exit"() {
|
|
25
|
+
for (const [peerDependencyName, peerDependencyNode] of Object.entries(peerDependencyMap)) if (!devDependencyNames.has(peerDependencyName)) {
|
|
26
|
+
const peerDependencyValue = peerDependencyNode.value;
|
|
27
|
+
context.report({
|
|
28
|
+
data: { name: peerDependencyName },
|
|
29
|
+
messageId: "devDependencyNotDefined",
|
|
30
|
+
node: peerDependencyNode,
|
|
31
|
+
suggest: devDependenciesObjectNode && isJSONStringLiteral(peerDependencyValue) ? [{
|
|
32
|
+
data: { name: peerDependencyName },
|
|
33
|
+
fix: (fixer) => {
|
|
34
|
+
const updatedDevDependencies = {
|
|
35
|
+
...JSON.parse(context.sourceCode.getText(devDependenciesObjectNode)),
|
|
36
|
+
[peerDependencyName]: peerDependencyValue.value
|
|
37
|
+
};
|
|
38
|
+
const sortedDevDependencies = Object.fromEntries(Object.entries(updatedDevDependencies).sort((a, b) => a[0] > b[0] ? 1 : -1));
|
|
39
|
+
return fixer.replaceText(devDependenciesObjectNode, JSON.stringify(sortedDevDependencies, null, 2).split("\n").join("\n "));
|
|
40
|
+
},
|
|
41
|
+
messageId: "addToDevDependencies"
|
|
42
|
+
}] : []
|
|
43
|
+
});
|
|
44
|
+
}
|
|
45
|
+
}
|
|
46
|
+
};
|
|
47
|
+
},
|
|
48
|
+
meta: {
|
|
49
|
+
docs: {
|
|
50
|
+
description: "Requires that all peer dependencies are also declared as dev dependencies",
|
|
51
|
+
recommended: true
|
|
52
|
+
},
|
|
53
|
+
hasSuggestions: true,
|
|
54
|
+
messages: {
|
|
55
|
+
addToDevDependencies: "Add \"{{ name }}\" to `devDependencies`",
|
|
56
|
+
devDependencyNotDefined: "Peer dependency \"{{ name }}\" is not also declared in `devDependencies`."
|
|
57
|
+
},
|
|
58
|
+
schema: [],
|
|
59
|
+
type: "problem"
|
|
60
|
+
},
|
|
61
|
+
name: "specify-peers-locally"
|
|
62
|
+
});
|
|
63
|
+
|
|
64
|
+
//#endregion
|
|
65
|
+
export { rule };
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "eslint-plugin-package-json",
|
|
3
|
-
"version": "0.
|
|
3
|
+
"version": "0.83.0",
|
|
4
4
|
"description": "Rules for consistent, readable, and valid package.json files. 🗂️",
|
|
5
5
|
"homepage": "https://github.com/JoshuaKGoldberg/eslint-plugin-package-json#readme",
|
|
6
6
|
"bugs": {
|