eslint 8.57.1 → 9.39.1

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 (458) hide show
  1. package/README.md +165 -115
  2. package/bin/eslint.js +112 -89
  3. package/conf/default-cli-options.js +22 -22
  4. package/conf/ecma-version.js +16 -0
  5. package/conf/globals.js +109 -94
  6. package/conf/replacements.json +24 -20
  7. package/conf/rule-type-list.json +89 -26
  8. package/lib/api.js +16 -20
  9. package/lib/cli-engine/cli-engine.js +841 -810
  10. package/lib/cli-engine/file-enumerator.js +384 -390
  11. package/lib/cli-engine/formatters/formatters-meta.json +17 -45
  12. package/lib/cli-engine/formatters/html.js +110 -102
  13. package/lib/cli-engine/formatters/json-with-metadata.js +5 -5
  14. package/lib/cli-engine/formatters/json.js +2 -2
  15. package/lib/cli-engine/formatters/stylish.js +97 -76
  16. package/lib/cli-engine/hash.js +1 -1
  17. package/lib/cli-engine/index.js +1 -1
  18. package/lib/cli-engine/lint-result-cache.js +165 -148
  19. package/lib/cli-engine/load-rules.js +17 -17
  20. package/lib/cli.js +481 -399
  21. package/lib/config/config-loader.js +816 -0
  22. package/lib/config/config.js +674 -0
  23. package/lib/config/default-config.js +57 -46
  24. package/lib/config/flat-config-array.js +170 -333
  25. package/lib/config/flat-config-schema.js +389 -389
  26. package/lib/config-api.js +12 -0
  27. package/lib/eslint/eslint-helpers.js +1196 -663
  28. package/lib/eslint/eslint.js +1262 -607
  29. package/lib/eslint/index.js +3 -3
  30. package/lib/eslint/legacy-eslint.js +786 -0
  31. package/lib/eslint/worker.js +173 -0
  32. package/lib/languages/js/index.js +336 -0
  33. package/lib/languages/js/source-code/index.js +7 -0
  34. package/lib/languages/js/source-code/source-code.js +1364 -0
  35. package/lib/languages/js/source-code/token-store/backward-token-comment-cursor.js +61 -0
  36. package/lib/languages/js/source-code/token-store/backward-token-cursor.js +57 -0
  37. package/lib/{source-code → languages/js/source-code}/token-store/cursor.js +36 -36
  38. package/lib/languages/js/source-code/token-store/cursors.js +120 -0
  39. package/lib/{source-code → languages/js/source-code}/token-store/decorative-cursor.js +17 -18
  40. package/lib/{source-code → languages/js/source-code}/token-store/filter-cursor.js +19 -20
  41. package/lib/languages/js/source-code/token-store/forward-token-comment-cursor.js +65 -0
  42. package/lib/languages/js/source-code/token-store/forward-token-cursor.js +62 -0
  43. package/lib/languages/js/source-code/token-store/index.js +721 -0
  44. package/lib/{source-code → languages/js/source-code}/token-store/limit-cursor.js +17 -18
  45. package/lib/languages/js/source-code/token-store/padded-token-cursor.js +45 -0
  46. package/lib/{source-code → languages/js/source-code}/token-store/skip-cursor.js +19 -20
  47. package/lib/languages/js/source-code/token-store/utils.js +110 -0
  48. package/lib/languages/js/validate-language-options.js +196 -0
  49. package/lib/linter/apply-disable-directives.js +490 -371
  50. package/lib/linter/code-path-analysis/code-path-analyzer.js +650 -674
  51. package/lib/linter/code-path-analysis/code-path-segment.js +215 -216
  52. package/lib/linter/code-path-analysis/code-path-state.js +2118 -2096
  53. package/lib/linter/code-path-analysis/code-path.js +307 -317
  54. package/lib/linter/code-path-analysis/debug-helpers.js +183 -163
  55. package/lib/linter/code-path-analysis/fork-context.js +297 -272
  56. package/lib/linter/code-path-analysis/id-generator.js +22 -23
  57. package/lib/linter/esquery.js +332 -0
  58. package/lib/linter/file-context.js +144 -0
  59. package/lib/linter/file-report.js +608 -0
  60. package/lib/linter/index.js +3 -5
  61. package/lib/linter/interpolate.js +38 -16
  62. package/lib/linter/linter.js +2328 -1785
  63. package/lib/linter/rule-fixer.js +136 -107
  64. package/lib/linter/rules.js +37 -46
  65. package/lib/linter/source-code-fixer.js +96 -94
  66. package/lib/linter/source-code-traverser.js +333 -0
  67. package/lib/linter/source-code-visitor.js +81 -0
  68. package/lib/linter/timing.js +145 -97
  69. package/lib/linter/vfile.js +115 -0
  70. package/lib/options.js +464 -326
  71. package/lib/rule-tester/index.js +3 -1
  72. package/lib/rule-tester/rule-tester.js +1371 -998
  73. package/lib/rules/accessor-pairs.js +333 -259
  74. package/lib/rules/array-bracket-newline.js +250 -220
  75. package/lib/rules/array-bracket-spacing.js +286 -229
  76. package/lib/rules/array-callback-return.js +401 -354
  77. package/lib/rules/array-element-newline.js +358 -295
  78. package/lib/rules/arrow-body-style.js +400 -278
  79. package/lib/rules/arrow-parens.js +206 -155
  80. package/lib/rules/arrow-spacing.js +169 -145
  81. package/lib/rules/block-scoped-var.js +125 -123
  82. package/lib/rules/block-spacing.js +186 -158
  83. package/lib/rules/brace-style.js +262 -181
  84. package/lib/rules/callback-return.js +203 -174
  85. package/lib/rules/camelcase.js +403 -380
  86. package/lib/rules/capitalized-comments.js +253 -228
  87. package/lib/rules/class-methods-use-this.js +231 -168
  88. package/lib/rules/comma-dangle.js +379 -328
  89. package/lib/rules/comma-spacing.js +193 -177
  90. package/lib/rules/comma-style.js +375 -298
  91. package/lib/rules/complexity.js +180 -144
  92. package/lib/rules/computed-property-spacing.js +236 -193
  93. package/lib/rules/consistent-return.js +181 -170
  94. package/lib/rules/consistent-this.js +167 -141
  95. package/lib/rules/constructor-super.js +418 -411
  96. package/lib/rules/curly.js +407 -468
  97. package/lib/rules/default-case-last.js +39 -32
  98. package/lib/rules/default-case.js +89 -83
  99. package/lib/rules/default-param-last.js +69 -53
  100. package/lib/rules/dot-location.js +122 -92
  101. package/lib/rules/dot-notation.js +193 -153
  102. package/lib/rules/eol-last.js +122 -102
  103. package/lib/rules/eqeqeq.js +191 -155
  104. package/lib/rules/for-direction.js +150 -122
  105. package/lib/rules/func-call-spacing.js +261 -213
  106. package/lib/rules/func-name-matching.js +294 -209
  107. package/lib/rules/func-names.js +165 -164
  108. package/lib/rules/func-style.js +209 -86
  109. package/lib/rules/function-call-argument-newline.js +152 -111
  110. package/lib/rules/function-paren-newline.js +349 -273
  111. package/lib/rules/generator-star-spacing.js +229 -192
  112. package/lib/rules/getter-return.js +208 -170
  113. package/lib/rules/global-require.js +85 -58
  114. package/lib/rules/grouped-accessor-pairs.js +201 -148
  115. package/lib/rules/guard-for-in.js +72 -63
  116. package/lib/rules/handle-callback-err.js +108 -87
  117. package/lib/rules/id-blacklist.js +182 -187
  118. package/lib/rules/id-denylist.js +174 -179
  119. package/lib/rules/id-length.js +197 -157
  120. package/lib/rules/id-match.js +350 -286
  121. package/lib/rules/implicit-arrow-linebreak.js +102 -61
  122. package/lib/rules/indent-legacy.js +1345 -1102
  123. package/lib/rules/indent.js +2272 -1741
  124. package/lib/rules/index.js +320 -294
  125. package/lib/rules/init-declarations.js +139 -106
  126. package/lib/rules/jsx-quotes.js +94 -64
  127. package/lib/rules/key-spacing.js +750 -615
  128. package/lib/rules/keyword-spacing.js +648 -587
  129. package/lib/rules/line-comment-position.js +143 -108
  130. package/lib/rules/linebreak-style.js +115 -88
  131. package/lib/rules/lines-around-comment.js +540 -430
  132. package/lib/rules/lines-around-directive.js +233 -185
  133. package/lib/rules/lines-between-class-members.js +305 -216
  134. package/lib/rules/logical-assignment-operators.js +582 -398
  135. package/lib/rules/max-classes-per-file.js +69 -68
  136. package/lib/rules/max-depth.js +146 -143
  137. package/lib/rules/max-len.js +473 -416
  138. package/lib/rules/max-lines-per-function.js +201 -176
  139. package/lib/rules/max-lines.js +158 -162
  140. package/lib/rules/max-nested-callbacks.js +102 -104
  141. package/lib/rules/max-params.js +102 -75
  142. package/lib/rules/max-statements-per-line.js +205 -180
  143. package/lib/rules/max-statements.js +168 -164
  144. package/lib/rules/multiline-comment-style.js +638 -460
  145. package/lib/rules/multiline-ternary.js +241 -158
  146. package/lib/rules/new-cap.js +233 -232
  147. package/lib/rules/new-parens.js +88 -61
  148. package/lib/rules/newline-after-var.js +287 -233
  149. package/lib/rules/newline-before-return.js +229 -204
  150. package/lib/rules/newline-per-chained-call.js +142 -109
  151. package/lib/rules/no-alert.js +90 -79
  152. package/lib/rules/no-array-constructor.js +175 -113
  153. package/lib/rules/no-async-promise-executor.js +30 -24
  154. package/lib/rules/no-await-in-loop.js +79 -70
  155. package/lib/rules/no-bitwise.js +113 -87
  156. package/lib/rules/no-buffer-constructor.js +61 -37
  157. package/lib/rules/no-caller.js +39 -33
  158. package/lib/rules/no-case-declarations.js +61 -45
  159. package/lib/rules/no-catch-shadow.js +76 -62
  160. package/lib/rules/no-class-assign.js +51 -48
  161. package/lib/rules/no-compare-neg-zero.js +62 -48
  162. package/lib/rules/no-cond-assign.js +148 -132
  163. package/lib/rules/no-confusing-arrow.js +98 -63
  164. package/lib/rules/no-console.js +202 -188
  165. package/lib/rules/no-const-assign.js +58 -41
  166. package/lib/rules/no-constant-binary-expression.js +501 -407
  167. package/lib/rules/no-constant-condition.js +158 -131
  168. package/lib/rules/no-constructor-return.js +49 -49
  169. package/lib/rules/no-continue.js +25 -26
  170. package/lib/rules/no-control-regex.js +125 -121
  171. package/lib/rules/no-debugger.js +28 -30
  172. package/lib/rules/no-delete-var.js +29 -29
  173. package/lib/rules/no-div-regex.js +47 -40
  174. package/lib/rules/no-dupe-args.js +79 -69
  175. package/lib/rules/no-dupe-class-members.js +102 -89
  176. package/lib/rules/no-dupe-else-if.js +100 -77
  177. package/lib/rules/no-dupe-keys.js +133 -110
  178. package/lib/rules/no-duplicate-case.js +50 -43
  179. package/lib/rules/no-duplicate-imports.js +266 -188
  180. package/lib/rules/no-else-return.js +430 -385
  181. package/lib/rules/no-empty-character-class.js +57 -50
  182. package/lib/rules/no-empty-function.js +197 -128
  183. package/lib/rules/no-empty-pattern.js +63 -56
  184. package/lib/rules/no-empty-static-block.js +61 -35
  185. package/lib/rules/no-empty.js +135 -85
  186. package/lib/rules/no-eq-null.js +37 -32
  187. package/lib/rules/no-eval.js +258 -249
  188. package/lib/rules/no-ex-assign.js +42 -39
  189. package/lib/rules/no-extend-native.js +161 -160
  190. package/lib/rules/no-extra-bind.js +201 -190
  191. package/lib/rules/no-extra-boolean-cast.js +398 -295
  192. package/lib/rules/no-extra-label.js +150 -130
  193. package/lib/rules/no-extra-parens.js +1654 -1307
  194. package/lib/rules/no-extra-semi.js +146 -126
  195. package/lib/rules/no-fallthrough.js +200 -136
  196. package/lib/rules/no-floating-decimal.js +74 -48
  197. package/lib/rules/no-func-assign.js +54 -55
  198. package/lib/rules/no-global-assign.js +78 -72
  199. package/lib/rules/no-implicit-coercion.js +350 -262
  200. package/lib/rules/no-implicit-globals.js +174 -133
  201. package/lib/rules/no-implied-eval.js +150 -112
  202. package/lib/rules/no-import-assign.js +145 -159
  203. package/lib/rules/no-inline-comments.js +101 -96
  204. package/lib/rules/no-inner-declarations.js +115 -78
  205. package/lib/rules/no-invalid-regexp.js +223 -174
  206. package/lib/rules/no-invalid-this.js +145 -117
  207. package/lib/rules/no-irregular-whitespace.js +266 -250
  208. package/lib/rules/no-iterator.js +29 -33
  209. package/lib/rules/no-label-var.js +59 -61
  210. package/lib/rules/no-labels.js +138 -131
  211. package/lib/rules/no-lone-blocks.js +127 -123
  212. package/lib/rules/no-lonely-if.js +105 -67
  213. package/lib/rules/no-loop-func.js +245 -184
  214. package/lib/rules/no-loss-of-precision.js +236 -201
  215. package/lib/rules/no-magic-numbers.js +339 -217
  216. package/lib/rules/no-misleading-character-class.js +548 -253
  217. package/lib/rules/no-mixed-operators.js +188 -164
  218. package/lib/rules/no-mixed-requires.js +253 -224
  219. package/lib/rules/no-mixed-spaces-and-tabs.js +135 -103
  220. package/lib/rules/no-multi-assign.js +46 -47
  221. package/lib/rules/no-multi-spaces.js +163 -125
  222. package/lib/rules/no-multi-str.js +42 -40
  223. package/lib/rules/no-multiple-empty-lines.js +196 -140
  224. package/lib/rules/no-native-reassign.js +90 -74
  225. package/lib/rules/no-negated-condition.js +79 -74
  226. package/lib/rules/no-negated-in-lhs.js +45 -32
  227. package/lib/rules/no-nested-ternary.js +33 -31
  228. package/lib/rules/no-new-func.js +71 -62
  229. package/lib/rules/no-new-native-nonconstructor.js +43 -39
  230. package/lib/rules/no-new-object.js +48 -39
  231. package/lib/rules/no-new-require.js +48 -31
  232. package/lib/rules/no-new-symbol.js +61 -43
  233. package/lib/rules/no-new-wrappers.js +43 -41
  234. package/lib/rules/no-new.js +28 -29
  235. package/lib/rules/no-nonoctal-decimal-escape.js +149 -121
  236. package/lib/rules/no-obj-calls.js +66 -53
  237. package/lib/rules/no-object-constructor.js +104 -97
  238. package/lib/rules/no-octal-escape.js +40 -43
  239. package/lib/rules/no-octal.js +29 -32
  240. package/lib/rules/no-param-reassign.js +236 -218
  241. package/lib/rules/no-path-concat.js +66 -51
  242. package/lib/rules/no-plusplus.js +60 -63
  243. package/lib/rules/no-process-env.js +49 -32
  244. package/lib/rules/no-process-exit.js +48 -28
  245. package/lib/rules/no-promise-executor-return.js +205 -204
  246. package/lib/rules/no-proto.js +26 -29
  247. package/lib/rules/no-prototype-builtins.js +146 -124
  248. package/lib/rules/no-redeclare.js +154 -155
  249. package/lib/rules/no-regex-spaces.js +183 -161
  250. package/lib/rules/no-restricted-exports.js +208 -174
  251. package/lib/rules/no-restricted-globals.js +254 -112
  252. package/lib/rules/no-restricted-imports.js +824 -384
  253. package/lib/rules/no-restricted-modules.js +222 -186
  254. package/lib/rules/no-restricted-properties.js +218 -153
  255. package/lib/rules/no-restricted-syntax.js +56 -52
  256. package/lib/rules/no-return-assign.js +56 -49
  257. package/lib/rules/no-return-await.js +147 -120
  258. package/lib/rules/no-script-url.js +53 -46
  259. package/lib/rules/no-self-assign.js +148 -145
  260. package/lib/rules/no-self-compare.js +63 -46
  261. package/lib/rules/no-sequences.js +135 -115
  262. package/lib/rules/no-setter-return.js +176 -178
  263. package/lib/rules/no-shadow-restricted-names.js +84 -36
  264. package/lib/rules/no-shadow.js +598 -310
  265. package/lib/rules/no-spaced-func.js +82 -60
  266. package/lib/rules/no-sparse-arrays.js +46 -28
  267. package/lib/rules/no-sync.js +61 -44
  268. package/lib/rules/no-tabs.js +83 -54
  269. package/lib/rules/no-template-curly-in-string.js +33 -32
  270. package/lib/rules/no-ternary.js +25 -28
  271. package/lib/rules/no-this-before-super.js +332 -298
  272. package/lib/rules/no-throw-literal.js +31 -36
  273. package/lib/rules/no-trailing-spaces.js +208 -174
  274. package/lib/rules/no-unassigned-vars.js +80 -0
  275. package/lib/rules/no-undef-init.js +86 -60
  276. package/lib/rules/no-undef.js +52 -47
  277. package/lib/rules/no-undefined.js +73 -74
  278. package/lib/rules/no-underscore-dangle.js +370 -322
  279. package/lib/rules/no-unexpected-multiline.js +112 -102
  280. package/lib/rules/no-unmodified-loop-condition.js +254 -254
  281. package/lib/rules/no-unneeded-ternary.js +212 -146
  282. package/lib/rules/no-unreachable-loop.js +145 -140
  283. package/lib/rules/no-unreachable.js +255 -248
  284. package/lib/rules/no-unsafe-finally.js +93 -85
  285. package/lib/rules/no-unsafe-negation.js +105 -81
  286. package/lib/rules/no-unsafe-optional-chaining.js +193 -177
  287. package/lib/rules/no-unused-expressions.js +199 -158
  288. package/lib/rules/no-unused-labels.js +139 -124
  289. package/lib/rules/no-unused-private-class-members.js +206 -182
  290. package/lib/rules/no-unused-vars.js +1708 -687
  291. package/lib/rules/no-use-before-define.js +327 -229
  292. package/lib/rules/no-useless-assignment.js +654 -0
  293. package/lib/rules/no-useless-backreference.js +212 -143
  294. package/lib/rules/no-useless-call.js +58 -53
  295. package/lib/rules/no-useless-catch.js +40 -40
  296. package/lib/rules/no-useless-computed-key.js +144 -108
  297. package/lib/rules/no-useless-concat.js +65 -59
  298. package/lib/rules/no-useless-constructor.js +160 -97
  299. package/lib/rules/no-useless-escape.js +364 -291
  300. package/lib/rules/no-useless-rename.js +183 -153
  301. package/lib/rules/no-useless-return.js +344 -307
  302. package/lib/rules/no-var.js +245 -212
  303. package/lib/rules/no-void.js +51 -46
  304. package/lib/rules/no-warning-comments.js +191 -183
  305. package/lib/rules/no-whitespace-before-property.js +131 -97
  306. package/lib/rules/no-with.js +24 -26
  307. package/lib/rules/nonblock-statement-body-position.js +149 -112
  308. package/lib/rules/object-curly-newline.js +306 -247
  309. package/lib/rules/object-curly-spacing.js +360 -296
  310. package/lib/rules/object-property-newline.js +137 -88
  311. package/lib/rules/object-shorthand.js +632 -500
  312. package/lib/rules/one-var-declaration-per-line.js +104 -82
  313. package/lib/rules/one-var.js +686 -536
  314. package/lib/rules/operator-assignment.js +219 -158
  315. package/lib/rules/operator-linebreak.js +295 -233
  316. package/lib/rules/padded-blocks.js +346 -290
  317. package/lib/rules/padding-line-between-statements.js +443 -421
  318. package/lib/rules/prefer-arrow-callback.js +371 -315
  319. package/lib/rules/prefer-const.js +418 -373
  320. package/lib/rules/prefer-destructuring.js +309 -278
  321. package/lib/rules/prefer-exponentiation-operator.js +176 -132
  322. package/lib/rules/prefer-named-capture-group.js +160 -141
  323. package/lib/rules/prefer-numeric-literals.js +121 -112
  324. package/lib/rules/prefer-object-has-own.js +116 -82
  325. package/lib/rules/prefer-object-spread.js +214 -193
  326. package/lib/rules/prefer-promise-reject-errors.js +140 -118
  327. package/lib/rules/prefer-reflect.js +126 -103
  328. package/lib/rules/prefer-regex-literals.js +561 -463
  329. package/lib/rules/prefer-rest-params.js +79 -80
  330. package/lib/rules/prefer-spread.js +47 -43
  331. package/lib/rules/prefer-template.js +266 -194
  332. package/lib/rules/preserve-caught-error.js +535 -0
  333. package/lib/rules/quote-props.js +373 -289
  334. package/lib/rules/quotes.js +374 -308
  335. package/lib/rules/radix.js +152 -134
  336. package/lib/rules/require-atomic-updates.js +316 -282
  337. package/lib/rules/require-await.js +153 -82
  338. package/lib/rules/require-unicode-regexp.js +296 -108
  339. package/lib/rules/require-yield.js +53 -54
  340. package/lib/rules/rest-spread-spacing.js +128 -98
  341. package/lib/rules/semi-spacing.js +281 -232
  342. package/lib/rules/semi-style.js +176 -116
  343. package/lib/rules/semi.js +456 -418
  344. package/lib/rules/sort-imports.js +307 -229
  345. package/lib/rules/sort-keys.js +219 -181
  346. package/lib/rules/sort-vars.js +127 -91
  347. package/lib/rules/space-before-blocks.js +199 -171
  348. package/lib/rules/space-before-function-paren.js +186 -148
  349. package/lib/rules/space-in-parens.js +359 -270
  350. package/lib/rules/space-infix-ops.js +237 -183
  351. package/lib/rules/space-unary-ops.js +356 -280
  352. package/lib/rules/spaced-comment.js +363 -301
  353. package/lib/rules/strict.js +266 -229
  354. package/lib/rules/switch-colon-spacing.js +130 -104
  355. package/lib/rules/symbol-description.js +45 -48
  356. package/lib/rules/template-curly-spacing.js +148 -124
  357. package/lib/rules/template-tag-spacing.js +98 -70
  358. package/lib/rules/unicode-bom.js +54 -54
  359. package/lib/rules/use-isnan.js +237 -110
  360. package/lib/rules/utils/ast-utils.js +2139 -1688
  361. package/lib/rules/utils/char-source.js +247 -0
  362. package/lib/rules/utils/fix-tracker.js +99 -88
  363. package/lib/rules/utils/keywords.js +59 -59
  364. package/lib/rules/utils/lazy-loading-rule-map.js +81 -78
  365. package/lib/rules/utils/regular-expressions.js +35 -19
  366. package/lib/rules/utils/unicode/index.js +9 -4
  367. package/lib/rules/utils/unicode/is-combining-character.js +1 -1
  368. package/lib/rules/utils/unicode/is-emoji-modifier.js +1 -1
  369. package/lib/rules/utils/unicode/is-regional-indicator-symbol.js +1 -1
  370. package/lib/rules/utils/unicode/is-surrogate-pair.js +1 -1
  371. package/lib/rules/valid-typeof.js +153 -109
  372. package/lib/rules/vars-on-top.js +152 -144
  373. package/lib/rules/wrap-iife.js +204 -173
  374. package/lib/rules/wrap-regex.js +77 -47
  375. package/lib/rules/yield-star-spacing.js +145 -116
  376. package/lib/rules/yoda.js +283 -274
  377. package/lib/services/parser-service.js +65 -0
  378. package/lib/services/processor-service.js +101 -0
  379. package/lib/services/suppressions-service.js +302 -0
  380. package/lib/services/warning-service.js +98 -0
  381. package/lib/shared/ajv.js +14 -14
  382. package/lib/shared/assert.js +21 -0
  383. package/lib/shared/ast-utils.js +7 -6
  384. package/lib/shared/deep-merge-arrays.js +62 -0
  385. package/lib/shared/directives.js +3 -2
  386. package/lib/shared/flags.js +108 -0
  387. package/lib/shared/logging.js +24 -16
  388. package/lib/shared/naming.js +109 -0
  389. package/lib/shared/option-utils.js +63 -0
  390. package/lib/shared/relative-module-resolver.js +18 -40
  391. package/lib/shared/runtime-info.js +138 -128
  392. package/lib/shared/serialization.js +78 -0
  393. package/lib/shared/severity.js +22 -22
  394. package/lib/shared/stats.js +30 -0
  395. package/lib/shared/string-utils.js +19 -21
  396. package/lib/shared/text-table.js +68 -0
  397. package/lib/shared/translate-cli-options.js +281 -0
  398. package/lib/shared/traverser.js +153 -146
  399. package/lib/types/config-api.d.ts +12 -0
  400. package/lib/types/index.d.ts +1473 -0
  401. package/lib/types/rules.d.ts +5589 -0
  402. package/lib/types/universal.d.ts +6 -0
  403. package/lib/types/use-at-your-own-risk.d.ts +87 -0
  404. package/lib/universal.js +10 -0
  405. package/lib/unsupported-api.js +8 -9
  406. package/messages/all-files-ignored.js +3 -3
  407. package/messages/all-matched-files-ignored.js +21 -0
  408. package/messages/config-file-missing.js +16 -0
  409. package/messages/config-plugin-missing.js +14 -0
  410. package/messages/config-serialize-function.js +30 -0
  411. package/messages/eslintrc-incompat.js +35 -16
  412. package/messages/eslintrc-plugins.js +8 -5
  413. package/messages/extend-config-missing.js +3 -3
  414. package/messages/failed-to-read-json.js +3 -3
  415. package/messages/file-not-found.js +3 -3
  416. package/messages/invalid-rule-options.js +2 -2
  417. package/messages/invalid-rule-severity.js +2 -2
  418. package/messages/no-config-found.js +4 -4
  419. package/messages/plugin-conflict.js +9 -9
  420. package/messages/plugin-invalid.js +4 -4
  421. package/messages/plugin-missing.js +4 -4
  422. package/messages/print-config-with-directory-path.js +2 -2
  423. package/messages/shared.js +6 -1
  424. package/messages/whitespace-found.js +3 -3
  425. package/package.json +105 -60
  426. package/conf/config-schema.js +0 -93
  427. package/lib/cli-engine/formatters/checkstyle.js +0 -60
  428. package/lib/cli-engine/formatters/compact.js +0 -60
  429. package/lib/cli-engine/formatters/jslint-xml.js +0 -41
  430. package/lib/cli-engine/formatters/junit.js +0 -82
  431. package/lib/cli-engine/formatters/tap.js +0 -95
  432. package/lib/cli-engine/formatters/unix.js +0 -58
  433. package/lib/cli-engine/formatters/visualstudio.js +0 -63
  434. package/lib/cli-engine/xml-escape.js +0 -34
  435. package/lib/config/flat-config-helpers.js +0 -111
  436. package/lib/config/rule-validator.js +0 -158
  437. package/lib/eslint/flat-eslint.js +0 -1159
  438. package/lib/linter/config-comment-parser.js +0 -185
  439. package/lib/linter/node-event-generator.js +0 -354
  440. package/lib/linter/report-translator.js +0 -369
  441. package/lib/linter/safe-emitter.js +0 -52
  442. package/lib/rule-tester/flat-rule-tester.js +0 -1131
  443. package/lib/rules/require-jsdoc.js +0 -122
  444. package/lib/rules/utils/patterns/letters.js +0 -36
  445. package/lib/rules/valid-jsdoc.js +0 -516
  446. package/lib/shared/config-validator.js +0 -347
  447. package/lib/shared/deprecation-warnings.js +0 -58
  448. package/lib/shared/types.js +0 -216
  449. package/lib/source-code/index.js +0 -5
  450. package/lib/source-code/source-code.js +0 -1055
  451. package/lib/source-code/token-store/backward-token-comment-cursor.js +0 -57
  452. package/lib/source-code/token-store/backward-token-cursor.js +0 -58
  453. package/lib/source-code/token-store/cursors.js +0 -90
  454. package/lib/source-code/token-store/forward-token-comment-cursor.js +0 -57
  455. package/lib/source-code/token-store/forward-token-cursor.js +0 -63
  456. package/lib/source-code/token-store/index.js +0 -627
  457. package/lib/source-code/token-store/padded-token-cursor.js +0 -38
  458. package/lib/source-code/token-store/utils.js +0 -107
