mirror of
https://github.com/flutter/flutter.git
synced 2025-06-03 00:51:18 +00:00

On `Podfile`: ```ruby flutter_application_path = '../flutter_module' load File.join(flutter_application_path, '.ios', 'Flutter', 'podhelper.rb') target 'OCProject' do # Comment the next line if you don't want to use dynamic frameworks use_frameworks! # Pods for OCProject # install_all_flutter_pods(flutter_application_path) # install_flutter_engine_pod(flutter_application_path) # install_flutter_application_pod(flutter_application_path) install_flutter_plugin_pods(flutter_application_path) end post_install do |installer| flutter_post_install(installer) end ``` Encountering the following error after executing `pod install`: ```shell pod install [!] Invalid `Podfile` file: undefined method `flutter_relative_path_from_podfile' for #<Pod::Podfile:0x000000010e74c520 @defined_in_file=#<Pathname:/Users/lxf/gitHub/flutter_hybrid_bug/OCProject/Podfile>, @internal_hash={}, @root_target_definitions=[#<Pod::Podfile::TargetDefinition label=Pods>], @current_target_definition=#<Pod::Podfile::TargetDefinition label=Pods>> relative = flutter_relative_path_from_podfile(export_script_directory) ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^. # from /Users/lxf/gitHub/flutter_hybrid_bug/OCProject/Podfile:17 # ------------------------------------------- # # install_flutter_plugin_pods(flutter_application_path) > install_flutter_application_pod(flutter_application_path) # # ------------------------------------------- ``` The `flutter_relative_path_from_podfile` method is in `flutter_tools/bin/podhelper.rb`, but now `flutter_tools/bin/podhelper.rb` is only required in `install_all_flutter_pods` in `podhelper.rb.tmpl`. Sometimes we only need to use the `install_flutter_plugin_pods` method in podhelper.rb. For example, using `Shorebird` in an iOS hybird app scenario, we need to build `Flutter.xcframework` and `App.xcframework` and embed them into the iOS native project. In order to avoid unnecessary conflicts, use `install_flutter_plugin_pods` method to install Flutter plugin pods. [Shorebird - Code Push In Hybrid Apps](https://docs.shorebird.dev/guides/hybrid-app/ios) So I adjust the position of `require File.expand_path(File.join('packages', 'flutter_tools', 'bin', 'podhelper'), flutter_root)`.
763 lines
26 KiB
Dart
763 lines
26 KiB
Dart
// 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:convert';
|
|
import 'dart:io';
|
|
import 'dart:typed_data';
|
|
|
|
import 'package:flutter_devicelab/framework/framework.dart';
|
|
import 'package:flutter_devicelab/framework/host_agent.dart';
|
|
import 'package:flutter_devicelab/framework/ios.dart';
|
|
import 'package:flutter_devicelab/framework/task_result.dart';
|
|
import 'package:flutter_devicelab/framework/utils.dart';
|
|
import 'package:path/path.dart' as path;
|
|
|
|
/// Tests that the Flutter module project template works and supports
|
|
/// adding Flutter to an existing iOS app.
|
|
Future<void> main() async {
|
|
await task(() async {
|
|
// Update pod repo.
|
|
await eval(
|
|
'pod',
|
|
<String>['repo', 'update'],
|
|
environment: <String, String>{
|
|
'LANG': 'en_US.UTF-8',
|
|
},
|
|
);
|
|
|
|
// This variable cannot be `late`, as we reference it in the `finally` block
|
|
// which may execute before this field has been initialized.
|
|
String? simulatorDeviceId;
|
|
section('Create Flutter module project');
|
|
|
|
final Directory tempDir = Directory.systemTemp.createTempSync('flutter_module_test.');
|
|
final Directory projectDir = Directory(path.join(tempDir.path, 'hello'));
|
|
try {
|
|
await inDirectory(tempDir, () async {
|
|
await flutter(
|
|
'create',
|
|
options: <String>[
|
|
'--org',
|
|
'io.flutter.devicelab',
|
|
'--template=module',
|
|
'hello',
|
|
],
|
|
);
|
|
});
|
|
|
|
// Copy test dart files to new module app.
|
|
final Directory flutterModuleLibSource = Directory(path.join(flutterDirectory.path, 'dev', 'integration_tests', 'ios_host_app', 'flutterapp', 'lib'));
|
|
final Directory flutterModuleLibDestination = Directory(path.join(projectDir.path, 'lib'));
|
|
|
|
// These test files don't have a .dart extension so the analyzer will ignore them. They aren't in a
|
|
// package and don't work on their own outside of the test module just created.
|
|
final File main = File(path.join(flutterModuleLibSource.path, 'main'));
|
|
main.copySync(path.join(flutterModuleLibDestination.path, 'main.dart'));
|
|
|
|
final File marquee = File(path.join(flutterModuleLibSource.path, 'marquee'));
|
|
marquee.copySync(path.join(flutterModuleLibDestination.path, 'marquee.dart'));
|
|
|
|
section('Create package with native assets');
|
|
|
|
await flutter(
|
|
'config',
|
|
options: <String>['--enable-native-assets'],
|
|
);
|
|
|
|
const String ffiPackageName = 'ffi_package';
|
|
await createFfiPackage(ffiPackageName, tempDir);
|
|
|
|
section('Add FFI package');
|
|
|
|
final File pubspec = File(path.join(projectDir.path, 'pubspec.yaml'));
|
|
String content = await pubspec.readAsString();
|
|
content = content.replaceFirst(
|
|
'dependencies:\n',
|
|
'''
|
|
dependencies:
|
|
$ffiPackageName:
|
|
path: ../$ffiPackageName
|
|
''',
|
|
);
|
|
await pubspec.writeAsString(content, flush: true);
|
|
await inDirectory(projectDir, () async {
|
|
await flutter(
|
|
'packages',
|
|
options: <String>['get'],
|
|
);
|
|
});
|
|
|
|
section('Build ephemeral host app in release mode without CocoaPods');
|
|
|
|
await inDirectory(projectDir, () async {
|
|
await flutter(
|
|
'build',
|
|
options: <String>['ios', '--no-codesign', '--verbose'],
|
|
);
|
|
});
|
|
|
|
// Check the tool is no longer copying to the legacy xcframework location.
|
|
checkDirectoryNotExists(path.join(projectDir.path, '.ios', 'Flutter', 'engine', 'Flutter.xcframework'));
|
|
|
|
final Directory ephemeralIOSHostApp = Directory(path.join(
|
|
projectDir.path,
|
|
'build',
|
|
'ios',
|
|
'iphoneos',
|
|
'Runner.app',
|
|
));
|
|
|
|
if (!exists(ephemeralIOSHostApp)) {
|
|
return TaskResult.failure('Failed to build ephemeral host .app');
|
|
}
|
|
|
|
if (!await _isAppAotBuild(ephemeralIOSHostApp)) {
|
|
return TaskResult.failure(
|
|
'Ephemeral host app ${ephemeralIOSHostApp.path} was not a release build as expected'
|
|
);
|
|
}
|
|
|
|
section('Build ephemeral host app in profile mode without CocoaPods');
|
|
|
|
await inDirectory(projectDir, () async {
|
|
await flutter(
|
|
'build',
|
|
options: <String>['ios', '--no-codesign', '--profile'],
|
|
);
|
|
});
|
|
|
|
if (!exists(ephemeralIOSHostApp)) {
|
|
return TaskResult.failure('Failed to build ephemeral host .app');
|
|
}
|
|
|
|
if (!await _isAppAotBuild(ephemeralIOSHostApp)) {
|
|
return TaskResult.failure(
|
|
'Ephemeral host app ${ephemeralIOSHostApp.path} was not a profile build as expected'
|
|
);
|
|
}
|
|
|
|
section('Clean build');
|
|
|
|
await inDirectory(projectDir, () async {
|
|
await flutter('clean');
|
|
});
|
|
|
|
section('Build ephemeral host app in debug mode for simulator without CocoaPods');
|
|
|
|
await inDirectory(projectDir, () async {
|
|
await flutter(
|
|
'build',
|
|
options: <String>['ios', '--no-codesign', '--simulator', '--debug'],
|
|
);
|
|
});
|
|
|
|
final Directory ephemeralSimulatorHostApp = Directory(path.join(
|
|
projectDir.path,
|
|
'build',
|
|
'ios',
|
|
'iphonesimulator',
|
|
'Runner.app',
|
|
));
|
|
|
|
if (!exists(ephemeralSimulatorHostApp)) {
|
|
return TaskResult.failure('Failed to build ephemeral host .app');
|
|
}
|
|
checkFileExists(path.join(ephemeralSimulatorHostApp.path, 'Frameworks', 'Flutter.framework', 'Flutter'));
|
|
|
|
if (!exists(File(path.join(
|
|
ephemeralSimulatorHostApp.path,
|
|
'Frameworks',
|
|
'App.framework',
|
|
'flutter_assets',
|
|
'isolate_snapshot_data',
|
|
)))) {
|
|
return TaskResult.failure(
|
|
'Ephemeral host app ${ephemeralSimulatorHostApp.path} was not a debug build as expected'
|
|
);
|
|
}
|
|
|
|
section('Clean build');
|
|
|
|
await inDirectory(projectDir, () async {
|
|
await flutter('clean');
|
|
});
|
|
|
|
// Make a fake Dart-only plugin, since there are no existing examples.
|
|
section('Create local plugin');
|
|
|
|
const String dartPluginName = 'dartplugin';
|
|
await _createFakeDartPlugin(dartPluginName, tempDir);
|
|
|
|
section('Add plugins');
|
|
|
|
content = content.replaceFirst(
|
|
'dependencies:\n',
|
|
// One framework, one Dart-only, one that does not support iOS, and one with a resource bundle.
|
|
'''
|
|
dependencies:
|
|
url_launcher: 6.0.20
|
|
android_alarm_manager: 2.0.2
|
|
google_sign_in_ios: 5.5.0
|
|
$dartPluginName:
|
|
path: ../$dartPluginName
|
|
''',
|
|
);
|
|
await pubspec.writeAsString(content, flush: true);
|
|
await inDirectory(projectDir, () async {
|
|
await flutter(
|
|
'packages',
|
|
options: <String>['get'],
|
|
);
|
|
});
|
|
|
|
section('Build ephemeral host app with CocoaPods');
|
|
|
|
await inDirectory(projectDir, () async {
|
|
await flutter(
|
|
'build',
|
|
options: <String>['ios', '--no-codesign', '-v'],
|
|
);
|
|
});
|
|
|
|
final bool ephemeralHostAppWithCocoaPodsBuilt = exists(ephemeralIOSHostApp);
|
|
|
|
if (!ephemeralHostAppWithCocoaPodsBuilt) {
|
|
return TaskResult.failure('Failed to build ephemeral host .app with CocoaPods');
|
|
}
|
|
|
|
final File podfileLockFile = File(path.join(projectDir.path, '.ios', 'Podfile.lock'));
|
|
final String podfileLockOutput = podfileLockFile.readAsStringSync();
|
|
if (!podfileLockOutput.contains(':path: Flutter')
|
|
|| !podfileLockOutput.contains(':path: Flutter/FlutterPluginRegistrant')
|
|
|| !podfileLockOutput.contains(':path: ".symlinks/plugins/url_launcher_ios/ios"')
|
|
|| podfileLockOutput.contains('android_alarm_manager')
|
|
|| podfileLockOutput.contains(dartPluginName)) {
|
|
print(podfileLockOutput);
|
|
return TaskResult.failure('Building ephemeral host app Podfile.lock does not contain expected pods');
|
|
}
|
|
|
|
checkFileExists(path.join(ephemeralIOSHostApp.path, 'Frameworks', 'url_launcher_ios.framework', 'url_launcher_ios'));
|
|
// Resources should be embedded.
|
|
checkDirectoryExists(path.join(ephemeralIOSHostApp.path, 'Frameworks', 'GoogleSignIn.framework', 'GoogleSignIn.bundle'));
|
|
checkFileExists(path.join(ephemeralIOSHostApp.path, 'Frameworks', 'Flutter.framework', 'Flutter'));
|
|
|
|
// Android-only, no embedded framework.
|
|
checkDirectoryNotExists(path.join(ephemeralIOSHostApp.path, 'Frameworks', 'android_alarm_manager.framework'));
|
|
|
|
// Dart-only, no embedded framework.
|
|
checkDirectoryNotExists(path.join(ephemeralIOSHostApp.path, 'Frameworks', '$dartPluginName.framework'));
|
|
|
|
// Native assets embedded, no embedded framework.
|
|
checkFileExists(path.join(ephemeralIOSHostApp.path, 'Frameworks', '$ffiPackageName.framework', ffiPackageName));
|
|
|
|
section('Clean and pub get module');
|
|
|
|
await inDirectory(projectDir, () async {
|
|
await flutter('clean');
|
|
});
|
|
|
|
await inDirectory(projectDir, () async {
|
|
await flutter('pub', options: <String>['get']);
|
|
});
|
|
|
|
section('Add to existing iOS Objective-C app');
|
|
|
|
final Directory objectiveCHostApp = Directory(path.join(tempDir.path, 'hello_host_app'));
|
|
mkdir(objectiveCHostApp);
|
|
recursiveCopy(
|
|
Directory(path.join(flutterDirectory.path, 'dev', 'integration_tests', 'ios_host_app')),
|
|
objectiveCHostApp,
|
|
);
|
|
|
|
final File objectiveCAnalyticsOutputFile = File(path.join(tempDir.path, 'analytics-objc.log'));
|
|
final Directory objectiveCBuildDirectory = Directory(path.join(tempDir.path, 'build-objc'));
|
|
|
|
await inDirectory(objectiveCHostApp, () async {
|
|
section('Validate iOS Objective-C host app Podfile');
|
|
|
|
final File podfile = File(path.join(objectiveCHostApp.path, 'Podfile'));
|
|
String podfileContent = await podfile.readAsString();
|
|
final String podFailure = await eval(
|
|
'pod',
|
|
<String>['install'],
|
|
environment: <String, String>{
|
|
'LANG': 'en_US.UTF-8',
|
|
},
|
|
canFail: true,
|
|
);
|
|
|
|
if (!podFailure.contains('Missing `flutter_post_install(installer)` in Podfile `post_install` block')
|
|
|| !podFailure.contains('Add `flutter_post_install(installer)` to your Podfile `post_install` block to build Flutter plugins')) {
|
|
print(podfileContent);
|
|
throw TaskResult.failure('pod install unexpectedly succeed without "flutter_post_install" post_install block');
|
|
}
|
|
podfileContent = '''
|
|
$podfileContent
|
|
|
|
post_install do |installer|
|
|
flutter_post_install(installer)
|
|
end
|
|
''';
|
|
await podfile.writeAsString(podfileContent, flush: true);
|
|
|
|
await exec(
|
|
'pod',
|
|
<String>['install'],
|
|
environment: <String, String>{
|
|
'LANG': 'en_US.UTF-8',
|
|
},
|
|
);
|
|
|
|
File hostPodfileLockFile = File(path.join(objectiveCHostApp.path, 'Podfile.lock'));
|
|
String hostPodfileLockOutput = hostPodfileLockFile.readAsStringSync();
|
|
if (!hostPodfileLockOutput.contains(':path: "../hello/.ios/Flutter"')
|
|
|| !hostPodfileLockOutput.contains(':path: "../hello/.ios/Flutter/FlutterPluginRegistrant"')
|
|
|| !hostPodfileLockOutput.contains(':path: "../hello/.ios/.symlinks/plugins/url_launcher_ios/ios"')
|
|
|| hostPodfileLockOutput.contains('android_alarm_manager')
|
|
|| hostPodfileLockOutput.contains(dartPluginName)) {
|
|
print(hostPodfileLockOutput);
|
|
throw TaskResult.failure('Building host app Podfile.lock does not contain expected pods');
|
|
}
|
|
|
|
section('Validate install_flutter_[engine_pod|plugin_pods|application_pod] methods in the Podfile can be executed normally');
|
|
|
|
podfileContent = podfileContent.replaceFirst('''
|
|
install_all_flutter_pods flutter_application_path
|
|
''', '''
|
|
install_flutter_engine_pod(flutter_application_path)
|
|
install_flutter_plugin_pods(flutter_application_path)
|
|
install_flutter_application_pod(flutter_application_path)
|
|
''');
|
|
await podfile.writeAsString(podfileContent, flush: true);
|
|
|
|
await exec(
|
|
'pod',
|
|
<String>['install'],
|
|
environment: <String, String>{
|
|
'LANG': 'en_US.UTF-8',
|
|
},
|
|
);
|
|
|
|
hostPodfileLockFile = File(path.join(objectiveCHostApp.path, 'Podfile.lock'));
|
|
hostPodfileLockOutput = hostPodfileLockFile.readAsStringSync();
|
|
if (!hostPodfileLockOutput.contains(':path: "../hello/.ios/Flutter"')
|
|
|| !hostPodfileLockOutput.contains(':path: "../hello/.ios/Flutter/FlutterPluginRegistrant"')
|
|
|| !hostPodfileLockOutput.contains(':path: "../hello/.ios/.symlinks/plugins/url_launcher_ios/ios"')
|
|
|| hostPodfileLockOutput.contains('android_alarm_manager')
|
|
|| hostPodfileLockOutput.contains(dartPluginName)) {
|
|
print(hostPodfileLockOutput);
|
|
throw TaskResult.failure('Building host app Podfile.lock does not contain expected pods');
|
|
}
|
|
|
|
// Check the tool is no longer copying to the legacy App.framework location.
|
|
final File dummyAppFramework = File(path.join(projectDir.path, '.ios', 'Flutter', 'App.framework', 'App'));
|
|
checkFileNotExists(dummyAppFramework.path);
|
|
|
|
section('Build iOS Objective-C host app');
|
|
|
|
await exec(
|
|
'xcodebuild',
|
|
<String>[
|
|
'-workspace',
|
|
'Host.xcworkspace',
|
|
'-scheme',
|
|
'Host',
|
|
'-configuration',
|
|
'Debug',
|
|
'CODE_SIGNING_ALLOWED=NO',
|
|
'CODE_SIGNING_REQUIRED=NO',
|
|
'CODE_SIGN_IDENTITY=-',
|
|
'EXPANDED_CODE_SIGN_IDENTITY=-',
|
|
'BUILD_DIR=${objectiveCBuildDirectory.path}',
|
|
'COMPILER_INDEX_STORE_ENABLE=NO',
|
|
],
|
|
environment: <String, String> {
|
|
'FLUTTER_ANALYTICS_LOG_FILE': objectiveCAnalyticsOutputFile.path,
|
|
},
|
|
);
|
|
});
|
|
|
|
final String hostAppDirectory = path.join(
|
|
objectiveCBuildDirectory.path,
|
|
'Debug-iphoneos',
|
|
'Host.app',
|
|
);
|
|
|
|
final bool existingAppBuilt = exists(File(path.join(
|
|
hostAppDirectory,
|
|
'Host',
|
|
)));
|
|
if (!existingAppBuilt) {
|
|
return TaskResult.failure('Failed to build existing Objective-C app .app');
|
|
}
|
|
|
|
final String hostFrameworksDirectory = path.join(
|
|
hostAppDirectory,
|
|
'Frameworks',
|
|
);
|
|
|
|
checkFileExists(path.join(
|
|
hostFrameworksDirectory,
|
|
'Flutter.framework',
|
|
'Flutter',
|
|
));
|
|
|
|
checkFileExists(path.join(
|
|
hostFrameworksDirectory,
|
|
'App.framework',
|
|
'flutter_assets',
|
|
'isolate_snapshot_data',
|
|
));
|
|
|
|
checkFileExists(path.join(
|
|
hostFrameworksDirectory,
|
|
'$ffiPackageName.framework',
|
|
ffiPackageName,
|
|
));
|
|
|
|
section('Check the NOTICE file is correct');
|
|
|
|
final String licenseFilePath = path.join(
|
|
hostFrameworksDirectory,
|
|
'App.framework',
|
|
'flutter_assets',
|
|
'NOTICES.Z',
|
|
);
|
|
checkFileExists(licenseFilePath);
|
|
|
|
await inDirectory(objectiveCBuildDirectory, () async {
|
|
final Uint8List licenseData = File(licenseFilePath).readAsBytesSync();
|
|
final String licenseString = utf8.decode(gzip.decode(licenseData));
|
|
if (!licenseString.contains('skia') || !licenseString.contains('Flutter Authors')) {
|
|
return TaskResult.failure('License content missing');
|
|
}
|
|
});
|
|
|
|
section('Check that the host build sends the correct analytics');
|
|
|
|
final String objectiveCAnalyticsOutput = objectiveCAnalyticsOutputFile.readAsStringSync();
|
|
if (!objectiveCAnalyticsOutput.contains('cd24: ios')
|
|
|| !objectiveCAnalyticsOutput.contains('cd25: true')
|
|
|| !objectiveCAnalyticsOutput.contains('viewName: assemble')) {
|
|
return TaskResult.failure(
|
|
'Building outer Objective-C app produced the following analytics: "$objectiveCAnalyticsOutput" '
|
|
'but not the expected strings: "cd24: ios", "cd25: true", "viewName: assemble"'
|
|
);
|
|
}
|
|
|
|
section('Archive iOS Objective-C host app');
|
|
|
|
await inDirectory(objectiveCHostApp, () async {
|
|
final Directory objectiveCBuildArchiveDirectory = Directory(path.join(tempDir.path, 'build-objc-archive'));
|
|
await exec(
|
|
'xcodebuild',
|
|
<String>[
|
|
'-workspace',
|
|
'Host.xcworkspace',
|
|
'-scheme',
|
|
'Host',
|
|
'-configuration',
|
|
'Release',
|
|
'CODE_SIGNING_ALLOWED=NO',
|
|
'CODE_SIGNING_REQUIRED=NO',
|
|
'CODE_SIGN_IDENTITY=-',
|
|
'EXPANDED_CODE_SIGN_IDENTITY=-',
|
|
'-archivePath',
|
|
objectiveCBuildArchiveDirectory.path,
|
|
'COMPILER_INDEX_STORE_ENABLE=NO',
|
|
'archive',
|
|
],
|
|
environment: <String, String> {
|
|
'FLUTTER_ANALYTICS_LOG_FILE': objectiveCAnalyticsOutputFile.path,
|
|
},
|
|
);
|
|
|
|
final String archivedAppPath = path.join(
|
|
'${objectiveCBuildArchiveDirectory.path}.xcarchive',
|
|
'Products',
|
|
'Applications',
|
|
'Host.app',
|
|
);
|
|
|
|
checkFileExists(path.join(
|
|
archivedAppPath,
|
|
'Host',
|
|
));
|
|
|
|
checkFileNotExists(path.join(
|
|
archivedAppPath,
|
|
'Frameworks',
|
|
'App.framework',
|
|
'flutter_assets',
|
|
'isolate_snapshot_data',
|
|
));
|
|
|
|
final String builtFlutterBinary = path.join(
|
|
archivedAppPath,
|
|
'Frameworks',
|
|
'Flutter.framework',
|
|
'Flutter',
|
|
);
|
|
checkFileExists(builtFlutterBinary);
|
|
if ((await fileType(builtFlutterBinary)).contains('armv7')) {
|
|
throw TaskResult.failure('Unexpected armv7 architecture slice in $builtFlutterBinary');
|
|
}
|
|
|
|
final String builtAppBinary = path.join(
|
|
archivedAppPath,
|
|
'Frameworks',
|
|
'App.framework',
|
|
'App',
|
|
);
|
|
checkFileExists(builtAppBinary);
|
|
if ((await fileType(builtAppBinary)).contains('armv7')) {
|
|
throw TaskResult.failure('Unexpected armv7 architecture slice in $builtAppBinary');
|
|
}
|
|
|
|
// Check native assets are bundled.
|
|
checkFileExists(path.join(
|
|
archivedAppPath,
|
|
'Frameworks',
|
|
'$ffiPackageName.framework',
|
|
ffiPackageName,
|
|
));
|
|
|
|
// The host app example builds plugins statically, url_launcher_ios.framework
|
|
// should not exist.
|
|
checkDirectoryNotExists(path.join(
|
|
archivedAppPath,
|
|
'Frameworks',
|
|
'url_launcher_ios.framework',
|
|
));
|
|
|
|
checkFileExists(path.join(
|
|
'${objectiveCBuildArchiveDirectory.path}.xcarchive',
|
|
'dSYMs',
|
|
'App.framework.dSYM',
|
|
'Contents',
|
|
'Resources',
|
|
'DWARF',
|
|
'App'
|
|
));
|
|
});
|
|
|
|
section('Run platform unit tests');
|
|
|
|
final String resultBundleTemp = Directory.systemTemp.createTempSync('flutter_module_test_ios_xcresult.').path;
|
|
await testWithNewIOSSimulator('TestAdd2AppSim', (String deviceId) async {
|
|
simulatorDeviceId = deviceId;
|
|
final String resultBundlePath = path.join(resultBundleTemp, 'result');
|
|
|
|
final int testResultExit = await exec(
|
|
'xcodebuild',
|
|
<String>[
|
|
'-workspace',
|
|
'Host.xcworkspace',
|
|
'-scheme',
|
|
'Host',
|
|
'-configuration',
|
|
'Debug',
|
|
'-destination',
|
|
'id=$deviceId',
|
|
'-resultBundlePath',
|
|
resultBundlePath,
|
|
'test',
|
|
'CODE_SIGNING_ALLOWED=NO',
|
|
'CODE_SIGNING_REQUIRED=NO',
|
|
'CODE_SIGN_IDENTITY=-',
|
|
'EXPANDED_CODE_SIGN_IDENTITY=-',
|
|
'COMPILER_INDEX_STORE_ENABLE=NO',
|
|
],
|
|
workingDirectory: objectiveCHostApp.path,
|
|
canFail: true,
|
|
);
|
|
|
|
if (testResultExit != 0) {
|
|
final Directory? dumpDirectory = hostAgent.dumpDirectory;
|
|
if (dumpDirectory != null) {
|
|
// Zip the test results to the artifacts directory for upload.
|
|
await inDirectory(resultBundleTemp, () {
|
|
final String zipPath = path.join(dumpDirectory.path,
|
|
'module_test_ios-objc-${DateTime.now().toLocal().toIso8601String()}.zip');
|
|
return exec(
|
|
'zip',
|
|
<String>[
|
|
'-r',
|
|
'-9',
|
|
'-q',
|
|
zipPath,
|
|
'result.xcresult',
|
|
],
|
|
canFail: true, // Best effort to get the logs.
|
|
);
|
|
});
|
|
}
|
|
|
|
throw TaskResult.failure('Platform unit tests failed');
|
|
}
|
|
});
|
|
|
|
section('Fail building existing Objective-C iOS app if flutter script fails');
|
|
final String xcodebuildOutput = await inDirectory<String>(objectiveCHostApp, () =>
|
|
eval(
|
|
'xcodebuild',
|
|
<String>[
|
|
'-workspace',
|
|
'Host.xcworkspace',
|
|
'-scheme',
|
|
'Host',
|
|
'-configuration',
|
|
'Debug',
|
|
'FLUTTER_ENGINE=bogus', // Force a Flutter error.
|
|
'CODE_SIGNING_ALLOWED=NO',
|
|
'CODE_SIGNING_REQUIRED=NO',
|
|
'CODE_SIGN_IDENTITY=-',
|
|
'EXPANDED_CODE_SIGN_IDENTITY=-',
|
|
'BUILD_DIR=${objectiveCBuildDirectory.path}',
|
|
'COMPILER_INDEX_STORE_ENABLE=NO',
|
|
],
|
|
canFail: true,
|
|
)
|
|
);
|
|
|
|
if (!xcodebuildOutput.contains('flutter --verbose --local-engine-src-path=bogus assemble') || // Verbose output
|
|
!xcodebuildOutput.contains('Unable to detect a Flutter engine build directory in bogus')) {
|
|
return TaskResult.failure('Host Objective-C app build succeeded though flutter script failed');
|
|
}
|
|
|
|
section('Add to existing iOS Swift app');
|
|
|
|
final Directory swiftHostApp = Directory(path.join(tempDir.path, 'hello_host_app_swift'));
|
|
mkdir(swiftHostApp);
|
|
recursiveCopy(
|
|
Directory(path.join(flutterDirectory.path, 'dev', 'integration_tests', 'ios_host_app_swift')),
|
|
swiftHostApp,
|
|
);
|
|
|
|
final File swiftAnalyticsOutputFile = File(path.join(tempDir.path, 'analytics-swift.log'));
|
|
final Directory swiftBuildDirectory = Directory(path.join(tempDir.path, 'build-swift'));
|
|
|
|
await inDirectory(swiftHostApp, () async {
|
|
await exec(
|
|
'pod',
|
|
<String>['install'],
|
|
environment: <String, String>{
|
|
'LANG': 'en_US.UTF-8',
|
|
},
|
|
);
|
|
await exec(
|
|
'xcodebuild',
|
|
<String>[
|
|
'-workspace',
|
|
'Host.xcworkspace',
|
|
'-scheme',
|
|
'Host',
|
|
'-configuration',
|
|
'Debug',
|
|
'CODE_SIGNING_ALLOWED=NO',
|
|
'CODE_SIGNING_REQUIRED=NO',
|
|
'CODE_SIGN_IDENTITY=-',
|
|
'EXPANDED_CODE_SIGN_IDENTITY=-',
|
|
'BUILD_DIR=${swiftBuildDirectory.path}',
|
|
'COMPILER_INDEX_STORE_ENABLE=NO',
|
|
],
|
|
environment: <String, String> {
|
|
'FLUTTER_ANALYTICS_LOG_FILE': swiftAnalyticsOutputFile.path,
|
|
},
|
|
);
|
|
});
|
|
|
|
final bool existingSwiftAppBuilt = exists(File(path.join(
|
|
swiftBuildDirectory.path,
|
|
'Debug-iphoneos',
|
|
'Host.app',
|
|
'Host',
|
|
)));
|
|
if (!existingSwiftAppBuilt) {
|
|
return TaskResult.failure('Failed to build existing Swift app .app');
|
|
}
|
|
|
|
final String swiftAnalyticsOutput = swiftAnalyticsOutputFile.readAsStringSync();
|
|
if (!swiftAnalyticsOutput.contains('cd24: ios')
|
|
|| !swiftAnalyticsOutput.contains('cd25: true')
|
|
|| !swiftAnalyticsOutput.contains('viewName: assemble')) {
|
|
return TaskResult.failure(
|
|
'Building outer Swift app produced the following analytics: "$swiftAnalyticsOutput" '
|
|
'but not the expected strings: "cd24: ios", "cd25: true", "viewName: assemble"'
|
|
);
|
|
}
|
|
|
|
return TaskResult.success(null);
|
|
} catch (e) {
|
|
return TaskResult.failure(e.toString());
|
|
} finally {
|
|
unawaited(removeIOSSimulator(simulatorDeviceId));
|
|
rmTree(tempDir);
|
|
}
|
|
});
|
|
}
|
|
|
|
Future<bool> _isAppAotBuild(Directory app) async {
|
|
final String binary = path.join(
|
|
app.path,
|
|
'Frameworks',
|
|
'App.framework',
|
|
'App',
|
|
);
|
|
|
|
final String symbolTable = await eval(
|
|
'nm',
|
|
<String> [
|
|
'-gU',
|
|
binary,
|
|
],
|
|
);
|
|
|
|
return symbolTable.contains('kDartIsolateSnapshotInstructions');
|
|
}
|
|
|
|
Future<void> _createFakeDartPlugin(String name, Directory parent) async {
|
|
// Start from a standard plugin template.
|
|
await inDirectory(parent, () async {
|
|
await flutter(
|
|
'create',
|
|
options: <String>[
|
|
'--org',
|
|
'io.flutter.devicelab',
|
|
'--template=plugin',
|
|
'--platforms=ios',
|
|
name,
|
|
],
|
|
);
|
|
});
|
|
|
|
final String pluginDir = path.join(parent.path, name);
|
|
|
|
// Convert the metadata to Dart-only.
|
|
final String dartPluginClass = 'DartClassFor$name';
|
|
final File pubspec = File(path.join(pluginDir, 'pubspec.yaml'));
|
|
String content = await pubspec.readAsString();
|
|
content = content.replaceAll(
|
|
RegExp(r' pluginClass: .*?\n'),
|
|
' dartPluginClass: $dartPluginClass\n',
|
|
);
|
|
await pubspec.writeAsString(content, flush: true);
|
|
|
|
// Add the Dart registration hook that the build will generate a call to.
|
|
final File dartCode = File(path.join(pluginDir, 'lib', '$name.dart'));
|
|
content = await dartCode.readAsString();
|
|
content = '''
|
|
$content
|
|
|
|
class $dartPluginClass {
|
|
static void registerWith() {}
|
|
}
|
|
''';
|
|
await dartCode.writeAsString(content, flush: true);
|
|
|
|
// Remove the native plugin code.
|
|
await Directory(path.join(pluginDir, 'ios')).delete(recursive: true);
|
|
}
|