| OLD | NEW |
| 1 // Copyright (c) 2015, the Dart project authors. Please see the AUTHORS file | 1 // Copyright (c) 2015, the Dart project authors. Please see the AUTHORS file |
| 2 // for details. All rights reserved. Use of this source code is governed by a | 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. | 3 // BSD-style license that can be found in the LICENSE file. |
| 4 | 4 |
| 5 part of dart.developer; | 5 part of dart.developer; |
| 6 | 6 |
| 7 class ServiceExtensionResponse { | 7 class ServiceExtensionResponse { |
| 8 final String _result; | 8 final String _result; |
| 9 final int _errorCode; | 9 final int _errorCode; |
| 10 final String _errorDetail; | 10 final String _errorDetail; |
| (...skipping 68 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 79 | 79 |
| 80 /// A service protocol extension handler. Registered with [registerExtension]. | 80 /// A service protocol extension handler. Registered with [registerExtension]. |
| 81 /// | 81 /// |
| 82 /// Must complete to a [ServiceExtensionResponse]. | 82 /// Must complete to a [ServiceExtensionResponse]. |
| 83 /// | 83 /// |
| 84 /// [method] - the method name. | 84 /// [method] - the method name. |
| 85 /// [parameters] - the parameters. | 85 /// [parameters] - the parameters. |
| 86 typedef Future<ServiceExtensionResponse> | 86 typedef Future<ServiceExtensionResponse> |
| 87 ServiceExtensionHandler(String method, Map parameters); | 87 ServiceExtensionHandler(String method, Map parameters); |
| 88 | 88 |
| 89 final _extensions = new Map<String, ServiceExtensionHandler>(); | |
| 90 | |
| 91 /// Register a [ServiceExtensionHandler] that will be invoked in this isolate | 89 /// Register a [ServiceExtensionHandler] that will be invoked in this isolate |
| 92 /// for [method]. | 90 /// for [method]. |
| 93 void registerExtension(String method, ServiceExtensionHandler handler) { | 91 void registerExtension(String method, ServiceExtensionHandler handler) { |
| 94 if (_extensions[method] != null) { | 92 if (method is! String) { |
| 93 throw new ArgumentError.value(method, |
| 94 'method', |
| 95 'Must be a String'); |
| 96 } |
| 97 if (_lookupExtension(method) != null) { |
| 95 throw new ArgumentError('Extension already registered: $method'); | 98 throw new ArgumentError('Extension already registered: $method'); |
| 96 } | 99 } |
| 97 if (handler is! ServiceExtensionHandler) { | 100 if (handler is! ServiceExtensionHandler) { |
| 98 throw new ArgumentError.value(handler, | 101 throw new ArgumentError.value(handler, |
| 99 'handler', | 102 'handler', |
| 100 'Must be a ServiceExtensionHandler'); | 103 'Must be a ServiceExtensionHandler'); |
| 101 } | 104 } |
| 102 _extensions[method] = handler; | 105 _registerExtension(method, handler); |
| 103 } | 106 } |
| 104 | 107 |
| 105 bool _scheduleExtension(String method, | 108 external ServiceExtensionHandler _lookupExtension(String method); |
| 106 List<String> parameterKeys, | 109 external _registerExtension(String method, ServiceExtensionHandler handler); |
| 107 List<String> parameterValues, | 110 |
| 108 SendPort replyPort, | 111 // This code is only invoked when there is no other Dart code on the stack. |
| 109 Object id) { | 112 _runExtension(ServiceExtensionHandler handler, |
| 110 ServiceExtensionHandler handler = _extensions[method]; | 113 String method, |
| 111 if (handler == null) { | 114 List<String> parameterKeys, |
| 112 return false; | 115 List<String> parameterValues, |
| 116 SendPort replyPort, |
| 117 Object id) { |
| 118 var parameters = {}; |
| 119 for (var i = 0; i < parameterKeys.length; i++) { |
| 120 parameters[parameterKeys[i]] = parameterValues[i]; |
| 113 } | 121 } |
| 114 // Defer execution of handler until next event loop. | 122 var response; |
| 115 Timer.run(() { | 123 try { |
| 116 var parameters = {}; | 124 response = handler(method, parameters); |
| 117 for (var i = 0; i < parameterKeys.length; i++) { | 125 } catch (e, st) { |
| 118 parameters[parameterKeys[i]] = parameterValues[i]; | 126 var errorDetails = (st == null) ? '$e' : '$e\n$st'; |
| 119 } | 127 response = new ServiceExtensionResponse.error( |
| 120 var response; | 128 ServiceExtensionResponse.kExtensionError, |
| 121 try { | 129 errorDetails); |
| 122 response = handler(method, parameters); | 130 _postResponse(replyPort, id, response); |
| 123 } catch (e, st) { | 131 return; |
| 124 var errorDetails = (st == null) ? '$e' : '$e\n$st'; | 132 } |
| 133 if (response is! Future) { |
| 134 response = new ServiceExtensionResponse.error( |
| 135 ServiceExtensionResponse.kExtensionError, |
| 136 "Extension handler must return a Future"); |
| 137 _postResponse(replyPort, id, response); |
| 138 return; |
| 139 } |
| 140 response.catchError((e, st) { |
| 141 // Catch any errors eagerly and wrap them in a ServiceExtensionResponse. |
| 142 var errorDetails = (st == null) ? '$e' : '$e\n$st'; |
| 143 return new ServiceExtensionResponse.error( |
| 144 ServiceExtensionResponse.kExtensionError, |
| 145 errorDetails); |
| 146 }).then((response) { |
| 147 // Post the valid response or the wrapped error after verifying that |
| 148 // the response is a ServiceExtensionResponse. |
| 149 if (response is! ServiceExtensionResponse) { |
| 125 response = new ServiceExtensionResponse.error( | 150 response = new ServiceExtensionResponse.error( |
| 126 ServiceExtensionResponse.kExtensionError, | 151 ServiceExtensionResponse.kExtensionError, |
| 127 errorDetails); | 152 "Extension handler must complete to a ServiceExtensionResponse"); |
| 128 _postResponse(replyPort, id, response); | |
| 129 return; | |
| 130 } | 153 } |
| 131 if (response is! Future) { | 154 _postResponse(replyPort, id, response); |
| 132 response = new ServiceExtensionResponse.error( | 155 }).catchError((e, st) { |
| 133 ServiceExtensionResponse.kExtensionError, | 156 // We do not expect any errors to occur in the .then or .catchError blocks |
| 134 "Extension handler must return a Future"); | 157 // but, suppress them just in case. |
| 135 _postResponse(replyPort, id, response); | |
| 136 return; | |
| 137 } | |
| 138 response.catchError((e, st) { | |
| 139 var errorDetails = (st == null) ? '$e' : '$e\n$st'; | |
| 140 return new ServiceExtensionResponse.error( | |
| 141 ServiceExtensionResponse.kExtensionError, | |
| 142 errorDetails); | |
| 143 }).then((response) { | |
| 144 if (response == null) { | |
| 145 response = new ServiceExtensionResponse.error( | |
| 146 ServiceExtensionResponse.kExtensionError, | |
| 147 "Extension handler returned null"); | |
| 148 } | |
| 149 _postResponse(replyPort, id, response); | |
| 150 }); | |
| 151 }); | 158 }); |
| 152 return true; | |
| 153 } | 159 } |
| 154 | 160 |
| 161 // This code is only invoked by _runExtension. |
| 155 _postResponse(SendPort replyPort, | 162 _postResponse(SendPort replyPort, |
| 156 Object id, | 163 Object id, |
| 157 ServiceExtensionResponse response) { | 164 ServiceExtensionResponse response) { |
| 158 assert(replyPort != null); | 165 assert(replyPort != null); |
| 159 if (id == null) { | 166 if (id == null) { |
| 160 // No id -> no response. | 167 // No id -> no response. |
| 161 replyPort.send(null); | 168 replyPort.send(null); |
| 162 return; | 169 return; |
| 163 } | 170 } |
| 164 assert(id != null); | 171 assert(id != null); |
| 165 StringBuffer sb = new StringBuffer(); | 172 StringBuffer sb = new StringBuffer(); |
| 166 sb.write('{"jsonrpc":"2.0",'); | 173 sb.write('{"jsonrpc":"2.0",'); |
| 167 if (response._isError()) { | 174 if (response._isError()) { |
| 168 sb.write('"error":'); | 175 sb.write('"error":'); |
| 169 } else { | 176 } else { |
| 170 sb.write('"result":'); | 177 sb.write('"result":'); |
| 171 } | 178 } |
| 172 sb.write('${response._toString()},'); | 179 sb.write('${response._toString()},'); |
| 173 if (id is String) { | 180 if (id is String) { |
| 174 sb.write('"id":"$id"}'); | 181 sb.write('"id":"$id"}'); |
| 175 } else { | 182 } else { |
| 176 sb.write('"id":$id}'); | 183 sb.write('"id":$id}'); |
| 177 } | 184 } |
| 178 replyPort.send(sb.toString()); | 185 replyPort.send(sb.toString()); |
| 179 } | 186 } |
| OLD | NEW |