@discourse/lint-configs 2.9.0 → 2.11.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/eslint-rules/deprecated-lookups.mjs +1 -2
- package/eslint-rules/discourse-common-imports.mjs +1 -2
- package/eslint-rules/i18n-import-location.mjs +4 -5
- package/eslint-rules/i18n-t.mjs +2 -3
- package/eslint-rules/line-after-imports.mjs +91 -0
- package/eslint-rules/lines-between-class-members.mjs +171 -0
- package/eslint-rules/{no-simple-queryselector.mjs → no-simple-query-selector.mjs} +1 -3
- package/eslint-rules/service-inject-import.mjs +1 -2
- package/eslint-rules/utils/tokens.mjs +128 -0
- package/eslint.config.mjs +1 -0
- package/eslint.mjs +12 -17
- package/package.json +6 -6
|
@@ -18,12 +18,11 @@ export default {
|
|
|
18
18
|
docs: {
|
|
19
19
|
description:
|
|
20
20
|
"replace deprecated resolver 'lookup' calls and modifyClass arguments with modern equivalents",
|
|
21
|
-
category: "Best Practices",
|
|
22
|
-
recommended: true,
|
|
23
21
|
},
|
|
24
22
|
fixable: "code",
|
|
25
23
|
schema: [], // no options
|
|
26
24
|
},
|
|
25
|
+
|
|
27
26
|
create(context) {
|
|
28
27
|
return {
|
|
29
28
|
CallExpression(node) {
|
|
@@ -4,12 +4,11 @@ export default {
|
|
|
4
4
|
docs: {
|
|
5
5
|
description:
|
|
6
6
|
"disallow imports from 'discourse-common' and replace with modern equivalents",
|
|
7
|
-
category: "Best Practices",
|
|
8
|
-
recommended: false,
|
|
9
7
|
},
|
|
10
8
|
fixable: "code",
|
|
11
9
|
schema: [], // no options
|
|
12
10
|
},
|
|
11
|
+
|
|
13
12
|
create(context) {
|
|
14
13
|
return {
|
|
15
14
|
ImportDeclaration(node) {
|
|
@@ -6,12 +6,11 @@ export default {
|
|
|
6
6
|
docs: {
|
|
7
7
|
description:
|
|
8
8
|
"disallow imports from 'i18n' and replace with 'discourse-i18n'",
|
|
9
|
-
category: "Best Practices",
|
|
10
|
-
recommended: false,
|
|
11
9
|
},
|
|
12
10
|
fixable: "code",
|
|
13
11
|
schema: [], // no options
|
|
14
12
|
},
|
|
13
|
+
|
|
15
14
|
create(context) {
|
|
16
15
|
return {
|
|
17
16
|
ImportDeclaration(node) {
|
|
@@ -27,12 +26,12 @@ export default {
|
|
|
27
26
|
}
|
|
28
27
|
|
|
29
28
|
if (
|
|
30
|
-
node.source.value.toLowerCase() === "discourse-common/helpers/i18n"
|
|
29
|
+
node.source.value.toLowerCase() === "discourse-common/helpers/i18n" ||
|
|
30
|
+
node.source.value.toLowerCase() === "discourse/helpers/i18n"
|
|
31
31
|
) {
|
|
32
32
|
context.report({
|
|
33
33
|
node,
|
|
34
|
-
message:
|
|
35
|
-
"Import from 'discourse-common/helpers/i18n' is not allowed. Use 'discourse-i18n' instead.",
|
|
34
|
+
message: `Import from '${node.source.value}' is not allowed. Use 'discourse-i18n' instead.`,
|
|
36
35
|
fix(fixer) {
|
|
37
36
|
const existingImport = context
|
|
38
37
|
.getSourceCode()
|
package/eslint-rules/i18n-t.mjs
CHANGED
|
@@ -5,12 +5,11 @@ export default {
|
|
|
5
5
|
type: "suggestion",
|
|
6
6
|
docs: {
|
|
7
7
|
description: "Use i18n(...) instead of 'I18n.t(...)'.",
|
|
8
|
-
category: "Best Practices",
|
|
9
|
-
recommended: false,
|
|
10
8
|
},
|
|
11
9
|
fixable: "code",
|
|
12
10
|
schema: [], // no options
|
|
13
11
|
},
|
|
12
|
+
|
|
14
13
|
create(context) {
|
|
15
14
|
const sourceCode = context.sourceCode ?? context.getSourceCode();
|
|
16
15
|
let alreadyFixedImport = false;
|
|
@@ -53,7 +52,7 @@ export default {
|
|
|
53
52
|
const fixes = [];
|
|
54
53
|
|
|
55
54
|
// Replace I18n.t with i18n
|
|
56
|
-
fixes.push(fixer.replaceText(node,
|
|
55
|
+
fixes.push(fixer.replaceText(node, "i18n"));
|
|
57
56
|
|
|
58
57
|
if (!alreadyFixedImport) {
|
|
59
58
|
const importDeclaration = i18nDefaultImport.node.parent;
|
|
@@ -0,0 +1,91 @@
|
|
|
1
|
+
import {
|
|
2
|
+
findFirstConsecutiveTokenBefore,
|
|
3
|
+
findLastConsecutiveTokenAfter,
|
|
4
|
+
getBoundaryTokens,
|
|
5
|
+
hasTokenOrCommentBetween,
|
|
6
|
+
} from "./utils/tokens.mjs";
|
|
7
|
+
|
|
8
|
+
function findLastIndexOfType(nodes, type) {
|
|
9
|
+
return nodes.findLastIndex((node) => node.type === type);
|
|
10
|
+
}
|
|
11
|
+
|
|
12
|
+
export default {
|
|
13
|
+
meta: {
|
|
14
|
+
type: "layout",
|
|
15
|
+
docs: {
|
|
16
|
+
description: "Require an empty line after the imports block",
|
|
17
|
+
},
|
|
18
|
+
fixable: "whitespace",
|
|
19
|
+
schema: [], // no options
|
|
20
|
+
},
|
|
21
|
+
|
|
22
|
+
create(context) {
|
|
23
|
+
const sourceCode = context.sourceCode;
|
|
24
|
+
|
|
25
|
+
return {
|
|
26
|
+
Program(node) {
|
|
27
|
+
const body = node.body;
|
|
28
|
+
const index = findLastIndexOfType(body, "ImportDeclaration");
|
|
29
|
+
|
|
30
|
+
if (index === -1) {
|
|
31
|
+
// No imports
|
|
32
|
+
return;
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
if (!body[index + 1]) {
|
|
36
|
+
// Nothing after imports
|
|
37
|
+
return;
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
const { curLast, nextFirst } = getBoundaryTokens(
|
|
41
|
+
sourceCode,
|
|
42
|
+
body[index],
|
|
43
|
+
body[index + 1]
|
|
44
|
+
);
|
|
45
|
+
|
|
46
|
+
const beforePadding = findLastConsecutiveTokenAfter(
|
|
47
|
+
sourceCode,
|
|
48
|
+
curLast,
|
|
49
|
+
nextFirst,
|
|
50
|
+
1
|
|
51
|
+
);
|
|
52
|
+
const afterPadding = findFirstConsecutiveTokenBefore(
|
|
53
|
+
sourceCode,
|
|
54
|
+
nextFirst,
|
|
55
|
+
curLast,
|
|
56
|
+
1
|
|
57
|
+
);
|
|
58
|
+
const isPadded =
|
|
59
|
+
afterPadding.loc.start.line - beforePadding.loc.end.line > 1;
|
|
60
|
+
const hasTokenInPadding = hasTokenOrCommentBetween(
|
|
61
|
+
sourceCode,
|
|
62
|
+
beforePadding,
|
|
63
|
+
afterPadding
|
|
64
|
+
);
|
|
65
|
+
const curLineLastToken = findLastConsecutiveTokenAfter(
|
|
66
|
+
sourceCode,
|
|
67
|
+
curLast,
|
|
68
|
+
nextFirst,
|
|
69
|
+
0
|
|
70
|
+
);
|
|
71
|
+
|
|
72
|
+
if (isPadded) {
|
|
73
|
+
return;
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
context.report({
|
|
77
|
+
node: body[index],
|
|
78
|
+
message: "Expected blank line after imports.",
|
|
79
|
+
|
|
80
|
+
fix(fixer) {
|
|
81
|
+
if (hasTokenInPadding) {
|
|
82
|
+
return null;
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
return fixer.insertTextAfter(curLineLastToken, "\n");
|
|
86
|
+
},
|
|
87
|
+
});
|
|
88
|
+
},
|
|
89
|
+
};
|
|
90
|
+
},
|
|
91
|
+
};
|
|
@@ -0,0 +1,171 @@
|
|
|
1
|
+
// Based on `@stylistic/js/lines-between-class-members`
|
|
2
|
+
// See: https://github.com/eslint-stylistic/eslint-stylistic/blob/d6809c910510a4477e01ea248071f0701d0af4ed/packages/eslint-plugin/rules/lines-between-class-members/lines-between-class-members._js_.ts
|
|
3
|
+
|
|
4
|
+
import {
|
|
5
|
+
findFirstConsecutiveTokenBefore,
|
|
6
|
+
findLastConsecutiveTokenAfter,
|
|
7
|
+
getBoundaryTokens,
|
|
8
|
+
hasTokenOrCommentBetween,
|
|
9
|
+
isTokenOnSameLine,
|
|
10
|
+
} from "./utils/tokens.mjs";
|
|
11
|
+
|
|
12
|
+
export default {
|
|
13
|
+
meta: {
|
|
14
|
+
type: "layout",
|
|
15
|
+
docs: {
|
|
16
|
+
description: "Require an empty line between class members",
|
|
17
|
+
},
|
|
18
|
+
fixable: "whitespace",
|
|
19
|
+
schema: [], // no options
|
|
20
|
+
messages: {
|
|
21
|
+
never: "Unexpected blank line between class members.",
|
|
22
|
+
always: "Expected blank line between class members.",
|
|
23
|
+
},
|
|
24
|
+
},
|
|
25
|
+
|
|
26
|
+
create(context) {
|
|
27
|
+
const configureList = [
|
|
28
|
+
{ blankLine: "always", prev: "service", next: "*" },
|
|
29
|
+
{ blankLine: "always", prev: "*", next: "method" },
|
|
30
|
+
{ blankLine: "always", prev: "method", next: "*" },
|
|
31
|
+
{ blankLine: "always", prev: "*", next: "template" },
|
|
32
|
+
];
|
|
33
|
+
const sourceCode = context.sourceCode;
|
|
34
|
+
|
|
35
|
+
/**
|
|
36
|
+
* Returns the type of the node.
|
|
37
|
+
* @param node The class member node to check.
|
|
38
|
+
* @returns The type string (see `configureList`)
|
|
39
|
+
* @private
|
|
40
|
+
*/
|
|
41
|
+
function nodeType(node) {
|
|
42
|
+
if (
|
|
43
|
+
node.type === "PropertyDefinition" &&
|
|
44
|
+
["service", "optionalService", "controller"].includes(
|
|
45
|
+
node.decorators?.[0]?.expression?.name ||
|
|
46
|
+
node.decorators?.[0]?.expression?.callee?.name
|
|
47
|
+
)
|
|
48
|
+
) {
|
|
49
|
+
return "service";
|
|
50
|
+
} else if (node.type === "PropertyDefinition") {
|
|
51
|
+
return "field";
|
|
52
|
+
} else if (node.type === "MethodDefinition") {
|
|
53
|
+
return "method";
|
|
54
|
+
} else if (node.type === "GlimmerTemplate") {
|
|
55
|
+
return "template";
|
|
56
|
+
} else {
|
|
57
|
+
return "other";
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
/**
|
|
62
|
+
* Checks whether the given node matches the given type.
|
|
63
|
+
* @param node The class member node to check.
|
|
64
|
+
* @param type The class member type to check.
|
|
65
|
+
* @returns `true` if the class member node matched the type.
|
|
66
|
+
* @private
|
|
67
|
+
*/
|
|
68
|
+
function match(node, type) {
|
|
69
|
+
if (type === "*") {
|
|
70
|
+
return true;
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
return nodeType(node) === type;
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
/**
|
|
77
|
+
* Finds the last matched configuration from the configureList.
|
|
78
|
+
* @param prevNode The previous node to match.
|
|
79
|
+
* @param nextNode The current node to match.
|
|
80
|
+
* @returns Padding type or `null` if no matches were found.
|
|
81
|
+
* @private
|
|
82
|
+
*/
|
|
83
|
+
function getPaddingType(prevNode, nextNode) {
|
|
84
|
+
for (let i = configureList.length - 1; i >= 0; --i) {
|
|
85
|
+
const configure = configureList[i];
|
|
86
|
+
const matched =
|
|
87
|
+
match(prevNode, configure.prev) && match(nextNode, configure.next);
|
|
88
|
+
|
|
89
|
+
if (matched) {
|
|
90
|
+
return configure.blankLine;
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
return null;
|
|
94
|
+
}
|
|
95
|
+
|
|
96
|
+
return {
|
|
97
|
+
ClassBody(node) {
|
|
98
|
+
const body = node.body;
|
|
99
|
+
|
|
100
|
+
for (let i = 0; i < body.length - 1; i++) {
|
|
101
|
+
const curFirst = sourceCode.getFirstToken(body[i]);
|
|
102
|
+
const { curLast, nextFirst } = getBoundaryTokens(
|
|
103
|
+
sourceCode,
|
|
104
|
+
body[i],
|
|
105
|
+
body[i + 1]
|
|
106
|
+
);
|
|
107
|
+
const singleLine = isTokenOnSameLine(curFirst, curLast);
|
|
108
|
+
const skip =
|
|
109
|
+
singleLine && nodeType(body[i]) === nodeType(body[i + 1]);
|
|
110
|
+
const beforePadding = findLastConsecutiveTokenAfter(
|
|
111
|
+
sourceCode,
|
|
112
|
+
curLast,
|
|
113
|
+
nextFirst,
|
|
114
|
+
1
|
|
115
|
+
);
|
|
116
|
+
const afterPadding = findFirstConsecutiveTokenBefore(
|
|
117
|
+
sourceCode,
|
|
118
|
+
nextFirst,
|
|
119
|
+
curLast,
|
|
120
|
+
1
|
|
121
|
+
);
|
|
122
|
+
const isPadded =
|
|
123
|
+
afterPadding.loc.start.line - beforePadding.loc.end.line > 1;
|
|
124
|
+
const hasTokenInPadding = hasTokenOrCommentBetween(
|
|
125
|
+
sourceCode,
|
|
126
|
+
beforePadding,
|
|
127
|
+
afterPadding
|
|
128
|
+
);
|
|
129
|
+
const curLineLastToken = findLastConsecutiveTokenAfter(
|
|
130
|
+
sourceCode,
|
|
131
|
+
curLast,
|
|
132
|
+
nextFirst,
|
|
133
|
+
0
|
|
134
|
+
);
|
|
135
|
+
const paddingType = getPaddingType(body[i], body[i + 1]);
|
|
136
|
+
|
|
137
|
+
if (paddingType === "never" && isPadded) {
|
|
138
|
+
context.report({
|
|
139
|
+
node: body[i + 1],
|
|
140
|
+
messageId: "never",
|
|
141
|
+
|
|
142
|
+
fix(fixer) {
|
|
143
|
+
if (hasTokenInPadding) {
|
|
144
|
+
return null;
|
|
145
|
+
}
|
|
146
|
+
|
|
147
|
+
return fixer.replaceTextRange(
|
|
148
|
+
[beforePadding.range[1], afterPadding.range[0]],
|
|
149
|
+
"\n"
|
|
150
|
+
);
|
|
151
|
+
},
|
|
152
|
+
});
|
|
153
|
+
} else if (paddingType === "always" && !skip && !isPadded) {
|
|
154
|
+
context.report({
|
|
155
|
+
node: body[i + 1],
|
|
156
|
+
messageId: "always",
|
|
157
|
+
|
|
158
|
+
fix(fixer) {
|
|
159
|
+
if (hasTokenInPadding) {
|
|
160
|
+
return null;
|
|
161
|
+
}
|
|
162
|
+
|
|
163
|
+
return fixer.insertTextAfter(curLineLastToken, "\n");
|
|
164
|
+
},
|
|
165
|
+
});
|
|
166
|
+
}
|
|
167
|
+
}
|
|
168
|
+
},
|
|
169
|
+
};
|
|
170
|
+
},
|
|
171
|
+
};
|
|
@@ -1,16 +1,14 @@
|
|
|
1
|
-
// no-queryselector-body-html.mjs
|
|
2
1
|
export default {
|
|
3
2
|
meta: {
|
|
4
3
|
type: "problem",
|
|
5
4
|
docs: {
|
|
6
5
|
description:
|
|
7
6
|
'disallow document.querySelector("body") and document.querySelector("html")',
|
|
8
|
-
category: "Best Practices",
|
|
9
|
-
recommended: false,
|
|
10
7
|
},
|
|
11
8
|
fixable: "code",
|
|
12
9
|
schema: [], // no options
|
|
13
10
|
},
|
|
11
|
+
|
|
14
12
|
create(context) {
|
|
15
13
|
return {
|
|
16
14
|
CallExpression(node) {
|
|
@@ -3,12 +3,11 @@ export default {
|
|
|
3
3
|
type: "suggestion",
|
|
4
4
|
docs: {
|
|
5
5
|
description: "Convert 'inject as service' to 'service'",
|
|
6
|
-
category: "Best Practices",
|
|
7
|
-
recommended: false,
|
|
8
6
|
},
|
|
9
7
|
fixable: "code",
|
|
10
8
|
schema: [], // no options
|
|
11
9
|
},
|
|
10
|
+
|
|
12
11
|
create(context) {
|
|
13
12
|
return {
|
|
14
13
|
ImportDeclaration(node) {
|
|
@@ -0,0 +1,128 @@
|
|
|
1
|
+
export function isTokenOnSameLine(left, right) {
|
|
2
|
+
return left?.loc?.end.line === right?.loc?.start.line;
|
|
3
|
+
}
|
|
4
|
+
|
|
5
|
+
export function isSemicolonToken(token) {
|
|
6
|
+
return token.value === ";" && token.type === "Punctuator";
|
|
7
|
+
}
|
|
8
|
+
|
|
9
|
+
/**
|
|
10
|
+
* Gets a pair of tokens that should be used to check lines between two class member nodes.
|
|
11
|
+
*
|
|
12
|
+
* In most cases, this returns the very last token of the current node and
|
|
13
|
+
* the very first token of the next node.
|
|
14
|
+
* For example:
|
|
15
|
+
*
|
|
16
|
+
* class C {
|
|
17
|
+
* x = 1; // curLast: `;` nextFirst: `in`
|
|
18
|
+
* in = 2
|
|
19
|
+
* }
|
|
20
|
+
*
|
|
21
|
+
* There is only one exception. If the given node ends with a semicolon, and it looks like
|
|
22
|
+
* a semicolon-less style's semicolon - one that is not on the same line as the preceding
|
|
23
|
+
* token, but is on the line where the next class member starts - this returns the preceding
|
|
24
|
+
* token and the semicolon as boundary tokens.
|
|
25
|
+
* For example:
|
|
26
|
+
*
|
|
27
|
+
* class C {
|
|
28
|
+
* x = 1 // curLast: `1` nextFirst: `;`
|
|
29
|
+
* ;in = 2
|
|
30
|
+
* }
|
|
31
|
+
* When determining the desired layout of the code, we should treat this semicolon as
|
|
32
|
+
* a part of the next class member node instead of the one it technically belongs to.
|
|
33
|
+
* @param curNode Current class member node.
|
|
34
|
+
* @param nextNode Next class member node.
|
|
35
|
+
* @returns The actual last token of `node`.
|
|
36
|
+
* @private
|
|
37
|
+
*/
|
|
38
|
+
export function getBoundaryTokens(sourceCode, curNode, nextNode) {
|
|
39
|
+
const lastToken = sourceCode.getLastToken(curNode);
|
|
40
|
+
const prevToken = sourceCode.getTokenBefore(lastToken);
|
|
41
|
+
const nextToken = sourceCode.getFirstToken(nextNode); // skip possible lone `;` between nodes
|
|
42
|
+
|
|
43
|
+
const isSemicolonLessStyle =
|
|
44
|
+
isSemicolonToken(lastToken) &&
|
|
45
|
+
!isTokenOnSameLine(prevToken, lastToken) &&
|
|
46
|
+
isTokenOnSameLine(lastToken, nextToken);
|
|
47
|
+
|
|
48
|
+
return isSemicolonLessStyle
|
|
49
|
+
? { curLast: prevToken, nextFirst: lastToken }
|
|
50
|
+
: { curLast: lastToken, nextFirst: nextToken };
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
/**
|
|
54
|
+
* Return the last token among the consecutive tokens that have no exceed max line difference in between, before the first token in the next member.
|
|
55
|
+
* @param prevLastToken The last token in the previous member node.
|
|
56
|
+
* @param nextFirstToken The first token in the next member node.
|
|
57
|
+
* @param maxLine The maximum number of allowed line difference between consecutive tokens.
|
|
58
|
+
* @returns The last token among the consecutive tokens.
|
|
59
|
+
*/
|
|
60
|
+
export function findLastConsecutiveTokenAfter(
|
|
61
|
+
sourceCode,
|
|
62
|
+
prevLastToken,
|
|
63
|
+
nextFirstToken,
|
|
64
|
+
maxLine
|
|
65
|
+
) {
|
|
66
|
+
const after = sourceCode.getTokenAfter(prevLastToken, {
|
|
67
|
+
includeComments: true,
|
|
68
|
+
});
|
|
69
|
+
|
|
70
|
+
if (
|
|
71
|
+
after !== nextFirstToken &&
|
|
72
|
+
after.loc.start.line - prevLastToken.loc.end.line <= maxLine
|
|
73
|
+
) {
|
|
74
|
+
return findLastConsecutiveTokenAfter(
|
|
75
|
+
sourceCode,
|
|
76
|
+
after,
|
|
77
|
+
nextFirstToken,
|
|
78
|
+
maxLine
|
|
79
|
+
);
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
return prevLastToken;
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
/**
|
|
86
|
+
* Return the first token among the consecutive tokens that have no exceed max line difference in between, after the last token in the previous member.
|
|
87
|
+
* @param nextFirstToken The first token in the next member node.
|
|
88
|
+
* @param prevLastToken The last token in the previous member node.
|
|
89
|
+
* @param maxLine The maximum number of allowed line difference between consecutive tokens.
|
|
90
|
+
* @returns The first token among the consecutive tokens.
|
|
91
|
+
*/
|
|
92
|
+
export function findFirstConsecutiveTokenBefore(
|
|
93
|
+
sourceCode,
|
|
94
|
+
nextFirstToken,
|
|
95
|
+
prevLastToken,
|
|
96
|
+
maxLine
|
|
97
|
+
) {
|
|
98
|
+
const before = sourceCode.getTokenBefore(nextFirstToken, {
|
|
99
|
+
includeComments: true,
|
|
100
|
+
});
|
|
101
|
+
|
|
102
|
+
if (
|
|
103
|
+
before !== prevLastToken &&
|
|
104
|
+
nextFirstToken.loc.start.line - before.loc.end.line <= maxLine
|
|
105
|
+
) {
|
|
106
|
+
return findFirstConsecutiveTokenBefore(
|
|
107
|
+
sourceCode,
|
|
108
|
+
before,
|
|
109
|
+
prevLastToken,
|
|
110
|
+
maxLine
|
|
111
|
+
);
|
|
112
|
+
}
|
|
113
|
+
|
|
114
|
+
return nextFirstToken;
|
|
115
|
+
}
|
|
116
|
+
|
|
117
|
+
/**
|
|
118
|
+
* Checks if there is a token or comment between two tokens.
|
|
119
|
+
* @param before The token before.
|
|
120
|
+
* @param after The token after.
|
|
121
|
+
* @returns True if there is a token or comment between two tokens.
|
|
122
|
+
*/
|
|
123
|
+
export function hasTokenOrCommentBetween(sourceCode, before, after) {
|
|
124
|
+
return (
|
|
125
|
+
sourceCode.getTokensBetween(before, after, { includeComments: true })
|
|
126
|
+
.length !== 0
|
|
127
|
+
);
|
|
128
|
+
}
|
package/eslint.config.mjs
CHANGED
package/eslint.mjs
CHANGED
|
@@ -2,11 +2,11 @@ import { createConfigItem } from "@babel/core";
|
|
|
2
2
|
import BabelParser from "@babel/eslint-parser";
|
|
3
3
|
import PluginProposalDecorators from "@babel/plugin-proposal-decorators";
|
|
4
4
|
import js from "@eslint/js";
|
|
5
|
-
import stylisticJs from "@stylistic/eslint-plugin-js";
|
|
6
5
|
import EmberESLintParser from "ember-eslint-parser";
|
|
7
6
|
import DecoratorPosition from "eslint-plugin-decorator-position";
|
|
8
7
|
import EmberPlugin from "eslint-plugin-ember";
|
|
9
8
|
import EmberRecommended from "eslint-plugin-ember/configs/recommended";
|
|
9
|
+
import ImportPlugin from "eslint-plugin-import";
|
|
10
10
|
import QUnitPlugin from "eslint-plugin-qunit";
|
|
11
11
|
import QUnitRecommended from "eslint-plugin-qunit/configs/recommended";
|
|
12
12
|
import SimpleImportSort from "eslint-plugin-simple-import-sort";
|
|
@@ -16,7 +16,9 @@ import deprecatedLookups from "./eslint-rules/deprecated-lookups.mjs";
|
|
|
16
16
|
import discourseCommonImports from "./eslint-rules/discourse-common-imports.mjs";
|
|
17
17
|
import i18nImport from "./eslint-rules/i18n-import-location.mjs";
|
|
18
18
|
import i18nT from "./eslint-rules/i18n-t.mjs";
|
|
19
|
-
import
|
|
19
|
+
import lineAfterImports from "./eslint-rules/line-after-imports.mjs";
|
|
20
|
+
import linesBetweenClassMembers from "./eslint-rules/lines-between-class-members.mjs";
|
|
21
|
+
import noSimpleQuerySelector from "./eslint-rules/no-simple-query-selector.mjs";
|
|
20
22
|
import serviceInjectImport from "./eslint-rules/service-inject-import.mjs";
|
|
21
23
|
|
|
22
24
|
// Copied from "ember-template-imports/lib/utils"
|
|
@@ -95,20 +97,22 @@ export default [
|
|
|
95
97
|
},
|
|
96
98
|
},
|
|
97
99
|
plugins: {
|
|
98
|
-
"@stylistic/js": stylisticJs,
|
|
99
100
|
ember: EmberPlugin,
|
|
100
101
|
"sort-class-members": SortClassMembers,
|
|
101
102
|
"decorator-position": DecoratorPosition,
|
|
102
103
|
"simple-import-sort": SimpleImportSort,
|
|
103
104
|
qunit: QUnitPlugin,
|
|
105
|
+
import: ImportPlugin,
|
|
104
106
|
discourse: {
|
|
105
107
|
rules: {
|
|
106
108
|
"i18n-import-location": i18nImport,
|
|
107
109
|
"i18n-t": i18nT,
|
|
108
110
|
"service-inject-import": serviceInjectImport,
|
|
109
|
-
"no-simple-
|
|
111
|
+
"no-simple-query-selector": noSimpleQuerySelector,
|
|
110
112
|
"deprecated-lookups": deprecatedLookups,
|
|
111
113
|
"discourse-common-imports": discourseCommonImports,
|
|
114
|
+
"lines-between-class-members": linesBetweenClassMembers,
|
|
115
|
+
"line-after-imports": lineAfterImports,
|
|
112
116
|
},
|
|
113
117
|
},
|
|
114
118
|
},
|
|
@@ -156,19 +160,9 @@ export default [
|
|
|
156
160
|
"valid-typeof": "error",
|
|
157
161
|
"wrap-iife": ["error", "inside"],
|
|
158
162
|
curly: "error",
|
|
159
|
-
"no-
|
|
163
|
+
"import/no-duplicates": "error",
|
|
160
164
|
"object-shorthand": ["error", "properties"],
|
|
161
165
|
"no-dupe-class-members": "error",
|
|
162
|
-
"@stylistic/js/lines-between-class-members": [
|
|
163
|
-
"error",
|
|
164
|
-
{
|
|
165
|
-
enforce: [
|
|
166
|
-
{ blankLine: "always", prev: "*", next: "method" },
|
|
167
|
-
{ blankLine: "always", prev: "method", next: "*" },
|
|
168
|
-
],
|
|
169
|
-
},
|
|
170
|
-
{ exceptAfterSingleLine: true },
|
|
171
|
-
],
|
|
172
166
|
"ember/no-classic-components": "off",
|
|
173
167
|
"ember/no-component-lifecycle-hooks": "off",
|
|
174
168
|
"ember/require-tagless-components": "off",
|
|
@@ -191,7 +185,6 @@ export default [
|
|
|
191
185
|
"ember/classic-decorator-hooks": "off",
|
|
192
186
|
"ember/classic-decorator-no-classic-methods": "off",
|
|
193
187
|
"ember/no-actions-hash": "off",
|
|
194
|
-
"ember/no-classic-classes": "off",
|
|
195
188
|
"ember/no-tracked-properties-from-args": "off",
|
|
196
189
|
"ember/no-jquery": "off",
|
|
197
190
|
"ember/no-runloop": "off",
|
|
@@ -287,9 +280,11 @@ export default [
|
|
|
287
280
|
"discourse/i18n-import-location": ["error"],
|
|
288
281
|
"discourse/i18n-t": ["error"],
|
|
289
282
|
"discourse/service-inject-import": ["error"],
|
|
290
|
-
"discourse/no-simple-
|
|
283
|
+
"discourse/no-simple-query-selector": ["error"],
|
|
291
284
|
"discourse/deprecated-lookups": ["error"],
|
|
292
285
|
"discourse/discourse-common-imports": ["error"],
|
|
286
|
+
"discourse/lines-between-class-members": ["error"],
|
|
287
|
+
"discourse/line-after-imports": ["error"],
|
|
293
288
|
},
|
|
294
289
|
},
|
|
295
290
|
{
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@discourse/lint-configs",
|
|
3
|
-
"version": "2.
|
|
3
|
+
"version": "2.11.0",
|
|
4
4
|
"description": "Shareable lint configs for Discourse core, plugins, and themes",
|
|
5
5
|
"author": "Discourse",
|
|
6
6
|
"license": "MIT",
|
|
@@ -33,11 +33,11 @@
|
|
|
33
33
|
"@babel/core": "^7.26.9",
|
|
34
34
|
"@babel/eslint-parser": "^7.26.8",
|
|
35
35
|
"@babel/plugin-proposal-decorators": "^7.25.9",
|
|
36
|
-
"
|
|
37
|
-
"
|
|
38
|
-
"eslint": "^9.21.0",
|
|
36
|
+
"ember-template-lint": "^7.0.1",
|
|
37
|
+
"eslint": "^9.22.0",
|
|
39
38
|
"eslint-plugin-decorator-position": "^6.0.0",
|
|
40
39
|
"eslint-plugin-ember": "^12.5.0",
|
|
40
|
+
"eslint-plugin-import": "^2.31.0",
|
|
41
41
|
"eslint-plugin-qunit": "^8.1.2",
|
|
42
42
|
"eslint-plugin-simple-import-sort": "^12.1.1",
|
|
43
43
|
"eslint-plugin-sort-class-members": "^1.21.0",
|
|
@@ -50,8 +50,8 @@
|
|
|
50
50
|
"typescript": "^5.8.2"
|
|
51
51
|
},
|
|
52
52
|
"peerDependencies": {
|
|
53
|
-
"ember-template-lint": "7.0.
|
|
54
|
-
"eslint": "9.
|
|
53
|
+
"ember-template-lint": "7.0.1",
|
|
54
|
+
"eslint": "9.22.0",
|
|
55
55
|
"prettier": "3.5.3",
|
|
56
56
|
"stylelint": "16.15.0"
|
|
57
57
|
}
|