-
Notifications
You must be signed in to change notification settings - Fork 13k
Closed
Labels
By DesignDeprecated - use "Working as Intended" or "Design Limitation" insteadDeprecated - use "Working as Intended" or "Design Limitation" insteadFixedA PR has been merged for this issueA PR has been merged for this issue
Description
Currently the following flawed code is permitted:
interface A { x: nuber; }
interface B { x: number, y: string }
function copyB(value: B) : B {
return { x: value.x, y: value.y };
}
let values : A[] = [];
let copied= values.map(copyB); // <-- no problem compiling, but a problem at runtime
Thank to the following line: https://github.com/Microsoft/TypeScript/blob/master/src/compiler/checker.ts#L5654
I suggest to give developers a flag that would disable such unsound behavior allowing function subtyping the way it is supposed to be: https://en.wikipedia.org/wiki/Subtyping#Function_types.
An outline of a pull request of how this might look like: https://github.com/aleksey-bykov/TypeScript/commit/34b8f9b5a937cb7bb471771d5c9b9c4d4f629fc8
Related issues: #5961, #3523, #4895, #5741, #3067, #222, #5673
styfle
Metadata
Metadata
Assignees
Labels
By DesignDeprecated - use "Working as Intended" or "Design Limitation" insteadDeprecated - use "Working as Intended" or "Design Limitation" insteadFixedA PR has been merged for this issueA PR has been merged for this issue
Type
Projects
Milestone
Relationships
Development
Select code repository
Activity
RyanCavanaugh commentedon Dec 14, 2015
I checked out your branch and it has hundreds of warnings in lib.d.ts. It would be good to see a demo branch where this isn't the case.
I cannot see how this would be useful in practice. Consider some normal code:
This errors in your branch:
So the user is supposed to write...
scan(<{}[]>[1, 2]);
? Or something else? It's very confusing that you'd have to write upcasts everywhere you used substitutability of arrays.Have you run this flag on your own code?
zpdDG4gta8XKpMCd commentedon Dec 14, 2015
Sorry for confusion. A blooper was fixed.
More working code: https://github.com/aleksey-bykov/TypeScript/commit/b5ce252342c26ebef2bde66036e58d6c05b0200b
Lib.d.ts compiles no problem.
Your example compiles too.
Here is a real project that is taking advantage of it as described in #4895
Instructions:
RyanCavanaugh commentedon Dec 14, 2015
I don't see what's being accomplished now
Same behavior with the initial example you posted
zpdDG4gta8XKpMCd commentedon Dec 15, 2015
you are right, surprisingly that example does not work, it must be a
different place in the compiler that I have yet to find and fix, which is
confusing because the following example works as expected:
does it mean that there is more than one place in the checker that does
matching? not sure how these 2 examples are different
any insights would be welcome
zpdDG4gta8XKpMCd commentedon Dec 15, 2015
oh crap, you are right, just realized I don't make sense, let me get my shit together
jeffreymorlan commentedon Dec 15, 2015
@Aleksey-Bykov: the check you actually want is the contravariant one:
isRelatedTo(t, s, reportErrors)
I tried checking some code with covariant parameters disallowed, and most errors found were
Subtype[]
not being assignable toSupertype[]
*, due to the.push
method (and similar errors with a few other generic container-ish classes). I think for a --noParameterCovariance flag to be viable we would need something like Java's wildcards, so you could doArray<? extends Supertype>
etc.*EDIT: None of the errors were exactly on two array types, one of them was always an interface extending an array. It looks like generics are always assumed to be covariant themselves, but structurally identical types are not.
RyanCavanaugh commentedon Dec 15, 2015
We chatted about this for a while in the team room.
I was hoping to have a sort of Socratic "But what about ..." inspection of this since I know we've argued about it for a long time, but let me relay some points @ahejlsberg raised that might inform your next direction here.
Your premise here is that it should be invalid to assign
(x: Derived) => void
to(x: Base => void)
, even thoughDerived
is assignable toBase
. Let's chase this to its logical conclusion.First off, the compiler assumes that
X<T>
is assignable toX<U>
ifT
is assignable toU
. We could either keep that assumption, or not.If we keep that assumption, then there is a gaping unsoundness hole in the
--noCovariance
flag:If we remove that assumption, then you get a) a massive perf hit, because all generic types have to be structurally checked and b) terrible semantics:
Second, it's unclear why one should take such umbrage with functions when this problem is omnipresent in the entire type system:
You could argue "this is why
Derived[]
shouldn't be assignable toBase[]
!", but that's not a language anyone wants to use in practice (what's the point of being super-sound if you have to have type assertions everywhere?).I think it's instructive to think about the signatures
contains
andpush
. It's clearly safe to callcontains(Base)
on aDerived[]
(this happens all the time). And it's clearly safe to callpush(Derived)
on aBase[]
(this happens all the time). But it's unsafe to callpush(Base)
on aDerived[]
.It's easy to convince yourself that you can just think about signatures without thinking about the implications on the types they are embedded in, but this is a fallacy. In a language without extensive const and in/out annotations, this is an unsoundness you have to accept.
It should be possible to write a TSLint rule that can detect most of the cases that humans would consider suspicious. But it's not going to be possible to wire this rule through the entire type system and get sane results.
zpdDG4gta8XKpMCd commentedon Dec 15, 2015
To me the most important point you made is that
in
andout
annotations are a must to be able do it right. I assume they are out of the list of TypeScript features. If so, it's the end of story.But to nourish this idea a bit more.
I see no reason why
Derived[]
can not be assigned toBase[]
as long as the set of methods ofBase[]
is shrunken to read only which is a fair thing to ask, isn't it? Whoever wants the dopush
onBase[]
has to take care of immutability.How come it is? Granted, for arrays it will work because they uses reference equality, but in general case, where equality is structural and hardcoded, it is not safe. To make it safe a comparison function should be passed too.
As long as
push
doesn't mutate the array it is safe. Which we know it does, so it's not safe.zpdDG4gta8XKpMCd commentedon Apr 11, 2016
there was a comment earlier, now it's gone.. anyway, i am glad someone brought it back to life
what i seen from my little exercise is that THE ONLY USE CASE for covariant function parameters are the event handlers
@RyanCavanaugh, am i wrong? where?
if so, it looks like such an overkill to allow the covariant parameters just to make stupid event handlers work
Instead, did you think of making event handlers discriminated by the event name WHOLE DIFFERENT METHODS merely making the event type argument value a CONTINUATION of the METHOD NAME?
so instead of
we get (not in syntax, but semantically) something like :
where
addEventListenerClick
is a method that registersclick
event handlers ONLY and has nothing to do with anything else12 remaining items