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,892 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @fileoverview Restrict usage of specified node imports.
|
|
3
|
+
* @author Guy Ellis
|
|
4
|
+
*/
|
|
5
|
+
"use strict";
|
|
6
|
+
|
|
7
|
+
//------------------------------------------------------------------------------
|
|
8
|
+
// Requirements
|
|
9
|
+
//------------------------------------------------------------------------------
|
|
10
|
+
|
|
11
|
+
const astUtils = require("./utils/ast-utils");
|
|
12
|
+
|
|
13
|
+
//------------------------------------------------------------------------------
|
|
14
|
+
// Helpers
|
|
15
|
+
//------------------------------------------------------------------------------
|
|
16
|
+
|
|
17
|
+
/**
|
|
18
|
+
* Format import names for error messages.
|
|
19
|
+
* @param {string[]} importNames The import names to format.
|
|
20
|
+
* @returns {string} The formatted import names.
|
|
21
|
+
*/
|
|
22
|
+
function formatImportNames(importNames) {
|
|
23
|
+
return new Intl.ListFormat("en-US").format(
|
|
24
|
+
importNames.map(name => `'${name}'`),
|
|
25
|
+
);
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
/**
|
|
29
|
+
* Returns "is" or "are" based on the number of import names.
|
|
30
|
+
* @param {string[]} importNames The import names to check.
|
|
31
|
+
* @returns {string} "is" if one import name, otherwise "are".
|
|
32
|
+
*/
|
|
33
|
+
function isOrAre(importNames) {
|
|
34
|
+
return importNames.length === 1 ? "is" : "are";
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
//------------------------------------------------------------------------------
|
|
38
|
+
// Rule Definition
|
|
39
|
+
//------------------------------------------------------------------------------
|
|
40
|
+
|
|
41
|
+
const ignore = require("ignore");
|
|
42
|
+
|
|
43
|
+
const arrayOfStringsOrObjects = {
|
|
44
|
+
type: "array",
|
|
45
|
+
items: {
|
|
46
|
+
anyOf: [
|
|
47
|
+
{ type: "string" },
|
|
48
|
+
{
|
|
49
|
+
type: "object",
|
|
50
|
+
properties: {
|
|
51
|
+
name: { type: "string" },
|
|
52
|
+
message: {
|
|
53
|
+
type: "string",
|
|
54
|
+
minLength: 1,
|
|
55
|
+
},
|
|
56
|
+
importNames: {
|
|
57
|
+
type: "array",
|
|
58
|
+
items: {
|
|
59
|
+
type: "string",
|
|
60
|
+
},
|
|
61
|
+
},
|
|
62
|
+
allowImportNames: {
|
|
63
|
+
type: "array",
|
|
64
|
+
items: {
|
|
65
|
+
type: "string",
|
|
66
|
+
},
|
|
67
|
+
},
|
|
68
|
+
allowTypeImports: {
|
|
69
|
+
type: "boolean",
|
|
70
|
+
description:
|
|
71
|
+
"Whether to allow type-only imports for a path.",
|
|
72
|
+
},
|
|
73
|
+
},
|
|
74
|
+
additionalProperties: false,
|
|
75
|
+
required: ["name"],
|
|
76
|
+
not: { required: ["importNames", "allowImportNames"] },
|
|
77
|
+
},
|
|
78
|
+
],
|
|
79
|
+
},
|
|
80
|
+
uniqueItems: true,
|
|
81
|
+
};
|
|
82
|
+
|
|
83
|
+
const arrayOfStringsOrObjectPatterns = {
|
|
84
|
+
anyOf: [
|
|
85
|
+
{
|
|
86
|
+
type: "array",
|
|
87
|
+
items: {
|
|
88
|
+
type: "string",
|
|
89
|
+
},
|
|
90
|
+
uniqueItems: true,
|
|
91
|
+
},
|
|
92
|
+
{
|
|
93
|
+
type: "array",
|
|
94
|
+
items: {
|
|
95
|
+
type: "object",
|
|
96
|
+
properties: {
|
|
97
|
+
importNames: {
|
|
98
|
+
type: "array",
|
|
99
|
+
items: {
|
|
100
|
+
type: "string",
|
|
101
|
+
},
|
|
102
|
+
minItems: 1,
|
|
103
|
+
uniqueItems: true,
|
|
104
|
+
},
|
|
105
|
+
allowImportNames: {
|
|
106
|
+
type: "array",
|
|
107
|
+
items: {
|
|
108
|
+
type: "string",
|
|
109
|
+
},
|
|
110
|
+
minItems: 1,
|
|
111
|
+
uniqueItems: true,
|
|
112
|
+
},
|
|
113
|
+
group: {
|
|
114
|
+
type: "array",
|
|
115
|
+
items: {
|
|
116
|
+
type: "string",
|
|
117
|
+
},
|
|
118
|
+
minItems: 1,
|
|
119
|
+
uniqueItems: true,
|
|
120
|
+
},
|
|
121
|
+
regex: {
|
|
122
|
+
type: "string",
|
|
123
|
+
},
|
|
124
|
+
importNamePattern: {
|
|
125
|
+
type: "string",
|
|
126
|
+
},
|
|
127
|
+
allowImportNamePattern: {
|
|
128
|
+
type: "string",
|
|
129
|
+
},
|
|
130
|
+
message: {
|
|
131
|
+
type: "string",
|
|
132
|
+
minLength: 1,
|
|
133
|
+
},
|
|
134
|
+
caseSensitive: {
|
|
135
|
+
type: "boolean",
|
|
136
|
+
},
|
|
137
|
+
allowTypeImports: {
|
|
138
|
+
type: "boolean",
|
|
139
|
+
description:
|
|
140
|
+
"Whether to allow type-only imports for a pattern.",
|
|
141
|
+
},
|
|
142
|
+
},
|
|
143
|
+
additionalProperties: false,
|
|
144
|
+
not: {
|
|
145
|
+
anyOf: [
|
|
146
|
+
{ required: ["importNames", "allowImportNames"] },
|
|
147
|
+
{
|
|
148
|
+
required: [
|
|
149
|
+
"importNamePattern",
|
|
150
|
+
"allowImportNamePattern",
|
|
151
|
+
],
|
|
152
|
+
},
|
|
153
|
+
{ required: ["importNames", "allowImportNamePattern"] },
|
|
154
|
+
{ required: ["importNamePattern", "allowImportNames"] },
|
|
155
|
+
{
|
|
156
|
+
required: [
|
|
157
|
+
"allowImportNames",
|
|
158
|
+
"allowImportNamePattern",
|
|
159
|
+
],
|
|
160
|
+
},
|
|
161
|
+
],
|
|
162
|
+
},
|
|
163
|
+
oneOf: [{ required: ["group"] }, { required: ["regex"] }],
|
|
164
|
+
},
|
|
165
|
+
uniqueItems: true,
|
|
166
|
+
},
|
|
167
|
+
],
|
|
168
|
+
};
|
|
169
|
+
|
|
170
|
+
/** @type {import('../types').Rule.RuleModule} */
|
|
171
|
+
module.exports = {
|
|
172
|
+
meta: {
|
|
173
|
+
type: "suggestion",
|
|
174
|
+
dialects: ["typescript", "javascript"],
|
|
175
|
+
language: "javascript",
|
|
176
|
+
|
|
177
|
+
docs: {
|
|
178
|
+
description: "Disallow specified modules when loaded by `import`",
|
|
179
|
+
recommended: false,
|
|
180
|
+
url: "https://eslint.org/docs/latest/rules/no-restricted-imports",
|
|
181
|
+
},
|
|
182
|
+
|
|
183
|
+
messages: {
|
|
184
|
+
path: "'{{importSource}}' import is restricted from being used.",
|
|
185
|
+
pathWithCustomMessage:
|
|
186
|
+
// eslint-disable-next-line eslint-plugin/report-message-format -- Custom message might not end in a period
|
|
187
|
+
"'{{importSource}}' import is restricted from being used. {{customMessage}}",
|
|
188
|
+
|
|
189
|
+
patterns:
|
|
190
|
+
"'{{importSource}}' import is restricted from being used by a pattern.",
|
|
191
|
+
patternWithCustomMessage:
|
|
192
|
+
// eslint-disable-next-line eslint-plugin/report-message-format -- Custom message might not end in a period
|
|
193
|
+
"'{{importSource}}' import is restricted from being used by a pattern. {{customMessage}}",
|
|
194
|
+
|
|
195
|
+
patternAndImportName:
|
|
196
|
+
"'{{importName}}' import from '{{importSource}}' is restricted from being used by a pattern.",
|
|
197
|
+
patternAndImportNameWithCustomMessage:
|
|
198
|
+
// eslint-disable-next-line eslint-plugin/report-message-format -- Custom message might not end in a period
|
|
199
|
+
"'{{importName}}' import from '{{importSource}}' is restricted from being used by a pattern. {{customMessage}}",
|
|
200
|
+
|
|
201
|
+
patternAndEverything:
|
|
202
|
+
"* import is invalid because {{importNames}} from '{{importSource}}' {{isOrAre}} restricted from being used by a pattern.",
|
|
203
|
+
|
|
204
|
+
patternAndEverythingWithRegexImportName:
|
|
205
|
+
"* import is invalid because import name matching '{{importNames}}' pattern from '{{importSource}}' is restricted from being used.",
|
|
206
|
+
patternAndEverythingWithCustomMessage:
|
|
207
|
+
// eslint-disable-next-line eslint-plugin/report-message-format -- Custom message might not end in a period
|
|
208
|
+
"* import is invalid because {{importNames}} from '{{importSource}}' {{isOrAre}} restricted from being used by a pattern. {{customMessage}}",
|
|
209
|
+
patternAndEverythingWithRegexImportNameAndCustomMessage:
|
|
210
|
+
// eslint-disable-next-line eslint-plugin/report-message-format -- Custom message might not end in a period
|
|
211
|
+
"* import is invalid because import name matching '{{importNames}}' pattern from '{{importSource}}' is restricted from being used. {{customMessage}}",
|
|
212
|
+
|
|
213
|
+
everything:
|
|
214
|
+
"* import is invalid because {{importNames}} from '{{importSource}}' {{isOrAre}} restricted.",
|
|
215
|
+
everythingWithCustomMessage:
|
|
216
|
+
// eslint-disable-next-line eslint-plugin/report-message-format -- Custom message might not end in a period
|
|
217
|
+
"* import is invalid because {{importNames}} from '{{importSource}}' {{isOrAre}} restricted. {{customMessage}}",
|
|
218
|
+
|
|
219
|
+
importName:
|
|
220
|
+
"'{{importName}}' import from '{{importSource}}' is restricted.",
|
|
221
|
+
importNameWithCustomMessage:
|
|
222
|
+
// eslint-disable-next-line eslint-plugin/report-message-format -- Custom message might not end in a period
|
|
223
|
+
"'{{importName}}' import from '{{importSource}}' is restricted. {{customMessage}}",
|
|
224
|
+
|
|
225
|
+
allowedImportName:
|
|
226
|
+
"'{{importName}}' import from '{{importSource}}' is restricted because only {{allowedImportNames}} {{isOrAre}} allowed.",
|
|
227
|
+
allowedImportNameWithCustomMessage:
|
|
228
|
+
// eslint-disable-next-line eslint-plugin/report-message-format -- Custom message might not end in a period
|
|
229
|
+
"'{{importName}}' import from '{{importSource}}' is restricted because only {{allowedImportNames}} {{isOrAre}} allowed. {{customMessage}}",
|
|
230
|
+
|
|
231
|
+
everythingWithAllowImportNames:
|
|
232
|
+
"* import is invalid because only {{allowedImportNames}} from '{{importSource}}' {{isOrAre}} allowed.",
|
|
233
|
+
everythingWithAllowImportNamesAndCustomMessage:
|
|
234
|
+
// eslint-disable-next-line eslint-plugin/report-message-format -- Custom message might not end in a period
|
|
235
|
+
"* import is invalid because only {{allowedImportNames}} from '{{importSource}}' {{isOrAre}} allowed. {{customMessage}}",
|
|
236
|
+
|
|
237
|
+
allowedImportNamePattern:
|
|
238
|
+
"'{{importName}}' import from '{{importSource}}' is restricted because only imports that match the pattern '{{allowedImportNamePattern}}' are allowed from '{{importSource}}'.",
|
|
239
|
+
allowedImportNamePatternWithCustomMessage:
|
|
240
|
+
// eslint-disable-next-line eslint-plugin/report-message-format -- Custom message might not end in a period
|
|
241
|
+
"'{{importName}}' import from '{{importSource}}' is restricted because only imports that match the pattern '{{allowedImportNamePattern}}' are allowed from '{{importSource}}'. {{customMessage}}",
|
|
242
|
+
|
|
243
|
+
everythingWithAllowedImportNamePattern:
|
|
244
|
+
"* import is invalid because only imports that match the pattern '{{allowedImportNamePattern}}' from '{{importSource}}' are allowed.",
|
|
245
|
+
everythingWithAllowedImportNamePatternWithCustomMessage:
|
|
246
|
+
// eslint-disable-next-line eslint-plugin/report-message-format -- Custom message might not end in a period
|
|
247
|
+
"* import is invalid because only imports that match the pattern '{{allowedImportNamePattern}}' from '{{importSource}}' are allowed. {{customMessage}}",
|
|
248
|
+
},
|
|
249
|
+
|
|
250
|
+
schema: {
|
|
251
|
+
anyOf: [
|
|
252
|
+
arrayOfStringsOrObjects,
|
|
253
|
+
{
|
|
254
|
+
type: "array",
|
|
255
|
+
items: [
|
|
256
|
+
{
|
|
257
|
+
type: "object",
|
|
258
|
+
properties: {
|
|
259
|
+
paths: arrayOfStringsOrObjects,
|
|
260
|
+
patterns: arrayOfStringsOrObjectPatterns,
|
|
261
|
+
},
|
|
262
|
+
additionalProperties: false,
|
|
263
|
+
},
|
|
264
|
+
],
|
|
265
|
+
additionalItems: false,
|
|
266
|
+
},
|
|
267
|
+
],
|
|
268
|
+
},
|
|
269
|
+
},
|
|
270
|
+
|
|
271
|
+
create(context) {
|
|
272
|
+
const sourceCode = context.sourceCode;
|
|
273
|
+
const options = Array.isArray(context.options) ? context.options : [];
|
|
274
|
+
const isPathAndPatternsObject =
|
|
275
|
+
typeof options[0] === "object" &&
|
|
276
|
+
(Object.hasOwn(options[0], "paths") ||
|
|
277
|
+
Object.hasOwn(options[0], "patterns"));
|
|
278
|
+
|
|
279
|
+
const restrictedPaths =
|
|
280
|
+
(isPathAndPatternsObject ? options[0].paths : context.options) ||
|
|
281
|
+
[];
|
|
282
|
+
const groupedRestrictedPaths = restrictedPaths.reduce(
|
|
283
|
+
(memo, importSource) => {
|
|
284
|
+
const path =
|
|
285
|
+
typeof importSource === "string"
|
|
286
|
+
? importSource
|
|
287
|
+
: importSource.name;
|
|
288
|
+
|
|
289
|
+
if (!memo[path]) {
|
|
290
|
+
memo[path] = [];
|
|
291
|
+
}
|
|
292
|
+
|
|
293
|
+
if (typeof importSource === "string") {
|
|
294
|
+
memo[path].push({});
|
|
295
|
+
} else {
|
|
296
|
+
memo[path].push({
|
|
297
|
+
message: importSource.message,
|
|
298
|
+
importNames: importSource.importNames,
|
|
299
|
+
allowImportNames: importSource.allowImportNames,
|
|
300
|
+
allowTypeImports: importSource.allowTypeImports,
|
|
301
|
+
});
|
|
302
|
+
}
|
|
303
|
+
return memo;
|
|
304
|
+
},
|
|
305
|
+
Object.create(null),
|
|
306
|
+
);
|
|
307
|
+
|
|
308
|
+
// Handle patterns too, either as strings or groups
|
|
309
|
+
let restrictedPatterns =
|
|
310
|
+
(isPathAndPatternsObject ? options[0].patterns : []) || [];
|
|
311
|
+
|
|
312
|
+
// standardize to array of objects if we have an array of strings
|
|
313
|
+
if (
|
|
314
|
+
restrictedPatterns.length > 0 &&
|
|
315
|
+
typeof restrictedPatterns[0] === "string"
|
|
316
|
+
) {
|
|
317
|
+
restrictedPatterns = [{ group: restrictedPatterns }];
|
|
318
|
+
}
|
|
319
|
+
|
|
320
|
+
// relative paths are supported for this rule
|
|
321
|
+
const restrictedPatternGroups = restrictedPatterns.map(
|
|
322
|
+
({
|
|
323
|
+
group,
|
|
324
|
+
regex,
|
|
325
|
+
message,
|
|
326
|
+
caseSensitive,
|
|
327
|
+
importNames,
|
|
328
|
+
importNamePattern,
|
|
329
|
+
allowImportNames,
|
|
330
|
+
allowImportNamePattern,
|
|
331
|
+
allowTypeImports,
|
|
332
|
+
}) => ({
|
|
333
|
+
...(group
|
|
334
|
+
? {
|
|
335
|
+
matcher: ignore({
|
|
336
|
+
allowRelativePaths: true,
|
|
337
|
+
ignorecase: !caseSensitive,
|
|
338
|
+
}).add(group),
|
|
339
|
+
}
|
|
340
|
+
: {}),
|
|
341
|
+
...(typeof regex === "string"
|
|
342
|
+
? {
|
|
343
|
+
regexMatcher: new RegExp(
|
|
344
|
+
regex,
|
|
345
|
+
caseSensitive ? "u" : "iu",
|
|
346
|
+
),
|
|
347
|
+
}
|
|
348
|
+
: {}),
|
|
349
|
+
customMessage: message,
|
|
350
|
+
importNames,
|
|
351
|
+
importNamePattern,
|
|
352
|
+
allowImportNames,
|
|
353
|
+
allowImportNamePattern,
|
|
354
|
+
allowTypeImports,
|
|
355
|
+
}),
|
|
356
|
+
);
|
|
357
|
+
|
|
358
|
+
// if no imports are restricted we don't need to check
|
|
359
|
+
if (
|
|
360
|
+
Object.keys(restrictedPaths).length === 0 &&
|
|
361
|
+
restrictedPatternGroups.length === 0
|
|
362
|
+
) {
|
|
363
|
+
return {};
|
|
364
|
+
}
|
|
365
|
+
|
|
366
|
+
/**
|
|
367
|
+
* Check if the node is a type-only import
|
|
368
|
+
* @param {ASTNode} node The node to check
|
|
369
|
+
* @returns {boolean} Whether the node is a type-only import
|
|
370
|
+
*/
|
|
371
|
+
function isTypeOnlyImport(node) {
|
|
372
|
+
return (
|
|
373
|
+
node.importKind === "type" ||
|
|
374
|
+
(node.specifiers?.length > 0 &&
|
|
375
|
+
node.specifiers.every(
|
|
376
|
+
specifier => specifier.importKind === "type",
|
|
377
|
+
))
|
|
378
|
+
);
|
|
379
|
+
}
|
|
380
|
+
|
|
381
|
+
/**
|
|
382
|
+
* Check if a specifier is type-only
|
|
383
|
+
* @param {ASTNode} specifier The specifier to check
|
|
384
|
+
* @returns {boolean} Whether the specifier is type-only
|
|
385
|
+
*/
|
|
386
|
+
function isTypeOnlySpecifier(specifier) {
|
|
387
|
+
return (
|
|
388
|
+
specifier.importKind === "type" ||
|
|
389
|
+
specifier.exportKind === "type"
|
|
390
|
+
);
|
|
391
|
+
}
|
|
392
|
+
|
|
393
|
+
/**
|
|
394
|
+
* Check if the node is a type-only export
|
|
395
|
+
* @param {ASTNode} node The node to check
|
|
396
|
+
* @returns {boolean} Whether the node is a type-only export
|
|
397
|
+
*/
|
|
398
|
+
function isTypeOnlyExport(node) {
|
|
399
|
+
return (
|
|
400
|
+
node.exportKind === "type" ||
|
|
401
|
+
(node.specifiers?.length > 0 &&
|
|
402
|
+
node.specifiers.every(
|
|
403
|
+
specifier => specifier.exportKind === "type",
|
|
404
|
+
))
|
|
405
|
+
);
|
|
406
|
+
}
|
|
407
|
+
|
|
408
|
+
/**
|
|
409
|
+
* Report a restricted path.
|
|
410
|
+
* @param {string} importSource path of the import
|
|
411
|
+
* @param {Map<string,Object[]>} importNames Map of import names that are being imported
|
|
412
|
+
* @param {node} node representing the restricted path reference
|
|
413
|
+
* @returns {void}
|
|
414
|
+
* @private
|
|
415
|
+
*/
|
|
416
|
+
function checkRestrictedPathAndReport(importSource, importNames, node) {
|
|
417
|
+
if (!Object.hasOwn(groupedRestrictedPaths, importSource)) {
|
|
418
|
+
return;
|
|
419
|
+
}
|
|
420
|
+
|
|
421
|
+
groupedRestrictedPaths[importSource].forEach(
|
|
422
|
+
restrictedPathEntry => {
|
|
423
|
+
const customMessage = restrictedPathEntry.message;
|
|
424
|
+
const restrictedImportNames =
|
|
425
|
+
restrictedPathEntry.importNames;
|
|
426
|
+
const allowedImportNames =
|
|
427
|
+
restrictedPathEntry.allowImportNames;
|
|
428
|
+
const allowTypeImports =
|
|
429
|
+
restrictedPathEntry.allowTypeImports;
|
|
430
|
+
|
|
431
|
+
// Skip if this is a type-only import and it's allowed for this specific entry
|
|
432
|
+
if (
|
|
433
|
+
allowTypeImports &&
|
|
434
|
+
(node.type === "ImportDeclaration" ||
|
|
435
|
+
node.type === "TSImportEqualsDeclaration") &&
|
|
436
|
+
isTypeOnlyImport(node)
|
|
437
|
+
) {
|
|
438
|
+
return;
|
|
439
|
+
}
|
|
440
|
+
|
|
441
|
+
// Skip if this is a type-only export and it's allowed for this specific entry
|
|
442
|
+
if (
|
|
443
|
+
allowTypeImports &&
|
|
444
|
+
(node.type === "ExportNamedDeclaration" ||
|
|
445
|
+
node.type === "ExportAllDeclaration") &&
|
|
446
|
+
isTypeOnlyExport(node)
|
|
447
|
+
) {
|
|
448
|
+
return;
|
|
449
|
+
}
|
|
450
|
+
|
|
451
|
+
if (!restrictedImportNames && !allowedImportNames) {
|
|
452
|
+
context.report({
|
|
453
|
+
node,
|
|
454
|
+
messageId: customMessage
|
|
455
|
+
? "pathWithCustomMessage"
|
|
456
|
+
: "path",
|
|
457
|
+
data: {
|
|
458
|
+
importSource,
|
|
459
|
+
customMessage,
|
|
460
|
+
},
|
|
461
|
+
});
|
|
462
|
+
|
|
463
|
+
return;
|
|
464
|
+
}
|
|
465
|
+
|
|
466
|
+
importNames.forEach((specifiers, importName) => {
|
|
467
|
+
if (importName === "*") {
|
|
468
|
+
const [specifier] = specifiers;
|
|
469
|
+
|
|
470
|
+
if (restrictedImportNames) {
|
|
471
|
+
context.report({
|
|
472
|
+
node,
|
|
473
|
+
messageId: customMessage
|
|
474
|
+
? "everythingWithCustomMessage"
|
|
475
|
+
: "everything",
|
|
476
|
+
loc: specifier.loc,
|
|
477
|
+
data: {
|
|
478
|
+
importSource,
|
|
479
|
+
importNames: formatImportNames(
|
|
480
|
+
restrictedImportNames,
|
|
481
|
+
),
|
|
482
|
+
isOrAre: isOrAre(restrictedImportNames),
|
|
483
|
+
customMessage,
|
|
484
|
+
},
|
|
485
|
+
});
|
|
486
|
+
} else if (allowedImportNames) {
|
|
487
|
+
context.report({
|
|
488
|
+
node,
|
|
489
|
+
messageId: customMessage
|
|
490
|
+
? "everythingWithAllowImportNamesAndCustomMessage"
|
|
491
|
+
: "everythingWithAllowImportNames",
|
|
492
|
+
loc: specifier.loc,
|
|
493
|
+
data: {
|
|
494
|
+
importSource,
|
|
495
|
+
allowedImportNames:
|
|
496
|
+
formatImportNames(
|
|
497
|
+
allowedImportNames,
|
|
498
|
+
),
|
|
499
|
+
isOrAre: isOrAre(allowedImportNames),
|
|
500
|
+
customMessage,
|
|
501
|
+
},
|
|
502
|
+
});
|
|
503
|
+
}
|
|
504
|
+
|
|
505
|
+
return;
|
|
506
|
+
}
|
|
507
|
+
|
|
508
|
+
if (
|
|
509
|
+
restrictedImportNames &&
|
|
510
|
+
restrictedImportNames.includes(importName)
|
|
511
|
+
) {
|
|
512
|
+
specifiers.forEach(specifier => {
|
|
513
|
+
// Skip if this is a type-only import specifier and type imports are allowed
|
|
514
|
+
if (
|
|
515
|
+
allowTypeImports &&
|
|
516
|
+
isTypeOnlySpecifier(specifier.specifier)
|
|
517
|
+
) {
|
|
518
|
+
return;
|
|
519
|
+
}
|
|
520
|
+
|
|
521
|
+
context.report({
|
|
522
|
+
node,
|
|
523
|
+
messageId: customMessage
|
|
524
|
+
? "importNameWithCustomMessage"
|
|
525
|
+
: "importName",
|
|
526
|
+
loc: specifier.loc,
|
|
527
|
+
data: {
|
|
528
|
+
importSource,
|
|
529
|
+
customMessage,
|
|
530
|
+
importName,
|
|
531
|
+
},
|
|
532
|
+
});
|
|
533
|
+
});
|
|
534
|
+
}
|
|
535
|
+
|
|
536
|
+
if (
|
|
537
|
+
allowedImportNames &&
|
|
538
|
+
!allowedImportNames.includes(importName)
|
|
539
|
+
) {
|
|
540
|
+
specifiers.forEach(specifier => {
|
|
541
|
+
// Skip if this is a type-only import specifier and type imports are allowed
|
|
542
|
+
if (
|
|
543
|
+
allowTypeImports &&
|
|
544
|
+
isTypeOnlySpecifier(specifier.specifier)
|
|
545
|
+
) {
|
|
546
|
+
return;
|
|
547
|
+
}
|
|
548
|
+
|
|
549
|
+
context.report({
|
|
550
|
+
node,
|
|
551
|
+
loc: specifier.loc,
|
|
552
|
+
messageId: customMessage
|
|
553
|
+
? "allowedImportNameWithCustomMessage"
|
|
554
|
+
: "allowedImportName",
|
|
555
|
+
data: {
|
|
556
|
+
importSource,
|
|
557
|
+
customMessage,
|
|
558
|
+
importName,
|
|
559
|
+
allowedImportNames:
|
|
560
|
+
formatImportNames(
|
|
561
|
+
allowedImportNames,
|
|
562
|
+
),
|
|
563
|
+
isOrAre: isOrAre(allowedImportNames),
|
|
564
|
+
},
|
|
565
|
+
});
|
|
566
|
+
});
|
|
567
|
+
}
|
|
568
|
+
});
|
|
569
|
+
},
|
|
570
|
+
);
|
|
571
|
+
}
|
|
572
|
+
|
|
573
|
+
/**
|
|
574
|
+
* Report a restricted path specifically for patterns.
|
|
575
|
+
* @param {node} node representing the restricted path reference
|
|
576
|
+
* @param {Object} group contains an Ignore instance for paths, the customMessage to show on failure,
|
|
577
|
+
* and any restricted import names that have been specified in the config
|
|
578
|
+
* @param {Map<string,Object[]>} importNames Map of import names that are being imported
|
|
579
|
+
* @param {string} importSource the import source string
|
|
580
|
+
* @returns {void}
|
|
581
|
+
* @private
|
|
582
|
+
*/
|
|
583
|
+
function reportPathForPatterns(node, group, importNames, importSource) {
|
|
584
|
+
// Skip if this is a type-only import and it's allowed
|
|
585
|
+
if (
|
|
586
|
+
group.allowTypeImports &&
|
|
587
|
+
(node.type === "ImportDeclaration" ||
|
|
588
|
+
node.type === "TSImportEqualsDeclaration") &&
|
|
589
|
+
isTypeOnlyImport(node)
|
|
590
|
+
) {
|
|
591
|
+
return;
|
|
592
|
+
}
|
|
593
|
+
|
|
594
|
+
// Skip if this is a type-only export and it's allowed
|
|
595
|
+
if (
|
|
596
|
+
group.allowTypeImports &&
|
|
597
|
+
(node.type === "ExportNamedDeclaration" ||
|
|
598
|
+
node.type === "ExportAllDeclaration") &&
|
|
599
|
+
isTypeOnlyExport(node)
|
|
600
|
+
) {
|
|
601
|
+
return;
|
|
602
|
+
}
|
|
603
|
+
|
|
604
|
+
const customMessage = group.customMessage;
|
|
605
|
+
const restrictedImportNames = group.importNames;
|
|
606
|
+
const restrictedImportNamePattern = group.importNamePattern
|
|
607
|
+
? new RegExp(group.importNamePattern, "u")
|
|
608
|
+
: null;
|
|
609
|
+
const allowedImportNames = group.allowImportNames;
|
|
610
|
+
const allowedImportNamePattern = group.allowImportNamePattern
|
|
611
|
+
? new RegExp(group.allowImportNamePattern, "u")
|
|
612
|
+
: null;
|
|
613
|
+
|
|
614
|
+
/**
|
|
615
|
+
* If we are not restricting to any specific import names and just the pattern itself,
|
|
616
|
+
* report the error and move on
|
|
617
|
+
*/
|
|
618
|
+
if (
|
|
619
|
+
!restrictedImportNames &&
|
|
620
|
+
!allowedImportNames &&
|
|
621
|
+
!restrictedImportNamePattern &&
|
|
622
|
+
!allowedImportNamePattern
|
|
623
|
+
) {
|
|
624
|
+
context.report({
|
|
625
|
+
node,
|
|
626
|
+
messageId: customMessage
|
|
627
|
+
? "patternWithCustomMessage"
|
|
628
|
+
: "patterns",
|
|
629
|
+
data: {
|
|
630
|
+
importSource,
|
|
631
|
+
customMessage,
|
|
632
|
+
},
|
|
633
|
+
});
|
|
634
|
+
return;
|
|
635
|
+
}
|
|
636
|
+
|
|
637
|
+
importNames.forEach((specifiers, importName) => {
|
|
638
|
+
if (importName === "*") {
|
|
639
|
+
const [specifier] = specifiers;
|
|
640
|
+
|
|
641
|
+
if (restrictedImportNames) {
|
|
642
|
+
context.report({
|
|
643
|
+
node,
|
|
644
|
+
messageId: customMessage
|
|
645
|
+
? "patternAndEverythingWithCustomMessage"
|
|
646
|
+
: "patternAndEverything",
|
|
647
|
+
loc: specifier.loc,
|
|
648
|
+
data: {
|
|
649
|
+
importSource,
|
|
650
|
+
importNames: formatImportNames(
|
|
651
|
+
restrictedImportNames,
|
|
652
|
+
),
|
|
653
|
+
isOrAre: isOrAre(restrictedImportNames),
|
|
654
|
+
customMessage,
|
|
655
|
+
},
|
|
656
|
+
});
|
|
657
|
+
} else if (allowedImportNames) {
|
|
658
|
+
context.report({
|
|
659
|
+
node,
|
|
660
|
+
messageId: customMessage
|
|
661
|
+
? "everythingWithAllowImportNamesAndCustomMessage"
|
|
662
|
+
: "everythingWithAllowImportNames",
|
|
663
|
+
loc: specifier.loc,
|
|
664
|
+
data: {
|
|
665
|
+
importSource,
|
|
666
|
+
allowedImportNames:
|
|
667
|
+
formatImportNames(allowedImportNames),
|
|
668
|
+
isOrAre: isOrAre(allowedImportNames),
|
|
669
|
+
customMessage,
|
|
670
|
+
},
|
|
671
|
+
});
|
|
672
|
+
} else if (allowedImportNamePattern) {
|
|
673
|
+
context.report({
|
|
674
|
+
node,
|
|
675
|
+
messageId: customMessage
|
|
676
|
+
? "everythingWithAllowedImportNamePatternWithCustomMessage"
|
|
677
|
+
: "everythingWithAllowedImportNamePattern",
|
|
678
|
+
loc: specifier.loc,
|
|
679
|
+
data: {
|
|
680
|
+
importSource,
|
|
681
|
+
allowedImportNamePattern,
|
|
682
|
+
customMessage,
|
|
683
|
+
},
|
|
684
|
+
});
|
|
685
|
+
} else {
|
|
686
|
+
context.report({
|
|
687
|
+
node,
|
|
688
|
+
messageId: customMessage
|
|
689
|
+
? "patternAndEverythingWithRegexImportNameAndCustomMessage"
|
|
690
|
+
: "patternAndEverythingWithRegexImportName",
|
|
691
|
+
loc: specifier.loc,
|
|
692
|
+
data: {
|
|
693
|
+
importSource,
|
|
694
|
+
importNames: restrictedImportNamePattern,
|
|
695
|
+
customMessage,
|
|
696
|
+
},
|
|
697
|
+
});
|
|
698
|
+
}
|
|
699
|
+
|
|
700
|
+
return;
|
|
701
|
+
}
|
|
702
|
+
|
|
703
|
+
if (
|
|
704
|
+
(restrictedImportNames &&
|
|
705
|
+
restrictedImportNames.includes(importName)) ||
|
|
706
|
+
(restrictedImportNamePattern &&
|
|
707
|
+
restrictedImportNamePattern.test(importName))
|
|
708
|
+
) {
|
|
709
|
+
specifiers.forEach(specifier => {
|
|
710
|
+
// Skip if this is a type-only import specifier and type imports are allowed
|
|
711
|
+
if (
|
|
712
|
+
group.allowTypeImports &&
|
|
713
|
+
isTypeOnlySpecifier(specifier.specifier)
|
|
714
|
+
) {
|
|
715
|
+
return;
|
|
716
|
+
}
|
|
717
|
+
|
|
718
|
+
context.report({
|
|
719
|
+
node,
|
|
720
|
+
messageId: customMessage
|
|
721
|
+
? "patternAndImportNameWithCustomMessage"
|
|
722
|
+
: "patternAndImportName",
|
|
723
|
+
loc: specifier.loc,
|
|
724
|
+
data: {
|
|
725
|
+
importSource,
|
|
726
|
+
customMessage,
|
|
727
|
+
importName,
|
|
728
|
+
},
|
|
729
|
+
});
|
|
730
|
+
});
|
|
731
|
+
}
|
|
732
|
+
|
|
733
|
+
if (
|
|
734
|
+
allowedImportNames &&
|
|
735
|
+
!allowedImportNames.includes(importName)
|
|
736
|
+
) {
|
|
737
|
+
specifiers.forEach(specifier => {
|
|
738
|
+
// Skip if this is a type-only import specifier and type imports are allowed
|
|
739
|
+
if (
|
|
740
|
+
group.allowTypeImports &&
|
|
741
|
+
isTypeOnlySpecifier(specifier.specifier)
|
|
742
|
+
) {
|
|
743
|
+
return;
|
|
744
|
+
}
|
|
745
|
+
|
|
746
|
+
context.report({
|
|
747
|
+
node,
|
|
748
|
+
messageId: customMessage
|
|
749
|
+
? "allowedImportNameWithCustomMessage"
|
|
750
|
+
: "allowedImportName",
|
|
751
|
+
loc: specifier.loc,
|
|
752
|
+
data: {
|
|
753
|
+
importSource,
|
|
754
|
+
customMessage,
|
|
755
|
+
importName,
|
|
756
|
+
allowedImportNames:
|
|
757
|
+
formatImportNames(allowedImportNames),
|
|
758
|
+
isOrAre: isOrAre(allowedImportNames),
|
|
759
|
+
},
|
|
760
|
+
});
|
|
761
|
+
});
|
|
762
|
+
} else if (
|
|
763
|
+
allowedImportNamePattern &&
|
|
764
|
+
!allowedImportNamePattern.test(importName)
|
|
765
|
+
) {
|
|
766
|
+
specifiers.forEach(specifier => {
|
|
767
|
+
// Skip if this is a type-only import specifier and type imports are allowed
|
|
768
|
+
if (
|
|
769
|
+
group.allowTypeImports &&
|
|
770
|
+
isTypeOnlySpecifier(specifier.specifier)
|
|
771
|
+
) {
|
|
772
|
+
return;
|
|
773
|
+
}
|
|
774
|
+
|
|
775
|
+
context.report({
|
|
776
|
+
node,
|
|
777
|
+
messageId: customMessage
|
|
778
|
+
? "allowedImportNamePatternWithCustomMessage"
|
|
779
|
+
: "allowedImportNamePattern",
|
|
780
|
+
loc: specifier.loc,
|
|
781
|
+
data: {
|
|
782
|
+
importSource,
|
|
783
|
+
customMessage,
|
|
784
|
+
importName,
|
|
785
|
+
allowedImportNamePattern,
|
|
786
|
+
},
|
|
787
|
+
});
|
|
788
|
+
});
|
|
789
|
+
}
|
|
790
|
+
});
|
|
791
|
+
}
|
|
792
|
+
|
|
793
|
+
/**
|
|
794
|
+
* Check if the given importSource is restricted by a pattern.
|
|
795
|
+
* @param {string} importSource path of the import
|
|
796
|
+
* @param {Object} group contains a Ignore instance for paths, and the customMessage to show if it fails
|
|
797
|
+
* @returns {boolean} whether the variable is a restricted pattern or not
|
|
798
|
+
* @private
|
|
799
|
+
*/
|
|
800
|
+
function isRestrictedPattern(importSource, group) {
|
|
801
|
+
return group.regexMatcher
|
|
802
|
+
? group.regexMatcher.test(importSource)
|
|
803
|
+
: group.matcher.ignores(importSource);
|
|
804
|
+
}
|
|
805
|
+
|
|
806
|
+
/**
|
|
807
|
+
* Checks a node to see if any problems should be reported.
|
|
808
|
+
* @param {ASTNode} node The node to check.
|
|
809
|
+
* @returns {void}
|
|
810
|
+
* @private
|
|
811
|
+
*/
|
|
812
|
+
function checkNode(node) {
|
|
813
|
+
const importSource = node.source.value.trim();
|
|
814
|
+
const importNames = new Map();
|
|
815
|
+
|
|
816
|
+
if (node.type === "ExportAllDeclaration") {
|
|
817
|
+
const starToken = sourceCode.getFirstToken(node, 1);
|
|
818
|
+
|
|
819
|
+
importNames.set("*", [{ loc: starToken.loc }]);
|
|
820
|
+
} else if (node.specifiers) {
|
|
821
|
+
for (const specifier of node.specifiers) {
|
|
822
|
+
let name;
|
|
823
|
+
const specifierData = { loc: specifier.loc, specifier };
|
|
824
|
+
|
|
825
|
+
if (specifier.type === "ImportDefaultSpecifier") {
|
|
826
|
+
name = "default";
|
|
827
|
+
} else if (specifier.type === "ImportNamespaceSpecifier") {
|
|
828
|
+
name = "*";
|
|
829
|
+
} else if (specifier.imported) {
|
|
830
|
+
name = astUtils.getModuleExportName(specifier.imported);
|
|
831
|
+
} else if (specifier.local) {
|
|
832
|
+
name = astUtils.getModuleExportName(specifier.local);
|
|
833
|
+
}
|
|
834
|
+
|
|
835
|
+
if (typeof name === "string") {
|
|
836
|
+
if (importNames.has(name)) {
|
|
837
|
+
importNames.get(name).push(specifierData);
|
|
838
|
+
} else {
|
|
839
|
+
importNames.set(name, [specifierData]);
|
|
840
|
+
}
|
|
841
|
+
}
|
|
842
|
+
}
|
|
843
|
+
}
|
|
844
|
+
|
|
845
|
+
checkRestrictedPathAndReport(importSource, importNames, node);
|
|
846
|
+
restrictedPatternGroups.forEach(group => {
|
|
847
|
+
if (isRestrictedPattern(importSource, group)) {
|
|
848
|
+
reportPathForPatterns(
|
|
849
|
+
node,
|
|
850
|
+
group,
|
|
851
|
+
importNames,
|
|
852
|
+
importSource,
|
|
853
|
+
);
|
|
854
|
+
}
|
|
855
|
+
});
|
|
856
|
+
}
|
|
857
|
+
|
|
858
|
+
return {
|
|
859
|
+
ImportDeclaration: checkNode,
|
|
860
|
+
ExportNamedDeclaration(node) {
|
|
861
|
+
if (node.source) {
|
|
862
|
+
checkNode(node);
|
|
863
|
+
}
|
|
864
|
+
},
|
|
865
|
+
ExportAllDeclaration: checkNode,
|
|
866
|
+
// Add support for TypeScript import equals declarations
|
|
867
|
+
TSImportEqualsDeclaration(node) {
|
|
868
|
+
if (node.moduleReference.type === "TSExternalModuleReference") {
|
|
869
|
+
const importSource = node.moduleReference.expression.value;
|
|
870
|
+
const importNames = new Map();
|
|
871
|
+
|
|
872
|
+
// Use existing logic with the actual node
|
|
873
|
+
checkRestrictedPathAndReport(
|
|
874
|
+
importSource,
|
|
875
|
+
importNames,
|
|
876
|
+
node,
|
|
877
|
+
);
|
|
878
|
+
restrictedPatternGroups.forEach(group => {
|
|
879
|
+
if (isRestrictedPattern(importSource, group)) {
|
|
880
|
+
reportPathForPatterns(
|
|
881
|
+
node,
|
|
882
|
+
group,
|
|
883
|
+
importNames,
|
|
884
|
+
importSource,
|
|
885
|
+
);
|
|
886
|
+
}
|
|
887
|
+
});
|
|
888
|
+
}
|
|
889
|
+
},
|
|
890
|
+
};
|
|
891
|
+
},
|
|
892
|
+
};
|