summaryrefslogtreecommitdiff
path: root/cli/js/streams/readable-stream-byob-reader.ts
blob: 0f9bfb037848d3006b8637e827c4c56f5e523037 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
// Forked from https://github.com/stardazed/sd-streams/tree/8928cf04b035fd02fb1340b7eb541c76be37e546
// Copyright (c) 2018-Present by Arthur Langereis - @zenmumbler MIT

/**
 * streams/readable-stream-byob-reader - ReadableStreamBYOBReader class implementation
 * Part of Stardazed
 * (c) 2018-Present by Arthur Langereis - @zenmumbler
 * https://github.com/stardazed/sd-streams
 */

import * as rs from "./readable-internals.ts";
import * as shared from "./shared-internals.ts";

export class SDReadableStreamBYOBReader
  implements rs.SDReadableStreamBYOBReader {
  [rs.closedPromise_]: shared.ControlledPromise<void>;
  [rs.ownerReadableStream_]: rs.SDReadableStream<ArrayBufferView> | undefined;
  [rs.readIntoRequests_]: Array<
    rs.ReadRequest<IteratorResult<ArrayBufferView>>
  >;

  constructor(stream: rs.SDReadableStream<ArrayBufferView>) {
    if (!rs.isReadableStream(stream)) {
      throw new TypeError();
    }
    if (
      !rs.isReadableByteStreamController(stream[rs.readableStreamController_])
    ) {
      throw new TypeError();
    }
    if (rs.isReadableStreamLocked(stream)) {
      throw new TypeError("The stream is locked.");
    }
    rs.readableStreamReaderGenericInitialize(this, stream);
    this[rs.readIntoRequests_] = [];
  }

  get closed(): Promise<void> {
    if (!rs.isReadableStreamBYOBReader(this)) {
      return Promise.reject(new TypeError());
    }
    return this[rs.closedPromise_].promise;
  }

  cancel(reason: shared.ErrorResult): Promise<void> {
    if (!rs.isReadableStreamBYOBReader(this)) {
      return Promise.reject(new TypeError());
    }
    const stream = this[rs.ownerReadableStream_];
    if (stream === undefined) {
      return Promise.reject(
        new TypeError("Reader is not associated with a stream")
      );
    }
    return rs.readableStreamCancel(stream, reason);
  }

  read(view: ArrayBufferView): Promise<IteratorResult<ArrayBufferView>> {
    if (!rs.isReadableStreamBYOBReader(this)) {
      return Promise.reject(new TypeError());
    }
    if (this[rs.ownerReadableStream_] === undefined) {
      return Promise.reject(
        new TypeError("Reader is not associated with a stream")
      );
    }
    if (!ArrayBuffer.isView(view)) {
      return Promise.reject(
        new TypeError("view argument must be a valid ArrayBufferView")
      );
    }
    // If ! IsDetachedBuffer(view.[[ViewedArrayBuffer]]) is true, return a promise rejected with a TypeError exception.
    if (view.byteLength === 0) {
      return Promise.reject(
        new TypeError("supplied buffer view must be > 0 bytes")
      );
    }
    return rs.readableStreamBYOBReaderRead(this, view, true);
  }

  releaseLock(): void {
    if (!rs.isReadableStreamBYOBReader(this)) {
      throw new TypeError();
    }
    if (this[rs.ownerReadableStream_] === undefined) {
      throw new TypeError("Reader is not associated with a stream");
    }
    if (this[rs.readIntoRequests_].length > 0) {
      throw new TypeError();
    }
    rs.readableStreamReaderGenericRelease(this);
  }
}