2020-01-02 15:13:47 -05:00
|
|
|
// Copyright 2018-2020 the Deno authors. All rights reserved. MIT license.
|
2020-06-02 00:37:59 +02:00
|
|
|
|
|
|
|
// Based on https://github.com/golang/go/blob/0452f9460f50f0f0aba18df43dc2b31906fb66cc/src/io/io.go
|
|
|
|
// Copyright 2009 The Go Authors. All rights reserved.
|
|
|
|
// Use of this source code is governed by a BSD-style
|
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
2019-03-08 09:25:16 +09:00
|
|
|
type Reader = Deno.Reader;
|
2020-04-01 15:23:39 -04:00
|
|
|
import { encode } from "../encoding/utf8.ts";
|
2019-02-11 08:49:48 +09:00
|
|
|
|
|
|
|
/** Reader utility for strings */
|
|
|
|
export class StringReader implements Reader {
|
|
|
|
private offs = 0;
|
|
|
|
private buf = new Uint8Array(encode(this.s));
|
|
|
|
|
|
|
|
constructor(private readonly s: string) {}
|
|
|
|
|
2020-04-28 17:40:43 +01:00
|
|
|
read(p: Uint8Array): Promise<number | null> {
|
2019-02-11 08:49:48 +09:00
|
|
|
const n = Math.min(p.byteLength, this.buf.byteLength - this.offs);
|
|
|
|
p.set(this.buf.slice(this.offs, this.offs + n));
|
|
|
|
this.offs += n;
|
2019-07-08 04:20:41 +09:00
|
|
|
if (n === 0) {
|
2020-04-28 17:40:43 +01:00
|
|
|
return Promise.resolve(null);
|
2019-07-08 04:20:41 +09:00
|
|
|
}
|
2020-03-20 14:38:34 +01:00
|
|
|
return Promise.resolve(n);
|
2019-02-11 08:49:48 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Reader utility for combining multiple readers */
|
|
|
|
export class MultiReader implements Reader {
|
|
|
|
private readonly readers: Reader[];
|
|
|
|
private currentIndex = 0;
|
|
|
|
|
|
|
|
constructor(...readers: Reader[]) {
|
|
|
|
this.readers = readers;
|
|
|
|
}
|
|
|
|
|
2020-04-28 17:40:43 +01:00
|
|
|
async read(p: Uint8Array): Promise<number | null> {
|
2019-02-11 08:49:48 +09:00
|
|
|
const r = this.readers[this.currentIndex];
|
2020-04-28 17:40:43 +01:00
|
|
|
if (!r) return null;
|
2019-07-08 04:20:41 +09:00
|
|
|
const result = await r.read(p);
|
2020-04-28 17:40:43 +01:00
|
|
|
if (result === null) {
|
2019-02-11 08:49:48 +09:00
|
|
|
this.currentIndex++;
|
2019-07-08 04:20:41 +09:00
|
|
|
return 0;
|
2019-02-11 08:49:48 +09:00
|
|
|
}
|
2019-07-08 04:20:41 +09:00
|
|
|
return result;
|
2019-02-11 08:49:48 +09:00
|
|
|
}
|
|
|
|
}
|
2020-06-02 00:37:59 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* A `LimitedReader` reads from `reader` but limits the amount of data returned to just `limit` bytes.
|
|
|
|
* Each call to `read` updates `limit` to reflect the new amount remaining.
|
|
|
|
* `read` returns `null` when `limit` <= `0` or
|
|
|
|
* when the underlying `reader` returns `null`.
|
|
|
|
*/
|
|
|
|
export class LimitedReader implements Deno.Reader {
|
|
|
|
constructor(public reader: Deno.Reader, public limit: number) {}
|
|
|
|
|
|
|
|
async read(p: Uint8Array): Promise<number | null> {
|
|
|
|
if (this.limit <= 0) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (p.length > this.limit) {
|
|
|
|
p = p.subarray(0, this.limit);
|
|
|
|
}
|
|
|
|
const n = await this.reader.read(p);
|
|
|
|
if (n == null) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.limit -= n;
|
|
|
|
return n;
|
|
|
|
}
|
|
|
|
}
|