-
Notifications
You must be signed in to change notification settings - Fork 12.8k
JS generic inference change after upgrading to 5.1 #55192
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
Comments
Broken in 4.8.4. |
Specifically, broken in #49740. I'll go look at why. |
Some observations given the simplified repro here: // @filename: second.ts
export declare function createSelector<TState, TProps extends any[]>(
selector: (s: TState, ...props: TProps) => void,
getDependents: (s: TState, ...props: TProps) => void
): void;
// @filename: first.js
import { createSelector } from './second';
export const getById = createSelector((state) => void 0, (state,id) => void 0,);
// @filename: first.ts
import { createSelector } from './second';
export const getById = createSelector((state: any) => void 0, (state: any,id) => void 0,);
That implies to me that the root of the difference is Javascript's defaulting type parameters without inferences to any instead of unknown -- and that there's still some kind of difference even if you explicitly provide @jakebailey You wrote the original PR. Is this result expected? Unexpected? As far as I can tell it's a pretty straightforward result of deleting some code, but I'm not sure. |
Wait what |
edit: |
This was a really long time ago, but I don't really understand how the two are related. |
It seems like the contextual type of the arrows -- |
The workaround I finally found was to add another type parameter: export declare function createSelector<TState, TProps extends any[], TDepProps extends TProps>(
selector: (s: TState, ...props: TProps) => void,
getDependents: (s: TState, ...props: TDepProps) => void
): void; I think this makes TS infer the type of |
This issue is specific to how untyped JS is handled, but I believe that this is a result of a broader problem that can even be observed in TS files. See my comment here. When it comes to what happens here is that this Based on that |
Bug Report
While working on upgrading Typescript to 5.1. from 4.7, I came across an issue with how generics are inferred with JS. Or, potentially not an issue, but a change in how it works that breaks existing code.
Importantly, this issue happens when you have a TS file importing a JS function importing a TS function. Previously, inference worked "good enough" for this scenario to work, but the change breaks it. (Unfortunately, this is a large, older codebase, so we migrate parts to TS as we can! In this case, there's a Typescript utility function, a JS redux selector using the utility, and a Typescript React component using the selector. But the only part that matters is the generic inference and having the JS file.)
🔎 Search Terms
Generics, arguments, JavaScript inference
🕗 Version & Regression Information
⏯ Playground Link
Since this requires JS code in between to TS files, I can't replicate it in the TS playground. However, you can create the three short files I mentioned below and you'll see the issue! That said, here's the working code in full TS. All that needs to change is for
getById
to be written in JS, and the error occurs.💻 Code
Consider this simplified higher order function. Fully implemented, it would be an optimized redux-style selector that only recalculates the selector when the dependents change. Both the selector and dependents have the same signature:
Here's a simple JS file with a new selector using that function:
And here's a TS file where the selector is used:
🙁 Actual behavior
The error is that
getById
expects the incorrect number of arguments. It can accept 2 arguments, but the type inference restricts it to 1. The issue is that in the JS file, the two argument functions have different signatures. Typescript infers theTProps
type based on the second function -- sogetById
is inferred as(state: any) => any
.Interestingly, this is only an issue when the selector is written in JS, not TS. When
getById
is implemented in TS 5.1, it uses the first function to infer the type ofgetById
-- so it becomes(state: State, id: number ) => string
.In other words, generic inference is inconsistent depending on if the function comes from JS or TS, despite most of the type information coming from a TS file in the first place!
I've not been able to find a workaround -- if we change to
Partial< TProps >
for the signature ofgetDependents
, we encounter issues in other selectors where the args are always defined (since Partial means some of those args can be undefined).🙂 Expected behavior
Before the update, Typescript seemingly inferred
TProps
based on both functions -- sogetById
becomes(state: any, id?: any) => any
.Now, I can understand that this situation is a bit tricky, and inferring based on both functions isn't ideal either. But I'm not sure how to express the types better here.
getDependents
will be called with all of the arguments, but it's not strictly necessary for the function body to accept all of those arguments when they might not be used. (And either way, this scenario works in TS, which indicates the basic premise is accepted by TS.)The text was updated successfully, but these errors were encountered: