bahlint 28.58.6934
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/LICENSE +19 -0
- package/README.md +370 -0
- package/bin/eslint.js +195 -0
- package/conf/ecma-version.js +16 -0
- package/conf/globals.js +169 -0
- package/conf/replacements.json +26 -0
- package/conf/rule-type-list.json +91 -0
- package/lib/api.js +39 -0
- package/lib/cli-engine/formatters/formatters-meta.json +22 -0
- package/lib/cli-engine/formatters/gasoline.js +168 -0
- package/lib/cli-engine/formatters/html.js +359 -0
- package/lib/cli-engine/formatters/json-with-metadata.js +16 -0
- package/lib/cli-engine/formatters/json.js +13 -0
- package/lib/cli-engine/formatters/stylish.js +153 -0
- package/lib/cli-engine/hash.js +35 -0
- package/lib/cli-engine/lint-result-cache.js +220 -0
- package/lib/cli.js +607 -0
- package/lib/config/config-loader.js +683 -0
- package/lib/config/config.js +674 -0
- package/lib/config/default-config.js +78 -0
- package/lib/config/flat-config-array.js +217 -0
- package/lib/config/flat-config-schema.js +598 -0
- package/lib/config-api.js +12 -0
- package/lib/eslint/eslint-helpers.js +1462 -0
- package/lib/eslint/eslint.js +1364 -0
- package/lib/eslint/index.js +7 -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 +1178 -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/languages/js/source-code/token-store/cursor.js +76 -0
- package/lib/languages/js/source-code/token-store/cursors.js +120 -0
- package/lib/languages/js/source-code/token-store/decorative-cursor.js +38 -0
- package/lib/languages/js/source-code/token-store/filter-cursor.js +42 -0
- 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 +695 -0
- package/lib/languages/js/source-code/token-store/limit-cursor.js +39 -0
- package/lib/languages/js/source-code/token-store/padded-token-cursor.js +45 -0
- package/lib/languages/js/source-code/token-store/skip-cursor.js +41 -0
- package/lib/languages/js/source-code/token-store/utils.js +131 -0
- package/lib/languages/js/validate-language-options.js +196 -0
- package/lib/linter/apply-disable-directives.js +583 -0
- package/lib/linter/code-path-analysis/code-path-analyzer.js +828 -0
- package/lib/linter/code-path-analysis/code-path-segment.js +262 -0
- package/lib/linter/code-path-analysis/code-path-state.js +2370 -0
- package/lib/linter/code-path-analysis/code-path.js +332 -0
- package/lib/linter/code-path-analysis/debug-helpers.js +223 -0
- package/lib/linter/code-path-analysis/fork-context.js +374 -0
- package/lib/linter/code-path-analysis/id-generator.js +44 -0
- package/lib/linter/esquery.js +332 -0
- package/lib/linter/file-context.js +88 -0
- package/lib/linter/file-report.js +604 -0
- package/lib/linter/index.js +11 -0
- package/lib/linter/interpolate.js +50 -0
- package/lib/linter/linter.js +1614 -0
- package/lib/linter/rule-fixer.js +199 -0
- package/lib/linter/source-code-fixer.js +154 -0
- package/lib/linter/source-code-traverser.js +333 -0
- package/lib/linter/source-code-visitor.js +81 -0
- package/lib/linter/timing.js +209 -0
- package/lib/linter/vfile.js +115 -0
- package/lib/options.js +416 -0
- package/lib/rule-tester/index.js +7 -0
- package/lib/rule-tester/rule-tester.js +1817 -0
- package/lib/rules/accessor-pairs.js +420 -0
- package/lib/rules/array-bracket-newline.js +291 -0
- package/lib/rules/array-bracket-spacing.js +301 -0
- package/lib/rules/array-callback-return.js +493 -0
- package/lib/rules/array-element-newline.js +374 -0
- package/lib/rules/arrow-body-style.js +418 -0
- package/lib/rules/arrow-parens.js +237 -0
- package/lib/rules/arrow-spacing.js +188 -0
- package/lib/rules/block-scoped-var.js +137 -0
- package/lib/rules/block-spacing.js +202 -0
- package/lib/rules/brace-style.js +278 -0
- package/lib/rules/callback-return.js +216 -0
- package/lib/rules/camelcase.js +422 -0
- package/lib/rules/capitalized-comments.js +325 -0
- package/lib/rules/class-methods-use-this.js +250 -0
- package/lib/rules/comma-dangle.js +424 -0
- package/lib/rules/comma-spacing.js +205 -0
- package/lib/rules/comma-style.js +391 -0
- package/lib/rules/complexity.js +201 -0
- package/lib/rules/computed-property-spacing.js +251 -0
- package/lib/rules/consistent-return.js +221 -0
- package/lib/rules/consistent-this.js +179 -0
- package/lib/rules/constructor-super.js +453 -0
- package/lib/rules/curly.js +425 -0
- package/lib/rules/default-case-last.js +51 -0
- package/lib/rules/default-case.js +103 -0
- package/lib/rules/default-param-last.js +78 -0
- package/lib/rules/dot-location.js +138 -0
- package/lib/rules/dot-notation.js +216 -0
- package/lib/rules/eol-last.js +135 -0
- package/lib/rules/eqeqeq.js +210 -0
- package/lib/rules/for-direction.js +168 -0
- package/lib/rules/func-call-spacing.js +281 -0
- package/lib/rules/func-name-matching.js +338 -0
- package/lib/rules/func-names.js +194 -0
- package/lib/rules/func-style.js +221 -0
- package/lib/rules/function-call-argument-newline.js +166 -0
- package/lib/rules/function-paren-newline.js +368 -0
- package/lib/rules/generator-star-spacing.js +246 -0
- package/lib/rules/getter-return.js +242 -0
- package/lib/rules/global-require.js +117 -0
- package/lib/rules/grouped-accessor-pairs.js +268 -0
- package/lib/rules/guard-for-in.js +85 -0
- package/lib/rules/handle-callback-err.js +122 -0
- package/lib/rules/id-blacklist.js +241 -0
- package/lib/rules/id-denylist.js +223 -0
- package/lib/rules/id-length.js +217 -0
- package/lib/rules/id-match.js +363 -0
- package/lib/rules/implicit-arrow-linebreak.js +125 -0
- package/lib/rules/indent-legacy.js +1369 -0
- package/lib/rules/indent.js +2334 -0
- package/lib/rules/index.js +332 -0
- package/lib/rules/init-declarations.js +172 -0
- package/lib/rules/jsx-quotes.js +128 -0
- package/lib/rules/key-spacing.js +822 -0
- package/lib/rules/keyword-spacing.js +701 -0
- package/lib/rules/line-comment-position.js +157 -0
- package/lib/rules/linebreak-style.js +135 -0
- package/lib/rules/lines-around-comment.js +581 -0
- package/lib/rules/lines-around-directive.js +249 -0
- package/lib/rules/lines-between-class-members.js +358 -0
- package/lib/rules/logical-assignment-operators.js +688 -0
- package/lib/rules/max-classes-per-file.js +90 -0
- package/lib/rules/max-depth.js +159 -0
- package/lib/rules/max-len.js +497 -0
- package/lib/rules/max-lines-per-function.js +238 -0
- package/lib/rules/max-lines.js +189 -0
- package/lib/rules/max-nested-callbacks.js +115 -0
- package/lib/rules/max-params.js +148 -0
- package/lib/rules/max-statements-per-line.js +224 -0
- package/lib/rules/max-statements.js +188 -0
- package/lib/rules/multiline-comment-style.js +652 -0
- package/lib/rules/multiline-ternary.js +257 -0
- package/lib/rules/new-cap.js +277 -0
- package/lib/rules/new-parens.js +120 -0
- package/lib/rules/newline-after-var.js +307 -0
- package/lib/rules/newline-before-return.js +242 -0
- package/lib/rules/newline-per-chained-call.js +159 -0
- package/lib/rules/no-alert.js +149 -0
- package/lib/rules/no-array-constructor.js +195 -0
- package/lib/rules/no-async-promise-executor.js +45 -0
- package/lib/rules/no-await-in-loop.js +115 -0
- package/lib/rules/no-bitwise.js +145 -0
- package/lib/rules/no-buffer-constructor.js +74 -0
- package/lib/rules/no-caller.js +52 -0
- package/lib/rules/no-case-declarations.js +80 -0
- package/lib/rules/no-catch-shadow.js +96 -0
- package/lib/rules/no-class-assign.js +66 -0
- package/lib/rules/no-compare-neg-zero.js +74 -0
- package/lib/rules/no-cond-assign.js +175 -0
- package/lib/rules/no-confusing-arrow.js +127 -0
- package/lib/rules/no-console.js +221 -0
- package/lib/rules/no-const-assign.js +73 -0
- package/lib/rules/no-constant-binary-expression.js +603 -0
- package/lib/rules/no-constant-condition.js +177 -0
- package/lib/rules/no-constructor-return.js +62 -0
- package/lib/rules/no-continue.js +38 -0
- package/lib/rules/no-control-regex.js +142 -0
- package/lib/rules/no-debugger.js +41 -0
- package/lib/rules/no-delete-var.js +42 -0
- package/lib/rules/no-div-regex.js +60 -0
- package/lib/rules/no-dupe-args.js +92 -0
- package/lib/rules/no-dupe-class-members.js +117 -0
- package/lib/rules/no-dupe-else-if.js +145 -0
- package/lib/rules/no-dupe-keys.js +165 -0
- package/lib/rules/no-duplicate-case.js +78 -0
- package/lib/rules/no-duplicate-imports.js +368 -0
- package/lib/rules/no-else-return.js +450 -0
- package/lib/rules/no-empty-character-class.js +83 -0
- package/lib/rules/no-empty-function.js +236 -0
- package/lib/rules/no-empty-pattern.js +85 -0
- package/lib/rules/no-empty-static-block.js +73 -0
- package/lib/rules/no-empty.js +153 -0
- package/lib/rules/no-eq-null.js +51 -0
- package/lib/rules/no-eval.js +295 -0
- package/lib/rules/no-ex-assign.js +57 -0
- package/lib/rules/no-extend-native.js +180 -0
- package/lib/rules/no-extra-bind.js +224 -0
- package/lib/rules/no-extra-boolean-cast.js +420 -0
- package/lib/rules/no-extra-label.js +169 -0
- package/lib/rules/no-extra-parens.js +1669 -0
- package/lib/rules/no-extra-semi.js +167 -0
- package/lib/rules/no-fallthrough.js +260 -0
- package/lib/rules/no-floating-decimal.js +99 -0
- package/lib/rules/no-func-assign.js +77 -0
- package/lib/rules/no-global-assign.js +101 -0
- package/lib/rules/no-implicit-coercion.js +468 -0
- package/lib/rules/no-implicit-globals.js +187 -0
- package/lib/rules/no-implied-eval.js +170 -0
- package/lib/rules/no-import-assign.js +227 -0
- package/lib/rules/no-inline-comments.js +115 -0
- package/lib/rules/no-inner-declarations.js +147 -0
- package/lib/rules/no-invalid-regexp.js +244 -0
- package/lib/rules/no-invalid-this.js +178 -0
- package/lib/rules/no-irregular-whitespace.js +292 -0
- package/lib/rules/no-iterator.js +48 -0
- package/lib/rules/no-label-var.js +78 -0
- package/lib/rules/no-labels.js +156 -0
- package/lib/rules/no-lone-blocks.js +140 -0
- package/lib/rules/no-lonely-if.js +126 -0
- package/lib/rules/no-loop-func.js +267 -0
- package/lib/rules/no-loss-of-precision.js +249 -0
- package/lib/rules/no-magic-numbers.js +365 -0
- package/lib/rules/no-misleading-character-class.js +595 -0
- package/lib/rules/no-mixed-operators.js +253 -0
- package/lib/rules/no-mixed-requires.js +267 -0
- package/lib/rules/no-mixed-spaces-and-tabs.js +148 -0
- package/lib/rules/no-multi-assign.js +66 -0
- package/lib/rules/no-multi-spaces.js +179 -0
- package/lib/rules/no-multi-str.js +67 -0
- package/lib/rules/no-multiple-empty-lines.js +210 -0
- package/lib/rules/no-native-reassign.js +114 -0
- package/lib/rules/no-negated-condition.js +100 -0
- package/lib/rules/no-negated-in-lhs.js +59 -0
- package/lib/rules/no-nested-ternary.js +46 -0
- package/lib/rules/no-new-func.js +96 -0
- package/lib/rules/no-new-native-nonconstructor.js +70 -0
- package/lib/rules/no-new-object.js +76 -0
- package/lib/rules/no-new-require.js +67 -0
- package/lib/rules/no-new-symbol.js +74 -0
- package/lib/rules/no-new-wrappers.js +62 -0
- package/lib/rules/no-new.js +42 -0
- package/lib/rules/no-nonoctal-decimal-escape.js +176 -0
- package/lib/rules/no-obj-calls.js +99 -0
- package/lib/rules/no-object-constructor.js +124 -0
- package/lib/rules/no-octal-escape.js +53 -0
- package/lib/rules/no-octal.js +42 -0
- package/lib/rules/no-param-reassign.js +248 -0
- package/lib/rules/no-path-concat.js +79 -0
- package/lib/rules/no-plusplus.js +102 -0
- package/lib/rules/no-process-env.js +68 -0
- package/lib/rules/no-process-exit.js +67 -0
- package/lib/rules/no-promise-executor-return.js +264 -0
- package/lib/rules/no-proto.js +45 -0
- package/lib/rules/no-prototype-builtins.js +181 -0
- package/lib/rules/no-redeclare.js +173 -0
- package/lib/rules/no-regex-spaces.js +219 -0
- package/lib/rules/no-restricted-exports.js +227 -0
- package/lib/rules/no-restricted-globals.js +266 -0
- package/lib/rules/no-restricted-imports.js +892 -0
- package/lib/rules/no-restricted-modules.js +249 -0
- package/lib/rules/no-restricted-properties.js +233 -0
- package/lib/rules/no-restricted-syntax.js +74 -0
- package/lib/rules/no-return-assign.js +87 -0
- package/lib/rules/no-return-await.js +162 -0
- package/lib/rules/no-script-url.js +68 -0
- package/lib/rules/no-self-assign.js +186 -0
- package/lib/rules/no-self-compare.js +77 -0
- package/lib/rules/no-sequences.js +158 -0
- package/lib/rules/no-setter-return.js +224 -0
- package/lib/rules/no-shadow-restricted-names.js +113 -0
- package/lib/rules/no-shadow.js +624 -0
- package/lib/rules/no-spaced-func.js +105 -0
- package/lib/rules/no-sparse-arrays.js +68 -0
- package/lib/rules/no-sync.js +81 -0
- package/lib/rules/no-tabs.js +110 -0
- package/lib/rules/no-template-curly-in-string.js +45 -0
- package/lib/rules/no-ternary.js +38 -0
- package/lib/rules/no-this-before-super.js +365 -0
- package/lib/rules/no-throw-literal.js +46 -0
- package/lib/rules/no-trailing-spaces.js +227 -0
- package/lib/rules/no-unassigned-vars.js +80 -0
- package/lib/rules/no-undef-init.js +101 -0
- package/lib/rules/no-undef.js +84 -0
- package/lib/rules/no-undefined.js +85 -0
- package/lib/rules/no-underscore-dangle.js +383 -0
- package/lib/rules/no-unexpected-multiline.js +130 -0
- package/lib/rules/no-unmodified-loop-condition.js +360 -0
- package/lib/rules/no-unneeded-ternary.js +232 -0
- package/lib/rules/no-unreachable-loop.js +190 -0
- package/lib/rules/no-unreachable.js +300 -0
- package/lib/rules/no-unsafe-finally.js +119 -0
- package/lib/rules/no-unsafe-negation.js +152 -0
- package/lib/rules/no-unsafe-optional-chaining.js +221 -0
- package/lib/rules/no-unused-expressions.js +227 -0
- package/lib/rules/no-unused-labels.js +158 -0
- package/lib/rules/no-unused-private-class-members.js +219 -0
- package/lib/rules/no-unused-vars.js +1739 -0
- package/lib/rules/no-use-before-define.js +446 -0
- package/lib/rules/no-useless-assignment.js +657 -0
- package/lib/rules/no-useless-backreference.js +263 -0
- package/lib/rules/no-useless-call.js +95 -0
- package/lib/rules/no-useless-catch.js +57 -0
- package/lib/rules/no-useless-computed-key.js +204 -0
- package/lib/rules/no-useless-concat.js +121 -0
- package/lib/rules/no-useless-constructor.js +262 -0
- package/lib/rules/no-useless-escape.js +406 -0
- package/lib/rules/no-useless-rename.js +202 -0
- package/lib/rules/no-useless-return.js +401 -0
- package/lib/rules/no-var.js +367 -0
- package/lib/rules/no-void.js +69 -0
- package/lib/rules/no-warning-comments.js +209 -0
- package/lib/rules/no-whitespace-before-property.js +150 -0
- package/lib/rules/no-with.js +37 -0
- package/lib/rules/nonblock-statement-body-position.js +164 -0
- package/lib/rules/object-curly-newline.js +383 -0
- package/lib/rules/object-curly-spacing.js +369 -0
- package/lib/rules/object-property-newline.js +151 -0
- package/lib/rules/object-shorthand.js +652 -0
- package/lib/rules/one-var-declaration-per-line.js +117 -0
- package/lib/rules/one-var.js +717 -0
- package/lib/rules/operator-assignment.js +270 -0
- package/lib/rules/operator-linebreak.js +315 -0
- package/lib/rules/padded-blocks.js +366 -0
- package/lib/rules/padding-line-between-statements.js +612 -0
- package/lib/rules/prefer-arrow-callback.js +437 -0
- package/lib/rules/prefer-const.js +546 -0
- package/lib/rules/prefer-destructuring.js +332 -0
- package/lib/rules/prefer-exponentiation-operator.js +235 -0
- package/lib/rules/prefer-named-capture-group.js +197 -0
- package/lib/rules/prefer-numeric-literals.js +157 -0
- package/lib/rules/prefer-object-has-own.js +148 -0
- package/lib/rules/prefer-object-spread.js +319 -0
- package/lib/rules/prefer-promise-reject-errors.js +154 -0
- package/lib/rules/prefer-reflect.js +150 -0
- package/lib/rules/prefer-regex-literals.js +605 -0
- package/lib/rules/prefer-rest-params.js +117 -0
- package/lib/rules/prefer-spread.js +91 -0
- package/lib/rules/prefer-template.js +347 -0
- package/lib/rules/preserve-caught-error.js +535 -0
- package/lib/rules/quote-props.js +394 -0
- package/lib/rules/quotes.js +416 -0
- package/lib/rules/radix.js +193 -0
- package/lib/rules/require-atomic-updates.js +365 -0
- package/lib/rules/require-await.js +184 -0
- package/lib/rules/require-unicode-regexp.js +317 -0
- package/lib/rules/require-yield.js +86 -0
- package/lib/rules/rest-spread-spacing.js +150 -0
- package/lib/rules/semi-spacing.js +297 -0
- package/lib/rules/semi-style.js +218 -0
- package/lib/rules/semi.js +476 -0
- package/lib/rules/sort-imports.js +319 -0
- package/lib/rules/sort-keys.js +268 -0
- package/lib/rules/sort-vars.js +140 -0
- package/lib/rules/space-before-blocks.js +232 -0
- package/lib/rules/space-before-function-paren.js +202 -0
- package/lib/rules/space-in-parens.js +374 -0
- package/lib/rules/space-infix-ops.js +249 -0
- package/lib/rules/space-unary-ops.js +400 -0
- package/lib/rules/spaced-comment.js +447 -0
- package/lib/rules/strict.js +314 -0
- package/lib/rules/switch-colon-spacing.js +158 -0
- package/lib/rules/symbol-description.js +70 -0
- package/lib/rules/template-curly-spacing.js +168 -0
- package/lib/rules/template-tag-spacing.js +121 -0
- package/lib/rules/unicode-bom.js +73 -0
- package/lib/rules/use-isnan.js +268 -0
- package/lib/rules/utils/ast-utils.js +2828 -0
- package/lib/rules/utils/char-source.js +247 -0
- package/lib/rules/utils/fix-tracker.js +125 -0
- package/lib/rules/utils/keywords.js +67 -0
- package/lib/rules/utils/lazy-loading-rule-map.js +118 -0
- package/lib/rules/utils/regular-expressions.js +58 -0
- package/lib/rules/utils/unicode/index.js +16 -0
- package/lib/rules/utils/unicode/is-combining-character.js +13 -0
- package/lib/rules/utils/unicode/is-emoji-modifier.js +13 -0
- package/lib/rules/utils/unicode/is-regional-indicator-symbol.js +13 -0
- package/lib/rules/utils/unicode/is-surrogate-pair.js +14 -0
- package/lib/rules/valid-typeof.js +171 -0
- package/lib/rules/vars-on-top.js +165 -0
- package/lib/rules/wrap-iife.js +238 -0
- package/lib/rules/wrap-regex.js +91 -0
- package/lib/rules/yield-star-spacing.js +158 -0
- package/lib/rules/yoda.js +362 -0
- package/lib/services/parser-service.js +64 -0
- package/lib/services/processor-service.js +100 -0
- package/lib/services/suppressions-service.js +302 -0
- package/lib/services/warning-service.js +87 -0
- package/lib/shared/ajv.js +34 -0
- package/lib/shared/assert.js +21 -0
- package/lib/shared/ast-utils.js +30 -0
- package/lib/shared/deep-merge-arrays.js +62 -0
- package/lib/shared/directives.js +16 -0
- package/lib/shared/flags.js +89 -0
- package/lib/shared/logging.js +38 -0
- package/lib/shared/naming.js +109 -0
- package/lib/shared/option-utils.js +63 -0
- package/lib/shared/relative-module-resolver.js +28 -0
- package/lib/shared/runtime-info.js +177 -0
- package/lib/shared/serialization.js +78 -0
- package/lib/shared/severity.js +49 -0
- package/lib/shared/stats.js +30 -0
- package/lib/shared/string-utils.js +58 -0
- package/lib/shared/text-table.js +68 -0
- package/lib/shared/translate-cli-options.js +223 -0
- package/lib/shared/traverser.js +202 -0
- package/lib/types/config-api.d.ts +12 -0
- package/lib/types/index.d.ts +1482 -0
- package/lib/types/rules.d.ts +5603 -0
- package/lib/types/universal.d.ts +6 -0
- package/lib/types/use-at-your-own-risk.d.ts +34 -0
- package/lib/universal.js +10 -0
- package/lib/unsupported-api.js +26 -0
- package/messages/all-files-ignored.js +16 -0
- 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 +117 -0
- package/messages/eslintrc-plugins.js +27 -0
- package/messages/extend-config-missing.js +13 -0
- package/messages/failed-to-read-json.js +11 -0
- package/messages/file-not-found.js +10 -0
- package/messages/invalid-rule-options.js +17 -0
- package/messages/invalid-rule-severity.js +13 -0
- package/messages/no-config-found.js +15 -0
- package/messages/plugin-conflict.js +22 -0
- package/messages/plugin-invalid.js +16 -0
- package/messages/plugin-missing.js +19 -0
- package/messages/print-config-with-directory-path.js +8 -0
- package/messages/shared.js +23 -0
- package/messages/whitespace-found.js +11 -0
- package/package.json +220 -0
|
@@ -0,0 +1,453 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @fileoverview A rule to verify `super()` callings in constructor.
|
|
3
|
+
* @author Toru Nagashima
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
"use strict";
|
|
7
|
+
|
|
8
|
+
//------------------------------------------------------------------------------
|
|
9
|
+
// Helpers
|
|
10
|
+
//------------------------------------------------------------------------------
|
|
11
|
+
|
|
12
|
+
/**
|
|
13
|
+
* Checks whether or not a given node is a constructor.
|
|
14
|
+
* @param {ASTNode} node A node to check. This node type is one of
|
|
15
|
+
* `Program`, `FunctionDeclaration`, `FunctionExpression`, and
|
|
16
|
+
* `ArrowFunctionExpression`.
|
|
17
|
+
* @returns {boolean} `true` if the node is a constructor.
|
|
18
|
+
*/
|
|
19
|
+
function isConstructorFunction(node) {
|
|
20
|
+
return (
|
|
21
|
+
node.type === "FunctionExpression" &&
|
|
22
|
+
node.parent.type === "MethodDefinition" &&
|
|
23
|
+
node.parent.kind === "constructor"
|
|
24
|
+
);
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
/**
|
|
28
|
+
* Checks whether a given node can be a constructor or not.
|
|
29
|
+
* @param {ASTNode} node A node to check.
|
|
30
|
+
* @returns {boolean} `true` if the node can be a constructor.
|
|
31
|
+
*/
|
|
32
|
+
function isPossibleConstructor(node) {
|
|
33
|
+
if (!node) {
|
|
34
|
+
return false;
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
switch (node.type) {
|
|
38
|
+
case "ClassExpression":
|
|
39
|
+
case "FunctionExpression":
|
|
40
|
+
case "ThisExpression":
|
|
41
|
+
case "MemberExpression":
|
|
42
|
+
case "CallExpression":
|
|
43
|
+
case "NewExpression":
|
|
44
|
+
case "ChainExpression":
|
|
45
|
+
case "YieldExpression":
|
|
46
|
+
case "TaggedTemplateExpression":
|
|
47
|
+
case "MetaProperty":
|
|
48
|
+
return true;
|
|
49
|
+
|
|
50
|
+
case "Identifier":
|
|
51
|
+
return node.name !== "undefined";
|
|
52
|
+
|
|
53
|
+
case "AssignmentExpression":
|
|
54
|
+
if (["=", "&&="].includes(node.operator)) {
|
|
55
|
+
return isPossibleConstructor(node.right);
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
if (["||=", "??="].includes(node.operator)) {
|
|
59
|
+
return (
|
|
60
|
+
isPossibleConstructor(node.left) ||
|
|
61
|
+
isPossibleConstructor(node.right)
|
|
62
|
+
);
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
/**
|
|
66
|
+
* All other assignment operators are mathematical assignment operators (arithmetic or bitwise).
|
|
67
|
+
* An assignment expression with a mathematical operator can either evaluate to a primitive value,
|
|
68
|
+
* or throw, depending on the operands. Thus, it cannot evaluate to a constructor function.
|
|
69
|
+
*/
|
|
70
|
+
return false;
|
|
71
|
+
|
|
72
|
+
case "LogicalExpression":
|
|
73
|
+
/*
|
|
74
|
+
* If the && operator short-circuits, the left side was falsy and therefore not a constructor, and if
|
|
75
|
+
* it doesn't short-circuit, it takes the value from the right side, so the right side must always be a
|
|
76
|
+
* possible constructor. A future improvement could verify that the left side could be truthy by
|
|
77
|
+
* excluding falsy literals.
|
|
78
|
+
*/
|
|
79
|
+
if (node.operator === "&&") {
|
|
80
|
+
return isPossibleConstructor(node.right);
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
return (
|
|
84
|
+
isPossibleConstructor(node.left) ||
|
|
85
|
+
isPossibleConstructor(node.right)
|
|
86
|
+
);
|
|
87
|
+
|
|
88
|
+
case "ConditionalExpression":
|
|
89
|
+
return (
|
|
90
|
+
isPossibleConstructor(node.alternate) ||
|
|
91
|
+
isPossibleConstructor(node.consequent)
|
|
92
|
+
);
|
|
93
|
+
|
|
94
|
+
case "SequenceExpression": {
|
|
95
|
+
const lastExpression = node.expressions.at(-1);
|
|
96
|
+
|
|
97
|
+
return isPossibleConstructor(lastExpression);
|
|
98
|
+
}
|
|
99
|
+
|
|
100
|
+
default:
|
|
101
|
+
return false;
|
|
102
|
+
}
|
|
103
|
+
}
|
|
104
|
+
|
|
105
|
+
/**
|
|
106
|
+
* A class to store information about a code path segment.
|
|
107
|
+
*/
|
|
108
|
+
class SegmentInfo {
|
|
109
|
+
/**
|
|
110
|
+
* Indicates if super() is called in all code paths.
|
|
111
|
+
* @type {boolean}
|
|
112
|
+
*/
|
|
113
|
+
calledInEveryPaths = false;
|
|
114
|
+
|
|
115
|
+
/**
|
|
116
|
+
* Indicates if super() is called in any code paths.
|
|
117
|
+
* @type {boolean}
|
|
118
|
+
*/
|
|
119
|
+
calledInSomePaths = false;
|
|
120
|
+
|
|
121
|
+
/**
|
|
122
|
+
* The nodes which have been validated and don't need to be reconsidered.
|
|
123
|
+
* @type {ASTNode[]}
|
|
124
|
+
*/
|
|
125
|
+
validNodes = [];
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
//------------------------------------------------------------------------------
|
|
129
|
+
// Rule Definition
|
|
130
|
+
//------------------------------------------------------------------------------
|
|
131
|
+
|
|
132
|
+
/** @type {import('../types').Rule.RuleModule} */
|
|
133
|
+
module.exports = {
|
|
134
|
+
meta: {
|
|
135
|
+
type: "problem",
|
|
136
|
+
|
|
137
|
+
docs: {
|
|
138
|
+
description: "Require `super()` calls in constructors",
|
|
139
|
+
recommended: true,
|
|
140
|
+
url: "https://eslint.org/docs/latest/rules/constructor-super",
|
|
141
|
+
},
|
|
142
|
+
|
|
143
|
+
schema: [],
|
|
144
|
+
|
|
145
|
+
messages: {
|
|
146
|
+
missingSome: "Lacked a call of 'super()' in some code paths.",
|
|
147
|
+
missingAll: "Expected to call 'super()'.",
|
|
148
|
+
|
|
149
|
+
duplicate: "Unexpected duplicate 'super()'.",
|
|
150
|
+
badSuper:
|
|
151
|
+
"Unexpected 'super()' because 'super' is not a constructor.",
|
|
152
|
+
},
|
|
153
|
+
},
|
|
154
|
+
|
|
155
|
+
create(context) {
|
|
156
|
+
/*
|
|
157
|
+
* {{hasExtends: boolean, scope: Scope, codePath: CodePath}[]}
|
|
158
|
+
* Information for each constructor.
|
|
159
|
+
* - upper: Information of the upper constructor.
|
|
160
|
+
* - hasExtends: A flag which shows whether own class has a valid `extends`
|
|
161
|
+
* part.
|
|
162
|
+
* - scope: The scope of own class.
|
|
163
|
+
* - codePath: The code path object of the constructor.
|
|
164
|
+
*/
|
|
165
|
+
let funcInfo = null;
|
|
166
|
+
|
|
167
|
+
/**
|
|
168
|
+
* @type {Record<string, SegmentInfo>}
|
|
169
|
+
*/
|
|
170
|
+
const segInfoMap = Object.create(null);
|
|
171
|
+
|
|
172
|
+
/**
|
|
173
|
+
* Gets the flag which shows `super()` is called in some paths.
|
|
174
|
+
* @param {CodePathSegment} segment A code path segment to get.
|
|
175
|
+
* @returns {boolean} The flag which shows `super()` is called in some paths
|
|
176
|
+
*/
|
|
177
|
+
function isCalledInSomePath(segment) {
|
|
178
|
+
return (
|
|
179
|
+
segment.reachable && segInfoMap[segment.id].calledInSomePaths
|
|
180
|
+
);
|
|
181
|
+
}
|
|
182
|
+
|
|
183
|
+
/**
|
|
184
|
+
* Determines if a segment has been seen in the traversal.
|
|
185
|
+
* @param {CodePathSegment} segment A code path segment to check.
|
|
186
|
+
* @returns {boolean} `true` if the segment has been seen.
|
|
187
|
+
*/
|
|
188
|
+
function hasSegmentBeenSeen(segment) {
|
|
189
|
+
return !!segInfoMap[segment.id];
|
|
190
|
+
}
|
|
191
|
+
|
|
192
|
+
/**
|
|
193
|
+
* Gets the flag which shows `super()` is called in all paths.
|
|
194
|
+
* @param {CodePathSegment} segment A code path segment to get.
|
|
195
|
+
* @returns {boolean} The flag which shows `super()` is called in all paths.
|
|
196
|
+
*/
|
|
197
|
+
function isCalledInEveryPath(segment) {
|
|
198
|
+
return (
|
|
199
|
+
segment.reachable && segInfoMap[segment.id].calledInEveryPaths
|
|
200
|
+
);
|
|
201
|
+
}
|
|
202
|
+
|
|
203
|
+
return {
|
|
204
|
+
/**
|
|
205
|
+
* Stacks a constructor information.
|
|
206
|
+
* @param {CodePath} codePath A code path which was started.
|
|
207
|
+
* @param {ASTNode} node The current node.
|
|
208
|
+
* @returns {void}
|
|
209
|
+
*/
|
|
210
|
+
onCodePathStart(codePath, node) {
|
|
211
|
+
if (isConstructorFunction(node)) {
|
|
212
|
+
// Class > ClassBody > MethodDefinition > FunctionExpression
|
|
213
|
+
const classNode = node.parent.parent.parent;
|
|
214
|
+
const superClass = classNode.superClass;
|
|
215
|
+
|
|
216
|
+
funcInfo = {
|
|
217
|
+
upper: funcInfo,
|
|
218
|
+
isConstructor: true,
|
|
219
|
+
hasExtends: Boolean(superClass),
|
|
220
|
+
superIsConstructor: isPossibleConstructor(superClass),
|
|
221
|
+
codePath,
|
|
222
|
+
currentSegments: new Set(),
|
|
223
|
+
};
|
|
224
|
+
} else {
|
|
225
|
+
funcInfo = {
|
|
226
|
+
upper: funcInfo,
|
|
227
|
+
isConstructor: false,
|
|
228
|
+
hasExtends: false,
|
|
229
|
+
superIsConstructor: false,
|
|
230
|
+
codePath,
|
|
231
|
+
currentSegments: new Set(),
|
|
232
|
+
};
|
|
233
|
+
}
|
|
234
|
+
},
|
|
235
|
+
|
|
236
|
+
/**
|
|
237
|
+
* Pops a constructor information.
|
|
238
|
+
* And reports if `super()` lacked.
|
|
239
|
+
* @param {CodePath} codePath A code path which was ended.
|
|
240
|
+
* @param {ASTNode} node The current node.
|
|
241
|
+
* @returns {void}
|
|
242
|
+
*/
|
|
243
|
+
onCodePathEnd(codePath, node) {
|
|
244
|
+
const hasExtends = funcInfo.hasExtends;
|
|
245
|
+
|
|
246
|
+
// Pop.
|
|
247
|
+
funcInfo = funcInfo.upper;
|
|
248
|
+
|
|
249
|
+
if (!hasExtends) {
|
|
250
|
+
return;
|
|
251
|
+
}
|
|
252
|
+
|
|
253
|
+
// Reports if `super()` lacked.
|
|
254
|
+
const returnedSegments = codePath.returnedSegments;
|
|
255
|
+
const calledInEveryPaths =
|
|
256
|
+
returnedSegments.every(isCalledInEveryPath);
|
|
257
|
+
const calledInSomePaths =
|
|
258
|
+
returnedSegments.some(isCalledInSomePath);
|
|
259
|
+
|
|
260
|
+
if (!calledInEveryPaths) {
|
|
261
|
+
context.report({
|
|
262
|
+
messageId: calledInSomePaths
|
|
263
|
+
? "missingSome"
|
|
264
|
+
: "missingAll",
|
|
265
|
+
node: node.parent,
|
|
266
|
+
});
|
|
267
|
+
}
|
|
268
|
+
},
|
|
269
|
+
|
|
270
|
+
/**
|
|
271
|
+
* Initialize information of a given code path segment.
|
|
272
|
+
* @param {CodePathSegment} segment A code path segment to initialize.
|
|
273
|
+
* @param {CodePathSegment} node Node that starts the segment.
|
|
274
|
+
* @returns {void}
|
|
275
|
+
*/
|
|
276
|
+
onCodePathSegmentStart(segment, node) {
|
|
277
|
+
funcInfo.currentSegments.add(segment);
|
|
278
|
+
|
|
279
|
+
if (!(funcInfo.isConstructor && funcInfo.hasExtends)) {
|
|
280
|
+
return;
|
|
281
|
+
}
|
|
282
|
+
|
|
283
|
+
// Initialize info.
|
|
284
|
+
const info = (segInfoMap[segment.id] = new SegmentInfo());
|
|
285
|
+
|
|
286
|
+
const seenPrevSegments =
|
|
287
|
+
segment.prevSegments.filter(hasSegmentBeenSeen);
|
|
288
|
+
|
|
289
|
+
// When there are previous segments, aggregates these.
|
|
290
|
+
if (seenPrevSegments.length > 0) {
|
|
291
|
+
info.calledInSomePaths =
|
|
292
|
+
seenPrevSegments.some(isCalledInSomePath);
|
|
293
|
+
info.calledInEveryPaths =
|
|
294
|
+
seenPrevSegments.every(isCalledInEveryPath);
|
|
295
|
+
}
|
|
296
|
+
|
|
297
|
+
/*
|
|
298
|
+
* ForStatement > *.update segments are a special case as they are created in advance,
|
|
299
|
+
* without seen previous segments. Since they logically don't affect `calledInEveryPaths`
|
|
300
|
+
* calculations, and they can never be a lone previous segment of another one, we'll set
|
|
301
|
+
* their `calledInEveryPaths` to `true` to effectively ignore them in those calculations.
|
|
302
|
+
* .
|
|
303
|
+
*/
|
|
304
|
+
if (
|
|
305
|
+
node.parent &&
|
|
306
|
+
node.parent.type === "ForStatement" &&
|
|
307
|
+
node.parent.update === node
|
|
308
|
+
) {
|
|
309
|
+
info.calledInEveryPaths = true;
|
|
310
|
+
}
|
|
311
|
+
},
|
|
312
|
+
|
|
313
|
+
onUnreachableCodePathSegmentStart(segment) {
|
|
314
|
+
funcInfo.currentSegments.add(segment);
|
|
315
|
+
},
|
|
316
|
+
|
|
317
|
+
onUnreachableCodePathSegmentEnd(segment) {
|
|
318
|
+
funcInfo.currentSegments.delete(segment);
|
|
319
|
+
},
|
|
320
|
+
|
|
321
|
+
onCodePathSegmentEnd(segment) {
|
|
322
|
+
funcInfo.currentSegments.delete(segment);
|
|
323
|
+
},
|
|
324
|
+
|
|
325
|
+
/**
|
|
326
|
+
* Update information of the code path segment when a code path was
|
|
327
|
+
* looped.
|
|
328
|
+
* @param {CodePathSegment} fromSegment The code path segment of the
|
|
329
|
+
* end of a loop.
|
|
330
|
+
* @param {CodePathSegment} toSegment A code path segment of the head
|
|
331
|
+
* of a loop.
|
|
332
|
+
* @returns {void}
|
|
333
|
+
*/
|
|
334
|
+
onCodePathSegmentLoop(fromSegment, toSegment) {
|
|
335
|
+
if (!(funcInfo.isConstructor && funcInfo.hasExtends)) {
|
|
336
|
+
return;
|
|
337
|
+
}
|
|
338
|
+
|
|
339
|
+
funcInfo.codePath.traverseSegments(
|
|
340
|
+
{ first: toSegment, last: fromSegment },
|
|
341
|
+
(segment, controller) => {
|
|
342
|
+
const info = segInfoMap[segment.id];
|
|
343
|
+
|
|
344
|
+
// skip segments after the loop
|
|
345
|
+
if (!info) {
|
|
346
|
+
controller.skip();
|
|
347
|
+
return;
|
|
348
|
+
}
|
|
349
|
+
|
|
350
|
+
const seenPrevSegments =
|
|
351
|
+
segment.prevSegments.filter(hasSegmentBeenSeen);
|
|
352
|
+
const calledInSomePreviousPaths =
|
|
353
|
+
seenPrevSegments.some(isCalledInSomePath);
|
|
354
|
+
const calledInEveryPreviousPaths =
|
|
355
|
+
seenPrevSegments.every(isCalledInEveryPath);
|
|
356
|
+
|
|
357
|
+
info.calledInSomePaths ||= calledInSomePreviousPaths;
|
|
358
|
+
info.calledInEveryPaths ||= calledInEveryPreviousPaths;
|
|
359
|
+
|
|
360
|
+
// If flags become true anew, reports the valid nodes.
|
|
361
|
+
if (calledInSomePreviousPaths) {
|
|
362
|
+
const nodes = info.validNodes;
|
|
363
|
+
|
|
364
|
+
info.validNodes = [];
|
|
365
|
+
|
|
366
|
+
for (let i = 0; i < nodes.length; ++i) {
|
|
367
|
+
const node = nodes[i];
|
|
368
|
+
|
|
369
|
+
context.report({
|
|
370
|
+
messageId: "duplicate",
|
|
371
|
+
node,
|
|
372
|
+
});
|
|
373
|
+
}
|
|
374
|
+
}
|
|
375
|
+
},
|
|
376
|
+
);
|
|
377
|
+
},
|
|
378
|
+
|
|
379
|
+
/**
|
|
380
|
+
* Checks for a call of `super()`.
|
|
381
|
+
* @param {ASTNode} node A CallExpression node to check.
|
|
382
|
+
* @returns {void}
|
|
383
|
+
*/
|
|
384
|
+
"CallExpression:exit"(node) {
|
|
385
|
+
if (!(funcInfo.isConstructor && funcInfo.hasExtends)) {
|
|
386
|
+
return;
|
|
387
|
+
}
|
|
388
|
+
|
|
389
|
+
// Skips except `super()`.
|
|
390
|
+
if (node.callee.type !== "Super") {
|
|
391
|
+
return;
|
|
392
|
+
}
|
|
393
|
+
|
|
394
|
+
// Reports if needed.
|
|
395
|
+
const segments = funcInfo.currentSegments;
|
|
396
|
+
let duplicate = false;
|
|
397
|
+
let info = null;
|
|
398
|
+
|
|
399
|
+
for (const segment of segments) {
|
|
400
|
+
if (segment.reachable) {
|
|
401
|
+
info = segInfoMap[segment.id];
|
|
402
|
+
|
|
403
|
+
duplicate = duplicate || info.calledInSomePaths;
|
|
404
|
+
info.calledInSomePaths = info.calledInEveryPaths = true;
|
|
405
|
+
}
|
|
406
|
+
}
|
|
407
|
+
|
|
408
|
+
if (info) {
|
|
409
|
+
if (duplicate) {
|
|
410
|
+
context.report({
|
|
411
|
+
messageId: "duplicate",
|
|
412
|
+
node,
|
|
413
|
+
});
|
|
414
|
+
} else if (!funcInfo.superIsConstructor) {
|
|
415
|
+
context.report({
|
|
416
|
+
messageId: "badSuper",
|
|
417
|
+
node,
|
|
418
|
+
});
|
|
419
|
+
} else {
|
|
420
|
+
info.validNodes.push(node);
|
|
421
|
+
}
|
|
422
|
+
}
|
|
423
|
+
},
|
|
424
|
+
|
|
425
|
+
/**
|
|
426
|
+
* Set the mark to the returned path as `super()` was called.
|
|
427
|
+
* @param {ASTNode} node A ReturnStatement node to check.
|
|
428
|
+
* @returns {void}
|
|
429
|
+
*/
|
|
430
|
+
ReturnStatement(node) {
|
|
431
|
+
if (!(funcInfo.isConstructor && funcInfo.hasExtends)) {
|
|
432
|
+
return;
|
|
433
|
+
}
|
|
434
|
+
|
|
435
|
+
// Skips if no argument.
|
|
436
|
+
if (!node.argument) {
|
|
437
|
+
return;
|
|
438
|
+
}
|
|
439
|
+
|
|
440
|
+
// Returning argument is a substitute of 'super()'.
|
|
441
|
+
const segments = funcInfo.currentSegments;
|
|
442
|
+
|
|
443
|
+
for (const segment of segments) {
|
|
444
|
+
if (segment.reachable) {
|
|
445
|
+
const info = segInfoMap[segment.id];
|
|
446
|
+
|
|
447
|
+
info.calledInSomePaths = info.calledInEveryPaths = true;
|
|
448
|
+
}
|
|
449
|
+
}
|
|
450
|
+
},
|
|
451
|
+
};
|
|
452
|
+
},
|
|
453
|
+
};
|