| // Copyright 2014 The Flutter Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| import 'dart:async'; |
| import 'dart:io' as io; |
| |
| import 'package:flutter_devicelab/framework/browser.dart'; |
| import 'package:shelf/shelf.dart'; |
| import 'package:shelf/shelf_io.dart' as shelf_io; |
| import 'package:shelf_static/shelf_static.dart'; |
| |
| /// Runs Chrome, opens the given `appUrl`, and returns the result reported by the |
| /// app. |
| /// |
| /// The app is served from the `appDirectory`. Typically, the app is built |
| /// using `flutter build web` and served from `build/web`. |
| /// |
| /// The launched app is expected to report the result by sending an HTTP POST |
| /// request to "/test-result" containing result data as plain text body of the |
| /// request. This function has no opinion about what that string contains. |
| Future<String> evalTestAppInChrome({ |
| required String appUrl, |
| required String appDirectory, |
| int serverPort = 8080, |
| int browserDebugPort = 8081, |
| }) async { |
| io.HttpServer? server; |
| Chrome? chrome; |
| try { |
| final Completer<String> resultCompleter = Completer<String>(); |
| server = await io.HttpServer.bind('localhost', serverPort); |
| final Cascade cascade = Cascade() |
| .add((Request request) async { |
| if (request.requestedUri.path.endsWith('/test-result')) { |
| resultCompleter.complete(await request.readAsString()); |
| return Response.ok('Test results received'); |
| } |
| return Response.notFound(''); |
| }) |
| .add(createStaticHandler(appDirectory)); |
| shelf_io.serveRequests(server, cascade.handler); |
| final io.Directory userDataDirectory = io.Directory.systemTemp.createTempSync( |
| 'flutter_chrome_user_data.', |
| ); |
| chrome = await Chrome.launch( |
| ChromeOptions( |
| headless: true, |
| debugPort: browserDebugPort, |
| url: appUrl, |
| userDataDirectory: userDataDirectory.path, |
| windowHeight: 500, |
| windowWidth: 500, |
| ), |
| onError: resultCompleter.completeError, |
| ); |
| return await resultCompleter.future; |
| } finally { |
| chrome?.stop(); |
| await server?.close(); |
| } |
| } |
| |
| typedef ServerRequestListener = void Function(Request); |
| |
| class AppServer { |
| AppServer._(this._server, this.chrome, this.onChromeError); |
| |
| static Future<AppServer> start({ |
| required String appUrl, |
| required String appDirectory, |
| required String cacheControl, |
| int serverPort = 8080, |
| int browserDebugPort = 8081, |
| bool headless = true, |
| List<Handler>? additionalRequestHandlers, |
| }) async { |
| io.HttpServer server; |
| Chrome chrome; |
| server = await io.HttpServer.bind('localhost', serverPort); |
| final Handler staticHandler = createStaticHandler(appDirectory, defaultDocument: 'index.html'); |
| Cascade cascade = Cascade(); |
| if (additionalRequestHandlers != null) { |
| for (final Handler handler in additionalRequestHandlers) { |
| cascade = cascade.add(handler); |
| } |
| } |
| cascade = cascade.add((Request request) async { |
| final Response response = await staticHandler(request); |
| return response.change(headers: <String, Object>{'cache-control': cacheControl}); |
| }); |
| shelf_io.serveRequests(server, cascade.handler); |
| final io.Directory userDataDirectory = io.Directory.systemTemp.createTempSync( |
| 'flutter_chrome_user_data.', |
| ); |
| final Completer<String> chromeErrorCompleter = Completer<String>(); |
| chrome = await Chrome.launch( |
| ChromeOptions( |
| headless: headless, |
| debugPort: browserDebugPort, |
| url: appUrl, |
| userDataDirectory: userDataDirectory.path, |
| ), |
| onError: chromeErrorCompleter.complete, |
| ); |
| return AppServer._(server, chrome, chromeErrorCompleter.future); |
| } |
| |
| final Future<String> onChromeError; |
| final io.HttpServer _server; |
| final Chrome chrome; |
| |
| Future<void> stop() async { |
| chrome.stop(); |
| await _server.close(); |
| } |
| } |