-
Notifications
You must be signed in to change notification settings - Fork 1.6k
/
websocket.dart
501 lines (454 loc) · 18.4 KB
/
websocket.dart
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
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
// Copyright (c) 2013, the Dart project authors. Please see the AUTHORS file
// for details. All rights reserved. Use of this source code is governed by a
// BSD-style license that can be found in the LICENSE file.
// @dart = 2.6
part of dart._http;
/**
* WebSocket status codes used when closing a WebSocket connection.
*/
abstract class WebSocketStatus {
static const int normalClosure = 1000;
static const int goingAway = 1001;
static const int protocolError = 1002;
static const int unsupportedData = 1003;
static const int reserved1004 = 1004;
static const int noStatusReceived = 1005;
static const int abnormalClosure = 1006;
static const int invalidFramePayloadData = 1007;
static const int policyViolation = 1008;
static const int messageTooBig = 1009;
static const int missingMandatoryExtension = 1010;
static const int internalServerError = 1011;
static const int reserved1015 = 1015;
@Deprecated("Use normalClosure instead")
static const int NORMAL_CLOSURE = normalClosure;
@Deprecated("Use goingAway instead")
static const int GOING_AWAY = goingAway;
@Deprecated("Use protocolError instead")
static const int PROTOCOL_ERROR = protocolError;
@Deprecated("Use unsupportedData instead")
static const int UNSUPPORTED_DATA = unsupportedData;
@Deprecated("Use reserved1004 instead")
static const int RESERVED_1004 = reserved1004;
@Deprecated("Use noStatusReceived instead")
static const int NO_STATUS_RECEIVED = noStatusReceived;
@Deprecated("Use abnormalClosure instead")
static const int ABNORMAL_CLOSURE = abnormalClosure;
@Deprecated("Use invalidFramePayloadData instead")
static const int INVALID_FRAME_PAYLOAD_DATA = invalidFramePayloadData;
@Deprecated("Use policyViolation instead")
static const int POLICY_VIOLATION = policyViolation;
@Deprecated("Use messageTooBig instead")
static const int MESSAGE_TOO_BIG = messageTooBig;
@Deprecated("Use missingMandatoryExtension instead")
static const int MISSING_MANDATORY_EXTENSION = missingMandatoryExtension;
@Deprecated("Use internalServerError instead")
static const int INTERNAL_SERVER_ERROR = internalServerError;
@Deprecated("Use reserved1015 instead")
static const int RESERVED_1015 = reserved1015;
}
/// Options controlling compression in a [WebSocket].
///
/// A [CompressionOptions] instance can be passed to [WebSocket.connect], or
/// used in other similar places where [WebSocket] compression is configured.
///
/// In most cases the default [compressionDefault] is sufficient, but in some
/// situations, it might be desirable to use different compression parameters,
/// for example to preserve memory on small devices.
class CompressionOptions {
/// Default [WebSocket] compression configuration.
///
/// Enables compression with default window sizes and no reuse. This is the
/// default options used by [WebSocket.connect] if no [CompressionOptions] is
/// supplied.
///
/// * `clientNoContextTakeover`: false
/// * `serverNoContextTakeover`: false
/// * `clientMaxWindowBits`: null (default maximal window size of 15 bits)
/// * `serverMaxWindowBits`: null (default maximal window size of 15 bits)
static const CompressionOptions compressionDefault =
const CompressionOptions();
@Deprecated("Use compressionDefault instead")
static const CompressionOptions DEFAULT = compressionDefault;
/// No-compression configuration.
///
/// Disables compression when used as compression configuration for a
/// [WebSocket].
static const CompressionOptions compressionOff =
const CompressionOptions(enabled: false);
@Deprecated("Use compressionOff instead")
static const CompressionOptions OFF = compressionOff;
/// Whether the client will reuse its compression instances.
final bool clientNoContextTakeover;
/// Whether the server will reuse its compression instances.
final bool serverNoContextTakeover;
/// The maximal window size bit count requested by the client.
///
/// The windows size for the compression is always a power of two, so the
/// number of bits precisely determines the window size.
///
/// If set to `null`, the client has no preference, and the compression can
/// use up to its default maximum window size of 15 bits depending on the
/// server's preference.
final int clientMaxWindowBits;
/// The maximal window size bit count requested by the server.
///
/// The windows size for the compression is always a power of two, so the
/// number of bits precisely determines the window size.
///
/// If set to `null`, the server has no preference, and the compression can
/// use up to its default maximum window size of 15 bits depending on the
/// client's preference.
final int serverMaxWindowBits;
/// Whether WebSocket compression is enabled.
///
/// If not enabled, the remaining fields have no effect, and the
/// [compressionOff] instance can, and should, be reused instead of creating a
/// new instance with compression disabled.
final bool enabled;
const CompressionOptions(
{this.clientNoContextTakeover = false,
this.serverNoContextTakeover = false,
this.clientMaxWindowBits,
this.serverMaxWindowBits,
this.enabled = true});
/// Parses list of requested server headers to return server compression
/// response headers.
///
/// Uses [serverMaxWindowBits] value if set, otherwise will attempt to use
/// value from headers. Defaults to [WebSocket.DEFAULT_WINDOW_BITS]. Returns a
/// [_CompressionMaxWindowBits] object which contains the response headers and
/// negotiated max window bits.
_CompressionMaxWindowBits _createServerResponseHeader(HeaderValue requested) {
var info = new _CompressionMaxWindowBits();
int mwb;
String part;
if (requested?.parameters != null) {
part = requested.parameters[_serverMaxWindowBits];
}
if (part != null) {
if (part.length >= 2 && part.startsWith('0')) {
throw new ArgumentError("Illegal 0 padding on value.");
} else {
mwb = serverMaxWindowBits == null
? int.tryParse(part) ?? _WebSocketImpl.DEFAULT_WINDOW_BITS
: serverMaxWindowBits;
info.headerValue = "; server_max_window_bits=${mwb}";
info.maxWindowBits = mwb;
}
} else {
info.headerValue = "";
info.maxWindowBits = _WebSocketImpl.DEFAULT_WINDOW_BITS;
}
return info;
}
/// Returns default values for client compression request headers.
String _createClientRequestHeader(HeaderValue requested, int size) {
var info = "";
// If responding to a valid request, specify size
if (requested != null) {
info = "; client_max_window_bits=$size";
} else {
// Client request. Specify default
if (clientMaxWindowBits == null) {
info = "; client_max_window_bits";
} else {
info = "; client_max_window_bits=$clientMaxWindowBits";
}
if (serverMaxWindowBits != null) {
info += "; server_max_window_bits=$serverMaxWindowBits";
}
}
return info;
}
/// Create a Compression Header.
///
/// If [requested] is null or contains client request headers, returns Client
/// compression request headers with default settings for
/// `client_max_window_bits` header value. If [requested] contains server
/// response headers this method returns a Server compression response header
/// negotiating the max window bits for both client and server as requested
/// `server_max_window_bits` value. This method returns a
/// [_CompressionMaxWindowBits] object with the response headers and
/// negotiated `maxWindowBits` value.
_CompressionMaxWindowBits _createHeader([HeaderValue requested]) {
var info = new _CompressionMaxWindowBits("", 0);
if (!enabled) {
return info;
}
info.headerValue = _WebSocketImpl.PER_MESSAGE_DEFLATE;
if (clientNoContextTakeover &&
(requested == null ||
(requested != null &&
requested.parameters.containsKey(_clientNoContextTakeover)))) {
info.headerValue += "; client_no_context_takeover";
}
if (serverNoContextTakeover &&
(requested == null ||
(requested != null &&
requested.parameters.containsKey(_serverNoContextTakeover)))) {
info.headerValue += "; server_no_context_takeover";
}
var headerList = _createServerResponseHeader(requested);
info.headerValue += headerList.headerValue;
info.maxWindowBits = headerList.maxWindowBits;
info.headerValue +=
_createClientRequestHeader(requested, info.maxWindowBits);
return info;
}
}
/**
* The [WebSocketTransformer] provides the ability to upgrade a
* [HttpRequest] to a [WebSocket] connection. It supports both
* upgrading a single [HttpRequest] and upgrading a stream of
* [HttpRequest]s.
*
* To upgrade a single [HttpRequest] use the static [upgrade] method.
*
* HttpServer server;
* server.listen((request) {
* if (...) {
* WebSocketTransformer.upgrade(request).then((websocket) {
* ...
* });
* } else {
* // Do normal HTTP request processing.
* }
* });
*
* To transform a stream of [HttpRequest] events as it implements a
* stream transformer that transforms a stream of HttpRequest into a
* stream of WebSockets by upgrading each HttpRequest from the HTTP or
* HTTPS server, to the WebSocket protocol.
*
* server.transform(new WebSocketTransformer()).listen((webSocket) => ...);
*
* This transformer strives to implement WebSockets as specified by RFC6455.
*/
abstract class WebSocketTransformer
implements StreamTransformer<HttpRequest, WebSocket> {
/**
* Create a new [WebSocketTransformer].
*
* If [protocolSelector] is provided, [protocolSelector] will be called to
* select what protocol to use, if any were provided by the client.
* [protocolSelector] is should return either a [String] or a [Future]
* completing with a [String]. The [String] must exist in the list of
* protocols.
*
* If [compression] is provided, the [WebSocket] created will be configured
* to negotiate with the specified [CompressionOptions]. If none is specified
* then the [WebSocket] will be created with the default [CompressionOptions].
*/
factory WebSocketTransformer(
{/*String|Future<String>*/ protocolSelector(List<String> protocols),
CompressionOptions compression: CompressionOptions.compressionDefault}) {
return new _WebSocketTransformerImpl(protocolSelector, compression);
}
/**
* Upgrades a [HttpRequest] to a [WebSocket] connection. If the
* request is not a valid WebSocket upgrade request an HTTP response
* with status code 500 will be returned. Otherwise the returned
* future will complete with the [WebSocket] when the upgrade process
* is complete.
*
* If [protocolSelector] is provided, [protocolSelector] will be called to
* select what protocol to use, if any were provided by the client.
* [protocolSelector] is should return either a [String] or a [Future]
* completing with a [String]. The [String] must exist in the list of
* protocols.
*
* If [compression] is provided, the [WebSocket] created will be configured
* to negotiate with the specified [CompressionOptions]. If none is specified
* then the [WebSocket] will be created with the default [CompressionOptions].
*/
static Future<WebSocket> upgrade(HttpRequest request,
{protocolSelector(List<String> protocols),
CompressionOptions compression: CompressionOptions.compressionDefault}) {
return _WebSocketTransformerImpl._upgrade(
request, protocolSelector, compression);
}
/**
* Checks whether the request is a valid WebSocket upgrade request.
*/
static bool isUpgradeRequest(HttpRequest request) {
return _WebSocketTransformerImpl._isUpgradeRequest(request);
}
}
/**
* A two-way HTTP communication object for client or server applications.
*
* The stream exposes the messages received. A text message will be of type
* `String` and a binary message will be of type `List<int>`.
*/
abstract class WebSocket
implements
Stream<dynamic /*String|List<int>*/ >,
StreamSink<dynamic /*String|List<int>*/ > {
/**
* Possible states of the connection.
*/
static const int connecting = 0;
static const int open = 1;
static const int closing = 2;
static const int closed = 3;
@Deprecated("Use connecting instead")
static const int CONNECTING = connecting;
@Deprecated("Use open instead")
static const int OPEN = open;
@Deprecated("Use closing instead")
static const int CLOSING = closing;
@Deprecated("Use closed instead")
static const int CLOSED = closed;
/**
* The interval between ping signals.
*
* A ping message is sent every [pingInterval], starting at the first
* [pingInterval] after a new value has been assigned or a pong message has
* been received. If a ping message is not answered by a pong message from the
* peer, the `WebSocket` is assumed disconnected and the connection is closed
* with a [WebSocketStatus.goingAway] close code. When a ping signal is sent,
* the pong message must be received within [pingInterval].
*
* There are never two outstanding pings at any given time, and the next ping
* timer starts when the pong is received.
*
* Set the [pingInterval] to `null` to disable sending ping messages.
*
* The default value is `null`.
*/
Duration pingInterval;
/**
* Create a new WebSocket connection. The URL supplied in [url]
* must use the scheme `ws` or `wss`.
*
* The [protocols] argument is specifying the subprotocols the
* client is willing to speak.
*
* The [headers] argument is specifying additional HTTP headers for
* setting up the connection. This would typically be the `Origin`
* header and potentially cookies. The keys of the map are the header
* fields and the values are either String or List<String>.
*
* If [headers] is provided, there are a number of headers
* which are controlled by the WebSocket connection process. These
* headers are:
*
* - `connection`
* - `sec-websocket-key`
* - `sec-websocket-protocol`
* - `sec-websocket-version`
* - `upgrade`
*
* If any of these are passed in the `headers` map they will be ignored.
*
* If the `url` contains user information this will be passed as basic
* authentication when setting up the connection.
*/
static Future<WebSocket> connect(String url,
{Iterable<String> protocols,
Map<String, dynamic> headers,
CompressionOptions compression:
CompressionOptions.compressionDefault}) =>
_WebSocketImpl.connect(url, protocols, headers, compression: compression);
@Deprecated('This constructor will be removed in Dart 2.0. Use `implements`'
' instead of `extends` if implementing this abstract class.')
WebSocket();
/**
* Creates a WebSocket from an already-upgraded socket.
*
* The initial WebSocket handshake must have occurred prior to this call. A
* WebSocket client can automatically perform the handshake using
* [WebSocket.connect], while a server can do so using
* [WebSocketTransformer.upgrade]. To manually upgrade an [HttpRequest],
* [HttpResponse.detachSocket] may be called.
*
* [protocol] should be the protocol negotiated by this handshake, if any.
*
* [serverSide] must be passed explicitly. If it's `false`, the WebSocket will
* act as the client and mask the messages it sends. If it's `true`, it will
* act as the server and will not mask its messages.
*
* If [compression] is provided, the [WebSocket] created will be configured
* to negotiate with the specified [CompressionOptions]. If none is specified
* then the [WebSocket] will be created with the default [CompressionOptions].
*/
factory WebSocket.fromUpgradedSocket(Socket socket,
{String protocol,
bool serverSide,
CompressionOptions compression: CompressionOptions.compressionDefault}) {
if (serverSide == null) {
throw new ArgumentError("The serverSide argument must be passed "
"explicitly to WebSocket.fromUpgradedSocket.");
}
return new _WebSocketImpl._fromSocket(
socket, protocol, compression, serverSide);
}
/**
* Returns the current state of the connection.
*/
int get readyState;
/**
* The extensions property is initially the empty string. After the
* WebSocket connection is established this string reflects the
* extensions used by the server.
*/
String get extensions;
/**
* The protocol property is initially the empty string. After the
* WebSocket connection is established the value is the subprotocol
* selected by the server. If no subprotocol is negotiated the
* value will remain [:null:].
*/
String get protocol;
/**
* The close code set when the WebSocket connection is closed. If
* there is no close code available this property will be [:null:]
*/
int get closeCode;
/**
* The close reason set when the WebSocket connection is closed. If
* there is no close reason available this property will be [:null:]
*/
String get closeReason;
/**
* Closes the WebSocket connection. Set the optional [code] and [reason]
* arguments to send close information to the remote peer. If they are
* omitted, the peer will see [WebSocketStatus.noStatusReceived] code
* with no reason.
*/
Future close([int code, String reason]);
/**
* Sends data on the WebSocket connection. The data in [data] must
* be either a `String`, or a `List<int>` holding bytes.
*/
void add(/*String|List<int>*/ data);
/**
* Sends data from a stream on WebSocket connection. Each data event from
* [stream] will be send as a single WebSocket frame. The data from [stream]
* must be either `String`s, or `List<int>`s holding bytes.
*/
Future addStream(Stream stream);
/**
* Sends a text message with the text represented by [bytes].
*
* The [bytes] should be valid UTF-8 encoded Unicode characters. If they are
* not, the receiving end will close the connection.
*/
void addUtf8Text(List<int> bytes);
/**
* Gets the user agent used for WebSocket connections.
*/
static String get userAgent => _WebSocketImpl.userAgent;
/**
* Sets the user agent to use for WebSocket connections.
*/
static set userAgent(String userAgent) {
_WebSocketImpl.userAgent = userAgent;
}
}
class WebSocketException implements IOException {
final String message;
const WebSocketException([this.message = ""]);
String toString() => "WebSocketException: $message";
}