-
-
Notifications
You must be signed in to change notification settings - Fork 30
/
Copy pathmeta-property-ordering.js
109 lines (95 loc) · 2.98 KB
/
meta-property-ordering.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
/**
* @fileoverview Enforces the order of meta properties
*/
'use strict';
const { getKeyName, getRuleInfo } = require('../utils');
// ------------------------------------------------------------------------------
// Rule Definition
// ------------------------------------------------------------------------------
/** @type {import('eslint').Rule.RuleModule} */
module.exports = {
meta: {
type: 'suggestion',
docs: {
description: 'Enforce the order of meta properties',
category: 'Rules',
recommended: false,
url: 'https://github.com/eslint-community/eslint-plugin-eslint-plugin/tree/HEAD/docs/rules/meta-property-ordering.md',
},
fixable: 'code',
schema: [
{
type: 'array',
description: 'What order to enforce for meta properties.',
elements: { type: 'string' },
},
],
messages: {
inconsistentOrder:
'The meta properties should be placed in a consistent order: [{{order}}].',
},
},
create(context) {
const sourceCode = context.sourceCode || context.getSourceCode(); // TODO: just use context.sourceCode when dropping eslint < v9
const ruleInfo = getRuleInfo(sourceCode);
if (!ruleInfo) {
return {};
}
const order = context.options[0] || [
'type',
'docs',
'fixable',
'hasSuggestions',
'deprecated',
'replacedBy',
'schema',
'defaultOptions', // https://github.com/eslint/rfcs/tree/main/designs/2023-rule-options-defaults
'messages',
];
const orderMap = new Map(order.map((name, i) => [name, i]));
return {
Program() {
if (!ruleInfo.meta || ruleInfo.meta.properties.length < 2) {
return;
}
const props = ruleInfo.meta.properties;
let last;
const violatingProps = props.filter((prop) => {
const curr = orderMap.has(getKeyName(prop))
? orderMap.get(getKeyName(prop))
: Number.POSITIVE_INFINITY;
return last > (last = curr);
});
if (violatingProps.length === 0) {
return;
}
const knownProps = props
.filter((prop) => orderMap.has(getKeyName(prop)))
.sort(
(a, b) => orderMap.get(getKeyName(a)) - orderMap.get(getKeyName(b)),
);
const unknownProps = props.filter(
(prop) => !orderMap.has(getKeyName(prop)),
);
for (const violatingProp of violatingProps) {
context.report({
node: violatingProp,
messageId: 'inconsistentOrder',
data: {
order: knownProps.map(getKeyName).join(', '),
},
fix(fixer) {
const expectedProps = [...knownProps, ...unknownProps];
return props.map((prop, k) => {
return fixer.replaceText(
prop,
sourceCode.getText(expectedProps[k]),
);
});
},
});
}
},
};
},
};