@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.
@@ -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()
@@ -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, `i18n`));
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
@@ -1,2 +1,3 @@
1
1
  import DiscourseRecommended from "./eslint.mjs";
2
+
2
3
  export default DiscourseRecommended;
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 noSimpleQueryselector from "./eslint-rules/no-simple-queryselector.mjs";
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-queryselector": noSimpleQueryselector,
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-duplicate-imports": "error",
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-queryselector": ["error"],
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.9.0",
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
- "@stylistic/eslint-plugin-js": "^4.2.0",
37
- "ember-template-lint": "^7.0.0",
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.0",
54
- "eslint": "9.21.0",
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
  }