diff options
| author | Bert Belder <bertbelder@gmail.com> | 2019-05-20 17:08:05 -0700 |
|---|---|---|
| committer | Bert Belder <bertbelder@gmail.com> | 2019-05-20 17:53:01 -0700 |
| commit | 16e52ee4b000870da8ef570e5182e3d14ca41d03 (patch) | |
| tree | 238a22753b9f0b4f3d90db54327a3cf8dcc2d00d | |
| parent | 79c88a46bbc3887560062e54a2dccaea4f213cf3 (diff) | |
Move collectUint8Arrays() to util/async.ts, clean up, fix bugs, add tests
Original: https://github.com/denoland/deno_std/commit/dcad420b92f79311b386ef524bd1487c04400cc0
| -rw-r--r-- | http/server.ts | 31 | ||||
| -rw-r--r-- | util/async.ts | 25 | ||||
| -rw-r--r-- | util/async_test.ts | 43 |
3 files changed, 73 insertions, 26 deletions
diff --git a/http/server.ts b/http/server.ts index 9ca6c4a89..da335a58c 100644 --- a/http/server.ts +++ b/http/server.ts @@ -8,7 +8,12 @@ import { BufReader, BufState, BufWriter } from "../io/bufio.ts"; import { TextProtoReader } from "../textproto/mod.ts"; import { STATUS_TEXT } from "./http_status.ts"; import { assert, fail } from "../testing/asserts.ts"; -import { deferred, Deferred, MuxAsyncIterator } from "../util/async.ts"; +import { + collectUint8Arrays, + deferred, + Deferred, + MuxAsyncIterator +} from "../util/async.ts"; function bufWriter(w: Writer): BufWriter { if (w instanceof BufWriter) { @@ -94,28 +99,6 @@ export async function writeResponse(w: Writer, r: Response): Promise<void> { await writer.flush(); } -async function readAllIterator( - it: AsyncIterableIterator<Uint8Array> -): Promise<Uint8Array> { - const chunks = []; - let len = 0; - for await (const chunk of it) { - chunks.push(chunk); - len += chunk.length; - } - if (chunks.length === 0) { - // No need for copy - return chunks[0]; - } - const collected = new Uint8Array(len); - let offset = 0; - for (let chunk of chunks) { - collected.set(chunk, offset); - offset += chunk.length; - } - return collected; -} - export class ServerRequest { url: string; method: string; @@ -203,7 +186,7 @@ export class ServerRequest { // Read the body of the request into a single Uint8Array public async body(): Promise<Uint8Array> { - return readAllIterator(this.bodyStream()); + return collectUint8Arrays(this.bodyStream()); } async respond(r: Response): Promise<void> { diff --git a/util/async.ts b/util/async.ts index f9f2477d0..15edb1b42 100644 --- a/util/async.ts +++ b/util/async.ts @@ -83,3 +83,28 @@ export class MuxAsyncIterator<T> implements AsyncIterable<T> { return this.iterate(); } } + +/** Collects all Uint8Arrays from an AsyncIterable and retuns a single + * Uint8Array with the concatenated contents of all the collected arrays. + */ +export async function collectUint8Arrays( + it: AsyncIterable<Uint8Array> +): Promise<Uint8Array> { + const chunks = []; + let length = 0; + for await (const chunk of it) { + chunks.push(chunk); + length += chunk.length; + } + if (chunks.length === 1) { + // No need to copy. + return chunks[0]; + } + const collected = new Uint8Array(length); + let offset = 0; + for (let chunk of chunks) { + collected.set(chunk, offset); + offset += chunk.length; + } + return collected; +} diff --git a/util/async_test.ts b/util/async_test.ts index c704002d4..adaac1e22 100644 --- a/util/async_test.ts +++ b/util/async_test.ts @@ -1,7 +1,7 @@ // Copyright 2018-2019 the Deno authors. All rights reserved. MIT license. import { test, runIfMain } from "../testing/mod.ts"; -import { assertEquals } from "../testing/asserts.ts"; -import { MuxAsyncIterator, deferred } from "./async.ts"; +import { assert, assertEquals, assertStrictEq } from "../testing/asserts.ts"; +import { collectUint8Arrays, deferred, MuxAsyncIterator } from "./async.ts"; test(async function asyncDeferred(): Promise<void> { const d = deferred<number>(); @@ -31,4 +31,43 @@ test(async function asyncMuxAsyncIterator(): Promise<void> { assertEquals(results.size, 6); }); +test(async function collectUint8Arrays0(): Promise<void> { + async function* gen(): AsyncIterableIterator<Uint8Array> {} + const result = await collectUint8Arrays(gen()); + assert(result instanceof Uint8Array); + assertEquals(result.length, 0); +}); + +test(async function collectUint8Arrays0(): Promise<void> { + async function* gen(): AsyncIterableIterator<Uint8Array> {} + const result = await collectUint8Arrays(gen()); + assert(result instanceof Uint8Array); + assertStrictEq(result.length, 0); +}); + +test(async function collectUint8Arrays1(): Promise<void> { + const buf = new Uint8Array([1, 2, 3]); + async function* gen(): AsyncIterableIterator<Uint8Array> { + yield buf; + } + const result = await collectUint8Arrays(gen()); + assertStrictEq(result, buf); + assertStrictEq(result.length, 3); +}); + +test(async function collectUint8Arrays4(): Promise<void> { + async function* gen(): AsyncIterableIterator<Uint8Array> { + yield new Uint8Array([1, 2, 3]); + yield new Uint8Array([]); + yield new Uint8Array([4, 5]); + yield new Uint8Array([6]); + } + const result = await collectUint8Arrays(gen()); + assert(result instanceof Uint8Array); + assertStrictEq(result.length, 6); + for (let i = 0; i < 6; i++) { + assertStrictEq(result[i], i + 1); + } +}); + runIfMain(import.meta); |
