Skip to content
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

Support for reverse mapped types with intersection constraint #55927

Closed
5 tasks done
jfet97 opened this issue Sep 30, 2023 · 1 comment Β· Fixed by #55811
Closed
5 tasks done

Support for reverse mapped types with intersection constraint #55927

jfet97 opened this issue Sep 30, 2023 · 1 comment Β· Fixed by #55811
Labels
Help Wanted You can do this Possible Improvement The current behavior isn't wrong, but it's possible to see that it might be better in some cases
Milestone

Comments

@jfet97
Copy link
Contributor

jfet97 commented Sep 30, 2023

πŸ” Search Terms

"reverse mapped types", "intersection"

βœ… Viability Checklist

  • This wouldn't be a breaking change in existing TypeScript/JavaScript code
  • This wouldn't change the runtime behavior of existing JavaScript code
  • This could be implemented without emitting different JS based on the types of the expressions
  • This isn't a runtime feature (e.g. library functionality, non-ECMAScript syntax with JavaScript output, new syntax sugar for JS, etc.)
  • This feature would agree with the rest of our Design Goals: https://github.com/Microsoft/TypeScript/wiki/TypeScript-Design-Goals

⭐ Suggestion

I'd like to see reverse mapped types capabilities enhanced to support an intersection constraint, providing users with the ability to enable EPC (Excess Property Checking) on type parameters inference sites.

Reverse mapped types are a powerful tool. Currently TypeScript is able to reverse three kind of mapped types:

  1. a homomorphic mapped type { [P in keyof T]: X }
  2. a mapped type { [P in K]: X }, where K is a type parameter
  3. a mapped type with an union constraint, as long as the union contains one of the constraints of the previous cases

This feature request asks support for a fourth case: a mapped type with an intersection constraint, as long as the intersection contains one of the constraints of the previous cases.

While the union constraint could be seen as a way to force the presence of some properties, the intersection one could be seen as a way to prevent the presence of extra properties.

May be closed by #55811.

πŸ“ƒ Motivating Example

In this example, taken from #12936, the type parameter U is the one that gets inferred by reversing the mapped type that has an intersection constraint: keyof U & keyof T. It should be inferred as { x: 1; y: "y"; z: string } or, even better, as { x: 1; y: "y"; } (with z stripped away because it wouldn't get through the application of the mapped type anyway). Either way, the application of the mapped type on the just inferred U imposes the absence of extra properties with respect to the ones declared in T = {x: number, y: string}.

const checkType = <T>() => <U extends T>(value: { [K in keyof U & keyof T]: U[K] }) => value;

const checked = checkType<{x: number, y: string}>()({
  x: 1,
  y: "y",
  z: "z", // <-- Object literal may only specify known properties, and 'z' does not exist in type '{ x: 1; y: "y"; }'.
});

πŸ’» Use Cases

  1. What do you want to use this for?
    It's a way to enable EPC on type parameters when it's really needed. Motivating comment.

  2. What shortcomings exist with current approaches/workarounds?
    One of the main workaround is the use of type functions like the following:

type NoExtra<T> = {
  [K in keyof T]: K extends keyof MyRefType ? T[K] : never
}

But then you get error messages like Type 'number' is not assignable to type 'never' that are bad for DX, expecially if compared with EPC ones.

What's good about this proposal, and the related PR, is the fact that it lets the user enable EPC on type parameters without fundamentally change how TS treats type parameters inference. It's just a nice side effect of enhancing reverse mapped types' capabilities.

@Andarist
Copy link
Contributor

It's also worth noting here that there is a strong relation between this and the other PR that - sort of - makes a very similar thing possible: #52972

I think that both can co-exist because one is about non-homomorphic mapped types and the other one is about homomorphic mapped types.

@jfet97 jfet97 changed the title Support for reverse mapped type with intersection constraint Support for reverse mapped types with intersection constraint Sep 30, 2023
@RyanCavanaugh RyanCavanaugh added Help Wanted You can do this Possible Improvement The current behavior isn't wrong, but it's possible to see that it might be better in some cases labels Oct 5, 2023
@RyanCavanaugh RyanCavanaugh added this to the Backlog milestone Oct 5, 2023
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
Help Wanted You can do this Possible Improvement The current behavior isn't wrong, but it's possible to see that it might be better in some cases
Projects
None yet
Development

Successfully merging a pull request may close this issue.

3 participants