-
Notifications
You must be signed in to change notification settings - Fork 634
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat(io):
iterateReader[Sync]()
(#4247)
- Loading branch information
Showing
5 changed files
with
232 additions
and
29 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,104 @@ | ||
// Copyright 2018-2024 the Deno authors. All rights reserved. MIT license. | ||
// This module is browser compatible. | ||
|
||
import { DEFAULT_BUFFER_SIZE } from "./_constants.ts"; | ||
import type { Reader, ReaderSync } from "./types.ts"; | ||
|
||
export type { Reader, ReaderSync }; | ||
|
||
/** | ||
* Turns a {@linkcode Reader} into an async iterator. | ||
* | ||
* @example | ||
* ```ts | ||
* import { iterateReader } from "https://deno.land/std@$STD_VERSION/io/iterate_reader.ts"; | ||
* | ||
* using file = await Deno.open("/etc/passwd"); | ||
* for await (const chunk of iterateReader(file)) { | ||
* console.log(chunk); | ||
* } | ||
* ``` | ||
* | ||
* Second argument can be used to tune size of a buffer. | ||
* Default size of the buffer is 32kB. | ||
* | ||
* @example | ||
* ```ts | ||
* import { iterateReader } from "https://deno.land/std@$STD_VERSION/io/iterate_reader.ts"; | ||
* | ||
* using file = await Deno.open("/etc/passwd"); | ||
* const iter = iterateReader(file, { | ||
* bufSize: 1024 * 1024 | ||
* }); | ||
* for await (const chunk of iter) { | ||
* console.log(chunk); | ||
* } | ||
* ``` | ||
*/ | ||
export async function* iterateReader( | ||
reader: Reader, | ||
options?: { | ||
bufSize?: number; | ||
}, | ||
): AsyncIterableIterator<Uint8Array> { | ||
const bufSize = options?.bufSize ?? DEFAULT_BUFFER_SIZE; | ||
const b = new Uint8Array(bufSize); | ||
while (true) { | ||
const result = await reader.read(b); | ||
if (result === null) { | ||
break; | ||
} | ||
|
||
yield b.slice(0, result); | ||
} | ||
} | ||
|
||
/** | ||
* Turns a {@linkcode ReaderSync} into an iterator. | ||
* | ||
* ```ts | ||
* import { iterateReaderSync } from "https://deno.land/std@$STD_VERSION/io/iterate_reader.ts"; | ||
* | ||
* using file = Deno.openSync("/etc/passwd"); | ||
* for (const chunk of iterateReaderSync(file)) { | ||
* console.log(chunk); | ||
* } | ||
* ``` | ||
* | ||
* Second argument can be used to tune size of a buffer. | ||
* Default size of the buffer is 32kB. | ||
* | ||
* ```ts | ||
* import { iterateReaderSync } from "https://deno.land/std@$STD_VERSION/io/iterate_reader.ts"; | ||
* using file = await Deno.open("/etc/passwd"); | ||
* const iter = iterateReaderSync(file, { | ||
* bufSize: 1024 * 1024 | ||
* }); | ||
* for (const chunk of iter) { | ||
* console.log(chunk); | ||
* } | ||
* ``` | ||
* | ||
* Iterator uses an internal buffer of fixed size for efficiency; it returns | ||
* a view on that buffer on each iteration. It is therefore caller's | ||
* responsibility to copy contents of the buffer if needed; otherwise the | ||
* next iteration will overwrite contents of previously returned chunk. | ||
*/ | ||
export function* iterateReaderSync( | ||
reader: ReaderSync, | ||
options?: { | ||
bufSize?: number; | ||
}, | ||
): IterableIterator<Uint8Array> { | ||
const bufSize = options?.bufSize ?? DEFAULT_BUFFER_SIZE; | ||
const b = new Uint8Array(bufSize); | ||
while (true) { | ||
const result = reader.readSync(b); | ||
if (result === null) { | ||
break; | ||
} | ||
|
||
yield b.slice(0, result); | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,111 @@ | ||
import { assertEquals } from "../assert/assert_equals.ts"; | ||
// Copyright 2018-2024 the Deno authors. All rights reserved. MIT license. | ||
|
||
import { iterateReader, iterateReaderSync } from "./iterate_reader.ts"; | ||
import { readerFromIterable } from "../streams/reader_from_iterable.ts"; | ||
import { delay } from "../async/delay.ts"; | ||
import type { Reader, ReaderSync } from "./types.ts"; | ||
|
||
Deno.test("iterateReader()", async () => { | ||
// ref: https://github.com/denoland/deno/issues/2330 | ||
const encoder = new TextEncoder(); | ||
|
||
class TestReader implements Reader { | ||
#offset = 0; | ||
#buf: Uint8Array; | ||
|
||
constructor(s: string) { | ||
this.#buf = new Uint8Array(encoder.encode(s)); | ||
} | ||
|
||
read(p: Uint8Array): Promise<number | null> { | ||
const n = Math.min(p.byteLength, this.#buf.byteLength - this.#offset); | ||
p.set(this.#buf.slice(this.#offset, this.#offset + n)); | ||
this.#offset += n; | ||
|
||
if (n === 0) { | ||
return Promise.resolve(null); | ||
} | ||
|
||
return Promise.resolve(n); | ||
} | ||
} | ||
|
||
const reader = new TestReader("hello world!"); | ||
|
||
let totalSize = 0; | ||
await Array.fromAsync( | ||
iterateReader(reader), | ||
(buf) => totalSize += buf.byteLength, | ||
); | ||
|
||
assertEquals(totalSize, 12); | ||
}); | ||
|
||
Deno.test("iterateReader() works with slow consumer", async () => { | ||
const a = new Uint8Array([97]); | ||
const b = new Uint8Array([98]); | ||
const iter = iterateReader(readerFromIterable([a, b])); | ||
const promises = []; | ||
for await (const bytes of iter) { | ||
promises.push(delay(10).then(() => bytes)); | ||
} | ||
assertEquals([a, b], await Promise.all(promises)); | ||
}); | ||
|
||
Deno.test("iterateReaderSync()", () => { | ||
// ref: https://github.com/denoland/deno/issues/2330 | ||
const encoder = new TextEncoder(); | ||
|
||
class TestReader implements ReaderSync { | ||
#offset = 0; | ||
#buf: Uint8Array; | ||
|
||
constructor(s: string) { | ||
this.#buf = new Uint8Array(encoder.encode(s)); | ||
} | ||
|
||
readSync(p: Uint8Array): number | null { | ||
const n = Math.min(p.byteLength, this.#buf.byteLength - this.#offset); | ||
p.set(this.#buf.slice(this.#offset, this.#offset + n)); | ||
this.#offset += n; | ||
|
||
if (n === 0) { | ||
return null; | ||
} | ||
|
||
return n; | ||
} | ||
} | ||
|
||
const reader = new TestReader("hello world!"); | ||
|
||
let totalSize = 0; | ||
for (const buf of iterateReaderSync(reader)) { | ||
totalSize += buf.byteLength; | ||
} | ||
|
||
assertEquals(totalSize, 12); | ||
}); | ||
|
||
Deno.test("iterateReaderSync() works with slow consumer", async () => { | ||
const a = new Uint8Array([97]); | ||
const b = new Uint8Array([98]); | ||
const data = [a, b]; | ||
const readerSync = { | ||
readSync(u8: Uint8Array) { | ||
const bytes = data.shift(); | ||
if (bytes) { | ||
u8.set(bytes); | ||
return bytes.length; | ||
} | ||
return null; | ||
}, | ||
}; | ||
const iter = iterateReaderSync(readerSync); | ||
const promises = []; | ||
for (const bytes of iter) { | ||
promises.push(delay(10).then(() => bytes)); | ||
} | ||
assertEquals([a, b], await Promise.all(promises)); | ||
}); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters