| Index: mojo/public/dart/third_party/test/lib/src/runner/vm/isolate_listener.dart
|
| diff --git a/mojo/public/dart/third_party/test/lib/src/runner/vm/isolate_listener.dart b/mojo/public/dart/third_party/test/lib/src/runner/vm/isolate_listener.dart
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..06e2d6a9693c4941ff8dd0e48c7cdd98af35cbb9
|
| --- /dev/null
|
| +++ b/mojo/public/dart/third_party/test/lib/src/runner/vm/isolate_listener.dart
|
| @@ -0,0 +1,162 @@
|
| +// Copyright (c) 2015, the Dart project authors. Please see the AUTHORS file
|
| +// for details. All rights reserved. Use of this source code is governed by a
|
| +// BSD-style license that can be found in the LICENSE file.
|
| +
|
| +library test.runner.vm.isolate_listener;
|
| +
|
| +import 'dart:isolate';
|
| +import 'dart:async';
|
| +
|
| +import 'package:stack_trace/stack_trace.dart';
|
| +
|
| +import '../../backend/declarer.dart';
|
| +import '../../backend/metadata.dart';
|
| +import '../../backend/suite.dart';
|
| +import '../../backend/test.dart';
|
| +import '../../backend/test_platform.dart';
|
| +import '../../util/io.dart';
|
| +import '../../util/remote_exception.dart';
|
| +import '../../utils.dart';
|
| +
|
| +/// A class that runs tests in a separate isolate and communicates the results
|
| +/// back to the main isolate.
|
| +class IsolateListener {
|
| + /// The test suite to run.
|
| + final Suite _suite;
|
| +
|
| + /// Extracts metadata about all the tests in the function returned by
|
| + /// [getMain] and sends information about them over [sendPort].
|
| + ///
|
| + /// The main function is wrapped in a closure so that we can handle it being
|
| + /// undefined here rather than in the generated code.
|
| + ///
|
| + /// Once that's done, this starts listening for commands about which tests to
|
| + /// run.
|
| + ///
|
| + /// [metadata] is the suite-level metadata defined at the top of the file.
|
| + static Future start(SendPort sendPort, Metadata metadata, Function getMain())
|
| + async {
|
| + // Capture any top-level errors (mostly lazy syntax errors, since other are
|
| + // caught below) and report them to the parent isolate. We set errors
|
| + // non-fatal because otherwise they'll be double-printed.
|
| + var errorPort = new ReceivePort();
|
| + Isolate.current.setErrorsFatal(false);
|
| + Isolate.current.addErrorListener(errorPort.sendPort);
|
| + errorPort.listen((message) {
|
| + // Masquerade as an IsoalteSpawnException because that's what this would
|
| + // be if the error had been detected statically.
|
| + var error = new IsolateSpawnException(message[0]);
|
| + var stackTrace =
|
| + message[1] == null ? new Trace([]) : new Trace.parse(message[1]);
|
| + sendPort.send({
|
| + "type": "error",
|
| + "error": RemoteException.serialize(error, stackTrace)
|
| + });
|
| + });
|
| +
|
| + var main;
|
| + try {
|
| + main = getMain();
|
| + } on NoSuchMethodError catch (_) {
|
| + _sendLoadException(sendPort, "No top-level main() function defined.");
|
| + return;
|
| + }
|
| +
|
| + if (main is! Function) {
|
| + _sendLoadException(sendPort, "Top-level main getter is not a function.");
|
| + return;
|
| + } else if (main is! AsyncFunction) {
|
| + _sendLoadException(
|
| + sendPort, "Top-level main() function takes arguments.");
|
| + return;
|
| + }
|
| +
|
| + var declarer = new Declarer();
|
| + try {
|
| + await runZoned(() => new Future.sync(main), zoneValues: {
|
| + #test.declarer: declarer
|
| + }, zoneSpecification: new ZoneSpecification(print: (_, __, ___, line) {
|
| + sendPort.send({"type": "print", "line": line});
|
| + }));
|
| + } catch (error, stackTrace) {
|
| + sendPort.send({
|
| + "type": "error",
|
| + "error": RemoteException.serialize(error, stackTrace)
|
| + });
|
| + return;
|
| + }
|
| +
|
| + var suite = new Suite(declarer.tests,
|
| + platform: TestPlatform.vm, os: currentOS, metadata: metadata);
|
| + new IsolateListener._(suite)._listen(sendPort);
|
| + }
|
| +
|
| + /// Sends a message over [sendPort] indicating that the tests failed to load.
|
| + ///
|
| + /// [message] should describe the failure.
|
| + static void _sendLoadException(SendPort sendPort, String message) {
|
| + sendPort.send({"type": "loadException", "message": message});
|
| + }
|
| +
|
| + IsolateListener._(this._suite);
|
| +
|
| + /// Send information about [_suite] across [sendPort] and start listening for
|
| + /// commands to run the tests.
|
| + void _listen(SendPort sendPort) {
|
| + var tests = [];
|
| + for (var i = 0; i < _suite.tests.length; i++) {
|
| + var test = _suite.tests[i];
|
| + var receivePort = new ReceivePort();
|
| + tests.add({
|
| + "name": test.name,
|
| + "metadata": test.metadata.serialize(),
|
| + "sendPort": receivePort.sendPort
|
| + });
|
| +
|
| + receivePort.listen((message) {
|
| + assert(message['command'] == 'run');
|
| + _runTest(test, message['reply']);
|
| + });
|
| + }
|
| +
|
| + sendPort.send({
|
| + "type": "success",
|
| + "tests": tests
|
| + });
|
| + }
|
| +
|
| + /// Runs [test] and sends the results across [sendPort].
|
| + void _runTest(Test test, SendPort sendPort) {
|
| + var liveTest = test.load(_suite);
|
| +
|
| + var receivePort = new ReceivePort();
|
| + sendPort.send({"type": "started", "reply": receivePort.sendPort});
|
| +
|
| + receivePort.listen((message) {
|
| + assert(message['command'] == 'close');
|
| + receivePort.close();
|
| + liveTest.close();
|
| + });
|
| +
|
| + liveTest.onStateChange.listen((state) {
|
| + sendPort.send({
|
| + "type": "state-change",
|
| + "status": state.status.name,
|
| + "result": state.result.name
|
| + });
|
| + });
|
| +
|
| + liveTest.onError.listen((asyncError) {
|
| + sendPort.send({
|
| + "type": "error",
|
| + "error": RemoteException.serialize(
|
| + asyncError.error, asyncError.stackTrace)
|
| + });
|
| + });
|
| +
|
| + liveTest.onPrint.listen((line) =>
|
| + sendPort.send({"type": "print", "line": line}));
|
| +
|
| + liveTest.run().then((_) => sendPort.send({"type": "complete"}));
|
| + }
|
| +}
|
|
|