Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(85)

Side by Side Diff: sdk/lib/_internal/pub/test/serve/utils.dart

Issue 207563002: Move pub's web socket API over to use JSON RPC 2.0. (Closed) Base URL: https://dart.googlecode.com/svn/branches/bleeding_edge/dart
Patch Set: code review Created 6 years, 9 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
OLDNEW
1 // Copyright (c) 2013, the Dart project authors. Please see the AUTHORS d.file 1 // Copyright (c) 2013, the Dart project authors. Please see the AUTHORS d.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 library pub_tests; 5 library pub_tests;
6 6
7 import 'dart:async'; 7 import 'dart:async';
8 import 'dart:convert'; 8 import 'dart:convert';
9 import 'dart:io'; 9 import 'dart:io';
10 10
(...skipping 39 matching lines...) Expand 10 before | Expand all | Expand 10 after
50 50
51 Future apply(Transform transform) { 51 Future apply(Transform transform) {
52 return transform.primaryInput.readAsString().then((contents) { 52 return transform.primaryInput.readAsString().then((contents) {
53 var id = transform.primaryInput.id.changeExtension(".out"); 53 var id = transform.primaryInput.id.changeExtension(".out");
54 transform.addOutput(new Asset.fromString(id, "\$contents.out")); 54 transform.addOutput(new Asset.fromString(id, "\$contents.out"));
55 }); 55 });
56 } 56 }
57 } 57 }
58 """; 58 """;
59 59
60 /// The web socket error code for a directory not being served.
61 const NOT_SERVED = 1;
62
60 /// Returns the source code for a Dart library defining a Transformer that 63 /// Returns the source code for a Dart library defining a Transformer that
61 /// rewrites Dart files. 64 /// rewrites Dart files.
62 /// 65 ///
63 /// The transformer defines a constant named TOKEN whose value is [id]. When the 66 /// The transformer defines a constant named TOKEN whose value is [id]. When the
64 /// transformer transforms another Dart file, it will look for a "TOKEN" 67 /// transformer transforms another Dart file, it will look for a "TOKEN"
65 /// constant definition there and modify it to include *this* transformer's 68 /// constant definition there and modify it to include *this* transformer's
66 /// TOKEN value as well. 69 /// TOKEN value as well.
67 /// 70 ///
68 /// If [import] is passed, it should be the name of a package that defines its 71 /// If [import] is passed, it should be the name of a package that defines its
69 /// own TOKEN constant. The primary library of that package will be imported 72 /// own TOKEN constant. The primary library of that package will be imported
(...skipping 217 matching lines...) Expand 10 before | Expand all | Expand 10 after
287 expect(_pubServer, isNotNull); 290 expect(_pubServer, isNotNull);
288 expect(_adminPort, isNotNull); 291 expect(_adminPort, isNotNull);
289 292
290 return WebSocket.connect("ws://127.0.0.1:$_adminPort").then((socket) { 293 return WebSocket.connect("ws://127.0.0.1:$_adminPort").then((socket) {
291 _webSocket = socket; 294 _webSocket = socket;
292 // TODO(rnystrom): Works around #13913. 295 // TODO(rnystrom): Works around #13913.
293 _webSocketBroadcastStream = _webSocket.asBroadcastStream(); 296 _webSocketBroadcastStream = _webSocket.asBroadcastStream();
294 }); 297 });
295 } 298 }
296 299
297 /// Sends [request] (an arbitrary JSON object) to the running pub serve's web 300 /// Sends a JSON RPC 2.0 request to the running pub serve's web socket
298 /// socket connection, waits for a reply, then verifies that the reply is 301 /// connection, waits for a reply, then verifies the result.
299 /// either equal to [replyEquals] or matches [replyMatches].
300 /// 302 ///
301 /// Only one of [replyEquals] or [replyMatches] may be provided. 303 /// This calls a method named [method] with the given [params]. [params] may
304 /// contain Futures, in which case this will wait until they've completed before
305 /// sending the request.
302 /// 306 ///
303 /// [request], [replyEquals], and [replyMatches] may contain futures, in which 307 /// The result is validated using [result], which may be a [Matcher] or a [Map]
304 /// case this will wait until they've completed before matching. 308 /// containing [Matcher]s and [Future]s. This will wait until any futures are
309 /// completed before sending the request.
305 /// 310 ///
306 /// If [encodeRequest] is `false`, then [request] will be sent as-is over the 311 /// Returns a [Future] that completes to the call's result.
307 /// socket. It omitted, request is JSON encoded to a string first. 312 Future<Map> expectWebSocketResult(String method, Map params, result) {
308 /// 313 return schedule(() {
309 /// Returns a [Future] that completes to the call's response. 314 return Future.wait([
310 Future<Map> expectWebSocketCall(request, {Map replyEquals, replyMatches, 315 _ensureWebSocket(),
311 bool encodeRequest: true}) { 316 awaitObject(params),
312 assert((replyEquals == null) != (replyMatches == null)); 317 awaitObject(result)
318 ]).then((results) {
319 var resolvedParams = results[1];
320 var resolvedResult = results[2];
313 321
314 return schedule(() => _ensureWebSocket().then((_) { 322 return _jsonRpcRequest(method, resolvedParams).then((response) {
315 var matcherFuture; 323 expect(response["result"], resolvedResult);
316 if (replyMatches != null) { 324 return response["result"];
317 matcherFuture = awaitObject(replyMatches);
318 } else {
319 matcherFuture = awaitObject(replyEquals).then((reply) => equals(reply));
320 }
321
322 return matcherFuture.then((matcher) {
323 return awaitObject(request).then((completeRequest) {
324 if (encodeRequest) completeRequest = JSON.encode(completeRequest);
325 _webSocket.add(completeRequest);
326
327 return _webSocketBroadcastStream.first.then((value) {
328 value = JSON.decode(value);
329 expect(value, matcher);
330 return value;
331 });
332 }); 325 });
333 }); 326 });
334 }), "send $request to web socket and expect reply $replyEquals"); 327 }, "send $method with $params to web socket and expect $result");
328 }
329
330 /// Sends a JSON RPC 2.0 request to the running pub serve's web socket
331 /// connection, waits for a reply, then verifies the error response.
332 ///
333 /// This calls a method named [method] with the given [params]. [params] may
334 /// contain Futures, in which case this will wait until they've completed before
335 /// sending the request.
336 ///
337 /// The error response is validated using [errorCode] and [errorMessage]. Both
338 /// of these must be provided. The error code is checked against [errorCode] and
339 /// the error message is checked against [errorMessage]. Either of these may be
340 /// matchers.
341 ///
342 /// Returns a [Future] that completes to the error's [data] field.
343 Future expectWebSocketError(String method, Map params, errorCode,
344 errorMessage) {
345 return schedule(() {
346 return Future.wait([
347 _ensureWebSocket(),
348 awaitObject(params)
349 ]).then((results) {
350 var resolvedParams = results[1];
351 return _jsonRpcRequest(method, resolvedParams);
352 }).then((response) {
353 expect(response["error"]["code"], errorCode);
354 expect(response["error"]["message"], errorMessage);
355
356 return response["error"]["data"];
357 });
358 }, "send $method with $params to web socket and expect error $errorCode");
359 }
360
361 /// The next id to use for a JSON-RPC 2.0 request.
362 var _rpcId = 0;
363
364 /// Sends a JSON-RPC 2.0 request calling [method] with [params].
365 ///
366 /// Returns the response object.
367 Future<Map> _jsonRpcRequest(String method, Map params) {
368 var id = _rpcId++;
369 _webSocket.add(JSON.encode({
370 "jsonrpc": "2.0",
371 "method": method,
372 "params": params,
373 "id": id
374 }));
375
376 return _webSocketBroadcastStream.first.then((value) {
377 value = JSON.decode(value);
378 currentSchedule.addDebugInfo(
379 "Web Socket request $method with params $params\n"
380 "Result: $value");
381
382 expect(value["id"], equals(id));
383 return value;
384 });
335 } 385 }
336 386
337 /// Returns a [Future] that completes to a URL string for the server serving 387 /// Returns a [Future] that completes to a URL string for the server serving
338 /// [path] from [root]. 388 /// [path] from [root].
339 /// 389 ///
340 /// If [root] is omitted, defaults to "web". If [path] is omitted, no path is 390 /// If [root] is omitted, defaults to "web". If [path] is omitted, no path is
341 /// included. The Future will complete once the server is up and running and 391 /// included. The Future will complete once the server is up and running and
342 /// the bound ports are known. 392 /// the bound ports are known.
343 Future<String> getServerUrl([String root, String path]) => 393 Future<String> getServerUrl([String root, String path]) =>
344 _portsCompleter.future.then((_) => _getServerUrlSync(root, path)); 394 _portsCompleter.future.then((_) => _getServerUrlSync(root, path));
(...skipping 12 matching lines...) Expand all
357 /// included. Unlike [getServerUrl], this should only be called after the ports 407 /// included. Unlike [getServerUrl], this should only be called after the ports
358 /// are known. 408 /// are known.
359 String _getServerUrlSync([String root, String path]) { 409 String _getServerUrlSync([String root, String path]) {
360 if (root == null) root = 'web'; 410 if (root == null) root = 'web';
361 expect(_ports, contains(root)); 411 expect(_ports, contains(root));
362 var url = "http://127.0.0.1:${_ports[root]}"; 412 var url = "http://127.0.0.1:${_ports[root]}";
363 if (path != null) url = "$url/$path"; 413 if (path != null) url = "$url/$path";
364 return url; 414 return url;
365 } 415 }
366 416
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698