yansu 0.0.0-beta.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (142) hide show
  1. package/README.md +1 -0
  2. package/bin.mjs +18 -0
  3. package/dist/cli.d.mts +2 -0
  4. package/dist/cli.mjs +68 -0
  5. package/dist/dirs.d.mts +4 -0
  6. package/dist/dirs.mjs +6 -0
  7. package/dist/index.d.mts +78 -0
  8. package/dist/index.mjs +121 -0
  9. package/dist/nitro.json +15 -0
  10. package/dist/public/_nuxt/BHIHUtPz.js +4 -0
  11. package/dist/public/_nuxt/CCgOhniq.js +1 -0
  12. package/dist/public/_nuxt/CR6vfq1R.js +1 -0
  13. package/dist/public/_nuxt/DmxzK6E8.js +1 -0
  14. package/dist/public/_nuxt/builds/latest.json +1 -0
  15. package/dist/public/_nuxt/builds/meta/e9079134-f314-434b-b3a7-092c4a1e187d.json +1 -0
  16. package/dist/public/_nuxt/entry.DJ3VOi_0.css +1 -0
  17. package/dist/public/_nuxt/error-404.lC6KBLNm.css +1 -0
  18. package/dist/public/_nuxt/error-500.NtBcR2wE.css +1 -0
  19. package/dist/server/chunks/_/error-500.mjs +9 -0
  20. package/dist/server/chunks/build/_plugin-vue_export-helper-DjsbPc54.mjs +29 -0
  21. package/dist/server/chunks/build/client.precomputed.mjs +3 -0
  22. package/dist/server/chunks/build/error-404-Du0ot2hm.mjs +369 -0
  23. package/dist/server/chunks/build/error-500-Y3RUV6n2.mjs +73 -0
  24. package/dist/server/chunks/build/server.mjs +727 -0
  25. package/dist/server/chunks/nitro/nitro.mjs +5176 -0
  26. package/dist/server/chunks/routes/api/metadata.json.mjs +147 -0
  27. package/dist/server/chunks/routes/api/platform.json.mjs +20 -0
  28. package/dist/server/chunks/routes/renderer.mjs +408 -0
  29. package/dist/server/chunks/virtual/_virtual_spa-template.mjs +3 -0
  30. package/dist/server/index.mjs +9 -0
  31. package/dist/server/node_modules/.nitro/@vue/shared@3.5.25/dist/shared.cjs.prod.js +604 -0
  32. package/dist/server/node_modules/.nitro/@vue/shared@3.5.25/package.json +47 -0
  33. package/dist/server/node_modules/.nitro/@vue/shared@3.5.26/dist/shared.cjs.prod.js +604 -0
  34. package/dist/server/node_modules/.nitro/@vue/shared@3.5.26/package.json +47 -0
  35. package/dist/server/node_modules/@babel/parser/lib/index.js +14662 -0
  36. package/dist/server/node_modules/@babel/parser/package.json +50 -0
  37. package/dist/server/node_modules/@vue/compiler-core/dist/compiler-core.cjs.prod.js +6763 -0
  38. package/dist/server/node_modules/@vue/compiler-core/package.json +58 -0
  39. package/dist/server/node_modules/@vue/compiler-dom/dist/compiler-dom.cjs.prod.js +689 -0
  40. package/dist/server/node_modules/@vue/compiler-dom/package.json +57 -0
  41. package/dist/server/node_modules/@vue/compiler-ssr/dist/compiler-ssr.cjs.js +1413 -0
  42. package/dist/server/node_modules/@vue/compiler-ssr/package.json +34 -0
  43. package/dist/server/node_modules/@vue/reactivity/dist/reactivity.cjs.prod.js +1853 -0
  44. package/dist/server/node_modules/@vue/reactivity/package.json +55 -0
  45. package/dist/server/node_modules/@vue/runtime-core/dist/runtime-core.cjs.prod.js +6770 -0
  46. package/dist/server/node_modules/@vue/runtime-core/package.json +52 -0
  47. package/dist/server/node_modules/@vue/runtime-dom/dist/runtime-dom.cjs.prod.js +1732 -0
  48. package/dist/server/node_modules/@vue/runtime-dom/package.json +60 -0
  49. package/dist/server/node_modules/@vue/server-renderer/dist/server-renderer.cjs.prod.js +879 -0
  50. package/dist/server/node_modules/@vue/server-renderer/package.json +55 -0
  51. package/dist/server/node_modules/ansis/index.cjs +1 -0
  52. package/dist/server/node_modules/ansis/index.mjs +1 -0
  53. package/dist/server/node_modules/ansis/package.json +25 -0
  54. package/dist/server/node_modules/birpc/dist/index.mjs +173 -0
  55. package/dist/server/node_modules/birpc/package.json +56 -0
  56. package/dist/server/node_modules/consola/dist/chunks/prompt.mjs +280 -0
  57. package/dist/server/node_modules/consola/dist/core.mjs +512 -0
  58. package/dist/server/node_modules/consola/dist/index.mjs +651 -0
  59. package/dist/server/node_modules/consola/dist/shared/consola.DRwqZj3T.mjs +72 -0
  60. package/dist/server/node_modules/consola/dist/shared/consola.DXBYu-KD.mjs +288 -0
  61. package/dist/server/node_modules/consola/package.json +136 -0
  62. package/dist/server/node_modules/devalue/index.js +4 -0
  63. package/dist/server/node_modules/devalue/package.json +37 -0
  64. package/dist/server/node_modules/devalue/src/base64.js +110 -0
  65. package/dist/server/node_modules/devalue/src/constants.js +6 -0
  66. package/dist/server/node_modules/devalue/src/parse.js +205 -0
  67. package/dist/server/node_modules/devalue/src/stringify.js +265 -0
  68. package/dist/server/node_modules/devalue/src/uneval.js +407 -0
  69. package/dist/server/node_modules/devalue/src/utils.js +118 -0
  70. package/dist/server/node_modules/entities/dist/commonjs/decode-codepoint.js +77 -0
  71. package/dist/server/node_modules/entities/dist/commonjs/decode.js +568 -0
  72. package/dist/server/node_modules/entities/dist/commonjs/generated/decode-data-html.js +7 -0
  73. package/dist/server/node_modules/entities/dist/commonjs/generated/decode-data-xml.js +7 -0
  74. package/dist/server/node_modules/entities/dist/commonjs/internal/bin-trie-flags.js +21 -0
  75. package/dist/server/node_modules/entities/dist/commonjs/internal/decode-shared.js +31 -0
  76. package/dist/server/node_modules/entities/dist/commonjs/package.json +3 -0
  77. package/dist/server/node_modules/entities/package.json +113 -0
  78. package/dist/server/node_modules/estree-walker/dist/umd/estree-walker.js +344 -0
  79. package/dist/server/node_modules/estree-walker/package.json +37 -0
  80. package/dist/server/node_modules/get-port-please/dist/index.mjs +430 -0
  81. package/dist/server/node_modules/get-port-please/package.json +39 -0
  82. package/dist/server/node_modules/hookable/dist/index.mjs +290 -0
  83. package/dist/server/node_modules/hookable/package.json +49 -0
  84. package/dist/server/node_modules/source-map-js/lib/array-set.js +121 -0
  85. package/dist/server/node_modules/source-map-js/lib/base64-vlq.js +140 -0
  86. package/dist/server/node_modules/source-map-js/lib/base64.js +67 -0
  87. package/dist/server/node_modules/source-map-js/lib/binary-search.js +111 -0
  88. package/dist/server/node_modules/source-map-js/lib/mapping-list.js +79 -0
  89. package/dist/server/node_modules/source-map-js/lib/quick-sort.js +132 -0
  90. package/dist/server/node_modules/source-map-js/lib/source-map-consumer.js +1188 -0
  91. package/dist/server/node_modules/source-map-js/lib/source-map-generator.js +444 -0
  92. package/dist/server/node_modules/source-map-js/lib/source-node.js +413 -0
  93. package/dist/server/node_modules/source-map-js/lib/util.js +594 -0
  94. package/dist/server/node_modules/source-map-js/package.json +71 -0
  95. package/dist/server/node_modules/source-map-js/source-map.js +8 -0
  96. package/dist/server/node_modules/structured-clone-es/dist/index.mjs +285 -0
  97. package/dist/server/node_modules/structured-clone-es/package.json +56 -0
  98. package/dist/server/node_modules/ufo/dist/index.mjs +638 -0
  99. package/dist/server/node_modules/ufo/package.json +47 -0
  100. package/dist/server/node_modules/unhead/dist/index.mjs +9 -0
  101. package/dist/server/node_modules/unhead/dist/parser.mjs +508 -0
  102. package/dist/server/node_modules/unhead/dist/plugins.mjs +101 -0
  103. package/dist/server/node_modules/unhead/dist/scripts.mjs +30 -0
  104. package/dist/server/node_modules/unhead/dist/server.mjs +182 -0
  105. package/dist/server/node_modules/unhead/dist/shared/unhead.B578PsDV.mjs +266 -0
  106. package/dist/server/node_modules/unhead/dist/shared/unhead.BPM0-cfG.mjs +44 -0
  107. package/dist/server/node_modules/unhead/dist/shared/unhead.BYvz9V1x.mjs +43 -0
  108. package/dist/server/node_modules/unhead/dist/shared/unhead.BpRRHAhY.mjs +194 -0
  109. package/dist/server/node_modules/unhead/dist/shared/unhead.CApf5sj3.mjs +148 -0
  110. package/dist/server/node_modules/unhead/dist/shared/unhead.DH45uomy.mjs +180 -0
  111. package/dist/server/node_modules/unhead/dist/shared/unhead.DQc16pHI.mjs +196 -0
  112. package/dist/server/node_modules/unhead/dist/shared/unhead.DZbvapt-.mjs +70 -0
  113. package/dist/server/node_modules/unhead/dist/shared/unhead.Djo8ep_Y.mjs +166 -0
  114. package/dist/server/node_modules/unhead/dist/shared/unhead.yem5I2v_.mjs +38 -0
  115. package/dist/server/node_modules/unhead/dist/utils.mjs +5 -0
  116. package/dist/server/node_modules/unhead/package.json +105 -0
  117. package/dist/server/node_modules/vue/dist/vue.cjs.js +80 -0
  118. package/dist/server/node_modules/vue/dist/vue.cjs.prod.js +66 -0
  119. package/dist/server/node_modules/vue/index.js +7 -0
  120. package/dist/server/node_modules/vue/index.mjs +1 -0
  121. package/dist/server/node_modules/vue/package.json +112 -0
  122. package/dist/server/node_modules/vue/server-renderer/index.mjs +1 -0
  123. package/dist/server/node_modules/vue-bundle-renderer/dist/runtime.mjs +301 -0
  124. package/dist/server/node_modules/vue-bundle-renderer/package.json +55 -0
  125. package/dist/server/node_modules/ws/lib/buffer-util.js +131 -0
  126. package/dist/server/node_modules/ws/lib/constants.js +19 -0
  127. package/dist/server/node_modules/ws/lib/event-target.js +292 -0
  128. package/dist/server/node_modules/ws/lib/extension.js +203 -0
  129. package/dist/server/node_modules/ws/lib/limiter.js +55 -0
  130. package/dist/server/node_modules/ws/lib/permessage-deflate.js +528 -0
  131. package/dist/server/node_modules/ws/lib/receiver.js +706 -0
  132. package/dist/server/node_modules/ws/lib/sender.js +602 -0
  133. package/dist/server/node_modules/ws/lib/stream.js +161 -0
  134. package/dist/server/node_modules/ws/lib/subprotocol.js +62 -0
  135. package/dist/server/node_modules/ws/lib/validation.js +152 -0
  136. package/dist/server/node_modules/ws/lib/websocket-server.js +554 -0
  137. package/dist/server/node_modules/ws/lib/websocket.js +1393 -0
  138. package/dist/server/node_modules/ws/package.json +69 -0
  139. package/dist/server/node_modules/ws/wrapper.mjs +8 -0
  140. package/dist/server/package.json +32 -0
  141. package/dist/shared/yansu.DmdNF4qf.mjs +151 -0
  142. package/package.json +50 -0
