-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathiter.ts
64 lines (58 loc) · 1.47 KB
/
iter.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
export function* iterable<T>(iter: Iterator<T>) {
while (true) {
const { value, done } = iter.next();
if (done) break;
yield value;
}
}
export function* take<T>(count: number, iter: Iterable<T>): Iterable<T> {
const it = iter[Symbol.iterator]();
for (let i = 0; i < count; ++i) {
const { value, done } = it.next();
if (done) break;
yield value;
}
}
export function* zip<T>(...iters: Iterable<T>[]): Iterable<T> {
for (const iter of iters) yield* iter;
}
export function* map<T, U>(iter: Iterable<T>, f: (item: T) => U): Iterable<U> {
for (const item of iter) yield f(item);
}
export function* chunks<T>(
chunkSize: number,
iter: Iterable<T>,
): Iterable<T[]> {
let chunk = [];
for (const item of iter) {
chunk.push(item);
if (chunk.length >= chunkSize) {
yield chunk;
chunk = [];
}
}
if (chunk.length > 0) yield chunk;
}
export function readableFromIterable<T>(
iter: Iterable<T> | AsyncIterable<T>,
): ReadableStream<T> {
const it =
(Symbol.iterator in iter
? iter[Symbol.iterator]()
: iter[Symbol.asyncIterator]());
return new ReadableStream({
pull: async (controller) => {
for (let i = 0; i < (controller.desiredSize ?? 1); ++i) {
const { value, done } = await it.next();
if (done) {
controller.close();
} else {
controller.enqueue(value);
}
}
},
cancel: async (reason) => {
await it.throw?.(reason);
},
});
}