|
1 | 1 | /**
|
2 |
| - * @fileoverview Rule to require JSON object keys to be sorted. Copied largely from https://github.com/eslint/eslint/blob/main/lib/rules/sort-keys.js |
| 2 | + * @fileoverview Rule to require JSON object keys to be sorted. |
| 3 | + * Copied largely from https://github.com/eslint/eslint/blob/main/lib/rules/sort-keys.js |
3 | 4 | * @author Robin Thomas
|
4 | 5 | */
|
5 | 6 |
|
| 7 | +//----------------------------------------------------------------------------- |
| 8 | +// Imports |
| 9 | +//----------------------------------------------------------------------------- |
| 10 | + |
6 | 11 | import naturalCompare from "natural-compare";
|
7 | 12 |
|
| 13 | +//----------------------------------------------------------------------------- |
| 14 | +// Type Definitions |
| 15 | +//----------------------------------------------------------------------------- |
| 16 | + |
| 17 | +/** @typedef {"sortKeys"} SortKeysMessageIds */ |
| 18 | + |
| 19 | +/** |
| 20 | + * @typedef {Object} SortOptions |
| 21 | + * @property {boolean} caseSensitive |
| 22 | + * @property {boolean} natural |
| 23 | + * @property {number} minKeys |
| 24 | + * @property {boolean} allowLineSeparatedGroups |
| 25 | + */ |
| 26 | + |
| 27 | +/** @typedef {"asc"|"desc"} SortDirection */ |
| 28 | +/** @typedef {[SortDirection, SortOptions]} SortKeysRuleOptions */ |
| 29 | +/** @typedef {import("../types.ts").JSONRuleDefinition<SortKeysRuleOptions, SortKeysMessageIds>} SortKeysRuleDefinition */ |
| 30 | +/** @typedef {(a:string,b:string) => boolean} Comparator */ |
| 31 | +/** @typedef {import("@humanwhocodes/momoa").MemberNode} MemberNode */ |
| 32 | + |
| 33 | +//----------------------------------------------------------------------------- |
| 34 | +// Helpers |
| 35 | +//----------------------------------------------------------------------------- |
| 36 | + |
8 | 37 | const hasNonWhitespace = /\S/u;
|
9 | 38 |
|
10 | 39 | const comparators = {
|
11 | 40 | ascending: {
|
12 | 41 | alphanumeric: {
|
| 42 | + /** @type {Comparator} */ |
13 | 43 | sensitive: (a, b) => a <= b,
|
| 44 | + |
| 45 | + /** @type {Comparator} */ |
14 | 46 | insensitive: (a, b) => a.toLowerCase() <= b.toLowerCase(),
|
15 | 47 | },
|
16 | 48 | natural: {
|
| 49 | + /** @type {Comparator} */ |
17 | 50 | sensitive: (a, b) => naturalCompare(a, b) <= 0,
|
| 51 | + |
| 52 | + /** @type {Comparator} */ |
18 | 53 | insensitive: (a, b) =>
|
19 | 54 | naturalCompare(a.toLowerCase(), b.toLowerCase()) <= 0,
|
20 | 55 | },
|
21 | 56 | },
|
22 | 57 | descending: {
|
23 | 58 | alphanumeric: {
|
| 59 | + /** @type {Comparator} */ |
24 | 60 | sensitive: (a, b) =>
|
25 | 61 | comparators.ascending.alphanumeric.sensitive(b, a),
|
| 62 | + |
| 63 | + /** @type {Comparator} */ |
26 | 64 | insensitive: (a, b) =>
|
27 | 65 | comparators.ascending.alphanumeric.insensitive(b, a),
|
28 | 66 | },
|
29 | 67 | natural: {
|
| 68 | + /** @type {Comparator} */ |
30 | 69 | sensitive: (a, b) => comparators.ascending.natural.sensitive(b, a),
|
| 70 | + |
| 71 | + /** @type {Comparator} */ |
31 | 72 | insensitive: (a, b) =>
|
32 | 73 | comparators.ascending.natural.insensitive(b, a),
|
33 | 74 | },
|
34 | 75 | },
|
35 | 76 | };
|
36 | 77 |
|
| 78 | +/** |
| 79 | + * Gets the MemberNode's string key value. |
| 80 | + * @param {MemberNode} member |
| 81 | + * @return {string} |
| 82 | + */ |
37 | 83 | function getKey(member) {
|
38 |
| - return member.name.type === `Identifier` |
| 84 | + return member.name.type === "Identifier" |
39 | 85 | ? member.name.name
|
40 | 86 | : member.name.value;
|
41 | 87 | }
|
42 | 88 |
|
43 |
| -export default { |
| 89 | +//----------------------------------------------------------------------------- |
| 90 | +// Rule Definition |
| 91 | +//----------------------------------------------------------------------------- |
| 92 | + |
| 93 | +/** @type {SortKeysRuleDefinition} */ |
| 94 | +const rule = { |
44 | 95 | meta: {
|
45 |
| - type: /** @type {const} */ ("suggestion"), |
| 96 | + type: "suggestion", |
46 | 97 |
|
47 | 98 | defaultOptions: [
|
48 | 99 | "asc",
|
@@ -112,8 +163,14 @@ export default {
|
112 | 163 | }
|
113 | 164 | }
|
114 | 165 |
|
115 |
| - // Note that there can be comments *inside* members, e.g. `{"foo: /* comment *\/ "bar"}`, but these are ignored when calculating line-separated groups |
| 166 | + /** |
| 167 | + * Checks if two members are line-separated. |
| 168 | + * @param {MemberNode} prevMember The previous member. |
| 169 | + * @param {MemberNode} member The current member. |
| 170 | + * @return {boolean} |
| 171 | + */ |
116 | 172 | function isLineSeparated(prevMember, member) {
|
| 173 | + // Note that there can be comments *inside* members, e.g. `{"foo: /* comment *\/ "bar"}`, but these are ignored when calculating line-separated groups |
117 | 174 | const prevMemberEndLine = prevMember.loc.end.line;
|
118 | 175 | const thisStartLine = member.loc.start.line;
|
119 | 176 | if (thisStartLine - prevMemberEndLine < 2) {
|
@@ -176,3 +233,5 @@ export default {
|
176 | 233 | };
|
177 | 234 | },
|
178 | 235 | };
|
| 236 | + |
| 237 | +export default rule; |
0 commit comments