blob: 16a9a0ce80c6de97600c391dffb63dfcafd4e263 [file] [log] [blame]
// Copyright (c) 2019, 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.
@TestOn('vm')
import 'dart:async';
import 'dart:io';
import 'package:shelf/shelf.dart' as shelf;
import 'package:shelf/shelf_io.dart' as io;
import 'package:shelf_static/shelf_static.dart';
import 'package:sse/server/sse_handler.dart';
import 'package:test/test.dart';
import 'package:webdriver/io.dart';
void main() {
HttpServer server;
WebDriver webdriver;
SseHandler handler;
Process chromeDriver;
setUpAll(() async {
try {
chromeDriver = await Process.start(
'chromedriver', ['--port=4444', '--url-base=wd/hub']);
} catch (e) {
throw StateError(
'Could not start ChromeDriver. Is it installed?\nError: $e');
}
});
tearDownAll(() {
chromeDriver.kill();
});
setUp(() async {
handler = SseHandler(Uri.parse('/test'));
var cascade = new shelf.Cascade()
.add(handler.handler)
.add(_faviconHandler)
.add(createStaticHandler('test/web',
listDirectories: true, defaultDocument: 'index.html'));
server = await io.serve(cascade.handler, 'localhost', 0);
webdriver = await createDriver(desired: {
'chromeOptions': {
'args': ['--headless']
}
});
});
tearDown(() async {
await webdriver.quit();
await server.close();
});
test('Can round trip messages', () async {
await webdriver.get('http://localhost:${server.port}');
var connection = await handler.connections.next;
connection.sink.add('blah');
expect(await connection.stream.first, 'blah');
});
test('Multiple clients can connect', () async {
var connections = handler.connections;
await webdriver.get('http://localhost:${server.port}');
await connections.next;
await webdriver.get('http://localhost:${server.port}');
await connections.next;
});
test('Routes data correctly', () async {
var connections = handler.connections;
await webdriver.get('http://localhost:${server.port}');
var connectionA = await connections.next;
connectionA.sink.add('foo');
expect(await connectionA.stream.first, 'foo');
await webdriver.get('http://localhost:${server.port}');
var connectionB = await connections.next;
connectionB.sink.add('bar');
expect(await connectionB.stream.first, 'bar');
});
test('Can close from the server', () async {
expect(handler.numberOfClients, 0);
await webdriver.get('http://localhost:${server.port}');
var connection = await handler.connections.next;
expect(handler.numberOfClients, 1);
await connection.sink.close();
await pumpEventQueue();
expect(handler.numberOfClients, 0);
});
test('Can close from the client-side', () async {
expect(handler.numberOfClients, 0);
await webdriver.get('http://localhost:${server.port}');
var connection = await handler.connections.next;
expect(handler.numberOfClients, 1);
var closeButton = await webdriver.findElement(const By.tagName('button'));
await closeButton.click();
// Should complete since the connection is closed.
await connection.stream.toList();
expect(handler.numberOfClients, 0);
});
test('Cancelling the listener closes the connection', () async {
expect(handler.numberOfClients, 0);
await webdriver.get('http://localhost:${server.port}');
var connection = await handler.connections.next;
expect(handler.numberOfClients, 1);
var sub = connection.stream.listen((_) {});
await sub.cancel();
await pumpEventQueue();
expect(handler.numberOfClients, 0);
});
test('Disconnects when navigating away', () async {
await webdriver.get('http://localhost:${server.port}');
expect(handler.numberOfClients, 1);
await webdriver.get('chrome://version/');
expect(handler.numberOfClients, 0);
});
}
FutureOr<shelf.Response> _faviconHandler(shelf.Request request) {
if (request.url.path.endsWith('favicon.ico')) {
return new shelf.Response.ok('');
}
return new shelf.Response.notFound('');
}