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
@@ -1,1055 +0,0 @@
1
- /**
2
- * @fileoverview Abstraction of JavaScript source code.
3
- * @author Nicholas C. Zakas
4
- */
5
- "use strict";
6
-
7
- //------------------------------------------------------------------------------
8
- // Requirements
9
- //------------------------------------------------------------------------------
10
-
11
- const
12
- { isCommentToken } = require("@eslint-community/eslint-utils"),
13
- TokenStore = require("./token-store"),
14
- astUtils = require("../shared/ast-utils"),
15
- Traverser = require("../shared/traverser"),
16
- globals = require("../../conf/globals"),
17
- {
18
- directivesPattern
19
- } = require("../shared/directives"),
20
-
21
- /* eslint-disable-next-line n/no-restricted-require -- Too messy to figure out right now. */
22
- ConfigCommentParser = require("../linter/config-comment-parser"),
23
- eslintScope = require("eslint-scope");
24
-
25
- //------------------------------------------------------------------------------
26
- // Type Definitions
27
- //------------------------------------------------------------------------------
28
-
29
- /** @typedef {import("eslint-scope").Variable} Variable */
30
-
31
- //------------------------------------------------------------------------------
32
- // Private
33
- //------------------------------------------------------------------------------
34
-
35
- const commentParser = new ConfigCommentParser();
36
-
37
- /**
38
- * Validates that the given AST has the required information.
39
- * @param {ASTNode} ast The Program node of the AST to check.
40
- * @throws {Error} If the AST doesn't contain the correct information.
41
- * @returns {void}
42
- * @private
43
- */
44
- function validate(ast) {
45
- if (!ast.tokens) {
46
- throw new Error("AST is missing the tokens array.");
47
- }
48
-
49
- if (!ast.comments) {
50
- throw new Error("AST is missing the comments array.");
51
- }
52
-
53
- if (!ast.loc) {
54
- throw new Error("AST is missing location information.");
55
- }
56
-
57
- if (!ast.range) {
58
- throw new Error("AST is missing range information");
59
- }
60
- }
61
-
62
- /**
63
- * Retrieves globals for the given ecmaVersion.
64
- * @param {number} ecmaVersion The version to retrieve globals for.
65
- * @returns {Object} The globals for the given ecmaVersion.
66
- */
67
- function getGlobalsForEcmaVersion(ecmaVersion) {
68
-
69
- switch (ecmaVersion) {
70
- case 3:
71
- return globals.es3;
72
-
73
- case 5:
74
- return globals.es5;
75
-
76
- default:
77
- if (ecmaVersion < 2015) {
78
- return globals[`es${ecmaVersion + 2009}`];
79
- }
80
-
81
- return globals[`es${ecmaVersion}`];
82
- }
83
- }
84
-
85
- /**
86
- * Check to see if its a ES6 export declaration.
87
- * @param {ASTNode} astNode An AST node.
88
- * @returns {boolean} whether the given node represents an export declaration.
89
- * @private
90
- */
91
- function looksLikeExport(astNode) {
92
- return astNode.type === "ExportDefaultDeclaration" || astNode.type === "ExportNamedDeclaration" ||
93
- astNode.type === "ExportAllDeclaration" || astNode.type === "ExportSpecifier";
94
- }
95
-
96
- /**
97
- * Merges two sorted lists into a larger sorted list in O(n) time.
98
- * @param {Token[]} tokens The list of tokens.
99
- * @param {Token[]} comments The list of comments.
100
- * @returns {Token[]} A sorted list of tokens and comments.
101
- * @private
102
- */
103
- function sortedMerge(tokens, comments) {
104
- const result = [];
105
- let tokenIndex = 0;
106
- let commentIndex = 0;
107
-
108
- while (tokenIndex < tokens.length || commentIndex < comments.length) {
109
- if (commentIndex >= comments.length || tokenIndex < tokens.length && tokens[tokenIndex].range[0] < comments[commentIndex].range[0]) {
110
- result.push(tokens[tokenIndex++]);
111
- } else {
112
- result.push(comments[commentIndex++]);
113
- }
114
- }
115
-
116
- return result;
117
- }
118
-
119
- /**
120
- * Normalizes a value for a global in a config
121
- * @param {(boolean|string|null)} configuredValue The value given for a global in configuration or in
122
- * a global directive comment
123
- * @returns {("readable"|"writeable"|"off")} The value normalized as a string
124
- * @throws Error if global value is invalid
125
- */
126
- function normalizeConfigGlobal(configuredValue) {
127
- switch (configuredValue) {
128
- case "off":
129
- return "off";
130
-
131
- case true:
132
- case "true":
133
- case "writeable":
134
- case "writable":
135
- return "writable";
136
-
137
- case null:
138
- case false:
139
- case "false":
140
- case "readable":
141
- case "readonly":
142
- return "readonly";
143
-
144
- default:
145
- throw new Error(`'${configuredValue}' is not a valid configuration for a global (use 'readonly', 'writable', or 'off')`);
146
- }
147
- }
148
-
149
- /**
150
- * Determines if two nodes or tokens overlap.
151
- * @param {ASTNode|Token} first The first node or token to check.
152
- * @param {ASTNode|Token} second The second node or token to check.
153
- * @returns {boolean} True if the two nodes or tokens overlap.
154
- * @private
155
- */
156
- function nodesOrTokensOverlap(first, second) {
157
- return (first.range[0] <= second.range[0] && first.range[1] >= second.range[0]) ||
158
- (second.range[0] <= first.range[0] && second.range[1] >= first.range[0]);
159
- }
160
-
161
- /**
162
- * Determines if two nodes or tokens have at least one whitespace character
163
- * between them. Order does not matter. Returns false if the given nodes or
164
- * tokens overlap.
165
- * @param {SourceCode} sourceCode The source code object.
166
- * @param {ASTNode|Token} first The first node or token to check between.
167
- * @param {ASTNode|Token} second The second node or token to check between.
168
- * @param {boolean} checkInsideOfJSXText If `true` is present, check inside of JSXText tokens for backward compatibility.
169
- * @returns {boolean} True if there is a whitespace character between
170
- * any of the tokens found between the two given nodes or tokens.
171
- * @public
172
- */
173
- function isSpaceBetween(sourceCode, first, second, checkInsideOfJSXText) {
174
- if (nodesOrTokensOverlap(first, second)) {
175
- return false;
176
- }
177
-
178
- const [startingNodeOrToken, endingNodeOrToken] = first.range[1] <= second.range[0]
179
- ? [first, second]
180
- : [second, first];
181
- const firstToken = sourceCode.getLastToken(startingNodeOrToken) || startingNodeOrToken;
182
- const finalToken = sourceCode.getFirstToken(endingNodeOrToken) || endingNodeOrToken;
183
- let currentToken = firstToken;
184
-
185
- while (currentToken !== finalToken) {
186
- const nextToken = sourceCode.getTokenAfter(currentToken, { includeComments: true });
187
-
188
- if (
189
- currentToken.range[1] !== nextToken.range[0] ||
190
-
191
- /*
192
- * For backward compatibility, check spaces in JSXText.
193
- * https://github.com/eslint/eslint/issues/12614
194
- */
195
- (
196
- checkInsideOfJSXText &&
197
- nextToken !== finalToken &&
198
- nextToken.type === "JSXText" &&
199
- /\s/u.test(nextToken.value)
200
- )
201
- ) {
202
- return true;
203
- }
204
-
205
- currentToken = nextToken;
206
- }
207
-
208
- return false;
209
- }
210
-
211
- //-----------------------------------------------------------------------------
212
- // Directive Comments
213
- //-----------------------------------------------------------------------------
214
-
215
- /**
216
- * Ensures that variables representing built-in properties of the Global Object,
217
- * and any globals declared by special block comments, are present in the global
218
- * scope.
219
- * @param {Scope} globalScope The global scope.
220
- * @param {Object|undefined} configGlobals The globals declared in configuration
221
- * @param {Object|undefined} inlineGlobals The globals declared in the source code
222
- * @returns {void}
223
- */
224
- function addDeclaredGlobals(globalScope, configGlobals = {}, inlineGlobals = {}) {
225
-
226
- // Define configured global variables.
227
- for (const id of new Set([...Object.keys(configGlobals), ...Object.keys(inlineGlobals)])) {
228
-
229
- /*
230
- * `normalizeConfigGlobal` will throw an error if a configured global value is invalid. However, these errors would
231
- * typically be caught when validating a config anyway (validity for inline global comments is checked separately).
232
- */
233
- const configValue = configGlobals[id] === void 0 ? void 0 : normalizeConfigGlobal(configGlobals[id]);
234
- const commentValue = inlineGlobals[id] && inlineGlobals[id].value;
235
- const value = commentValue || configValue;
236
- const sourceComments = inlineGlobals[id] && inlineGlobals[id].comments;
237
-
238
- if (value === "off") {
239
- continue;
240
- }
241
-
242
- let variable = globalScope.set.get(id);
243
-
244
- if (!variable) {
245
- variable = new eslintScope.Variable(id, globalScope);
246
-
247
- globalScope.variables.push(variable);
248
- globalScope.set.set(id, variable);
249
- }
250
-
251
- variable.eslintImplicitGlobalSetting = configValue;
252
- variable.eslintExplicitGlobal = sourceComments !== void 0;
253
- variable.eslintExplicitGlobalComments = sourceComments;
254
- variable.writeable = (value === "writable");
255
- }
256
-
257
- /*
258
- * "through" contains all references which definitions cannot be found.
259
- * Since we augment the global scope using configuration, we need to update
260
- * references and remove the ones that were added by configuration.
261
- */
262
- globalScope.through = globalScope.through.filter(reference => {
263
- const name = reference.identifier.name;
264
- const variable = globalScope.set.get(name);
265
-
266
- if (variable) {
267
-
268
- /*
269
- * Links the variable and the reference.
270
- * And this reference is removed from `Scope#through`.
271
- */
272
- reference.resolved = variable;
273
- variable.references.push(reference);
274
-
275
- return false;
276
- }
277
-
278
- return true;
279
- });
280
- }
281
-
282
- /**
283
- * Sets the given variable names as exported so they won't be triggered by
284
- * the `no-unused-vars` rule.
285
- * @param {eslint.Scope} globalScope The global scope to define exports in.
286
- * @param {Record<string,string>} variables An object whose keys are the variable
287
- * names to export.
288
- * @returns {void}
289
- */
290
- function markExportedVariables(globalScope, variables) {
291
-
292
- Object.keys(variables).forEach(name => {
293
- const variable = globalScope.set.get(name);
294
-
295
- if (variable) {
296
- variable.eslintUsed = true;
297
- variable.eslintExported = true;
298
- }
299
- });
300
-
301
- }
302
-
303
- //------------------------------------------------------------------------------
304
- // Public Interface
305
- //------------------------------------------------------------------------------
306
-
307
- const caches = Symbol("caches");
308
-
309
- /**
310
- * Represents parsed source code.
311
- */
312
- class SourceCode extends TokenStore {
313
-
314
- /**
315
- * @param {string|Object} textOrConfig The source code text or config object.
316
- * @param {string} textOrConfig.text The source code text.
317
- * @param {ASTNode} textOrConfig.ast The Program node of the AST representing the code. This AST should be created from the text that BOM was stripped.
318
- * @param {Object|null} textOrConfig.parserServices The parser services.
319
- * @param {ScopeManager|null} textOrConfig.scopeManager The scope of this source code.
320
- * @param {Object|null} textOrConfig.visitorKeys The visitor keys to traverse AST.
321
- * @param {ASTNode} [astIfNoConfig] The Program node of the AST representing the code. This AST should be created from the text that BOM was stripped.
322
- */
323
- constructor(textOrConfig, astIfNoConfig) {
324
- let text, ast, parserServices, scopeManager, visitorKeys;
325
-
326
- // Process overloading.
327
- if (typeof textOrConfig === "string") {
328
- text = textOrConfig;
329
- ast = astIfNoConfig;
330
- } else if (typeof textOrConfig === "object" && textOrConfig !== null) {
331
- text = textOrConfig.text;
332
- ast = textOrConfig.ast;
333
- parserServices = textOrConfig.parserServices;
334
- scopeManager = textOrConfig.scopeManager;
335
- visitorKeys = textOrConfig.visitorKeys;
336
- }
337
-
338
- validate(ast);
339
- super(ast.tokens, ast.comments);
340
-
341
- /**
342
- * General purpose caching for the class.
343
- */
344
- this[caches] = new Map([
345
- ["scopes", new WeakMap()],
346
- ["vars", new Map()],
347
- ["configNodes", void 0]
348
- ]);
349
-
350
- /**
351
- * The flag to indicate that the source code has Unicode BOM.
352
- * @type {boolean}
353
- */
354
- this.hasBOM = (text.charCodeAt(0) === 0xFEFF);
355
-
356
- /**
357
- * The original text source code.
358
- * BOM was stripped from this text.
359
- * @type {string}
360
- */
361
- this.text = (this.hasBOM ? text.slice(1) : text);
362
-
363
- /**
364
- * The parsed AST for the source code.
365
- * @type {ASTNode}
366
- */
367
- this.ast = ast;
368
-
369
- /**
370
- * The parser services of this source code.
371
- * @type {Object}
372
- */
373
- this.parserServices = parserServices || {};
374
-
375
- /**
376
- * The scope of this source code.
377
- * @type {ScopeManager|null}
378
- */
379
- this.scopeManager = scopeManager || null;
380
-
381
- /**
382
- * The visitor keys to traverse AST.
383
- * @type {Object}
384
- */
385
- this.visitorKeys = visitorKeys || Traverser.DEFAULT_VISITOR_KEYS;
386
-
387
- // Check the source text for the presence of a shebang since it is parsed as a standard line comment.
388
- const shebangMatched = this.text.match(astUtils.shebangPattern);
389
- const hasShebang = shebangMatched && ast.comments.length && ast.comments[0].value === shebangMatched[1];
390
-
391
- if (hasShebang) {
392
- ast.comments[0].type = "Shebang";
393
- }
394
-
395
- this.tokensAndComments = sortedMerge(ast.tokens, ast.comments);
396
-
397
- /**
398
- * The source code split into lines according to ECMA-262 specification.
399
- * This is done to avoid each rule needing to do so separately.
400
- * @type {string[]}
401
- */
402
- this.lines = [];
403
- this.lineStartIndices = [0];
404
-
405
- const lineEndingPattern = astUtils.createGlobalLinebreakMatcher();
406
- let match;
407
-
408
- /*
409
- * Previously, this was implemented using a regex that
410
- * matched a sequence of non-linebreak characters followed by a
411
- * linebreak, then adding the lengths of the matches. However,
412
- * this caused a catastrophic backtracking issue when the end
413
- * of a file contained a large number of non-newline characters.
414
- * To avoid this, the current implementation just matches newlines
415
- * and uses match.index to get the correct line start indices.
416
- */
417
- while ((match = lineEndingPattern.exec(this.text))) {
418
- this.lines.push(this.text.slice(this.lineStartIndices[this.lineStartIndices.length - 1], match.index));
419
- this.lineStartIndices.push(match.index + match[0].length);
420
- }
421
- this.lines.push(this.text.slice(this.lineStartIndices[this.lineStartIndices.length - 1]));
422
-
423
- // Cache for comments found using getComments().
424
- this._commentCache = new WeakMap();
425
-
426
- // don't allow further modification of this object
427
- Object.freeze(this);
428
- Object.freeze(this.lines);
429
- }
430
-
431
- /**
432
- * Split the source code into multiple lines based on the line delimiters.
433
- * @param {string} text Source code as a string.
434
- * @returns {string[]} Array of source code lines.
435
- * @public
436
- */
437
- static splitLines(text) {
438
- return text.split(astUtils.createGlobalLinebreakMatcher());
439
- }
440
-
441
- /**
442
- * Gets the source code for the given node.
443
- * @param {ASTNode} [node] The AST node to get the text for.
444
- * @param {int} [beforeCount] The number of characters before the node to retrieve.
445
- * @param {int} [afterCount] The number of characters after the node to retrieve.
446
- * @returns {string} The text representing the AST node.
447
- * @public
448
- */
449
- getText(node, beforeCount, afterCount) {
450
- if (node) {
451
- return this.text.slice(Math.max(node.range[0] - (beforeCount || 0), 0),
452
- node.range[1] + (afterCount || 0));
453
- }
454
- return this.text;
455
- }
456
-
457
- /**
458
- * Gets the entire source text split into an array of lines.
459
- * @returns {Array} The source text as an array of lines.
460
- * @public
461
- */
462
- getLines() {
463
- return this.lines;
464
- }
465
-
466
- /**
467
- * Retrieves an array containing all comments in the source code.
468
- * @returns {ASTNode[]} An array of comment nodes.
469
- * @public
470
- */
471
- getAllComments() {
472
- return this.ast.comments;
473
- }
474
-
475
- /**
476
- * Gets all comments for the given node.
477
- * @param {ASTNode} node The AST node to get the comments for.
478
- * @returns {Object} An object containing a leading and trailing array
479
- * of comments indexed by their position.
480
- * @public
481
- * @deprecated replaced by getCommentsBefore(), getCommentsAfter(), and getCommentsInside().
482
- */
483
- getComments(node) {
484
- if (this._commentCache.has(node)) {
485
- return this._commentCache.get(node);
486
- }
487
-
488
- const comments = {
489
- leading: [],
490
- trailing: []
491
- };
492
-
493
- /*
494
- * Return all comments as leading comments of the Program node when
495
- * there is no executable code.
496
- */
497
- if (node.type === "Program") {
498
- if (node.body.length === 0) {
499
- comments.leading = node.comments;
500
- }
501
- } else {
502
-
503
- /*
504
- * Return comments as trailing comments of nodes that only contain
505
- * comments (to mimic the comment attachment behavior present in Espree).
506
- */
507
- if ((node.type === "BlockStatement" || node.type === "ClassBody") && node.body.length === 0 ||
508
- node.type === "ObjectExpression" && node.properties.length === 0 ||
509
- node.type === "ArrayExpression" && node.elements.length === 0 ||
510
- node.type === "SwitchStatement" && node.cases.length === 0
511
- ) {
512
- comments.trailing = this.getTokens(node, {
513
- includeComments: true,
514
- filter: isCommentToken
515
- });
516
- }
517
-
518
- /*
519
- * Iterate over tokens before and after node and collect comment tokens.
520
- * Do not include comments that exist outside of the parent node
521
- * to avoid duplication.
522
- */
523
- let currentToken = this.getTokenBefore(node, { includeComments: true });
524
-
525
- while (currentToken && isCommentToken(currentToken)) {
526
- if (node.parent && node.parent.type !== "Program" && (currentToken.start < node.parent.start)) {
527
- break;
528
- }
529
- comments.leading.push(currentToken);
530
- currentToken = this.getTokenBefore(currentToken, { includeComments: true });
531
- }
532
-
533
- comments.leading.reverse();
534
-
535
- currentToken = this.getTokenAfter(node, { includeComments: true });
536
-
537
- while (currentToken && isCommentToken(currentToken)) {
538
- if (node.parent && node.parent.type !== "Program" && (currentToken.end > node.parent.end)) {
539
- break;
540
- }
541
- comments.trailing.push(currentToken);
542
- currentToken = this.getTokenAfter(currentToken, { includeComments: true });
543
- }
544
- }
545
-
546
- this._commentCache.set(node, comments);
547
- return comments;
548
- }
549
-
550
- /**
551
- * Retrieves the JSDoc comment for a given node.
552
- * @param {ASTNode} node The AST node to get the comment for.
553
- * @returns {Token|null} The Block comment token containing the JSDoc comment
554
- * for the given node or null if not found.
555
- * @public
556
- * @deprecated
557
- */
558
- getJSDocComment(node) {
559
-
560
- /**
561
- * Checks for the presence of a JSDoc comment for the given node and returns it.
562
- * @param {ASTNode} astNode The AST node to get the comment for.
563
- * @returns {Token|null} The Block comment token containing the JSDoc comment
564
- * for the given node or null if not found.
565
- * @private
566
- */
567
- const findJSDocComment = astNode => {
568
- const tokenBefore = this.getTokenBefore(astNode, { includeComments: true });
569
-
570
- if (
571
- tokenBefore &&
572
- isCommentToken(tokenBefore) &&
573
- tokenBefore.type === "Block" &&
574
- tokenBefore.value.charAt(0) === "*" &&
575
- astNode.loc.start.line - tokenBefore.loc.end.line <= 1
576
- ) {
577
- return tokenBefore;
578
- }
579
-
580
- return null;
581
- };
582
- let parent = node.parent;
583
-
584
- switch (node.type) {
585
- case "ClassDeclaration":
586
- case "FunctionDeclaration":
587
- return findJSDocComment(looksLikeExport(parent) ? parent : node);
588
-
589
- case "ClassExpression":
590
- return findJSDocComment(parent.parent);
591
-
592
- case "ArrowFunctionExpression":
593
- case "FunctionExpression":
594
- if (parent.type !== "CallExpression" && parent.type !== "NewExpression") {
595
- while (
596
- !this.getCommentsBefore(parent).length &&
597
- !/Function/u.test(parent.type) &&
598
- parent.type !== "MethodDefinition" &&
599
- parent.type !== "Property"
600
- ) {
601
- parent = parent.parent;
602
-
603
- if (!parent) {
604
- break;
605
- }
606
- }
607
-
608
- if (parent && parent.type !== "FunctionDeclaration" && parent.type !== "Program") {
609
- return findJSDocComment(parent);
610
- }
611
- }
612
-
613
- return findJSDocComment(node);
614
-
615
- // falls through
616
- default:
617
- return null;
618
- }
619
- }
620
-
621
- /**
622
- * Gets the deepest node containing a range index.
623
- * @param {int} index Range index of the desired node.
624
- * @returns {ASTNode} The node if found or null if not found.
625
- * @public
626
- */
627
- getNodeByRangeIndex(index) {
628
- let result = null;
629
-
630
- Traverser.traverse(this.ast, {
631
- visitorKeys: this.visitorKeys,
632
- enter(node) {
633
- if (node.range[0] <= index && index < node.range[1]) {
634
- result = node;
635
- } else {
636
- this.skip();
637
- }
638
- },
639
- leave(node) {
640
- if (node === result) {
641
- this.break();
642
- }
643
- }
644
- });
645
-
646
- return result;
647
- }
648
-
649
- /**
650
- * Determines if two nodes or tokens have at least one whitespace character
651
- * between them. Order does not matter. Returns false if the given nodes or
652
- * tokens overlap.
653
- * @param {ASTNode|Token} first The first node or token to check between.
654
- * @param {ASTNode|Token} second The second node or token to check between.
655
- * @returns {boolean} True if there is a whitespace character between
656
- * any of the tokens found between the two given nodes or tokens.
657
- * @public
658
- */
659
- isSpaceBetween(first, second) {
660
- return isSpaceBetween(this, first, second, false);
661
- }
662
-
663
- /**
664
- * Determines if two nodes or tokens have at least one whitespace character
665
- * between them. Order does not matter. Returns false if the given nodes or
666
- * tokens overlap.
667
- * For backward compatibility, this method returns true if there are
668
- * `JSXText` tokens that contain whitespaces between the two.
669
- * @param {ASTNode|Token} first The first node or token to check between.
670
- * @param {ASTNode|Token} second The second node or token to check between.
671
- * @returns {boolean} True if there is a whitespace character between
672
- * any of the tokens found between the two given nodes or tokens.
673
- * @deprecated in favor of isSpaceBetween().
674
- * @public
675
- */
676
- isSpaceBetweenTokens(first, second) {
677
- return isSpaceBetween(this, first, second, true);
678
- }
679
-
680
- /**
681
- * Converts a source text index into a (line, column) pair.
682
- * @param {number} index The index of a character in a file
683
- * @throws {TypeError} If non-numeric index or index out of range.
684
- * @returns {Object} A {line, column} location object with a 0-indexed column
685
- * @public
686
- */
687
- getLocFromIndex(index) {
688
- if (typeof index !== "number") {
689
- throw new TypeError("Expected `index` to be a number.");
690
- }
691
-
692
- if (index < 0 || index > this.text.length) {
693
- throw new RangeError(`Index out of range (requested index ${index}, but source text has length ${this.text.length}).`);
694
- }
695
-
696
- /*
697
- * For an argument of this.text.length, return the location one "spot" past the last character
698
- * of the file. If the last character is a linebreak, the location will be column 0 of the next
699
- * line; otherwise, the location will be in the next column on the same line.
700
- *
701
- * See getIndexFromLoc for the motivation for this special case.
702
- */
703
- if (index === this.text.length) {
704
- return { line: this.lines.length, column: this.lines[this.lines.length - 1].length };
705
- }
706
-
707
- /*
708
- * To figure out which line index is on, determine the last place at which index could
709
- * be inserted into lineStartIndices to keep the list sorted.
710
- */
711
- const lineNumber = index >= this.lineStartIndices[this.lineStartIndices.length - 1]
712
- ? this.lineStartIndices.length
713
- : this.lineStartIndices.findIndex(el => index < el);
714
-
715
- return { line: lineNumber, column: index - this.lineStartIndices[lineNumber - 1] };
716
- }
717
-
718
- /**
719
- * Converts a (line, column) pair into a range index.
720
- * @param {Object} loc A line/column location
721
- * @param {number} loc.line The line number of the location (1-indexed)
722
- * @param {number} loc.column The column number of the location (0-indexed)
723
- * @throws {TypeError|RangeError} If `loc` is not an object with a numeric
724
- * `line` and `column`, if the `line` is less than or equal to zero or
725
- * the line or column is out of the expected range.
726
- * @returns {number} The range index of the location in the file.
727
- * @public
728
- */
729
- getIndexFromLoc(loc) {
730
- if (typeof loc !== "object" || typeof loc.line !== "number" || typeof loc.column !== "number") {
731
- throw new TypeError("Expected `loc` to be an object with numeric `line` and `column` properties.");
732
- }
733
-
734
- if (loc.line <= 0) {
735
- throw new RangeError(`Line number out of range (line ${loc.line} requested). Line numbers should be 1-based.`);
736
- }
737
-
738
- if (loc.line > this.lineStartIndices.length) {
739
- throw new RangeError(`Line number out of range (line ${loc.line} requested, but only ${this.lineStartIndices.length} lines present).`);
740
- }
741
-
742
- const lineStartIndex = this.lineStartIndices[loc.line - 1];
743
- const lineEndIndex = loc.line === this.lineStartIndices.length ? this.text.length : this.lineStartIndices[loc.line];
744
- const positionIndex = lineStartIndex + loc.column;
745
-
746
- /*
747
- * By design, getIndexFromLoc({ line: lineNum, column: 0 }) should return the start index of
748
- * the given line, provided that the line number is valid element of this.lines. Since the
749
- * last element of this.lines is an empty string for files with trailing newlines, add a
750
- * special case where getting the index for the first location after the end of the file
751
- * will return the length of the file, rather than throwing an error. This allows rules to
752
- * use getIndexFromLoc consistently without worrying about edge cases at the end of a file.
753
- */
754
- if (
755
- loc.line === this.lineStartIndices.length && positionIndex > lineEndIndex ||
756
- loc.line < this.lineStartIndices.length && positionIndex >= lineEndIndex
757
- ) {
758
- throw new RangeError(`Column number out of range (column ${loc.column} requested, but the length of line ${loc.line} is ${lineEndIndex - lineStartIndex}).`);
759
- }
760
-
761
- return positionIndex;
762
- }
763
-
764
- /**
765
- * Gets the scope for the given node
766
- * @param {ASTNode} currentNode The node to get the scope of
767
- * @returns {eslint-scope.Scope} The scope information for this node
768
- * @throws {TypeError} If the `currentNode` argument is missing.
769
- */
770
- getScope(currentNode) {
771
-
772
- if (!currentNode) {
773
- throw new TypeError("Missing required argument: node.");
774
- }
775
-
776
- // check cache first
777
- const cache = this[caches].get("scopes");
778
- const cachedScope = cache.get(currentNode);
779
-
780
- if (cachedScope) {
781
- return cachedScope;
782
- }
783
-
784
- // On Program node, get the outermost scope to avoid return Node.js special function scope or ES modules scope.
785
- const inner = currentNode.type !== "Program";
786
-
787
- for (let node = currentNode; node; node = node.parent) {
788
- const scope = this.scopeManager.acquire(node, inner);
789
-
790
- if (scope) {
791
- if (scope.type === "function-expression-name") {
792
- cache.set(currentNode, scope.childScopes[0]);
793
- return scope.childScopes[0];
794
- }
795
-
796
- cache.set(currentNode, scope);
797
- return scope;
798
- }
799
- }
800
-
801
- cache.set(currentNode, this.scopeManager.scopes[0]);
802
- return this.scopeManager.scopes[0];
803
- }
804
-
805
- /**
806
- * Get the variables that `node` defines.
807
- * This is a convenience method that passes through
808
- * to the same method on the `scopeManager`.
809
- * @param {ASTNode} node The node for which the variables are obtained.
810
- * @returns {Array<Variable>} An array of variable nodes representing
811
- * the variables that `node` defines.
812
- */
813
- getDeclaredVariables(node) {
814
- return this.scopeManager.getDeclaredVariables(node);
815
- }
816
-
817
- /* eslint-disable class-methods-use-this -- node is owned by SourceCode */
818
- /**
819
- * Gets all the ancestors of a given node
820
- * @param {ASTNode} node The node
821
- * @returns {Array<ASTNode>} All the ancestor nodes in the AST, not including the provided node, starting
822
- * from the root node at index 0 and going inwards to the parent node.
823
- * @throws {TypeError} When `node` is missing.
824
- */
825
- getAncestors(node) {
826
-
827
- if (!node) {
828
- throw new TypeError("Missing required argument: node.");
829
- }
830
-
831
- const ancestorsStartingAtParent = [];
832
-
833
- for (let ancestor = node.parent; ancestor; ancestor = ancestor.parent) {
834
- ancestorsStartingAtParent.push(ancestor);
835
- }
836
-
837
- return ancestorsStartingAtParent.reverse();
838
- }
839
- /* eslint-enable class-methods-use-this -- node is owned by SourceCode */
840
-
841
- /**
842
- * Marks a variable as used in the current scope
843
- * @param {string} name The name of the variable to mark as used.
844
- * @param {ASTNode} [refNode] The closest node to the variable reference.
845
- * @returns {boolean} True if the variable was found and marked as used, false if not.
846
- */
847
- markVariableAsUsed(name, refNode = this.ast) {
848
-
849
- const currentScope = this.getScope(refNode);
850
- let initialScope = currentScope;
851
-
852
- /*
853
- * When we are in an ESM or CommonJS module, we need to start searching
854
- * from the top-level scope, not the global scope. For ESM the top-level
855
- * scope is the module scope; for CommonJS the top-level scope is the
856
- * outer function scope.
857
- *
858
- * Without this check, we might miss a variable declared with `var` at
859
- * the top-level because it won't exist in the global scope.
860
- */
861
- if (
862
- currentScope.type === "global" &&
863
- currentScope.childScopes.length > 0 &&
864
-
865
- // top-level scopes refer to a `Program` node
866
- currentScope.childScopes[0].block === this.ast
867
- ) {
868
- initialScope = currentScope.childScopes[0];
869
- }
870
-
871
- for (let scope = initialScope; scope; scope = scope.upper) {
872
- const variable = scope.variables.find(scopeVar => scopeVar.name === name);
873
-
874
- if (variable) {
875
- variable.eslintUsed = true;
876
- return true;
877
- }
878
- }
879
-
880
- return false;
881
- }
882
-
883
-
884
- /**
885
- * Returns an array of all inline configuration nodes found in the
886
- * source code.
887
- * @returns {Array<Token>} An array of all inline configuration nodes.
888
- */
889
- getInlineConfigNodes() {
890
-
891
- // check the cache first
892
- let configNodes = this[caches].get("configNodes");
893
-
894
- if (configNodes) {
895
- return configNodes;
896
- }
897
-
898
- // calculate fresh config nodes
899
- configNodes = this.ast.comments.filter(comment => {
900
-
901
- // shebang comments are never directives
902
- if (comment.type === "Shebang") {
903
- return false;
904
- }
905
-
906
- const { directivePart } = commentParser.extractDirectiveComment(comment.value);
907
-
908
- const directiveMatch = directivesPattern.exec(directivePart);
909
-
910
- if (!directiveMatch) {
911
- return false;
912
- }
913
-
914
- // only certain comment types are supported as line comments
915
- return comment.type !== "Line" || !!/^eslint-disable-(next-)?line$/u.test(directiveMatch[1]);
916
- });
917
-
918
- this[caches].set("configNodes", configNodes);
919
-
920
- return configNodes;
921
- }
922
-
923
- /**
924
- * Applies language options sent in from the core.
925
- * @param {Object} languageOptions The language options for this run.
926
- * @returns {void}
927
- */
928
- applyLanguageOptions(languageOptions) {
929
-
930
- /*
931
- * Add configured globals and language globals
932
- *
933
- * Using Object.assign instead of object spread for performance reasons
934
- * https://github.com/eslint/eslint/issues/16302
935
- */
936
- const configGlobals = Object.assign(
937
- Object.create(null), // https://github.com/eslint/eslint/issues/18363
938
- getGlobalsForEcmaVersion(languageOptions.ecmaVersion),
939
- languageOptions.sourceType === "commonjs" ? globals.commonjs : void 0,
940
- languageOptions.globals
941
- );
942
- const varsCache = this[caches].get("vars");
943
-
944
- varsCache.set("configGlobals", configGlobals);
945
- }
946
-
947
- /**
948
- * Applies configuration found inside of the source code. This method is only
949
- * called when ESLint is running with inline configuration allowed.
950
- * @returns {{problems:Array<Problem>,configs:{config:FlatConfigArray,node:ASTNode}}} Information
951
- * that ESLint needs to further process the inline configuration.
952
- */
953
- applyInlineConfig() {
954
-
955
- const problems = [];
956
- const configs = [];
957
- const exportedVariables = {};
958
- const inlineGlobals = Object.create(null);
959
-
960
- this.getInlineConfigNodes().forEach(comment => {
961
-
962
- const { directiveText, directiveValue } = commentParser.parseDirective(comment);
963
-
964
- switch (directiveText) {
965
- case "exported":
966
- Object.assign(exportedVariables, commentParser.parseStringConfig(directiveValue, comment));
967
- break;
968
-
969
- case "globals":
970
- case "global":
971
- for (const [id, { value }] of Object.entries(commentParser.parseStringConfig(directiveValue, comment))) {
972
- let normalizedValue;
973
-
974
- try {
975
- normalizedValue = normalizeConfigGlobal(value);
976
- } catch (err) {
977
- problems.push({
978
- ruleId: null,
979
- loc: comment.loc,
980
- message: err.message
981
- });
982
- continue;
983
- }
984
-
985
- if (inlineGlobals[id]) {
986
- inlineGlobals[id].comments.push(comment);
987
- inlineGlobals[id].value = normalizedValue;
988
- } else {
989
- inlineGlobals[id] = {
990
- comments: [comment],
991
- value: normalizedValue
992
- };
993
- }
994
- }
995
- break;
996
-
997
- case "eslint": {
998
- const parseResult = commentParser.parseJsonConfig(directiveValue, comment.loc);
999
-
1000
- if (parseResult.success) {
1001
- configs.push({
1002
- config: {
1003
- rules: parseResult.config
1004
- },
1005
- node: comment
1006
- });
1007
- } else {
1008
- problems.push(parseResult.error);
1009
- }
1010
-
1011
- break;
1012
- }
1013
-
1014
- // no default
1015
- }
1016
- });
1017
-
1018
- // save all the new variables for later
1019
- const varsCache = this[caches].get("vars");
1020
-
1021
- varsCache.set("inlineGlobals", inlineGlobals);
1022
- varsCache.set("exportedVariables", exportedVariables);
1023
-
1024
- return {
1025
- configs,
1026
- problems
1027
- };
1028
- }
1029
-
1030
- /**
1031
- * Called by ESLint core to indicate that it has finished providing
1032
- * information. We now add in all the missing variables and ensure that
1033
- * state-changing methods cannot be called by rules.
1034
- * @returns {void}
1035
- */
1036
- finalize() {
1037
-
1038
- // Step 1: ensure that all of the necessary variables are up to date
1039
- const varsCache = this[caches].get("vars");
1040
- const globalScope = this.scopeManager.scopes[0];
1041
- const configGlobals = varsCache.get("configGlobals");
1042
- const inlineGlobals = varsCache.get("inlineGlobals");
1043
- const exportedVariables = varsCache.get("exportedVariables");
1044
-
1045
- addDeclaredGlobals(globalScope, configGlobals, inlineGlobals);
1046
-
1047
- if (exportedVariables) {
1048
- markExportedVariables(globalScope, exportedVariables);
1049
- }
1050
-
1051
- }
1052
-
1053
- }
1054
-
1055
- module.exports = SourceCode;