supercat666's picture
add igv
78c921d
raw
history blame
4.24 kB
// Licensed to the Apache Software Foundation (ASF) under one
// or more contributor license agreements. See the NOTICE file
// distributed with this work for additional information
// regarding copyright ownership. The ASF licenses this file
// to you under the Apache License, Version 2.0 (the
// "License"); you may not use this file except in compliance
// with the License. You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing,
// software distributed under the License is distributed on an
// "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
// KIND, either express or implied. See the License for the
// specific language governing permissions and limitations
// under the License.
import { toUint8Array } from '../../util/buffer.js';
import { ReadableDOMStreamOptions } from '../../io/interfaces.js';
import { isIterable, isAsyncIterable } from '../../util/compat.js';
/** @ignore */
type SourceIterator<T> = Generator<T, void, number | null>;
/** @ignore */
type AsyncSourceIterator<T> = AsyncGenerator<T, void, number | null>;
/** @ignore */
export function toDOMStream<T>(source: Iterable<T> | AsyncIterable<T>, options?: ReadableDOMStreamOptions): ReadableStream<T> {
if (isAsyncIterable<T>(source)) { return asyncIterableAsReadableDOMStream(source, options); }
if (isIterable<T>(source)) { return iterableAsReadableDOMStream(source, options); }
/* istanbul ignore next */
throw new Error(`toDOMStream() must be called with an Iterable or AsyncIterable`);
}
/** @ignore */
function iterableAsReadableDOMStream<T>(source: Iterable<T>, options?: ReadableDOMStreamOptions) {
let it: SourceIterator<T> | null = null;
const bm = (options?.type === 'bytes') || false;
const hwm = options?.highWaterMark || (2 ** 24);
return new ReadableStream<T>({
...options as any,
start(controller) { next(controller, it || (it = source[Symbol.iterator]() as SourceIterator<T>)); },
pull(controller) { it ? (next(controller, it)) : controller.close(); },
cancel() { (it?.return && it.return() || true) && (it = null); }
}, { highWaterMark: bm ? hwm : undefined, ...options });
function next(controller: ReadableStreamDefaultController<T>, it: SourceIterator<T>) {
let buf: Uint8Array;
let r: IteratorResult<T> | null = null;
let size = controller.desiredSize || null;
while (!(r = it.next(bm ? size : null)).done) {
if (ArrayBuffer.isView(r.value) && (buf = toUint8Array(r.value))) {
size != null && bm && (size = size - buf.byteLength + 1);
r.value = <any>buf;
}
controller.enqueue(r.value);
if (size != null && --size <= 0) { return; }
}
controller.close();
}
}
/** @ignore */
function asyncIterableAsReadableDOMStream<T>(source: AsyncIterable<T>, options?: ReadableDOMStreamOptions) {
let it: AsyncSourceIterator<T> | null = null;
const bm = (options?.type === 'bytes') || false;
const hwm = options?.highWaterMark || (2 ** 24);
return new ReadableStream<T>({
...options as any,
async start(controller) { await next(controller, it || (it = source[Symbol.asyncIterator]() as AsyncSourceIterator<T>)); },
async pull(controller) { it ? (await next(controller, it)) : controller.close(); },
async cancel() { (it?.return && await it.return() || true) && (it = null); },
}, { highWaterMark: bm ? hwm : undefined, ...options });
async function next(controller: ReadableStreamDefaultController<T>, it: AsyncSourceIterator<T>) {
let buf: Uint8Array;
let r: IteratorResult<T> | null = null;
let size = controller.desiredSize || null;
while (!(r = await it.next(bm ? size : null)).done) {
if (ArrayBuffer.isView(r.value) && (buf = toUint8Array(r.value))) {
size != null && bm && (size = size - buf.byteLength + 1);
r.value = <any>buf;
}
controller.enqueue(r.value);
if (size != null && --size <= 0) { return; }
}
controller.close();
}
}