๐Ÿ“ฆ socketio / socket.io-deno

๐Ÿ“„ websocket.ts ยท 68 lines
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
68import { getLogger } from "../../../../deps.ts";
import { Transport } from "../transport.ts";
import { Packet, Parser, RawData } from "../../../engine.io-parser/mod.ts";

export class WS extends Transport {
  private socket?: WebSocket;

  public get name() {
    return "websocket";
  }

  public get upgradesTo(): string[] {
    return [];
  }

  public send(packets: Packet[]) {
    for (const packet of packets) {
      Parser.encodePacket(packet, true, (data: RawData) => {
        if (this.writable && this.socket?.readyState === WebSocket.OPEN) {
          this.socket?.send(data);
        }
      });
    }
  }

  public onRequest(req: Request): Promise<Response> {
    const { socket, response } = Deno.upgradeWebSocket(req);

    this.socket = socket;

    socket.onopen = () => {
      getLogger("engine.io").debug(
        "[websocket] transport is now writable",
      );
      this.writable = true;
      this.emitReserved("drain");
    };

    socket.onmessage = ({ data }) => {
      // note: we use the length of the string here, which might be different from the number of bytes (up to 4 bytes)
      const byteLength = typeof data === "string"
        ? data.length
        : data.byteLength;
      if (byteLength > this.opts.maxHttpBufferSize) {
        return this.onError("payload too large");
      } else {
        this.onData(data);
      }
    };

    socket.onclose = (closeEvent) => {
      getLogger("engine.io").debug(
        `[websocket] onclose with code ${closeEvent.code}`,
      );
      this.writable = false;
      this.onClose();
    };

    // note: response.headers is immutable, so it seems we can't add headers here

    return Promise.resolve(response);
  }

  protected doClose() {
    this.socket?.close();
  }
}