Unverified Commit 611e5cb4 authored by Jonah Williams's avatar Jonah Williams Committed by GitHub

[flutter_tools] update chrome tests to remove globals, mock processmanager, mock process (#51533)

parent ee4b40a1
...@@ -4,99 +4,126 @@ ...@@ -4,99 +4,126 @@
import 'dart:async'; import 'dart:async';
import 'package:file/memory.dart';
import 'package:flutter_tools/src/base/file_system.dart'; import 'package:flutter_tools/src/base/file_system.dart';
import 'package:flutter_tools/src/base/logger.dart';
import 'package:flutter_tools/src/base/os.dart'; import 'package:flutter_tools/src/base/os.dart';
import 'package:flutter_tools/src/convert.dart';
import 'package:flutter_tools/src/web/chrome.dart'; import 'package:flutter_tools/src/web/chrome.dart';
import 'package:flutter_tools/src/globals.dart' as globals;
import 'package:mockito/mockito.dart'; import 'package:mockito/mockito.dart';
import 'package:process/process.dart';
import 'package:platform/platform.dart'; import 'package:platform/platform.dart';
import '../../src/common.dart'; import '../../src/common.dart';
import '../../src/mocks.dart'; import '../../src/context.dart';
import '../../src/testbed.dart';
const List<String> _kChromeArgs = <String>[
'--disable-background-timer-throttling',
'--disable-extensions',
'--disable-popup-blocking',
'--bwsi',
'--no-first-run',
'--no-default-browser-check',
'--disable-default-apps',
'--disable-translate',
];
const String kDevtoolsStderr = '\n\nDevTools listening\n\n';
void main() { void main() {
Testbed testbed; ChromeLauncher chromeLauncher;
Completer<int> exitCompleter; FileSystem fileSystem;
Platform platform;
FakeProcessManager processManager;
OperatingSystemUtils operatingSystemUtils;
MockLogger logger;
setUp(() { setUp(() {
final MockPlatform platform = MockPlatform(); logger = MockLogger();
final MockOperatingSystemUtils os = MockOperatingSystemUtils(); operatingSystemUtils = MockOperatingSystemUtils();
exitCompleter = Completer<int>.sync(); when(operatingSystemUtils.findFreePort())
when(platform.isWindows).thenReturn(false);
testbed = Testbed(overrides: <Type, Generator>{
ProcessManager: () => MockProcessManager(),
Platform: () => platform,
OperatingSystemUtils: () => os,
}, setup: () {
when(os.findFreePort()).thenAnswer((Invocation invocation) async {
return 1234;
});
when(platform.environment).thenReturn(<String, String>{
kChromeEnvironment: 'example_chrome',
});
when(globals.processManager.start(any))
.thenAnswer((Invocation invocation) async { .thenAnswer((Invocation invocation) async {
return FakeProcess( return 1234;
exitCode: exitCompleter.future, });
stdout: const Stream<List<int>>.empty(), platform = FakePlatform(operatingSystem: 'macos', environment: <String, String>{
stderr: Stream<List<int>>.fromIterable(<List<int>>[ kChromeEnvironment: 'example_chrome',
utf8.encode('\n\nDevTools listening\n\n'),
]),
);
});
}); });
fileSystem = MemoryFileSystem.test();
processManager = FakeProcessManager.list(<FakeCommand>[]);
chromeLauncher = ChromeLauncher(
fileSystem: fileSystem,
platform: platform,
processManager: processManager,
operatingSystemUtils: operatingSystemUtils,
logger: logger,
);
}); });
tearDown(() { tearDown(() {
resetChromeForTesting(); resetChromeForTesting();
}); });
List<String> expectChromeArgs({int debugPort = 1234}) { test('can launch chrome and connect to the devtools', () async {
return <String>[ processManager.addCommand(const FakeCommand(
'example_chrome', command: <String>[
'--remote-debugging-port=$debugPort', 'example_chrome',
'--disable-background-timer-throttling', '--user-data-dir=/.tmp_rand0/flutter_tool.rand0',
'--disable-extensions', '--remote-debugging-port=1234',
'--disable-popup-blocking', ..._kChromeArgs,
'--bwsi', 'example_url',
'--no-first-run', ],
'--no-default-browser-check', stderr: kDevtoolsStderr,
'--disable-default-apps', ));
'--disable-translate',
await chromeLauncher.launch(
'example_url', 'example_url',
]; skipCheck: true,
} );
});
test('can launch chrome and connect to the devtools', () => testbed.run(() async {
await globals.chromeLauncher.launch('example_url', skipCheck: true);
final VerificationResult result = verify(globals.processManager.start(captureAny));
expect(result.captured.single, containsAll(expectChromeArgs()));
expect(result.captured.single, isNot(contains('--window-size=2400,1800')));
}));
test('can launch chrome with a custom debug port', () => testbed.run(() async {
await globals.chromeLauncher.launch('example_url', skipCheck: true, debugPort: 10000);
final VerificationResult result = verify(globals.processManager.start(captureAny));
expect(result.captured.single, containsAll(expectChromeArgs(debugPort: 10000)));
expect(result.captured.single, isNot(contains('--window-size=2400,1800')));
}));
test('can launch chrome headless', () => testbed.run(() async {
await globals.chromeLauncher.launch('example_url', skipCheck: true, headless: true);
final VerificationResult result = verify(globals.processManager.start(captureAny));
expect(result.captured.single, containsAll(expectChromeArgs())); test('can launch chrome with a custom debug port', () async {
expect(result.captured.single, contains('--window-size=2400,1800')); processManager.addCommand(const FakeCommand(
})); command: <String>[
'example_chrome',
'--user-data-dir=/.tmp_rand0/flutter_tool.rand0',
'--remote-debugging-port=10000',
..._kChromeArgs,
'example_url',
],
stderr: kDevtoolsStderr,
));
await chromeLauncher.launch(
'example_url',
skipCheck: true,
debugPort: 10000,
);
});
test('can launch chrome headless', () async {
processManager.addCommand(const FakeCommand(
command: <String>[
'example_chrome',
'--user-data-dir=/.tmp_rand0/flutter_tool.rand0',
'--remote-debugging-port=1234',
..._kChromeArgs,
'--headless',
'--disable-gpu',
'--no-sandbox',
'--window-size=2400,1800',
'example_url',
],
stderr: kDevtoolsStderr,
));
await chromeLauncher.launch(
'example_url',
skipCheck: true,
headless: true,
);
});
test('can seed chrome temp directory with existing preferences', () => testbed.run(() async { test('can seed chrome temp directory with existing preferences', () async {
final Directory dataDir = globals.fs.directory('chrome-stuff'); final Completer<void> exitCompleter = Completer<void>.sync();
final Directory dataDir = fileSystem.directory('chrome-stuff');
final File preferencesFile = dataDir final File preferencesFile = dataDir
.childDirectory('Default') .childDirectory('Default')
.childFile('preferences'); .childFile('preferences');
...@@ -104,12 +131,22 @@ void main() { ...@@ -104,12 +131,22 @@ void main() {
..createSync(recursive: true) ..createSync(recursive: true)
..writeAsStringSync('example'); ..writeAsStringSync('example');
await globals.chromeLauncher.launch('example_url', skipCheck: true, dataDir: dataDir); processManager.addCommand(FakeCommand(command: const <String>[
final VerificationResult result = verify(globals.processManager.start(captureAny)); 'example_chrome',
final String arg = (result.captured.single as List<String>) '--user-data-dir=/.tmp_rand0/flutter_tool.rand0',
.firstWhere((String arg) => arg.startsWith('--user-data-dir=')); '--remote-debugging-port=1234',
final Directory tempDirectory = globals.fs.directory(arg.split('=')[1]); ..._kChromeArgs,
final File tempFile = tempDirectory 'example_url',
], completer: exitCompleter));
await chromeLauncher.launch(
'example_url',
skipCheck: true,
dataDir: dataDir,
);
final File tempFile = fileSystem
.directory('.tmp_rand0/flutter_tool.rand0')
.childDirectory('Default') .childDirectory('Default')
.childFile('preferences'); .childFile('preferences');
...@@ -118,13 +155,12 @@ void main() { ...@@ -118,13 +155,12 @@ void main() {
// write crash to file: // write crash to file:
tempFile.writeAsStringSync('"exit_type":"Crashed"'); tempFile.writeAsStringSync('"exit_type":"Crashed"');
exitCompleter.complete(0); exitCompleter.complete();
// writes non-crash back to dart_tool // writes non-crash back to dart_tool
expect(preferencesFile.readAsStringSync(), '"exit_type":"Normal"'); expect(preferencesFile.readAsStringSync(), '"exit_type":"Normal"');
})); });
} }
class MockProcessManager extends Mock implements ProcessManager {}
class MockPlatform extends Mock implements Platform {}
class MockOperatingSystemUtils extends Mock implements OperatingSystemUtils {} class MockOperatingSystemUtils extends Mock implements OperatingSystemUtils {}
class MockLogger extends Mock implements Logger {}
...@@ -2,6 +2,7 @@ ...@@ -2,6 +2,7 @@
// Use of this source code is governed by a BSD-style license that can be // Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file. // found in the LICENSE file.
import 'dart:async';
import 'dart:convert'; import 'dart:convert';
import 'dart:io' as io show ProcessSignal; import 'dart:io' as io show ProcessSignal;
...@@ -26,6 +27,7 @@ class FakeCommand { ...@@ -26,6 +27,7 @@ class FakeCommand {
this.exitCode = 0, this.exitCode = 0,
this.stdout = '', this.stdout = '',
this.stderr = '', this.stderr = '',
this.completer,
}) : assert(command != null), }) : assert(command != null),
assert(duration != null), assert(duration != null),
assert(exitCode != null), assert(exitCode != null),
...@@ -78,6 +80,10 @@ class FakeCommand { ...@@ -78,6 +80,10 @@ class FakeCommand {
/// returned in one go. /// returned in one go.
final String stderr; final String stderr;
/// If provided, allows the command completion to be blocked until the future
/// resolves.
final Completer<void> completer;
static bool _listEquals<T>(List<T> a, List<T> b) { static bool _listEquals<T>(List<T> a, List<T> b) {
if (a == null) { if (a == null) {
return b == null; return b == null;
...@@ -123,10 +129,14 @@ class _FakeProcess implements Process { ...@@ -123,10 +129,14 @@ class _FakeProcess implements Process {
this._stderr, this._stderr,
this.stdin, this.stdin,
this._stdout, this._stdout,
Completer<void> completer,
) : exitCode = Future<void>.delayed(duration).then((void value) { ) : exitCode = Future<void>.delayed(duration).then((void value) {
if (onRun != null) { if (onRun != null) {
onRun(); onRun();
} }
if (completer != null) {
return completer.future.then((void _) => _exitCode);
}
return _exitCode; return _exitCode;
}), }),
stderr = Stream<List<int>>.value(utf8.encode(_stderr)), stderr = Stream<List<int>>.value(utf8.encode(_stderr)),
...@@ -184,6 +194,14 @@ abstract class FakeProcessManager implements ProcessManager { ...@@ -184,6 +194,14 @@ abstract class FakeProcessManager implements ProcessManager {
FakeProcessManager._(); FakeProcessManager._();
/// Adds a new [FakeCommand] to the current process manager.
///
/// This can be used to configure test expectations after the [ProcessManager] has been
/// provided to another interface.
///
/// This is a no-op on [FakeProcessManager.any].
void addCommand(FakeCommand command);
@protected @protected
FakeCommand findCommand(List<String> command, String workingDirectory, Map<String, String> environment); FakeCommand findCommand(List<String> command, String workingDirectory, Map<String, String> environment);
...@@ -200,6 +218,7 @@ abstract class FakeProcessManager implements ProcessManager { ...@@ -200,6 +218,7 @@ abstract class FakeProcessManager implements ProcessManager {
fakeCommand.stderr, fakeCommand.stderr,
null, // stdin null, // stdin
fakeCommand.stdout, fakeCommand.stdout,
fakeCommand.completer,
); );
} }
...@@ -277,6 +296,9 @@ class _FakeAnyProcessManager extends FakeProcessManager { ...@@ -277,6 +296,9 @@ class _FakeAnyProcessManager extends FakeProcessManager {
stderr: '', stderr: '',
); );
} }
@override
void addCommand(FakeCommand command) { }
} }
class _SequenceProcessManager extends FakeProcessManager { class _SequenceProcessManager extends FakeProcessManager {
...@@ -298,4 +320,9 @@ class _SequenceProcessManager extends FakeProcessManager { ...@@ -298,4 +320,9 @@ class _SequenceProcessManager extends FakeProcessManager {
); );
return _commands.removeAt(0); return _commands.removeAt(0);
} }
@override
void addCommand(FakeCommand command) {
_commands.add(command);
}
} }
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment