@fookiejs/eslint-plugin 0.1.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.
Files changed (210) hide show
  1. package/README.md +512 -0
  2. package/dist/index.d.ts +164 -0
  3. package/dist/index.d.ts.map +1 -0
  4. package/dist/index.js +167 -0
  5. package/dist/index.js.map +1 -0
  6. package/dist/rules/consistent-return.d.ts +7 -0
  7. package/dist/rules/consistent-return.d.ts.map +1 -0
  8. package/dist/rules/consistent-return.js +56 -0
  9. package/dist/rules/consistent-return.js.map +1 -0
  10. package/dist/rules/exhaustive-switch.d.ts +5 -0
  11. package/dist/rules/exhaustive-switch.d.ts.map +1 -0
  12. package/dist/rules/exhaustive-switch.js +83 -0
  13. package/dist/rules/exhaustive-switch.js.map +1 -0
  14. package/dist/rules/index.d.ts +50 -0
  15. package/dist/rules/index.d.ts.map +1 -0
  16. package/dist/rules/index.js +50 -0
  17. package/dist/rules/index.js.map +1 -0
  18. package/dist/rules/no-any.d.ts +5 -0
  19. package/dist/rules/no-any.d.ts.map +1 -0
  20. package/dist/rules/no-any.js +20 -0
  21. package/dist/rules/no-any.js.map +1 -0
  22. package/dist/rules/no-array-mutating-methods.d.ts +5 -0
  23. package/dist/rules/no-array-mutating-methods.d.ts.map +1 -0
  24. package/dist/rules/no-array-mutating-methods.js +65 -0
  25. package/dist/rules/no-array-mutating-methods.js.map +1 -0
  26. package/dist/rules/no-async-without-await.d.ts +5 -0
  27. package/dist/rules/no-async-without-await.d.ts.map +1 -0
  28. package/dist/rules/no-async-without-await.js +45 -0
  29. package/dist/rules/no-async-without-await.js.map +1 -0
  30. package/dist/rules/no-catch-instanceof.d.ts +5 -0
  31. package/dist/rules/no-catch-instanceof.d.ts.map +1 -0
  32. package/dist/rules/no-catch-instanceof.js +55 -0
  33. package/dist/rules/no-catch-instanceof.js.map +1 -0
  34. package/dist/rules/no-catch-unknown.d.ts +5 -0
  35. package/dist/rules/no-catch-unknown.d.ts.map +1 -0
  36. package/dist/rules/no-catch-unknown.js +32 -0
  37. package/dist/rules/no-catch-unknown.js.map +1 -0
  38. package/dist/rules/no-class-mutation.d.ts +5 -0
  39. package/dist/rules/no-class-mutation.d.ts.map +1 -0
  40. package/dist/rules/no-class-mutation.js +44 -0
  41. package/dist/rules/no-class-mutation.js.map +1 -0
  42. package/dist/rules/no-comments.d.ts +5 -0
  43. package/dist/rules/no-comments.d.ts.map +1 -0
  44. package/dist/rules/no-comments.js +27 -0
  45. package/dist/rules/no-comments.js.map +1 -0
  46. package/dist/rules/no-default-export.d.ts +5 -0
  47. package/dist/rules/no-default-export.d.ts.map +1 -0
  48. package/dist/rules/no-default-export.js +22 -0
  49. package/dist/rules/no-default-export.js.map +1 -0
  50. package/dist/rules/no-define-property.d.ts +5 -0
  51. package/dist/rules/no-define-property.d.ts.map +1 -0
  52. package/dist/rules/no-define-property.js +37 -0
  53. package/dist/rules/no-define-property.js.map +1 -0
  54. package/dist/rules/no-delete.d.ts +5 -0
  55. package/dist/rules/no-delete.d.ts.map +1 -0
  56. package/dist/rules/no-delete.js +35 -0
  57. package/dist/rules/no-delete.js.map +1 -0
  58. package/dist/rules/no-empty-string.d.ts +5 -0
  59. package/dist/rules/no-empty-string.d.ts.map +1 -0
  60. package/dist/rules/no-empty-string.js +24 -0
  61. package/dist/rules/no-empty-string.js.map +1 -0
  62. package/dist/rules/no-eval.d.ts +5 -0
  63. package/dist/rules/no-eval.d.ts.map +1 -0
  64. package/dist/rules/no-eval.js +33 -0
  65. package/dist/rules/no-eval.js.map +1 -0
  66. package/dist/rules/no-floating-promise.d.ts +5 -0
  67. package/dist/rules/no-floating-promise.d.ts.map +1 -0
  68. package/dist/rules/no-floating-promise.js +74 -0
  69. package/dist/rules/no-floating-promise.js.map +1 -0
  70. package/dist/rules/no-for-in.d.ts +5 -0
  71. package/dist/rules/no-for-in.d.ts.map +1 -0
  72. package/dist/rules/no-for-in.js +24 -0
  73. package/dist/rules/no-for-in.js.map +1 -0
  74. package/dist/rules/no-generic-names.d.ts +12 -0
  75. package/dist/rules/no-generic-names.d.ts.map +1 -0
  76. package/dist/rules/no-generic-names.js +104 -0
  77. package/dist/rules/no-generic-names.js.map +1 -0
  78. package/dist/rules/no-implicit-coercion.d.ts +7 -0
  79. package/dist/rules/no-implicit-coercion.d.ts.map +1 -0
  80. package/dist/rules/no-implicit-coercion.js +33 -0
  81. package/dist/rules/no-implicit-coercion.js.map +1 -0
  82. package/dist/rules/no-legacy-globals.d.ts +7 -0
  83. package/dist/rules/no-legacy-globals.d.ts.map +1 -0
  84. package/dist/rules/no-legacy-globals.js +58 -0
  85. package/dist/rules/no-legacy-globals.js.map +1 -0
  86. package/dist/rules/no-loop-func.d.ts +5 -0
  87. package/dist/rules/no-loop-func.d.ts.map +1 -0
  88. package/dist/rules/no-loop-func.js +50 -0
  89. package/dist/rules/no-loop-func.js.map +1 -0
  90. package/dist/rules/no-map-set-mutation.d.ts +5 -0
  91. package/dist/rules/no-map-set-mutation.d.ts.map +1 -0
  92. package/dist/rules/no-map-set-mutation.js +69 -0
  93. package/dist/rules/no-map-set-mutation.js.map +1 -0
  94. package/dist/rules/no-mutable-exports.d.ts +5 -0
  95. package/dist/rules/no-mutable-exports.d.ts.map +1 -0
  96. package/dist/rules/no-mutable-exports.js +29 -0
  97. package/dist/rules/no-mutable-exports.js.map +1 -0
  98. package/dist/rules/no-nan-array-indexof.d.ts +5 -0
  99. package/dist/rules/no-nan-array-indexof.d.ts.map +1 -0
  100. package/dist/rules/no-nan-array-indexof.js +33 -0
  101. package/dist/rules/no-nan-array-indexof.js.map +1 -0
  102. package/dist/rules/no-nan-in-math-result.d.ts +5 -0
  103. package/dist/rules/no-nan-in-math-result.d.ts.map +1 -0
  104. package/dist/rules/no-nan-in-math-result.js +58 -0
  105. package/dist/rules/no-nan-in-math-result.js.map +1 -0
  106. package/dist/rules/no-nan.d.ts +5 -0
  107. package/dist/rules/no-nan.d.ts.map +1 -0
  108. package/dist/rules/no-nan.js +26 -0
  109. package/dist/rules/no-nan.js.map +1 -0
  110. package/dist/rules/no-new-wrappers.d.ts +5 -0
  111. package/dist/rules/no-new-wrappers.d.ts.map +1 -0
  112. package/dist/rules/no-new-wrappers.js +47 -0
  113. package/dist/rules/no-new-wrappers.js.map +1 -0
  114. package/dist/rules/no-new.d.ts +5 -0
  115. package/dist/rules/no-new.d.ts.map +1 -0
  116. package/dist/rules/no-new.js +111 -0
  117. package/dist/rules/no-new.js.map +1 -0
  118. package/dist/rules/no-non-null-assertion.d.ts +5 -0
  119. package/dist/rules/no-non-null-assertion.d.ts.map +1 -0
  120. package/dist/rules/no-non-null-assertion.js +22 -0
  121. package/dist/rules/no-non-null-assertion.js.map +1 -0
  122. package/dist/rules/no-null-undefined.d.ts +7 -0
  123. package/dist/rules/no-null-undefined.d.ts.map +1 -0
  124. package/dist/rules/no-null-undefined.js +34 -0
  125. package/dist/rules/no-null-undefined.js.map +1 -0
  126. package/dist/rules/no-nullish-operators.d.ts +7 -0
  127. package/dist/rules/no-nullish-operators.d.ts.map +1 -0
  128. package/dist/rules/no-nullish-operators.js +31 -0
  129. package/dist/rules/no-nullish-operators.js.map +1 -0
  130. package/dist/rules/no-object-assign-mutation.d.ts +5 -0
  131. package/dist/rules/no-object-assign-mutation.d.ts.map +1 -0
  132. package/dist/rules/no-object-assign-mutation.js +36 -0
  133. package/dist/rules/no-object-assign-mutation.js.map +1 -0
  134. package/dist/rules/no-param-reassign.d.ts +5 -0
  135. package/dist/rules/no-param-reassign.d.ts.map +1 -0
  136. package/dist/rules/no-param-reassign.js +89 -0
  137. package/dist/rules/no-param-reassign.js.map +1 -0
  138. package/dist/rules/no-parseint-nan.d.ts +5 -0
  139. package/dist/rules/no-parseint-nan.d.ts.map +1 -0
  140. package/dist/rules/no-parseint-nan.js +64 -0
  141. package/dist/rules/no-parseint-nan.js.map +1 -0
  142. package/dist/rules/no-placeholder-names.d.ts +12 -0
  143. package/dist/rules/no-placeholder-names.d.ts.map +1 -0
  144. package/dist/rules/no-placeholder-names.js +107 -0
  145. package/dist/rules/no-placeholder-names.js.map +1 -0
  146. package/dist/rules/no-process-env.d.ts +5 -0
  147. package/dist/rules/no-process-env.d.ts.map +1 -0
  148. package/dist/rules/no-process-env.js +31 -0
  149. package/dist/rules/no-process-env.js.map +1 -0
  150. package/dist/rules/no-prototype-mutation.d.ts +5 -0
  151. package/dist/rules/no-prototype-mutation.d.ts.map +1 -0
  152. package/dist/rules/no-prototype-mutation.js +30 -0
  153. package/dist/rules/no-prototype-mutation.js.map +1 -0
  154. package/dist/rules/no-require.d.ts +5 -0
  155. package/dist/rules/no-require.d.ts.map +1 -0
  156. package/dist/rules/no-require.js +37 -0
  157. package/dist/rules/no-require.js.map +1 -0
  158. package/dist/rules/no-shadow.d.ts +5 -0
  159. package/dist/rules/no-shadow.d.ts.map +1 -0
  160. package/dist/rules/no-shadow.js +103 -0
  161. package/dist/rules/no-shadow.js.map +1 -0
  162. package/dist/rules/no-string-concat.d.ts +5 -0
  163. package/dist/rules/no-string-concat.d.ts.map +1 -0
  164. package/dist/rules/no-string-concat.js +36 -0
  165. package/dist/rules/no-string-concat.js.map +1 -0
  166. package/dist/rules/no-throw-literal.d.ts +5 -0
  167. package/dist/rules/no-throw-literal.d.ts.map +1 -0
  168. package/dist/rules/no-throw-literal.js +33 -0
  169. package/dist/rules/no-throw-literal.js.map +1 -0
  170. package/dist/rules/no-type-assertion.d.ts +5 -0
  171. package/dist/rules/no-type-assertion.d.ts.map +1 -0
  172. package/dist/rules/no-type-assertion.js +25 -0
  173. package/dist/rules/no-type-assertion.js.map +1 -0
  174. package/dist/rules/no-typeof.d.ts +5 -0
  175. package/dist/rules/no-typeof.d.ts.map +1 -0
  176. package/dist/rules/no-typeof.js +24 -0
  177. package/dist/rules/no-typeof.js.map +1 -0
  178. package/dist/rules/no-unknown.d.ts +5 -0
  179. package/dist/rules/no-unknown.d.ts.map +1 -0
  180. package/dist/rules/no-unknown.js +22 -0
  181. package/dist/rules/no-unknown.js.map +1 -0
  182. package/dist/rules/prefer-includes.d.ts +5 -0
  183. package/dist/rules/prefer-includes.d.ts.map +1 -0
  184. package/dist/rules/prefer-includes.js +50 -0
  185. package/dist/rules/prefer-includes.js.map +1 -0
  186. package/dist/rules/prefer-readonly-params.d.ts +5 -0
  187. package/dist/rules/prefer-readonly-params.d.ts.map +1 -0
  188. package/dist/rules/prefer-readonly-params.js +100 -0
  189. package/dist/rules/prefer-readonly-params.js.map +1 -0
  190. package/dist/rules/require-boolean-condition.d.ts +5 -0
  191. package/dist/rules/require-boolean-condition.d.ts.map +1 -0
  192. package/dist/rules/require-boolean-condition.js +60 -0
  193. package/dist/rules/require-boolean-condition.js.map +1 -0
  194. package/dist/rules/require-curly.d.ts +5 -0
  195. package/dist/rules/require-curly.d.ts.map +1 -0
  196. package/dist/rules/require-curly.js +55 -0
  197. package/dist/rules/require-curly.js.map +1 -0
  198. package/dist/rules/require-explicit-return-type.d.ts +5 -0
  199. package/dist/rules/require-explicit-return-type.d.ts.map +1 -0
  200. package/dist/rules/require-explicit-return-type.js +66 -0
  201. package/dist/rules/require-explicit-return-type.js.map +1 -0
  202. package/dist/rules/require-private-constructor.d.ts +5 -0
  203. package/dist/rules/require-private-constructor.d.ts.map +1 -0
  204. package/dist/rules/require-private-constructor.js +34 -0
  205. package/dist/rules/require-private-constructor.js.map +1 -0
  206. package/dist/rules/same-type-comparison.d.ts +5 -0
  207. package/dist/rules/same-type-comparison.d.ts.map +1 -0
  208. package/dist/rules/same-type-comparison.js +71 -0
  209. package/dist/rules/same-type-comparison.js.map +1 -0
  210. package/package.json +46 -0
@@ -0,0 +1,50 @@
1
+ import { ESLintUtils, TSESTree } from "@typescript-eslint/utils";
2
+ const createRule = ESLintUtils.RuleCreator((name) => `https://github.com/fookiejs/eslint-plugin-fookie/blob/main/docs/rules/${name}.md`);
3
+ function isIndexOfCall(node) {
4
+ return (node.type === "CallExpression" &&
5
+ node.callee.type === "MemberExpression" &&
6
+ node.callee.property.type === "Identifier" &&
7
+ node.callee.property.name === "indexOf");
8
+ }
9
+ function isMinusOne(node) {
10
+ return (node.type === "UnaryExpression" &&
11
+ node.operator === "-" &&
12
+ node.argument.type === "Literal" &&
13
+ node.argument.value === 1);
14
+ }
15
+ function isZero(node) {
16
+ return node.type === "Literal" && node.value === 0;
17
+ }
18
+ export const preferIncludes = createRule({
19
+ name: "prefer-includes",
20
+ meta: {
21
+ type: "suggestion",
22
+ docs: {
23
+ description: "Require .includes() instead of .indexOf() comparisons",
24
+ },
25
+ schema: [],
26
+ messages: {
27
+ preferIncludes: "Use '.includes()' instead of '.indexOf()' comparison.",
28
+ },
29
+ },
30
+ defaultOptions: [],
31
+ create(context) {
32
+ return {
33
+ BinaryExpression(node) {
34
+ const { left, right, operator } = node;
35
+ const isComparisonWithIndexOf = (["!==", "!=", "===", "=="].includes(operator) &&
36
+ isIndexOfCall(left) &&
37
+ isMinusOne(right)) ||
38
+ (operator === ">=" && isIndexOfCall(left) && isZero(right)) ||
39
+ (operator === ">" && isIndexOfCall(left) && isMinusOne(right)) ||
40
+ (["!==", "!=", "===", "=="].includes(operator) &&
41
+ isIndexOfCall(right) &&
42
+ isMinusOne(left));
43
+ if (isComparisonWithIndexOf) {
44
+ context.report({ node, messageId: "preferIncludes" });
45
+ }
46
+ },
47
+ };
48
+ },
49
+ });
50
+ //# sourceMappingURL=prefer-includes.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"prefer-includes.js","sourceRoot":"","sources":["../../src/rules/prefer-includes.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAE,QAAQ,EAAE,MAAM,0BAA0B,CAAA;AAEhE,MAAM,UAAU,GAAG,WAAW,CAAC,WAAW,CACxC,CAAC,IAAI,EAAE,EAAE,CACP,yEAAyE,IAAI,KAAK,CACrF,CAAA;AAKD,SAAS,aAAa,CAAC,IAAmB;IACxC,OAAO,CACL,IAAI,CAAC,IAAI,KAAK,gBAAgB;QAC9B,IAAI,CAAC,MAAM,CAAC,IAAI,KAAK,kBAAkB;QACvC,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,IAAI,KAAK,YAAY;QAC1C,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,IAAI,KAAK,SAAS,CACxC,CAAA;AACH,CAAC;AAED,SAAS,UAAU,CAAC,IAAmB;IACrC,OAAO,CACL,IAAI,CAAC,IAAI,KAAK,iBAAiB;QAC/B,IAAI,CAAC,QAAQ,KAAK,GAAG;QACrB,IAAI,CAAC,QAAQ,CAAC,IAAI,KAAK,SAAS;QAChC,IAAI,CAAC,QAAQ,CAAC,KAAK,KAAK,CAAC,CAC1B,CAAA;AACH,CAAC;AAED,SAAS,MAAM,CAAC,IAAmB;IACjC,OAAO,IAAI,CAAC,IAAI,KAAK,SAAS,IAAI,IAAI,CAAC,KAAK,KAAK,CAAC,CAAA;AACpD,CAAC;AAED,MAAM,CAAC,MAAM,cAAc,GAAG,UAAU,CAAsB;IAC5D,IAAI,EAAE,iBAAiB;IACvB,IAAI,EAAE;QACJ,IAAI,EAAE,YAAY;QAClB,IAAI,EAAE;YACJ,WAAW,EAAE,uDAAuD;SACrE;QACD,MAAM,EAAE,EAAE;QACV,QAAQ,EAAE;YACR,cAAc,EACZ,uDAAuD;SAC1D;KACF;IACD,cAAc,EAAE,EAAE;IAClB,MAAM,CAAC,OAAO;QACZ,OAAO;YACL,gBAAgB,CAAC,IAA+B;gBAC9C,MAAM,EAAE,IAAI,EAAE,KAAK,EAAE,QAAQ,EAAE,GAAG,IAAI,CAAA;gBAEtC,MAAM,uBAAuB,GAC3B,CAAC,CAAC,KAAK,EAAE,IAAI,EAAE,KAAK,EAAE,IAAI,CAAC,CAAC,QAAQ,CAAC,QAAQ,CAAC;oBAC5C,aAAa,CAAC,IAAI,CAAC;oBACnB,UAAU,CAAC,KAAK,CAAC,CAAC;oBACpB,CAAC,QAAQ,KAAK,IAAI,IAAI,aAAa,CAAC,IAAI,CAAC,IAAI,MAAM,CAAC,KAAK,CAAC,CAAC;oBAC3D,CAAC,QAAQ,KAAK,GAAG,IAAI,aAAa,CAAC,IAAI,CAAC,IAAI,UAAU,CAAC,KAAK,CAAC,CAAC;oBAC9D,CAAC,CAAC,KAAK,EAAE,IAAI,EAAE,KAAK,EAAE,IAAI,CAAC,CAAC,QAAQ,CAAC,QAAQ,CAAC;wBAC5C,aAAa,CAAC,KAAK,CAAC;wBACpB,UAAU,CAAC,IAAI,CAAC,CAAC,CAAA;gBAErB,IAAI,uBAAuB,EAAE,CAAC;oBAC5B,OAAO,CAAC,MAAM,CAAC,EAAE,IAAI,EAAE,SAAS,EAAE,gBAAgB,EAAE,CAAC,CAAA;gBACvD,CAAC;YACH,CAAC;SACF,CAAA;IACH,CAAC;CACF,CAAC,CAAA"}
@@ -0,0 +1,5 @@
1
+ import { ESLintUtils } from "@typescript-eslint/utils";
2
+ export declare const preferReadonlyParams: ESLintUtils.RuleModule<"preferReadonlyParams", [], unknown, ESLintUtils.RuleListener> & {
3
+ name: string;
4
+ };
5
+ //# sourceMappingURL=prefer-readonly-params.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"prefer-readonly-params.d.ts","sourceRoot":"","sources":["../../src/rules/prefer-readonly-params.ts"],"names":[],"mappings":"AAAA,OAAO,EAAkB,WAAW,EAAY,MAAM,0BAA0B,CAAA;AA2BhF,eAAO,MAAM,oBAAoB;;CAuF/B,CAAA"}
@@ -0,0 +1,100 @@
1
+ import { AST_NODE_TYPES, ESLintUtils, TSESTree } from "@typescript-eslint/utils";
2
+ const createRule = ESLintUtils.RuleCreator((name) => `https://github.com/fookiejs/eslint-plugin-fookie/blob/main/docs/rules/${name}.md`);
3
+ function hasReadonlyModifier(typeAnnotation) {
4
+ const t = typeAnnotation.typeAnnotation;
5
+ if (t.type === AST_NODE_TYPES.TSTypeOperator && t.operator === "readonly")
6
+ return true;
7
+ return false;
8
+ }
9
+ function isArrayType(t) {
10
+ return t.type === AST_NODE_TYPES.TSArrayType || t.type === AST_NODE_TYPES.TSTupleType;
11
+ }
12
+ function isGenericArray(t) {
13
+ if (t.type !== AST_NODE_TYPES.TSTypeReference)
14
+ return false;
15
+ const name = t.typeName;
16
+ if (name.type !== AST_NODE_TYPES.Identifier)
17
+ return false;
18
+ return name.name === "Array" || name.name === "ReadonlyArray";
19
+ }
20
+ export const preferReadonlyParams = createRule({
21
+ name: "prefer-readonly-params",
22
+ meta: {
23
+ type: "suggestion",
24
+ docs: {
25
+ description: "Require array function parameters to be typed as readonly",
26
+ },
27
+ schema: [],
28
+ messages: {
29
+ preferReadonlyParams: "Array parameter '{{name}}' should be typed as readonly: 'readonly {{type}}'.",
30
+ },
31
+ },
32
+ defaultOptions: [],
33
+ create(context) {
34
+ function checkParam(param) {
35
+ if (param.type !== AST_NODE_TYPES.Identifier)
36
+ return;
37
+ const ann = param.typeAnnotation;
38
+ if (!ann)
39
+ return;
40
+ const t = ann.typeAnnotation;
41
+ if (isArrayType(t)) {
42
+ context.report({
43
+ node: param,
44
+ messageId: "preferReadonlyParams",
45
+ data: {
46
+ name: param.name,
47
+ type: context.getSourceCode().getText(t),
48
+ },
49
+ });
50
+ return;
51
+ }
52
+ if (isGenericArray(t) && t.typeName.type === AST_NODE_TYPES.Identifier) {
53
+ const typeName = t.typeName;
54
+ if (typeName.name === "Array") {
55
+ context.report({
56
+ node: param,
57
+ messageId: "preferReadonlyParams",
58
+ data: {
59
+ name: param.name,
60
+ type: context.getSourceCode().getText(t),
61
+ },
62
+ });
63
+ }
64
+ }
65
+ }
66
+ function checkFunction(node) {
67
+ for (const param of node.params) {
68
+ if (param.type === AST_NODE_TYPES.RestElement) {
69
+ const inner = param.argument;
70
+ if (inner.type === AST_NODE_TYPES.Identifier) {
71
+ const ann = inner.typeAnnotation;
72
+ if (!ann)
73
+ continue;
74
+ const t = ann.typeAnnotation;
75
+ if (isArrayType(t) || isGenericArray(t)) {
76
+ if (!hasReadonlyModifier(ann)) {
77
+ context.report({
78
+ node: param,
79
+ messageId: "preferReadonlyParams",
80
+ data: {
81
+ name: inner.name,
82
+ type: context.getSourceCode().getText(t),
83
+ },
84
+ });
85
+ }
86
+ }
87
+ }
88
+ continue;
89
+ }
90
+ checkParam(param);
91
+ }
92
+ }
93
+ return {
94
+ FunctionDeclaration: checkFunction,
95
+ FunctionExpression: checkFunction,
96
+ ArrowFunctionExpression: checkFunction,
97
+ };
98
+ },
99
+ });
100
+ //# sourceMappingURL=prefer-readonly-params.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"prefer-readonly-params.js","sourceRoot":"","sources":["../../src/rules/prefer-readonly-params.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,5 @@
1
+ import { ESLintUtils } from "@typescript-eslint/utils";
2
+ export declare const requireBooleanCondition: ESLintUtils.RuleModule<"requireBoolean", [], unknown, ESLintUtils.RuleListener> & {
3
+ name: string;
4
+ };
5
+ //# sourceMappingURL=require-boolean-condition.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"require-boolean-condition.d.ts","sourceRoot":"","sources":["../../src/rules/require-boolean-condition.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAY,MAAM,0BAA0B,CAAA;AAiBhE,eAAO,MAAM,uBAAuB;;CAoDlC,CAAA"}
@@ -0,0 +1,60 @@
1
+ import { ESLintUtils, TSESTree } from "@typescript-eslint/utils";
2
+ import * as ts from "typescript";
3
+ const createRule = ESLintUtils.RuleCreator((name) => `https://github.com/fookiejs/eslint-plugin-fookie/blob/main/docs/rules/${name}.md`);
4
+ function isPurelyBoolean(type) {
5
+ if ((type.flags & (ts.TypeFlags.Boolean | ts.TypeFlags.BooleanLiteral)) !== 0)
6
+ return true;
7
+ if (type.isUnion())
8
+ return type.types.every(isPurelyBoolean);
9
+ return false;
10
+ }
11
+ export const requireBooleanCondition = createRule({
12
+ name: "require-boolean-condition",
13
+ meta: {
14
+ type: "problem",
15
+ docs: {
16
+ description: "Require conditions in if/while/ternary to be explicitly boolean, not truthy/falsy",
17
+ },
18
+ schema: [],
19
+ messages: {
20
+ requireBoolean: "Condition has type '{{type}}'. Use an explicit boolean expression (e.g. x !== undefined, x.length > 0).",
21
+ },
22
+ },
23
+ defaultOptions: [],
24
+ create(context) {
25
+ const services = ESLintUtils.getParserServices(context);
26
+ const checker = services.program.getTypeChecker();
27
+ function check(conditionNode) {
28
+ const tsNode = services.esTreeNodeToTSNodeMap.get(conditionNode);
29
+ const type = checker.getTypeAtLocation(tsNode);
30
+ if ((type.flags & ts.TypeFlags.Any) !== 0)
31
+ return;
32
+ if (!isPurelyBoolean(type)) {
33
+ context.report({
34
+ node: conditionNode,
35
+ messageId: "requireBoolean",
36
+ data: { type: checker.typeToString(type) },
37
+ });
38
+ }
39
+ }
40
+ return {
41
+ "IfStatement"(node) {
42
+ check(node.test);
43
+ },
44
+ "WhileStatement"(node) {
45
+ check(node.test);
46
+ },
47
+ "DoWhileStatement"(node) {
48
+ check(node.test);
49
+ },
50
+ "ForStatement"(node) {
51
+ if (node.test !== null)
52
+ check(node.test);
53
+ },
54
+ "ConditionalExpression"(node) {
55
+ check(node.test);
56
+ },
57
+ };
58
+ },
59
+ });
60
+ //# sourceMappingURL=require-boolean-condition.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"require-boolean-condition.js","sourceRoot":"","sources":["../../src/rules/require-boolean-condition.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAE,QAAQ,EAAE,MAAM,0BAA0B,CAAA;AAChE,OAAO,KAAK,EAAE,MAAM,YAAY,CAAA;AAEhC,MAAM,UAAU,GAAG,WAAW,CAAC,WAAW,CACxC,CAAC,IAAI,EAAE,EAAE,CACP,yEAAyE,IAAI,KAAK,CACrF,CAAA;AAKD,SAAS,eAAe,CAAC,IAAa;IACpC,IAAI,CAAC,IAAI,CAAC,KAAK,GAAG,CAAC,EAAE,CAAC,SAAS,CAAC,OAAO,GAAG,EAAE,CAAC,SAAS,CAAC,cAAc,CAAC,CAAC,KAAK,CAAC;QAAE,OAAO,IAAI,CAAA;IAC1F,IAAI,IAAI,CAAC,OAAO,EAAE;QAAE,OAAO,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,eAAe,CAAC,CAAA;IAC5D,OAAO,KAAK,CAAA;AACd,CAAC;AAED,MAAM,CAAC,MAAM,uBAAuB,GAAG,UAAU,CAAsB;IACrE,IAAI,EAAE,2BAA2B;IACjC,IAAI,EAAE;QACJ,IAAI,EAAE,SAAS;QACf,IAAI,EAAE;YACJ,WAAW,EACT,mFAAmF;SACtF;QACD,MAAM,EAAE,EAAE;QACV,QAAQ,EAAE;YACR,cAAc,EACZ,yGAAyG;SAC5G;KACF;IACD,cAAc,EAAE,EAAE;IAClB,MAAM,CAAC,OAAO;QACZ,MAAM,QAAQ,GAAG,WAAW,CAAC,iBAAiB,CAAC,OAAO,CAAC,CAAA;QACvD,MAAM,OAAO,GAAG,QAAQ,CAAC,OAAO,CAAC,cAAc,EAAE,CAAA;QAEjD,SAAS,KAAK,CAAC,aAAkC;YAC/C,MAAM,MAAM,GAAG,QAAQ,CAAC,qBAAqB,CAAC,GAAG,CAAC,aAAa,CAAC,CAAA;YAChE,MAAM,IAAI,GAAG,OAAO,CAAC,iBAAiB,CAAC,MAAM,CAAC,CAAA;YAE9C,IAAI,CAAC,IAAI,CAAC,KAAK,GAAG,EAAE,CAAC,SAAS,CAAC,GAAG,CAAC,KAAK,CAAC;gBAAE,OAAM;YAEjD,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,EAAE,CAAC;gBAC3B,OAAO,CAAC,MAAM,CAAC;oBACb,IAAI,EAAE,aAAa;oBACnB,SAAS,EAAE,gBAAgB;oBAC3B,IAAI,EAAE,EAAE,IAAI,EAAE,OAAO,CAAC,YAAY,CAAC,IAAI,CAAC,EAAE;iBAC3C,CAAC,CAAA;YACJ,CAAC;QACH,CAAC;QAED,OAAO;YACL,aAAa,CAAC,IAA0B;gBACtC,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;YAClB,CAAC;YACD,gBAAgB,CAAC,IAA6B;gBAC5C,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;YAClB,CAAC;YACD,kBAAkB,CAAC,IAA+B;gBAChD,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;YAClB,CAAC;YACD,cAAc,CAAC,IAA2B;gBACxC,IAAI,IAAI,CAAC,IAAI,KAAK,IAAI;oBAAE,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;YAC1C,CAAC;YACD,uBAAuB,CAAC,IAAoC;gBAC1D,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;YAClB,CAAC;SACF,CAAA;IACH,CAAC;CACF,CAAC,CAAA"}
@@ -0,0 +1,5 @@
1
+ import { ESLintUtils } from "@typescript-eslint/utils";
2
+ export declare const requireCurly: ESLintUtils.RuleModule<"requireCurly", [], unknown, ESLintUtils.RuleListener> & {
3
+ name: string;
4
+ };
5
+ //# sourceMappingURL=require-curly.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"require-curly.d.ts","sourceRoot":"","sources":["../../src/rules/require-curly.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAY,MAAM,0BAA0B,CAAA;AAchE,eAAO,MAAM,YAAY;;CAgDvB,CAAA"}
@@ -0,0 +1,55 @@
1
+ import { ESLintUtils, TSESTree } from "@typescript-eslint/utils";
2
+ const createRule = ESLintUtils.RuleCreator((name) => `https://github.com/fookiejs/eslint-plugin-fookie/blob/main/docs/rules/${name}.md`);
3
+ function isBlock(node) {
4
+ return node.type === "BlockStatement";
5
+ }
6
+ export const requireCurly = createRule({
7
+ name: "require-curly",
8
+ meta: {
9
+ type: "suggestion",
10
+ docs: { description: "Require curly braces for all control flow statements" },
11
+ schema: [],
12
+ messages: {
13
+ requireCurly: "Always use curly braces after '{{keyword}}'.",
14
+ },
15
+ },
16
+ defaultOptions: [],
17
+ create(context) {
18
+ return {
19
+ IfStatement(node) {
20
+ if (!isBlock(node.consequent)) {
21
+ context.report({ node, messageId: "requireCurly", data: { keyword: "if" } });
22
+ }
23
+ if (node.alternate && !isBlock(node.alternate) && node.alternate.type !== "IfStatement") {
24
+ context.report({ node: node.alternate, messageId: "requireCurly", data: { keyword: "else" } });
25
+ }
26
+ },
27
+ ForStatement(node) {
28
+ if (!isBlock(node.body)) {
29
+ context.report({ node, messageId: "requireCurly", data: { keyword: "for" } });
30
+ }
31
+ },
32
+ ForOfStatement(node) {
33
+ if (!isBlock(node.body)) {
34
+ context.report({ node, messageId: "requireCurly", data: { keyword: "for...of" } });
35
+ }
36
+ },
37
+ ForInStatement(node) {
38
+ if (!isBlock(node.body)) {
39
+ context.report({ node, messageId: "requireCurly", data: { keyword: "for...in" } });
40
+ }
41
+ },
42
+ WhileStatement(node) {
43
+ if (!isBlock(node.body)) {
44
+ context.report({ node, messageId: "requireCurly", data: { keyword: "while" } });
45
+ }
46
+ },
47
+ DoWhileStatement(node) {
48
+ if (!isBlock(node.body)) {
49
+ context.report({ node, messageId: "requireCurly", data: { keyword: "do...while" } });
50
+ }
51
+ },
52
+ };
53
+ },
54
+ });
55
+ //# sourceMappingURL=require-curly.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"require-curly.js","sourceRoot":"","sources":["../../src/rules/require-curly.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,5 @@
1
+ import { ESLintUtils } from "@typescript-eslint/utils";
2
+ export declare const requireExplicitReturnType: ESLintUtils.RuleModule<"requireReturnType", [], unknown, ESLintUtils.RuleListener> & {
3
+ name: string;
4
+ };
5
+ //# sourceMappingURL=require-explicit-return-type.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"require-explicit-return-type.d.ts","sourceRoot":"","sources":["../../src/rules/require-explicit-return-type.ts"],"names":[],"mappings":"AAAA,OAAO,EAAkB,WAAW,EAAY,MAAM,0BAA0B,CAAA;AAkBhF,eAAO,MAAM,yBAAyB;;CAoEpC,CAAA"}
@@ -0,0 +1,66 @@
1
+ import { AST_NODE_TYPES, ESLintUtils, TSESTree } from "@typescript-eslint/utils";
2
+ const createRule = ESLintUtils.RuleCreator((name) => `https://github.com/fookiejs/eslint-plugin-fookie/blob/main/docs/rules/${name}.md`);
3
+ function isExported(node) {
4
+ const parent = node.parent;
5
+ if (!parent)
6
+ return false;
7
+ if (parent.type === AST_NODE_TYPES.ExportNamedDeclaration)
8
+ return true;
9
+ if (parent.type === AST_NODE_TYPES.ExportDefaultDeclaration)
10
+ return true;
11
+ return false;
12
+ }
13
+ export const requireExplicitReturnType = createRule({
14
+ name: "require-explicit-return-type",
15
+ meta: {
16
+ type: "suggestion",
17
+ docs: {
18
+ description: "Require explicit return type on exported functions and class methods",
19
+ },
20
+ schema: [],
21
+ messages: {
22
+ requireReturnType: "Exported functions and class methods must have an explicit return type annotation.",
23
+ },
24
+ },
25
+ defaultOptions: [],
26
+ create(context) {
27
+ function check(node) {
28
+ if (node.returnType)
29
+ return;
30
+ if (isExported(node)) {
31
+ context.report({ node, messageId: "requireReturnType" });
32
+ return;
33
+ }
34
+ const parent = node.parent;
35
+ if (parent.type === AST_NODE_TYPES.MethodDefinition) {
36
+ if (parent.accessibility !== "private" &&
37
+ parent.accessibility !== "protected") {
38
+ context.report({ node, messageId: "requireReturnType" });
39
+ }
40
+ return;
41
+ }
42
+ if (parent.type === AST_NODE_TYPES.PropertyDefinition) {
43
+ if (parent.accessibility !== "private" &&
44
+ parent.accessibility !== "protected") {
45
+ context.report({ node, messageId: "requireReturnType" });
46
+ }
47
+ return;
48
+ }
49
+ if (parent.type === AST_NODE_TYPES.VariableDeclarator) {
50
+ const grandParent = parent.parent;
51
+ if (grandParent.type === AST_NODE_TYPES.VariableDeclaration) {
52
+ const greatGrandParent = grandParent.parent;
53
+ if (greatGrandParent.type === AST_NODE_TYPES.ExportNamedDeclaration) {
54
+ context.report({ node, messageId: "requireReturnType" });
55
+ }
56
+ }
57
+ }
58
+ }
59
+ return {
60
+ FunctionDeclaration: check,
61
+ FunctionExpression: check,
62
+ ArrowFunctionExpression: check,
63
+ };
64
+ },
65
+ });
66
+ //# sourceMappingURL=require-explicit-return-type.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"require-explicit-return-type.js","sourceRoot":"","sources":["../../src/rules/require-explicit-return-type.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,cAAc,EAAE,WAAW,EAAE,QAAQ,EAAE,MAAM,0BAA0B,CAAA;AAEhF,MAAM,UAAU,GAAG,WAAW,CAAC,WAAW,CACxC,CAAC,IAAI,EAAE,EAAE,CACP,yEAAyE,IAAI,KAAK,CACrF,CAAA;AAKD,SAAS,UAAU,CAAC,IAAmB;IACrC,MAAM,MAAM,GAAG,IAAI,CAAC,MAAM,CAAA;IAC1B,IAAI,CAAC,MAAM;QAAE,OAAO,KAAK,CAAA;IACzB,IAAI,MAAM,CAAC,IAAI,KAAK,cAAc,CAAC,sBAAsB;QAAE,OAAO,IAAI,CAAA;IACtE,IAAI,MAAM,CAAC,IAAI,KAAK,cAAc,CAAC,wBAAwB;QAAE,OAAO,IAAI,CAAA;IACxE,OAAO,KAAK,CAAA;AACd,CAAC;AAED,MAAM,CAAC,MAAM,yBAAyB,GAAG,UAAU,CAAsB;IACvE,IAAI,EAAE,8BAA8B;IACpC,IAAI,EAAE;QACJ,IAAI,EAAE,YAAY;QAClB,IAAI,EAAE;YACJ,WAAW,EACT,sEAAsE;SACzE;QACD,MAAM,EAAE,EAAE;QACV,QAAQ,EAAE;YACR,iBAAiB,EACf,oFAAoF;SACvF;KACF;IACD,cAAc,EAAE,EAAE;IAClB,MAAM,CAAC,OAAO;QACZ,SAAS,KAAK,CACZ,IAGoC;YAEpC,IAAI,IAAI,CAAC,UAAU;gBAAE,OAAM;YAE3B,IAAI,UAAU,CAAC,IAAI,CAAC,EAAE,CAAC;gBACrB,OAAO,CAAC,MAAM,CAAC,EAAE,IAAI,EAAE,SAAS,EAAE,mBAAmB,EAAE,CAAC,CAAA;gBACxD,OAAM;YACR,CAAC;YAED,MAAM,MAAM,GAAG,IAAI,CAAC,MAAM,CAAA;YAE1B,IAAI,MAAM,CAAC,IAAI,KAAK,cAAc,CAAC,gBAAgB,EAAE,CAAC;gBACpD,IACE,MAAM,CAAC,aAAa,KAAK,SAAS;oBAClC,MAAM,CAAC,aAAa,KAAK,WAAW,EACpC,CAAC;oBACD,OAAO,CAAC,MAAM,CAAC,EAAE,IAAI,EAAE,SAAS,EAAE,mBAAmB,EAAE,CAAC,CAAA;gBAC1D,CAAC;gBACD,OAAM;YACR,CAAC;YAED,IAAI,MAAM,CAAC,IAAI,KAAK,cAAc,CAAC,kBAAkB,EAAE,CAAC;gBACtD,IACE,MAAM,CAAC,aAAa,KAAK,SAAS;oBAClC,MAAM,CAAC,aAAa,KAAK,WAAW,EACpC,CAAC;oBACD,OAAO,CAAC,MAAM,CAAC,EAAE,IAAI,EAAE,SAAS,EAAE,mBAAmB,EAAE,CAAC,CAAA;gBAC1D,CAAC;gBACD,OAAM;YACR,CAAC;YAED,IAAI,MAAM,CAAC,IAAI,KAAK,cAAc,CAAC,kBAAkB,EAAE,CAAC;gBACtD,MAAM,WAAW,GAAG,MAAM,CAAC,MAAM,CAAA;gBACjC,IAAI,WAAW,CAAC,IAAI,KAAK,cAAc,CAAC,mBAAmB,EAAE,CAAC;oBAC5D,MAAM,gBAAgB,GAAG,WAAW,CAAC,MAAM,CAAA;oBAC3C,IAAI,gBAAgB,CAAC,IAAI,KAAK,cAAc,CAAC,sBAAsB,EAAE,CAAC;wBACpE,OAAO,CAAC,MAAM,CAAC,EAAE,IAAI,EAAE,SAAS,EAAE,mBAAmB,EAAE,CAAC,CAAA;oBAC1D,CAAC;gBACH,CAAC;YACH,CAAC;QACH,CAAC;QAED,OAAO;YACL,mBAAmB,EAAE,KAAK;YAC1B,kBAAkB,EAAE,KAAK;YACzB,uBAAuB,EAAE,KAAK;SAC/B,CAAA;IACH,CAAC;CACF,CAAC,CAAA"}
@@ -0,0 +1,5 @@
1
+ import { ESLintUtils } from "@typescript-eslint/utils";
2
+ export declare const requirePrivateConstructor: ESLintUtils.RuleModule<"requirePrivateConstructor", [], unknown, ESLintUtils.RuleListener> & {
3
+ name: string;
4
+ };
5
+ //# sourceMappingURL=require-private-constructor.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"require-private-constructor.d.ts","sourceRoot":"","sources":["../../src/rules/require-private-constructor.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAY,MAAM,0BAA0B,CAAA;AAUhE,eAAO,MAAM,yBAAyB;;CAoCpC,CAAA"}
@@ -0,0 +1,34 @@
1
+ import { ESLintUtils, TSESTree } from "@typescript-eslint/utils";
2
+ const createRule = ESLintUtils.RuleCreator((name) => `https://github.com/fookiejs/eslint-plugin-fookie/blob/main/docs/rules/${name}.md`);
3
+ export const requirePrivateConstructor = createRule({
4
+ name: "require-private-constructor",
5
+ meta: {
6
+ type: "suggestion",
7
+ docs: {
8
+ description: "Require class constructors to be private. Forces use of static factory methods.",
9
+ },
10
+ schema: [],
11
+ messages: {
12
+ requirePrivateConstructor: "Constructor must be private. Use a static factory method like '{{name}}.create(...)' instead of 'new {{name}}(...)'.",
13
+ },
14
+ },
15
+ defaultOptions: [],
16
+ create(context) {
17
+ return {
18
+ MethodDefinition(node) {
19
+ if (node.kind !== "constructor")
20
+ return;
21
+ if (node.accessibility === "private")
22
+ return;
23
+ const classNode = node.parent.parent;
24
+ const name = classNode.id?.name ?? "Class";
25
+ context.report({
26
+ node,
27
+ messageId: "requirePrivateConstructor",
28
+ data: { name },
29
+ });
30
+ },
31
+ };
32
+ },
33
+ });
34
+ //# sourceMappingURL=require-private-constructor.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"require-private-constructor.js","sourceRoot":"","sources":["../../src/rules/require-private-constructor.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAE,QAAQ,EAAE,MAAM,0BAA0B,CAAA;AAEhE,MAAM,UAAU,GAAG,WAAW,CAAC,WAAW,CACxC,CAAC,IAAI,EAAE,EAAE,CACP,yEAAyE,IAAI,KAAK,CACrF,CAAA;AAKD,MAAM,CAAC,MAAM,yBAAyB,GAAG,UAAU,CAAsB;IACvE,IAAI,EAAE,6BAA6B;IACnC,IAAI,EAAE;QACJ,IAAI,EAAE,YAAY;QAClB,IAAI,EAAE;YACJ,WAAW,EACT,iFAAiF;SACpF;QACD,MAAM,EAAE,EAAE;QACV,QAAQ,EAAE;YACR,yBAAyB,EACvB,sHAAsH;SACzH;KACF;IACD,cAAc,EAAE,EAAE;IAClB,MAAM,CAAC,OAAO;QACZ,OAAO;YACL,gBAAgB,CAAC,IAA+B;gBAC9C,IAAI,IAAI,CAAC,IAAI,KAAK,aAAa;oBAAE,OAAM;gBACvC,IAAI,IAAI,CAAC,aAAa,KAAK,SAAS;oBAAE,OAAM;gBAE5C,MAAM,SAAS,GAAG,IAAI,CAAC,MAAM,CAAC,MAEF,CAAA;gBAE5B,MAAM,IAAI,GACR,SAAS,CAAC,EAAE,EAAE,IAAI,IAAI,OAAO,CAAA;gBAE/B,OAAO,CAAC,MAAM,CAAC;oBACb,IAAI;oBACJ,SAAS,EAAE,2BAA2B;oBACtC,IAAI,EAAE,EAAE,IAAI,EAAE;iBACf,CAAC,CAAA;YACJ,CAAC;SACF,CAAA;IACH,CAAC;CACF,CAAC,CAAA"}
@@ -0,0 +1,5 @@
1
+ import { ESLintUtils } from "@typescript-eslint/utils";
2
+ export declare const sameTypeComparison: ESLintUtils.RuleModule<"mismatchedTypes", [], unknown, ESLintUtils.RuleListener> & {
3
+ name: string;
4
+ };
5
+ //# sourceMappingURL=same-type-comparison.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"same-type-comparison.d.ts","sourceRoot":"","sources":["../../src/rules/same-type-comparison.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAY,MAAM,0BAA0B,CAAA;AAgChE,eAAO,MAAM,kBAAkB;;CA6C7B,CAAA"}
@@ -0,0 +1,71 @@
1
+ import { ESLintUtils, TSESTree } from "@typescript-eslint/utils";
2
+ import * as ts from "typescript";
3
+ const createRule = ESLintUtils.RuleCreator((name) => `https://github.com/fookiejs/eslint-plugin-fookie/blob/main/docs/rules/${name}.md`);
4
+ function isStringBased(type) {
5
+ if ((type.flags & ts.TypeFlags.String) !== 0)
6
+ return true;
7
+ if ((type.flags & ts.TypeFlags.StringLiteral) !== 0)
8
+ return true;
9
+ if (type.isUnion())
10
+ return type.types.every(isStringBased);
11
+ return false;
12
+ }
13
+ function isNumberBased(type) {
14
+ if ((type.flags & ts.TypeFlags.Number) !== 0)
15
+ return true;
16
+ if ((type.flags & ts.TypeFlags.NumberLiteral) !== 0)
17
+ return true;
18
+ if (type.isUnion())
19
+ return type.types.every(isNumberBased);
20
+ return false;
21
+ }
22
+ function areCompatible(a, b, checker) {
23
+ if ((a.flags & ts.TypeFlags.Any) !== 0 || (b.flags & ts.TypeFlags.Any) !== 0)
24
+ return true;
25
+ if (isStringBased(a) && isStringBased(b))
26
+ return true;
27
+ if (isNumberBased(a) && isNumberBased(b))
28
+ return true;
29
+ return checker.isTypeAssignableTo(a, b) || checker.isTypeAssignableTo(b, a);
30
+ }
31
+ export const sameTypeComparison = createRule({
32
+ name: "same-type-comparison",
33
+ meta: {
34
+ type: "problem",
35
+ docs: {
36
+ description: "Require both sides of === and !== to have the same TypeScript type",
37
+ },
38
+ schema: [],
39
+ messages: {
40
+ mismatchedTypes: "Comparing '{{left}}' with '{{right}}' — both sides must be the same type.",
41
+ },
42
+ },
43
+ defaultOptions: [],
44
+ create(context) {
45
+ const services = ESLintUtils.getParserServices(context);
46
+ const checker = services.program.getTypeChecker();
47
+ return {
48
+ BinaryExpression(node) {
49
+ const op = node.operator;
50
+ if (op !== "===" && op !== "!==" && op !== "==" && op !== "!=") {
51
+ return;
52
+ }
53
+ const leftTsNode = services.esTreeNodeToTSNodeMap.get(node.left);
54
+ const rightTsNode = services.esTreeNodeToTSNodeMap.get(node.right);
55
+ const leftType = checker.getTypeAtLocation(leftTsNode);
56
+ const rightType = checker.getTypeAtLocation(rightTsNode);
57
+ if (!areCompatible(leftType, rightType, checker)) {
58
+ context.report({
59
+ node,
60
+ messageId: "mismatchedTypes",
61
+ data: {
62
+ left: checker.typeToString(leftType),
63
+ right: checker.typeToString(rightType),
64
+ },
65
+ });
66
+ }
67
+ },
68
+ };
69
+ },
70
+ });
71
+ //# sourceMappingURL=same-type-comparison.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"same-type-comparison.js","sourceRoot":"","sources":["../../src/rules/same-type-comparison.ts"],"names":[],"mappings":"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"}
package/package.json ADDED
@@ -0,0 +1,46 @@
1
+ {
2
+ "name": "@fookiejs/eslint-plugin",
3
+ "version": "0.1.0",
4
+ "description": "ESLint plugin for TypeScript code quality — naming, structure, type safety.",
5
+ "type": "module",
6
+ "main": "./dist/index.js",
7
+ "types": "./dist/index.d.ts",
8
+ "exports": {
9
+ ".": {
10
+ "types": "./dist/index.d.ts",
11
+ "import": "./dist/index.js"
12
+ }
13
+ },
14
+ "files": [
15
+ "dist"
16
+ ],
17
+ "scripts": {
18
+ "build": "tsc -p tsconfig.build.json",
19
+ "test": "node --import tsx/esm --test 'tests/**/*.test.ts'",
20
+ "lint": "eslint src tests"
21
+ },
22
+ "keywords": [
23
+ "eslint",
24
+ "typescript",
25
+ "linter"
26
+ ],
27
+ "peerDependencies": {
28
+ "@typescript-eslint/parser": ">=7.0.0",
29
+ "eslint": ">=8.0.0",
30
+ "typescript": ">=5.0.0"
31
+ },
32
+ "dependencies": {
33
+ "@typescript-eslint/utils": "^8.0.0"
34
+ },
35
+ "publishConfig": {
36
+ "access": "public"
37
+ },
38
+ "devDependencies": {
39
+ "@types/node": "^22.0.0",
40
+ "@typescript-eslint/parser": "^8.0.0",
41
+ "@typescript-eslint/rule-tester": "^8.0.0",
42
+ "eslint": "^10.6.0",
43
+ "tsx": "^4.22.4",
44
+ "typescript": "^5.7.0"
45
+ }
46
+ }