-
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
Assignability of callback parameter defined with conditional tuple type #26013
Comments
For const sourceCount = getParameterCount(source);
const sourceGenericRestType = getGenericRestType(source);
const targetGenericRestType = sourceGenericRestType ? getGenericRestType(target) : undefined;
if (sourceGenericRestType && !(targetGenericRestType && sourceCount === targetCount)) {
return Ternary.False;
} I'm unsure what this code is supposed to be doing. |
@ahejlsberg was working on a fix for assigning a type to a conditional type if it is assignable to the two branches. |
Unless I'm missing something, the code is bailing out at the point I quoted before it even tests assignability to the conditional type, so something additional will need to be done. |
The conditional type is irrelevant here. The examples can be simplified to just: declare var f1: (...args: any[]) => void;
declare var f2: <T extends any[]>(...args: T) => void;
f1 = f2; // Ok
f2 = f1; // Ok
declare var cb1: (cb: (...args: any[]) => void) => void;
declare var cb2: <T extends any[]>(cb: (...args: T) => void) => void;
cb1 = cb2; // Ok
cb2 = cb1; // Error In the second example, the declare var ff1: () => any[];
declare var ff2: <T extends any[]>() => T;
ff1 = ff2; // Ok
ff2 = ff1; // Error The second assignment fails because @mattmccutchen The code you're looking at is indeed the place we error. It basically says that a generic rest parameter in the source must be matched by a generic rest parameter in the target, because only another generic type could be assignable here (i.e. for a source |
This is inconsistent with the behavior when the rest parameter is a non-generic tuple type. For example: declare var cb1: (cb: (...args: any[]) => void) => void;
declare var cb2: (cb: (...args: [any]) => void) => void;
cb1 = cb2; // Ok
cb2 = cb1; // Ok
declare var ff1: () => any[];
declare var ff2: () => [any];
ff1 = ff2; // Ok
ff2 = ff1; // Error So I think there is a bug here: either |
@mattmccutchen that's because that effectively gets flattened to declare var cb1: (cb: (...args: any[]) => void) => void;
declare var cb2: (cb: (arg0: any) => void) => void;
cb1 = cb2;
cb2 = cb1; and Which is definitely an inconsistency in how tuple types and parameter lists are compared, but it seems unrelated here. The thing is that you really don't know whether |
TypeScript doesn't normally allow that: interface MySuperCoolArray<E> extends Array<E> {
hello: number;
}
// Error: A rest parameter must be of an array type.
declare var cb2: (cb: (...args: MySuperCoolArray<any>) => void) => void; So in principle, when we see: declare var cb2: <T extends any[]>(cb: (...args: T) => void) => void; we should have a constraint that As an aside, this isn't an error on the playground and it probably should be: interface MySuperCoolArray<E> extends Array<E> {
hello: number;
}
declare var cb2: <T extends MySuperCoolArray<any>>(cb: (...args: T) => void) => void; |
This issue has been marked 'Working as Intended' and has seen no recent activity. It has been automatically closed for house-keeping purposes. |
TypeScript Version: 3.1.0-dev.20180727
Code
Expected behavior:
I am able to construct callback signatures with generic conditional types and tuples that are assignable with the maximally general callback signature (
cb1
).Actual behavior:
cb1
cannot be assigned tocb2
, but removing the generic parameter alacb3
makes assignability work as expected.The text was updated successfully, but these errors were encountered: