prefer-nullish-coalescing
Enforce using the nullish coalescing operator instead of logical assignments or chaining.
Extending "plugin:@typescript-eslint/stylistic-type-checked"
in an ESLint configuration enables this rule.
Some problems reported by this rule are manually fixable by editor suggestions.
This rule requires type information to run.
The ??
nullish coalescing runtime operator allows providing a default value when dealing with null
or undefined
.
Because the nullish coalescing operator only coalesces when the original value is null
or undefined
, it is much safer than relying upon logical OR operator chaining ||
, which coalesces on any falsy value.
This rule reports when you may consider replacing:
- An
||
operator with??
- An
||=
operator with??=
This rule will not work as expected if strictNullChecks
is not enabled.
module.exports = {
"rules": {
"@typescript-eslint/prefer-nullish-coalescing": "error"
}
};
Options​
This rule accepts the following options
type Options = [
{
allowRuleToRunWithoutStrictNullChecksIKnowWhatIAmDoing?: boolean;
ignoreConditionalTests?: boolean;
ignoreMixedLogicalExpressions?: boolean;
ignorePrimitives?:
| {
bigint?: boolean;
boolean?: boolean;
number?: boolean;
string?: boolean;
[k: string]: unknown;
}
| true;
ignoreTernaryTests?: boolean;
},
];
const defaultOptions: Options = [
{
allowRuleToRunWithoutStrictNullChecksIKnowWhatIAmDoing: false,
ignoreConditionalTests: false,
ignoreTernaryTests: false,
ignoreMixedLogicalExpressions: false,
ignorePrimitives: {
bigint: false,
boolean: false,
number: false,
string: false,
},
},
];
ignoreTernaryTests
​
Setting this option to true
will cause the rule to ignore any ternary expressions that could be simplified by using the nullish coalescing operator. This is set to false
by default.
Incorrect code for ignoreTernaryTests: false
, and correct code for ignoreTernaryTests: true
:
const foo: any = 'bar';
foo !== undefined && foo !== null ? foo : 'a string';
foo === undefined || foo === null ? 'a string' : foo;
foo == undefined ? 'a string' : foo;
foo == null ? 'a string' : foo;
const foo: string | undefined = 'bar';
foo !== undefined ? foo : 'a string';
foo === undefined ? 'a string' : foo;
const foo: string | null = 'bar';
foo !== null ? foo : 'a string';
foo === null ? 'a string' : foo;
Open in PlaygroundCorrect code for ignoreTernaryTests: false
:
const foo: any = 'bar';
foo ?? 'a string';
foo ?? 'a string';
foo ?? 'a string';
foo ?? 'a string';
const foo: string | undefined = 'bar';
foo ?? 'a string';
foo ?? 'a string';
const foo: string | null = 'bar';
foo ?? 'a string';
foo ?? 'a string';
Open in PlaygroundignoreConditionalTests
​
Setting this option to true
will cause the rule to ignore any cases that are located within a conditional test. This is set to false
by default.
Generally expressions within conditional tests intentionally use the falsy fallthrough behavior of the logical or operator, meaning that fixing the operator to the nullish coalesce operator could cause bugs.
If you're looking to enforce stricter conditional tests, you should consider using the strict-boolean-expressions
rule.
Incorrect code for ignoreConditionalTests: false
, and correct code for ignoreConditionalTests: true
:
declare const a: string | null;
declare const b: string | null;
if (a || b) {
}
if ((a ||= b)) {
}
while (a || b) {}
while ((a ||= b)) {}
do {} while (a || b);
for (let i = 0; a || b; i += 1) {}
a || b ? true : false;
Open in PlaygroundCorrect code for ignoreConditionalTests: false
:
declare const a: string | null;
declare const b: string | null;
if (a ?? b) {
}
if ((a ??= b)) {
}
while (a ?? b) {}
while ((a ??= b)) {}
do {} while (a ?? b);
for (let i = 0; a ?? b; i += 1) {}
a ?? b ? true : false;
Open in PlaygroundignoreMixedLogicalExpressions
​
Setting this option to true
will cause the rule to ignore any logical or expressions that are part of a mixed logical expression (with &&
). This is set to false
by default.
Generally expressions within mixed logical expressions intentionally use the falsy fallthrough behavior of the logical or operator, meaning that fixing the operator to the nullish coalesce operator could cause bugs.
If you're looking to enforce stricter conditional tests, you should consider using the strict-boolean-expressions
rule.
Incorrect code for ignoreMixedLogicalExpressions: false
, and correct code for ignoreMixedLogicalExpressions: true
:
declare const a: string | null;
declare const b: string | null;
declare const c: string | null;
declare const d: string | null;
a || (b && c);
a ||= b && c;
(a && b) || c || d;
a || (b && c) || d;
a || (b && c && d);
Open in PlaygroundCorrect code for ignoreMixedLogicalExpressions: false
:
declare const a: string | null;
declare const b: string | null;
declare const c: string | null;
declare const d: string | null;
a ?? (b && c);
a ??= b && c;
(a && b) ?? c ?? d;
a ?? (b && c) ?? d;
a ?? (b && c && d);
Open in PlaygroundNOTE: Errors for this specific case will be presented as suggestions (see below), instead of fixes. This is because it is not always safe to automatically convert ||
to ??
within a mixed logical expression, as we cannot tell the intended precedence of the operator. Note that by design, ??
requires parentheses when used with &&
or ||
in the same expression.
ignorePrimitives
​
If you would like to ignore expressions containing operands of certain primitive types that can be falsy then you may pass an object containing a boolean value for each primitive:
string: true
, ignoresnull
orundefined
unions withstring
(default: false).number: true
, ignoresnull
orundefined
unions withnumber
(default: false).bigint: true
, ignoresnull
orundefined
unions withbigint
(default: false).boolean: true
, ignoresnull
orundefined
unions withboolean
(default: false).
Incorrect code for ignorePrimitives: { string: false }
, and correct code for ignorePrimitives: { string: true }
:
const foo: string | undefined = 'bar';
foo || 'a string';
Open in PlaygroundCorrect code for both ignorePrimitives: { string: false }
and ignorePrimitives: { string: true }
:
const foo: string | undefined = 'bar';
foo ?? 'a string';
Open in PlaygroundAlso, if you would like to ignore all primitives types, you can set ignorePrimitives: true
. It is equivalent to ignorePrimitives: { string: true, number: true, bigint: true, boolean: true }
.
When Not To Use It​
If you are not using TypeScript 3.7 (or greater), then you will not be able to use this rule, as the operator is not supported.