Skip to content

Commit 2e3e6a2

Browse files
committedJul 3, 2020
Support lazy UnionIterator.
Closes #19
1 parent acd905e commit 2e3e6a2

File tree

2 files changed

+179
-13
lines changed

2 files changed

+179
-13
lines changed
 

‎asynciterator.ts

+37-13
Original file line numberDiff line numberDiff line change
@@ -1418,7 +1418,7 @@ export class MultiTransformIterator<S, D = S> extends TransformIterator<S, D> {
14181418
*/
14191419
export class UnionIterator<T> extends BufferedIterator<T> {
14201420
private _sources : InternalSource<T>[] = [];
1421-
private _sourcesComplete = true;
1421+
private _pending? : { sources?: AsyncIterator<AsyncIterator<T>> };
14221422
private _currentSource = -1;
14231423

14241424
/**
@@ -1427,30 +1427,49 @@ export class UnionIterator<T> extends BufferedIterator<T> {
14271427
@param {object} [options] Settings of the iterator
14281428
*/
14291429
constructor(sources: AsyncIteratorOrArray<AsyncIterator<T>>,
1430-
options?: BufferedIteratorOptions) {
1430+
options: BufferedIteratorOptions = {}) {
14311431
super(options);
1432+
const autoStart = options.autoStart !== false;
14321433

1434+
// Sources have been passed as an iterator
1435+
if (isEventEmitter(sources)) {
1436+
sources.on('error', error => this.emit('error', error));
1437+
this._pending = { sources };
1438+
if (autoStart)
1439+
this._loadSources();
1440+
}
14331441
// Sources have been passed as a non-empty array
1434-
if (Array.isArray(sources) && sources.length > 0) {
1442+
else if (Array.isArray(sources) && sources.length > 0) {
14351443
for (const source of sources)
14361444
this._addSource(source as InternalSource<T>);
14371445
}
1438-
// Sources have been passed as an open iterator
1439-
else if (isEventEmitter(sources) && !sources.done) {
1440-
this._sourcesComplete = false;
1446+
// Sources are an empty list
1447+
else if (autoStart) {
1448+
this.close();
1449+
}
1450+
}
1451+
1452+
// Loads sources passed as an iterator
1453+
protected _loadSources() {
1454+
// Obtain sources iterator
1455+
const sources = this._pending!.sources!;
1456+
delete this._pending!.sources;
1457+
1458+
// Close immediately if done
1459+
if (sources.done) {
1460+
delete this._pending;
1461+
this.close();
1462+
}
1463+
// Otherwise, set up source reading
1464+
else {
14411465
sources.on('data', source => {
14421466
this._addSource(source as InternalSource<T>);
14431467
this._fillBufferAsync();
14441468
});
14451469
sources.on('end', () => {
1446-
this._sourcesComplete = true;
1470+
delete this._pending;
14471471
this._fillBuffer();
14481472
});
1449-
sources.on('error', error => this.emit('error', error));
1450-
}
1451-
// Sources are an empty list
1452-
else {
1453-
this.close();
14541473
}
14551474
}
14561475

@@ -1478,6 +1497,10 @@ export class UnionIterator<T> extends BufferedIterator<T> {
14781497

14791498
// Reads items from the next sources
14801499
protected _read(count: number, done: () => void): void {
1500+
// Start source loading if needed
1501+
if (this._pending?.sources)
1502+
this._loadSources();
1503+
14811504
// Try to read `count` items
14821505
let lastCount = 0, item : T | null;
14831506
while (lastCount !== (lastCount = count)) {
@@ -1493,8 +1516,9 @@ export class UnionIterator<T> extends BufferedIterator<T> {
14931516
}
14941517
}
14951518
}
1519+
14961520
// Close this iterator if all of its sources have been read
1497-
if (this._sourcesComplete && this._sources.length === 0)
1521+
if (!this._pending && this._sources.length === 0)
14981522
this.close();
14991523
done();
15001524
}

‎test/UnionIterator-test.js

+142
Original file line numberDiff line numberDiff line change
@@ -80,6 +80,31 @@ describe('UnionIterator', () => {
8080
});
8181
});
8282

83+
describe('when constructed with an array of 0 sources without autoStart', () => {
84+
let iterator;
85+
before(() => {
86+
const sources = [];
87+
iterator = new UnionIterator(sources, { autoStart: false });
88+
});
89+
90+
describe('before reading', () => {
91+
it('should not have ended', () => {
92+
iterator.ended.should.be.false;
93+
});
94+
});
95+
96+
describe('after reading', () => {
97+
before(done => {
98+
iterator.read();
99+
queueMicrotask(done);
100+
});
101+
102+
it('should have ended', () => {
103+
iterator.ended.should.be.true;
104+
});
105+
});
106+
});
107+
83108
describe('when constructed with an array of 1 source', () => {
84109
let iterator;
85110
before(() => {
@@ -127,6 +152,31 @@ describe('UnionIterator', () => {
127152
});
128153
});
129154

155+
describe('when constructed with an iterator of 0 sources without autoStart', () => {
156+
let iterator;
157+
before(() => {
158+
const sources = [];
159+
iterator = new UnionIterator(new ArrayIterator(sources), { autoStart: false });
160+
});
161+
162+
describe('before reading', () => {
163+
it('should not have ended', () => {
164+
iterator.ended.should.be.false;
165+
});
166+
});
167+
168+
describe('after reading', () => {
169+
before(done => {
170+
iterator.read();
171+
queueMicrotask(done);
172+
});
173+
174+
it('should have ended', () => {
175+
iterator.ended.should.be.true;
176+
});
177+
});
178+
});
179+
130180
describe('when constructed with an iterator of 1 source', () => {
131181
let iterator;
132182
before(() => {
@@ -166,6 +216,98 @@ describe('UnionIterator', () => {
166216
});
167217
});
168218

219+
describe('when constructed with an iterator and with autoStart', () => {
220+
let iterator, sourceIterator;
221+
before(() => {
222+
const sources = [range(0, 2), range(3, 6)];
223+
sourceIterator = new ArrayIterator(sources);
224+
sinon.spy(sourceIterator, 'read');
225+
iterator = new UnionIterator(sourceIterator, { autoStart: true });
226+
});
227+
228+
describe('before reading', () => {
229+
it('should have read the sources', () => {
230+
sourceIterator.read.should.have.been.called;
231+
});
232+
233+
it('should not have ended', () => {
234+
iterator.ended.should.be.false;
235+
});
236+
237+
it('should pass errors', () => {
238+
const callback = sinon.spy();
239+
const error = new Error('error');
240+
iterator.once('error', callback);
241+
sourceIterator.emit('error', error);
242+
callback.should.have.been.calledOnce;
243+
callback.should.have.been.calledWith(error);
244+
});
245+
});
246+
247+
describe('after reading', () => {
248+
let items;
249+
before(async () => {
250+
items = (await toArray(iterator)).sort();
251+
});
252+
253+
it('should have emitted all items', () => {
254+
items.should.eql([0, 1, 2, 3, 4, 5, 6]);
255+
});
256+
257+
it('should have ended', () => {
258+
iterator.ended.should.be.true;
259+
});
260+
});
261+
});
262+
263+
describe('when constructed with an iterator and without autoStart', () => {
264+
let iterator, sourceIterator;
265+
before(() => {
266+
const sources = [range(0, 2), range(3, 6)];
267+
sourceIterator = new ArrayIterator(sources);
268+
sinon.spy(sourceIterator, 'read');
269+
iterator = new UnionIterator(sourceIterator, { autoStart: false });
270+
});
271+
272+
describe('before reading', () => {
273+
it('should not have read the sources', () => {
274+
sourceIterator.read.should.not.have.been.called;
275+
});
276+
277+
it('should not have ended', () => {
278+
iterator.ended.should.be.false;
279+
});
280+
281+
it('should pass errors', () => {
282+
const callback = sinon.spy();
283+
const error = new Error('error');
284+
iterator.once('error', callback);
285+
sourceIterator.emit('error', error);
286+
callback.should.have.been.calledOnce;
287+
callback.should.have.been.calledWith(error);
288+
});
289+
});
290+
291+
describe('after reading', () => {
292+
let items;
293+
before(async () => {
294+
items = (await toArray(iterator)).sort();
295+
});
296+
297+
it('should have read the sources', () => {
298+
sourceIterator.read.should.have.been.called;
299+
});
300+
301+
it('should have emitted all items', () => {
302+
items.should.eql([0, 1, 2, 3, 4, 5, 6]);
303+
});
304+
305+
it('should have ended', () => {
306+
iterator.ended.should.be.true;
307+
});
308+
});
309+
});
310+
169311
describe('a UnionIterator with two sources', () => {
170312
let iterator, sources;
171313

0 commit comments

Comments
 (0)
Please sign in to comment.