blob: 53350d91793b56625e2db02a25aa2f4cd567f40b [file] [log] [blame]
// 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.
@JS()
library test.host;
import 'dart:async';
import 'dart:convert';
import 'dart:html';
import 'package:stack_trace/stack_trace.dart';
import 'package:stream_channel/stream_channel.dart';
import 'package:js/js.dart';
/// A class defined in content shell, used to control its behavior.
@JS()
class _TestRunner {
external void waitUntilDone();
}
/// Returns the current content shell runner, or `null` if none exists.
@JS()
external _TestRunner get testRunner;
/// A class that exposes the test API to JS.
///
/// These are exposed so that tools like IDEs can interact with them via remote
/// debugging.
@JS()
@anonymous
class _JSApi {
/// Causes the test runner to resume running, as though the user had clicked
/// the "play" button.
external Function get resume;
/// Causes the test runner to restart the current test once it finishes
/// running.
external Function get restartCurrent;
external factory _JSApi({void resume(), void restartCurrent()});
}
/// Sets the top-level `dartTest` object so that it's visible to JS.
@JS("dartTest")
external set _jsApi(_JSApi api);
/// The iframes created for each loaded test suite, indexed by the suite id.
final _iframes = new Map<int, IFrameElement>();
/// Subscriptions created for each loaded test suite, indexed by the suite id.
final _subscriptions = new Map<int, List<StreamSubscription>>();
/// The URL for the current page.
final _currentUrl = Uri.parse(window.location.href);
/// Code that runs in the browser and loads test suites at the server's behest.
///
/// One instance of this runs for each browser. When the server tells it to load
/// a test, it starts an iframe pointing at that test's code; from then on, it
/// just relays messages between the two.
///
/// The browser uses two layers of [MultiChannel]s when communicating with the
/// server:
///
/// server
/// │
/// (WebSocket)
/// │
/// ┏━ host.html ━━━━━━━━┿━━━━━━━━━━━━━━━━━┓
/// ┃ │ ┃
/// ┃ ┌──────┬───MultiChannel─────┐ ┃
/// ┃ │ │ │ │ │ ┃
/// ┃ host suite suite suite suite ┃
/// ┃ │ │ │ │ ┃
/// ┗━━━━━━━━━━━┿━━━━━━┿━━━━━━┿━━━━━━┿━━━━━┛
/// │ │ │ │
/// │ ... ... ...
/// │
/// (postMessage)
/// │
/// ┏━ suite.html (in iframe) ┿━━━━━━━━━━━━━━━━━━━━━━━━━━┓
/// ┃ │ ┃
/// ┃ ┌──────────MultiChannel┬─────────┐ ┃
/// ┃ │ │ │ │ │ ┃
/// ┃ IframeListener test test test running test ┃
/// ┃ ┃
/// ┗━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━┛
///
/// The host (this code) has a [MultiChannel] that splits the WebSocket
/// connection with the server. One connection is used for the host itself to
/// receive messages like "load a suite at this URL", and the rest are connected
/// to each test suite's iframe via `postMessage`.
///
/// Each iframe then has its own [MultiChannel] which takes its `postMessage`
/// connection and splits it again. One connection is used for the
/// [IframeListener], which sends messages like "here are all the tests in this
/// suite". The rest are used for each test, receiving messages like "start
/// running". A new connection is also created whenever a test begins running to
/// send status messages about its progress.
///
/// It's of particular note that the suite's [MultiChannel] connection uses the
/// host's purely as a transport layer; neither is aware that the other is also
/// using [MultiChannel]. This is necessary, since the host doesn't share memory
/// with the suites and thus can't share its [MultiChannel] with them, but it
/// does mean that the server needs to be sure to nest its [MultiChannel]s at
/// the same place the client does.
void main() {
// This tells content_shell not to close immediately after the page has
// rendered.
testRunner?.waitUntilDone();
if (_currentUrl.queryParameters['debug'] == 'true') {
document.body.classes.add('debug');
}
runZoned(() {
var serverChannel = _connectToServer();
serverChannel.stream.listen((message) {
if (message['command'] == 'loadSuite') {
var suiteChannel = serverChannel.virtualChannel(message['channel']);
var iframeChannel = _connectToIframe(message['url'], message['id']);
suiteChannel.pipe(iframeChannel);
} else if (message['command'] == 'displayPause') {
document.body.classes.add('paused');
} else if (message['command'] == 'resume') {
document.body.classes.remove('paused');
} else {
assert(message['command'] == 'closeSuite');
_iframes.remove(message['id']).remove();
for (var subscription in _subscriptions.remove(message['id'])) {
subscription.cancel();
}
}
});
// Send periodic pings to the test runner so it can know when the browser is
// paused for debugging.
new Timer.periodic(new Duration(seconds: 1),
(_) => serverChannel.sink.add({"command": "ping"}));
var play = document.querySelector("#play");
play.onClick.listen((_) {
if (!document.body.classes.remove('paused')) return;
serverChannel.sink.add({"command": "resume"});
});
_jsApi = new _JSApi(resume: allowInterop(() {
if (!document.body.classes.remove('paused')) return;
serverChannel.sink.add({"command": "resume"});
}), restartCurrent: allowInterop(() {
serverChannel.sink.add({"command": "restart"});
}));
}, onError: (error, stackTrace) {
print("$error\n${new Trace.from(stackTrace).terse}");
});
}
/// Creates a [MultiChannel] connection to the server, using a [WebSocket] as
/// the underlying protocol.
MultiChannel _connectToServer() {
// The `managerUrl` query parameter contains the WebSocket URL of the remote
// [BrowserManager] with which this communicates.
var webSocket = new WebSocket(_currentUrl.queryParameters['managerUrl']);
var controller = new StreamChannelController(sync: true);
webSocket.onMessage.listen((message) {
controller.local.sink.add(JSON.decode(message.data));
});
controller.local.stream
.listen((message) => webSocket.send(JSON.encode(message)));
return new MultiChannel(controller.foreign);
}
/// Creates an iframe with `src` [url] and establishes a connection to it using
/// `postMessage`.
///
/// [id] identifies the suite loaded in this iframe.
StreamChannel _connectToIframe(String url, int id) {
var iframe = new IFrameElement();
_iframes[id] = iframe;
iframe.src = url;
document.body.children.add(iframe);
var controller = new StreamChannelController(sync: true);
// Use this to avoid sending a message to the iframe before it's sent a
// message to us. This ensures that no messages get dropped on the floor.
var readyCompleter = new Completer();
var subscriptions = <StreamSubscription>[];
_subscriptions[id] = subscriptions;
// TODO(nweiz): use MessageChannel once Firefox supports it
// (http://caniuse.com/#search=MessageChannel).
subscriptions.add(window.onMessage.listen((message) {
// A message on the Window can theoretically come from any website. It's
// very unlikely that a malicious site would care about hacking someone's
// unit tests, let alone be able to find the test server while it's
// running, but it's good practice to check the origin anyway.
if (message.origin != window.location.origin) return;
// TODO(nweiz): Stop manually checking href here once issue 22554 is
// fixed.
if (message.data["href"] != iframe.src) return;
message.stopPropagation();
// This message indicates that the iframe is actively listening for events.
if (message.data["ready"] == true) {
readyCompleter.complete();
} else {
controller.local.sink.add(message.data["data"]);
}
}));
subscriptions.add(controller.local.stream.listen((message) async {
await readyCompleter.future;
iframe.contentWindow.postMessage(message, window.location.origin);
}));
return controller.foreign;
}