mirror of
https://github.com/denoland/std.git
synced 2024-11-21 20:50:22 +00:00
d102a10235
* refactor: import from `@std/assert` * update
111 lines
3.5 KiB
TypeScript
111 lines
3.5 KiB
TypeScript
// Copyright 2018-2024 the Deno authors. All rights reserved. MIT license.
|
|
// This module is browser compatible.
|
|
|
|
/**
|
|
* Convert the generator function into a {@linkcode TransformStream}.
|
|
*
|
|
* @typeparam I The type of the chunks in the source stream.
|
|
* @typeparam O The type of the chunks in the transformed stream.
|
|
* @param transformer A function to transform.
|
|
* @param writableStrategy An object that optionally defines a queuing strategy for the stream.
|
|
* @param readableStrategy An object that optionally defines a queuing strategy for the stream.
|
|
* @returns A {@linkcode TransformStream} that transforms the source stream as defined by the provided transformer.
|
|
*
|
|
* @example Build a transform stream that multiplies each value by 100
|
|
* ```ts
|
|
* import { toTransformStream } from "@std/streams/to-transform-stream";
|
|
* import { assertEquals } from "@std/assert";
|
|
*
|
|
* const stream = ReadableStream.from([0, 1, 2])
|
|
* .pipeThrough(toTransformStream(async function* (src) {
|
|
* for await (const chunk of src) {
|
|
* yield chunk * 100;
|
|
* }
|
|
* }));
|
|
*
|
|
* assertEquals(
|
|
* await Array.fromAsync(stream),
|
|
* [0, 100, 200],
|
|
* );
|
|
* ```
|
|
*
|
|
* @example JSON Lines
|
|
* ```ts
|
|
* import { TextLineStream } from "@std/streams/text-line-stream";
|
|
* import { toTransformStream } from "@std/streams/to-transform-stream";
|
|
* import { assertEquals } from "@std/assert";
|
|
*
|
|
* const stream = ReadableStream.from([
|
|
* '{"name": "Alice", "age": ',
|
|
* '30}\n{"name": "Bob", "age"',
|
|
* ": 25}\n",
|
|
* ]);
|
|
*
|
|
* type Person = { name: string; age: number };
|
|
*
|
|
* // Split the stream by newline and parse each line as a JSON object
|
|
* const jsonStream = stream.pipeThrough(new TextLineStream())
|
|
* .pipeThrough(toTransformStream(async function* (src) {
|
|
* for await (const chunk of src) {
|
|
* if (chunk.trim().length === 0) {
|
|
* continue;
|
|
* }
|
|
* yield JSON.parse(chunk) as Person;
|
|
* }
|
|
* }));
|
|
*
|
|
* assertEquals(
|
|
* await Array.fromAsync(jsonStream),
|
|
* [{ "name": "Alice", "age": 30 }, { "name": "Bob", "age": 25 }],
|
|
* );
|
|
* ```
|
|
*/
|
|
export function toTransformStream<I, O>(
|
|
transformer: (src: ReadableStream<I>) => Iterable<O> | AsyncIterable<O>,
|
|
writableStrategy?: QueuingStrategy<I>,
|
|
readableStrategy?: QueuingStrategy<O>,
|
|
): TransformStream<I, O> {
|
|
const {
|
|
writable,
|
|
readable,
|
|
} = new TransformStream<I, I>(undefined, writableStrategy);
|
|
|
|
const iterable = transformer(readable);
|
|
const iterator: Iterator<O> | AsyncIterator<O> =
|
|
(iterable as AsyncIterable<O>)[Symbol.asyncIterator]?.() ??
|
|
(iterable as Iterable<O>)[Symbol.iterator]?.();
|
|
return {
|
|
writable,
|
|
readable: new ReadableStream<O>({
|
|
async pull(controller) {
|
|
let result: IteratorResult<O>;
|
|
try {
|
|
result = await iterator.next();
|
|
} catch (error) {
|
|
// Propagate error to stream from iterator
|
|
// If the stream status is "errored", it will be thrown, but ignore.
|
|
await readable.cancel(error).catch(() => {});
|
|
controller.error(error);
|
|
return;
|
|
}
|
|
if (result.done) {
|
|
controller.close();
|
|
return;
|
|
}
|
|
controller.enqueue(result.value);
|
|
},
|
|
async cancel(reason) {
|
|
// Propagate cancellation to readable and iterator
|
|
if (typeof iterator.throw === "function") {
|
|
try {
|
|
await iterator.throw(reason);
|
|
} catch {
|
|
/* `iterator.throw()` always throws on site. We catch it. */
|
|
}
|
|
}
|
|
await readable.cancel(reason);
|
|
},
|
|
}, readableStrategy),
|
|
};
|
|
}
|