| OLD | NEW |
| (Empty) |
| 1 // Copyright (c) 2014, the Dart project authors. Please see the AUTHORS file | |
| 2 // for details. All rights reserved. Use of this source code is governed by a | |
| 3 // BSD-style license that can be found in the LICENSE file. | |
| 4 | |
| 5 library service_common; | |
| 6 | |
| 7 import 'dart:async'; | |
| 8 import 'dart:convert'; | |
| 9 import 'dart:typed_data'; | |
| 10 | |
| 11 import 'package:logging/logging.dart'; | |
| 12 import 'package:observatory/service.dart'; | |
| 13 | |
| 14 // Export the service library. | |
| 15 export 'package:observatory/service.dart'; | |
| 16 | |
| 17 /// Description of a VM target. | |
| 18 class WebSocketVMTarget { | |
| 19 // Last time this VM has been connected to. | |
| 20 int lastConnectionTime = 0; | |
| 21 bool get hasEverConnected => lastConnectionTime > 0; | |
| 22 | |
| 23 // Chrome VM or standalone; | |
| 24 bool chrome = false; | |
| 25 bool get standalone => !chrome; | |
| 26 | |
| 27 // User defined name. | |
| 28 String name; | |
| 29 // Network address of VM. | |
| 30 String networkAddress; | |
| 31 | |
| 32 WebSocketVMTarget(this.networkAddress) { | |
| 33 name = networkAddress; | |
| 34 } | |
| 35 | |
| 36 WebSocketVMTarget.fromMap(Map json) { | |
| 37 lastConnectionTime = json['lastConnectionTime']; | |
| 38 chrome = json['chrome']; | |
| 39 name = json['name']; | |
| 40 networkAddress = json['networkAddress']; | |
| 41 if (name == null) { | |
| 42 name = networkAddress; | |
| 43 } | |
| 44 } | |
| 45 | |
| 46 Map toJson() { | |
| 47 return { | |
| 48 'lastConnectionTime': lastConnectionTime, | |
| 49 'chrome': chrome, | |
| 50 'name': name, | |
| 51 'networkAddress': networkAddress, | |
| 52 }; | |
| 53 } | |
| 54 } | |
| 55 | |
| 56 class _WebSocketRequest { | |
| 57 final String id; | |
| 58 final Completer<String> completer; | |
| 59 _WebSocketRequest(this.id) | |
| 60 : completer = new Completer<String>(); | |
| 61 } | |
| 62 | |
| 63 /// Minimal common interface for 'WebSocket' in [dart:io] and [dart:html]. | |
| 64 abstract class CommonWebSocket { | |
| 65 void connect(String address, | |
| 66 void onOpen(), | |
| 67 void onMessage(dynamic data), | |
| 68 void onError(), | |
| 69 void onClose()); | |
| 70 bool get isOpen; | |
| 71 void send(dynamic data); | |
| 72 void close(); | |
| 73 Future<ByteData> nonStringToByteData(dynamic data); | |
| 74 } | |
| 75 | |
| 76 /// A [CommonWebSocketVM] communicates with a Dart VM over a CommonWebSocket. | |
| 77 /// The Dart VM can be embedded in Chromium or standalone. In the case of | |
| 78 /// Chromium, we make the service requests via the Chrome Remote Debugging | |
| 79 /// Protocol. | |
| 80 abstract class CommonWebSocketVM extends VM { | |
| 81 final Completer _connected = new Completer(); | |
| 82 final Completer _disconnected = new Completer(); | |
| 83 final WebSocketVMTarget target; | |
| 84 final Map<String, _WebSocketRequest> _delayedRequests = | |
| 85 new Map<String, _WebSocketRequest>(); | |
| 86 final Map<String, _WebSocketRequest> _pendingRequests = | |
| 87 new Map<String, _WebSocketRequest>(); | |
| 88 int _requestSerial = 0; | |
| 89 bool _hasInitiatedConnect = false; | |
| 90 bool _hasFinishedConnect = false; | |
| 91 Utf8Decoder _utf8Decoder = new Utf8Decoder(); | |
| 92 | |
| 93 CommonWebSocket _webSocket; | |
| 94 | |
| 95 CommonWebSocketVM(this.target, this._webSocket) { | |
| 96 assert(target != null); | |
| 97 } | |
| 98 | |
| 99 void _notifyConnect() { | |
| 100 _hasFinishedConnect = true; | |
| 101 if (!_connected.isCompleted) { | |
| 102 Logger.root.info('WebSocketVM connection opened: ${target.networkAddress}'
); | |
| 103 _connected.complete(this); | |
| 104 } | |
| 105 } | |
| 106 Future get onConnect => _connected.future; | |
| 107 void _notifyDisconnect() { | |
| 108 if (!_hasFinishedConnect) { | |
| 109 return; | |
| 110 } | |
| 111 if (!_disconnected.isCompleted) { | |
| 112 Logger.root.info('WebSocketVM connection error: ${target.networkAddress}')
; | |
| 113 _disconnected.complete(this); | |
| 114 } | |
| 115 } | |
| 116 Future get onDisconnect => _disconnected.future; | |
| 117 | |
| 118 void disconnect() { | |
| 119 if (_hasInitiatedConnect) { | |
| 120 _webSocket.close(); | |
| 121 } | |
| 122 _cancelAllRequests(); | |
| 123 _notifyDisconnect(); | |
| 124 } | |
| 125 | |
| 126 Future<String> getString(String id) { | |
| 127 if (!_hasInitiatedConnect) { | |
| 128 _hasInitiatedConnect = true; | |
| 129 _webSocket.connect( | |
| 130 target.networkAddress, _onOpen, _onMessage, _onError, _onClose); | |
| 131 } | |
| 132 return _makeRequest(id); | |
| 133 } | |
| 134 | |
| 135 /// Add a request for [id] to pending requests. | |
| 136 Future<String> _makeRequest(String id) { | |
| 137 assert(_hasInitiatedConnect); | |
| 138 // Create request. | |
| 139 String serial = (_requestSerial++).toString(); | |
| 140 var request = new _WebSocketRequest(id); | |
| 141 if (_webSocket.isOpen) { | |
| 142 // Already connected, send request immediately. | |
| 143 _sendRequest(serial, request); | |
| 144 } else { | |
| 145 // Not connected yet, add to delayed requests. | |
| 146 _delayedRequests[serial] = request; | |
| 147 } | |
| 148 return request.completer.future; | |
| 149 } | |
| 150 | |
| 151 void _onClose() { | |
| 152 _cancelAllRequests(); | |
| 153 _notifyDisconnect(); | |
| 154 } | |
| 155 | |
| 156 // WebSocket error event handler. | |
| 157 void _onError() { | |
| 158 _cancelAllRequests(); | |
| 159 _notifyDisconnect(); | |
| 160 } | |
| 161 | |
| 162 // WebSocket open event handler. | |
| 163 void _onOpen() { | |
| 164 target.lastConnectionTime = new DateTime.now().millisecondsSinceEpoch; | |
| 165 _sendAllDelayedRequests(); | |
| 166 _notifyConnect(); | |
| 167 } | |
| 168 | |
| 169 // WebSocket message event handler. | |
| 170 void _onMessage(dynamic data) { | |
| 171 if (data is! String) { | |
| 172 _webSocket.nonStringToByteData(data).then((ByteData bytes) { | |
| 173 // See format spec. in VMs Service::SendEvent. | |
| 174 int offset = 0; | |
| 175 int metaSize = bytes.getUint64(offset, Endianness.BIG_ENDIAN); | |
| 176 offset += 8; | |
| 177 var meta = _utf8Decoder.convert(new Uint8List.view( | |
| 178 bytes.buffer, bytes.offsetInBytes + offset, metaSize)); | |
| 179 offset += metaSize; | |
| 180 var data = new ByteData.view( | |
| 181 bytes.buffer, | |
| 182 bytes.offsetInBytes + offset, | |
| 183 bytes.lengthInBytes - offset); | |
| 184 postEventMessage(meta, data); | |
| 185 }); | |
| 186 return; | |
| 187 } | |
| 188 var map = JSON.decode(data); | |
| 189 if (map == null) { | |
| 190 Logger.root.severe('WebSocketVM got empty message'); | |
| 191 return; | |
| 192 } | |
| 193 // Extract serial and response. | |
| 194 var serial; | |
| 195 var response; | |
| 196 if (target.chrome) { | |
| 197 if (map['method'] != 'Dart.observatoryData') { | |
| 198 // ignore devtools protocol spam. | |
| 199 return; | |
| 200 } | |
| 201 serial = map['params']['id'].toString(); | |
| 202 response = map['params']['data']; | |
| 203 } else { | |
| 204 serial = map['seq']; | |
| 205 response = map['response']; | |
| 206 } | |
| 207 if (serial == null) { | |
| 208 // Messages without sequence numbers are asynchronous events | |
| 209 // from the vm. | |
| 210 postEventMessage(response); | |
| 211 return; | |
| 212 } | |
| 213 // Complete request. | |
| 214 var request = _pendingRequests.remove(serial); | |
| 215 if (request == null) { | |
| 216 Logger.root.severe('Received unexpected message: ${map}'); | |
| 217 return; | |
| 218 } | |
| 219 request.completer.complete(response); | |
| 220 } | |
| 221 | |
| 222 String _generateNetworkError(String userMessage) { | |
| 223 return JSON.encode({ | |
| 224 'type': 'ServiceException', | |
| 225 'id': '', | |
| 226 'kind': 'NetworkException', | |
| 227 'message': userMessage | |
| 228 }); | |
| 229 } | |
| 230 | |
| 231 void _cancelRequests(Map<String, _WebSocketRequest> requests) { | |
| 232 requests.forEach((String serial, _WebSocketRequest request) { | |
| 233 request.completer.complete( | |
| 234 _generateNetworkError('WebSocket disconnected')); | |
| 235 }); | |
| 236 requests.clear(); | |
| 237 } | |
| 238 | |
| 239 /// Cancel all pending and delayed requests by completing them with an error. | |
| 240 void _cancelAllRequests() { | |
| 241 if (_pendingRequests.length > 0) { | |
| 242 Logger.root.info('Cancelling all pending requests.'); | |
| 243 _cancelRequests(_pendingRequests); | |
| 244 } | |
| 245 if (_delayedRequests.length > 0) { | |
| 246 Logger.root.info('Cancelling all delayed requests.'); | |
| 247 _cancelRequests(_delayedRequests); | |
| 248 } | |
| 249 } | |
| 250 | |
| 251 /// Send all delayed requests. | |
| 252 void _sendAllDelayedRequests() { | |
| 253 assert(_webSocket.isOpen); | |
| 254 if (_delayedRequests.length == 0) { | |
| 255 return; | |
| 256 } | |
| 257 Logger.root.info('Sending all delayed requests.'); | |
| 258 // Send all delayed requests. | |
| 259 _delayedRequests.forEach(_sendRequest); | |
| 260 // Clear all delayed requests. | |
| 261 _delayedRequests.clear(); | |
| 262 } | |
| 263 | |
| 264 /// Send the request over WebSocket. | |
| 265 void _sendRequest(String serial, _WebSocketRequest request) { | |
| 266 assert (_webSocket.isOpen); | |
| 267 if (!request.id.endsWith('/profile/tag')) { | |
| 268 Logger.root.info('GET ${request.id} from ${target.networkAddress}'); | |
| 269 } | |
| 270 // Mark request as pending. | |
| 271 assert(_pendingRequests.containsKey(serial) == false); | |
| 272 _pendingRequests[serial] = request; | |
| 273 var message; | |
| 274 // Encode message. | |
| 275 if (target.chrome) { | |
| 276 message = JSON.encode({ | |
| 277 'id': int.parse(serial), | |
| 278 'method': 'Dart.observatoryQuery', | |
| 279 'params': { | |
| 280 'id': serial, | |
| 281 'query': request.id | |
| 282 } | |
| 283 }); | |
| 284 } else { | |
| 285 message = JSON.encode({'seq': serial, 'request': request.id}); | |
| 286 } | |
| 287 // Send message. | |
| 288 _webSocket.send(message); | |
| 289 } | |
| 290 } | |
| OLD | NEW |