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 | |
63 /// Returns the source code for a Dart library defining a Transformer that | 60 /// Returns the source code for a Dart library defining a Transformer that |
64 /// rewrites Dart files. | 61 /// rewrites Dart files. |
65 /// | 62 /// |
66 /// The transformer defines a constant named TOKEN whose value is [id]. When the | 63 /// The transformer defines a constant named TOKEN whose value is [id]. When the |
67 /// transformer transforms another Dart file, it will look for a "TOKEN" | 64 /// transformer transforms another Dart file, it will look for a "TOKEN" |
68 /// constant definition there and modify it to include *this* transformer's | 65 /// constant definition there and modify it to include *this* transformer's |
69 /// TOKEN value as well. | 66 /// TOKEN value as well. |
70 /// | 67 /// |
71 /// If [import] is passed, it should be the name of a package that defines its | 68 /// If [import] is passed, it should be the name of a package that defines its |
72 /// own TOKEN constant. The primary library of that package will be imported | 69 /// 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... |
290 expect(_pubServer, isNotNull); | 287 expect(_pubServer, isNotNull); |
291 expect(_adminPort, isNotNull); | 288 expect(_adminPort, isNotNull); |
292 | 289 |
293 return WebSocket.connect("ws://127.0.0.1:$_adminPort").then((socket) { | 290 return WebSocket.connect("ws://127.0.0.1:$_adminPort").then((socket) { |
294 _webSocket = socket; | 291 _webSocket = socket; |
295 // TODO(rnystrom): Works around #13913. | 292 // TODO(rnystrom): Works around #13913. |
296 _webSocketBroadcastStream = _webSocket.asBroadcastStream(); | 293 _webSocketBroadcastStream = _webSocket.asBroadcastStream(); |
297 }); | 294 }); |
298 } | 295 } |
299 | 296 |
300 /// Sends a JSON RPC 2.0 request to the running pub serve's web socket | 297 /// Sends [request] (an arbitrary JSON object) to the running pub serve's web |
301 /// connection, waits for a reply, then verifies the result. | 298 /// socket connection, waits for a reply, then verifies that the reply is |
| 299 /// either equal to [replyEquals] or matches [replyMatches]. |
302 /// | 300 /// |
303 /// This calls a method named [method] with the given [params]. [params] may | 301 /// Only one of [replyEquals] or [replyMatches] may be provided. |
304 /// contain Futures, in which case this will wait until they've completed before | |
305 /// sending the request. | |
306 /// | 302 /// |
307 /// The result is validated using [result], which may be a [Matcher] or a [Map] | 303 /// [request], [replyEquals], and [replyMatches] may contain futures, in which |
308 /// containing [Matcher]s and [Future]s. This will wait until any futures are | 304 /// case this will wait until they've completed before matching. |
309 /// completed before sending the request. | |
310 /// | 305 /// |
311 /// Returns a [Future] that completes to the call's result. | 306 /// If [encodeRequest] is `false`, then [request] will be sent as-is over the |
312 Future<Map> expectWebSocketResult(String method, Map params, result) { | 307 /// socket. It omitted, request is JSON encoded to a string first. |
313 return schedule(() { | 308 /// |
314 return Future.wait([ | 309 /// Returns a [Future] that completes to the call's response. |
315 _ensureWebSocket(), | 310 Future<Map> expectWebSocketCall(request, {Map replyEquals, replyMatches, |
316 awaitObject(params), | 311 bool encodeRequest: true}) { |
317 awaitObject(result) | 312 assert((replyEquals == null) != (replyMatches == null)); |
318 ]).then((results) { | |
319 var resolvedParams = results[1]; | |
320 var resolvedResult = results[2]; | |
321 | 313 |
322 return _jsonRpcRequest(method, resolvedParams).then((response) { | 314 return schedule(() => _ensureWebSocket().then((_) { |
323 expect(response["result"], resolvedResult); | 315 var matcherFuture; |
324 return response["result"]; | 316 if (replyMatches != null) { |
| 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 }); |
325 }); | 332 }); |
326 }); | 333 }); |
327 }, "send $method with $params to web socket and expect $result"); | 334 }), "send $request to web socket and expect reply $replyEquals"); |
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 }); | |
385 } | 335 } |
386 | 336 |
387 /// Returns a [Future] that completes to a URL string for the server serving | 337 /// Returns a [Future] that completes to a URL string for the server serving |
388 /// [path] from [root]. | 338 /// [path] from [root]. |
389 /// | 339 /// |
390 /// If [root] is omitted, defaults to "web". If [path] is omitted, no path is | 340 /// If [root] is omitted, defaults to "web". If [path] is omitted, no path is |
391 /// included. The Future will complete once the server is up and running and | 341 /// included. The Future will complete once the server is up and running and |
392 /// the bound ports are known. | 342 /// the bound ports are known. |
393 Future<String> getServerUrl([String root, String path]) => | 343 Future<String> getServerUrl([String root, String path]) => |
394 _portsCompleter.future.then((_) => _getServerUrlSync(root, path)); | 344 _portsCompleter.future.then((_) => _getServerUrlSync(root, path)); |
(...skipping 12 matching lines...) Expand all Loading... |
407 /// included. Unlike [getServerUrl], this should only be called after the ports | 357 /// included. Unlike [getServerUrl], this should only be called after the ports |
408 /// are known. | 358 /// are known. |
409 String _getServerUrlSync([String root, String path]) { | 359 String _getServerUrlSync([String root, String path]) { |
410 if (root == null) root = 'web'; | 360 if (root == null) root = 'web'; |
411 expect(_ports, contains(root)); | 361 expect(_ports, contains(root)); |
412 var url = "http://127.0.0.1:${_ports[root]}"; | 362 var url = "http://127.0.0.1:${_ports[root]}"; |
413 if (path != null) url = "$url/$path"; | 363 if (path != null) url = "$url/$path"; |
414 return url; | 364 return url; |
415 } | 365 } |
416 | 366 |
OLD | NEW |