Narrow type of variable when declared as literal (also tuples)
See original GitHub issueTypeScript Version: 2.4.1
Code
The following doesn’t compile, because x
has inferred type string
. I think it would be helpful if it did, but I still want x
to have inferred type string
:
function blah(arg: "foo" | "bar") {
}
let x = "foo";
blah(x);
x = "something else";
Desired behavior:
This would compile.
Actual behavior:
demo.ts(4,6): error TS2345: Argument of type 'string' is not assignable to parameter of type '"foo" | "bar"'.
Suggestion
The type of x
should be inferred to be string
, but it should be narrowed to "foo"
via flow-sensitive typing, much like the below program:
function blah(arg: "foo" | "bar") {
}
let x = "foo";
if (x != "foo") throw "impossible"; // NOTE: codegen would NOT insert this
blah(x);
x = "something else";
Note that this program currently works as desired: x
is still inferred to have type string
, but the guard immediately narrows it to type "foo"
. The assignment after the function call widens x
’s type back to string
.
My suggestion is for TS to treat declarations that assign variables to literals (as in the first example) to automatically perform this flow-sensitive type narrowing, without the need for an unnecessary guard.
I’m suggesting it only for declarations, not general assignments (so only with let
, var
, and const
) or other operations. In addition, I’m only suggesting it for assignments to literal values (literal strings or literal numbers) without function calls, operations, or casts.
Syntax Changes
Nothing changes in syntax.
Code Generation
Nothing changes in code generation.
Semantic Changes
Additional flow-sensitive type narrowing occurs when variables are declared as literals, essentially making the existing
let x = "foo";
behave the same as (in checking, but not in codegen)
let x = "foo";
if (x != "foo") throw "unreachable";
Reverse Compatibility
Due to x
being assigned a more-precise type than it was in the past, some code is now marked as unreachable/invalid that previously passed:
let x = "foo";
if (x == "bar") { // error TS2365: Operator '==' cannot be applied to types '"foo"' and '"blah"'.
// ...
}
The error is correct (in that the comparison could never succeed) but it’s still possibly undesirable.
I don’t know how commonly this occurs in production code for it to be a problem. If it is common, this change could be put behind a flag, or there could be adjustments made to these error cases so that the narrowing doesn’t occur (or still narrows, but won’t cause complaints from the compiler in these error cases).
If the existing behavior is strongly needed, then the change can be circumvented by using an as
cast or an indirection through a function, although these are a little awkward:
let x = "foo" as string;
let x = (() => "foo")()
Tuples and Object Literals
It would be helpful if this extended also to literal objects or literal arrays (as tuples).
For example, the following could compile:
function blah(arg: [number, number]) {
}
let x = [1, 3];
blah(x);
x = [1, 2, 3];
with x
again having inferred type number[]
but being flow-typed to [number, number]
. The same basic considerations apply here as above.
Similarly, it could be useful for objects to have similar flow-typing, although I’m not sure if this introduces new soundness holes:
function get(): number {
return 0;
}
function blah(arg: {a: "foo" | "bar", b: number}) {
// (...)
}
let y = get(); // y: number
let x = {a: "foo", b: y}; // x: {a: string, b: number}
// x is narrowed to {a: "foo", b: number}
blah(x); // this compiles due to x's narrowed type
x.a = "something else"; // this is accepted, because x: {a: string, b: number}.
See https://github.com/Microsoft/TypeScript/issues/16276 and https://github.com/Microsoft/TypeScript/issues/16360 and probably others for related but different approaches to take here.
Issue Analytics
- State:
- Created 6 years ago
- Reactions:22
- Comments:6 (3 by maintainers)
Top GitHub Comments
The more I think about this the more I think it would be highly desirable. I cannot think of any case that would be negatively impacted by making literals be a more narrow type since you can always widen the type. But I can see a lot of positive changes coming from this since you cannot automatically narrow a wide type.
Consider these examples:
A fix in this simple case would be to add e.g.
as Animal
toconst charlie = ...
.Here the fix is quite annoying, you need to go in and declare the type of every nested schema by making all strings string literals.
I would love it if we could move forward on this. Is there something that I could contribute with? Do we need to do some kind of research on how this would affect current projects using TypeScript?
For Tuples the way I’m doing it currently is by using
as const
alongside aWriteable
helper to remove thereadonly
modifier, something like this:Playground Link