-
Notifications
You must be signed in to change notification settings - Fork 12.8k
New issue
Have a question about this project? # for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “#”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? # to your account
In mapped types with key remapping, any property involving a generic type argument is dropped. #53357
Comments
I bet that your real-life example looks quite different and I'm not sure if that would be fixable. This simple case could potentially be fixed but this conditional type stays deferred in the generic context. I'm not exactly sure why but there are some tests with The potential fix that I had in mind for this is pretty easy: diff --git a/src/compiler/checker.ts b/src/compiler/checker.ts
index b975b98488..0a781288bd 100644
--- a/src/compiler/checker.ts
+++ b/src/compiler/checker.ts
@@ -17780,7 +17780,7 @@ export function createTypeChecker(host: TypeCheckerHost): TypeChecker {
// Instantiate the extends type including inferences for 'infer T' type parameters
const inferredExtendsType = combinedMapper ? instantiateType(root.extendsType, combinedMapper) : extendsType;
// We attempt to resolve the conditional type only when the check and extends types are non-generic
- if (!checkTypeDeferred && !isDeferredType(inferredExtendsType, checkTuples)) {
+ if (checkType === inferredExtendsType || !checkTypeDeferred && !isDeferredType(inferredExtendsType, checkTuples)) {
// Return falseType for a definitely false extends check. We check an instantiations of the two
// types with type parameters mapped to the wildcard type, the most permissive instantiations
// possible (the wildcard type is assignable to and from all types). If those are not related, |
Do you think the problem is in conditional types? If I test the equivalent Noop without the
This comment might give us a hint:
So maybe it's that I'm not sure about my terminology here, so please correct me if what I'm saying makes no sense... |
Yes, that's exactly what happens. This conditional type stays deferred and thus it fails |
Is there a reason why key remapping should drop deferred types, then, or is it just incomplete? |
How do you access a property name that is typed using a deferred type? This conditional type decides if that property got filtered out from the type or not. If TS can't resolve it immediately then it can't assume that it's there so it becomes inaccessible. |
That makes sense! So in other words, "How can you name a property based on a type that hasn't been resolved?" In the cases I provide, though, the only possible outcomes are Maybe typescript should attempt to resolve the conditional before evaluating the property name? That way it would be able to tell whether or not it can name the property, even if the type contains generic arguments. |
I think that it's best to visualize this outcome as equivalent to: type Preserved = { a: 1, b: 2 }
type Filtered = { a: 1 }
type Both = Preserved | Filtered
type Ok = Both['a']
type Err = Both['b'] Conceptually a very similar thing happens, at the generic level if your property name stays deferred it can end up being both ( |
But in the example I provide:
There isn't anything indeterminate emitted... the conditional type's output is always P. If typescript tried to resolve this, it would know that the fact that T[P] is generic doesn't matter - and know enough to leave the property included. Instead it doesn't even attempt it... it just sees a deferred type and lazily (in the sense of little or no effort) drops it. |
This is what I mentioned in one of the previous comments (here). I'm not sure why this particular type has to be deferred. |
Ahhh... okay a little more context then... I came across the problem trying to filter our properties that have been explicitly marked as
In the above, I expected So I tried a different approach, using a branded type to mark properties I wanted dropped:
In this approach, I flipped the
That's what led me to try a noop conditional where both branches return The same is not true in a simple Identity mapped type, where
So I thought this inconsistency might be a bug! Edited to emphasise that in the last case |
The inconsistency is an intentional difference. You're one of the lucky 10,000 today to learn about homomorphic mapped types - see #26063 |
@RyanCavanaugh do you happen to know why |
I don't quite see the link between what I'm pointing out and homomorphic mapped types, @RyanCavanaugh do you think you could explain? Is it that the I don't understand why In fact, in the noop key remapping, properties whose type are explicitly e.g.
In the above:
But in case 3 it's dropped...? What am I missing here? P.S. I've actually noticed this in many cases that the possibility of a generic type being assigned |
I think the example given is degenerate to the point that it's not useful to discuss. If you write |
I mean, I was only using the example to illustrate a broader issue... which was the inability to pass generic types through any kind of key remapping. A more concrete example would be to filter out properties that are of type
But then I noticed that the I thought the whole point of |
I think that the current logic is fine and that it's very consistent with the rest of TypeScript. As I mentioned, given that this object depends on a generic it's essentially equivalent within this method to smth like: type Args = { bar: Date, value: Z } | { bar: Date } TypeScript doesn't allow you to access properties that are members of some union constituents - you can only access common keys. This is exactly how it works in concrete scenarios: type Concrete = { bar: Date, value: number } | { bar: Date }
declare const c: Concrete
// Property 'value' does not exist on type 'Concrete'.
// Property 'value' does not exist on type '{ bar: Date; }'
c.value
Similar limitations are quite common in TypeScript. As long as things stay generic the reasoning capabilities are limited - often for good reasons though (and I think that this one is a good reason). I think that perhaps what you would like to ask for here is the ability to narrow this implicit union using type FilterNeverProps<T> = {
[P in keyof T as T[P] extends never ? never : P]: T[P];
};
class Test<Z> {
foo(args: FilterNeverProps<{ key: never; value: Z; bar: Date }>) {
if ("value" in args) {
args.value; // error but it wouldn't have to be
}
}
} |
Yeah, actually maybe this is a better way to think about it... So that if But in practice when am I ever going to pass E.g. I would be looking for something like:
Can't do that though because never is assignable to everything... I really appreciate the patience and discussion, by the way, thank you for sticking with me! |
E.g. even if I constrain
Or am I still misunderstanding something? |
That's just caused by what you have already mentioned - |
Yep, got that - I suppose my point is that maybe there should be a way to escape the |
This issue has been marked 'Working as Intended' and has seen no recent activity. It has been automatically closed for house-keeping purposes. |
Bug Report
When performing key remapping in mapped types, any property involving a generic type argument is dropped from the resulting type.
🔎 Search Terms
Mapped types with generic properties
Key remapping with generic properties
Properties with generic type dropped from mapped type with 'as' key remapping
🕗 Version & Regression Information
I first noticed it in
4.9.5
and still exhibited innightly
(5.1.*
). Tried every version in playground down to4.1.*
, same behavior.This is the behavior in every version I tried, and I reviewed the FAQ for entries about
Type System Behavior
,Classes
,Generics
.⏯ Playground Link
Playground link with relevant code
💻 Code
🙁 Actual behavior
value
is omitted from the resulting type, even though all propertiesP
are emitted. Any property that has a type involving a generic type parameter always gets dropped from the key remapping.🙂 Expected behavior
value
to be on the resulting type asZ
, since its key would be emitted and type looked up byT[P]
.The text was updated successfully, but these errors were encountered: