OLD | NEW |
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 Loading... |
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 Loading... |
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 Loading... |
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 |
OLD | NEW |