Unverified Commit 8a33d244 authored by Zachary Anderson's avatar Zachary Anderson Committed by GitHub

[flutter_tool] Remove the synchronous -showBuildSettings (#40435)

parent ef126954
...@@ -328,7 +328,7 @@ abstract class IOSApp extends ApplicationPackage { ...@@ -328,7 +328,7 @@ abstract class IOSApp extends ApplicationPackage {
); );
} }
factory IOSApp.fromIosProject(IosProject project) { static Future<IOSApp> fromIosProject(IosProject project) {
if (getCurrentHostPlatform() != HostPlatform.darwin_x64) { if (getCurrentHostPlatform() != HostPlatform.darwin_x64) {
return null; return null;
} }
...@@ -345,7 +345,7 @@ abstract class IOSApp extends ApplicationPackage { ...@@ -345,7 +345,7 @@ abstract class IOSApp extends ApplicationPackage {
printError('Expected ios/Runner.xcodeproj/project.pbxproj but this file is missing.'); printError('Expected ios/Runner.xcodeproj/project.pbxproj but this file is missing.');
return null; return null;
} }
return BuildableIOSApp(project); return BuildableIOSApp.fromProject(project);
} }
@override @override
...@@ -357,7 +357,13 @@ abstract class IOSApp extends ApplicationPackage { ...@@ -357,7 +357,13 @@ abstract class IOSApp extends ApplicationPackage {
} }
class BuildableIOSApp extends IOSApp { class BuildableIOSApp extends IOSApp {
BuildableIOSApp(this.project) : super(projectBundleId: project.productBundleIdentifier); BuildableIOSApp(this.project, String projectBundleId)
: super(projectBundleId: projectBundleId);
static Future<BuildableIOSApp> fromProject(IosProject project) async {
final String projectBundleId = await project.productBundleIdentifier;
return BuildableIOSApp(project, projectBundleId);
}
final IosProject project; final IosProject project;
...@@ -416,7 +422,7 @@ class ApplicationPackageStore { ...@@ -416,7 +422,7 @@ class ApplicationPackageStore {
android ??= await AndroidApk.fromAndroidProject(FlutterProject.current().android); android ??= await AndroidApk.fromAndroidProject(FlutterProject.current().android);
return android; return android;
case TargetPlatform.ios: case TargetPlatform.ios:
iOS ??= IOSApp.fromIosProject(FlutterProject.current().ios); iOS ??= await IOSApp.fromIosProject(FlutterProject.current().ios);
return iOS; return iOS;
case TargetPlatform.fuchsia: case TargetPlatform.fuchsia:
fuchsia ??= FuchsiaApp.fromFuchsiaProject(FlutterProject.current().fuchsia); fuchsia ??= FuchsiaApp.fromFuchsiaProject(FlutterProject.current().fuchsia);
......
...@@ -352,7 +352,7 @@ class CreateCommand extends FlutterCommand { ...@@ -352,7 +352,7 @@ class CreateCommand extends FlutterCommand {
String organization = argResults['org']; String organization = argResults['org'];
if (!argResults.wasParsed('org')) { if (!argResults.wasParsed('org')) {
final FlutterProject project = FlutterProject.fromDirectory(projectDir); final FlutterProject project = FlutterProject.fromDirectory(projectDir);
final Set<String> existingOrganizations = project.organizationNames; final Set<String> existingOrganizations = await project.organizationNames;
if (existingOrganizations.length == 1) { if (existingOrganizations.length == 1) {
organization = existingOrganizations.first; organization = existingOrganizations.first;
} else if (1 < existingOrganizations.length) { } else if (1 < existingOrganizations.length) {
......
...@@ -231,7 +231,7 @@ class RunCommand extends RunCommandBase { ...@@ -231,7 +231,7 @@ class RunCommand extends RunCommandBase {
hostLanguage.add(androidProject.isKotlin ? 'kotlin' : 'java'); hostLanguage.add(androidProject.isKotlin ? 'kotlin' : 'java');
} }
if (iosProject != null && iosProject.exists) { if (iosProject != null && iosProject.exists) {
hostLanguage.add(iosProject.isSwift ? 'swift' : 'objc'); hostLanguage.add(await iosProject.isSwift ? 'swift' : 'objc');
} }
return <CustomDimensions, String>{ return <CustomDimensions, String>{
......
...@@ -95,7 +95,7 @@ final RegExp _certificateOrganizationalUnitExtractionPattern = RegExp(r'OU=([a-z ...@@ -95,7 +95,7 @@ final RegExp _certificateOrganizationalUnitExtractionPattern = RegExp(r'OU=([a-z
Future<Map<String, String>> getCodeSigningIdentityDevelopmentTeam({ Future<Map<String, String>> getCodeSigningIdentityDevelopmentTeam({
BuildableIOSApp iosApp, BuildableIOSApp iosApp,
}) async { }) async {
final Map<String, String> buildSettings = iosApp.project.buildSettings; final Map<String, String> buildSettings = await iosApp.project.buildSettings;
if (buildSettings == null) { if (buildSettings == null) {
return null; return null;
} }
......
...@@ -258,32 +258,9 @@ class XcodeProjectInterpreter { ...@@ -258,32 +258,9 @@ class XcodeProjectInterpreter {
return _minorVersion; return _minorVersion;
} }
/// Synchronously retrieve xcode build settings. Prefer using the async
/// version below.
Map<String, String> getBuildSettings(String projectPath, String target) {
try {
final String out = processUtils.runSync(
<String>[
_executable,
'-project',
fs.path.absolute(projectPath),
'-target',
target,
'-showBuildSettings',
],
throwOnError: true,
workingDirectory: projectPath,
).stdout.trim();
return parseXcodeBuildSettings(out);
} on ProcessException catch (error) {
printTrace('Unexpected failure to get the build settings: $error.');
return const <String, String>{};
}
}
/// Asynchronously retrieve xcode build settings. This one is preferred for /// Asynchronously retrieve xcode build settings. This one is preferred for
/// new call-sites. /// new call-sites.
Future<Map<String, String>> getBuildSettingsAsync( Future<Map<String, String>> getBuildSettings(
String projectPath, String target, { String projectPath, String target, {
Duration timeout = const Duration(minutes: 1), Duration timeout = const Duration(minutes: 1),
}) async { }) async {
......
...@@ -37,7 +37,7 @@ Future<void> processPodsIfNeeded(XcodeBasedProject xcodeProject, ...@@ -37,7 +37,7 @@ Future<void> processPodsIfNeeded(XcodeBasedProject xcodeProject,
final bool didPodInstall = await cocoaPods.processPods( final bool didPodInstall = await cocoaPods.processPods(
xcodeProject: xcodeProject, xcodeProject: xcodeProject,
engineDir: flutterFrameworkDir(buildMode), engineDir: flutterFrameworkDir(buildMode),
isSwift: xcodeProject.isSwift, isSwift: await xcodeProject.isSwift,
dependenciesChanged: !fingerprinter.doesFingerprintMatch(), dependenciesChanged: !fingerprinter.doesFingerprintMatch(),
); );
if (didPodInstall) { if (didPodInstall) {
......
...@@ -213,7 +213,7 @@ class CocoaPods { ...@@ -213,7 +213,7 @@ class CocoaPods {
if (xcodeProject is MacOSProject) { if (xcodeProject is MacOSProject) {
podfileTemplateName = 'Podfile-macos'; podfileTemplateName = 'Podfile-macos';
} else { } else {
final bool isSwift = (await xcodeProjectInterpreter.getBuildSettingsAsync( final bool isSwift = (await xcodeProjectInterpreter.getBuildSettings(
runnerProject.path, runnerProject.path,
'Runner', 'Runner',
)).containsKey('SWIFT_VERSION'); )).containsKey('SWIFT_VERSION');
......
...@@ -88,13 +88,13 @@ class FlutterProject { ...@@ -88,13 +88,13 @@ class FlutterProject {
/// The set of organization names found in this project as /// The set of organization names found in this project as
/// part of iOS product bundle identifier, Android application ID, or /// part of iOS product bundle identifier, Android application ID, or
/// Gradle group ID. /// Gradle group ID.
Set<String> get organizationNames { Future<Set<String>> get organizationNames async {
final List<String> candidates = <String>[ final List<String> candidates = <String>[
ios.productBundleIdentifier, await ios.productBundleIdentifier,
android.applicationId, android.applicationId,
android.group, android.group,
example.android.applicationId, example.android.applicationId,
example.ios.productBundleIdentifier, await example.ios.productBundleIdentifier,
]; ];
return Set<String>.from(candidates return Set<String>.from(candidates
.map<String>(_organizationNameFromPackageName) .map<String>(_organizationNameFromPackageName)
...@@ -280,7 +280,7 @@ abstract class XcodeBasedProject { ...@@ -280,7 +280,7 @@ abstract class XcodeBasedProject {
File get podManifestLock; File get podManifestLock;
/// True if the host app project is using Swift. /// True if the host app project is using Swift.
bool get isSwift; Future<bool> get isSwift;
} }
/// Represents the iOS sub-project of a Flutter project. /// Represents the iOS sub-project of a Flutter project.
...@@ -365,7 +365,7 @@ class IosProject implements XcodeBasedProject { ...@@ -365,7 +365,7 @@ class IosProject implements XcodeBasedProject {
/// The product bundle identifier of the host app, or null if not set or if /// The product bundle identifier of the host app, or null if not set or if
/// iOS tooling needed to read it is not installed. /// iOS tooling needed to read it is not installed.
String get productBundleIdentifier { Future<String> get productBundleIdentifier async {
String fromPlist; String fromPlist;
try { try {
fromPlist = PlistParser.instance.getValueFromFile( fromPlist = PlistParser.instance.getValueFromFile(
...@@ -386,24 +386,26 @@ class IosProject implements XcodeBasedProject { ...@@ -386,24 +386,26 @@ class IosProject implements XcodeBasedProject {
} }
if (fromPlist != null && xcode.xcodeProjectInterpreter.isInstalled) { if (fromPlist != null && xcode.xcodeProjectInterpreter.isInstalled) {
// General case: perform variable substitution using build settings. // General case: perform variable substitution using build settings.
return xcode.substituteXcodeVariables(fromPlist, buildSettings); return xcode.substituteXcodeVariables(fromPlist, await buildSettings);
} }
return null; return null;
} }
@override @override
bool get isSwift => buildSettings?.containsKey('SWIFT_VERSION') ?? false; Future<bool> get isSwift async =>
(await buildSettings)?.containsKey('SWIFT_VERSION') ?? false;
/// The build settings for the host app of this project, as a detached map. /// The build settings for the host app of this project, as a detached map.
/// ///
/// Returns null, if iOS tooling is unavailable. /// Returns null, if iOS tooling is unavailable.
Map<String, String> get buildSettings { Future<Map<String, String>> get buildSettings async {
if (!xcode.xcodeProjectInterpreter.isInstalled) { if (!xcode.xcodeProjectInterpreter.isInstalled) {
return null; return null;
} }
_buildSettings ??= _buildSettings ??= await xcode.xcodeProjectInterpreter.getBuildSettings(
xcode.xcodeProjectInterpreter.getBuildSettings(xcodeProject.path, xcodeProject.path,
_hostAppBundleName); _hostAppBundleName,
);
return _buildSettings; return _buildSettings;
} }
...@@ -726,7 +728,7 @@ class MacOSProject implements XcodeBasedProject { ...@@ -726,7 +728,7 @@ class MacOSProject implements XcodeBasedProject {
Directory get xcodeWorkspace => _macOSDirectory.childDirectory('$_hostAppBundleName.xcworkspace'); Directory get xcodeWorkspace => _macOSDirectory.childDirectory('$_hostAppBundleName.xcworkspace');
@override @override
bool get isSwift => true; Future<bool> get isSwift async => true;
/// The file where the Xcode build will write the name of the built app. /// The file where the Xcode build will write the name of the built app.
/// ///
......
...@@ -325,7 +325,8 @@ void main() { ...@@ -325,7 +325,8 @@ void main() {
testUsingContext('returns null when there is no ios or .ios directory', () async { testUsingContext('returns null when there is no ios or .ios directory', () async {
fs.file('pubspec.yaml').createSync(); fs.file('pubspec.yaml').createSync();
fs.file('.packages').createSync(); fs.file('.packages').createSync();
final BuildableIOSApp iosApp = IOSApp.fromIosProject(FlutterProject.fromDirectory(fs.currentDirectory).ios); final BuildableIOSApp iosApp = await IOSApp.fromIosProject(
FlutterProject.fromDirectory(fs.currentDirectory).ios);
expect(iosApp, null); expect(iosApp, null);
}, overrides: overrides); }, overrides: overrides);
...@@ -334,7 +335,8 @@ void main() { ...@@ -334,7 +335,8 @@ void main() {
fs.file('pubspec.yaml').createSync(); fs.file('pubspec.yaml').createSync();
fs.file('.packages').createSync(); fs.file('.packages').createSync();
fs.file('ios/FooBar.xcodeproj').createSync(recursive: true); fs.file('ios/FooBar.xcodeproj').createSync(recursive: true);
final BuildableIOSApp iosApp = IOSApp.fromIosProject(FlutterProject.fromDirectory(fs.currentDirectory).ios); final BuildableIOSApp iosApp = await IOSApp.fromIosProject(
FlutterProject.fromDirectory(fs.currentDirectory).ios);
expect(iosApp, null); expect(iosApp, null);
}, overrides: overrides); }, overrides: overrides);
...@@ -343,7 +345,8 @@ void main() { ...@@ -343,7 +345,8 @@ void main() {
fs.file('pubspec.yaml').createSync(); fs.file('pubspec.yaml').createSync();
fs.file('.packages').createSync(); fs.file('.packages').createSync();
fs.file('ios/Runner.xcodeproj').createSync(recursive: true); fs.file('ios/Runner.xcodeproj').createSync(recursive: true);
final BuildableIOSApp iosApp = IOSApp.fromIosProject(FlutterProject.fromDirectory(fs.currentDirectory).ios); final BuildableIOSApp iosApp = await IOSApp.fromIosProject(
FlutterProject.fromDirectory(fs.currentDirectory).ios);
expect(iosApp, null); expect(iosApp, null);
}, overrides: overrides); }, overrides: overrides);
......
...@@ -707,7 +707,7 @@ void main() { ...@@ -707,7 +707,7 @@ void main() {
await runner.run(<String>['create', '--template=app', '--no-pub', '--org', 'com.example', tmpProjectDir]); await runner.run(<String>['create', '--template=app', '--no-pub', '--org', 'com.example', tmpProjectDir]);
FlutterProject project = FlutterProject.fromDirectory(fs.directory(tmpProjectDir)); FlutterProject project = FlutterProject.fromDirectory(fs.directory(tmpProjectDir));
expect( expect(
project.ios.productBundleIdentifier, await project.ios.productBundleIdentifier,
'com.example.helloFlutter', 'com.example.helloFlutter',
); );
expect( expect(
...@@ -719,7 +719,7 @@ void main() { ...@@ -719,7 +719,7 @@ void main() {
await runner.run(<String>['create', '--template=app', '--no-pub', '--org', 'abc^*.1#@', tmpProjectDir]); await runner.run(<String>['create', '--template=app', '--no-pub', '--org', 'abc^*.1#@', tmpProjectDir]);
project = FlutterProject.fromDirectory(fs.directory(tmpProjectDir)); project = FlutterProject.fromDirectory(fs.directory(tmpProjectDir));
expect( expect(
project.ios.productBundleIdentifier, await project.ios.productBundleIdentifier,
'abc.1.testAbc', 'abc.1.testAbc',
); );
expect( expect(
...@@ -731,7 +731,7 @@ void main() { ...@@ -731,7 +731,7 @@ void main() {
await runner.run(<String>['create', '--template=app', '--no-pub', '--org', '#+^%', tmpProjectDir]); await runner.run(<String>['create', '--template=app', '--no-pub', '--org', '#+^%', tmpProjectDir]);
project = FlutterProject.fromDirectory(fs.directory(tmpProjectDir)); project = FlutterProject.fromDirectory(fs.directory(tmpProjectDir));
expect( expect(
project.ios.productBundleIdentifier, await project.ios.productBundleIdentifier,
'flutterProject.untitled', 'flutterProject.untitled',
); );
expect( expect(
...@@ -856,7 +856,7 @@ void main() { ...@@ -856,7 +856,7 @@ void main() {
await _createProject(projectDir, <String>[], <String>[]); await _createProject(projectDir, <String>[], <String>[]);
final FlutterProject project = FlutterProject.fromDirectory(projectDir); final FlutterProject project = FlutterProject.fromDirectory(projectDir);
expect( expect(
project.ios.productBundleIdentifier, await project.ios.productBundleIdentifier,
'com.bar.foo.flutterProject', 'com.bar.foo.flutterProject',
); );
}, timeout: allowForRemotePubInvocation); }, timeout: allowForRemotePubInvocation);
...@@ -896,7 +896,7 @@ void main() { ...@@ -896,7 +896,7 @@ void main() {
await _createProject(projectDir, <String>['--no-pub'], <String>[]); await _createProject(projectDir, <String>['--no-pub'], <String>[]);
final FlutterProject project = FlutterProject.fromDirectory(projectDir); final FlutterProject project = FlutterProject.fromDirectory(projectDir);
expect( expect(
project.ios.productBundleIdentifier, await project.ios.productBundleIdentifier,
'com.bar.foo.flutterProject', 'com.bar.foo.flutterProject',
); );
}, timeout: allowForCreateFlutterProject); }, timeout: allowForCreateFlutterProject);
...@@ -929,7 +929,7 @@ void main() { ...@@ -929,7 +929,7 @@ void main() {
); );
final FlutterProject project = FlutterProject.fromDirectory(projectDir); final FlutterProject project = FlutterProject.fromDirectory(projectDir);
expect( expect(
project.example.ios.productBundleIdentifier, await project.example.ios.productBundleIdentifier,
'com.bar.foo.flutterProjectExample', 'com.bar.foo.flutterProjectExample',
); );
}, timeout: allowForCreateFlutterProject); }, timeout: allowForCreateFlutterProject);
......
...@@ -27,16 +27,18 @@ void main() { ...@@ -27,16 +27,18 @@ void main() {
BuildableIOSApp app; BuildableIOSApp app;
AnsiTerminal testTerminal; AnsiTerminal testTerminal;
setUp(() { setUp(() async {
mockProcessManager = MockProcessManager(); mockProcessManager = MockProcessManager();
mockConfig = MockConfig(); mockConfig = MockConfig();
mockIosProject = MockIosProject(); mockIosProject = MockIosProject();
when(mockIosProject.buildSettings).thenReturn(<String, String>{ when(mockIosProject.buildSettings).thenAnswer((_) {
return Future<Map<String, String>>.value(<String, String>{
'For our purposes': 'a non-empty build settings map is valid', 'For our purposes': 'a non-empty build settings map is valid',
}); });
});
testTerminal = TestTerminal(); testTerminal = TestTerminal();
testTerminal.usesTerminalUi = true; testTerminal.usesTerminalUi = true;
app = BuildableIOSApp(mockIosProject); app = await BuildableIOSApp.fromProject(mockIosProject);
}); });
testUsingContext('No auto-sign if Xcode project settings are not available', () async { testUsingContext('No auto-sign if Xcode project settings are not available', () async {
...@@ -46,9 +48,11 @@ void main() { ...@@ -46,9 +48,11 @@ void main() {
}); });
testUsingContext('No discovery if development team specified in Xcode project', () async { testUsingContext('No discovery if development team specified in Xcode project', () async {
when(mockIosProject.buildSettings).thenReturn(<String, String>{ when(mockIosProject.buildSettings).thenAnswer((_) {
return Future<Map<String, String>>.value(<String, String>{
'DEVELOPMENT_TEAM': 'abc', 'DEVELOPMENT_TEAM': 'abc',
}); });
});
final Map<String, String> signingConfigs = await getCodeSigningIdentityDevelopmentTeam(iosApp: app); final Map<String, String> signingConfigs = await getCodeSigningIdentityDevelopmentTeam(iosApp: app);
expect(signingConfigs, isNull); expect(signingConfigs, isNull);
expect(testLogger.statusText, equals( expect(testLogger.statusText, equals(
......
...@@ -295,8 +295,8 @@ void main() { ...@@ -295,8 +295,8 @@ void main() {
projectDir.path, projectDir.path,
]); ]);
final IOSApp app = final IOSApp app = await AbsoluteBuildableIOSApp.fromProject(
AbsoluteBuildableIOSApp(FlutterProject.fromDirectory(projectDir).ios); FlutterProject.fromDirectory(projectDir).ios);
final IOSDevice device = IOSDevice('123'); final IOSDevice device = IOSDevice('123');
// Pre-create the expected build products. // Pre-create the expected build products.
...@@ -547,7 +547,7 @@ Runner(UIKit)[297] <Notice>: E is for enpitsu" ...@@ -547,7 +547,7 @@ Runner(UIKit)[297] <Notice>: E is for enpitsu"
final IOSDevice device = IOSDevice('123456'); final IOSDevice device = IOSDevice('123456');
final DeviceLogReader logReader = device.getLogReader( final DeviceLogReader logReader = device.getLogReader(
app: BuildableIOSApp(mockIosProject), app: await BuildableIOSApp.fromProject(mockIosProject),
); );
final List<String> lines = await logReader.logLines.toList(); final List<String> lines = await logReader.logLines.toList();
...@@ -572,7 +572,7 @@ Runner(libsystem_asl.dylib)[297] <Notice>: libMobileGestalt ...@@ -572,7 +572,7 @@ Runner(libsystem_asl.dylib)[297] <Notice>: libMobileGestalt
final IOSDevice device = IOSDevice('123456'); final IOSDevice device = IOSDevice('123456');
final DeviceLogReader logReader = device.getLogReader( final DeviceLogReader logReader = device.getLogReader(
app: BuildableIOSApp(mockIosProject), app: await BuildableIOSApp.fromProject(mockIosProject),
); );
final List<String> lines = await logReader.logLines.toList(); final List<String> lines = await logReader.logLines.toList();
...@@ -630,7 +630,13 @@ flutter: ...@@ -630,7 +630,13 @@ flutter:
} }
class AbsoluteBuildableIOSApp extends BuildableIOSApp { class AbsoluteBuildableIOSApp extends BuildableIOSApp {
AbsoluteBuildableIOSApp(IosProject project) : super(project); AbsoluteBuildableIOSApp(IosProject project, String projectBundleId) :
super(project, projectBundleId);
static Future<AbsoluteBuildableIOSApp> fromProject(IosProject project) async {
final String projectBundleId = await project.productBundleIdentifier;
return AbsoluteBuildableIOSApp(project, projectBundleId);
}
@override @override
String get deviceBundlePath => String get deviceBundlePath =>
......
...@@ -338,7 +338,7 @@ void main() { ...@@ -338,7 +338,7 @@ void main() {
final IOSSimulator device = IOSSimulator('123456', simulatorCategory: 'iOS 11.0'); final IOSSimulator device = IOSSimulator('123456', simulatorCategory: 'iOS 11.0');
final DeviceLogReader logReader = device.getLogReader( final DeviceLogReader logReader = device.getLogReader(
app: BuildableIOSApp(mockIosProject), app: await BuildableIOSApp.fromProject(mockIosProject),
); );
final List<String> lines = await logReader.logLines.toList(); final List<String> lines = await logReader.logLines.toList();
......
...@@ -145,13 +145,13 @@ void main() { ...@@ -145,13 +145,13 @@ void main() {
expect(xcodeProjectInterpreter.isInstalled, isTrue); expect(xcodeProjectInterpreter.isInstalled, isTrue);
}); });
testUsingOsxContext('build settings is empty when xcodebuild failed to get the build settings', () { testUsingOsxContext('build settings is empty when xcodebuild failed to get the build settings', () async {
when(mockProcessManager.runSync( when(mockProcessManager.runSync(
argThat(contains(xcodebuild)), argThat(contains(xcodebuild)),
workingDirectory: anyNamed('workingDirectory'), workingDirectory: anyNamed('workingDirectory'),
environment: anyNamed('environment'))) environment: anyNamed('environment')))
.thenReturn(ProcessResult(0, 1, '', '')); .thenReturn(ProcessResult(0, 1, '', ''));
expect(xcodeProjectInterpreter.getBuildSettings('', ''), const <String, String>{}); expect(await xcodeProjectInterpreter.getBuildSettings('', ''), const <String, String>{});
}); });
testUsingContext('build settings flakes', () async { testUsingContext('build settings flakes', () async {
...@@ -160,7 +160,7 @@ void main() { ...@@ -160,7 +160,7 @@ void main() {
flakes: 1, flakes: 1,
delay: delay + const Duration(seconds: 1), delay: delay + const Duration(seconds: 1),
); );
expect(await xcodeProjectInterpreter.getBuildSettingsAsync( expect(await xcodeProjectInterpreter.getBuildSettings(
'', '', timeout: delay), '', '', timeout: delay),
const <String, String>{}); const <String, String>{});
// build settings times out and is killed once, then succeeds. // build settings times out and is killed once, then succeeds.
......
...@@ -173,7 +173,7 @@ void main() { ...@@ -173,7 +173,7 @@ void main() {
testUsingContext('creates swift Podfile if swift', () async { testUsingContext('creates swift Podfile if swift', () async {
when(mockXcodeProjectInterpreter.isInstalled).thenReturn(true); when(mockXcodeProjectInterpreter.isInstalled).thenReturn(true);
when(mockXcodeProjectInterpreter.getBuildSettingsAsync(any, any)) when(mockXcodeProjectInterpreter.getBuildSettings(any, any))
.thenAnswer((_) async => <String, String>{ .thenAnswer((_) async => <String, String>{
'SWIFT_VERSION': '4.0', 'SWIFT_VERSION': '4.0',
}); });
......
...@@ -257,16 +257,19 @@ void main() { ...@@ -257,16 +257,19 @@ void main() {
testInMemory('default host app language', () async { testInMemory('default host app language', () async {
final FlutterProject project = await someProject(); final FlutterProject project = await someProject();
expect(project.ios.isSwift, isFalse); expect(await project.ios.isSwift, isFalse);
expect(project.android.isKotlin, isFalse); expect(project.android.isKotlin, isFalse);
}); });
testUsingContext('swift and kotlin host app language', () async { testUsingContext('swift and kotlin host app language', () async {
final FlutterProject project = await someProject(); final FlutterProject project = await someProject();
when(mockXcodeProjectInterpreter.getBuildSettings(any, any)).thenReturn(<String, String>{ when(mockXcodeProjectInterpreter.getBuildSettings(any, any)).thenAnswer(
(_) {
return Future<Map<String, String>>.value(<String, String>{
'SWIFT_VERSION': '4.0', 'SWIFT_VERSION': '4.0',
}); });
});
addAndroidGradleFile(project.directory, addAndroidGradleFile(project.directory,
gradleFileContent: () { gradleFileContent: () {
return ''' return '''
...@@ -274,7 +277,7 @@ apply plugin: 'com.android.application' ...@@ -274,7 +277,7 @@ apply plugin: 'com.android.application'
apply plugin: 'kotlin-android' apply plugin: 'kotlin-android'
'''; ''';
}); });
expect(project.ios.isSwift, isTrue); expect(await project.ios.isSwift, isTrue);
expect(project.android.isKotlin, isTrue); expect(project.android.isKotlin, isTrue);
}, overrides: <Type, Generator>{ }, overrides: <Type, Generator>{
FileSystem: () => fs, FileSystem: () => fs,
...@@ -306,19 +309,19 @@ apply plugin: 'kotlin-android' ...@@ -306,19 +309,19 @@ apply plugin: 'kotlin-android'
testWithMocks('null, if no pbxproj or plist entries', () async { testWithMocks('null, if no pbxproj or plist entries', () async {
final FlutterProject project = await someProject(); final FlutterProject project = await someProject();
expect(project.ios.productBundleIdentifier, isNull); expect(await project.ios.productBundleIdentifier, isNull);
}); });
testWithMocks('from pbxproj file, if no plist', () async { testWithMocks('from pbxproj file, if no plist', () async {
final FlutterProject project = await someProject(); final FlutterProject project = await someProject();
addIosProjectFile(project.directory, projectFileContent: () { addIosProjectFile(project.directory, projectFileContent: () {
return projectFileWithBundleId('io.flutter.someProject'); return projectFileWithBundleId('io.flutter.someProject');
}); });
expect(project.ios.productBundleIdentifier, 'io.flutter.someProject'); expect(await project.ios.productBundleIdentifier, 'io.flutter.someProject');
}); });
testWithMocks('from plist, if no variables', () async { testWithMocks('from plist, if no variables', () async {
final FlutterProject project = await someProject(); final FlutterProject project = await someProject();
when(mockPlistUtils.getValueFromFile(any, any)).thenReturn('io.flutter.someProject'); when(mockPlistUtils.getValueFromFile(any, any)).thenReturn('io.flutter.someProject');
expect(project.ios.productBundleIdentifier, 'io.flutter.someProject'); expect(await project.ios.productBundleIdentifier, 'io.flutter.someProject');
}); });
testWithMocks('from pbxproj and plist, if default variable', () async { testWithMocks('from pbxproj and plist, if default variable', () async {
final FlutterProject project = await someProject(); final FlutterProject project = await someProject();
...@@ -326,56 +329,60 @@ apply plugin: 'kotlin-android' ...@@ -326,56 +329,60 @@ apply plugin: 'kotlin-android'
return projectFileWithBundleId('io.flutter.someProject'); return projectFileWithBundleId('io.flutter.someProject');
}); });
when(mockPlistUtils.getValueFromFile(any, any)).thenReturn('\$(PRODUCT_BUNDLE_IDENTIFIER)'); when(mockPlistUtils.getValueFromFile(any, any)).thenReturn('\$(PRODUCT_BUNDLE_IDENTIFIER)');
expect(project.ios.productBundleIdentifier, 'io.flutter.someProject'); expect(await project.ios.productBundleIdentifier, 'io.flutter.someProject');
}); });
testWithMocks('from pbxproj and plist, by substitution', () async { testWithMocks('from pbxproj and plist, by substitution', () async {
final FlutterProject project = await someProject(); final FlutterProject project = await someProject();
when(mockXcodeProjectInterpreter.getBuildSettings(any, any)).thenReturn(<String, String>{ when(mockXcodeProjectInterpreter.getBuildSettings(any, any)).thenAnswer(
(_) {
return Future<Map<String,String>>.value(<String, String>{
'PRODUCT_BUNDLE_IDENTIFIER': 'io.flutter.someProject', 'PRODUCT_BUNDLE_IDENTIFIER': 'io.flutter.someProject',
'SUFFIX': 'suffix', 'SUFFIX': 'suffix',
}); });
}
);
when(mockPlistUtils.getValueFromFile(any, any)).thenReturn('\$(PRODUCT_BUNDLE_IDENTIFIER).\$(SUFFIX)'); when(mockPlistUtils.getValueFromFile(any, any)).thenReturn('\$(PRODUCT_BUNDLE_IDENTIFIER).\$(SUFFIX)');
expect(project.ios.productBundleIdentifier, 'io.flutter.someProject.suffix'); expect(await project.ios.productBundleIdentifier, 'io.flutter.someProject.suffix');
}); });
testWithMocks('empty surrounded by quotes', () async { testWithMocks('empty surrounded by quotes', () async {
final FlutterProject project = await someProject(); final FlutterProject project = await someProject();
addIosProjectFile(project.directory, projectFileContent: () { addIosProjectFile(project.directory, projectFileContent: () {
return projectFileWithBundleId('', qualifier: '"'); return projectFileWithBundleId('', qualifier: '"');
}); });
expect(project.ios.productBundleIdentifier, ''); expect(await project.ios.productBundleIdentifier, '');
}); });
testWithMocks('surrounded by double quotes', () async { testWithMocks('surrounded by double quotes', () async {
final FlutterProject project = await someProject(); final FlutterProject project = await someProject();
addIosProjectFile(project.directory, projectFileContent: () { addIosProjectFile(project.directory, projectFileContent: () {
return projectFileWithBundleId('io.flutter.someProject', qualifier: '"'); return projectFileWithBundleId('io.flutter.someProject', qualifier: '"');
}); });
expect(project.ios.productBundleIdentifier, 'io.flutter.someProject'); expect(await project.ios.productBundleIdentifier, 'io.flutter.someProject');
}); });
testWithMocks('surrounded by single quotes', () async { testWithMocks('surrounded by single quotes', () async {
final FlutterProject project = await someProject(); final FlutterProject project = await someProject();
addIosProjectFile(project.directory, projectFileContent: () { addIosProjectFile(project.directory, projectFileContent: () {
return projectFileWithBundleId('io.flutter.someProject', qualifier: '\''); return projectFileWithBundleId('io.flutter.someProject', qualifier: '\'');
}); });
expect(project.ios.productBundleIdentifier, 'io.flutter.someProject'); expect(await project.ios.productBundleIdentifier, 'io.flutter.someProject');
}); });
}); });
group('organization names set', () { group('organization names set', () {
testInMemory('is empty, if project not created', () async { testInMemory('is empty, if project not created', () async {
final FlutterProject project = await someProject(); final FlutterProject project = await someProject();
expect(project.organizationNames, isEmpty); expect(await project.organizationNames, isEmpty);
}); });
testInMemory('is empty, if no platform folders exist', () async { testInMemory('is empty, if no platform folders exist', () async {
final FlutterProject project = await someProject(); final FlutterProject project = await someProject();
project.directory.createSync(); project.directory.createSync();
expect(project.organizationNames, isEmpty); expect(await project.organizationNames, isEmpty);
}); });
testInMemory('is populated from iOS bundle identifier', () async { testInMemory('is populated from iOS bundle identifier', () async {
final FlutterProject project = await someProject(); final FlutterProject project = await someProject();
addIosProjectFile(project.directory, projectFileContent: () { addIosProjectFile(project.directory, projectFileContent: () {
return projectFileWithBundleId('io.flutter.someProject', qualifier: '\''); return projectFileWithBundleId('io.flutter.someProject', qualifier: '\'');
}); });
expect(project.organizationNames, <String>['io.flutter']); expect(await project.organizationNames, <String>['io.flutter']);
}); });
testInMemory('is populated from Android application ID', () async { testInMemory('is populated from Android application ID', () async {
final FlutterProject project = await someProject(); final FlutterProject project = await someProject();
...@@ -383,14 +390,14 @@ apply plugin: 'kotlin-android' ...@@ -383,14 +390,14 @@ apply plugin: 'kotlin-android'
gradleFileContent: () { gradleFileContent: () {
return gradleFileWithApplicationId('io.flutter.someproject'); return gradleFileWithApplicationId('io.flutter.someproject');
}); });
expect(project.organizationNames, <String>['io.flutter']); expect(await project.organizationNames, <String>['io.flutter']);
}); });
testInMemory('is populated from iOS bundle identifier in plugin example', () async { testInMemory('is populated from iOS bundle identifier in plugin example', () async {
final FlutterProject project = await someProject(); final FlutterProject project = await someProject();
addIosProjectFile(project.example.directory, projectFileContent: () { addIosProjectFile(project.example.directory, projectFileContent: () {
return projectFileWithBundleId('io.flutter.someProject', qualifier: '\''); return projectFileWithBundleId('io.flutter.someProject', qualifier: '\'');
}); });
expect(project.organizationNames, <String>['io.flutter']); expect(await project.organizationNames, <String>['io.flutter']);
}); });
testInMemory('is populated from Android application ID in plugin example', () async { testInMemory('is populated from Android application ID in plugin example', () async {
final FlutterProject project = await someProject(); final FlutterProject project = await someProject();
...@@ -398,12 +405,12 @@ apply plugin: 'kotlin-android' ...@@ -398,12 +405,12 @@ apply plugin: 'kotlin-android'
gradleFileContent: () { gradleFileContent: () {
return gradleFileWithApplicationId('io.flutter.someproject'); return gradleFileWithApplicationId('io.flutter.someproject');
}); });
expect(project.organizationNames, <String>['io.flutter']); expect(await project.organizationNames, <String>['io.flutter']);
}); });
testInMemory('is populated from Android group in plugin', () async { testInMemory('is populated from Android group in plugin', () async {
final FlutterProject project = await someProject(); final FlutterProject project = await someProject();
addAndroidWithGroup(project.directory, 'io.flutter.someproject'); addAndroidWithGroup(project.directory, 'io.flutter.someproject');
expect(project.organizationNames, <String>['io.flutter']); expect(await project.organizationNames, <String>['io.flutter']);
}); });
testInMemory('is singleton, if sources agree', () async { testInMemory('is singleton, if sources agree', () async {
final FlutterProject project = await someProject(); final FlutterProject project = await someProject();
...@@ -414,7 +421,7 @@ apply plugin: 'kotlin-android' ...@@ -414,7 +421,7 @@ apply plugin: 'kotlin-android'
gradleFileContent: () { gradleFileContent: () {
return gradleFileWithApplicationId('io.flutter.someproject'); return gradleFileWithApplicationId('io.flutter.someproject');
}); });
expect(project.organizationNames, <String>['io.flutter']); expect(await project.organizationNames, <String>['io.flutter']);
}); });
testInMemory('is non-singleton, if sources disagree', () async { testInMemory('is non-singleton, if sources disagree', () async {
final FlutterProject project = await someProject(); final FlutterProject project = await someProject();
...@@ -426,7 +433,7 @@ apply plugin: 'kotlin-android' ...@@ -426,7 +433,7 @@ apply plugin: 'kotlin-android'
return gradleFileWithApplicationId('io.clutter.someproject'); return gradleFileWithApplicationId('io.clutter.someproject');
}); });
expect( expect(
project.organizationNames, await project.organizationNames,
<String>['io.flutter', 'io.clutter'], <String>['io.flutter', 'io.clutter'],
); );
}); });
......
...@@ -333,12 +333,7 @@ class FakeXcodeProjectInterpreter implements XcodeProjectInterpreter { ...@@ -333,12 +333,7 @@ class FakeXcodeProjectInterpreter implements XcodeProjectInterpreter {
int get minorVersion => 2; int get minorVersion => 2;
@override @override
Map<String, String> getBuildSettings(String projectPath, String target) { Future<Map<String, String>> getBuildSettings(
return <String, String>{};
}
@override
Future<Map<String, String>> getBuildSettingsAsync(
String projectPath, String projectPath,
String target, { String target, {
Duration timeout = const Duration(minutes: 1), Duration timeout = const Duration(minutes: 1),
......
...@@ -26,7 +26,11 @@ import 'package:process/process.dart'; ...@@ -26,7 +26,11 @@ import 'package:process/process.dart';
import 'common.dart'; import 'common.dart';
final Generator kNoColorTerminalPlatform = () => FakePlatform.fromPlatform(const LocalPlatform())..stdoutSupportsAnsi = false; final Generator kNoColorTerminalPlatform = () {
return FakePlatform.fromPlatform(
const LocalPlatform()
)..stdoutSupportsAnsi = false;
};
class MockApplicationPackageStore extends ApplicationPackageStore { class MockApplicationPackageStore extends ApplicationPackageStore {
MockApplicationPackageStore() : super( MockApplicationPackageStore() : super(
...@@ -36,7 +40,7 @@ class MockApplicationPackageStore extends ApplicationPackageStore { ...@@ -36,7 +40,7 @@ class MockApplicationPackageStore extends ApplicationPackageStore {
versionCode: 1, versionCode: 1,
launchActivity: 'io.flutter.android.mock.MockActivity', launchActivity: 'io.flutter.android.mock.MockActivity',
), ),
iOS: BuildableIOSApp(MockIosProject()) iOS: BuildableIOSApp(MockIosProject(), MockIosProject.bundleId)
); );
} }
...@@ -513,8 +517,10 @@ class MockPollingDeviceDiscovery extends PollingDeviceDiscovery { ...@@ -513,8 +517,10 @@ class MockPollingDeviceDiscovery extends PollingDeviceDiscovery {
} }
class MockIosProject extends Mock implements IosProject { class MockIosProject extends Mock implements IosProject {
static const String bundleId = 'com.example.test';
@override @override
String get productBundleIdentifier => 'com.example.test'; Future<String> get productBundleIdentifier async => bundleId;
@override @override
String get hostAppBundleName => 'Runner.app'; String get hostAppBundleName => 'Runner.app';
......
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