@@ -0,0 +1,1393 @@
1
+ /* eslint no-unused-vars: ["error", { "varsIgnorePattern": "^Duplex|Readable$", "caughtErrors": "none" }] */
2
+
3
+ 'use strict';
4
+
5
+ const EventEmitter = require('events');
6
+ const https = require('https');
7
+ const http = require('http');
8
+ const net = require('net');
9
+ const tls = require('tls');
10
+ const { randomBytes, createHash } = require('crypto');
11
+ const { Duplex, Readable } = require('stream');
12
+ const { URL } = require('url');
13
+
14
+ const PerMessageDeflate = require('./permessage-deflate');
15
+ const Receiver = require('./receiver');
16
+ const Sender = require('./sender');
17
+ const { isBlob } = require('./validation');
18
+
19
+ const {
20
+ BINARY_TYPES,
21
+ CLOSE_TIMEOUT,
22
+ EMPTY_BUFFER,
23
+ GUID,
24
+ kForOnEventAttribute,
25
+ kListener,
26
+ kStatusCode,
27
+ kWebSocket,
28
+ NOOP
29
+ } = require('./constants');
30
+ const {
31
+ EventTarget: { addEventListener, removeEventListener }
32
+ } = require('./event-target');
33
+ const { format, parse } = require('./extension');
34
+ const { toBuffer } = require('./buffer-util');
35
+
36
+ const kAborted = Symbol('kAborted');
37
+ const protocolVersions = [8, 13];
38
+ const readyStates = ['CONNECTING', 'OPEN', 'CLOSING', 'CLOSED'];
39
+ const subprotocolRegex = /^[!#$%&'*+\-.0-9A-Z^_`|a-z~]+$/;
40
+
41
+ /**
42
+ * Class representing a WebSocket.
43
+ *
44
+ * @extends EventEmitter
45
+ */
46
+ class WebSocket extends EventEmitter {
47
+ /**
48
+ * Create a new `WebSocket`.
49
+ *
50
+ * @param {(String|URL)} address The URL to which to connect
51
+ * @param {(String|String[])} [protocols] The subprotocols
52
+ * @param {Object} [options] Connection options
53
+ */
54
+ constructor(address, protocols, options) {
55
+ super();
56
+
57
+ this._binaryType = BINARY_TYPES[0];
58
+ this._closeCode = 1006;
59
+ this._closeFrameReceived = false;
60
+ this._closeFrameSent = false;
61
+ this._closeMessage = EMPTY_BUFFER;
62
+ this._closeTimer = null;
63
+ this._errorEmitted = false;
64
+ this._extensions = {};
65
+ this._paused = false;
66
+ this._protocol = '';
67
+ this._readyState = WebSocket.CONNECTING;
68
+ this._receiver = null;
69
+ this._sender = null;
70
+ this._socket = null;
71
+
72
+ if (address !== null) {
73
+ this._bufferedAmount = 0;
74
+ this._isServer = false;
75
+ this._redirects = 0;
76
+
77
+ if (protocols === undefined) {
78
+ protocols = [];
79
+ } else if (!Array.isArray(protocols)) {
80
+ if (typeof protocols === 'object' && protocols !== null) {
81
+ options = protocols;
82
+ protocols = [];
83
+ } else {
84
+ protocols = [protocols];
85
+ }
86
+ }
87
+
88
+ initAsClient(this, address, protocols, options);
89
+ } else {
90
+ this._autoPong = options.autoPong;
91
+ this._closeTimeout = options.closeTimeout;
92
+ this._isServer = true;
93
+ }
94
+ }
95
+
96
+ /**
97
+ * For historical reasons, the custom "nodebuffer" type is used by the default
98
+ * instead of "blob".
99
+ *
100
+ * @type {String}
101
+ */
102
+ get binaryType() {
103
+ return this._binaryType;
104
+ }
105
+
106
+ set binaryType(type) {
107
+ if (!BINARY_TYPES.includes(type)) return;
108
+
109
+ this._binaryType = type;
110
+
111
+ //
112
+ // Allow to change `binaryType` on the fly.
113
+ //
114
+ if (this._receiver) this._receiver._binaryType = type;
115
+ }
116
+
117
+ /**
118
+ * @type {Number}
119
+ */
120
+ get bufferedAmount() {
121
+ if (!this._socket) return this._bufferedAmount;
122
+
123
+ return this._socket._writableState.length + this._sender._bufferedBytes;
124
+ }
125
+
126
+ /**
127
+ * @type {String}
128
+ */
129
+ get extensions() {
130
+ return Object.keys(this._extensions).join();
131
+ }
132
+
133
+ /**
134
+ * @type {Boolean}
135
+ */
136
+ get isPaused() {
137
+ return this._paused;
138
+ }
139
+
140
+ /**
141
+ * @type {Function}
142
+ */
143
+ /* istanbul ignore next */
144
+ get onclose() {
145
+ return null;
146
+ }
147
+
148
+ /**
149
+ * @type {Function}
150
+ */
151
+ /* istanbul ignore next */
152
+ get onerror() {
153
+ return null;
154
+ }
155
+
156
+ /**
157
+ * @type {Function}
158
+ */
159
+ /* istanbul ignore next */
160
+ get onopen() {
161
+ return null;
162
+ }
163
+
164
+ /**
165
+ * @type {Function}
166
+ */
167
+ /* istanbul ignore next */
168
+ get onmessage() {
169
+ return null;
170
+ }
171
+
172
+ /**
173
+ * @type {String}
174
+ */
175
+ get protocol() {
176
+ return this._protocol;
177
+ }
178
+
179
+ /**
180
+ * @type {Number}
181
+ */
182
+ get readyState() {
183
+ return this._readyState;
184
+ }
185
+
186
+ /**
187
+ * @type {String}
188
+ */
189
+ get url() {
190
+ return this._url;
191
+ }
192
+
193
+ /**
194
+ * Set up the socket and the internal resources.
195
+ *
196
+ * @param {Duplex} socket The network socket between the server and client
197
+ * @param {Buffer} head The first packet of the upgraded stream
198
+ * @param {Object} options Options object
199
+ * @param {Boolean} [options.allowSynchronousEvents=false] Specifies whether
200
+ * any of the `'message'`, `'ping'`, and `'pong'` events can be emitted
201
+ * multiple times in the same tick
202
+ * @param {Function} [options.generateMask] The function used to generate the
203
+ * masking key
204
+ * @param {Number} [options.maxPayload=0] The maximum allowed message size
205
+ * @param {Boolean} [options.skipUTF8Validation=false] Specifies whether or
206
+ * not to skip UTF-8 validation for text and close messages
207
+ * @private
208
+ */
209
+ setSocket(socket, head, options) {
210
+ const receiver = new Receiver({
211
+ allowSynchronousEvents: options.allowSynchronousEvents,
212
+ binaryType: this.binaryType,
213
+ extensions: this._extensions,
214
+ isServer: this._isServer,
215
+ maxPayload: options.maxPayload,
216
+ skipUTF8Validation: options.skipUTF8Validation
217
+ });
218
+
219
+ const sender = new Sender(socket, this._extensions, options.generateMask);
220
+
221
+ this._receiver = receiver;
222
+ this._sender = sender;
223
+ this._socket = socket;
224
+
225
+ receiver[kWebSocket] = this;
226
+ sender[kWebSocket] = this;
227
+ socket[kWebSocket] = this;
228
+
229
+ receiver.on('conclude', receiverOnConclude);
230
+ receiver.on('drain', receiverOnDrain);
231
+ receiver.on('error', receiverOnError);
232
+ receiver.on('message', receiverOnMessage);
233
+ receiver.on('ping', receiverOnPing);
234
+ receiver.on('pong', receiverOnPong);
235
+
236
+ sender.onerror = senderOnError;
237
+
238
+ //
239
+ // These methods may not be available if `socket` is just a `Duplex`.
240
+ //
241
+ if (socket.setTimeout) socket.setTimeout(0);
242
+ if (socket.setNoDelay) socket.setNoDelay();
243
+
244
+ if (head.length > 0) socket.unshift(head);
245
+
246
+ socket.on('close', socketOnClose);
247
+ socket.on('data', socketOnData);
248
+ socket.on('end', socketOnEnd);
249
+ socket.on('error', socketOnError);
250
+
251
+ this._readyState = WebSocket.OPEN;
252
+ this.emit('open');
253
+ }
254
+
255
+ /**
256
+ * Emit the `'close'` event.
257
+ *
258
+ * @private
259
+ */
260
+ emitClose() {
261
+ if (!this._socket) {
262
+ this._readyState = WebSocket.CLOSED;
263
+ this.emit('close', this._closeCode, this._closeMessage);
264
+ return;
265
+ }
266
+
267
+ if (this._extensions[PerMessageDeflate.extensionName]) {
268
+ this._extensions[PerMessageDeflate.extensionName].cleanup();
269
+ }
270
+
271
+ this._receiver.removeAllListeners();
272
+ this._readyState = WebSocket.CLOSED;
273
+ this.emit('close', this._closeCode, this._closeMessage);
274
+ }
275
+
276
+ /**
277
+ * Start a closing handshake.
278
+ *
279
+ * +----------+ +-----------+ +----------+
280
+ * - - -|ws.close()|-->|close frame|-->|ws.close()|- - -
281
+ * | +----------+ +-----------+ +----------+ |
282
+ * +----------+ +-----------+ |
283
+ * CLOSING |ws.close()|<--|close frame|<--+-----+ CLOSING
284
+ * +----------+ +-----------+ |
285
+ * | | | +---+ |
286
+ * +------------------------+-->|fin| - - - -
287
+ * | +---+ | +---+
288
+ * - - - - -|fin|<---------------------+
289
+ * +---+
290
+ *
291
+ * @param {Number} [code] Status code explaining why the connection is closing
292
+ * @param {(String|Buffer)} [data] The reason why the connection is
293
+ * closing
294
+ * @public
295
+ */
296
+ close(code, data) {
297
+ if (this.readyState === WebSocket.CLOSED) return;
298
+ if (this.readyState === WebSocket.CONNECTING) {
299
+ const msg = 'WebSocket was closed before the connection was established';
300
+ abortHandshake(this, this._req, msg);
301
+ return;
302
+ }
303
+
304
+ if (this.readyState === WebSocket.CLOSING) {
305
+ if (
306
+ this._closeFrameSent &&
307
+ (this._closeFrameReceived || this._receiver._writableState.errorEmitted)
308
+ ) {
309
+ this._socket.end();
310
+ }
311
+
312
+ return;
313
+ }
314
+
315
+ this._readyState = WebSocket.CLOSING;
316
+ this._sender.close(code, data, !this._isServer, (err) => {
317
+ //
318
+ // This error is handled by the `'error'` listener on the socket. We only
319
+ // want to know if the close frame has been sent here.
320
+ //
321
+ if (err) return;
322
+
323
+ this._closeFrameSent = true;
324
+
325
+ if (
326
+ this._closeFrameReceived ||
327
+ this._receiver._writableState.errorEmitted
328
+ ) {
329
+ this._socket.end();
330
+ }
331
+ });
332
+
333
+ setCloseTimer(this);
334
+ }
335
+
336
+ /**
337
+ * Pause the socket.
338
+ *
339
+ * @public
340
+ */
341
+ pause() {
342
+ if (
343
+ this.readyState === WebSocket.CONNECTING ||
344
+ this.readyState === WebSocket.CLOSED
345
+ ) {
346
+ return;
347
+ }
348
+
349
+ this._paused = true;
350
+ this._socket.pause();
351
+ }
352
+
353
+ /**
354
+ * Send a ping.
355
+ *
356
+ * @param {*} [data] The data to send
357
+ * @param {Boolean} [mask] Indicates whether or not to mask `data`
358
+ * @param {Function} [cb] Callback which is executed when the ping is sent
359
+ * @public
360
+ */
361
+ ping(data, mask, cb) {
362
+ if (this.readyState === WebSocket.CONNECTING) {
363
+ throw new Error('WebSocket is not open: readyState 0 (CONNECTING)');
364
+ }
365
+
366
+ if (typeof data === 'function') {
367
+ cb = data;
368
+ data = mask = undefined;
369
+ } else if (typeof mask === 'function') {
370
+ cb = mask;
371
+ mask = undefined;
372
+ }
373
+
374
+ if (typeof data === 'number') data = data.toString();
375
+
376
+ if (this.readyState !== WebSocket.OPEN) {
377
+ sendAfterClose(this, data, cb);
378
+ return;
379
+ }
380
+
381
+ if (mask === undefined) mask = !this._isServer;
382
+ this._sender.ping(data || EMPTY_BUFFER, mask, cb);
383
+ }
384
+
385
+ /**
386
+ * Send a pong.
387
+ *
388
+ * @param {*} [data] The data to send
389
+ * @param {Boolean} [mask] Indicates whether or not to mask `data`
390
+ * @param {Function} [cb] Callback which is executed when the pong is sent
391
+ * @public
392
+ */
393
+ pong(data, mask, cb) {
394
+ if (this.readyState === WebSocket.CONNECTING) {
395
+ throw new Error('WebSocket is not open: readyState 0 (CONNECTING)');
396
+ }
397
+
398
+ if (typeof data === 'function') {
399
+ cb = data;
400
+ data = mask = undefined;
401
+ } else if (typeof mask === 'function') {
402
+ cb = mask;
403
+ mask = undefined;
404
+ }
405
+
406
+ if (typeof data === 'number') data = data.toString();
407
+
408
+ if (this.readyState !== WebSocket.OPEN) {
409
+ sendAfterClose(this, data, cb);
410
+ return;
411
+ }
412
+
413
+ if (mask === undefined) mask = !this._isServer;
414
+ this._sender.pong(data || EMPTY_BUFFER, mask, cb);
415
+ }
416
+
417
+ /**
418
+ * Resume the socket.
419
+ *
420
+ * @public
421
+ */
422
+ resume() {
423
+ if (
424
+ this.readyState === WebSocket.CONNECTING ||
425
+ this.readyState === WebSocket.CLOSED
426
+ ) {
427
+ return;
428
+ }
429
+
430
+ this._paused = false;
431
+ if (!this._receiver._writableState.needDrain) this._socket.resume();
432
+ }
433
+
434
+ /**
435
+ * Send a data message.
436
+ *
437
+ * @param {*} data The message to send
438
+ * @param {Object} [options] Options object
439
+ * @param {Boolean} [options.binary] Specifies whether `data` is binary or
440
+ * text
441
+ * @param {Boolean} [options.compress] Specifies whether or not to compress
442
+ * `data`
443
+ * @param {Boolean} [options.fin=true] Specifies whether the fragment is the
444
+ * last one
445
+ * @param {Boolean} [options.mask] Specifies whether or not to mask `data`
446
+ * @param {Function} [cb] Callback which is executed when data is written out
447
+ * @public
448
+ */
449
+ send(data, options, cb) {
450
+ if (this.readyState === WebSocket.CONNECTING) {
451
+ throw new Error('WebSocket is not open: readyState 0 (CONNECTING)');
452
+ }
453
+
454
+ if (typeof options === 'function') {
455
+ cb = options;
456
+ options = {};
457
+ }
458
+
459
+ if (typeof data === 'number') data = data.toString();
460
+
461
+ if (this.readyState !== WebSocket.OPEN) {
462
+ sendAfterClose(this, data, cb);
463
+ return;
464
+ }
465
+
466
+ const opts = {
467
+ binary: typeof data !== 'string',
468
+ mask: !this._isServer,
469
+ compress: true,
470
+ fin: true,
471
+ ...options
472
+ };
473
+
474
+ if (!this._extensions[PerMessageDeflate.extensionName]) {
475
+ opts.compress = false;
476
+ }
477
+
478
+ this._sender.send(data || EMPTY_BUFFER, opts, cb);
479
+ }
480
+
481
+ /**
482
+ * Forcibly close the connection.
483
+ *
484
+ * @public
485
+ */
486
+ terminate() {
487
+ if (this.readyState === WebSocket.CLOSED) return;
488
+ if (this.readyState === WebSocket.CONNECTING) {
489
+ const msg = 'WebSocket was closed before the connection was established';
490
+ abortHandshake(this, this._req, msg);
491
+ return;
492
+ }
493
+
494
+ if (this._socket) {
495
+ this._readyState = WebSocket.CLOSING;
496
+ this._socket.destroy();
497
+ }
498
+ }
499
+ }
500
+
501
+ /**
502
+ * @constant {Number} CONNECTING
503
+ * @memberof WebSocket
504
+ */
505
+ Object.defineProperty(WebSocket, 'CONNECTING', {
506
+ enumerable: true,
507
+ value: readyStates.indexOf('CONNECTING')
508
+ });
509
+
510
+ /**
511
+ * @constant {Number} CONNECTING
512
+ * @memberof WebSocket.prototype
513
+ */
514
+ Object.defineProperty(WebSocket.prototype, 'CONNECTING', {
515
+ enumerable: true,
516
+ value: readyStates.indexOf('CONNECTING')
517
+ });
518
+
519
+ /**
520
+ * @constant {Number} OPEN
521
+ * @memberof WebSocket
522
+ */
523
+ Object.defineProperty(WebSocket, 'OPEN', {
524
+ enumerable: true,
525
+ value: readyStates.indexOf('OPEN')
526
+ });
527
+
528
+ /**
529
+ * @constant {Number} OPEN
530
+ * @memberof WebSocket.prototype
531
+ */
532
+ Object.defineProperty(WebSocket.prototype, 'OPEN', {
533
+ enumerable: true,
534
+ value: readyStates.indexOf('OPEN')
535
+ });
536
+
537
+ /**
538
+ * @constant {Number} CLOSING
539
+ * @memberof WebSocket
540
+ */
541
+ Object.defineProperty(WebSocket, 'CLOSING', {
542
+ enumerable: true,
543
+ value: readyStates.indexOf('CLOSING')
544
+ });
545
+
546
+ /**
547
+ * @constant {Number} CLOSING
548
+ * @memberof WebSocket.prototype
549
+ */
550
+ Object.defineProperty(WebSocket.prototype, 'CLOSING', {
551
+ enumerable: true,
552
+ value: readyStates.indexOf('CLOSING')
553
+ });
554
+
555
+ /**
556
+ * @constant {Number} CLOSED
557
+ * @memberof WebSocket
558
+ */
559
+ Object.defineProperty(WebSocket, 'CLOSED', {
560
+ enumerable: true,
561
+ value: readyStates.indexOf('CLOSED')
562
+ });
563
+
564
+ /**
565
+ * @constant {Number} CLOSED
566
+ * @memberof WebSocket.prototype
567
+ */
568
+ Object.defineProperty(WebSocket.prototype, 'CLOSED', {
569
+ enumerable: true,
570
+ value: readyStates.indexOf('CLOSED')
571
+ });
572
+
573
+ [
574
+ 'binaryType',
575
+ 'bufferedAmount',
576
+ 'extensions',
577
+ 'isPaused',
578
+ 'protocol',
579
+ 'readyState',
580
+ 'url'
581
+ ].forEach((property) => {
582
+ Object.defineProperty(WebSocket.prototype, property, { enumerable: true });
583
+ });
584
+
585
+ //
586
+ // Add the `onopen`, `onerror`, `onclose`, and `onmessage` attributes.
587
+ // See https://html.spec.whatwg.org/multipage/comms.html#the-websocket-interface
588
+ //
589
+ ['open', 'error', 'close', 'message'].forEach((method) => {
590
+ Object.defineProperty(WebSocket.prototype, `on${method}`, {
591
+ enumerable: true,
592
+ get() {
593
+ for (const listener of this.listeners(method)) {
594
+ if (listener[kForOnEventAttribute]) return listener[kListener];
595
+ }
596
+
597
+ return null;
598
+ },
599
+ set(handler) {
600
+ for (const listener of this.listeners(method)) {
601
+ if (listener[kForOnEventAttribute]) {
602
+ this.removeListener(method, listener);
603
+ break;
604
+ }
605
+ }
606
+
607
+ if (typeof handler !== 'function') return;
608
+
609
+ this.addEventListener(method, handler, {
610
+ [kForOnEventAttribute]: true
611
+ });
612
+ }
613
+ });
614
+ });
615
+
616
+ WebSocket.prototype.addEventListener = addEventListener;
617
+ WebSocket.prototype.removeEventListener = removeEventListener;
618
+
619
+ module.exports = WebSocket;
620
+
621
+ /**
622
+ * Initialize a WebSocket client.
623
+ *
624
+ * @param {WebSocket} websocket The client to initialize
625
+ * @param {(String|URL)} address The URL to which to connect
626
+ * @param {Array} protocols The subprotocols
627
+ * @param {Object} [options] Connection options
628
+ * @param {Boolean} [options.allowSynchronousEvents=true] Specifies whether any
629
+ * of the `'message'`, `'ping'`, and `'pong'` events can be emitted multiple
630
+ * times in the same tick
631
+ * @param {Boolean} [options.autoPong=true] Specifies whether or not to
632
+ * automatically send a pong in response to a ping
633
+ * @param {Number} [options.closeTimeout=30000] Duration in milliseconds to wait
634
+ * for the closing handshake to finish after `websocket.close()` is called
635
+ * @param {Function} [options.finishRequest] A function which can be used to
636
+ * customize the headers of each http request before it is sent
637
+ * @param {Boolean} [options.followRedirects=false] Whether or not to follow
638
+ * redirects
639
+ * @param {Function} [options.generateMask] The function used to generate the
640
+ * masking key
641
+ * @param {Number} [options.handshakeTimeout] Timeout in milliseconds for the
642
+ * handshake request
643
+ * @param {Number} [options.maxPayload=104857600] The maximum allowed message
644
+ * size
645
+ * @param {Number} [options.maxRedirects=10] The maximum number of redirects
646
+ * allowed
647
+ * @param {String} [options.origin] Value of the `Origin` or
648
+ * `Sec-WebSocket-Origin` header
649
+ * @param {(Boolean|Object)} [options.perMessageDeflate=true] Enable/disable
650
+ * permessage-deflate
651
+ * @param {Number} [options.protocolVersion=13] Value of the
652
+ * `Sec-WebSocket-Version` header
653
+ * @param {Boolean} [options.skipUTF8Validation=false] Specifies whether or
654
+ * not to skip UTF-8 validation for text and close messages
655
+ * @private
656
+ */
657
+ function initAsClient(websocket, address, protocols, options) {
658
+ const opts = {
659
+ allowSynchronousEvents: true,
660
+ autoPong: true,
661
+ closeTimeout: CLOSE_TIMEOUT,
662
+ protocolVersion: protocolVersions[1],
663
+ maxPayload: 100 * 1024 * 1024,
664
+ skipUTF8Validation: false,
665
+ perMessageDeflate: true,
666
+ followRedirects: false,
667
+ maxRedirects: 10,
668
+ ...options,
669
+ socketPath: undefined,
670
+ hostname: undefined,
671
+ protocol: undefined,
672
+ timeout: undefined,
673
+ method: 'GET',
674
+ host: undefined,
675
+ path: undefined,
676
+ port: undefined
677
+ };
678
+
679
+ websocket._autoPong = opts.autoPong;
680
+ websocket._closeTimeout = opts.closeTimeout;
681
+
682
+ if (!protocolVersions.includes(opts.protocolVersion)) {
683
+ throw new RangeError(
684
+ `Unsupported protocol version: ${opts.protocolVersion} ` +
685
+ `(supported versions: ${protocolVersions.join(', ')})`
686
+ );
687
+ }
688
+
689
+ let parsedUrl;
690
+
691
+ if (address instanceof URL) {
692
+ parsedUrl = address;
693
+ } else {
694
+ try {
695
+ parsedUrl = new URL(address);
696
+ } catch (e) {
697
+ throw new SyntaxError(`Invalid URL: ${address}`);
698
+ }
699
+ }
700
+
701
+ if (parsedUrl.protocol === 'http:') {
702
+ parsedUrl.protocol = 'ws:';
703
+ } else if (parsedUrl.protocol === 'https:') {
704
+ parsedUrl.protocol = 'wss:';
705
+ }
706
+
707
+ websocket._url = parsedUrl.href;
708
+
709
+ const isSecure = parsedUrl.protocol === 'wss:';
710
+ const isIpcUrl = parsedUrl.protocol === 'ws+unix:';
711
+ let invalidUrlMessage;
712
+
713
+ if (parsedUrl.protocol !== 'ws:' && !isSecure && !isIpcUrl) {
714
+ invalidUrlMessage =
715
+ 'The URL\'s protocol must be one of "ws:", "wss:", ' +
716
+ '"http:", "https:", or "ws+unix:"';
717
+ } else if (isIpcUrl && !parsedUrl.pathname) {
718
+ invalidUrlMessage = "The URL's pathname is empty";
719
+ } else if (parsedUrl.hash) {
720
+ invalidUrlMessage = 'The URL contains a fragment identifier';
721
+ }
722
+
723
+ if (invalidUrlMessage) {
724
+ const err = new SyntaxError(invalidUrlMessage);
725
+
726
+ if (websocket._redirects === 0) {
727
+ throw err;
728
+ } else {
729
+ emitErrorAndClose(websocket, err);
730
+ return;
731
+ }
732
+ }
733
+
734
+ const defaultPort = isSecure ? 443 : 80;
735
+ const key = randomBytes(16).toString('base64');
736
+ const request = isSecure ? https.request : http.request;
737
+ const protocolSet = new Set();
738
+ let perMessageDeflate;
739
+
740
+ opts.createConnection =
741
+ opts.createConnection || (isSecure ? tlsConnect : netConnect);
742
+ opts.defaultPort = opts.defaultPort || defaultPort;
743
+ opts.port = parsedUrl.port || defaultPort;
744
+ opts.host = parsedUrl.hostname.startsWith('[')
745
+ ? parsedUrl.hostname.slice(1, -1)
746
+ : parsedUrl.hostname;
747
+ opts.headers = {
748
+ ...opts.headers,
749
+ 'Sec-WebSocket-Version': opts.protocolVersion,
750
+ 'Sec-WebSocket-Key': key,
751
+ Connection: 'Upgrade',
752
+ Upgrade: 'websocket'
753
+ };
754
+ opts.path = parsedUrl.pathname + parsedUrl.search;
755
+ opts.timeout = opts.handshakeTimeout;
756
+
757
+ if (opts.perMessageDeflate) {
758
+ perMessageDeflate = new PerMessageDeflate(
759
+ opts.perMessageDeflate !== true ? opts.perMessageDeflate : {},
760
+ false,
761
+ opts.maxPayload
762
+ );
763
+ opts.headers['Sec-WebSocket-Extensions'] = format({
764
+ [PerMessageDeflate.extensionName]: perMessageDeflate.offer()
765
+ });
766
+ }
767
+ if (protocols.length) {
768
+ for (const protocol of protocols) {
769
+ if (
770
+ typeof protocol !== 'string' ||
771
+ !subprotocolRegex.test(protocol) ||
772
+ protocolSet.has(protocol)
773
+ ) {
774
+ throw new SyntaxError(
775
+ 'An invalid or duplicated subprotocol was specified'
776
+ );
777
+ }
778
+
779
+ protocolSet.add(protocol);
780
+ }
781
+
782
+ opts.headers['Sec-WebSocket-Protocol'] = protocols.join(',');
783
+ }
784
+ if (opts.origin) {
785
+ if (opts.protocolVersion < 13) {
786
+ opts.headers['Sec-WebSocket-Origin'] = opts.origin;
787
+ } else {
788
+ opts.headers.Origin = opts.origin;
789
+ }
790
+ }
791
+ if (parsedUrl.username || parsedUrl.password) {
792
+ opts.auth = `${parsedUrl.username}:${parsedUrl.password}`;
793
+ }
794
+
795
+ if (isIpcUrl) {
796
+ const parts = opts.path.split(':');
797
+
798
+ opts.socketPath = parts[0];
799
+ opts.path = parts[1];
800
+ }
801
+
802
+ let req;
803
+
804
+ if (opts.followRedirects) {
805
+ if (websocket._redirects === 0) {
806
+ websocket._originalIpc = isIpcUrl;
807
+ websocket._originalSecure = isSecure;
808
+ websocket._originalHostOrSocketPath = isIpcUrl
809
+ ? opts.socketPath
810
+ : parsedUrl.host;
811
+
812
+ const headers = options && options.headers;
813
+
814
+ //
815
+ // Shallow copy the user provided options so that headers can be changed
816
+ // without mutating the original object.
817
+ //
818
+ options = { ...options, headers: {} };
819
+
820
+ if (headers) {
821
+ for (const [key, value] of Object.entries(headers)) {
822
+ options.headers[key.toLowerCase()] = value;
823
+ }
824
+ }
825
+ } else if (websocket.listenerCount('redirect') === 0) {
826
+ const isSameHost = isIpcUrl
827
+ ? websocket._originalIpc
828
+ ? opts.socketPath === websocket._originalHostOrSocketPath
829
+ : false
830
+ : websocket._originalIpc
831
+ ? false
832
+ : parsedUrl.host === websocket._originalHostOrSocketPath;
833
+
834
+ if (!isSameHost || (websocket._originalSecure && !isSecure)) {
835
+ //
836
+ // Match curl 7.77.0 behavior and drop the following headers. These
837
+ // headers are also dropped when following a redirect to a subdomain.
838
+ //
839
+ delete opts.headers.authorization;
840
+ delete opts.headers.cookie;
841
+
842
+ if (!isSameHost) delete opts.headers.host;
843
+
844
+ opts.auth = undefined;
845
+ }
846
+ }
847
+
848
+ //
849
+ // Match curl 7.77.0 behavior and make the first `Authorization` header win.
850
+ // If the `Authorization` header is set, then there is nothing to do as it
851
+ // will take precedence.
852
+ //
853
+ if (opts.auth && !options.headers.authorization) {
854
+ options.headers.authorization =
855
+ 'Basic ' + Buffer.from(opts.auth).toString('base64');
856
+ }
857
+
858
+ req = websocket._req = request(opts);
859
+
860
+ if (websocket._redirects) {
861
+ //
862
+ // Unlike what is done for the `'upgrade'` event, no early exit is
863
+ // triggered here if the user calls `websocket.close()` or
864
+ // `websocket.terminate()` from a listener of the `'redirect'` event. This
865
+ // is because the user can also call `request.destroy()` with an error
866
+ // before calling `websocket.close()` or `websocket.terminate()` and this
867
+ // would result in an error being emitted on the `request` object with no
868
+ // `'error'` event listeners attached.
869
+ //
870
+ websocket.emit('redirect', websocket.url, req);
871
+ }
872
+ } else {
873
+ req = websocket._req = request(opts);
874
+ }
875
+
876
+ if (opts.timeout) {
877
+ req.on('timeout', () => {
878
+ abortHandshake(websocket, req, 'Opening handshake has timed out');
879
+ });
880
+ }
881
+
882
+ req.on('error', (err) => {
883
+ if (req === null || req[kAborted]) return;
884
+
885
+ req = websocket._req = null;
886
+ emitErrorAndClose(websocket, err);
887
+ });
888
+
889
+ req.on('response', (res) => {
890
+ const location = res.headers.location;
891
+ const statusCode = res.statusCode;
892
+
893
+ if (
894
+ location &&
895
+ opts.followRedirects &&
896
+ statusCode >= 300 &&
897
+ statusCode < 400
898
+ ) {
899
+ if (++websocket._redirects > opts.maxRedirects) {
900
+ abortHandshake(websocket, req, 'Maximum redirects exceeded');
901
+ return;
902
+ }
903
+
904
+ req.abort();
905
+
906
+ let addr;
907
+
908
+ try {
909
+ addr = new URL(location, address);
910
+ } catch (e) {
911
+ const err = new SyntaxError(`Invalid URL: ${location}`);
912
+ emitErrorAndClose(websocket, err);
913
+ return;
914
+ }
915
+
916
+ initAsClient(websocket, addr, protocols, options);
917
+ } else if (!websocket.emit('unexpected-response', req, res)) {
918
+ abortHandshake(
919
+ websocket,
920
+ req,
921
+ `Unexpected server response: ${res.statusCode}`
922
+ );
923
+ }
924
+ });
925
+
926
+ req.on('upgrade', (res, socket, head) => {
927
+ websocket.emit('upgrade', res);
928
+
929
+ //
930
+ // The user may have closed the connection from a listener of the
931
+ // `'upgrade'` event.
932
+ //
933
+ if (websocket.readyState !== WebSocket.CONNECTING) return;
934
+
935
+ req = websocket._req = null;
936
+
937
+ const upgrade = res.headers.upgrade;
938
+
939
+ if (upgrade === undefined || upgrade.toLowerCase() !== 'websocket') {
940
+ abortHandshake(websocket, socket, 'Invalid Upgrade header');
941
+ return;
942
+ }
943
+
944
+ const digest = createHash('sha1')
945
+ .update(key + GUID)
946
+ .digest('base64');
947
+
948
+ if (res.headers['sec-websocket-accept'] !== digest) {
949
+ abortHandshake(websocket, socket, 'Invalid Sec-WebSocket-Accept header');
950
+ return;
951
+ }
952
+
953
+ const serverProt = res.headers['sec-websocket-protocol'];
954
+ let protError;
955
+
956
+ if (serverProt !== undefined) {
957
+ if (!protocolSet.size) {
958
+ protError = 'Server sent a subprotocol but none was requested';
959
+ } else if (!protocolSet.has(serverProt)) {
960
+ protError = 'Server sent an invalid subprotocol';
961
+ }
962
+ } else if (protocolSet.size) {
963
+ protError = 'Server sent no subprotocol';
964
+ }
965
+
966
+ if (protError) {
967
+ abortHandshake(websocket, socket, protError);
968
+ return;
969
+ }
970
+
971
+ if (serverProt) websocket._protocol = serverProt;
972
+
973
+ const secWebSocketExtensions = res.headers['sec-websocket-extensions'];
974
+
975
+ if (secWebSocketExtensions !== undefined) {
976
+ if (!perMessageDeflate) {
977
+ const message =
978
+ 'Server sent a Sec-WebSocket-Extensions header but no extension ' +
979
+ 'was requested';
980
+ abortHandshake(websocket, socket, message);
981
+ return;
982
+ }
983
+
984
+ let extensions;
985
+
986
+ try {
987
+ extensions = parse(secWebSocketExtensions);
988
+ } catch (err) {
989
+ const message = 'Invalid Sec-WebSocket-Extensions header';
990
+ abortHandshake(websocket, socket, message);
991
+ return;
992
+ }
993
+
994
+ const extensionNames = Object.keys(extensions);
995
+
996
+ if (
997
+ extensionNames.length !== 1 ||
998
+ extensionNames[0] !== PerMessageDeflate.extensionName
999
+ ) {
1000
+ const message = 'Server indicated an extension that was not requested';
1001
+ abortHandshake(websocket, socket, message);
1002
+ return;
1003
+ }
1004
+
1005
+ try {
1006
+ perMessageDeflate.accept(extensions[PerMessageDeflate.extensionName]);
1007
+ } catch (err) {
1008
+ const message = 'Invalid Sec-WebSocket-Extensions header';
1009
+ abortHandshake(websocket, socket, message);
1010
+ return;
1011
+ }
1012
+
1013
+ websocket._extensions[PerMessageDeflate.extensionName] =
1014
+ perMessageDeflate;
1015
+ }
1016
+
1017
+ websocket.setSocket(socket, head, {
1018
+ allowSynchronousEvents: opts.allowSynchronousEvents,
1019
+ generateMask: opts.generateMask,
1020
+ maxPayload: opts.maxPayload,
1021
+ skipUTF8Validation: opts.skipUTF8Validation
1022
+ });
1023
+ });
1024
+
1025
+ if (opts.finishRequest) {
1026
+ opts.finishRequest(req, websocket);
1027
+ } else {
1028
+ req.end();
1029
+ }
1030
+ }
1031
+
1032
+ /**
1033
+ * Emit the `'error'` and `'close'` events.
1034
+ *
1035
+ * @param {WebSocket} websocket The WebSocket instance
1036
+ * @param {Error} The error to emit
1037
+ * @private
1038
+ */
1039
+ function emitErrorAndClose(websocket, err) {
1040
+ websocket._readyState = WebSocket.CLOSING;
1041
+ //
1042
+ // The following assignment is practically useless and is done only for
1043
+ // consistency.
1044
+ //
1045
+ websocket._errorEmitted = true;
1046
+ websocket.emit('error', err);
1047
+ websocket.emitClose();
1048
+ }
1049
+
1050
+ /**
1051
+ * Create a `net.Socket` and initiate a connection.
1052
+ *
1053
+ * @param {Object} options Connection options
1054
+ * @return {net.Socket} The newly created socket used to start the connection
1055
+ * @private
1056
+ */
1057
+ function netConnect(options) {
1058
+ options.path = options.socketPath;
1059
+ return net.connect(options);
1060
+ }
1061
+
1062
+ /**
1063
+ * Create a `tls.TLSSocket` and initiate a connection.
1064
+ *
1065
+ * @param {Object} options Connection options
1066
+ * @return {tls.TLSSocket} The newly created socket used to start the connection
1067
+ * @private
1068
+ */
1069
+ function tlsConnect(options) {
1070
+ options.path = undefined;
1071
+
1072
+ if (!options.servername && options.servername !== '') {
1073
+ options.servername = net.isIP(options.host) ? '' : options.host;
1074
+ }
1075
+
1076
+ return tls.connect(options);
1077
+ }
1078
+
1079
+ /**
1080
+ * Abort the handshake and emit an error.
1081
+ *
1082
+ * @param {WebSocket} websocket The WebSocket instance
1083
+ * @param {(http.ClientRequest|net.Socket|tls.Socket)} stream The request to
1084
+ * abort or the socket to destroy
1085
+ * @param {String} message The error message
1086
+ * @private
1087
+ */
1088
+ function abortHandshake(websocket, stream, message) {
1089
+ websocket._readyState = WebSocket.CLOSING;
1090
+
1091
+ const err = new Error(message);
1092
+ Error.captureStackTrace(err, abortHandshake);
1093
+
1094
+ if (stream.setHeader) {
1095
+ stream[kAborted] = true;
1096
+ stream.abort();
1097
+
1098
+ if (stream.socket && !stream.socket.destroyed) {
1099
+ //
1100
+ // On Node.js >= 14.3.0 `request.abort()` does not destroy the socket if
1101
+ // called after the request completed. See
1102
+ // https://github.com/websockets/ws/issues/1869.
1103
+ //
1104
+ stream.socket.destroy();
1105
+ }
1106
+
1107
+ process.nextTick(emitErrorAndClose, websocket, err);
1108
+ } else {
1109
+ stream.destroy(err);
1110
+ stream.once('error', websocket.emit.bind(websocket, 'error'));
1111
+ stream.once('close', websocket.emitClose.bind(websocket));
1112
+ }
1113
+ }
1114
+
1115
+ /**
1116
+ * Handle cases where the `ping()`, `pong()`, or `send()` methods are called
1117
+ * when the `readyState` attribute is `CLOSING` or `CLOSED`.
1118
+ *
1119
+ * @param {WebSocket} websocket The WebSocket instance
1120
+ * @param {*} [data] The data to send
1121
+ * @param {Function} [cb] Callback
1122
+ * @private
1123
+ */
1124
+ function sendAfterClose(websocket, data, cb) {
1125
+ if (data) {
1126
+ const length = isBlob(data) ? data.size : toBuffer(data).length;
1127
+
1128
+ //
1129
+ // The `_bufferedAmount` property is used only when the peer is a client and
1130
+ // the opening handshake fails. Under these circumstances, in fact, the
1131
+ // `setSocket()` method is not called, so the `_socket` and `_sender`
1132
+ // properties are set to `null`.
1133
+ //
1134
+ if (websocket._socket) websocket._sender._bufferedBytes += length;
1135
+ else websocket._bufferedAmount += length;
1136
+ }
1137
+
1138
+ if (cb) {
1139
+ const err = new Error(
1140
+ `WebSocket is not open: readyState ${websocket.readyState} ` +
1141
+ `(${readyStates[websocket.readyState]})`
1142
+ );
1143
+ process.nextTick(cb, err);
1144
+ }
1145
+ }
1146
+
1147
+ /**
1148
+ * The listener of the `Receiver` `'conclude'` event.
1149
+ *
1150
+ * @param {Number} code The status code
1151
+ * @param {Buffer} reason The reason for closing
1152
+ * @private
1153
+ */
1154
+ function receiverOnConclude(code, reason) {
1155
+ const websocket = this[kWebSocket];
1156
+
1157
+ websocket._closeFrameReceived = true;
1158
+ websocket._closeMessage = reason;
1159
+ websocket._closeCode = code;
1160
+
1161
+ if (websocket._socket[kWebSocket] === undefined) return;
1162
+
1163
+ websocket._socket.removeListener('data', socketOnData);
1164
+ process.nextTick(resume, websocket._socket);
1165
+
1166
+ if (code === 1005) websocket.close();
1167
+ else websocket.close(code, reason);
1168
+ }
1169
+
1170
+ /**
1171
+ * The listener of the `Receiver` `'drain'` event.
1172
+ *
1173
+ * @private
1174
+ */
1175
+ function receiverOnDrain() {
1176
+ const websocket = this[kWebSocket];
1177
+
1178
+ if (!websocket.isPaused) websocket._socket.resume();
1179
+ }
1180
+
1181
+ /**
1182
+ * The listener of the `Receiver` `'error'` event.
1183
+ *
1184
+ * @param {(RangeError|Error)} err The emitted error
1185
+ * @private
1186
+ */
1187
+ function receiverOnError(err) {
1188
+ const websocket = this[kWebSocket];
1189
+
1190
+ if (websocket._socket[kWebSocket] !== undefined) {
1191
+ websocket._socket.removeListener('data', socketOnData);
1192
+
1193
+ //
1194
+ // On Node.js < 14.0.0 the `'error'` event is emitted synchronously. See
1195
+ // https://github.com/websockets/ws/issues/1940.
1196
+ //
1197
+ process.nextTick(resume, websocket._socket);
1198
+
1199
+ websocket.close(err[kStatusCode]);
1200
+ }
1201
+
1202
+ if (!websocket._errorEmitted) {
1203
+ websocket._errorEmitted = true;
1204
+ websocket.emit('error', err);
1205
+ }
1206
+ }
1207
+
1208
+ /**
1209
+ * The listener of the `Receiver` `'finish'` event.
1210
+ *
1211
+ * @private
1212
+ */
1213
+ function receiverOnFinish() {
1214
+ this[kWebSocket].emitClose();
1215
+ }
1216
+
1217
+ /**
1218
+ * The listener of the `Receiver` `'message'` event.
1219
+ *
1220
+ * @param {Buffer|ArrayBuffer|Buffer[])} data The message
1221
+ * @param {Boolean} isBinary Specifies whether the message is binary or not
1222
+ * @private
1223
+ */
1224
+ function receiverOnMessage(data, isBinary) {
1225
+ this[kWebSocket].emit('message', data, isBinary);
1226
+ }
1227
+
1228
+ /**
1229
+ * The listener of the `Receiver` `'ping'` event.
1230
+ *
1231
+ * @param {Buffer} data The data included in the ping frame
1232
+ * @private
1233
+ */
1234
+ function receiverOnPing(data) {
1235
+ const websocket = this[kWebSocket];
1236
+
1237
+ if (websocket._autoPong) websocket.pong(data, !this._isServer, NOOP);
1238
+ websocket.emit('ping', data);
1239
+ }
1240
+
1241
+ /**
1242
+ * The listener of the `Receiver` `'pong'` event.
1243
+ *
1244
+ * @param {Buffer} data The data included in the pong frame
1245
+ * @private
1246
+ */
1247
+ function receiverOnPong(data) {
1248
+ this[kWebSocket].emit('pong', data);
1249
+ }
1250
+
1251
+ /**
1252
+ * Resume a readable stream
1253
+ *
1254
+ * @param {Readable} stream The readable stream
1255
+ * @private
1256
+ */
1257
+ function resume(stream) {
1258
+ stream.resume();
1259
+ }
1260
+
1261
+ /**
1262
+ * The `Sender` error event handler.
1263
+ *
1264
+ * @param {Error} The error
1265
+ * @private
1266
+ */
1267
+ function senderOnError(err) {
1268
+ const websocket = this[kWebSocket];
1269
+
1270
+ if (websocket.readyState === WebSocket.CLOSED) return;
1271
+ if (websocket.readyState === WebSocket.OPEN) {
1272
+ websocket._readyState = WebSocket.CLOSING;
1273
+ setCloseTimer(websocket);
1274
+ }
1275
+
1276
+ //
1277
+ // `socket.end()` is used instead of `socket.destroy()` to allow the other
1278
+ // peer to finish sending queued data. There is no need to set a timer here
1279
+ // because `CLOSING` means that it is already set or not needed.
1280
+ //
1281
+ this._socket.end();
1282
+
1283
+ if (!websocket._errorEmitted) {
1284
+ websocket._errorEmitted = true;
1285
+ websocket.emit('error', err);
1286
+ }
1287
+ }
1288
+
1289
+ /**
1290
+ * Set a timer to destroy the underlying raw socket of a WebSocket.
1291
+ *
1292
+ * @param {WebSocket} websocket The WebSocket instance
1293
+ * @private
1294
+ */
1295
+ function setCloseTimer(websocket) {
1296
+ websocket._closeTimer = setTimeout(
1297
+ websocket._socket.destroy.bind(websocket._socket),
1298
+ websocket._closeTimeout
1299
+ );
1300
+ }
1301
+
1302
+ /**
1303
+ * The listener of the socket `'close'` event.
1304
+ *
1305
+ * @private
1306
+ */
1307
+ function socketOnClose() {
1308
+ const websocket = this[kWebSocket];
1309
+
1310
+ this.removeListener('close', socketOnClose);
1311
+ this.removeListener('data', socketOnData);
1312
+ this.removeListener('end', socketOnEnd);
1313
+
1314
+ websocket._readyState = WebSocket.CLOSING;
1315
+
1316
+ //
1317
+ // The close frame might not have been received or the `'end'` event emitted,
1318
+ // for example, if the socket was destroyed due to an error. Ensure that the
1319
+ // `receiver` stream is closed after writing any remaining buffered data to
1320
+ // it. If the readable side of the socket is in flowing mode then there is no
1321
+ // buffered data as everything has been already written. If instead, the
1322
+ // socket is paused, any possible buffered data will be read as a single
1323
+ // chunk.
1324
+ //
1325
+ if (
1326
+ !this._readableState.endEmitted &&
1327
+ !websocket._closeFrameReceived &&
1328
+ !websocket._receiver._writableState.errorEmitted &&
1329
+ this._readableState.length !== 0
1330
+ ) {
1331
+ const chunk = this.read(this._readableState.length);
1332
+
1333
+ websocket._receiver.write(chunk);
1334
+ }
1335
+
1336
+ websocket._receiver.end();
1337
+
1338
+ this[kWebSocket] = undefined;
1339
+
1340
+ clearTimeout(websocket._closeTimer);
1341
+
1342
+ if (
1343
+ websocket._receiver._writableState.finished ||
1344
+ websocket._receiver._writableState.errorEmitted
1345
+ ) {
1346
+ websocket.emitClose();
1347
+ } else {
1348
+ websocket._receiver.on('error', receiverOnFinish);
1349
+ websocket._receiver.on('finish', receiverOnFinish);
1350
+ }
1351
+ }
1352
+
1353
+ /**
1354
+ * The listener of the socket `'data'` event.
1355
+ *
1356
+ * @param {Buffer} chunk A chunk of data
1357
+ * @private
1358
+ */
1359
+ function socketOnData(chunk) {
1360
+ if (!this[kWebSocket]._receiver.write(chunk)) {
1361
+ this.pause();
1362
+ }
1363
+ }
1364
+
1365
+ /**
1366
+ * The listener of the socket `'end'` event.
1367
+ *
1368
+ * @private
1369
+ */
1370
+ function socketOnEnd() {
1371
+ const websocket = this[kWebSocket];
1372
+
1373
+ websocket._readyState = WebSocket.CLOSING;
1374
+ websocket._receiver.end();
1375
+ this.end();
1376
+ }
1377
+
1378
+ /**
1379
+ * The listener of the socket `'error'` event.
1380
+ *
1381
+ * @private
1382
+ */
1383
+ function socketOnError() {
1384
+ const websocket = this[kWebSocket];
1385
+
1386
+ this.removeListener('error', socketOnError);
1387
+ this.on('error', NOOP);
1388
+
1389
+ if (websocket) {
1390
+ websocket._readyState = WebSocket.CLOSING;
1391
+ this.destroy();
1392
+ }
1393
+ }