| // Copyright 2015 The Chromium 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:convert'; |
| import 'dart:io' as io show IOSink, ProcessSignal, Stdout, StdoutException; |
| |
| import 'package:flutter_tools/src/android/android_device.dart'; |
| import 'package:flutter_tools/src/android/android_sdk.dart' show AndroidSdk; |
| import 'package:flutter_tools/src/application_package.dart'; |
| import 'package:flutter_tools/src/base/file_system.dart' hide IOSink; |
| import 'package:flutter_tools/src/base/io.dart'; |
| import 'package:flutter_tools/src/base/platform.dart'; |
| import 'package:flutter_tools/src/build_info.dart'; |
| import 'package:flutter_tools/src/compile.dart'; |
| import 'package:flutter_tools/src/devfs.dart'; |
| import 'package:flutter_tools/src/device.dart'; |
| import 'package:flutter_tools/src/ios/devices.dart'; |
| import 'package:flutter_tools/src/ios/simulators.dart'; |
| import 'package:flutter_tools/src/project.dart'; |
| import 'package:flutter_tools/src/runner/flutter_command.dart'; |
| import 'package:mockito/mockito.dart'; |
| import 'package:process/process.dart'; |
| |
| import 'common.dart'; |
| |
| class MockApplicationPackageStore extends ApplicationPackageStore { |
| MockApplicationPackageStore() : super( |
| android: AndroidApk( |
| id: 'io.flutter.android.mock', |
| file: fs.file('/mock/path/to/android/SkyShell.apk'), |
| versionCode: 1, |
| launchActivity: 'io.flutter.android.mock.MockActivity', |
| ), |
| iOS: BuildableIOSApp(MockIosProject()) |
| ); |
| } |
| |
| class MockApplicationPackageFactory extends Mock implements ApplicationPackageFactory { |
| final MockApplicationPackageStore _store = MockApplicationPackageStore(); |
| |
| @override |
| Future<ApplicationPackage> getPackageForPlatform( |
| TargetPlatform platform, { |
| File applicationBinary, |
| }) async { |
| return _store.getPackageForPlatform(platform); |
| } |
| } |
| |
| /// An SDK installation with several SDK levels (19, 22, 23). |
| class MockAndroidSdk extends Mock implements AndroidSdk { |
| static Directory createSdkDirectory({ |
| bool withAndroidN = false, |
| String withNdkDir, |
| int ndkVersion = 16, |
| bool withNdkSysroot = false, |
| bool withSdkManager = true, |
| bool withPlatformTools = true, |
| bool withBuildTools = true, |
| }) { |
| final Directory dir = fs.systemTempDirectory.createTempSync('flutter_mock_android_sdk.'); |
| final String exe = platform.isWindows ? '.exe' : ''; |
| final String bat = platform.isWindows ? '.bat' : ''; |
| |
| _createDir(dir, 'licenses'); |
| |
| if (withPlatformTools) { |
| _createSdkFile(dir, 'platform-tools/adb$exe'); |
| } |
| |
| if (withBuildTools) { |
| _createSdkFile(dir, 'build-tools/19.1.0/aapt$exe'); |
| _createSdkFile(dir, 'build-tools/22.0.1/aapt$exe'); |
| _createSdkFile(dir, 'build-tools/23.0.2/aapt$exe'); |
| if (withAndroidN) |
| _createSdkFile(dir, 'build-tools/24.0.0-preview/aapt$exe'); |
| } |
| |
| _createSdkFile(dir, 'platforms/android-22/android.jar'); |
| _createSdkFile(dir, 'platforms/android-23/android.jar'); |
| if (withAndroidN) { |
| _createSdkFile(dir, 'platforms/android-N/android.jar'); |
| _createSdkFile(dir, 'platforms/android-N/build.prop', contents: _buildProp); |
| } |
| |
| if (withSdkManager) |
| _createSdkFile(dir, 'tools/bin/sdkmanager$bat'); |
| |
| if (withNdkDir != null) { |
| final String ndkToolchainBin = fs.path.join( |
| 'ndk-bundle', |
| 'toolchains', |
| 'arm-linux-androideabi-4.9', |
| 'prebuilt', |
| withNdkDir, |
| 'bin', |
| ); |
| final String ndkCompiler = fs.path.join( |
| ndkToolchainBin, |
| 'arm-linux-androideabi-gcc', |
| ); |
| final String ndkLinker = fs.path.join( |
| ndkToolchainBin, |
| 'arm-linux-androideabi-ld', |
| ); |
| _createSdkFile(dir, ndkCompiler); |
| _createSdkFile(dir, ndkLinker); |
| _createSdkFile(dir, fs.path.join('ndk-bundle', 'source.properties'), contents: ''' |
| Pkg.Desc = Android NDK[] |
| Pkg.Revision = $ndkVersion.1.5063045 |
| |
| '''); |
| } |
| if (withNdkSysroot) { |
| final String armPlatform = fs.path.join( |
| 'ndk-bundle', |
| 'platforms', |
| 'android-9', |
| 'arch-arm', |
| ); |
| _createDir(dir, armPlatform); |
| } |
| |
| return dir; |
| } |
| |
| static void _createSdkFile(Directory dir, String filePath, { String contents }) { |
| final File file = dir.childFile(filePath); |
| file.createSync(recursive: true); |
| if (contents != null) { |
| file.writeAsStringSync(contents, flush: true); |
| } |
| } |
| |
| static void _createDir(Directory dir, String path) { |
| final Directory directory = fs.directory(fs.path.join(dir.path, path)); |
| directory.createSync(recursive: true); |
| } |
| |
| static const String _buildProp = r''' |
| ro.build.version.incremental=1624448 |
| ro.build.version.sdk=24 |
| ro.build.version.codename=REL |
| '''; |
| } |
| |
| /// A strategy for creating Process objects from a list of commands. |
| typedef ProcessFactory = Process Function(List<String> command); |
| |
| /// A ProcessManager that starts Processes by delegating to a ProcessFactory. |
| class MockProcessManager implements ProcessManager { |
| ProcessFactory processFactory = (List<String> commands) => MockProcess(); |
| bool succeed = true; |
| List<String> commands; |
| |
| @override |
| bool canRun(dynamic command, { String workingDirectory }) => succeed; |
| |
| @override |
| Future<Process> start( |
| List<dynamic> command, { |
| String workingDirectory, |
| Map<String, String> environment, |
| bool includeParentEnvironment = true, |
| bool runInShell = false, |
| ProcessStartMode mode = ProcessStartMode.normal, |
| }) { |
| if (!succeed) { |
| final String executable = command[0]; |
| final List<String> arguments = command.length > 1 ? command.sublist(1) : <String>[]; |
| throw ProcessException(executable, arguments); |
| } |
| |
| commands = command; |
| return Future<Process>.value(processFactory(command)); |
| } |
| |
| @override |
| dynamic noSuchMethod(Invocation invocation) => null; |
| } |
| |
| /// A process that exits successfully with no output and ignores all input. |
| class MockProcess extends Mock implements Process { |
| MockProcess({ |
| this.pid = 1, |
| Future<int> exitCode, |
| Stream<List<int>> stdin, |
| this.stdout = const Stream<List<int>>.empty(), |
| this.stderr = const Stream<List<int>>.empty(), |
| }) : exitCode = exitCode ?? Future<int>.value(0), |
| stdin = stdin ?? MemoryIOSink(); |
| |
| @override |
| final int pid; |
| |
| @override |
| final Future<int> exitCode; |
| |
| @override |
| final io.IOSink stdin; |
| |
| @override |
| final Stream<List<int>> stdout; |
| |
| @override |
| final Stream<List<int>> stderr; |
| } |
| |
| /// A fake process implemenation which can be provided all necessary values. |
| class FakeProcess implements Process { |
| FakeProcess({ |
| this.pid = 1, |
| Future<int> exitCode, |
| Stream<List<int>> stdin, |
| this.stdout = const Stream<List<int>>.empty(), |
| this.stderr = const Stream<List<int>>.empty(), |
| }) : exitCode = exitCode ?? Future<int>.value(0), |
| stdin = stdin ?? MemoryIOSink(); |
| |
| @override |
| final int pid; |
| |
| @override |
| final Future<int> exitCode; |
| |
| @override |
| final io.IOSink stdin; |
| |
| @override |
| final Stream<List<int>> stdout; |
| |
| @override |
| final Stream<List<int>> stderr; |
| |
| @override |
| bool kill([io.ProcessSignal signal = io.ProcessSignal.sigterm]) { |
| return true; |
| } |
| } |
| |
| /// A process that prompts the user to proceed, then asynchronously writes |
| /// some lines to stdout before it exits. |
| class PromptingProcess implements Process { |
| Future<void> showPrompt(String prompt, List<String> outputLines) async { |
| _stdoutController.add(utf8.encode(prompt)); |
| final List<int> bytesOnStdin = await _stdin.future; |
| // Echo stdin to stdout. |
| _stdoutController.add(bytesOnStdin); |
| if (bytesOnStdin[0] == utf8.encode('y')[0]) { |
| for (final String line in outputLines) |
| _stdoutController.add(utf8.encode('$line\n')); |
| } |
| await _stdoutController.close(); |
| } |
| |
| final StreamController<List<int>> _stdoutController = StreamController<List<int>>(); |
| final CompleterIOSink _stdin = CompleterIOSink(); |
| |
| @override |
| Stream<List<int>> get stdout => _stdoutController.stream; |
| |
| @override |
| Stream<List<int>> get stderr => const Stream<List<int>>.empty(); |
| |
| @override |
| IOSink get stdin => _stdin; |
| |
| @override |
| Future<int> get exitCode async { |
| await _stdoutController.done; |
| return 0; |
| } |
| |
| @override |
| dynamic noSuchMethod(Invocation invocation) => null; |
| } |
| |
| /// An IOSink that completes a future with the first line written to it. |
| class CompleterIOSink extends MemoryIOSink { |
| final Completer<List<int>> _completer = Completer<List<int>>(); |
| |
| Future<List<int>> get future => _completer.future; |
| |
| @override |
| void add(List<int> data) { |
| if (!_completer.isCompleted) |
| _completer.complete(data); |
| super.add(data); |
| } |
| } |
| |
| /// An IOSink that collects whatever is written to it. |
| class MemoryIOSink implements IOSink { |
| @override |
| Encoding encoding = utf8; |
| |
| final List<List<int>> writes = <List<int>>[]; |
| |
| @override |
| void add(List<int> data) { |
| writes.add(data); |
| } |
| |
| @override |
| Future<void> addStream(Stream<List<int>> stream) { |
| final Completer<void> completer = Completer<void>(); |
| stream.listen((List<int> data) { |
| add(data); |
| }).onDone(() => completer.complete()); |
| return completer.future; |
| } |
| |
| @override |
| void writeCharCode(int charCode) { |
| add(<int>[charCode]); |
| } |
| |
| @override |
| void write(Object obj) { |
| add(encoding.encode('$obj')); |
| } |
| |
| @override |
| void writeln([ Object obj = '' ]) { |
| add(encoding.encode('$obj\n')); |
| } |
| |
| @override |
| void writeAll(Iterable<dynamic> objects, [ String separator = '' ]) { |
| bool addSeparator = false; |
| for (dynamic object in objects) { |
| if (addSeparator) { |
| write(separator); |
| } |
| write(object); |
| addSeparator = true; |
| } |
| } |
| |
| @override |
| void addError(dynamic error, [ StackTrace stackTrace ]) { |
| throw UnimplementedError(); |
| } |
| |
| @override |
| Future<void> get done => close(); |
| |
| @override |
| Future<void> close() async { } |
| |
| @override |
| Future<void> flush() async { } |
| } |
| |
| class MemoryStdout extends MemoryIOSink implements io.Stdout { |
| @override |
| bool get hasTerminal => _hasTerminal; |
| set hasTerminal(bool value) { |
| assert(value != null); |
| _hasTerminal = value; |
| } |
| bool _hasTerminal = true; |
| |
| @override |
| io.IOSink get nonBlocking => this; |
| |
| @override |
| bool get supportsAnsiEscapes => _supportsAnsiEscapes; |
| set supportsAnsiEscapes(bool value) { |
| assert(value != null); |
| _supportsAnsiEscapes = value; |
| } |
| bool _supportsAnsiEscapes = true; |
| |
| @override |
| int get terminalColumns { |
| if (_terminalColumns != null) |
| return _terminalColumns; |
| throw const io.StdoutException('unspecified mock value'); |
| } |
| set terminalColumns(int value) => _terminalColumns = value; |
| int _terminalColumns; |
| |
| @override |
| int get terminalLines { |
| if (_terminalLines != null) |
| return _terminalLines; |
| throw const io.StdoutException('unspecified mock value'); |
| } |
| set terminalLines(int value) => _terminalLines = value; |
| int _terminalLines; |
| } |
| |
| /// A Stdio that collects stdout and supports simulated stdin. |
| class MockStdio extends Stdio { |
| final MemoryStdout _stdout = MemoryStdout(); |
| final MemoryIOSink _stderr = MemoryIOSink(); |
| final StreamController<List<int>> _stdin = StreamController<List<int>>(); |
| |
| @override |
| MemoryStdout get stdout => _stdout; |
| |
| @override |
| MemoryIOSink get stderr => _stderr; |
| |
| @override |
| Stream<List<int>> get stdin => _stdin.stream; |
| |
| void simulateStdin(String line) { |
| _stdin.add(utf8.encode('$line\n')); |
| } |
| |
| List<String> get writtenToStdout => _stdout.writes.map<String>(_stdout.encoding.decode).toList(); |
| List<String> get writtenToStderr => _stderr.writes.map<String>(_stderr.encoding.decode).toList(); |
| } |
| |
| class MockPollingDeviceDiscovery extends PollingDeviceDiscovery { |
| MockPollingDeviceDiscovery() : super('mock'); |
| |
| final List<Device> _devices = <Device>[]; |
| final StreamController<Device> _onAddedController = StreamController<Device>.broadcast(); |
| final StreamController<Device> _onRemovedController = StreamController<Device>.broadcast(); |
| |
| @override |
| Future<List<Device>> pollingGetDevices() async => _devices; |
| |
| @override |
| bool get supportsPlatform => true; |
| |
| @override |
| bool get canListAnything => true; |
| |
| void addDevice(MockAndroidDevice device) { |
| _devices.add(device); |
| |
| _onAddedController.add(device); |
| } |
| |
| @override |
| Future<List<Device>> get devices async => _devices; |
| |
| @override |
| Stream<Device> get onAdded => _onAddedController.stream; |
| |
| @override |
| Stream<Device> get onRemoved => _onRemovedController.stream; |
| } |
| |
| class MockIosProject extends Mock implements IosProject { |
| @override |
| String get productBundleIdentifier => 'com.example.test'; |
| |
| @override |
| String get hostAppBundleName => 'Runner.app'; |
| } |
| |
| class MockAndroidDevice extends Mock implements AndroidDevice { |
| @override |
| Future<TargetPlatform> get targetPlatform async => TargetPlatform.android_arm; |
| |
| @override |
| bool isSupported() => true; |
| |
| @override |
| bool isSupportedForProject(FlutterProject flutterProject) => true; |
| } |
| |
| class MockIOSDevice extends Mock implements IOSDevice { |
| @override |
| Future<TargetPlatform> get targetPlatform async => TargetPlatform.ios; |
| |
| @override |
| bool isSupported() => true; |
| |
| @override |
| bool isSupportedForProject(FlutterProject flutterProject) => true; |
| } |
| |
| class MockIOSSimulator extends Mock implements IOSSimulator { |
| @override |
| Future<TargetPlatform> get targetPlatform async => TargetPlatform.ios; |
| |
| @override |
| bool isSupported() => true; |
| |
| @override |
| bool isSupportedForProject(FlutterProject flutterProject) => true; |
| } |
| |
| class MockDeviceLogReader extends DeviceLogReader { |
| @override |
| String get name => 'MockLogReader'; |
| |
| final StreamController<String> _linesController = StreamController<String>.broadcast(); |
| |
| @override |
| Stream<String> get logLines => _linesController.stream; |
| |
| void addLine(String line) => _linesController.add(line); |
| |
| void dispose() { |
| _linesController.close(); |
| } |
| } |
| |
| void applyMocksToCommand(FlutterCommand command) { |
| command |
| ..applicationPackages = MockApplicationPackageStore(); |
| } |
| |
| /// Common functionality for tracking mock interaction |
| class BasicMock { |
| final List<String> messages = <String>[]; |
| |
| void expectMessages(List<String> expectedMessages) { |
| final List<String> actualMessages = List<String>.from(messages); |
| messages.clear(); |
| expect(actualMessages, unorderedEquals(expectedMessages)); |
| } |
| |
| bool contains(String match) { |
| print('Checking for `$match` in:'); |
| print(messages); |
| final bool result = messages.contains(match); |
| messages.clear(); |
| return result; |
| } |
| } |
| |
| class MockDevFSOperations extends BasicMock implements DevFSOperations { |
| Map<Uri, DevFSContent> devicePathToContent = <Uri, DevFSContent>{}; |
| |
| @override |
| Future<Uri> create(String fsName) async { |
| messages.add('create $fsName'); |
| return Uri.parse('file:///$fsName'); |
| } |
| |
| @override |
| Future<dynamic> destroy(String fsName) async { |
| messages.add('destroy $fsName'); |
| } |
| |
| @override |
| Future<dynamic> writeFile(String fsName, Uri deviceUri, DevFSContent content) async { |
| String message = 'writeFile $fsName $deviceUri'; |
| if (content is DevFSFileContent) { |
| message += ' ${content.file.path}'; |
| } |
| messages.add(message); |
| devicePathToContent[deviceUri] = content; |
| } |
| } |
| |
| class MockResidentCompiler extends BasicMock implements ResidentCompiler { |
| @override |
| void accept() { } |
| |
| @override |
| Future<CompilerOutput> reject() async { return null; } |
| |
| @override |
| void reset() { } |
| |
| @override |
| Future<dynamic> shutdown() async { } |
| |
| @override |
| Future<CompilerOutput> compileExpression( |
| String expression, |
| List<String> definitions, |
| List<String> typeDefinitions, |
| String libraryUri, |
| String klass, |
| bool isStatic, |
| ) async { |
| return null; |
| } |
| @override |
| Future<CompilerOutput> recompile(String mainPath, List<Uri> invalidatedFiles, { String outputPath, String packagesFilePath }) async { |
| fs.file(outputPath).createSync(recursive: true); |
| fs.file(outputPath).writeAsStringSync('compiled_kernel_output'); |
| return CompilerOutput(outputPath, 0, <Uri>[]); |
| } |
| } |
| |
| /// A fake implementation of [ProcessResult]. |
| class FakeProcessResult implements ProcessResult { |
| FakeProcessResult({ |
| this.exitCode = 0, |
| this.pid = 1, |
| this.stderr, |
| this.stdout, |
| }); |
| |
| @override |
| final int exitCode; |
| |
| @override |
| final int pid; |
| |
| @override |
| final dynamic stderr; |
| |
| @override |
| final dynamic stdout; |
| |
| @override |
| String toString() => stdout?.toString() ?? stderr?.toString() ?? runtimeType.toString(); |
| } |