// 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; 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()) ); } /// 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 { } } /// A Stdio that collects stdout and supports simulated stdin. class MockStdio extends Stdio { final MemoryIOSink _stdout = MemoryIOSink(); final MemoryIOSink _stderr = MemoryIOSink(); final StreamController<List<int>> _stdin = StreamController<List<int>>(); @override IOSink get stdout => _stdout; @override IOSink 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>[]); } }