TypeScript: PSA: potential lib breaking change after iterator-helpers proposal
lib Update Request
Configuration Check
My compilation target is ESNext and my lib is the default.
Missing / Incorrect Definition
The TS lib defs for Iterator, Iterable, and IterableIterator treat them as if they are all fictitious interfaces (a.k.a. protocols). The lib def’s inheritance structure looks like this:
interface Iterator<T, TReturn = any, TNext = undefined> {
// NOTE: 'next' is defined using a tuple to ensure we report the correct assignability errors in all places.
next(...args: [] | [TNext]): IteratorResult<T, TReturn>;
return?(value?: TReturn): IteratorResult<T, TReturn>;
throw?(e?: any): IteratorResult<T, TReturn>;
}
interface Iterable<T> {
[Symbol.iterator](): Iterator<T>;
}
interface IterableIterator<T> extends Iterator<T> {
[Symbol.iterator](): IterableIterator<T>;
}
However, while Iterable and arguably IterableIterator are protocols, Iterator is an actual JS entity. See MDN docs: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Iterator This has some practical issues, because what TS terms as IterableIterator is the actual JS Iterator class, but what TS terms as Iterator is what JS regards as “iterator-like” or “iterator protocol”. For example, the def for new Set().values() says it returns an IterableIterator, while in fact it should return an instance of the JS Iterator instance.
This becomes more of an issue after the iterator-helpers proposal, which is going to expose Iterator as an actual global that people can use as extends. For example:
class MyClass {
static #MyClassIterator = class extends Iterator {
next() { ... }
};
[Symbol.iterator]() {
return new MyClass.#MyClassIterator();
}
}
Under the current lib def, #MyClassIterator won’t be seen as iterable by TS, because it extends Iterator, not IterableIterator.
On the other hand, if we change the definition of Iterator to say it has [Symbol.iterator] (mirroring what happens in JS), then it will break code of the following kind:
function stepIterator(it: Iterator) {
return it.next();
}
stepIterator({ next() {} });
Of course you could assume that almost all userland iterators are built-in and therefore inherit from Iterator and have @@iterator, but it’s a breaking change nonetheless.
The question thus arises that when iterator-helpers lands, where should we put the new definitions on. Right now, I’m using core-js to polyfill these methods, and when writing declarations, I chose to put them on IterableIterator, so that new Set().values() works:
declare global {
interface IterableIterator<T> {
filter(
predicate: (value: T, index: number) => unknown,
): IterableIterator<T>;
map<U>(mapper: (value: T, index: number) => U): IterableIterator<U>;
toArray(): T[];
}
}
…but this is suboptimal, for obvious reasons (it means the Iterator class doesn’t actually have these methods).
Sample Code
Documentation Link
About this issue
- Original URL
- State: open
- Created a year ago
- Reactions: 7
- Comments: 15 (14 by maintainers)
A concrete proposal to maybe get the ball rolling:
Iteratorinterface toIteratorLike(or some other better name). Change all internal uses ofIteratorto refer toIteratorLikeinstead, but no need to renameIterableIteratoror anything else.abstract class Iterator { ... }modeling the actualIteratorclass in the language.This is a breaking change for anyone using the
Iteratortype explicitly, but hopefully it’s not that many - there’s not much reason to explicitly use theIteratortype right now. There are only 176 files in DefinitelyTyped which useIterator<right now, and manual inspection reveals that more than half of those aren’t referring to the TSIteratortype (e.g. x, x, x).The main way that this would break is that places which declared to need an
Iterator<T>would start giving an error if provided with{ next() {...} }, which previously worked. There do not appear to be that many cases of this. Also, types in dependencies which declare themselves to provideIterator<T>and give{ next() {...} }would appear to have.mapetc, when in fact that is missing - this gives you the wrong typing, but not in a way which causes errors for existing code.Now that iterator helpers are shipped in Chrome and Edge, meaning it will also ship in the next NodeJS release this month (and there are polyfills available too for other engines that currently have it behind experimental flags) it would be really nice if TypeScript could tackle typing the iterator helpers. Given the current naming conflict, it is difficult to write custom typings to fill in the gap in userland (and I haven’t found any successful attempts in DT or polyfill packages).
IterableIteratoris possibly the best place for them, but in the event that is too much of a break we may introduce aNativeIterator<T>orBuiltinIterator<T>interface to avoid breaks.The biggest issue I see is that we can’t properly model the type of the native
Iteratorconstructor without usingclass, because the nativeIteratorclass from the iterator helpers proposal is essentiallyabstractbecause you must still definenext:While we can model the abstract constructor using an
abstract newconstructor type, theabstract newsyntax isn’t supported on construct signatures in an interface. We also have no way to indicate anabstractmethod in an interface at the moment. If we did, a non-classversion might look like this:I may have a solution to use
classwithout polluting the global scope. I’ll test it out tomorrow.I’m confused; Iterator.prototype.next exists, why would you need to define it yourself?