File: RTCDataChannel-worker.js

package info (click to toggle)
firefox 147.0.3-1
  • links: PTS, VCS
  • area: main
  • in suites: sid
  • size: 4,683,320 kB
  • sloc: cpp: 7,607,359; javascript: 6,533,295; ansic: 3,775,223; python: 1,415,500; xml: 634,561; asm: 438,949; java: 186,241; sh: 62,752; makefile: 18,079; objc: 13,092; perl: 12,808; yacc: 4,583; cs: 3,846; pascal: 3,448; lex: 1,720; ruby: 1,003; php: 436; lisp: 258; awk: 247; sql: 66; sed: 54; csh: 10; exp: 6
file content (124 lines) | stat: -rw-r--r-- 3,978 bytes parent folder | download | duplicates (3)
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
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
console.log(`Creating worker...`);
try {
  const channels = new Map();

  function getChannel(label) {
    const dc = channels.get(label);
    if (!dc) throw new Error(`No datachannel found for label ${label}`);
    return dc;
  }

  function reportWorkerError(label, message) {
    self.postMessage({ type: 'workerError', label, result: message });
  }

  function wireEvents(dc) {
    const eventTypes = ['open', 'bufferedamountlow', 'error', 'closing', 'close', 'message'];
    for (const type of eventTypes) {
      dc.addEventListener(type, ({data, origin}) => {
        if (type == 'message') {
          // Events aren't transferable in general, have to reconstruct :(
          self.postMessage({type, label: dc.label, result: {data, origin}});
        } else {
          self.postMessage({type, label: dc.label});
        }
      });
    }
  }

  onmessage = ({data}) => {
    const { type, label, arg } = data;

    try {
      switch (type) {
        case 'init': {
          const channel = arg;
          // We do not put errors in an initResponse message; those are for
          // RTCDataChannel errors only.
          if (channels.has(channel.label)) {
            throw new Error('Duplicate RTCDataChannel label');
          }
          channels.set(channel.label, channel);
          wireEvents(channel);
          self.postMessage({
            type: 'initResponse',
            label: channel.label,
            result: {
              label: channel.label,
              ordered: channel.ordered,
              maxPacketLifeTime: channel.maxPacketLifeTime,
              maxRetransmits: channel.maxRetransmits,
              protocol: channel.protocol,
              negotiated: channel.negotiated,
              id: channel.id,
              readyState: channel.readyState,
              bufferedAmount: channel.bufferedAmount,
              binaryType: channel.binaryType,
              bufferedAmountLowThreshold: channel.bufferedAmountLowThreshold,
            },
          });
          break;
        }

        case 'send': {
          // try block is only for RTCDataChannel errors
          const channel = getChannel(label);
          try {
            channel.send(arg);
            self.postMessage({ type: 'sendResponse', label, result: undefined });
          } catch (e) {
            // Man it would be nice if we could do "const error = {...} = e"
            const error = { name: e.name, message: e.message };
            self.postMessage({ type: 'sendResponse', label, result: { error } });
          }
          break;
        }

        case 'close': {
          // RTCDataChannel.close() does not throw; any error here is ours
          getChannel(label).close();
          self.postMessage({ type: 'closeResponse', label, result: undefined });
          break;
        }

        case 'setBufferedAmountLowThreshold': {
          // This is fire-and-forget called by a setter, any error will be
          // dealt with by the code in _errorPromise.
          getChannel(label).bufferedAmountLowThreshold = arg;
          break;
        }

        case 'setBinaryType': {
          // This is fire-and-forget called by a setter
          getChannel(label).binaryType = arg;
          break;
        }

        case 'queryState': {
          const channel = getChannel(label);
          self.postMessage({
            type: 'queryStateResponse',
            label,
            result: {
              id: channel.id,
              readyState: channel.readyState,
              bufferedAmount: channel.bufferedAmount,
            },
          });
          break;
        }

        default:
          console.log(`Unknown type`);
          reportWorkerError(label, `Received unknown message type: ${type}`);
      }
    } catch (err) {
      console.log(`Unhandled error`);
      reportWorkerError(label, `In handling ${type} request, got ${err.message}`);
    }
  };

} catch (e) {
  console.log(`Creation failed! ${e.message}`);
}