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.
- package/README.md +165 -115
- package/bin/eslint.js +112 -89
- package/conf/default-cli-options.js +22 -22
- package/conf/ecma-version.js +16 -0
- package/conf/globals.js +109 -94
- package/conf/replacements.json +24 -20
- package/conf/rule-type-list.json +89 -26
- package/lib/api.js +16 -20
- package/lib/cli-engine/cli-engine.js +841 -810
- package/lib/cli-engine/file-enumerator.js +384 -390
- package/lib/cli-engine/formatters/formatters-meta.json +17 -45
- package/lib/cli-engine/formatters/html.js +110 -102
- package/lib/cli-engine/formatters/json-with-metadata.js +5 -5
- package/lib/cli-engine/formatters/json.js +2 -2
- package/lib/cli-engine/formatters/stylish.js +97 -76
- package/lib/cli-engine/hash.js +1 -1
- package/lib/cli-engine/index.js +1 -1
- package/lib/cli-engine/lint-result-cache.js +165 -148
- package/lib/cli-engine/load-rules.js +17 -17
- package/lib/cli.js +481 -399
- package/lib/config/config-loader.js +816 -0
- package/lib/config/config.js +674 -0
- package/lib/config/default-config.js +57 -46
- package/lib/config/flat-config-array.js +170 -333
- package/lib/config/flat-config-schema.js +389 -389
- package/lib/config-api.js +12 -0
- package/lib/eslint/eslint-helpers.js +1196 -663
- package/lib/eslint/eslint.js +1262 -607
- package/lib/eslint/index.js +3 -3
- package/lib/eslint/legacy-eslint.js +786 -0
- package/lib/eslint/worker.js +173 -0
- package/lib/languages/js/index.js +336 -0
- package/lib/languages/js/source-code/index.js +7 -0
- package/lib/languages/js/source-code/source-code.js +1364 -0
- package/lib/languages/js/source-code/token-store/backward-token-comment-cursor.js +61 -0
- package/lib/languages/js/source-code/token-store/backward-token-cursor.js +57 -0
- package/lib/{source-code → languages/js/source-code}/token-store/cursor.js +36 -36
- package/lib/languages/js/source-code/token-store/cursors.js +120 -0
- package/lib/{source-code → languages/js/source-code}/token-store/decorative-cursor.js +17 -18
- package/lib/{source-code → languages/js/source-code}/token-store/filter-cursor.js +19 -20
- package/lib/languages/js/source-code/token-store/forward-token-comment-cursor.js +65 -0
- package/lib/languages/js/source-code/token-store/forward-token-cursor.js +62 -0
- package/lib/languages/js/source-code/token-store/index.js +721 -0
- package/lib/{source-code → languages/js/source-code}/token-store/limit-cursor.js +17 -18
- package/lib/languages/js/source-code/token-store/padded-token-cursor.js +45 -0
- package/lib/{source-code → languages/js/source-code}/token-store/skip-cursor.js +19 -20
- package/lib/languages/js/source-code/token-store/utils.js +110 -0
- package/lib/languages/js/validate-language-options.js +196 -0
- package/lib/linter/apply-disable-directives.js +490 -371
- package/lib/linter/code-path-analysis/code-path-analyzer.js +650 -674
- package/lib/linter/code-path-analysis/code-path-segment.js +215 -216
- package/lib/linter/code-path-analysis/code-path-state.js +2118 -2096
- package/lib/linter/code-path-analysis/code-path.js +307 -317
- package/lib/linter/code-path-analysis/debug-helpers.js +183 -163
- package/lib/linter/code-path-analysis/fork-context.js +297 -272
- package/lib/linter/code-path-analysis/id-generator.js +22 -23
- package/lib/linter/esquery.js +332 -0
- package/lib/linter/file-context.js +144 -0
- package/lib/linter/file-report.js +608 -0
- package/lib/linter/index.js +3 -5
- package/lib/linter/interpolate.js +38 -16
- package/lib/linter/linter.js +2328 -1785
- package/lib/linter/rule-fixer.js +136 -107
- package/lib/linter/rules.js +37 -46
- package/lib/linter/source-code-fixer.js +96 -94
- package/lib/linter/source-code-traverser.js +333 -0
- package/lib/linter/source-code-visitor.js +81 -0
- package/lib/linter/timing.js +145 -97
- package/lib/linter/vfile.js +115 -0
- package/lib/options.js +464 -326
- package/lib/rule-tester/index.js +3 -1
- package/lib/rule-tester/rule-tester.js +1371 -998
- package/lib/rules/accessor-pairs.js +333 -259
- package/lib/rules/array-bracket-newline.js +250 -220
- package/lib/rules/array-bracket-spacing.js +286 -229
- package/lib/rules/array-callback-return.js +401 -354
- package/lib/rules/array-element-newline.js +358 -295
- package/lib/rules/arrow-body-style.js +400 -278
- package/lib/rules/arrow-parens.js +206 -155
- package/lib/rules/arrow-spacing.js +169 -145
- package/lib/rules/block-scoped-var.js +125 -123
- package/lib/rules/block-spacing.js +186 -158
- package/lib/rules/brace-style.js +262 -181
- package/lib/rules/callback-return.js +203 -174
- package/lib/rules/camelcase.js +403 -380
- package/lib/rules/capitalized-comments.js +253 -228
- package/lib/rules/class-methods-use-this.js +231 -168
- package/lib/rules/comma-dangle.js +379 -328
- package/lib/rules/comma-spacing.js +193 -177
- package/lib/rules/comma-style.js +375 -298
- package/lib/rules/complexity.js +180 -144
- package/lib/rules/computed-property-spacing.js +236 -193
- package/lib/rules/consistent-return.js +181 -170
- package/lib/rules/consistent-this.js +167 -141
- package/lib/rules/constructor-super.js +418 -411
- package/lib/rules/curly.js +407 -468
- package/lib/rules/default-case-last.js +39 -32
- package/lib/rules/default-case.js +89 -83
- package/lib/rules/default-param-last.js +69 -53
- package/lib/rules/dot-location.js +122 -92
- package/lib/rules/dot-notation.js +193 -153
- package/lib/rules/eol-last.js +122 -102
- package/lib/rules/eqeqeq.js +191 -155
- package/lib/rules/for-direction.js +150 -122
- package/lib/rules/func-call-spacing.js +261 -213
- package/lib/rules/func-name-matching.js +294 -209
- package/lib/rules/func-names.js +165 -164
- package/lib/rules/func-style.js +209 -86
- package/lib/rules/function-call-argument-newline.js +152 -111
- package/lib/rules/function-paren-newline.js +349 -273
- package/lib/rules/generator-star-spacing.js +229 -192
- package/lib/rules/getter-return.js +208 -170
- package/lib/rules/global-require.js +85 -58
- package/lib/rules/grouped-accessor-pairs.js +201 -148
- package/lib/rules/guard-for-in.js +72 -63
- package/lib/rules/handle-callback-err.js +108 -87
- package/lib/rules/id-blacklist.js +182 -187
- package/lib/rules/id-denylist.js +174 -179
- package/lib/rules/id-length.js +197 -157
- package/lib/rules/id-match.js +350 -286
- package/lib/rules/implicit-arrow-linebreak.js +102 -61
- package/lib/rules/indent-legacy.js +1345 -1102
- package/lib/rules/indent.js +2272 -1741
- package/lib/rules/index.js +320 -294
- package/lib/rules/init-declarations.js +139 -106
- package/lib/rules/jsx-quotes.js +94 -64
- package/lib/rules/key-spacing.js +750 -615
- package/lib/rules/keyword-spacing.js +648 -587
- package/lib/rules/line-comment-position.js +143 -108
- package/lib/rules/linebreak-style.js +115 -88
- package/lib/rules/lines-around-comment.js +540 -430
- package/lib/rules/lines-around-directive.js +233 -185
- package/lib/rules/lines-between-class-members.js +305 -216
- package/lib/rules/logical-assignment-operators.js +582 -398
- package/lib/rules/max-classes-per-file.js +69 -68
- package/lib/rules/max-depth.js +146 -143
- package/lib/rules/max-len.js +473 -416
- package/lib/rules/max-lines-per-function.js +201 -176
- package/lib/rules/max-lines.js +158 -162
- package/lib/rules/max-nested-callbacks.js +102 -104
- package/lib/rules/max-params.js +102 -75
- package/lib/rules/max-statements-per-line.js +205 -180
- package/lib/rules/max-statements.js +168 -164
- package/lib/rules/multiline-comment-style.js +638 -460
- package/lib/rules/multiline-ternary.js +241 -158
- package/lib/rules/new-cap.js +233 -232
- package/lib/rules/new-parens.js +88 -61
- package/lib/rules/newline-after-var.js +287 -233
- package/lib/rules/newline-before-return.js +229 -204
- package/lib/rules/newline-per-chained-call.js +142 -109
- package/lib/rules/no-alert.js +90 -79
- package/lib/rules/no-array-constructor.js +175 -113
- package/lib/rules/no-async-promise-executor.js +30 -24
- package/lib/rules/no-await-in-loop.js +79 -70
- package/lib/rules/no-bitwise.js +113 -87
- package/lib/rules/no-buffer-constructor.js +61 -37
- package/lib/rules/no-caller.js +39 -33
- package/lib/rules/no-case-declarations.js +61 -45
- package/lib/rules/no-catch-shadow.js +76 -62
- package/lib/rules/no-class-assign.js +51 -48
- package/lib/rules/no-compare-neg-zero.js +62 -48
- package/lib/rules/no-cond-assign.js +148 -132
- package/lib/rules/no-confusing-arrow.js +98 -63
- package/lib/rules/no-console.js +202 -188
- package/lib/rules/no-const-assign.js +58 -41
- package/lib/rules/no-constant-binary-expression.js +501 -407
- package/lib/rules/no-constant-condition.js +158 -131
- package/lib/rules/no-constructor-return.js +49 -49
- package/lib/rules/no-continue.js +25 -26
- package/lib/rules/no-control-regex.js +125 -121
- package/lib/rules/no-debugger.js +28 -30
- package/lib/rules/no-delete-var.js +29 -29
- package/lib/rules/no-div-regex.js +47 -40
- package/lib/rules/no-dupe-args.js +79 -69
- package/lib/rules/no-dupe-class-members.js +102 -89
- package/lib/rules/no-dupe-else-if.js +100 -77
- package/lib/rules/no-dupe-keys.js +133 -110
- package/lib/rules/no-duplicate-case.js +50 -43
- package/lib/rules/no-duplicate-imports.js +266 -188
- package/lib/rules/no-else-return.js +430 -385
- package/lib/rules/no-empty-character-class.js +57 -50
- package/lib/rules/no-empty-function.js +197 -128
- package/lib/rules/no-empty-pattern.js +63 -56
- package/lib/rules/no-empty-static-block.js +61 -35
- package/lib/rules/no-empty.js +135 -85
- package/lib/rules/no-eq-null.js +37 -32
- package/lib/rules/no-eval.js +258 -249
- package/lib/rules/no-ex-assign.js +42 -39
- package/lib/rules/no-extend-native.js +161 -160
- package/lib/rules/no-extra-bind.js +201 -190
- package/lib/rules/no-extra-boolean-cast.js +398 -295
- package/lib/rules/no-extra-label.js +150 -130
- package/lib/rules/no-extra-parens.js +1654 -1307
- package/lib/rules/no-extra-semi.js +146 -126
- package/lib/rules/no-fallthrough.js +200 -136
- package/lib/rules/no-floating-decimal.js +74 -48
- package/lib/rules/no-func-assign.js +54 -55
- package/lib/rules/no-global-assign.js +78 -72
- package/lib/rules/no-implicit-coercion.js +350 -262
- package/lib/rules/no-implicit-globals.js +174 -133
- package/lib/rules/no-implied-eval.js +150 -112
- package/lib/rules/no-import-assign.js +145 -159
- package/lib/rules/no-inline-comments.js +101 -96
- package/lib/rules/no-inner-declarations.js +115 -78
- package/lib/rules/no-invalid-regexp.js +223 -174
- package/lib/rules/no-invalid-this.js +145 -117
- package/lib/rules/no-irregular-whitespace.js +266 -250
- package/lib/rules/no-iterator.js +29 -33
- package/lib/rules/no-label-var.js +59 -61
- package/lib/rules/no-labels.js +138 -131
- package/lib/rules/no-lone-blocks.js +127 -123
- package/lib/rules/no-lonely-if.js +105 -67
- package/lib/rules/no-loop-func.js +245 -184
- package/lib/rules/no-loss-of-precision.js +236 -201
- package/lib/rules/no-magic-numbers.js +339 -217
- package/lib/rules/no-misleading-character-class.js +548 -253
- package/lib/rules/no-mixed-operators.js +188 -164
- package/lib/rules/no-mixed-requires.js +253 -224
- package/lib/rules/no-mixed-spaces-and-tabs.js +135 -103
- package/lib/rules/no-multi-assign.js +46 -47
- package/lib/rules/no-multi-spaces.js +163 -125
- package/lib/rules/no-multi-str.js +42 -40
- package/lib/rules/no-multiple-empty-lines.js +196 -140
- package/lib/rules/no-native-reassign.js +90 -74
- package/lib/rules/no-negated-condition.js +79 -74
- package/lib/rules/no-negated-in-lhs.js +45 -32
- package/lib/rules/no-nested-ternary.js +33 -31
- package/lib/rules/no-new-func.js +71 -62
- package/lib/rules/no-new-native-nonconstructor.js +43 -39
- package/lib/rules/no-new-object.js +48 -39
- package/lib/rules/no-new-require.js +48 -31
- package/lib/rules/no-new-symbol.js +61 -43
- package/lib/rules/no-new-wrappers.js +43 -41
- package/lib/rules/no-new.js +28 -29
- package/lib/rules/no-nonoctal-decimal-escape.js +149 -121
- package/lib/rules/no-obj-calls.js +66 -53
- package/lib/rules/no-object-constructor.js +104 -97
- package/lib/rules/no-octal-escape.js +40 -43
- package/lib/rules/no-octal.js +29 -32
- package/lib/rules/no-param-reassign.js +236 -218
- package/lib/rules/no-path-concat.js +66 -51
- package/lib/rules/no-plusplus.js +60 -63
- package/lib/rules/no-process-env.js +49 -32
- package/lib/rules/no-process-exit.js +48 -28
- package/lib/rules/no-promise-executor-return.js +205 -204
- package/lib/rules/no-proto.js +26 -29
- package/lib/rules/no-prototype-builtins.js +146 -124
- package/lib/rules/no-redeclare.js +154 -155
- package/lib/rules/no-regex-spaces.js +183 -161
- package/lib/rules/no-restricted-exports.js +208 -174
- package/lib/rules/no-restricted-globals.js +254 -112
- package/lib/rules/no-restricted-imports.js +824 -384
- package/lib/rules/no-restricted-modules.js +222 -186
- package/lib/rules/no-restricted-properties.js +218 -153
- package/lib/rules/no-restricted-syntax.js +56 -52
- package/lib/rules/no-return-assign.js +56 -49
- package/lib/rules/no-return-await.js +147 -120
- package/lib/rules/no-script-url.js +53 -46
- package/lib/rules/no-self-assign.js +148 -145
- package/lib/rules/no-self-compare.js +63 -46
- package/lib/rules/no-sequences.js +135 -115
- package/lib/rules/no-setter-return.js +176 -178
- package/lib/rules/no-shadow-restricted-names.js +84 -36
- package/lib/rules/no-shadow.js +598 -310
- package/lib/rules/no-spaced-func.js +82 -60
- package/lib/rules/no-sparse-arrays.js +46 -28
- package/lib/rules/no-sync.js +61 -44
- package/lib/rules/no-tabs.js +83 -54
- package/lib/rules/no-template-curly-in-string.js +33 -32
- package/lib/rules/no-ternary.js +25 -28
- package/lib/rules/no-this-before-super.js +332 -298
- package/lib/rules/no-throw-literal.js +31 -36
- package/lib/rules/no-trailing-spaces.js +208 -174
- package/lib/rules/no-unassigned-vars.js +80 -0
- package/lib/rules/no-undef-init.js +86 -60
- package/lib/rules/no-undef.js +52 -47
- package/lib/rules/no-undefined.js +73 -74
- package/lib/rules/no-underscore-dangle.js +370 -322
- package/lib/rules/no-unexpected-multiline.js +112 -102
- package/lib/rules/no-unmodified-loop-condition.js +254 -254
- package/lib/rules/no-unneeded-ternary.js +212 -146
- package/lib/rules/no-unreachable-loop.js +145 -140
- package/lib/rules/no-unreachable.js +255 -248
- package/lib/rules/no-unsafe-finally.js +93 -85
- package/lib/rules/no-unsafe-negation.js +105 -81
- package/lib/rules/no-unsafe-optional-chaining.js +193 -177
- package/lib/rules/no-unused-expressions.js +199 -158
- package/lib/rules/no-unused-labels.js +139 -124
- package/lib/rules/no-unused-private-class-members.js +206 -182
- package/lib/rules/no-unused-vars.js +1708 -687
- package/lib/rules/no-use-before-define.js +327 -229
- package/lib/rules/no-useless-assignment.js +654 -0
- package/lib/rules/no-useless-backreference.js +212 -143
- package/lib/rules/no-useless-call.js +58 -53
- package/lib/rules/no-useless-catch.js +40 -40
- package/lib/rules/no-useless-computed-key.js +144 -108
- package/lib/rules/no-useless-concat.js +65 -59
- package/lib/rules/no-useless-constructor.js +160 -97
- package/lib/rules/no-useless-escape.js +364 -291
- package/lib/rules/no-useless-rename.js +183 -153
- package/lib/rules/no-useless-return.js +344 -307
- package/lib/rules/no-var.js +245 -212
- package/lib/rules/no-void.js +51 -46
- package/lib/rules/no-warning-comments.js +191 -183
- package/lib/rules/no-whitespace-before-property.js +131 -97
- package/lib/rules/no-with.js +24 -26
- package/lib/rules/nonblock-statement-body-position.js +149 -112
- package/lib/rules/object-curly-newline.js +306 -247
- package/lib/rules/object-curly-spacing.js +360 -296
- package/lib/rules/object-property-newline.js +137 -88
- package/lib/rules/object-shorthand.js +632 -500
- package/lib/rules/one-var-declaration-per-line.js +104 -82
- package/lib/rules/one-var.js +686 -536
- package/lib/rules/operator-assignment.js +219 -158
- package/lib/rules/operator-linebreak.js +295 -233
- package/lib/rules/padded-blocks.js +346 -290
- package/lib/rules/padding-line-between-statements.js +443 -421
- package/lib/rules/prefer-arrow-callback.js +371 -315
- package/lib/rules/prefer-const.js +418 -373
- package/lib/rules/prefer-destructuring.js +309 -278
- package/lib/rules/prefer-exponentiation-operator.js +176 -132
- package/lib/rules/prefer-named-capture-group.js +160 -141
- package/lib/rules/prefer-numeric-literals.js +121 -112
- package/lib/rules/prefer-object-has-own.js +116 -82
- package/lib/rules/prefer-object-spread.js +214 -193
- package/lib/rules/prefer-promise-reject-errors.js +140 -118
- package/lib/rules/prefer-reflect.js +126 -103
- package/lib/rules/prefer-regex-literals.js +561 -463
- package/lib/rules/prefer-rest-params.js +79 -80
- package/lib/rules/prefer-spread.js +47 -43
- package/lib/rules/prefer-template.js +266 -194
- package/lib/rules/preserve-caught-error.js +535 -0
- package/lib/rules/quote-props.js +373 -289
- package/lib/rules/quotes.js +374 -308
- package/lib/rules/radix.js +152 -134
- package/lib/rules/require-atomic-updates.js +316 -282
- package/lib/rules/require-await.js +153 -82
- package/lib/rules/require-unicode-regexp.js +296 -108
- package/lib/rules/require-yield.js +53 -54
- package/lib/rules/rest-spread-spacing.js +128 -98
- package/lib/rules/semi-spacing.js +281 -232
- package/lib/rules/semi-style.js +176 -116
- package/lib/rules/semi.js +456 -418
- package/lib/rules/sort-imports.js +307 -229
- package/lib/rules/sort-keys.js +219 -181
- package/lib/rules/sort-vars.js +127 -91
- package/lib/rules/space-before-blocks.js +199 -171
- package/lib/rules/space-before-function-paren.js +186 -148
- package/lib/rules/space-in-parens.js +359 -270
- package/lib/rules/space-infix-ops.js +237 -183
- package/lib/rules/space-unary-ops.js +356 -280
- package/lib/rules/spaced-comment.js +363 -301
- package/lib/rules/strict.js +266 -229
- package/lib/rules/switch-colon-spacing.js +130 -104
- package/lib/rules/symbol-description.js +45 -48
- package/lib/rules/template-curly-spacing.js +148 -124
- package/lib/rules/template-tag-spacing.js +98 -70
- package/lib/rules/unicode-bom.js +54 -54
- package/lib/rules/use-isnan.js +237 -110
- package/lib/rules/utils/ast-utils.js +2139 -1688
- package/lib/rules/utils/char-source.js +247 -0
- package/lib/rules/utils/fix-tracker.js +99 -88
- package/lib/rules/utils/keywords.js +59 -59
- package/lib/rules/utils/lazy-loading-rule-map.js +81 -78
- package/lib/rules/utils/regular-expressions.js +35 -19
- package/lib/rules/utils/unicode/index.js +9 -4
- package/lib/rules/utils/unicode/is-combining-character.js +1 -1
- package/lib/rules/utils/unicode/is-emoji-modifier.js +1 -1
- package/lib/rules/utils/unicode/is-regional-indicator-symbol.js +1 -1
- package/lib/rules/utils/unicode/is-surrogate-pair.js +1 -1
- package/lib/rules/valid-typeof.js +153 -109
- package/lib/rules/vars-on-top.js +152 -144
- package/lib/rules/wrap-iife.js +204 -173
- package/lib/rules/wrap-regex.js +77 -47
- package/lib/rules/yield-star-spacing.js +145 -116
- package/lib/rules/yoda.js +283 -274
- package/lib/services/parser-service.js +65 -0
- package/lib/services/processor-service.js +101 -0
- package/lib/services/suppressions-service.js +302 -0
- package/lib/services/warning-service.js +98 -0
- package/lib/shared/ajv.js +14 -14
- package/lib/shared/assert.js +21 -0
- package/lib/shared/ast-utils.js +7 -6
- package/lib/shared/deep-merge-arrays.js +62 -0
- package/lib/shared/directives.js +3 -2
- package/lib/shared/flags.js +108 -0
- package/lib/shared/logging.js +24 -16
- package/lib/shared/naming.js +109 -0
- package/lib/shared/option-utils.js +63 -0
- package/lib/shared/relative-module-resolver.js +18 -40
- package/lib/shared/runtime-info.js +138 -128
- package/lib/shared/serialization.js +78 -0
- package/lib/shared/severity.js +22 -22
- package/lib/shared/stats.js +30 -0
- package/lib/shared/string-utils.js +19 -21
- package/lib/shared/text-table.js +68 -0
- package/lib/shared/translate-cli-options.js +281 -0
- package/lib/shared/traverser.js +153 -146
- package/lib/types/config-api.d.ts +12 -0
- package/lib/types/index.d.ts +1473 -0
- package/lib/types/rules.d.ts +5589 -0
- package/lib/types/universal.d.ts +6 -0
- package/lib/types/use-at-your-own-risk.d.ts +87 -0
- package/lib/universal.js +10 -0
- package/lib/unsupported-api.js +8 -9
- package/messages/all-files-ignored.js +3 -3
- package/messages/all-matched-files-ignored.js +21 -0
- package/messages/config-file-missing.js +16 -0
- package/messages/config-plugin-missing.js +14 -0
- package/messages/config-serialize-function.js +30 -0
- package/messages/eslintrc-incompat.js +35 -16
- package/messages/eslintrc-plugins.js +8 -5
- package/messages/extend-config-missing.js +3 -3
- package/messages/failed-to-read-json.js +3 -3
- package/messages/file-not-found.js +3 -3
- package/messages/invalid-rule-options.js +2 -2
- package/messages/invalid-rule-severity.js +2 -2
- package/messages/no-config-found.js +4 -4
- package/messages/plugin-conflict.js +9 -9
- package/messages/plugin-invalid.js +4 -4
- package/messages/plugin-missing.js +4 -4
- package/messages/print-config-with-directory-path.js +2 -2
- package/messages/shared.js +6 -1
- package/messages/whitespace-found.js +3 -3
- package/package.json +105 -60
- package/conf/config-schema.js +0 -93
- package/lib/cli-engine/formatters/checkstyle.js +0 -60
- package/lib/cli-engine/formatters/compact.js +0 -60
- package/lib/cli-engine/formatters/jslint-xml.js +0 -41
- package/lib/cli-engine/formatters/junit.js +0 -82
- package/lib/cli-engine/formatters/tap.js +0 -95
- package/lib/cli-engine/formatters/unix.js +0 -58
- package/lib/cli-engine/formatters/visualstudio.js +0 -63
- package/lib/cli-engine/xml-escape.js +0 -34
- package/lib/config/flat-config-helpers.js +0 -111
- package/lib/config/rule-validator.js +0 -158
- package/lib/eslint/flat-eslint.js +0 -1159
- package/lib/linter/config-comment-parser.js +0 -185
- package/lib/linter/node-event-generator.js +0 -354
- package/lib/linter/report-translator.js +0 -369
- package/lib/linter/safe-emitter.js +0 -52
- package/lib/rule-tester/flat-rule-tester.js +0 -1131
- package/lib/rules/require-jsdoc.js +0 -122
- package/lib/rules/utils/patterns/letters.js +0 -36
- package/lib/rules/valid-jsdoc.js +0 -516
- package/lib/shared/config-validator.js +0 -347
- package/lib/shared/deprecation-warnings.js +0 -58
- package/lib/shared/types.js +0 -216
- package/lib/source-code/index.js +0 -5
- package/lib/source-code/source-code.js +0 -1055
- package/lib/source-code/token-store/backward-token-comment-cursor.js +0 -57
- package/lib/source-code/token-store/backward-token-cursor.js +0 -58
- package/lib/source-code/token-store/cursors.js +0 -90
- package/lib/source-code/token-store/forward-token-comment-cursor.js +0 -57
- package/lib/source-code/token-store/forward-token-cursor.js +0 -63
- package/lib/source-code/token-store/index.js +0 -627
- package/lib/source-code/token-store/padded-token-cursor.js +0 -38
- package/lib/source-code/token-store/utils.js +0 -107
|
@@ -5,84 +5,122 @@
|
|
|
5
5
|
"use strict";
|
|
6
6
|
|
|
7
7
|
//------------------------------------------------------------------------------
|
|
8
|
-
//
|
|
8
|
+
// Requirements
|
|
9
9
|
//------------------------------------------------------------------------------
|
|
10
10
|
|
|
11
|
-
|
|
12
|
-
module.exports = {
|
|
13
|
-
meta: {
|
|
14
|
-
type: "suggestion",
|
|
15
|
-
|
|
16
|
-
docs: {
|
|
17
|
-
description: "Disallow `if` statements as the only statement in `else` blocks",
|
|
18
|
-
recommended: false,
|
|
19
|
-
url: "https://eslint.org/docs/latest/rules/no-lonely-if"
|
|
20
|
-
},
|
|
11
|
+
const astUtils = require("./utils/ast-utils");
|
|
21
12
|
|
|
22
|
-
|
|
23
|
-
|
|
13
|
+
//------------------------------------------------------------------------------
|
|
14
|
+
// Rule Definition
|
|
15
|
+
//------------------------------------------------------------------------------
|
|
24
16
|
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
17
|
+
/** @type {import('../types').Rule.RuleModule} */
|
|
18
|
+
module.exports = {
|
|
19
|
+
meta: {
|
|
20
|
+
type: "suggestion",
|
|
29
21
|
|
|
30
|
-
|
|
31
|
-
|
|
22
|
+
docs: {
|
|
23
|
+
description:
|
|
24
|
+
"Disallow `if` statements as the only statement in `else` blocks",
|
|
25
|
+
recommended: false,
|
|
26
|
+
frozen: true,
|
|
27
|
+
url: "https://eslint.org/docs/latest/rules/no-lonely-if",
|
|
28
|
+
},
|
|
32
29
|
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
const parent = node.parent,
|
|
36
|
-
grandparent = parent.parent;
|
|
30
|
+
schema: [],
|
|
31
|
+
fixable: "code",
|
|
37
32
|
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
node,
|
|
44
|
-
messageId: "unexpectedLonelyIf",
|
|
45
|
-
fix(fixer) {
|
|
46
|
-
const openingElseCurly = sourceCode.getFirstToken(parent);
|
|
47
|
-
const closingElseCurly = sourceCode.getLastToken(parent);
|
|
48
|
-
const elseKeyword = sourceCode.getTokenBefore(openingElseCurly);
|
|
49
|
-
const tokenAfterElseBlock = sourceCode.getTokenAfter(closingElseCurly);
|
|
50
|
-
const lastIfToken = sourceCode.getLastToken(node.consequent);
|
|
51
|
-
const sourceText = sourceCode.getText();
|
|
33
|
+
messages: {
|
|
34
|
+
unexpectedLonelyIf:
|
|
35
|
+
"Unexpected if as the only statement in an else block.",
|
|
36
|
+
},
|
|
37
|
+
},
|
|
52
38
|
|
|
53
|
-
|
|
54
|
-
|
|
39
|
+
create(context) {
|
|
40
|
+
const sourceCode = context.sourceCode;
|
|
55
41
|
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
42
|
+
return {
|
|
43
|
+
IfStatement(node) {
|
|
44
|
+
const parent = node.parent,
|
|
45
|
+
grandparent = parent.parent;
|
|
59
46
|
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
47
|
+
if (
|
|
48
|
+
parent &&
|
|
49
|
+
parent.type === "BlockStatement" &&
|
|
50
|
+
parent.body.length === 1 &&
|
|
51
|
+
!astUtils.areBracesNecessary(parent, sourceCode) &&
|
|
52
|
+
grandparent &&
|
|
53
|
+
grandparent.type === "IfStatement" &&
|
|
54
|
+
parent === grandparent.alternate
|
|
55
|
+
) {
|
|
56
|
+
context.report({
|
|
57
|
+
node,
|
|
58
|
+
messageId: "unexpectedLonelyIf",
|
|
59
|
+
fix(fixer) {
|
|
60
|
+
const openingElseCurly =
|
|
61
|
+
sourceCode.getFirstToken(parent);
|
|
62
|
+
const closingElseCurly =
|
|
63
|
+
sourceCode.getLastToken(parent);
|
|
64
|
+
const elseKeyword =
|
|
65
|
+
sourceCode.getTokenBefore(openingElseCurly);
|
|
66
|
+
const tokenAfterElseBlock =
|
|
67
|
+
sourceCode.getTokenAfter(closingElseCurly);
|
|
68
|
+
const lastIfToken = sourceCode.getLastToken(
|
|
69
|
+
node.consequent,
|
|
70
|
+
);
|
|
71
|
+
const sourceText = sourceCode.getText();
|
|
69
72
|
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
73
|
+
if (
|
|
74
|
+
sourceText
|
|
75
|
+
.slice(
|
|
76
|
+
openingElseCurly.range[1],
|
|
77
|
+
node.range[0],
|
|
78
|
+
)
|
|
79
|
+
.trim() ||
|
|
80
|
+
sourceText
|
|
81
|
+
.slice(
|
|
82
|
+
node.range[1],
|
|
83
|
+
closingElseCurly.range[0],
|
|
84
|
+
)
|
|
85
|
+
.trim()
|
|
86
|
+
) {
|
|
87
|
+
// Don't fix if there are any non-whitespace characters interfering (e.g. comments)
|
|
88
|
+
return null;
|
|
89
|
+
}
|
|
76
90
|
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
91
|
+
if (
|
|
92
|
+
node.consequent.type !== "BlockStatement" &&
|
|
93
|
+
lastIfToken.value !== ";" &&
|
|
94
|
+
tokenAfterElseBlock &&
|
|
95
|
+
(node.consequent.loc.end.line ===
|
|
96
|
+
tokenAfterElseBlock.loc.start.line ||
|
|
97
|
+
/^[([/+`-]/u.test(
|
|
98
|
+
tokenAfterElseBlock.value,
|
|
99
|
+
) ||
|
|
100
|
+
lastIfToken.value === "++" ||
|
|
101
|
+
lastIfToken.value === "--")
|
|
102
|
+
) {
|
|
103
|
+
/*
|
|
104
|
+
* If the `if` statement has no block, and is not followed by a semicolon, make sure that fixing
|
|
105
|
+
* the issue would not change semantics due to ASI. If this would happen, don't do a fix.
|
|
106
|
+
*/
|
|
107
|
+
return null;
|
|
108
|
+
}
|
|
86
109
|
|
|
87
|
-
|
|
110
|
+
return fixer.replaceTextRange(
|
|
111
|
+
[
|
|
112
|
+
openingElseCurly.range[0],
|
|
113
|
+
closingElseCurly.range[1],
|
|
114
|
+
],
|
|
115
|
+
(elseKeyword.range[1] ===
|
|
116
|
+
openingElseCurly.range[0]
|
|
117
|
+
? " "
|
|
118
|
+
: "") + sourceCode.getText(node),
|
|
119
|
+
);
|
|
120
|
+
},
|
|
121
|
+
});
|
|
122
|
+
}
|
|
123
|
+
},
|
|
124
|
+
};
|
|
125
|
+
},
|
|
88
126
|
};
|
|
@@ -9,198 +9,259 @@
|
|
|
9
9
|
// Helpers
|
|
10
10
|
//------------------------------------------------------------------------------
|
|
11
11
|
|
|
12
|
-
|
|
13
|
-
* Gets the containing loop node of a specified node.
|
|
14
|
-
*
|
|
15
|
-
* We don't need to check nested functions, so this ignores those.
|
|
16
|
-
* `Scope.through` contains references of nested functions.
|
|
17
|
-
* @param {ASTNode} node An AST node to get.
|
|
18
|
-
* @returns {ASTNode|null} The containing loop node of the specified node, or
|
|
19
|
-
* `null`.
|
|
20
|
-
*/
|
|
21
|
-
function getContainingLoopNode(node) {
|
|
22
|
-
for (let currentNode = node; currentNode.parent; currentNode = currentNode.parent) {
|
|
23
|
-
const parent = currentNode.parent;
|
|
24
|
-
|
|
25
|
-
switch (parent.type) {
|
|
26
|
-
case "WhileStatement":
|
|
27
|
-
case "DoWhileStatement":
|
|
28
|
-
return parent;
|
|
29
|
-
|
|
30
|
-
case "ForStatement":
|
|
31
|
-
|
|
32
|
-
// `init` is outside of the loop.
|
|
33
|
-
if (parent.init !== currentNode) {
|
|
34
|
-
return parent;
|
|
35
|
-
}
|
|
36
|
-
break;
|
|
37
|
-
|
|
38
|
-
case "ForInStatement":
|
|
39
|
-
case "ForOfStatement":
|
|
40
|
-
|
|
41
|
-
// `right` is outside of the loop.
|
|
42
|
-
if (parent.right !== currentNode) {
|
|
43
|
-
return parent;
|
|
44
|
-
}
|
|
45
|
-
break;
|
|
46
|
-
|
|
47
|
-
case "ArrowFunctionExpression":
|
|
48
|
-
case "FunctionExpression":
|
|
49
|
-
case "FunctionDeclaration":
|
|
50
|
-
|
|
51
|
-
// We don't need to check nested functions.
|
|
52
|
-
return null;
|
|
53
|
-
|
|
54
|
-
default:
|
|
55
|
-
break;
|
|
56
|
-
}
|
|
57
|
-
}
|
|
58
|
-
|
|
59
|
-
return null;
|
|
60
|
-
}
|
|
61
|
-
|
|
62
|
-
/**
|
|
63
|
-
* Gets the containing loop node of a given node.
|
|
64
|
-
* If the loop was nested, this returns the most outer loop.
|
|
65
|
-
* @param {ASTNode} node A node to get. This is a loop node.
|
|
66
|
-
* @param {ASTNode|null} excludedNode A node that the result node should not
|
|
67
|
-
* include.
|
|
68
|
-
* @returns {ASTNode} The most outer loop node.
|
|
69
|
-
*/
|
|
70
|
-
function getTopLoopNode(node, excludedNode) {
|
|
71
|
-
const border = excludedNode ? excludedNode.range[1] : 0;
|
|
72
|
-
let retv = node;
|
|
73
|
-
let containingLoopNode = node;
|
|
74
|
-
|
|
75
|
-
while (containingLoopNode && containingLoopNode.range[0] >= border) {
|
|
76
|
-
retv = containingLoopNode;
|
|
77
|
-
containingLoopNode = getContainingLoopNode(containingLoopNode);
|
|
78
|
-
}
|
|
79
|
-
|
|
80
|
-
return retv;
|
|
81
|
-
}
|
|
12
|
+
const CONSTANT_BINDINGS = new Set(["const", "using", "await using"]);
|
|
82
13
|
|
|
83
14
|
/**
|
|
84
|
-
*
|
|
85
|
-
*
|
|
86
|
-
* @
|
|
87
|
-
* @param {eslint-scope.Reference} reference A reference to check.
|
|
88
|
-
* @returns {boolean} `true` if the reference is safe or not.
|
|
15
|
+
* Identifies is a node is a FunctionExpression which is part of an IIFE
|
|
16
|
+
* @param {ASTNode} node Node to test
|
|
17
|
+
* @returns {boolean} True if it's an IIFE
|
|
89
18
|
*/
|
|
90
|
-
function
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
// Variables which are declared by `const` is safe.
|
|
99
|
-
if (kind === "const") {
|
|
100
|
-
return true;
|
|
101
|
-
}
|
|
102
|
-
|
|
103
|
-
/*
|
|
104
|
-
* Variables which are declared by `let` in the loop is safe.
|
|
105
|
-
* It's a different instance from the next loop step's.
|
|
106
|
-
*/
|
|
107
|
-
if (kind === "let" &&
|
|
108
|
-
declaration.range[0] > loopNode.range[0] &&
|
|
109
|
-
declaration.range[1] < loopNode.range[1]
|
|
110
|
-
) {
|
|
111
|
-
return true;
|
|
112
|
-
}
|
|
113
|
-
|
|
114
|
-
/*
|
|
115
|
-
* WriteReferences which exist after this border are unsafe because those
|
|
116
|
-
* can modify the variable.
|
|
117
|
-
*/
|
|
118
|
-
const border = getTopLoopNode(
|
|
119
|
-
loopNode,
|
|
120
|
-
(kind === "let") ? declaration : null
|
|
121
|
-
).range[0];
|
|
122
|
-
|
|
123
|
-
/**
|
|
124
|
-
* Checks whether a given reference is safe or not.
|
|
125
|
-
* The reference is every reference of the upper scope's variable we are
|
|
126
|
-
* looking now.
|
|
127
|
-
*
|
|
128
|
-
* It's safe if the reference matches one of the following condition.
|
|
129
|
-
* - is readonly.
|
|
130
|
-
* - doesn't exist inside a local function and after the border.
|
|
131
|
-
* @param {eslint-scope.Reference} upperRef A reference to check.
|
|
132
|
-
* @returns {boolean} `true` if the reference is safe.
|
|
133
|
-
*/
|
|
134
|
-
function isSafeReference(upperRef) {
|
|
135
|
-
const id = upperRef.identifier;
|
|
136
|
-
|
|
137
|
-
return (
|
|
138
|
-
!upperRef.isWrite() ||
|
|
139
|
-
variable.scope.variableScope === upperRef.from.variableScope &&
|
|
140
|
-
id.range[0] < border
|
|
141
|
-
);
|
|
142
|
-
}
|
|
143
|
-
|
|
144
|
-
return Boolean(variable) && variable.references.every(isSafeReference);
|
|
19
|
+
function isIIFE(node) {
|
|
20
|
+
return (
|
|
21
|
+
(node.type === "FunctionExpression" ||
|
|
22
|
+
node.type === "ArrowFunctionExpression") &&
|
|
23
|
+
node.parent &&
|
|
24
|
+
node.parent.type === "CallExpression" &&
|
|
25
|
+
node.parent.callee === node
|
|
26
|
+
);
|
|
145
27
|
}
|
|
146
28
|
|
|
147
29
|
//------------------------------------------------------------------------------
|
|
148
30
|
// Rule Definition
|
|
149
31
|
//------------------------------------------------------------------------------
|
|
150
32
|
|
|
151
|
-
/** @type {import('../
|
|
33
|
+
/** @type {import('../types').Rule.RuleModule} */
|
|
152
34
|
module.exports = {
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
|
|
163
|
-
|
|
164
|
-
|
|
165
|
-
|
|
166
|
-
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
|
|
171
|
-
|
|
172
|
-
|
|
173
|
-
|
|
174
|
-
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
|
|
178
|
-
|
|
179
|
-
|
|
180
|
-
|
|
181
|
-
|
|
182
|
-
|
|
183
|
-
|
|
184
|
-
|
|
185
|
-
|
|
186
|
-
|
|
187
|
-
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
|
|
191
|
-
|
|
192
|
-
|
|
193
|
-
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
197
|
-
|
|
198
|
-
|
|
199
|
-
|
|
200
|
-
|
|
201
|
-
|
|
202
|
-
|
|
203
|
-
|
|
204
|
-
|
|
205
|
-
|
|
35
|
+
meta: {
|
|
36
|
+
type: "suggestion",
|
|
37
|
+
dialects: ["typescript", "javascript"],
|
|
38
|
+
language: "javascript",
|
|
39
|
+
|
|
40
|
+
docs: {
|
|
41
|
+
description:
|
|
42
|
+
"Disallow function declarations that contain unsafe references inside loop statements",
|
|
43
|
+
recommended: false,
|
|
44
|
+
url: "https://eslint.org/docs/latest/rules/no-loop-func",
|
|
45
|
+
},
|
|
46
|
+
|
|
47
|
+
schema: [],
|
|
48
|
+
|
|
49
|
+
messages: {
|
|
50
|
+
unsafeRefs:
|
|
51
|
+
"Function declared in a loop contains unsafe references to variable(s) {{ varNames }}.",
|
|
52
|
+
},
|
|
53
|
+
},
|
|
54
|
+
|
|
55
|
+
create(context) {
|
|
56
|
+
const SKIPPED_IIFE_NODES = new Set();
|
|
57
|
+
const sourceCode = context.sourceCode;
|
|
58
|
+
|
|
59
|
+
/**
|
|
60
|
+
* Gets the containing loop node of a specified node.
|
|
61
|
+
*
|
|
62
|
+
* We don't need to check nested functions, so this ignores those, with the exception of IIFE.
|
|
63
|
+
* `Scope.through` contains references of nested functions.
|
|
64
|
+
* @param {ASTNode} node An AST node to get.
|
|
65
|
+
* @returns {ASTNode|null} The containing loop node of the specified node, or
|
|
66
|
+
* `null`.
|
|
67
|
+
*/
|
|
68
|
+
function getContainingLoopNode(node) {
|
|
69
|
+
for (
|
|
70
|
+
let currentNode = node;
|
|
71
|
+
currentNode.parent;
|
|
72
|
+
currentNode = currentNode.parent
|
|
73
|
+
) {
|
|
74
|
+
const parent = currentNode.parent;
|
|
75
|
+
|
|
76
|
+
switch (parent.type) {
|
|
77
|
+
case "WhileStatement":
|
|
78
|
+
case "DoWhileStatement":
|
|
79
|
+
return parent;
|
|
80
|
+
|
|
81
|
+
case "ForStatement":
|
|
82
|
+
// `init` is outside of the loop.
|
|
83
|
+
if (parent.init !== currentNode) {
|
|
84
|
+
return parent;
|
|
85
|
+
}
|
|
86
|
+
break;
|
|
87
|
+
|
|
88
|
+
case "ForInStatement":
|
|
89
|
+
case "ForOfStatement":
|
|
90
|
+
// `right` is outside of the loop.
|
|
91
|
+
if (parent.right !== currentNode) {
|
|
92
|
+
return parent;
|
|
93
|
+
}
|
|
94
|
+
break;
|
|
95
|
+
|
|
96
|
+
case "ArrowFunctionExpression":
|
|
97
|
+
case "FunctionExpression":
|
|
98
|
+
case "FunctionDeclaration":
|
|
99
|
+
// We need to check nested functions only in case of IIFE.
|
|
100
|
+
if (SKIPPED_IIFE_NODES.has(parent)) {
|
|
101
|
+
break;
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
return null;
|
|
105
|
+
default:
|
|
106
|
+
break;
|
|
107
|
+
}
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
return null;
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
/**
|
|
114
|
+
* Gets the containing loop node of a given node.
|
|
115
|
+
* If the loop was nested, this returns the most outer loop.
|
|
116
|
+
* @param {ASTNode} node A node to get. This is a loop node.
|
|
117
|
+
* @param {ASTNode|null} excludedNode A node that the result node should not
|
|
118
|
+
* include.
|
|
119
|
+
* @returns {ASTNode} The most outer loop node.
|
|
120
|
+
*/
|
|
121
|
+
function getTopLoopNode(node, excludedNode) {
|
|
122
|
+
const border = excludedNode ? excludedNode.range[1] : 0;
|
|
123
|
+
let retv = node;
|
|
124
|
+
let containingLoopNode = node;
|
|
125
|
+
|
|
126
|
+
while (
|
|
127
|
+
containingLoopNode &&
|
|
128
|
+
containingLoopNode.range[0] >= border
|
|
129
|
+
) {
|
|
130
|
+
retv = containingLoopNode;
|
|
131
|
+
containingLoopNode = getContainingLoopNode(containingLoopNode);
|
|
132
|
+
}
|
|
133
|
+
|
|
134
|
+
return retv;
|
|
135
|
+
}
|
|
136
|
+
|
|
137
|
+
/**
|
|
138
|
+
* Checks whether a given reference which refers to an upper scope's variable is
|
|
139
|
+
* safe or not.
|
|
140
|
+
* @param {ASTNode} loopNode A containing loop node.
|
|
141
|
+
* @param {eslint-scope.Reference} reference A reference to check.
|
|
142
|
+
* @returns {boolean} `true` if the reference is safe or not.
|
|
143
|
+
*/
|
|
144
|
+
function isSafe(loopNode, reference) {
|
|
145
|
+
const variable = reference.resolved;
|
|
146
|
+
const definition = variable && variable.defs[0];
|
|
147
|
+
const declaration = definition && definition.parent;
|
|
148
|
+
const kind =
|
|
149
|
+
declaration && declaration.type === "VariableDeclaration"
|
|
150
|
+
? declaration.kind
|
|
151
|
+
: "";
|
|
152
|
+
|
|
153
|
+
// Constant variables are safe.
|
|
154
|
+
if (CONSTANT_BINDINGS.has(kind)) {
|
|
155
|
+
return true;
|
|
156
|
+
}
|
|
157
|
+
|
|
158
|
+
/*
|
|
159
|
+
* Variables which are declared by `let` in the loop is safe.
|
|
160
|
+
* It's a different instance from the next loop step's.
|
|
161
|
+
*/
|
|
162
|
+
if (
|
|
163
|
+
kind === "let" &&
|
|
164
|
+
declaration.range[0] > loopNode.range[0] &&
|
|
165
|
+
declaration.range[1] < loopNode.range[1]
|
|
166
|
+
) {
|
|
167
|
+
return true;
|
|
168
|
+
}
|
|
169
|
+
|
|
170
|
+
/*
|
|
171
|
+
* WriteReferences which exist after this border are unsafe because those
|
|
172
|
+
* can modify the variable.
|
|
173
|
+
*/
|
|
174
|
+
const border = getTopLoopNode(
|
|
175
|
+
loopNode,
|
|
176
|
+
kind === "let" ? declaration : null,
|
|
177
|
+
).range[0];
|
|
178
|
+
|
|
179
|
+
/**
|
|
180
|
+
* Checks whether a given reference is safe or not.
|
|
181
|
+
* The reference is every reference of the upper scope's variable we are
|
|
182
|
+
* looking now.
|
|
183
|
+
*
|
|
184
|
+
* It's safe if the reference matches one of the following condition.
|
|
185
|
+
* - is readonly.
|
|
186
|
+
* - doesn't exist inside a local function and after the border.
|
|
187
|
+
* @param {eslint-scope.Reference} upperRef A reference to check.
|
|
188
|
+
* @returns {boolean} `true` if the reference is safe.
|
|
189
|
+
*/
|
|
190
|
+
function isSafeReference(upperRef) {
|
|
191
|
+
const id = upperRef.identifier;
|
|
192
|
+
|
|
193
|
+
return (
|
|
194
|
+
!upperRef.isWrite() ||
|
|
195
|
+
(variable.scope.variableScope ===
|
|
196
|
+
upperRef.from.variableScope &&
|
|
197
|
+
id.range[0] < border)
|
|
198
|
+
);
|
|
199
|
+
}
|
|
200
|
+
|
|
201
|
+
return (
|
|
202
|
+
Boolean(variable) && variable.references.every(isSafeReference)
|
|
203
|
+
);
|
|
204
|
+
}
|
|
205
|
+
|
|
206
|
+
/**
|
|
207
|
+
* Reports functions which match the following condition:
|
|
208
|
+
*
|
|
209
|
+
* - has a loop node in ancestors.
|
|
210
|
+
* - has any references which refers to an unsafe variable.
|
|
211
|
+
* @param {ASTNode} node The AST node to check.
|
|
212
|
+
* @returns {void}
|
|
213
|
+
*/
|
|
214
|
+
function checkForLoops(node) {
|
|
215
|
+
const loopNode = getContainingLoopNode(node);
|
|
216
|
+
|
|
217
|
+
if (!loopNode) {
|
|
218
|
+
return;
|
|
219
|
+
}
|
|
220
|
+
|
|
221
|
+
const references = sourceCode.getScope(node).through;
|
|
222
|
+
|
|
223
|
+
// Check if the function is not asynchronous or a generator function
|
|
224
|
+
if (!(node.async || node.generator)) {
|
|
225
|
+
if (isIIFE(node)) {
|
|
226
|
+
const isFunctionExpression =
|
|
227
|
+
node.type === "FunctionExpression";
|
|
228
|
+
|
|
229
|
+
// Check if the function is referenced elsewhere in the code
|
|
230
|
+
const isFunctionReferenced =
|
|
231
|
+
isFunctionExpression && node.id
|
|
232
|
+
? references.some(
|
|
233
|
+
r => r.identifier.name === node.id.name,
|
|
234
|
+
)
|
|
235
|
+
: false;
|
|
236
|
+
|
|
237
|
+
if (!isFunctionReferenced) {
|
|
238
|
+
SKIPPED_IIFE_NODES.add(node);
|
|
239
|
+
return;
|
|
240
|
+
}
|
|
241
|
+
}
|
|
242
|
+
}
|
|
243
|
+
|
|
244
|
+
const unsafeRefs = [
|
|
245
|
+
...new Set(
|
|
246
|
+
references
|
|
247
|
+
.filter(r => r.resolved && !isSafe(loopNode, r))
|
|
248
|
+
.map(r => r.identifier.name),
|
|
249
|
+
),
|
|
250
|
+
];
|
|
251
|
+
|
|
252
|
+
if (unsafeRefs.length > 0) {
|
|
253
|
+
context.report({
|
|
254
|
+
node,
|
|
255
|
+
messageId: "unsafeRefs",
|
|
256
|
+
data: { varNames: `'${unsafeRefs.join("', '")}'` },
|
|
257
|
+
});
|
|
258
|
+
}
|
|
259
|
+
}
|
|
260
|
+
|
|
261
|
+
return {
|
|
262
|
+
ArrowFunctionExpression: checkForLoops,
|
|
263
|
+
FunctionExpression: checkForLoops,
|
|
264
|
+
FunctionDeclaration: checkForLoops,
|
|
265
|
+
};
|
|
266
|
+
},
|
|
206
267
|
};
|