2020-01-02 15:13:47 -05:00
|
|
|
// Copyright 2018-2020 the Deno authors. All rights reserved. MIT license.
|
2018-09-27 00:56:39 -04:00
|
|
|
// Interfaces 100% copied from Go.
|
|
|
|
// Documentation liberally lifted from them too.
|
|
|
|
// Thank you! We love Go!
|
|
|
|
|
2019-10-31 23:57:09 +09:00
|
|
|
export const EOF: unique symbol = Symbol("EOF");
|
|
|
|
export type EOF = typeof EOF;
|
2018-09-27 00:56:39 -04:00
|
|
|
|
2019-02-18 15:26:41 -08:00
|
|
|
// Seek whence values.
|
|
|
|
// https://golang.org/pkg/io/#pkg-constants
|
2020-03-02 10:19:42 -05:00
|
|
|
/** **UNSTABLE**: might remove `"SEEK_"` prefix. Might not use all-caps. */
|
2019-02-18 15:26:41 -08:00
|
|
|
export enum SeekMode {
|
|
|
|
SEEK_START = 0,
|
|
|
|
SEEK_CURRENT = 1,
|
|
|
|
SEEK_END = 2
|
|
|
|
}
|
|
|
|
|
2018-09-27 00:56:39 -04:00
|
|
|
// Reader is the interface that wraps the basic read() method.
|
|
|
|
// https://golang.org/pkg/io/#Reader
|
2020-03-02 10:19:42 -05:00
|
|
|
/** **UNSTABLE**: might make `Reader` into iterator of some sort. */
|
2018-09-27 00:56:39 -04:00
|
|
|
export interface Reader {
|
2020-03-02 10:19:42 -05:00
|
|
|
/** Reads up to `p.byteLength` bytes into `p`. It resolves to the number of
|
|
|
|
* bytes read (`0` < `n` <= `p.byteLength`) and rejects if any error
|
|
|
|
* encountered. Even if `read()` resolves to `n` < `p.byteLength`, it may
|
|
|
|
* use all of `p` as scratch space during the call. If some data is
|
|
|
|
* available but not `p.byteLength` bytes, `read()` conventionally resolves
|
|
|
|
* to what is available instead of waiting for more.
|
2018-10-15 07:29:50 +11:00
|
|
|
*
|
2020-03-02 10:19:42 -05:00
|
|
|
* When `read()` encounters end-of-file condition, it resolves to
|
|
|
|
* `Deno.EOF` symbol.
|
2018-10-15 07:29:50 +11:00
|
|
|
*
|
2019-07-06 23:16:03 +09:00
|
|
|
* When `read()` encounters an error, it rejects with an error.
|
2018-10-15 07:29:50 +11:00
|
|
|
*
|
2019-07-06 23:16:03 +09:00
|
|
|
* Callers should always process the `n` > `0` bytes returned before
|
2020-03-02 10:19:42 -05:00
|
|
|
* considering the `EOF`. Doing so correctly handles I/O errors that happen
|
2019-07-06 23:16:03 +09:00
|
|
|
* after reading some bytes and also both of the allowed EOF behaviors.
|
2018-10-15 07:29:50 +11:00
|
|
|
*
|
2020-03-02 10:19:42 -05:00
|
|
|
* Implementations should not retain a reference to `p`.
|
2018-10-15 07:29:50 +11:00
|
|
|
*/
|
2019-07-06 23:16:03 +09:00
|
|
|
read(p: Uint8Array): Promise<number | EOF>;
|
2018-09-27 00:56:39 -04:00
|
|
|
}
|
|
|
|
|
2019-03-28 04:29:36 +01:00
|
|
|
export interface SyncReader {
|
2020-03-02 10:19:42 -05:00
|
|
|
/** Reads up to `p.byteLength` bytes into `p`. It resolves to the number
|
|
|
|
* of bytes read (`0` < `n` <= `p.byteLength`) and rejects if any error
|
|
|
|
* encountered. Even if `read()` returns `n` < `p.byteLength`, it may use
|
|
|
|
* all of `p` as scratch space during the call. If some data is available
|
|
|
|
* but not `p.byteLength` bytes, `read()` conventionally returns what is
|
|
|
|
* available instead of waiting for more.
|
|
|
|
*
|
|
|
|
* When `readSync()` encounters end-of-file condition, it returns `Deno.EOF`
|
|
|
|
* symbol.
|
|
|
|
*
|
|
|
|
* When `readSync()` encounters an error, it throws with an error.
|
|
|
|
*
|
|
|
|
* Callers should always process the `n` > `0` bytes returned before
|
|
|
|
* considering the `EOF`. Doing so correctly handles I/O errors that happen
|
|
|
|
* after reading some bytes and also both of the allowed EOF behaviors.
|
|
|
|
*
|
|
|
|
* Implementations should not retain a reference to `p`.
|
|
|
|
*/
|
2019-07-06 23:16:03 +09:00
|
|
|
readSync(p: Uint8Array): number | EOF;
|
2019-03-28 04:29:36 +01:00
|
|
|
}
|
|
|
|
|
2018-09-27 00:56:39 -04:00
|
|
|
// Writer is the interface that wraps the basic write() method.
|
|
|
|
// https://golang.org/pkg/io/#Writer
|
|
|
|
export interface Writer {
|
2020-03-02 10:19:42 -05:00
|
|
|
/** Writes `p.byteLength` bytes from `p` to the underlying data stream. It
|
|
|
|
* resolves to the number of bytes written from `p` (`0` <= `n` <=
|
|
|
|
* `p.byteLength`) or reject with the error encountered that caused the
|
|
|
|
* write to stop early. `write()` must reject with a non-null error if
|
|
|
|
* would resolve to `n` < `p.byteLength`. `write()` must not modify the
|
|
|
|
* slice data, even temporarily.
|
2018-10-15 07:29:50 +11:00
|
|
|
*
|
2020-03-02 10:19:42 -05:00
|
|
|
* Implementations should not retain a reference to `p`.
|
2018-10-15 07:29:50 +11:00
|
|
|
*/
|
2018-11-08 01:18:21 -05:00
|
|
|
write(p: Uint8Array): Promise<number>;
|
2018-09-27 00:56:39 -04:00
|
|
|
}
|
|
|
|
|
2019-03-28 04:29:36 +01:00
|
|
|
export interface SyncWriter {
|
2020-03-02 10:19:42 -05:00
|
|
|
/** Writes `p.byteLength` bytes from `p` to the underlying data
|
|
|
|
* stream. It returns the number of bytes written from `p` (`0` <= `n`
|
|
|
|
* <= `p.byteLength`) and any error encountered that caused the write to
|
|
|
|
* stop early. `writeSync()` must throw a non-null error if it returns `n` <
|
|
|
|
* `p.byteLength`. `writeSync()` must not modify the slice data, even
|
|
|
|
* temporarily.
|
|
|
|
*
|
|
|
|
* Implementations should not retain a reference to `p`.
|
|
|
|
*/
|
2019-03-28 04:29:36 +01:00
|
|
|
writeSync(p: Uint8Array): number;
|
|
|
|
}
|
2020-03-02 10:19:42 -05:00
|
|
|
|
2018-09-27 00:56:39 -04:00
|
|
|
// https://golang.org/pkg/io/#Closer
|
|
|
|
export interface Closer {
|
|
|
|
// The behavior of Close after the first call is undefined. Specific
|
|
|
|
// implementations may document their own behavior.
|
|
|
|
close(): void;
|
|
|
|
}
|
|
|
|
|
|
|
|
// https://golang.org/pkg/io/#Seeker
|
|
|
|
export interface Seeker {
|
2018-10-15 07:29:50 +11:00
|
|
|
/** Seek sets the offset for the next `read()` or `write()` to offset,
|
2020-03-02 10:19:42 -05:00
|
|
|
* interpreted according to `whence`: `SEEK_START` means relative to the
|
|
|
|
* start of the file, `SEEK_CURRENT` means relative to the current offset,
|
|
|
|
* and `SEEK_END` means relative to the end.
|
2018-10-15 07:29:50 +11:00
|
|
|
*
|
|
|
|
* Seeking to an offset before the start of the file is an error. Seeking to
|
2020-03-02 17:44:46 +01:00
|
|
|
* any positive offset is legal, but the behavior of subsequent I/O operations
|
|
|
|
* on the underlying object is implementation-dependent. It returns the cursor
|
|
|
|
* position.
|
2018-10-15 07:29:50 +11:00
|
|
|
*/
|
2020-03-02 17:44:46 +01:00
|
|
|
seek(offset: number, whence: SeekMode): Promise<number>;
|
2018-09-27 00:56:39 -04:00
|
|
|
}
|
|
|
|
|
2019-03-28 04:29:36 +01:00
|
|
|
export interface SyncSeeker {
|
2020-03-02 17:44:46 +01:00
|
|
|
seekSync(offset: number, whence: SeekMode): number;
|
2019-03-28 04:29:36 +01:00
|
|
|
}
|
|
|
|
|
2018-09-27 00:56:39 -04:00
|
|
|
// https://golang.org/pkg/io/#ReadCloser
|
2018-10-26 12:14:06 -04:00
|
|
|
export interface ReadCloser extends Reader, Closer {}
|
2018-09-27 00:56:39 -04:00
|
|
|
|
|
|
|
// https://golang.org/pkg/io/#WriteCloser
|
|
|
|
export interface WriteCloser extends Writer, Closer {}
|
|
|
|
|
|
|
|
// https://golang.org/pkg/io/#ReadSeeker
|
|
|
|
export interface ReadSeeker extends Reader, Seeker {}
|
|
|
|
|
|
|
|
// https://golang.org/pkg/io/#WriteSeeker
|
|
|
|
export interface WriteSeeker extends Writer, Seeker {}
|
|
|
|
|
|
|
|
// https://golang.org/pkg/io/#ReadWriteCloser
|
|
|
|
export interface ReadWriteCloser extends Reader, Writer, Closer {}
|
|
|
|
|
|
|
|
// https://golang.org/pkg/io/#ReadWriteSeeker
|
|
|
|
export interface ReadWriteSeeker extends Reader, Writer, Seeker {}
|
|
|
|
|
2020-03-02 10:19:42 -05:00
|
|
|
/** Copies from `src` to `dst` until either `EOF` is reached on `src` or an
|
|
|
|
* error occurs. It resolves to the number of bytes copied or rejects with
|
|
|
|
* the first error encountered while copying.
|
2018-10-15 07:29:50 +11:00
|
|
|
*
|
|
|
|
* Because `copy()` is defined to read from `src` until `EOF`, it does not
|
|
|
|
* treat an `EOF` from `read()` as an error to be reported.
|
|
|
|
*/
|
2018-09-27 00:56:39 -04:00
|
|
|
// https://golang.org/pkg/io/#Copy
|
|
|
|
export async function copy(dst: Writer, src: Reader): Promise<number> {
|
|
|
|
let n = 0;
|
2018-10-15 07:29:50 +11:00
|
|
|
const b = new Uint8Array(32 * 1024);
|
2018-09-27 00:56:39 -04:00
|
|
|
let gotEOF = false;
|
|
|
|
while (gotEOF === false) {
|
|
|
|
const result = await src.read(b);
|
2019-07-06 23:16:03 +09:00
|
|
|
if (result === EOF) {
|
2018-09-27 00:56:39 -04:00
|
|
|
gotEOF = true;
|
2019-07-06 23:16:03 +09:00
|
|
|
} else {
|
|
|
|
n += await dst.write(b.subarray(0, result));
|
2018-09-27 00:56:39 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return n;
|
|
|
|
}
|
2018-10-31 15:29:13 +01:00
|
|
|
|
2018-11-09 15:07:48 +11:00
|
|
|
/** Turns `r` into async iterator.
|
2018-10-31 15:29:13 +01:00
|
|
|
*
|
2019-01-29 12:24:40 -06:00
|
|
|
* for await (const chunk of toAsyncIterator(reader)) {
|
2020-03-02 10:19:42 -05:00
|
|
|
* console.log(chunk);
|
2018-11-09 15:07:48 +11:00
|
|
|
* }
|
2018-10-31 15:29:13 +01:00
|
|
|
*/
|
2018-11-08 01:18:21 -05:00
|
|
|
export function toAsyncIterator(r: Reader): AsyncIterableIterator<Uint8Array> {
|
2018-10-31 15:29:13 +01:00
|
|
|
const b = new Uint8Array(1024);
|
|
|
|
return {
|
2019-04-21 16:40:10 -04:00
|
|
|
[Symbol.asyncIterator](): AsyncIterableIterator<Uint8Array> {
|
2018-10-31 15:29:13 +01:00
|
|
|
return this;
|
|
|
|
},
|
|
|
|
|
2018-11-08 01:18:21 -05:00
|
|
|
async next(): Promise<IteratorResult<Uint8Array>> {
|
2018-10-31 15:29:13 +01:00
|
|
|
const result = await r.read(b);
|
2019-07-06 23:16:03 +09:00
|
|
|
if (result === EOF) {
|
|
|
|
return { value: new Uint8Array(), done: true };
|
|
|
|
}
|
2019-05-11 16:05:56 +02:00
|
|
|
|
2018-10-31 15:29:13 +01:00
|
|
|
return {
|
2019-07-06 23:16:03 +09:00
|
|
|
value: b.subarray(0, result),
|
2019-05-11 16:05:56 +02:00
|
|
|
done: false
|
2018-10-31 15:29:13 +01:00
|
|
|
};
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|