@@ -0,0 +1,786 @@
1
+ /**
2
+ * @fileoverview Main API Class
3
+ * @author Kai Cataldo
4
+ * @author Toru Nagashima
5
+ */
6
+
7
+ "use strict";
8
+
9
+ //------------------------------------------------------------------------------
10
+ // Requirements
11
+ //------------------------------------------------------------------------------
12
+
13
+ const path = require("node:path");
14
+ const fs = require("node:fs");
15
+ const { promisify } = require("node:util");
16
+ const {
17
+ CLIEngine,
18
+ getCLIEngineInternalSlots,
19
+ } = require("../cli-engine/cli-engine");
20
+ const BuiltinRules = require("../rules");
21
+ const {
22
+ Legacy: {
23
+ ConfigOps: { getRuleSeverity },
24
+ },
25
+ } = require("@eslint/eslintrc");
26
+ const { version } = require("../../package.json");
27
+
28
+ //------------------------------------------------------------------------------
29
+ // Typedefs
30
+ //------------------------------------------------------------------------------
31
+
32
+ /** @typedef {import("../cli-engine/cli-engine").LintReport} CLIEngineLintReport */
33
+ /** @typedef {import("../types").ESLint.ConfigData} ConfigData */
34
+ /** @typedef {import("../types").ESLint.DeprecatedRuleUse} DeprecatedRuleInfo */
35
+ /** @typedef {import("../types").Linter.LintMessage} LintMessage */
36
+ /** @typedef {import("../types").ESLint.LintResult} LintResult */
37
+ /** @typedef {import("../types").ESLint.Plugin} Plugin */
38
+ /** @typedef {import("../types").ESLint.ResultsMeta} ResultsMeta */
39
+ /** @typedef {import("../types").Rule.RuleModule} Rule */
40
+ /** @typedef {import("../types").Linter.SuppressedLintMessage} SuppressedLintMessage */
41
+
42
+ /**
43
+ * The main formatter object.
44
+ * @typedef LoadedFormatter
45
+ * @property {(results: LintResult[], resultsMeta: ResultsMeta) => string | Promise<string>} format format function.
46
+ */
47
+
48
+ /**
49
+ * The options with which to configure the LegacyESLint instance.
50
+ * @typedef {Object} LegacyESLintOptions
51
+ * @property {boolean} [allowInlineConfig] Enable or disable inline configuration comments.
52
+ * @property {ConfigData} [baseConfig] Base config object, extended by all configs used with this instance
53
+ * @property {boolean} [cache] Enable result caching.
54
+ * @property {string} [cacheLocation] The cache file to use instead of .eslintcache.
55
+ * @property {"metadata" | "content"} [cacheStrategy] The strategy used to detect changed files.
56
+ * @property {string} [cwd] The value to use for the current working directory.
57
+ * @property {boolean} [errorOnUnmatchedPattern] If `false` then `ESLint#lintFiles()` doesn't throw even if no target files found. Defaults to `true`.
58
+ * @property {string[]} [extensions] An array of file extensions to check.
59
+ * @property {boolean|Function} [fix] Execute in autofix mode. If a function, should return a boolean.
60
+ * @property {string[]} [fixTypes] Array of rule types to apply fixes for.
61
+ * @property {boolean} [globInputPaths] Set to false to skip glob resolution of input file paths to lint (default: true). If false, each input file paths is assumed to be a non-glob path to an existing file.
62
+ * @property {boolean} [ignore] False disables use of .eslintignore.
63
+ * @property {string} [ignorePath] The ignore file to use instead of .eslintignore.
64
+ * @property {ConfigData} [overrideConfig] Override config object, overrides all configs used with this instance
65
+ * @property {string} [overrideConfigFile] The configuration file to use.
66
+ * @property {Record<string,Plugin>|null} [plugins] Preloaded plugins. This is a map-like object, keys are plugin IDs and each value is implementation.
67
+ * @property {"error" | "warn" | "off"} [reportUnusedDisableDirectives] the severity to report unused eslint-disable directives.
68
+ * @property {string} [resolvePluginsRelativeTo] The folder where plugins should be resolved from, defaulting to the CWD.
69
+ * @property {string[]} [rulePaths] An array of directories to load custom rules from.
70
+ * @property {boolean} [useEslintrc] False disables looking for .eslintrc.* files.
71
+ * @property {boolean} [passOnNoPatterns=false] When set to true, missing patterns cause
72
+ * the linting operation to short circuit and not report any failures.
73
+ */
74
+
75
+ /**
76
+ * A rules metadata object.
77
+ * @typedef {Object} RulesMeta
78
+ * @property {string} id The plugin ID.
79
+ * @property {Object} definition The plugin definition.
80
+ */
81
+
82
+ /**
83
+ * Private members for the `ESLint` instance.
84
+ * @typedef {Object} ESLintPrivateMembers
85
+ * @property {CLIEngine} cliEngine The wrapped CLIEngine instance.
86
+ * @property {LegacyESLintOptions} options The options used to instantiate the ESLint instance.
87
+ */
88
+
89
+ //------------------------------------------------------------------------------
90
+ // Helpers
91
+ //------------------------------------------------------------------------------
92
+
93
+ const writeFile = promisify(fs.writeFile);
94
+
95
+ /**
96
+ * The map with which to store private class members.
97
+ * @type {WeakMap<ESLint, ESLintPrivateMembers>}
98
+ */
99
+ const privateMembersMap = new WeakMap();
100
+
101
+ /**
102
+ * Check if a given value is a non-empty string or not.
103
+ * @param {any} value The value to check.
104
+ * @returns {boolean} `true` if `value` is a non-empty string.
105
+ */
106
+ function isNonEmptyString(value) {
107
+ return typeof value === "string" && value.trim() !== "";
108
+ }
109
+
110
+ /**
111
+ * Check if a given value is an array of non-empty strings or not.
112
+ * @param {any} value The value to check.
113
+ * @returns {boolean} `true` if `value` is an array of non-empty strings.
114
+ */
115
+ function isArrayOfNonEmptyString(value) {
116
+ return (
117
+ Array.isArray(value) && value.length && value.every(isNonEmptyString)
118
+ );
119
+ }
120
+
121
+ /**
122
+ * Check if a given value is an empty array or an array of non-empty strings.
123
+ * @param {any} value The value to check.
124
+ * @returns {boolean} `true` if `value` is an empty array or an array of non-empty
125
+ * strings.
126
+ */
127
+ function isEmptyArrayOrArrayOfNonEmptyString(value) {
128
+ return Array.isArray(value) && value.every(isNonEmptyString);
129
+ }
130
+
131
+ /**
132
+ * Check if a given value is a valid fix type or not.
133
+ * @param {any} value The value to check.
134
+ * @returns {boolean} `true` if `value` is valid fix type.
135
+ */
136
+ function isFixType(value) {
137
+ return (
138
+ value === "directive" ||
139
+ value === "problem" ||
140
+ value === "suggestion" ||
141
+ value === "layout"
142
+ );
143
+ }
144
+
145
+ /**
146
+ * Check if a given value is an array of fix types or not.
147
+ * @param {any} value The value to check.
148
+ * @returns {boolean} `true` if `value` is an array of fix types.
149
+ */
150
+ function isFixTypeArray(value) {
151
+ return Array.isArray(value) && value.every(isFixType);
152
+ }
153
+
154
+ /**
155
+ * The error for invalid options.
156
+ */
157
+ class ESLintInvalidOptionsError extends Error {
158
+ constructor(messages) {
159
+ super(`Invalid Options:\n- ${messages.join("\n- ")}`);
160
+ this.code = "ESLINT_INVALID_OPTIONS";
161
+ Error.captureStackTrace(this, ESLintInvalidOptionsError);
162
+ }
163
+ }
164
+
165
+ /**
166
+ * Validates and normalizes options for the wrapped CLIEngine instance.
167
+ * @param {LegacyESLintOptions} options The options to process.
168
+ * @throws {ESLintInvalidOptionsError} If of any of a variety of type errors.
169
+ * @returns {LegacyESLintOptions} The normalized options.
170
+ */
171
+ function processOptions({
172
+ allowInlineConfig = true, // ← we cannot use `overrideConfig.noInlineConfig` instead because `allowInlineConfig` has side-effect that suppress warnings that show inline configs are ignored.
173
+ baseConfig = null,
174
+ cache = false,
175
+ cacheLocation = ".eslintcache",
176
+ cacheStrategy = "metadata",
177
+ cwd = process.cwd(),
178
+ errorOnUnmatchedPattern = true,
179
+ extensions = null, // ← should be null by default because if it's an array then it suppresses RFC20 feature.
180
+ fix = false,
181
+ fixTypes = null, // ← should be null by default because if it's an array then it suppresses rules that don't have the `meta.type` property.
182
+ flags /* eslint-disable-line no-unused-vars -- leaving for compatibility with ESLint#hasFlag */,
183
+ globInputPaths = true,
184
+ ignore = true,
185
+ ignorePath = null, // ← should be null by default because if it's a string then it may throw ENOENT.
186
+ overrideConfig = null,
187
+ overrideConfigFile = null,
188
+ plugins = {},
189
+ reportUnusedDisableDirectives = null, // ← should be null by default because if it's a string then it overrides the 'reportUnusedDisableDirectives' setting in config files. And we cannot use `overrideConfig.reportUnusedDisableDirectives` instead because we cannot configure the `error` severity with that.
190
+ resolvePluginsRelativeTo = null, // ← should be null by default because if it's a string then it suppresses RFC47 feature.
191
+ rulePaths = [],
192
+ useEslintrc = true,
193
+ passOnNoPatterns = false,
194
+ ...unknownOptions
195
+ }) {
196
+ const errors = [];
197
+ const unknownOptionKeys = Object.keys(unknownOptions);
198
+
199
+ if (unknownOptionKeys.length >= 1) {
200
+ errors.push(`Unknown options: ${unknownOptionKeys.join(", ")}`);
201
+ if (unknownOptionKeys.includes("cacheFile")) {
202
+ errors.push(
203
+ "'cacheFile' has been removed. Please use the 'cacheLocation' option instead.",
204
+ );
205
+ }
206
+ if (unknownOptionKeys.includes("configFile")) {
207
+ errors.push(
208
+ "'configFile' has been removed. Please use the 'overrideConfigFile' option instead.",
209
+ );
210
+ }
211
+ if (unknownOptionKeys.includes("envs")) {
212
+ errors.push(
213
+ "'envs' has been removed. Please use the 'overrideConfig.env' option instead.",
214
+ );
215
+ }
216
+ if (unknownOptionKeys.includes("globals")) {
217
+ errors.push(
218
+ "'globals' has been removed. Please use the 'overrideConfig.globals' option instead.",
219
+ );
220
+ }
221
+ if (unknownOptionKeys.includes("ignorePattern")) {
222
+ errors.push(
223
+ "'ignorePattern' has been removed. Please use the 'overrideConfig.ignorePatterns' option instead.",
224
+ );
225
+ }
226
+ if (unknownOptionKeys.includes("parser")) {
227
+ errors.push(
228
+ "'parser' has been removed. Please use the 'overrideConfig.parser' option instead.",
229
+ );
230
+ }
231
+ if (unknownOptionKeys.includes("parserOptions")) {
232
+ errors.push(
233
+ "'parserOptions' has been removed. Please use the 'overrideConfig.parserOptions' option instead.",
234
+ );
235
+ }
236
+ if (unknownOptionKeys.includes("rules")) {
237
+ errors.push(
238
+ "'rules' has been removed. Please use the 'overrideConfig.rules' option instead.",
239
+ );
240
+ }
241
+ }
242
+ if (typeof allowInlineConfig !== "boolean") {
243
+ errors.push("'allowInlineConfig' must be a boolean.");
244
+ }
245
+ if (typeof baseConfig !== "object") {
246
+ errors.push("'baseConfig' must be an object or null.");
247
+ }
248
+ if (typeof cache !== "boolean") {
249
+ errors.push("'cache' must be a boolean.");
250
+ }
251
+ if (!isNonEmptyString(cacheLocation)) {
252
+ errors.push("'cacheLocation' must be a non-empty string.");
253
+ }
254
+ if (cacheStrategy !== "metadata" && cacheStrategy !== "content") {
255
+ errors.push('\'cacheStrategy\' must be any of "metadata", "content".');
256
+ }
257
+ if (!isNonEmptyString(cwd) || !path.isAbsolute(cwd)) {
258
+ errors.push("'cwd' must be an absolute path.");
259
+ }
260
+ if (typeof errorOnUnmatchedPattern !== "boolean") {
261
+ errors.push("'errorOnUnmatchedPattern' must be a boolean.");
262
+ }
263
+ if (
264
+ !isEmptyArrayOrArrayOfNonEmptyString(extensions) &&
265
+ extensions !== null
266
+ ) {
267
+ errors.push(
268
+ "'extensions' must be an array of non-empty strings or null.",
269
+ );
270
+ }
271
+ if (typeof fix !== "boolean" && typeof fix !== "function") {
272
+ errors.push("'fix' must be a boolean or a function.");
273
+ }
274
+ if (fixTypes !== null && !isFixTypeArray(fixTypes)) {
275
+ errors.push(
276
+ '\'fixTypes\' must be an array of any of "directive", "problem", "suggestion", and "layout".',
277
+ );
278
+ }
279
+ if (typeof globInputPaths !== "boolean") {
280
+ errors.push("'globInputPaths' must be a boolean.");
281
+ }
282
+ if (typeof ignore !== "boolean") {
283
+ errors.push("'ignore' must be a boolean.");
284
+ }
285
+ if (!isNonEmptyString(ignorePath) && ignorePath !== null) {
286
+ errors.push("'ignorePath' must be a non-empty string or null.");
287
+ }
288
+ if (typeof overrideConfig !== "object") {
289
+ errors.push("'overrideConfig' must be an object or null.");
290
+ }
291
+ if (!isNonEmptyString(overrideConfigFile) && overrideConfigFile !== null) {
292
+ errors.push("'overrideConfigFile' must be a non-empty string or null.");
293
+ }
294
+ if (typeof plugins !== "object") {
295
+ errors.push("'plugins' must be an object or null.");
296
+ } else if (plugins !== null && Object.keys(plugins).includes("")) {
297
+ errors.push("'plugins' must not include an empty string.");
298
+ }
299
+ if (Array.isArray(plugins)) {
300
+ errors.push(
301
+ "'plugins' doesn't add plugins to configuration to load. Please use the 'overrideConfig.plugins' option instead.",
302
+ );
303
+ }
304
+ if (
305
+ reportUnusedDisableDirectives !== "error" &&
306
+ reportUnusedDisableDirectives !== "warn" &&
307
+ reportUnusedDisableDirectives !== "off" &&
308
+ reportUnusedDisableDirectives !== null
309
+ ) {
310
+ errors.push(
311
+ '\'reportUnusedDisableDirectives\' must be any of "error", "warn", "off", and null.',
312
+ );
313
+ }
314
+ if (
315
+ !isNonEmptyString(resolvePluginsRelativeTo) &&
316
+ resolvePluginsRelativeTo !== null
317
+ ) {
318
+ errors.push(
319
+ "'resolvePluginsRelativeTo' must be a non-empty string or null.",
320
+ );
321
+ }
322
+ if (!isEmptyArrayOrArrayOfNonEmptyString(rulePaths)) {
323
+ errors.push("'rulePaths' must be an array of non-empty strings.");
324
+ }
325
+ if (typeof useEslintrc !== "boolean") {
326
+ errors.push("'useEslintrc' must be a boolean.");
327
+ }
328
+ if (typeof passOnNoPatterns !== "boolean") {
329
+ errors.push("'passOnNoPatterns' must be a boolean.");
330
+ }
331
+
332
+ if (errors.length > 0) {
333
+ throw new ESLintInvalidOptionsError(errors);
334
+ }
335
+
336
+ return {
337
+ allowInlineConfig,
338
+ baseConfig,
339
+ cache,
340
+ cacheLocation,
341
+ cacheStrategy,
342
+ configFile: overrideConfigFile,
343
+ cwd: path.normalize(cwd),
344
+ errorOnUnmatchedPattern,
345
+ extensions,
346
+ fix,
347
+ fixTypes,
348
+ flags: [], // LegacyESLint does not support flags, so just ignore them.
349
+ globInputPaths,
350
+ ignore,
351
+ ignorePath,
352
+ reportUnusedDisableDirectives,
353
+ resolvePluginsRelativeTo,
354
+ rulePaths,
355
+ useEslintrc,
356
+ passOnNoPatterns,
357
+ };
358
+ }
359
+
360
+ /**
361
+ * Check if a value has one or more properties and that value is not undefined.
362
+ * @param {any} obj The value to check.
363
+ * @returns {boolean} `true` if `obj` has one or more properties that value is not undefined.
364
+ */
365
+ function hasDefinedProperty(obj) {
366
+ if (typeof obj === "object" && obj !== null) {
367
+ for (const key in obj) {
368
+ if (typeof obj[key] !== "undefined") {
369
+ return true;
370
+ }
371
+ }
372
+ }
373
+ return false;
374
+ }
375
+
376
+ /**
377
+ * Create rulesMeta object.
378
+ * @param {Map<string,Rule>} rules a map of rules from which to generate the object.
379
+ * @returns {Object} metadata for all enabled rules.
380
+ */
381
+ function createRulesMeta(rules) {
382
+ return Array.from(rules).reduce((retVal, [id, rule]) => {
383
+ retVal[id] = rule.meta;
384
+ return retVal;
385
+ }, {});
386
+ }
387
+
388
+ /** @type {WeakMap<ExtractedConfig, DeprecatedRuleInfo[]>} */
389
+ const usedDeprecatedRulesCache = new WeakMap();
390
+
391
+ /**
392
+ * Create used deprecated rule list.
393
+ * @param {CLIEngine} cliEngine The CLIEngine instance.
394
+ * @param {string} maybeFilePath The absolute path to a lint target file or `"<text>"`.
395
+ * @returns {DeprecatedRuleInfo[]} The used deprecated rule list.
396
+ */
397
+ function getOrFindUsedDeprecatedRules(cliEngine, maybeFilePath) {
398
+ const {
399
+ configArrayFactory,
400
+ options: { cwd },
401
+ } = getCLIEngineInternalSlots(cliEngine);
402
+ const filePath = path.isAbsolute(maybeFilePath)
403
+ ? maybeFilePath
404
+ : path.join(cwd, "__placeholder__.js");
405
+ const configArray = configArrayFactory.getConfigArrayForFile(filePath);
406
+ const config = configArray.extractConfig(filePath);
407
+
408
+ // Most files use the same config, so cache it.
409
+ if (!usedDeprecatedRulesCache.has(config)) {
410
+ const pluginRules = configArray.pluginRules;
411
+ const retv = [];
412
+
413
+ for (const [ruleId, ruleConf] of Object.entries(config.rules)) {
414
+ if (getRuleSeverity(ruleConf) === 0) {
415
+ continue;
416
+ }
417
+ const rule = pluginRules.get(ruleId) || BuiltinRules.get(ruleId);
418
+ const meta = rule && rule.meta;
419
+
420
+ if (meta && meta.deprecated) {
421
+ retv.push({ ruleId, replacedBy: meta.replacedBy || [] });
422
+ }
423
+ }
424
+
425
+ usedDeprecatedRulesCache.set(config, Object.freeze(retv));
426
+ }
427
+
428
+ return usedDeprecatedRulesCache.get(config);
429
+ }
430
+
431
+ /**
432
+ * Processes the linting results generated by a CLIEngine linting report to
433
+ * match the ESLint class's API.
434
+ * @param {CLIEngine} cliEngine The CLIEngine instance.
435
+ * @param {CLIEngineLintReport} report The CLIEngine linting report to process.
436
+ * @returns {LintResult[]} The processed linting results.
437
+ */
438
+ function processCLIEngineLintReport(cliEngine, { results }) {
439
+ const descriptor = {
440
+ configurable: true,
441
+ enumerable: true,
442
+ get() {
443
+ return getOrFindUsedDeprecatedRules(cliEngine, this.filePath);
444
+ },
445
+ };
446
+
447
+ for (const result of results) {
448
+ Object.defineProperty(result, "usedDeprecatedRules", descriptor);
449
+ }
450
+
451
+ return results;
452
+ }
453
+
454
+ /**
455
+ * An Array.prototype.sort() compatible compare function to order results by their file path.
456
+ * @param {LintResult} a The first lint result.
457
+ * @param {LintResult} b The second lint result.
458
+ * @returns {number} An integer representing the order in which the two results should occur.
459
+ */
460
+ function compareResultsByFilePath(a, b) {
461
+ if (a.filePath < b.filePath) {
462
+ return -1;
463
+ }
464
+
465
+ if (a.filePath > b.filePath) {
466
+ return 1;
467
+ }
468
+
469
+ return 0;
470
+ }
471
+
472
+ /**
473
+ * Main API.
474
+ */
475
+ class LegacyESLint {
476
+ /**
477
+ * The type of configuration used by this class.
478
+ * @type {string}
479
+ */
480
+ static configType = "eslintrc";
481
+
482
+ /**
483
+ * Creates a new instance of the main ESLint API.
484
+ * @param {LegacyESLintOptions} options The options for this instance.
485
+ */
486
+ constructor(options = {}) {
487
+ const processedOptions = processOptions(options);
488
+ const cliEngine = new CLIEngine(processedOptions, {
489
+ preloadedPlugins: options.plugins,
490
+ });
491
+ const { configArrayFactory, lastConfigArrays } =
492
+ getCLIEngineInternalSlots(cliEngine);
493
+ let updated = false;
494
+
495
+ /*
496
+ * Address `overrideConfig` to set override config.
497
+ * Operate the `configArrayFactory` internal slot directly because this
498
+ * functionality doesn't exist as the public API of CLIEngine.
499
+ */
500
+ if (hasDefinedProperty(options.overrideConfig)) {
501
+ configArrayFactory.setOverrideConfig(options.overrideConfig);
502
+ updated = true;
503
+ }
504
+
505
+ // Update caches.
506
+ if (updated) {
507
+ configArrayFactory.clearCache();
508
+ lastConfigArrays[0] = configArrayFactory.getConfigArrayForFile();
509
+ }
510
+
511
+ // Initialize private properties.
512
+ privateMembersMap.set(this, {
513
+ cliEngine,
514
+ options: processedOptions,
515
+ });
516
+ }
517
+
518
+ /**
519
+ * The version text.
520
+ * @type {string}
521
+ */
522
+ static get version() {
523
+ return version;
524
+ }
525
+
526
+ /**
527
+ * Outputs fixes from the given results to files.
528
+ * @param {LintResult[]} results The lint results.
529
+ * @returns {Promise<void>} Returns a promise that is used to track side effects.
530
+ */
531
+ static async outputFixes(results) {
532
+ if (!Array.isArray(results)) {
533
+ throw new Error("'results' must be an array");
534
+ }
535
+
536
+ await Promise.all(
537
+ results
538
+ .filter(result => {
539
+ if (typeof result !== "object" || result === null) {
540
+ throw new Error("'results' must include only objects");
541
+ }
542
+ return (
543
+ typeof result.output === "string" &&
544
+ path.isAbsolute(result.filePath)
545
+ );
546
+ })
547
+ .map(r => writeFile(r.filePath, r.output)),
548
+ );
549
+ }
550
+
551
+ /**
552
+ * Returns results that only contains errors.
553
+ * @param {LintResult[]} results The results to filter.
554
+ * @returns {LintResult[]} The filtered results.
555
+ */
556
+ static getErrorResults(results) {
557
+ return CLIEngine.getErrorResults(results);
558
+ }
559
+
560
+ /**
561
+ * Returns meta objects for each rule represented in the lint results.
562
+ * @param {LintResult[]} results The results to fetch rules meta for.
563
+ * @returns {Object} A mapping of ruleIds to rule meta objects.
564
+ */
565
+ getRulesMetaForResults(results) {
566
+ const resultRuleIds = new Set();
567
+
568
+ // first gather all ruleIds from all results
569
+
570
+ for (const result of results) {
571
+ for (const { ruleId } of result.messages) {
572
+ resultRuleIds.add(ruleId);
573
+ }
574
+ for (const { ruleId } of result.suppressedMessages) {
575
+ resultRuleIds.add(ruleId);
576
+ }
577
+ }
578
+
579
+ // create a map of all rules in the results
580
+
581
+ const { cliEngine } = privateMembersMap.get(this);
582
+ const rules = cliEngine.getRules();
583
+ const resultRules = new Map();
584
+
585
+ for (const [ruleId, rule] of rules) {
586
+ if (resultRuleIds.has(ruleId)) {
587
+ resultRules.set(ruleId, rule);
588
+ }
589
+ }
590
+
591
+ return createRulesMeta(resultRules);
592
+ }
593
+
594
+ /* eslint-disable no-unused-vars, class-methods-use-this -- leaving for compatibility with ESLint#hasFlag */
595
+ /**
596
+ * Indicates if the given feature flag is enabled for this instance. For this
597
+ * class, this always returns `false` because it does not support feature flags.
598
+ * @param {string} flag The feature flag to check.
599
+ * @returns {boolean} Always false.
600
+ */
601
+ hasFlag(flag) {
602
+ return false;
603
+ }
604
+ /* eslint-enable no-unused-vars, class-methods-use-this -- reenable rules for the rest of the file */
605
+
606
+ /**
607
+ * Executes the current configuration on an array of file and directory names.
608
+ * @param {string[]} patterns An array of file and directory names.
609
+ * @returns {Promise<LintResult[]>} The results of linting the file patterns given.
610
+ */
611
+ async lintFiles(patterns) {
612
+ const { cliEngine, options } = privateMembersMap.get(this);
613
+
614
+ if (
615
+ options.passOnNoPatterns &&
616
+ (patterns === "" ||
617
+ (Array.isArray(patterns) && patterns.length === 0))
618
+ ) {
619
+ return [];
620
+ }
621
+
622
+ if (!isNonEmptyString(patterns) && !isArrayOfNonEmptyString(patterns)) {
623
+ throw new Error(
624
+ "'patterns' must be a non-empty string or an array of non-empty strings",
625
+ );
626
+ }
627
+
628
+ return processCLIEngineLintReport(
629
+ cliEngine,
630
+ cliEngine.executeOnFiles(patterns),
631
+ );
632
+ }
633
+
634
+ /**
635
+ * Executes the current configuration on text.
636
+ * @param {string} code A string of JavaScript code to lint.
637
+ * @param {Object} [options] The options.
638
+ * @param {string} [options.filePath] The path to the file of the source code.
639
+ * @param {boolean} [options.warnIgnored] When set to true, warn if given filePath is an ignored path.
640
+ * @returns {Promise<LintResult[]>} The results of linting the string of code given.
641
+ */
642
+ async lintText(code, options = {}) {
643
+ if (typeof code !== "string") {
644
+ throw new Error("'code' must be a string");
645
+ }
646
+ if (typeof options !== "object") {
647
+ throw new Error("'options' must be an object, null, or undefined");
648
+ }
649
+ const {
650
+ filePath,
651
+ warnIgnored = false,
652
+ ...unknownOptions
653
+ } = options || {};
654
+
655
+ const unknownOptionKeys = Object.keys(unknownOptions);
656
+
657
+ if (unknownOptionKeys.length > 0) {
658
+ throw new Error(
659
+ `'options' must not include the unknown option(s): ${unknownOptionKeys.join(", ")}`,
660
+ );
661
+ }
662
+
663
+ if (filePath !== void 0 && !isNonEmptyString(filePath)) {
664
+ throw new Error(
665
+ "'options.filePath' must be a non-empty string or undefined",
666
+ );
667
+ }
668
+ if (typeof warnIgnored !== "boolean") {
669
+ throw new Error(
670
+ "'options.warnIgnored' must be a boolean or undefined",
671
+ );
672
+ }
673
+
674
+ const { cliEngine } = privateMembersMap.get(this);
675
+
676
+ return processCLIEngineLintReport(
677
+ cliEngine,
678
+ cliEngine.executeOnText(code, filePath, warnIgnored),
679
+ );
680
+ }
681
+
682
+ /**
683
+ * Returns the formatter representing the given formatter name.
684
+ * @param {string} [name] The name of the formatter to load.
685
+ * The following values are allowed:
686
+ * - `undefined` ... Load `stylish` builtin formatter.
687
+ * - A builtin formatter name ... Load the builtin formatter.
688
+ * - A third-party formatter name:
689
+ * - `foo` → `eslint-formatter-foo`
690
+ * - `@foo` → `@foo/eslint-formatter`
691
+ * - `@foo/bar` → `@foo/eslint-formatter-bar`
692
+ * - A file path ... Load the file.
693
+ * @returns {Promise<LoadedFormatter>} A promise resolving to the formatter object.
694
+ * This promise will be rejected if the given formatter was not found or not
695
+ * a function.
696
+ */
697
+ async loadFormatter(name = "stylish") {
698
+ if (typeof name !== "string") {
699
+ throw new Error("'name' must be a string");
700
+ }
701
+
702
+ const { cliEngine, options } = privateMembersMap.get(this);
703
+ const formatter = cliEngine.getFormatter(name);
704
+
705
+ if (typeof formatter !== "function") {
706
+ throw new Error(
707
+ `Formatter must be a function, but got a ${typeof formatter}.`,
708
+ );
709
+ }
710
+
711
+ return {
712
+ /**
713
+ * The main formatter method.
714
+ * @param {LintResult[]} results The lint results to format.
715
+ * @param {ResultsMeta} resultsMeta Warning count and max threshold.
716
+ * @returns {string | Promise<string>} The formatted lint results.
717
+ */
718
+ format(results, resultsMeta) {
719
+ let rulesMeta = null;
720
+
721
+ results.sort(compareResultsByFilePath);
722
+
723
+ return formatter(results, {
724
+ ...resultsMeta,
725
+ get cwd() {
726
+ return options.cwd;
727
+ },
728
+ get rulesMeta() {
729
+ if (!rulesMeta) {
730
+ rulesMeta = createRulesMeta(cliEngine.getRules());
731
+ }
732
+
733
+ return rulesMeta;
734
+ },
735
+ });
736
+ },
737
+ };
738
+ }
739
+
740
+ /**
741
+ * Returns a configuration object for the given file based on the CLI options.
742
+ * This is the same logic used by the ESLint CLI executable to determine
743
+ * configuration for each file it processes.
744
+ * @param {string} filePath The path of the file to retrieve a config object for.
745
+ * @returns {Promise<ConfigData>} A configuration object for the file.
746
+ */
747
+ async calculateConfigForFile(filePath) {
748
+ if (!isNonEmptyString(filePath)) {
749
+ throw new Error("'filePath' must be a non-empty string");
750
+ }
751
+ const { cliEngine } = privateMembersMap.get(this);
752
+
753
+ return cliEngine.getConfigForFile(filePath);
754
+ }
755
+
756
+ /**
757
+ * Checks if a given path is ignored by ESLint.
758
+ * @param {string} filePath The path of the file to check.
759
+ * @returns {Promise<boolean>} Whether or not the given path is ignored.
760
+ */
761
+ async isPathIgnored(filePath) {
762
+ if (!isNonEmptyString(filePath)) {
763
+ throw new Error("'filePath' must be a non-empty string");
764
+ }
765
+ const { cliEngine } = privateMembersMap.get(this);
766
+
767
+ return cliEngine.isPathIgnored(filePath);
768
+ }
769
+ }
770
+
771
+ //------------------------------------------------------------------------------
772
+ // Public Interface
773
+ //------------------------------------------------------------------------------
774
+
775
+ module.exports = {
776
+ LegacyESLint,
777
+
778
+ /**
779
+ * Get the private class members of a given ESLint instance for tests.
780
+ * @param {ESLint} instance The ESLint instance to get.
781
+ * @returns {ESLintPrivateMembers} The instance's private class members.
782
+ */
783
+ getESLintPrivateMembers(instance) {
784
+ return privateMembersMap.get(instance);
785
+ },
786
+ };