123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976 |
- /* eslint-env node */
- const detectDeprecations = !!process.env.SENTRY_DETECT_DEPRECATIONS;
- const baseRules = {
- /**
- * Strict mode
- */
- // https://eslint.org/docs/rules/strict
- strict: ['error', 'global'],
- /**
- * Variables
- */
- // https://eslint.org/docs/rules/no-shadow-restricted-names
- 'no-shadow-restricted-names': ['error'],
- /**
- * Possible errors
- */
- // https://eslint.org/docs/rules/no-cond-assign
- 'no-cond-assign': ['error', 'always'],
- // https://eslint.org/docs/rules/no-alert
- 'no-alert': ['error'],
- // https://eslint.org/docs/rules/no-constant-condition
- 'no-constant-condition': ['warn'],
- // https://eslint.org/docs/rules/no-empty
- 'no-empty': ['error'],
- // https://eslint.org/docs/rules/no-ex-assign
- 'no-ex-assign': ['error'],
- // https://eslint.org/docs/rules/no-extra-boolean-cast
- 'no-extra-boolean-cast': ['error'],
- // https://eslint.org/docs/rules/no-func-assign
- 'no-func-assign': ['error'],
- // https://eslint.org/docs/rules/no-inner-declarations
- 'no-inner-declarations': ['error'],
- // https://eslint.org/docs/rules/no-invalid-regexp
- 'no-invalid-regexp': ['error'],
- // https://eslint.org/docs/rules/no-irregular-whitespace
- 'no-irregular-whitespace': ['error'],
- // https://eslint.org/docs/rules/no-obj-calls
- 'no-obj-calls': ['error'],
- // https://eslint.org/docs/rules/no-sparse-arrays
- 'no-sparse-arrays': ['error'],
- // https://eslint.org/docs/rules/block-scoped-var
- 'block-scoped-var': ['error'],
- /**
- * Best practices
- */
- // https://eslint.org/docs/rules/consistent-return
- 'consistent-return': ['error'],
- // https://eslint.org/docs/rules/default-case
- 'default-case': ['error'],
- // https://eslint.org/docs/rules/dot-notation
- 'dot-notation': [
- 'error',
- {
- allowKeywords: true,
- },
- ],
- // https://eslint.org/docs/rules/guard-for-in [REVISIT ME]
- 'guard-for-in': ['off'],
- // https://eslint.org/docs/rules/no-caller
- 'no-caller': ['error'],
- // https://eslint.org/docs/rules/no-eval
- 'no-eval': ['error'],
- // https://eslint.org/docs/rules/no-extend-native
- 'no-extend-native': ['error'],
- // https://eslint.org/docs/rules/no-extra-bind
- 'no-extra-bind': ['error'],
- // https://eslint.org/docs/rules/no-fallthrough
- 'no-fallthrough': ['error'],
- // https://eslint.org/docs/rules/no-floating-decimal
- 'no-floating-decimal': ['error'],
- // https://eslint.org/docs/rules/no-implied-eval
- 'no-implied-eval': ['error'],
- // https://eslint.org/docs/rules/no-lone-blocks
- 'no-lone-blocks': ['error'],
- // https://eslint.org/docs/rules/no-loop-func
- 'no-loop-func': ['error'],
- // https://eslint.org/docs/rules/no-multi-str
- 'no-multi-str': ['error'],
- // https://eslint.org/docs/rules/no-native-reassign
- 'no-native-reassign': ['error'],
- // https://eslint.org/docs/rules/no-new
- 'no-new': ['error'],
- // https://eslint.org/docs/rules/no-new-func
- 'no-new-func': ['error'],
- // https://eslint.org/docs/rules/no-new-wrappers
- 'no-new-wrappers': ['error'],
- // https://eslint.org/docs/rules/no-octal
- 'no-octal': ['error'],
- // https://eslint.org/docs/rules/no-octal-escape
- 'no-octal-escape': ['error'],
- // https://eslint.org/docs/rules/no-param-reassign [REVISIT ME]
- 'no-param-reassign': ['off'],
- // https://eslint.org/docs/rules/no-proto
- 'no-proto': ['error'],
- // https://eslint.org/docs/rules/no-return-assign
- 'no-return-assign': ['error'],
- // https://eslint.org/docs/rules/no-script-url
- 'no-script-url': ['error'],
- // https://eslint.org/docs/rules/no-self-compare
- 'no-self-compare': ['error'],
- // https://eslint.org/docs/rules/no-sequences
- 'no-sequences': ['error'],
- // https://eslint.org/docs/rules/no-throw-literal
- 'no-throw-literal': ['error'],
- // https://eslint.org/docs/rules/no-with
- 'no-with': ['error'],
- // https://eslint.org/docs/rules/radix
- radix: ['error'],
- // https://eslint.org/docs/rules/space-in-brackets.html
- 'computed-property-spacing': ['error', 'never'],
- // https://eslint.org/docs/rules/space-in-brackets.html
- 'array-bracket-spacing': ['error', 'never'],
- // https://eslint.org/docs/rules/space-in-brackets.html
- 'object-curly-spacing': ['error', 'never'],
- // https://eslint.org/docs/rules/object-shorthand
- 'object-shorthand': ['error', 'properties'],
- // https://eslint.org/docs/rules/space-infix-ops.html
- 'space-infix-ops': ['error'],
- // https://eslint.org/docs/rules/vars-on-top
- 'vars-on-top': ['off'],
- // https://eslint.org/docs/rules/wrap-iife
- 'wrap-iife': ['error', 'any'],
- // https://eslint.org/docs/rules/array-callback-return
- 'array-callback-return': ['error'],
- // https://eslint.org/docs/rules/yoda
- yoda: ['error'],
- // https://eslint.org/docs/rules/no-else-return
- 'no-else-return': ['error', {allowElseIf: false}],
- // https://eslint.org/docs/rules/require-await
- 'require-await': ['error'],
- // https://eslint.org/docs/rules/multiline-comment-style
- 'multiline-comment-style': ['error', 'separate-lines'],
- // https://eslint.org/docs/rules/spaced-comment
- 'spaced-comment': [
- 'error',
- 'always',
- {
- line: {markers: ['/'], exceptions: ['-', '+']},
- block: {exceptions: ['*'], balanced: true},
- },
- ],
- };
- const reactReactRules = {
- // https://github.com/yannickcr/eslint-plugin-react/blob/master/docs/rules/display-name.md
- 'react/display-name': ['off'],
- // https://github.com/yannickcr/eslint-plugin-react/blob/master/docs/rules/no-multi-comp.md
- 'react/no-multi-comp': [
- 'off',
- {
- ignoreStateless: true,
- },
- ],
- // https://github.com/yannickcr/eslint-plugin-react/blob/master/docs/rules/jsx-fragments.md
- 'react/jsx-fragments': ['error', 'element'],
- // https://github.com/yannickcr/eslint-plugin-react/blob/master/docs/rules/jsx-handler-names.md
- // Ensures that any component or prop methods used to handle events are correctly prefixed.
- 'react/jsx-handler-names': [
- 'off',
- {
- eventHandlerPrefix: 'handle',
- eventHandlerPropPrefix: 'on',
- },
- ],
- // https://github.com/yannickcr/eslint-plugin-react/blob/master/docs/rules/jsx-key.md
- 'react/jsx-key': ['error'],
- // https://github.com/yannickcr/eslint-plugin-react/blob/master/docs/rules/jsx-no-undef.md
- 'react/jsx-no-undef': ['error'],
- // Disabled as we use the newer JSX transform babel plugin.
- // https://github.com/yannickcr/eslint-plugin-react/blob/master/docs/rules/jsx-uses-react.md
- 'react/jsx-uses-react': ['off'],
- // https://github.com/yannickcr/eslint-plugin-react/blob/master/docs/rules/jsx-uses-vars.md
- 'react/jsx-uses-vars': ['error'],
- /**
- * Deprecation related rules
- */
- // https://github.com/yannickcr/eslint-plugin-react/blob/master/docs/rules/no-deprecated.md
- 'react/no-deprecated': ['error'],
- // Prevent usage of the return value of React.render
- // deprecation: https://facebook.github.io/react/docs/react-dom.html#render
- // https://github.com/yannickcr/eslint-plugin-react/blob/master/docs/rules/no-render-return-value.md
- 'react/no-render-return-value': ['error'],
- // Children should always be actual children, not passed in as a prop.
- // When using JSX, the children should be nested between the opening and closing tags. When not using JSX, the children should be passed as additional arguments to React.createElement.
- // https://github.com/yannickcr/eslint-plugin-react/blob/master/docs/rules/no-children-prop.md
- 'react/no-children-prop': ['error'],
- // This rule helps prevent problems caused by using children and the dangerouslySetInnerHTML prop at the same time.
- // React will throw a warning if this rule is ignored.
- // https://github.com/yannickcr/eslint-plugin-react/blob/master/docs/rules/no-danger-with-children.md
- 'react/no-danger-with-children': ['error'],
- // Prevent direct mutation of this.state
- // https://github.com/yannickcr/eslint-plugin-react/blob/master/docs/rules/no-direct-mutation-state.md
- 'react/no-direct-mutation-state': ['error'],
- // https://github.com/yannickcr/eslint-plugin-react/blob/master/docs/rules/no-did-mount-set-state.md
- 'react/no-did-mount-set-state': ['error'],
- // https://github.com/yannickcr/eslint-plugin-react/blob/master/docs/rules/no-did-update-set-state.md"
- 'react/no-did-update-set-state': ['error'],
- // https://github.com/yannickcr/eslint-plugin-react/blob/master/docs/rules/no-redundant-should-component-update.md
- 'react/no-redundant-should-component-update': ['error'],
- // https://github.com/yannickcr/eslint-plugin-react/blob/master/docs/rules/no-typos.md
- 'react/no-typos': ['error'],
- // Prevent invalid characters from appearing in markup
- // https://github.com/yannickcr/eslint-plugin-react/blob/master/docs/rules/no-unescaped-entities.md
- 'react/no-unescaped-entities': ['off'],
- // https://github.com/yannickcr/eslint-plugin-react/blob/master/docs/rules/no-unknown-property.md
- 'react/no-unknown-property': ['error', {ignore: ['css']}],
- // https://github.com/yannickcr/eslint-plugin-react/blob/master/docs/rules/no-unused-prop-types.md
- // Disabled since this currently fails to correctly detect a lot of
- // typescript prop type usage.
- 'react/no-unused-prop-types': ['off'],
- // We do not need proptypes since we're using typescript
- 'react/prop-types': ['off'],
- // When writing the render method in a component it is easy to forget to return the JSX content.
- // This rule will warn if the return statement is missing.
- // https://github.com/yannickcr/eslint-plugin-react/blob/master/docs/rules/require-render-return.md
- 'react/require-render-return': ['error'],
- // Disabled as we are using the newer JSX transform babel plugin.
- // https://github.com/yannickcr/eslint-plugin-react/blob/master/docs/rules/react-in-jsx-scope.md
- 'react/react-in-jsx-scope': ['off'],
- // https://github.com/yannickcr/eslint-plugin-react/blob/master/docs/rules/self-closing-comp.md
- 'react/self-closing-comp': ['error'],
- // This also causes issues with typescript
- // See: https://github.com/yannickcr/eslint-plugin-react/issues/2066
- //
- // https://github.com/yannickcr/eslint-plugin-react/blob/master/docs/rules/sort-comp.md
- 'react/sort-comp': ['warn'],
- // Disabled because of prettier
- // https://github.com/yannickcr/eslint-plugin-react/blob/master/docs/rules/wrap-multilines.md
- 'react/jsx-wrap-multilines': ['off'],
- // Consistent <Component booleanProp /> (never add ={true})
- // https://github.com/yannickcr/eslint-plugin-react/blob/master/docs/rules/jsx-boolean-value.md
- 'react/jsx-boolean-value': ['error', 'never'],
- // Consistent function component declaration styles
- // https://github.com/jsx-eslint/eslint-plugin-react/blob/master/docs/rules/function-component-definition.md
- 'react/function-component-definition': [
- 'error',
- {namedComponents: 'function-declaration'},
- ],
- };
- const reactImportRules = {
- // Not recommended to be enabled with typescript-eslint
- // https://typescript-eslint.io/linting/troubleshooting/performance-troubleshooting/#eslint-plugin-import
- 'import/no-unresolved': ['off'],
- 'import/named': ['off'],
- 'import/default': ['off'],
- 'import/export': ['off'],
- 'import/no-named-as-default-member': ['off'],
- // Redflags
- // do not allow a default import name to match a named export (airbnb: error)
- // Issue with `DefaultIssuePlugin` and `app/plugins/index`
- // https://github.com/benmosher/eslint-plugin-import/blob/master/docs/rules/no-named-as-default.md
- 'import/no-named-as-default': ['off'],
- // disallow use of jsdoc-marked-deprecated imports
- // https://github.com/benmosher/eslint-plugin-import/blob/master/docs/rules/no-deprecated.md
- 'import/no-deprecated': ['off'],
- // Forbid mutable exports (airbnb: error)
- // https://github.com/benmosher/eslint-plugin-import/blob/master/docs/rules/no-mutable-exports.md
- // TODO: enable?
- 'import/no-mutable-exports': ['off'],
- // disallow require()
- // https://github.com/benmosher/eslint-plugin-import/blob/master/docs/rules/no-commonjs.md
- 'import/no-commonjs': ['off'],
- // disallow AMD require/define
- // https://github.com/benmosher/eslint-plugin-import/blob/master/docs/rules/no-amd.md
- 'import/no-amd': ['error'],
- // disallow duplicate imports
- // https://github.com/benmosher/eslint-plugin-import/blob/master/docs/rules/no-duplicates.md
- 'import/no-duplicates': ['error'],
- // disallow namespace imports
- // https://github.com/benmosher/eslint-plugin-import/blob/master/docs/rules/no-namespace.md
- 'import/no-namespace': ['off'],
- // Ensure consistent use of file extension within the import path
- // https://github.com/benmosher/eslint-plugin-import/blob/master/docs/rules/extensions.md
- // TODO this fucks up getsentry
- 'import/extensions': [
- 'off',
- 'always',
- {
- js: 'never',
- jsx: 'never',
- },
- ],
- // Require a newline after the last import/require in a group
- // https://github.com/benmosher/eslint-plugin-import/blob/master/docs/rules/newline-after-import.md
- 'import/newline-after-import': ['error'],
- // Require modules with a single export to use a default export (airbnb: error)
- // https://github.com/benmosher/eslint-plugin-import/blob/master/docs/rules/prefer-default-export.md
- 'import/prefer-default-export': ['off'],
- // Restrict which files can be imported in a given folder
- // https://github.com/benmosher/eslint-plugin-import/blob/master/docs/rules/no-restricted-paths.md
- 'import/no-restricted-paths': ['off'],
- // Forbid modules to have too many dependencies
- // https://github.com/benmosher/eslint-plugin-import/blob/master/docs/rules/max-dependencies.md
- 'import/max-dependencies': ['off', {max: 10}],
- // Forbid import of modules using absolute paths
- // https://github.com/benmosher/eslint-plugin-import/blob/master/docs/rules/no-absolute-path.md
- 'import/no-absolute-path': ['error'],
- // Forbid require() calls with expressions (airbnb: error)
- // https://github.com/benmosher/eslint-plugin-import/blob/master/docs/rules/no-dynamic-require.md
- 'import/no-dynamic-require': ['off'],
- // Use webpack default chunk names
- 'import/dynamic-import-chunkname': ['off'],
- // prevent importing the submodules of other modules
- // https://github.com/benmosher/eslint-plugin-import/blob/master/docs/rules/no-internal-modules.md
- 'import/no-internal-modules': [
- 'off',
- {
- allow: [],
- },
- ],
- // Warn if a module could be mistakenly parsed as a script by a consumer
- // leveraging Unambiguous JavaScript Grammar
- // https://github.com/benmosher/eslint-plugin-import/blob/master/docs/rules/unambiguous.md
- // this should not be enabled until this proposal has at least been *presented* to TC39.
- // At the moment, it"s not a thing.
- 'import/unambiguous': ['off'],
- // Forbid Webpack loader syntax in imports
- // https://github.com/benmosher/eslint-plugin-import/blob/master/docs/rules/no-webpack-loader-syntax.md
- 'import/no-webpack-loader-syntax': ['error'],
- // Prevent unassigned imports
- // https://github.com/benmosher/eslint-plugin-import/blob/master/docs/rules/no-unassigned-import.md
- // importing for side effects is perfectly acceptable, if you need side effects.
- 'import/no-unassigned-import': ['off'],
- // Prevent importing the default as if it were named
- // https://github.com/benmosher/eslint-plugin-import/blob/master/docs/rules/no-named-default.md
- 'import/no-named-default': ['error'],
- // Reports if a module"s default export is unnamed
- // https://github.com/benmosher/eslint-plugin-import/blob/d9b712ac7fd1fddc391f7b234827925c160d956f/docs/rules/no-anonymous-default-export.md
- 'import/no-anonymous-default-export': [
- 'error',
- {
- allowArray: false,
- allowArrowFunction: false,
- allowAnonymousClass: false,
- allowAnonymousFunction: false,
- allowCallExpression: true,
- allowLiteral: false,
- allowObject: false,
- },
- ],
- };
- const reactJestRules = {
- 'jest/no-disabled-tests': 'error',
- };
- const reactRules = {
- ...reactReactRules,
- ...reactImportRules,
- ...reactJestRules,
- /**
- * React hooks
- */
- 'react-hooks/exhaustive-deps': 'error',
- // Biome not yet enforcing all parts of this rule https://github.com/biomejs/biome/issues/1984
- 'react-hooks/rules-of-hooks': 'error',
- /**
- * Custom
- */
- // highlights literals in JSX components w/o translation tags
- 'getsentry/jsx-needs-il8n': ['off'],
- 'testing-library/render-result-naming-convention': 'off',
- 'testing-library/no-unnecessary-act': 'off',
- // Disabled as we have many tests which render as simple validations
- 'jest/expect-expect': 'off',
- // Disabled as we have some comment out tests that cannot be
- // uncommented due to typescript errors.
- 'jest/no-commented-out-tests': 'off',
- // Disabled as we do sometimes have conditional expects
- 'jest/no-conditional-expect': 'off',
- // Useful for exporting some test utilities
- 'jest/no-export': 'off',
- 'typescript-sort-keys/interface': [
- 'error',
- 'asc',
- {caseSensitive: true, natural: false, requiredFirst: true},
- ],
- };
- const appRules = {
- /**
- * emotion rules for v10
- *
- * This probably aren't as necessary anymore, but let's remove when we move to v11
- */
- '@emotion/jsx-import': 'off',
- '@emotion/no-vanilla': 'error',
- '@emotion/import-from-emotion': 'error',
- '@emotion/styled-import': 'error',
- // no-undef is redundant with typescript as tsc will complain
- // A downside is that we won't get eslint errors about it, but your editors should
- // support tsc errors so....
- // https://eslint.org/docs/rules/no-undef
- 'no-undef': 'off',
- // Let formatter handle this
- 'arrow-body-style': 'off',
- /**
- * Need to use typescript version of these rules
- * https://eslint.org/docs/rules/no-shadow
- */
- 'no-shadow': 'off',
- '@typescript-eslint/no-shadow': 'error',
- // This only override the `args` rule (which is "none"). There are too many errors and it's difficult to manually
- // fix them all, so we'll have to incrementally update.
- // https://eslint.org/docs/rules/no-unused-vars
- 'no-unused-vars': 'off',
- '@typescript-eslint/no-unused-vars': [
- 'error',
- {
- vars: 'all',
- args: 'all',
- // TODO(scttcper): We could enable this to enforce catch (error)
- // https://eslint.org/docs/latest/rules/no-unused-vars#caughterrors
- caughtErrors: 'none',
- // Ignore vars that start with an underscore
- // e.g. if you want to omit a property using object spread:
- //
- // const {name: _name, ...props} = this.props;
- //
- varsIgnorePattern: '^_',
- argsIgnorePattern: '^_',
- destructuredArrayIgnorePattern: '^_',
- },
- ],
- // https://eslint.org/docs/rules/no-use-before-define
- 'no-use-before-define': 'off',
- // This seems to have been turned on while previously it had been off
- '@typescript-eslint/no-use-before-define': ['off'],
- /**
- * Restricted imports, e.g. deprecated libraries, etc
- *
- * See: https://eslint.org/docs/rules/no-restricted-imports
- */
- 'no-restricted-imports': [
- 'error',
- {
- paths: [
- {
- name: 'enzyme',
- message:
- 'Please import from `sentry-test/enzyme` instead. See: https://github.com/getsentry/frontend-handbook#undefined-theme-properties-in-tests for more information',
- },
- {
- name: '@testing-library/react',
- message:
- 'Please import from `sentry-test/reactTestingLibrary` instead so that we can ensure consistency throughout the codebase',
- },
- {
- name: '@testing-library/react-hooks',
- message:
- 'Please import from `sentry-test/reactTestingLibrary` instead so that we can ensure consistency throughout the codebase',
- },
- {
- name: '@testing-library/user-event',
- message:
- 'Please import from `sentry-test/reactTestingLibrary` instead so that we can ensure consistency throughout the codebase',
- },
- {
- name: '@sentry/browser',
- message:
- 'Please import from `@sentry/react` to ensure consistency throughout the codebase.',
- },
- {
- name: 'marked',
- message:
- "Please import marked from 'app/utils/marked' so that we can ensure sanitation of marked output",
- },
- {
- name: 'lodash',
- message:
- "Please import lodash utilities individually. e.g. `import isEqual from 'lodash/isEqual';`. See https://github.com/getsentry/frontend-handbook#lodash from for information",
- },
- {
- name: 'lodash/get',
- message:
- 'Optional chaining `?.` and nullish coalescing operators `??` are available and preferred over using `lodash/get`. See https://github.com/getsentry/frontend-handbook#new-syntax for more information',
- },
- {
- name: 'react-bootstrap',
- message:
- 'Avoid usage of any react-bootstrap components as it will soon be removed',
- },
- {
- name: 'sentry/utils/theme',
- importNames: ['lightColors', 'darkColors'],
- message:
- "'lightColors' and 'darkColors' exports intended for use in Storybook only. Instead, use theme prop from emotion or the useTheme hook.",
- },
- {
- name: 'react-router',
- importNames: ['withRouter'],
- message:
- "Use 'useLocation', 'useParams', 'useNavigate', 'useRoutes' from sentry/utils instead.",
- },
- {
- name: 'sentry/utils/withSentryRouter',
- importNames: ['withSentryRouter'],
- message:
- "Use 'useLocation', 'useParams', 'useNavigate', 'useRoutes' from sentry/utils instead.",
- },
- ],
- },
- ],
- /**
- * Better import sorting
- */
- 'sort-imports': 'off',
- 'import/order': 'off',
- 'simple-import-sort/imports': [
- 'error',
- {
- groups: [
- // Side effect imports.
- ['^\\u0000'],
- // Node.js builtins.
- // biome-ignore lint/correctness/noNodejsModules: Need to get the list of things!
- [`^(${require('node:module').builtinModules.join('|')})(/|$)`],
- // Packages. `react` related packages come first.
- ['^react', '^@?\\w'],
- // Test should be separate from the app
- ['^(sentry-test|getsentry-test)(/.*|$)'],
- // Internal packages.
- ['^(sentry-locale|sentry-images)(/.*|$)'],
- ['^(getsentry-images)(/.*|$)'],
- ['^(app|sentry)(/.*|$)'],
- // Getsentry packages.
- ['^(admin|getsentry)(/.*|$)'],
- // Style imports.
- ['^.+\\.less$'],
- // Parent imports. Put `..` last.
- ['^\\.\\.(?!/?$)', '^\\.\\./?$'],
- // Other relative imports. Put same-folder imports and `.` last.
- ['^\\./(?=.*/)(?!/?$)', '^\\.(?!/?$)', '^\\./?$'],
- ],
- },
- ],
- 'sentry/no-digits-in-tn': ['error'],
- 'sentry/no-dynamic-translations': ['error'],
- // https://github.com/xojs/eslint-config-xo-typescript/blob/9791a067d6a119a21a4db72c02f1da95e25ffbb6/index.js#L95
- '@typescript-eslint/no-restricted-types': [
- 'error',
- {
- types: {
- // TODO(scttcper): Turn object on to make our types more strict
- // object: {
- // message: 'The `object` type is hard to use. Use `Record<string, unknown>` instead. See: https://github.com/typescript-eslint/typescript-eslint/pull/848',
- // fixWith: 'Record<string, unknown>'
- // },
- Buffer: {
- message:
- 'Use Uint8Array instead. See: https://sindresorhus.com/blog/goodbye-nodejs-buffer',
- suggest: ['Uint8Array'],
- },
- '[]': "Don't use the empty array type `[]`. It only allows empty arrays. Use `SomeType[]` instead.",
- '[[]]':
- "Don't use `[[]]`. It only allows an array with a single element which is an empty array. Use `SomeType[][]` instead.",
- '[[[]]]': "Don't use `[[[]]]`. Use `SomeType[][][]` instead.",
- },
- },
- ],
- // TODO(scttcper): Turn no-empty-object-type on to make our types more strict
- // '@typescript-eslint/no-empty-object-type': 'error',
- // TODO(scttcper): Turn no-function on to make our types more strict
- // '@typescript-eslint/no-unsafe-function-type': 'error',
- '@typescript-eslint/no-wrapper-object-types': 'error',
- // Naming convention enforcements
- '@typescript-eslint/naming-convention': [
- 'error',
- {
- selector: 'typeLike',
- format: ['PascalCase'],
- leadingUnderscore: 'allow',
- },
- {
- selector: 'enumMember',
- format: ['UPPER_CASE'],
- },
- ],
- // Don't allow lookbehind expressions in regexp as they crash safari
- // We've accidentally used lookbehinds a few times and caused problems.
- 'no-lookahead-lookbehind-regexp/no-lookahead-lookbehind-regexp': [
- 'error',
- 'no-lookbehind',
- 'no-negative-lookbehind',
- ],
- };
- const strictRules = {
- // https://eslint.org/docs/rules/no-console
- 'no-console': ['error'],
- // https://github.com/yannickcr/eslint-plugin-react/blob/master/docs/rules/no-is-mounted.md
- 'react/no-is-mounted': ['error'],
- // https://github.com/yannickcr/eslint-plugin-react/blob/master/docs/rules/no-find-dom-node.md
- // Recommended to use callback refs instead
- 'react/no-find-dom-node': ['error'],
- // https://github.com/yannickcr/eslint-plugin-react/blob/master/docs/rules/no-string-refs.md
- // This is now considered legacy, callback refs preferred
- 'react/no-string-refs': ['error'],
- 'jest/no-large-snapshots': ['error', {maxSize: 2000}],
- 'sentry/no-styled-shortcut': ['error'],
- };
- const extendsList = [
- 'plugin:jest/recommended',
- 'plugin:jest-dom/recommended',
- 'plugin:import/typescript',
- ];
- if (detectDeprecations) {
- extendsList.push('plugin:deprecation/recommended');
- }
- module.exports = {
- root: true,
- extends: extendsList,
- plugins: [
- 'jest-dom',
- 'testing-library',
- 'typescript-sort-keys',
- 'react-hooks',
- '@typescript-eslint',
- '@emotion',
- 'import',
- 'react',
- 'sentry',
- 'simple-import-sort',
- 'no-lookahead-lookbehind-regexp',
- ],
- parser: '@typescript-eslint/parser',
- parserOptions: detectDeprecations
- ? {
- warnOnUnsupportedTypeScriptVersion: false,
- ecmaVersion: 6,
- sourceType: 'module',
- ecmaFeatures: {
- jsx: true,
- modules: true,
- legacyDecorators: true,
- },
- project: './tsconfig.json',
- }
- : {
- warnOnUnsupportedTypeScriptVersion: false,
- ecmaVersion: 6,
- sourceType: 'module',
- ecmaFeatures: {
- jsx: true,
- modules: true,
- legacyDecorators: true,
- },
- },
- env: {
- browser: true,
- es6: true,
- jest: true,
- jquery: true, // hard-loaded into vendor.js
- },
- globals: {
- require: false,
- expect: false,
- MockApiClient: true,
- tick: true,
- jest: true,
- },
- settings: {
- react: {
- version: '17.0.2', // React version, can not `detect` because of getsentry
- },
- 'import/parsers': {
- '@typescript-eslint/parser': ['.ts', '.tsx'],
- },
- 'import/resolver': {
- typescript: {},
- },
- 'import/extensions': ['.js', '.jsx'],
- },
- rules: {
- ...baseRules,
- ...reactRules,
- ...appRules,
- ...strictRules,
- 'react-hooks/rules-of-hooks': 'error',
- 'react-hooks/exhaustive-deps': [
- 'error',
- {additionalHooks: '(useEffectAfterFirstRender|useMemoWithPrevious)'},
- ],
- 'no-restricted-imports': [
- 'error',
- {
- patterns: [
- {
- group: ['sentry/components/devtoolbar/*'],
- message: 'Do not depend on toolbar internals',
- },
- ],
- paths: [
- {
- name: '@testing-library/react',
- message:
- 'Please import from `sentry-test/reactTestingLibrary` instead so that we can ensure consistency throughout the codebase',
- },
- {
- name: '@testing-library/react-hooks',
- message:
- 'Please import from `sentry-test/reactTestingLibrary` instead so that we can ensure consistency throughout the codebase',
- },
- {
- name: '@testing-library/user-event',
- message:
- 'Please import from `sentry-test/reactTestingLibrary` instead so that we can ensure consistency throughout the codebase',
- },
- {
- name: '@sentry/browser',
- message:
- 'Please import from `@sentry/react` to ensure consistency throughout the codebase.',
- },
- {
- name: 'marked',
- message:
- "Please import marked from 'app/utils/marked' so that we can ensure sanitation of marked output",
- },
- {
- name: 'lodash',
- message:
- "Please import lodash utilities individually. e.g. `import isEqual from 'lodash/isEqual';`. See https://github.com/getsentry/frontend-handbook#lodash from for information",
- },
- {
- name: 'lodash/get',
- message:
- 'Optional chaining `?.` and nullish coalescing operators `??` are available and preferred over using `lodash/get`. See https://github.com/getsentry/frontend-handbook#new-syntax for more information',
- },
- {
- name: 'sentry/utils/theme',
- importNames: ['lightColors', 'darkColors'],
- message:
- "'lightColors' and 'darkColors' exports intended for use in Storybook only. Instead, use theme prop from emotion or the useTheme hook.",
- },
- {
- name: 'react-router',
- importNames: ['withRouter'],
- message:
- "Use 'useLocation', 'useParams', 'useNavigate', 'useRoutes' from sentry/utils instead.",
- },
- {
- name: 'sentry/utils/withSentryRouter',
- importNames: ['withSentryRouter'],
- message:
- "Use 'useLocation', 'useParams', 'useNavigate', 'useRoutes' from sentry/utils instead.",
- },
- {
- name: 'qs',
- message: 'Please use query-string instead of qs',
- },
- {
- name: 'moment',
- message: 'Please import moment-timezone instead of moment',
- },
- ],
- },
- ],
- // TODO(@anonrig): Remove this from eslint-sentry-config
- 'space-infix-ops': 'off',
- 'object-shorthand': 'off',
- 'object-curly-spacing': 'off',
- 'import/no-amd': 'off',
- 'no-danger-with-children': 'off',
- 'no-fallthrough': 'off',
- 'no-obj-calls': 'off',
- 'array-bracket-spacing': 'off',
- 'computed-property-spacing': 'off',
- 'react/no-danger-with-children': 'off',
- 'jest/no-disabled-tests': 'off',
- },
- // JSON file formatting is handled by Biome. ESLint should not be linting
- // and formatting these files.
- ignorePatterns: ['*.json'],
- overrides: [
- {
- files: ['static/app/components/devtoolbar/**/*.{ts,tsx}'],
- rules: {
- 'no-restricted-imports': [
- 'error',
- {
- paths: [
- {
- name: 'sentry/utils/queryClient',
- message:
- 'Import from `@tanstack/react-query` and `./hooks/useFetchApiData` or `./hooks/useFetchInfiniteApiData` instead.',
- },
- ],
- },
- ],
- },
- },
- {
- files: ['static/**/*.spec.{ts,js}', 'tests/js/**/*.{ts,js}'],
- extends: ['plugin:testing-library/react', ...extendsList],
- rules: {
- ...baseRules,
- ...reactRules,
- ...appRules,
- ...strictRules,
- // TODO(@anonrig): Remove this from eslint-sentry-config
- 'space-infix-ops': 'off',
- 'object-shorthand': 'off',
- 'object-curly-spacing': 'off',
- 'import/no-amd': 'off',
- 'no-danger-with-children': 'off',
- 'no-fallthrough': 'off',
- 'no-obj-calls': 'off',
- 'array-bracket-spacing': 'off',
- 'computed-property-spacing': 'off',
- 'react/no-danger-with-children': 'off',
- 'jest/no-disabled-tests': 'off',
- },
- },
- {
- // We specify rules explicitly for the sdk-loader here so we do not have
- // eslint ignore comments included in the source file, which is consumed
- // by users.
- files: ['**/js-sdk-loader.ts'],
- rules: {
- 'no-console': 'off',
- },
- },
- ],
- };
|