Index: runtime/bin/websocket_impl.dart |
diff --git a/runtime/bin/websocket_impl.dart b/runtime/bin/websocket_impl.dart |
deleted file mode 100644 |
index 0d2089d0d7c6e6e5cd71fcdbd69b22a048a02e30..0000000000000000000000000000000000000000 |
--- a/runtime/bin/websocket_impl.dart |
+++ /dev/null |
@@ -1,872 +0,0 @@ |
-// Copyright (c) 2012, 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. |
- |
-const String _webSocketGUID = "258EAFA5-E914-47DA-95CA-C5AB0DC85B11"; |
- |
-class _WebSocketMessageType { |
- static const int NONE = 0; |
- static const int BINARY = 1; |
- static const int TEXT = 2; |
-} |
- |
- |
-class _WebSocketOpcode { |
- static const int CONTINUATION = 0; |
- static const int TEXT = 1; |
- static const int BINARY = 2; |
- static const int RESERVED_3 = 3; |
- static const int RESERVED_4 = 4; |
- static const int RESERVED_5 = 5; |
- static const int RESERVED_6 = 6; |
- static const int RESERVED_7 = 7; |
- static const int CLOSE = 8; |
- static const int PING = 9; |
- static const int PONG = 10; |
- static const int RESERVED_B = 11; |
- static const int RESERVED_C = 12; |
- static const int RESERVED_D = 13; |
- static const int RESERVED_E = 14; |
- static const int RESERVED_F = 15; |
-} |
- |
-/** |
- * The web socket protocol processor handles the protocol byte stream |
- * which is supplied through the [:update:] and [:closed:] |
- * methods. As the protocol is processed the following callbacks are |
- * called: |
- * |
- * [:onMessageStart:] |
- * [:onMessageData:] |
- * [:onMessageEnd:] |
- * [:onClosed:] |
- * [:onError:] |
- * |
- */ |
-class _WebSocketProtocolProcessor { |
- static const int START = 0; |
- static const int LEN_FIRST = 1; |
- static const int LEN_REST = 2; |
- static const int MASK = 3; |
- static const int PAYLOAD = 4; |
- static const int CLOSED = 5; |
- static const int FAILURE = 6; |
- |
- _WebSocketProtocolProcessor() { |
- _prepareForNextFrame(); |
- _currentMessageType = _WebSocketMessageType.NONE; |
- } |
- |
- /** |
- * Process data received from the underlying communication channel. |
- */ |
- void update(List<int> buffer, int offset, int count) { |
- int index = offset; |
- int lastIndex = offset + count; |
- try { |
- if (_state == CLOSED) { |
- throw new WebSocketException("Data on closed connection"); |
- } |
- if (_state == FAILURE) { |
- throw new WebSocketException("Data on failed connection"); |
- } |
- while ((index < lastIndex) && _state != CLOSED && _state != FAILURE) { |
- int byte = buffer[index]; |
- switch (_state) { |
- case START: |
- _fin = (byte & 0x80) != 0; |
- _opcode = (byte & 0xF); |
- switch (_opcode) { |
- case _WebSocketOpcode.CONTINUATION: |
- if (_currentMessageType == _WebSocketMessageType.NONE) { |
- throw new WebSocketException("Protocol error"); |
- } |
- break; |
- |
- case _WebSocketOpcode.TEXT: |
- if (_currentMessageType != _WebSocketMessageType.NONE) { |
- throw new WebSocketException("Protocol error"); |
- } |
- _currentMessageType = _WebSocketMessageType.TEXT; |
- if (onMessageStart !== null) { |
- onMessageStart(_WebSocketMessageType.TEXT); |
- } |
- break; |
- |
- case _WebSocketOpcode.BINARY: |
- if (_currentMessageType != _WebSocketMessageType.NONE) { |
- throw new WebSocketException("Protocol error"); |
- } |
- _currentMessageType = _WebSocketMessageType.BINARY; |
- if (onMessageStart !== null) { |
- onMessageStart(_WebSocketMessageType.BINARY); |
- } |
- break; |
- |
- case _WebSocketOpcode.CLOSE: |
- case _WebSocketOpcode.PING: |
- case _WebSocketOpcode.PONG: |
- // Control frames cannot be fragmented. |
- if (!_fin) throw new WebSocketException("Protocol error"); |
- break; |
- |
- default: |
- throw new WebSocketException("Protocol error"); |
- } |
- _state = LEN_FIRST; |
- break; |
- |
- case LEN_FIRST: |
- _masked = (byte & 0x80) != 0; |
- _len = byte & 0x7F; |
- if (_isControlFrame() && _len > 126) { |
- throw new WebSocketException("Protocol error"); |
- } |
- if (_len < 126) { |
- _lengthDone(); |
- } else if (_len == 126) { |
- _len = 0; |
- _remainingLenBytes = 2; |
- _state = LEN_REST; |
- } else if (_len == 127) { |
- _len = 0; |
- _remainingLenBytes = 8; |
- _state = LEN_REST; |
- } |
- break; |
- |
- case LEN_REST: |
- _len = _len << 8 | byte; |
- _remainingLenBytes--; |
- if (_remainingLenBytes == 0) { |
- _lengthDone(); |
- } |
- break; |
- |
- case MASK: |
- _maskingKey = _maskingKey << 8 | byte; |
- _remainingMaskingKeyBytes--; |
- if (_remainingMaskingKeyBytes == 0) { |
- _maskDone(); |
- } |
- break; |
- |
- case PAYLOAD: |
- // The payload is not handled one byte at a time but in blocks. |
- int payload; |
- if (lastIndex - index <= _remainingPayloadBytes) { |
- payload = lastIndex - index; |
- } else { |
- payload = _remainingPayloadBytes; |
- } |
- _remainingPayloadBytes -= payload; |
- |
- // Unmask payload if masked. |
- if (_masked) { |
- for (int i = 0; i < payload; i++) { |
- int maskingByte = |
- ((_maskingKey >> ((3 - _unmaskingIndex) * 8)) & 0xFF); |
- buffer[index + i] = buffer[index + i] ^ maskingByte; |
- _unmaskingIndex = (_unmaskingIndex + 1) % 4; |
- } |
- } |
- |
- if (_isControlFrame()) { |
- if (payload > 0) { |
- // Allocate a buffer for collecting the control frame |
- // payload if any. |
- if (_controlPayload == null) { |
- _controlPayload = new List<int>(); |
- } |
- _controlPayload.addAll(buffer.getRange(index, payload)); |
- index += payload; |
- } |
- |
- if (_remainingPayloadBytes == 0) { |
- _controlFrameEnd(); |
- } |
- } else { |
- switch (_currentMessageType) { |
- case _WebSocketMessageType.NONE: |
- throw new WebSocketException("Protocol error"); |
- |
- case _WebSocketMessageType.TEXT: |
- case _WebSocketMessageType.BINARY: |
- if (onMessageData !== null) { |
- onMessageData(buffer, index, payload); |
- } |
- index += payload; |
- if (_remainingPayloadBytes == 0) { |
- _messageFrameEnd(); |
- } |
- break; |
- |
- default: |
- throw new WebSocketException("Protocol error"); |
- } |
- } |
- |
- // Hack - as we always do index++ below. |
- index--; |
- } |
- |
- // Move to the next byte. |
- index++; |
- } |
- } catch (e) { |
- if (onClosed !== null) onClosed(WebSocketStatus.PROTOCOL_ERROR, |
- "Protocol error"); |
- _state = FAILURE; |
- } |
- } |
- |
- /** |
- * Indicate that the underlying communication channel has been closed. |
- */ |
- void closed() { |
- if (_state == START || _state == CLOSED || _state == FAILURE) return; |
- if (onClosed !== null) onClosed(WebSocketStatus.ABNORMAL_CLOSURE, |
- "Connection closed unexpectedly"); |
- _state = CLOSED; |
- } |
- |
- void _lengthDone() { |
- if (_masked) { |
- _state = MASK; |
- _remainingMaskingKeyBytes = 4; |
- } else { |
- _remainingPayloadBytes = _len; |
- _startPayload(); |
- } |
- } |
- |
- void _maskDone() { |
- _remainingPayloadBytes = _len; |
- _startPayload(); |
- } |
- |
- void _startPayload() { |
- // If there is no actual payload perform perform callbacks without |
- // going through the PAYLOAD state. |
- if (_remainingPayloadBytes == 0) { |
- if (_isControlFrame()) { |
- switch (_opcode) { |
- case _WebSocketOpcode.CLOSE: |
- if (onClosed !== null) onClosed(1005, ""); |
- _state = CLOSED; |
- break; |
- case _WebSocketOpcode.PING: |
- if (onPing !== null) onPing(null); |
- break; |
- case _WebSocketOpcode.PONG: |
- if (onPong !== null) onPong(null); |
- break; |
- } |
- _prepareForNextFrame(); |
- } else { |
- _messageFrameEnd(); |
- } |
- } else { |
- _state = PAYLOAD; |
- } |
- } |
- |
- void _messageFrameEnd() { |
- if (_fin) { |
- if (onMessageEnd !== null) onMessageEnd(); |
- _currentMessageType = _WebSocketMessageType.NONE; |
- } |
- _prepareForNextFrame(); |
- } |
- |
- void _controlFrameEnd() { |
- switch (_opcode) { |
- case _WebSocketOpcode.CLOSE: |
- int status = WebSocketStatus.NO_STATUS_RECEIVED; |
- String reason = ""; |
- if (_controlPayload.length > 0) { |
- if (_controlPayload.length == 1) { |
- throw new WebSocketException("Protocol error"); |
- } |
- status = _controlPayload[0] << 8 | _controlPayload[1]; |
- if (status == WebSocketStatus.NO_STATUS_RECEIVED) { |
- throw new WebSocketException("Protocol error"); |
- } |
- if (_controlPayload.length > 2) { |
- var decoder = _StringDecoders.decoder(Encoding.UTF_8); |
- decoder.write( |
- _controlPayload.getRange(2, _controlPayload.length - 2)); |
- reason = decoder.decoded(); |
- } |
- } |
- if (onClosed !== null) onClosed(status, reason); |
- _state = CLOSED; |
- break; |
- |
- case _WebSocketOpcode.PING: |
- if (onPing !== null) onPing(_controlPayload); |
- break; |
- |
- case _WebSocketOpcode.PONG: |
- if (onPong !== null) onPong(_controlPayload); |
- break; |
- } |
- _prepareForNextFrame(); |
- } |
- |
- bool _isControlFrame() { |
- return _opcode == _WebSocketOpcode.CLOSE || |
- _opcode == _WebSocketOpcode.PING || |
- _opcode == _WebSocketOpcode.PONG; |
- } |
- |
- void _prepareForNextFrame() { |
- if (_state != CLOSED && _state != FAILURE) _state = START; |
- _fin = null; |
- _opcode = null; |
- _len = null; |
- _masked = null; |
- _maskingKey = 0; |
- _remainingLenBytes = null; |
- _remainingMaskingKeyBytes = null; |
- _remainingPayloadBytes = null; |
- _unmaskingIndex = 0; |
- _controlPayload = null; |
- } |
- |
- int _state; |
- bool _fin; |
- int _opcode; |
- int _len; |
- bool _masked; |
- int _maskingKey; |
- int _remainingLenBytes; |
- int _remainingMaskingKeyBytes; |
- int _remainingPayloadBytes; |
- int _unmaskingIndex; |
- |
- int _currentMessageType; |
- List<int> _controlPayload; |
- |
- Function onMessageStart; |
- Function onMessageData; |
- Function onMessageEnd; |
- Function onPing; |
- Function onPong; |
- Function onClosed; |
-} |
- |
- |
-class _WebSocketConnectionBase { |
- void _socketConnected(Socket socket) { |
- _socket = socket; |
- _socket.onError = (e) => _socket.close(); |
- } |
- |
- void _startProcessing(List<int> unparsedData) { |
- _WebSocketProtocolProcessor processor = new _WebSocketProtocolProcessor(); |
- processor.onMessageStart = _onWebSocketMessageStart; |
- processor.onMessageData = _onWebSocketMessageData; |
- processor.onMessageEnd = _onWebSocketMessageEnd; |
- processor.onPing = _onWebSocketPing; |
- processor.onPong = _onWebSocketPong; |
- processor.onClosed = _onWebSocketClosed; |
- if (unparsedData !== null) { |
- processor.update(unparsedData, 0, unparsedData.length); |
- } |
- _socket.onData = () { |
- int available = _socket.available(); |
- List<int> data = new List<int>(available); |
- int read = _socket.readList(data, 0, available); |
- processor.update(data, 0, read); |
- }; |
- _socket.onClosed = () { |
- processor.closed(); |
- if (_closeSent) { |
- // Got socket close in response to close frame. Don't treat |
- // that as an error. |
- if (_closeTimer !== null) _closeTimer.cancel(); |
- } else { |
- if (_onClosed !== null) _onClosed(WebSocketStatus.ABNORMAL_CLOSURE, |
- "Unexpected close"); |
- } |
- _socket.close(); |
- }; |
- } |
- |
- void set onMessage(void callback(Object message)) { |
- _onMessage = callback; |
- } |
- |
- void set onClosed(void callback(int status, String reason)) { |
- _onClosed = callback; |
- } |
- |
- send(message) { |
- if (_closeSent) { |
- throw new WebSocketException("Connection closed"); |
- } |
- List<int> data; |
- int opcode; |
- if (message !== null) { |
- if (message is String) { |
- opcode = _WebSocketOpcode.TEXT; |
- data = _StringEncoders.encoder(Encoding.UTF_8).encodeString(message); |
- } else { |
- if (message is !List<int>) { |
- throw new ArgumentError(message); |
- } |
- opcode = _WebSocketOpcode.BINARY; |
- data = message; |
- } |
- } else { |
- opcode = _WebSocketOpcode.TEXT; |
- } |
- _sendFrame(opcode, data); |
- } |
- |
- close([int status, String reason]) { |
- if (status == WebSocketStatus.RESERVED_1004 || |
- status == WebSocketStatus.NO_STATUS_RECEIVED || |
- status == WebSocketStatus.RESERVED_1015) { |
- throw new WebSocketException("Reserved status code $status"); |
- } |
- |
- if (_closeSent) return; |
- List<int> data; |
- if (status !== null) { |
- data = new List<int>(); |
- data.add((status >> 8) & 0xFF); |
- data.add(status & 0xFF); |
- if (reason !== null) { |
- data.addAll( |
- _StringEncoders.encoder(Encoding.UTF_8).encodeString(reason)); |
- } |
- } |
- _sendFrame(_WebSocketOpcode.CLOSE, data); |
- |
- if (_closeReceived) { |
- // Close the socket when the close frame has been sent - if it |
- // does not take too long. |
- _socket.outputStream.close(); |
- _socket.outputStream.onClosed = () { |
- if (_closeTimer !== null) _closeTimer.cancel(); |
- _socket.close(); |
- }; |
- _closeTimer = new Timer(5000, (t) { |
- _socket.close(); |
- }); |
- } else { |
- // Half close the socket and expect a close frame in response |
- // before closing the socket. If a close frame does not arrive |
- // within a reasonable amount of time just close the socket. |
- _socket.outputStream.close(); |
- _closeTimer = new Timer(5000, (t) { |
- _socket.close(); |
- }); |
- } |
- _closeSent = true; |
- } |
- |
- int get hashCode => _hash; |
- |
- _onWebSocketMessageStart(int type) { |
- _currentMessageType = type; |
- if (_currentMessageType == _WebSocketMessageType.TEXT) { |
- _decoder = _StringDecoders.decoder(Encoding.UTF_8); |
- } else { |
- _outputStream = new ListOutputStream(); |
- } |
- } |
- |
- _onWebSocketMessageData(List<int> buffer, int offset, int count) { |
- if (_currentMessageType == _WebSocketMessageType.TEXT) { |
- _decoder.write(buffer.getRange(offset, count)); |
- } else { |
- _outputStream.write(buffer.getRange(offset, count)); |
- } |
- } |
- |
- _onWebSocketMessageEnd() { |
- if (_onMessage !== null) { |
- if (_currentMessageType == _WebSocketMessageType.TEXT) { |
- _onMessage(_decoder.decoded()); |
- } else { |
- _onMessage(_outputStream.read()); |
- } |
- } |
- _decoder = null; |
- _outputStream = null; |
- } |
- |
- _onWebSocketPing(List<int> payload) { |
- _sendFrame(_WebSocketOpcode.PONG, payload); |
- } |
- |
- _onWebSocketPong(List<int> payload) { |
- // Currently pong messages are ignored. |
- } |
- |
- _onWebSocketClosed(int status, String reason) { |
- _closeReceived = true; |
- if (_onClosed !== null) _onClosed(status, reason); |
- if (_closeSent) { |
- // Got close frame in response to close frame. Now close the socket. |
- if (_closeTimer !== null) _closeTimer.cancel(); |
- _socket.close(); |
- } else { |
- if (status != WebSocketStatus.NO_STATUS_RECEIVED) { |
- close(status); |
- } else { |
- close(); |
- } |
- } |
- } |
- |
- _sendFrame(int opcode, [List<int> data]) { |
- bool mask = false; // Masking not implemented for server. |
- int dataLength = data == null ? 0 : data.length; |
- // Determine the header size. |
- int headerSize = (mask) ? 6 : 2; |
- if (dataLength > 65535) { |
- headerSize += 8; |
- } else if (dataLength > 125) { |
- headerSize += 2; |
- } |
- List<int> header = new List<int>(headerSize); |
- int index = 0; |
- // Set FIN and opcode. |
- header[index++] = 0x80 | opcode; |
- // Determine size and position of length field. |
- int lengthBytes = 1; |
- int firstLengthByte = 1; |
- if (dataLength > 65535) { |
- header[index++] = 127; |
- lengthBytes = 8; |
- } else if (dataLength > 125) { |
- header[index++] = 126; |
- lengthBytes = 2; |
- } |
- // Write the length in network byte order into the header. |
- for (int i = 0; i < lengthBytes; i++) { |
- header[index++] = dataLength >> (((lengthBytes - 1) - i) * 8) & 0xFF; |
- } |
- assert(index == headerSize); |
- _socket.outputStream.write(header); |
- if (data !== null) { |
- _socket.outputStream.write(data); |
- } |
- } |
- |
- Socket _socket; |
- Timer _closeTimer; |
- int _hash; |
- |
- Function _onMessage; |
- Function _onClosed; |
- |
- int _currentMessageType = _WebSocketMessageType.NONE; |
- _StringDecoder _decoder; |
- ListOutputStream _outputStream; |
- bool _closeReceived = false; |
- bool _closeSent = false; |
-} |
- |
- |
-class _WebSocketConnection |
- extends _WebSocketConnectionBase implements WebSocketConnection { |
- _WebSocketConnection(DetachedSocket detached) { |
- _hash = detached.socket.hashCode; |
- _socketConnected(detached.socket); |
- _startProcessing(detached.unparsedData); |
- } |
-} |
- |
- |
-class _WebSocketHandler implements WebSocketHandler { |
- void onRequest(HttpRequest request, HttpResponse response) { |
- // Check that this is a web socket upgrade. |
- if (!_isWebSocketUpgrade(request)) { |
- response.statusCode = HttpStatus.BAD_REQUEST; |
- response.outputStream.close(); |
- return; |
- } |
- |
- // Send the upgrade response. |
- response.statusCode = HttpStatus.SWITCHING_PROTOCOLS; |
- response.headers.add(HttpHeaders.CONNECTION, "Upgrade"); |
- response.headers.add(HttpHeaders.UPGRADE, "websocket"); |
- String key = request.headers.value("Sec-WebSocket-Key"); |
- SHA1 sha1 = new SHA1(); |
- sha1.update("$key$_webSocketGUID".charCodes); |
- String accept = _Base64._encode(sha1.digest()); |
- response.headers.add("Sec-WebSocket-Accept", accept); |
- response.contentLength = 0; |
- |
- // Upgrade the connection and get the underlying socket. |
- WebSocketConnection conn = |
- new _WebSocketConnection(response.detachSocket()); |
- if (_onOpen !== null) _onOpen(conn); |
- } |
- |
- void set onOpen(callback(WebSocketConnection connection)) { |
- _onOpen = callback; |
- } |
- |
- bool _isWebSocketUpgrade(HttpRequest request) { |
- if (request.method != "GET") { |
- return false; |
- } |
- if (request.headers[HttpHeaders.CONNECTION] == null) { |
- return false; |
- } |
- bool isUpgrade = false; |
- request.headers[HttpHeaders.CONNECTION].forEach((String value) { |
- if (value.toLowerCase() == "upgrade") isUpgrade = true; |
- }); |
- if (!isUpgrade) return false; |
- String upgrade = request.headers.value(HttpHeaders.UPGRADE); |
- if (upgrade == null || upgrade.toLowerCase() != "websocket") { |
- return false; |
- } |
- String version = request.headers.value("Sec-WebSocket-Version"); |
- if (version == null || version != "13") { |
- return false; |
- } |
- String key = request.headers.value("Sec-WebSocket-Key"); |
- if (key == null) { |
- return false; |
- } |
- return true; |
- } |
- |
- Function _onOpen; |
-} |
- |
- |
-class _WebSocketClientConnection |
- extends _WebSocketConnectionBase implements WebSocketClientConnection { |
- _WebSocketClientConnection(HttpClientConnection this._conn, |
- [List<String> protocols]) { |
- _conn.onRequest = _onHttpClientRequest; |
- _conn.onResponse = _onHttpClientResponse; |
- _conn.onError = (e) { |
- if (_onClosed !== null) { |
- _onClosed(WebSocketStatus.ABNORMAL_CLOSURE, "$e"); |
- } |
- }; |
- |
- // Generate the nonce now as it is also used to set the hash code. |
- _generateNonceAndHash(); |
- } |
- |
- void set onRequest(void callback(HttpClientRequest request)) { |
- _onRequest = callback; |
- } |
- |
- void set onOpen(void callback()) { |
- _onOpen = callback; |
- } |
- |
- void set onNoUpgrade(void callback(HttpClientResponse request)) { |
- _onNoUpgrade = callback; |
- } |
- |
- void _onHttpClientRequest(HttpClientRequest request) { |
- if (_onRequest !== null) { |
- _onRequest(request); |
- } |
- // Setup the initial handshake. |
- request.headers.add(HttpHeaders.CONNECTION, "upgrade"); |
- request.headers.set(HttpHeaders.UPGRADE, "websocket"); |
- request.headers.set("Sec-WebSocket-Key", _nonce); |
- request.headers.set("Sec-WebSocket-Version", "13"); |
- request.contentLength = 0; |
- request.outputStream.close(); |
- } |
- |
- void _onHttpClientResponse(HttpClientResponse response) { |
- if (response.statusCode != HttpStatus.SWITCHING_PROTOCOLS) { |
- if (_onNoUpgrade !== null) { |
- _onNoUpgrade(response); |
- } else { |
- _conn.detachSocket().socket.close(); |
- throw new WebSocketException("Protocol upgrade refused"); |
- } |
- return; |
- } |
- |
- if (!_isWebSocketUpgrade(response)) { |
- _conn.detachSocket().socket.close(); |
- throw new WebSocketException("Protocol upgrade failed"); |
- return; |
- } |
- |
- // Connection upgrade successful. |
- DetachedSocket detached = _conn.detachSocket(); |
- _socketConnected(detached.socket); |
- if (_onOpen !== null) _onOpen(); |
- _startProcessing(detached.unparsedData); |
- } |
- |
- void _generateNonceAndHash() { |
- Random random = new Random(); |
- assert(_nonce == null); |
- void intToBigEndianBytes(int value, List<int> bytes, int offset) { |
- bytes[offset] = (value >> 24) & 0xFF; |
- bytes[offset + 1] = (value >> 16) & 0xFF; |
- bytes[offset + 2] = (value >> 8) & 0xFF; |
- bytes[offset + 3] = value & 0xFF; |
- } |
- |
- // Generate 16 random bytes. Use the last four bytes for the hash code. |
- List<int> nonce = new List<int>(16); |
- for (int i = 0; i < 4; i++) { |
- int r = random.nextInt(0x100000000); |
- intToBigEndianBytes(r, nonce, i * 4); |
- } |
- _nonce = _Base64._encode(nonce); |
- _hash = random.nextInt(0x100000000); |
- } |
- |
- bool _isWebSocketUpgrade(HttpClientResponse response) { |
- if (response.headers[HttpHeaders.CONNECTION] == null) { |
- return false; |
- } |
- bool isUpgrade = false; |
- response.headers[HttpHeaders.CONNECTION].forEach((String value) { |
- if (value.toLowerCase() == "upgrade") isUpgrade = true; |
- }); |
- if (!isUpgrade) return false; |
- String upgrade = response.headers.value(HttpHeaders.UPGRADE); |
- if (upgrade == null || upgrade.toLowerCase() != "websocket") { |
- return false; |
- } |
- String accept = response.headers.value("Sec-WebSocket-Accept"); |
- if (accept == null) { |
- return false; |
- } |
- SHA1 sha1 = new SHA1(); |
- sha1.update("$_nonce$_webSocketGUID".charCodes); |
- List<int> expectedAccept = sha1.digest(); |
- List<int> receivedAccept = _Base64._decode(accept); |
- if (expectedAccept.length != receivedAccept.length) return false; |
- for (int i = 0; i < expectedAccept.length; i++) { |
- if (expectedAccept[i] != receivedAccept[i]) return false; |
- } |
- return true; |
- } |
- |
- Function _onRequest; |
- Function _onOpen; |
- Function _onNoUpgrade; |
- HttpClientConnection _conn; |
- String _nonce; |
-} |
- |
- |
-class _WebSocket implements WebSocket { |
- _WebSocket(String url, [protocols]) { |
- Uri uri = new Uri.fromString(url); |
- if (uri.scheme != "ws") { |
- throw new WebSocketException("Unsupported URL scheme ${uri.scheme}"); |
- } |
- if (uri.userInfo != "") { |
- throw new WebSocketException("Unsupported user info ${uri.userInfo}"); |
- } |
- int port = uri.port == 0 ? HttpClient.DEFAULT_HTTP_PORT : uri.port; |
- String path = uri.path; |
- if (path.length == 0) path = "/"; |
- if (uri.query != "") { |
- if (uri.fragment != "") { |
- path = "${path}?${uri.query}#${uri.fragment}"; |
- } else { |
- path = "${path}?${uri.query}"; |
- } |
- } |
- |
- HttpClient client = new HttpClient(); |
- HttpClientConnection conn = client.open("GET", uri.domain, port, path); |
- if (protocols is String) protocols = [protocols]; |
- _wsconn = new WebSocketClientConnection(conn, protocols); |
- _wsconn.onOpen = () { |
- // HTTP client not needed after socket have been detached. |
- client.shutdown(); |
- client = null; |
- _readyState = WebSocket.OPEN; |
- if (_onopen !== null) _onopen(); |
- }; |
- _wsconn.onMessage = (message) { |
- if (_onmessage !== null) { |
- _onmessage(new _WebSocketMessageEvent(message)); |
- } |
- }; |
- _wsconn.onClosed = (status, reason) { |
- _readyState = WebSocket.CLOSED; |
- if (_onclose !== null) { |
- _onclose(new _WebSocketCloseEvent(true, status, reason)); |
- } |
- }; |
- _wsconn.onNoUpgrade = (response) { |
- if (_onclose !== null) { |
- _onclose( |
- new _WebSocketCloseEvent(true, |
- WebSocketStatus.ABNORMAL_CLOSURE, |
- "Connection not upgraded")); |
- } |
- }; |
- } |
- |
- int get readyState => _readyState; |
- int get bufferedAmount => 0; |
- |
- void set onopen(Function callback) { |
- _onopen = callback; |
- } |
- |
- void set onerror(Function callback) {} |
- |
- void set onclose(Function callback) { |
- _onclose = callback; |
- } |
- |
- String get extensions => null; |
- String get protocol => null; |
- |
- void close(int code, String reason) { |
- if (_readyState < WebSocket.CLOSING) _readyState = WebSocket.CLOSING; |
- _wsconn.close(code, reason); |
- } |
- |
- void set onmessage(Function callback) { |
- _onmessage = callback; |
- } |
- |
- void send(data) { |
- _wsconn.send(data); |
- } |
- |
- WebSocketClientConnection _wsconn; |
- int _readyState = WebSocket.CONNECTING; |
- Function _onopen; |
- Function _onclose; |
- Function _onmessage; |
-} |
- |
- |
-class _WebSocketMessageEvent implements MessageEvent { |
- _WebSocketMessageEvent(this._data); |
- get data => _data; |
- var _data; |
-} |
- |
- |
-class _WebSocketCloseEvent implements CloseEvent { |
- _WebSocketCloseEvent(this._wasClean, this._code, this._reason); |
- bool get wasClean => _wasClean; |
- int get code => _code; |
- String get reason => _reason; |
- bool _wasClean; |
- int _code; |
- String _reason; |
-} |