website/docs/linting/rule-reference.md
import {SinceVersion} from '../../components/VersionTags';
all {#toc-all}While all isn't technically a lint rule, it's worth mentioning here. all sets the default
level for lint rules that don't have a level set explicitly. all can only
occur as the first entry in a .flowconfig or as the first rule in a --lints
flag. It's not allowed in comments at all because it would have different
semantics than would be expected.
ambiguous-object-type {#toc-ambiguous-object-type}Triggers when you use object type syntax without explicitly specifying exactness or inexactness.
This lint setting is ignored when exact_by_default is set to false.
// flowlint ambiguous-object-type:error
type A = {x: number}; // Error
type B = {x: number, ...} // Ok
type C = {| x: number |} // Ok
deprecated-type {#toc-deprecated-type}Triggered on the bool type, which is just an alias for boolean. Just use boolean instead.
// flowlint deprecated-type:error
type A = Array<bool>; // Error
implicit-inexact-object {#toc-implicit-inexact-object}Like ambiguous-object-type, except triggers even when the exact_by_default option is set to false.
nested-component {#toc-nested-component}Triggers when a component is defined directly inside another component or hook. Nested components are problematic because React cannot preserve the state of a nested component across re-renders of the parent — each render creates a brand new component type, so React always unmounts and remounts it.
This lint is enabled as an error by default. It applies to component syntax declarations, which require component_syntax=true in your .flowconfig.
import * as React from 'react';
component Outer() {
component Inner() { // Error
return null;
}
return <Inner />;
}
The lint also fires when a component is defined inside a hook:
import * as React from 'react';
hook useItems() {
component ItemView() { // Error
return null;
}
return ItemView;
}
To fix, move the nested component to the top level:
import * as React from 'react';
component Inner() {
return null;
}
component Outer() {
return <Inner />;
}
nested-hook {#toc-nested-hook}Triggers when a hook is defined directly inside another component or hook. Nested hooks are problematic because they break the Rules of Hooks — React relies on hooks being called in a consistent order at the top level of a component or hook, and a nested hook definition obscures the call structure.
This lint is enabled as an error by default. It applies to hook syntax declarations, which require component_syntax=true in your .flowconfig.
import * as React from 'react';
import {useState} from 'react';
component Foo() {
hook useNested() { // Error
return useState();
}
return null;
}
The lint also fires when a hook is defined inside another hook:
import {useState} from 'react';
hook useFoo() {
hook useNested() { // Error
return useState();
}
return useNested();
}
To fix, move the nested hook to the top level:
import * as React from 'react';
import {useState} from 'react';
hook useNested() {
return useState();
}
component Foo() {
useNested();
return null;
}
libdef-override <SinceVersion version="0.265" /> {#toc-libdef-override}Triggers when a library definition overrides an existing built-in definition. This can happen when a .js.flow library file or a flow-typed stub re-declares a global variable, type, or module that Flow already provides in its builtins. Overriding built-in definitions can lead to surprising behaviors, because the order in which library files are loaded affects which definition wins.
This lint is enabled as an error by default.
There are two forms of this error. The first is a name override, which fires when a library definition re-declares a global variable or type that already exists:
// In a library definition file (.js.flow or [libs])
declare const globalThis: mixed; // Error: overrides built-in globalThis
declare type React$Node = string; // Error: overrides built-in React$Node
The second is a module override, which fires when a library definition re-declares a module that is already declared elsewhere:
// In a library definition file
declare module 'react' { // Error: overrides built-in react module
declare module.exports: any;
}
This error also fires when the same library file is included twice (for example, if both a directory and its subdirectory are listed in the [libs] section of .flowconfig), causing all declarations in that file to conflict with themselves.
To fix the error, remove the redundant declaration from your library definition. If you intentionally need to override a built-in, you can suppress the error with a $FlowFixMe comment:
// $FlowFixMe[libdef-override]
declare type React$Node = string | number | null;
You can also disable this lint for an entire project in your .flowconfig:
[lints]
libdef-override=off
internal-type {#toc-internal-type}Triggers when you use an internal Flow type directly in your code. Internal types are implementation details that have public-facing alternatives you should use instead.
This lint is enabled as an error by default.
There are two categories of internal types that trigger this lint:
Dollar-prefixed utility types such as $Omit, $Enum, and $EnumValue have non-dollar equivalents (Omit, Enum, EnumValue):
type A = $Omit<{x: number, y: number}, 'x'>; // Error
type B = Omit<{x: number, y: number}, 'x'>; // Ok
React$ types such as React$Node and React$ElementConfig should be accessed via the React namespace using dot syntax (React.Node, React.ElementConfig):
import * as React from 'react';
type A = React$Node; // Error
type B = React.Node; // Ok
nonstrict-import {#toc-nonstrict-import}Used in conjunction with Flow Strict. Triggers when importing a non @flow strict module. When enabled, dependencies of a @flow strict module must also be @flow strict.
react-intrinsic-overlap {#toc-react-intrinsic-overlap}Triggers when a local definition shares its name with an intrinsic JSX element (such as div, span, or input) and has a type that could be used as a React component. Because JSX treats lowercase element names as intrinsics, writing <div /> always refers to the HTML element, never to a local binding called div. If the local binding is a function, class, abstract component, callable object, or mixed, this overlap is likely a mistake that leads to confusing behavior.
This lint is off by default.
A local function whose name overlaps with an intrinsic:
// flowlint react-intrinsic-overlap:error
import * as React from 'react';
declare function div(): React.Node; // ERROR
<div />;
A local variable with a non-component type does not trigger the lint, because there is no ambiguity:
// flowlint react-intrinsic-overlap:error
import * as React from 'react';
const div = 2 / 3;
<div />; // OK
To fix the error, rename the local definition so it does not collide with an intrinsic element name:
// flowlint react-intrinsic-overlap:error
import * as React from 'react';
declare function MyDiv(): React.Node;
<MyDiv />;
sketchy-null {#toc-sketchy-null}Triggers when you do an existence check on a value that can be either null/undefined or falsey.
For example:
// flowlint sketchy-null:error
const x: ?number = 5;
if (x) {} // Error: sketchy because x could be either null or 0.
const y: number = 5;
if (y) {} // not sketchy because y can't be null, only 0.
const z: ?{foo: number} = {foo: 5};
if (z) {} // not sketchy, because z can't be falsey, only null/undefined.
Setting sketchy-null sets the level for all sketchy null checks, but there are more granular rules for particular types. These are:
sketchy-null-boolsketchy-null-numbersketchy-null-stringsketchy-null-mixedsketchy-null-bigintThe type-specific variants are useful for specifying that some types of sketchy null checks are acceptable while others should be errors/warnings. For example, if you want to allow boolean sketchy null checks (for the pattern of treating undefined optional booleans as false) but forbid other types of sketchy null checks, you can do so with this .flowconfig [lints] section:
[lints]
sketchy-null=warn
sketchy-null-bool=off
and now
function foo (bar: ?boolean): void {
if (bar) {
...
} else {
...
}
}
doesn't report a warning.
Suppressing one type of sketchy null check only suppresses that type, so, for example
// flowlint sketchy-null:error, sketchy-null-bool:off
const x: ?(number | boolean) = 0;
if (x) {} // Error
would still have a sketchy-null-number error on line 3.
sketchy-number {#toc-sketchy-number}Triggers when a number is used in a manner which may lead to unexpected results if the value is falsy.
Currently, this lint triggers if a number appears in:
&& expression.As a motivating example, consider this common idiom in React:
{showFoo && <Foo />}
Here, showFoo is a boolean which controls whether or not to display the <Foo /> element. If showFoo is true, then this evaluates to {<Foo />}. If showFoo is false, then this evaluates to {false}, which doesn't display anything.
Now suppose that instead of a boolean, we have a numerical value representing, say, the number of comments on a post. We want to display a count of the comments, unless there are no comments. We might naively try to do something similar to the boolean case:
{count && <>[{count} comments]</>}
If count is, say, 5, then this displays "[5 comments]". However, if count is 0, then this displays "0" instead of displaying nothing. (This problem is unique to number because 0 and NaN are the only falsy values which React renders with a visible result.) This could be subtly dangerous: if this immediately follows another numerical value, it might appear to the user that we have multiplied that value by 10! Instead, we should do a proper conditional check:
{count ? <>[{count} comments]</> : null}
unclear-type {#toc-unclear-type}Triggers when you use any, Object, or Function as type annotations. These
types are unsafe.
// flowlint unclear-type:error
declare const a: any; // Error
declare const c: Object; // Error
declare const d: Function; // Error
unnecessary-invariant {#toc-unnecessary-invariant}Triggers when you use invariant to check a condition which we know must be truthy based on the available type information. This is quite conservative: for example, if all we know about the condition is that it is a boolean, then the lint will not fire even if the condition must be true at runtime.
Note that this lint does not trigger when we know a condition is always false. It is a common idiom to use invariant() or invariant(false, ...) to throw in code that should be unreachable.
// flowlint unnecessary-invariant:error
declare function invariant(boolean): void;
declare const x: Array<string>; // Array is truthy
invariant(x); // Error
unnecessary-optional-chain {#toc-unnecessary-optional-chain}Triggers when you use ?. where it isn't needed. This comes in two main flavors. The first is when the left-hand-side cannot be nullish:
// flowlint unnecessary-optional-chain:error
type Foo = {
bar: number
}
declare const foo: Foo;
foo?.bar; // Error
The second is when the left-hand-side could be nullish, but the short-circuiting behavior of ?. is sufficient to handle it anyway:
// flowlint unnecessary-optional-chain:error
type Foo = {
bar: {
baz: number
}
}
declare const foo: ?Foo;
foo?.bar?.baz; // Error
In the second example, the first use of ?. is valid, since foo is potentially nullish, but the second use of ?. is unnecessary. The left-hand-side of the second ?. (foo?.bar) can only be nullish as a result of foo being nullish, and when foo is nullish, short-circuiting lets us avoid the second ?. altogether!
foo?.bar.baz;
This makes it clear to the reader that bar is not a potentially nullish property.
unsafe-getters-setters {#toc-unsafe-getters-setters}Triggers when you use getters or setters. Getters and setters can have side effects and are unsafe.
For example:
// flowlint unsafe-getters-setters:error
let a = 1;
const o = {
get a() { return a; }, // Error: unsafe-getters-setters
set b(x: number) { a = x; }, // Error: unsafe-getters-setters
c: 10,
};
unsafe-object-assign {#toc-unsafe-object-assign}Triggers on any use of Object.assign. Flow's support for Object.assign is unsafe, and you should use object spreads instead.
This lint is enabled as an error by default.
Object.assign mutates the first argument in place, which is difficult to track through the type system. The spread syntax ({...obj1, ...obj2}) is a safer alternative because it always creates a new object, and Flow can type it precisely.
const defaults = {color: 'red', size: 10};
const overrides = {color: 'blue'};
const config = Object.assign(defaults, overrides); // Error
Use the spread syntax instead:
const defaults = {color: 'red', size: 10};
const overrides = {color: 'blue'};
const config = {...defaults, ...overrides}; // Ok
untyped-import {#toc-untyped-import}Triggers when you import from an untyped file. Importing from an untyped file
results in those imports being typed as any, which is unsafe.
untyped-type-import {#toc-untyped-type-import}Triggers when you import a type from an untyped file. Importing a type from an
untyped file results in an any alias, which is typically not the intended behavior.
Enabling this lint brings extra attention to this case and can help improve Flow
coverage of typed files by limiting the spread of implicit any types.
unused-promise {#toc-unused-promise}Triggers when a Promise is unused. This can be dangerous, because errors are potentially unhandled, and the code may not execute in the desired order.
A promise can be "used" by...
awaiting it.then with a rejection handler (i.e., with two arguments).catch.finallyFor example:
// flowlint unused-promise:error
declare function foo(): Promise<void>;
async function bar() {
await foo(); // ok
foo(); // error, we forgot to await!
}
function baz() {
foo().catch(err => {console.log(err)}); // ok
foo(); // error
}
You can explicitly ignore the promise with the void operator (e.g., void foo();).
Note: As of v0.201.0, this rule subsumed the unused-promise-in-async-scope and unused-promise-in-sync-scope rules.
.flowconfig [lints] — configuring lint severities project-wide