-
Notifications
You must be signed in to change notification settings - Fork 12.8k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Variable isn't narrowed within a capturing closure #35124
Comments
Duplicate of #9998. Typescript warns you that the let i: number | undefined;
i = 0;
let j:number = i+1; // works
const func = (k: number) => k === i + 1; // error: Object i is possibly undefined
i = undefined;
func(3); |
The more I see this issue come up, the more I wonder if it’s possible to do better. The analyzer can’t look into function calls, but would it be possible to notice that the variable is not accessed after the closure is created, so the last narrowing can be maintained? Seems like it should be easy enough to determine that syntactically, since the function is already inline. If the function modifies the variable itself, that’s trickier, but that problem seems similar to loop analysis on the surface. |
@j-oliveras which section of the linked issue are you referring to? I'm okay with closing this as a dupe if it's covered by the other issue, but I couldn't find my case on first glance. Anyway, I understand your point. I was hoping the compiler would realize that |
This is kind of the inverse of #9998, but not a direct dupe I don’t think. #9998 is about what happens with local narrowings when you call a function (answer: nothing, the compiler is optimistic and assumes no side effects); the issue here is about what happens to narrowings inside a function that closes over the narrowed variable (answer: narrowings are discarded at function boundaries, with the exception of IIFEs). This particular issue has come up many times in the past though. |
I don't think there's anything better than the current behavior in this case. I assume the suggestion isn't: "uncallable functions should have no type checking inside their implementation" (the arrow function in the example code is uncallable because it's neither immediately invoked nor assigned to anything) So, assuming the function is meant to be callable, what better could be done here? function foo(i: number | undefined) {
i + 1; // no error! foo() is only called safely
}
foo(1);
foo(2);
// foo(undefined); // commented out and then if you uncomment Since that doesn't make much sense, I'm guessing there's some other suggestion being made here, but I'm not seeing it. IIFEs are already inlined because it's one of the few places we can say that the unsafe situation is known to be impossible and not just fortuitously avoided. What better behavior are we hoping for here? |
If |
Yes, it's meant to be callable. I just omitted that from my example for the sake of brevity. |
function foo(i: number | undefined) {
i + 1; // no error! foo() is only called safely
}
foo(1);
foo(2);
// foo(undefined); // commented out I would expect this to error regardless of my proposed behavior, because |
If @fatcerberus is accurately describing the problem, can we change the example code to something that demonstrates the relevant issue? Like, maybe this: function foo() {
let i: number | undefined;
i = 0;
return (k: number) => k === i + 1; // error that i is possibly undefined
}
console.log(foo()(1)); // true
console.log(foo()(0)); // false |
If I had to draw the most realistic picture of a warning on a closed-over variable that really shouldn't exist, it'd look like this: function makeAdder(n?: number) {
n = n ?? 0;
return (m: number) => n + m;
} It seems like the rule is "If a closure is lexically after all assignments to a bare identifier, then the last narrowing can apply". You can break this, though: function adderWrapper(n?: number) {
function setValue(v?: number) {
n = v;
}
n = n ?? 0;
return {
setValue,
add: (m: number) => m + n
}
}
const k = adderWrapper(0);
k.setValue(undefined);
k.add(3); // NaN So the rule would need to be amended to "All assignments to a bare identifier occur lexically and CFA-before the closure expression", though possibly that could be broken too. |
@RyanCavanaugh nailed what I was shooting for. From my outsider perspective, it definitely seems like this could be enabled using a combination of existing lexical and CFA data, though it would be interesting to explore the edge cases, if any indeed exist. |
@RyanCavanaugh the I ended up working around the issue by simply assigning the default value in the function declaration: function makeAdder(n: number = 0) {
return (m: number) => n + m;
} |
Perhaps this should be a separate issue, but is there any way to make this do the right thing with declare function getApple(): string | undefined
function eat() {
const apple = getApple()
apple // type: string | undefined
if (apple === undefined) return
apple // type: string
function vomit() {
apple // type: string | undefined
}
const digest = () => {
apple // type: string
}
} I'm guessing this behavior is because named functions are "lifted" (or whatever it is called) and can in theory be called prior to their definition. It feels like this could be detected, but maybe that is a lot of complexity for something that can be worked earound with lambdas? |
Is this the same issue? Or should I create a separate? function fooOk() {
let r = "b";
() => r;
}
function fooError() {
let r // Variable 'r' implicitly has type 'any' in some locations where its type cannot be determined.(7034)
r = "b";
() => r; // Variable 'r' implicitly has an 'any' type.(7005)
} |
TypeScript 3.7.2
Playground link
Compiler Options:
Input:
Output:
Expected behavior:
The compiler should not complain about the last
i+1
because it's clearly anumber
type after assigning0
to it.I suspect the closure uses the type of
i
from thelet
statement and ignores it being narrowed down later on. Is this expected behavior?The text was updated successfully, but these errors were encountered: