-
Notifications
You must be signed in to change notification settings - Fork 674
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
Inlining wrappers over createSelector causes selector args to be of any type #559
Comments
Huh. I'll be honest, I have no idea what's going on here and am not really even sure how to start debugging this one :) (This goes along with my Twitter comment the other day that "what we really need is a types-level debugger that would tell us how TS is analyzing and transforming types at each step in the process".) Given that there seem to be workarounds and that I've got other stuff going on, I'm afraid this one is going to be relatively low priority, and I'm doubtful at first glance that I'd even be able to figure out what's happening or what a solution is. I may toss it out on Twitter or something and see if anyone else has a clue what's going on here. |
Thanks for the response! It's not urgent at all, the workarounds work fine for now.
It would be awesome to see that. |
Yeah. If you look at the commit/issue history of this repo over the last few weeks, you can see that I (and a few others I've dragged in) have been iterating on the types repeatedly. Every time we fix one issue, something else pops up. I feel pretty good about the types in 4.1.5, tbh - they pass all the typetest scenarios we've got (and I've added a bunch of additional scenarios based on these issues), and based on the thought and design we've put into how they work, they seem to be doing what they should logically in combining parameters. That process would have been much easier if I'd had better ways to inspect the intermediate steps of the types, though :) |
Potentially the same issue also bites me on the other end — in the composed selector calculation, here's a small sample: interface Foo<A> {
foo: A
}
// some selector
function foos(s: RootState) {
return [] as Foo<string>[]
}
// extract loses type information when used with createSelector
function extract<A>(items: Foo<A>[]) {
return items[0].foo
}
const foosDerived = createSelector(foos, extract)
// This type is unknown when it should be a Foo
const shouldBeFooButIsUnknown = foosDerived(null as any) I believe we see this with both TS 4.4.2 and 4.5.2, hit it when upgrading from reselect 3.0.0 to 4.1.5 |
@garrettm what is the inferred type of Also, what happens if you call |
@markerikson here's a more full repro: import {createSelector} from 'reselect'
interface Foo<A = unknown> {
bar: A
}
function values<A>(_: {[k: string]: A}): A[] {
const result: A[] = []
const ks = Object.keys(_)
for (const key of ks) {
result.push(_[key])
}
return result
}
interface State {
foos: {[k: string]: Foo<string>}
}
const state: State = {foos: {}}
const foos = (s: State) => s.foos
const foosList = createSelector(foos, values)
const thisShouldBeFoos: Foo[] = foosList(state)
const foosListWorking = createSelector(foos, f => values(f))
const thisIsFoos: Foo[] = foosListWorking(state) in this example, I see this error: typescript$ tsc inference.ts --target es2017 --moduleResolution node
inference.ts:25:7 - error TS2322: Type 'unknown[]' is not assignable to type 'Foo<unknown>[]'.
Property 'bar' is missing in type '{}' but required in type 'Foo<unknown>'.
25 const thisShouldBeFoos: Foo[] = foosList(state)
~~~~~~~~~~~~~~~~
inference.ts:4:3
4 bar: A
~~~
'bar' is declared here.
Found 1 error. Even though both calls are identical, there's just an extra function call in the working one. I see |
@garrettm fwiw, while you're definitely seeing something, it looks to be a very different problem than what was originally described in this issue. Pasting that example into the TS playground, the type of const foosList: (state: {
foos: {
[k: string]: Foo<string>;
};
}) => unknown[] So, somehow the final return type is the problem here, and I'm not sure where it's going wrong in the middle. |
Think this is the result of using |
Huh. That... sounds very plausible, yeah. |
I don't think this is an issue anymore as the types were simplified and tested prior to v5. |
When upgrading from 4.0.0 to 4.1.5, our selectors started to complain about
any
types when we inline ourcreateSelector
wrapper:TS7006: Parameter 'featureA' implicitly has an 'any' type.
TS7006: Parameter 'featureB' implicitly has an 'any' type.
TypeScript version is 4.4.2.
CodeSandbox example
Here's a code sandbox example of the issue (line 66):
https://codesandbox.io/s/reselect-typescript-issues-xsgeq?file=/src/App.tsx
Description
Code example:
Here
createPreferenceSelector
is our helper to build typed preference selectors. It's included into code sandbox example:Extracting preference selector into its own variable solves the issue:
Removing the helper entirely also solves the issue:
While it didn't look like a huge issue, it made me very curious about why inlining caused that issue. For me it looks like all three examples should work just fine.
After some testing it looks like this issue was introduced in 4.1.0.
The text was updated successfully, but these errors were encountered: