Add --strictObjectIterables
(bikeshed): exclude primitive (string
) from Iterable
typesΒ #59676
Description
π Search Terms
"iterable object", "string"
β Viability Checklist
- This wouldn't be a breaking change in existing TypeScript/JavaScript code
- This wouldn't change the runtime behavior of existing JavaScript code
- This could be implemented without emitting different JS based on the types of the expressions
- This isn't a runtime feature (e.g. library functionality, non-ECMAScript syntax with JavaScript output, new syntax sugar for JS, etc.)
- This isn't a request to add a new utility type: https://github.com/microsoft/TypeScript/wiki/No-New-Utility-Types
- This feature would agree with the rest of our Design Goals: https://github.com/Microsoft/TypeScript/wiki/TypeScript-Design-Goals
β Suggestion
At the TC39 meeting in 2024-07, it was decided that Iterable
expects objects.
Reject primitives in iterable-taking positions
Any time an iterable or async-iterable value (a value that has a
Symbol.iterator
orSymbol.asyncIterator
method) is expected, primitives should be treated as if they were not iterable. Usually, this will mean throwing aTypeError
. If the user provides a primitive wrapper Object such as a String Object, however, it should be treated like any other Object.Although primitive Strings are default iterable (
String.prototype
has aSymbol.iterator
method which enumerates code points), it is now considered a mistake to iterate a String without specifying whether the String is providing an abstraction over code units, code points, grapheme clusters, or something else.NB: This convention is new as of 2024, and most earlier parts of the language do not follow it. In particular, positional destructuring (both binding and assignment), array spread, argument spread, for-of loops,
yield *
, theSet
andAggregateError
constructors,Object.groupBy
,Map.groupBy
,Promise.all
,Promise.allSettled
,Promise.any
,Promise.race
,Array.from
, the staticfrom
methods on typed array constructors, andIterator.from
(Stage 3 at time of writing) all accept primitives where iterables are expected.
To follow this decision, the Iterable
and string
types should be separated, and APIs that accept both types should be required to explicitly specify Iterable<any> | string
. Since this would be a breaking change, how about adding a new --strictObjectIterables
(bikeshed) option?
In practice, the upcoming ReadableStream.from
will accept Iterable<any>
and AsyncIterable<any>
, but will be restricted to objects. whatwg/streams#1310
π Motivating Example
Enabling the --strictObjectIterables
option raises a type error in the following example:
const iterable: Iterable<any> = "string";
π» Use Cases
-
What do you want to use this for?
Used for
ReadableStream.from
and other APIs to be added in the future. -
What shortcomings exist with current approaches?
Iterable
object andstring
are not distinguished by default. -
What workarounds are you using in the meantime?
Maybe
Iterable<any> & object
can rejectstring
.
Activity