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

tuple type don't check number of elements #5259

Closed
co-dh opened this issue Oct 14, 2015 · 16 comments
Closed

tuple type don't check number of elements #5259

co-dh opened this issue Oct 14, 2015 · 16 comments
Assignees
Labels
Bug A bug in TypeScript By Design Deprecated - use "Working as Intended" or "Design Limitation" instead

Comments

@co-dh
Copy link

co-dh commented Oct 14, 2015

The following compiles,

    var foo:[string, string] = ['','', '', '']

while this one not:

    var foo:[string, string] = ['','', '', 1]

The first one should not compiles, since the initializer has 3 elements while the type says 2.

@sandersn
Copy link
Member

Tuples are basically objects with numeric properties. So foo: [string, string] is like saying foo: {0: string, 1: string}. The value you assign to foo has an extra property '2', but that's OK in Typescript.

The second doesn't compile for a different reason, which is just that the array ['', '', '', 1] is typed as (number | string)[], and number is not compatible with any of the types in [string, string] -- even though the number doesn't occur until the third element.

The first reason basically makes sense to me, although it would be nice if Typescript had better checking here.

The second reason is a side effect of using arrays to represent tuples, and is a bit unfortunate. For example, this doesn't compile either:

var bar: (number | string)[] = [1, 2, ''];
var foo: [number, number, string] = bar; // (number | string)[] is not assignable to [number, number, string]

var baz: [number, number, string] = [1, 2, '']; // OK!

@seanfreiburg
Copy link

Second doesn't compile because 1 isn't a string. You gave it a type of [string,string].

@mhegazy mhegazy added the Question An issue which isn't directly actionable in code label Oct 16, 2015
@sandersn
Copy link
Member

According to the spec section 3.3.3, the type [number, string] is equivalent to an interface that extends Array:

interface NSPair extends Array<number | string> { 0: number, 1: string }

The unfortunate side effect of this is that extra elements in a tuple assignment are ignored, but only if they are of a type that is already present in the tuple. That's because [1, "foo", 2]: (number | string)[], which NSPair extends, but [1, "foo", false]: (number | string | boolean)[], which it does not.

I think this is a bug in the spec, but I'm not at all sure how to fix it. Maybe an ad-hoc rule that says the lengths have to match, or maybe there's an existing way to fix it in the language.

@sandersn sandersn added the Bug A bug in TypeScript label Oct 16, 2015
@mhegazy mhegazy removed the Question An issue which isn't directly actionable in code label Oct 28, 2015
@JsonFreeman
Copy link
Contributor

Tuples are already quite unsafe because you can do the following:

var tuple: [number, string] = [0, ""];
for (var i in tuple) {
    tuple[i] = ""; // Succeeds because string is assignable to number | string
}
tuple[0]; // type is number, but should be string

I think the goal of giving the user the right type needs to be relaxed for tuples, relative to other parts of the language. On the one hand, there are no restrictions on what users can do with tuples, but on the other hand, users may make strong assumptions about what the tuple will contain at each index (that's what makes it a tuple). This disconnect between expectation and reality should probably be addressed more broadly if any changes to the type checking rules for tuples are considered.

@ahejlsberg
Copy link
Member

This isn't a bug, it is working as designed. Our view is that tuple is an array for which we have static knowledge of the individual types of the first N elements and for which the element type is the union of the types of those N elements. We allow more than N elements to be present (since there's really not much we can do to prohibit it), but we do require those elements to have a compatible type.

@ahejlsberg ahejlsberg added the By Design Deprecated - use "Working as Intended" or "Design Limitation" instead label Dec 8, 2015
@mortb
Copy link

mortb commented Jun 28, 2016

Must say I feel a bit sad about this.

I felt a bit excited when I read about tuples in the typescript handbook:
https://www.typescriptlang.org/docs/handbook/basic-types.html#tuple
When realizing that there is no checking on the number of array elements the tuple feature doesn't seem as attractive as it did in my first impression.
After a second read in the typescript handbook I see that this limitation it is mentioned:

When accessing an element outside the set of known indices, a union type is used instead

But I was expecting something to use like C# tuples: (dotnet/roslyn#347)
Oh well...

@aboglioli
Copy link

Tuples should have a fixed size or number of elements. That's a property:
Properties of tuples

@RyanCavanaugh
Copy link
Member

Identical is not the same as subtype.

@wclr
Copy link

wclr commented Apr 7, 2017

So there is no way any way to fixate number of elements of the tuple or list in typing? To me this is a big disadvantage.

@RyanCavanaugh
Copy link
Member

type fixed = [number, number] & { 2: void };
// Error
var x: fixed = [3, 3, 3];

@wclr
Copy link

wclr commented Apr 7, 2017

but it doesn't work with 2 either:

type fixed = [number, number] & { 2: void };
// Error
var x: fixed = [3, 3];

Type '[number, number]' is not assignable to type 'fixed'.
Type '[number, number]' is not assignable to type '{ 2: void; }'.
Property '2' is missing in type '[number, number]'.

@RyanCavanaugh
Copy link
Member

Sorry

type fixed = [number, number] & { 2?: void };
// Error
var x: fixed = [3, 3, 3];
// OK
var y: fixed = [3, 3];

@wclr
Copy link

wclr commented Apr 7, 2017

@RyanCavanaugh cool, thank you! Btw when was this feature added?

@RyanCavanaugh
Copy link
Member

That's worked for as long as intersection types (or tuples? whichever came first) have existed

@arnauorriols
Copy link

arnauorriols commented Nov 16, 2017

This isn't a bug, it is working as designed. Our view is that tuple is an array for which we have static knowledge of the individual types of the first N elements and for which the element type is the union of the types of those N elements. We allow more than N elements to be present (since there's really not much we can do to prohibit it), but we do require those elements to have a compatible type.

This is the boldest "it's not a bug, it's a feature" case I've seen in years. Respect.

We allow more than N elements to be present (since there's really not much we can do to prohibit it), but we do require those elements to have a compatible type.

Maybe it's not simple to prohibit the number of items a tuple contains, but what about reading the tuple? Could be prohibited using an index larger than the tuple size?

const t: [number, number] = [1, 2, 3];  // Works, apparently not possible to prohibit consistently
t[1];  // works
t[2];  // Error, [number, number] only has 2 elements

Regarding prohibiting the number of elements a tuple has, I imagine you say it's not possible because of the mutable nature of the arrays in javascript. However, the majority of languages that contain the tuple data structure define them as immutable, even in those languages prone to mutability, like Python. Have you considered making tuples immutable? That could be done easily, and would allow typescript to stop saying that a tuple has unlimited size by design...

@RyanCavanaugh
Copy link
Member

@arnauorriols see #17765

@microsoft microsoft locked and limited conversation to collaborators Jun 19, 2018
Sign up for free to subscribe to this conversation on GitHub. Already have an account? Sign in.
Labels
Bug A bug in TypeScript By Design Deprecated - use "Working as Intended" or "Design Limitation" instead
Projects
None yet
Development

No branches or pull requests