Unverified Commit a737c86a authored by Mikkel Nygaard Ravn's avatar Mikkel Nygaard Ravn Committed by GitHub

Avoid null manifests in FlutterProject (#20332)

parent 142cbae7
...@@ -203,18 +203,13 @@ distributionUrl=https\\://services.gradle.org/distributions/gradle-$gradleVersio ...@@ -203,18 +203,13 @@ distributionUrl=https\\://services.gradle.org/distributions/gradle-$gradleVersio
/// Overwrite local.properties in the specified Flutter project's Android /// Overwrite local.properties in the specified Flutter project's Android
/// sub-project, if needed. /// sub-project, if needed.
/// ///
/// Throws tool exit, if `pubspec.yaml` is invalid. /// If [requireAndroidSdk] is true (the default) and no Android SDK is found,
/// /// this will fail with a [ToolExit].
/// If [requireSdk] is `true` this will fail with a tool exit if no Android Sdk
/// is found.
Future<void> updateLocalProperties({ Future<void> updateLocalProperties({
@required FlutterProject project, @required FlutterProject project,
BuildInfo buildInfo, BuildInfo buildInfo,
bool requireAndroidSdk = true, bool requireAndroidSdk = true,
}) async { }) async {
if (project.manifest == null) {
throwToolExit('Invalid `pubspec.yaml`');
}
if (requireAndroidSdk && androidSdk == null) { if (requireAndroidSdk && androidSdk == null) {
throwToolExit('Unable to locate Android SDK. Please run `flutter doctor` for more details.'); throwToolExit('Unable to locate Android SDK. Please run `flutter doctor` for more details.');
} }
......
...@@ -28,14 +28,6 @@ class FlutterManifest { ...@@ -28,14 +28,6 @@ class FlutterManifest {
return manifest; return manifest;
} }
/// Returns a mock manifest with the given contents.
static FlutterManifest mock(Map<String, dynamic> contents) {
final FlutterManifest manifest = new FlutterManifest._();
manifest._descriptor = contents ?? const <String, dynamic>{};
manifest._flutterDescriptor = manifest._descriptor['flutter'] ?? const <String, dynamic>{};
return manifest;
}
/// Returns null on invalid manifest. Returns empty manifest on missing file. /// Returns null on invalid manifest. Returns empty manifest on missing file.
static Future<FlutterManifest> createFromPath(String path) async { static Future<FlutterManifest> createFromPath(String path) async {
if (path == null || !fs.isFileSync(path)) if (path == null || !fs.isFileSync(path))
...@@ -78,8 +70,10 @@ class FlutterManifest { ...@@ -78,8 +70,10 @@ class FlutterManifest {
/// A map representation of the `flutter` section in the `pubspec.yaml` file. /// A map representation of the `flutter` section in the `pubspec.yaml` file.
Map<String, dynamic> _flutterDescriptor; Map<String, dynamic> _flutterDescriptor;
/// True if the `pubspec.yaml` file does not exist.
bool get isEmpty => _descriptor.isEmpty; bool get isEmpty => _descriptor.isEmpty;
/// The string value of the top-level `name` property in the `pubspec.yaml` file.
String get appName => _descriptor['name'] ?? ''; String get appName => _descriptor['name'] ?? '';
/// The version String from the `pubspec.yaml` file. /// The version String from the `pubspec.yaml` file.
......
...@@ -16,7 +16,6 @@ import '../base/process_manager.dart'; ...@@ -16,7 +16,6 @@ import '../base/process_manager.dart';
import '../base/utils.dart'; import '../base/utils.dart';
import '../build_info.dart'; import '../build_info.dart';
import '../cache.dart'; import '../cache.dart';
import '../flutter_manifest.dart';
import '../globals.dart'; import '../globals.dart';
import '../project.dart'; import '../project.dart';
...@@ -59,20 +58,19 @@ Future<void> updateGeneratedXcodeProperties({ ...@@ -59,20 +58,19 @@ Future<void> updateGeneratedXcodeProperties({
localsBuffer.writeln('SYMROOT=\${SOURCE_ROOT}/../${getIosBuildDirectory()}'); localsBuffer.writeln('SYMROOT=\${SOURCE_ROOT}/../${getIosBuildDirectory()}');
final FlutterManifest manifest = project.manifest; if (!project.isModule) {
if (!manifest.isModule) {
// For module projects we do not want to write the FLUTTER_FRAMEWORK_DIR // For module projects we do not want to write the FLUTTER_FRAMEWORK_DIR
// explicitly. Rather we rely on the xcode backend script and the Podfile // explicitly. Rather we rely on the xcode backend script and the Podfile
// logic to derive it from FLUTTER_ROOT and FLUTTER_BUILD_MODE. // logic to derive it from FLUTTER_ROOT and FLUTTER_BUILD_MODE.
localsBuffer.writeln('FLUTTER_FRAMEWORK_DIR=${flutterFrameworkDir(buildInfo.mode)}'); localsBuffer.writeln('FLUTTER_FRAMEWORK_DIR=${flutterFrameworkDir(buildInfo.mode)}');
} }
final String buildName = buildInfo?.buildName ?? manifest.buildName; final String buildName = buildInfo?.buildName ?? project.manifest.buildName;
if (buildName != null) { if (buildName != null) {
localsBuffer.writeln('FLUTTER_BUILD_NAME=$buildName'); localsBuffer.writeln('FLUTTER_BUILD_NAME=$buildName');
} }
final int buildNumber = buildInfo?.buildNumber ?? manifest.buildNumber; final int buildNumber = buildInfo?.buildNumber ?? project.manifest.buildNumber;
if (buildNumber != null) { if (buildNumber != null) {
localsBuffer.writeln('FLUTTER_BUILD_NUMBER=$buildNumber'); localsBuffer.writeln('FLUTTER_BUILD_NUMBER=$buildNumber');
} }
......
...@@ -8,6 +8,7 @@ import 'dart:convert'; ...@@ -8,6 +8,7 @@ import 'dart:convert';
import 'package:meta/meta.dart'; import 'package:meta/meta.dart';
import 'android/gradle.dart' as gradle; import 'android/gradle.dart' as gradle;
import 'base/common.dart';
import 'base/file_system.dart'; import 'base/file_system.dart';
import 'build_info.dart'; import 'build_info.dart';
import 'bundle.dart' as bundle; import 'bundle.dart' as bundle;
...@@ -28,34 +29,39 @@ import 'template.dart'; ...@@ -28,34 +29,39 @@ import 'template.dart';
/// cached. /// cached.
class FlutterProject { class FlutterProject {
@visibleForTesting @visibleForTesting
FlutterProject(this.directory, this.manifest, this._exampleManifest); FlutterProject(this.directory, this.manifest, this._exampleManifest)
: assert(directory != null),
assert(manifest != null),
assert(_exampleManifest != null);
/// Returns a future that completes with a FlutterProject view of the given directory. /// Returns a future that completes with a [FlutterProject] view of the given directory
/// or a ToolExit error, if `pubspec.yaml` or `example/pubspec.yaml` is invalid.
static Future<FlutterProject> fromDirectory(Directory directory) async { static Future<FlutterProject> fromDirectory(Directory directory) async {
final FlutterManifest manifest = await FlutterManifest.createFromPath( assert(directory != null);
final FlutterManifest manifest = await _readManifest(
directory.childFile(bundle.defaultManifestPath).path, directory.childFile(bundle.defaultManifestPath).path,
); );
final Directory exampleDirectory = _exampleDirectory(directory); final FlutterManifest exampleManifest = await _readManifest(
final FlutterManifest exampleManifest = await FlutterManifest.createFromPath( _exampleDirectory(directory).childFile(bundle.defaultManifestPath).path,
exampleDirectory.childFile(bundle.defaultManifestPath).path,
); );
return new FlutterProject(directory, manifest, exampleManifest); return new FlutterProject(directory, manifest, exampleManifest);
} }
/// Returns a future that completes with a FlutterProject view of the current directory. /// Returns a future that completes with a [FlutterProject] view of the current directory.
/// or a ToolExit error, if `pubspec.yaml` or `example/pubspec.yaml` is invalid.
static Future<FlutterProject> current() => fromDirectory(fs.currentDirectory); static Future<FlutterProject> current() => fromDirectory(fs.currentDirectory);
/// Returns a future that completes with a FlutterProject view of the given directory. /// Returns a future that completes with a [FlutterProject] view of the given directory.
/// or a ToolExit error, if `pubspec.yaml` or `example/pubspec.yaml` is invalid.
static Future<FlutterProject> fromPath(String path) => fromDirectory(fs.directory(path)); static Future<FlutterProject> fromPath(String path) => fromDirectory(fs.directory(path));
/// The location of this project. /// The location of this project.
final Directory directory; final Directory directory;
/// The manifest of this project, or null, if `pubspec.yaml` is invalid. /// The manifest of this project.
final FlutterManifest manifest; final FlutterManifest manifest;
/// The manifest of the example sub-project of this project, or null, if /// The manifest of the example sub-project of this project.
/// `example/pubspec.yaml` is invalid.
final FlutterManifest _exampleManifest; final FlutterManifest _exampleManifest;
/// Asynchronously returns the organization names found in this project as /// Asynchronously returns the organization names found in this project as
...@@ -96,7 +102,8 @@ class FlutterProject { ...@@ -96,7 +102,8 @@ class FlutterProject {
FlutterManifest.empty(), FlutterManifest.empty(),
); );
bool get isModule => manifest != null && manifest.isModule; /// True, if this project is a Flutter module.
bool get isModule => manifest.isModule;
/// True, if this project has an example application. /// True, if this project has an example application.
bool get hasExampleApp => _exampleDirectory(directory).existsSync(); bool get hasExampleApp => _exampleDirectory(directory).existsSync();
...@@ -104,6 +111,18 @@ class FlutterProject { ...@@ -104,6 +111,18 @@ class FlutterProject {
/// The directory that will contain the example if an example exists. /// The directory that will contain the example if an example exists.
static Directory _exampleDirectory(Directory directory) => directory.childDirectory('example'); static Directory _exampleDirectory(Directory directory) => directory.childDirectory('example');
/// Reads and validates the `pubspec.yaml` file at [path], asynchronously
/// returning a [FlutterManifest] representation of the contents.
///
/// Completes with an empty [FlutterManifest], if the file does not exist.
/// Completes with a ToolExit on validation error.
static Future<FlutterManifest> _readManifest(String path) async {
final FlutterManifest manifest = await FlutterManifest.createFromPath(path);
if (manifest == null)
throwToolExit('Please correct the pubspec.yaml file at $path');
return manifest;
}
/// Generates project files necessary to make Gradle builds work on Android /// Generates project files necessary to make Gradle builds work on Android
/// and CocoaPods+Xcode work on iOS, for app and module projects only. /// and CocoaPods+Xcode work on iOS, for app and module projects only.
Future<void> ensureReadyForPlatformSpecificTooling() async { Future<void> ensureReadyForPlatformSpecificTooling() async {
......
...@@ -3,7 +3,9 @@ ...@@ -3,7 +3,9 @@
// found in the LICENSE file. // found in the LICENSE file.
import 'dart:async'; import 'dart:async';
import 'package:flutter_tools/src/base/common.dart';
import 'package:flutter_tools/src/base/context.dart'; import 'package:flutter_tools/src/base/context.dart';
import 'package:flutter_tools/src/base/platform.dart';
import 'package:flutter_tools/src/cache.dart'; import 'package:flutter_tools/src/cache.dart';
import 'package:flutter_tools/src/flutter_manifest.dart'; import 'package:flutter_tools/src/flutter_manifest.dart';
import 'package:flutter_tools/src/project.dart'; import 'package:flutter_tools/src/project.dart';
...@@ -11,27 +13,97 @@ import 'package:flutter_tools/src/base/file_system.dart'; ...@@ -11,27 +13,97 @@ import 'package:flutter_tools/src/base/file_system.dart';
import 'package:test/test.dart'; import 'package:test/test.dart';
import 'package:file/file.dart'; import 'package:file/file.dart';
import 'package:file/memory.dart'; import 'package:file/memory.dart';
import 'src/common.dart';
import 'src/context.dart'; import 'src/context.dart';
void main() { void main() {
group('Project', () { group('Project', () {
testInMemory('knows location', () async { group('construction', () {
final Directory directory = fs.directory('myproject'); testInMemory('fails on null directory', () async {
expect( await expectLater(
(await FlutterProject.fromDirectory(directory)).directory.absolute.path, FlutterProject.fromDirectory(null),
directory.absolute.path, throwsA(const isInstanceOf<AssertionError>()),
); );
expect( });
(await FlutterProject.fromPath(directory.path)).directory.absolute.path,
directory.absolute.path, Future<Null> expectToolExitLater(Future<dynamic> future, Matcher messageMatcher) async {
); try {
expect( await future;
(await FlutterProject.current()).directory.absolute.path, fail('ToolExit expected, but nothing thrown');
fs.currentDirectory.absolute.path, } on ToolExit catch(e) {
); expect(e.message, messageMatcher);
} catch(e, trace) {
fail('ToolExit expected, got $e\n$trace');
}
}
testInMemory('fails on invalid pubspec.yaml', () async {
final Directory directory = fs.directory('myproject');
directory.childFile('pubspec.yaml')
..createSync(recursive: true)
..writeAsStringSync(invalidPubspec);
await expectToolExitLater(
FlutterProject.fromDirectory(directory),
contains('pubspec.yaml'),
);
});
testInMemory('fails on invalid example/pubspec.yaml', () async {
final Directory directory = fs.directory('myproject');
directory.childDirectory('example').childFile('pubspec.yaml')
..createSync(recursive: true)
..writeAsStringSync(invalidPubspec);
await expectToolExitLater(
FlutterProject.fromDirectory(directory),
contains('pubspec.yaml'),
);
});
testInMemory('treats missing pubspec.yaml as empty', () async {
final Directory directory = fs.directory('myproject')
..createSync(recursive: true);
expect(
(await FlutterProject.fromDirectory(directory)).manifest.isEmpty,
true,
);
});
testInMemory('reads valid pubspec.yaml', () async {
final Directory directory = fs.directory('myproject');
directory.childFile('pubspec.yaml')
..createSync(recursive: true)
..writeAsStringSync(validPubspec);
expect(
(await FlutterProject.fromDirectory(directory)).manifest.appName,
'hello',
);
});
testInMemory('sets up location', () async {
final Directory directory = fs.directory('myproject');
expect(
(await FlutterProject.fromDirectory(directory)).directory.absolute.path,
directory.absolute.path,
);
expect(
(await FlutterProject.fromPath(directory.path)).directory.absolute.path,
directory.absolute.path,
);
expect(
(await FlutterProject.current()).directory.absolute.path,
fs.currentDirectory.absolute.path,
);
});
}); });
group('ensure ready for platform-specific tooling', () { group('ensure ready for platform-specific tooling', () {
void expectExists(FileSystemEntity entity) {
expect(entity.existsSync(), isTrue);
}
void expectNotExists(FileSystemEntity entity) {
expect(entity.existsSync(), isFalse);
}
testInMemory('does nothing, if project is not created', () async { testInMemory('does nothing, if project is not created', () async {
final FlutterProject project = new FlutterProject( final FlutterProject project = new FlutterProject(
fs.directory('not_created'), fs.directory('not_created'),
...@@ -39,137 +111,130 @@ void main() { ...@@ -39,137 +111,130 @@ void main() {
FlutterManifest.empty(), FlutterManifest.empty(),
); );
await project.ensureReadyForPlatformSpecificTooling(); await project.ensureReadyForPlatformSpecificTooling();
expect(project.directory.existsSync(), isFalse); expectNotExists(project.directory);
}); });
testInMemory('does nothing in plugin or package root project', () async { testInMemory('does nothing in plugin or package root project', () async {
final FlutterProject project = aPluginProject(); final FlutterProject project = await aPluginProject();
await project.ensureReadyForPlatformSpecificTooling(); await project.ensureReadyForPlatformSpecificTooling();
expect(project.ios.directory.childFile('Runner/GeneratedPluginRegistrant.h').existsSync(), isFalse); expectNotExists(project.ios.directory.childDirectory('Runner').childFile('GeneratedPluginRegistrant.h'));
expect(project.android.directory.childFile( expectNotExists(androidPluginRegistrant(project.android.directory.childDirectory('app')));
'app/src/main/java/io/flutter/plugins/GeneratedPluginRegistrant.java', expectNotExists(project.ios.directory.childDirectory('Flutter').childFile('Generated.xcconfig'));
).existsSync(), isFalse); expectNotExists(project.android.directory.childFile('local.properties'));
expect(project.ios.directory.childFile('Flutter/Generated.xcconfig').existsSync(), isFalse);
expect(project.android.directory.childFile('local.properties').existsSync(), isFalse);
}); });
testInMemory('injects plugins for iOS', () async { testInMemory('injects plugins for iOS', () async {
final FlutterProject project = someProject(); final FlutterProject project = await someProject();
await project.ensureReadyForPlatformSpecificTooling(); await project.ensureReadyForPlatformSpecificTooling();
expect(project.ios.directory.childFile('Runner/GeneratedPluginRegistrant.h').existsSync(), isTrue); expectExists(project.ios.directory.childDirectory('Runner').childFile('GeneratedPluginRegistrant.h'));
}); });
testInMemory('generates Xcode configuration for iOS', () async { testInMemory('generates Xcode configuration for iOS', () async {
final FlutterProject project = someProject(); final FlutterProject project = await someProject();
await project.ensureReadyForPlatformSpecificTooling(); await project.ensureReadyForPlatformSpecificTooling();
expect(project.ios.directory.childFile('Flutter/Generated.xcconfig').existsSync(), isTrue); expectExists(project.ios.directory.childDirectory('Flutter').childFile('Generated.xcconfig'));
}); });
testInMemory('injects plugins for Android', () async { testInMemory('injects plugins for Android', () async {
final FlutterProject project = someProject(); final FlutterProject project = await someProject();
await project.ensureReadyForPlatformSpecificTooling(); await project.ensureReadyForPlatformSpecificTooling();
expect(project.android.directory.childFile( expectExists(androidPluginRegistrant(project.android.directory.childDirectory('app')));
'app/src/main/java/io/flutter/plugins/GeneratedPluginRegistrant.java',
).existsSync(), isTrue);
}); });
testInMemory('updates local properties for Android', () async { testInMemory('updates local properties for Android', () async {
final FlutterProject project = someProject(); final FlutterProject project = await someProject();
await project.ensureReadyForPlatformSpecificTooling(); await project.ensureReadyForPlatformSpecificTooling();
expect(project.android.directory.childFile('local.properties').existsSync(), isTrue); expectExists(project.android.directory.childFile('local.properties'));
}); });
testInMemory('creates Android library in module', () async { testInMemory('creates Android library in module', () async {
final FlutterProject project = aModuleProject(); final FlutterProject project = await aModuleProject();
await project.ensureReadyForPlatformSpecificTooling(); await project.ensureReadyForPlatformSpecificTooling();
expect(project.android.directory.childFile('template_content').existsSync(), isTrue); expectExists(project.android.directory.childFile('settings.gradle'));
expect(project.android.directory.childFile('local.properties').existsSync(), isTrue); expectExists(project.android.directory.childFile('local.properties'));
expect(project.android.directory.childFile( expectExists(androidPluginRegistrant(project.android.directory.childDirectory('Flutter')));
'Flutter/src/main/java/io/flutter/plugins/GeneratedPluginRegistrant.java',
).existsSync(), isTrue);
}); });
testInMemory('creates iOS pod in module', () async { testInMemory('creates iOS pod in module', () async {
final FlutterProject project = aModuleProject(); final FlutterProject project = await aModuleProject();
await project.ensureReadyForPlatformSpecificTooling(); await project.ensureReadyForPlatformSpecificTooling();
final Directory flutter = project.ios.directory.childDirectory('Flutter'); final Directory flutter = project.ios.directory.childDirectory('Flutter');
expect(flutter.childFile('template_content').existsSync(), isTrue); expectExists(flutter.childFile('podhelper.rb'));
expect(flutter.childFile('Generated.xcconfig').existsSync(), isTrue); expectExists(flutter.childFile('Generated.xcconfig'));
expect(flutter.childFile( final Directory pluginRegistrantClasses = flutter
'FlutterPluginRegistrant/Classes/GeneratedPluginRegistrant.h', .childDirectory('FlutterPluginRegistrant')
).existsSync(), isTrue); .childDirectory('Classes');
expect(flutter.childFile( expectExists(pluginRegistrantClasses.childFile('GeneratedPluginRegistrant.h'));
'FlutterPluginRegistrant/Classes/GeneratedPluginRegistrant.m', expectExists(pluginRegistrantClasses.childFile('GeneratedPluginRegistrant.m'));
).existsSync(), isTrue);
}); });
}); });
group('module status', () { group('module status', () {
testInMemory('is known for module', () async { testInMemory('is known for module', () async {
final FlutterProject project = aModuleProject(); final FlutterProject project = await aModuleProject();
expect(project.isModule, isTrue); expect(project.isModule, isTrue);
expect(project.android.isModule, isTrue); expect(project.android.isModule, isTrue);
expect(project.ios.isModule, isTrue); expect(project.ios.isModule, isTrue);
expect(project.android.directory.path, startsWith('module_project/.android')); expect(project.android.directory.basename, '.android');
expect(project.ios.directory.path, startsWith('module_project/.ios')); expect(project.ios.directory.basename, '.ios');
}); });
testInMemory('is known for non-module', () async { testInMemory('is known for non-module', () async {
final FlutterProject project = someProject(); final FlutterProject project = await someProject();
expect(project.isModule, isFalse); expect(project.isModule, isFalse);
expect(project.android.isModule, isFalse); expect(project.android.isModule, isFalse);
expect(project.ios.isModule, isFalse); expect(project.ios.isModule, isFalse);
expect(project.android.directory.path, startsWith('some_project/android')); expect(project.android.directory.basename, 'android');
expect(project.ios.directory.path, startsWith('some_project/ios')); expect(project.ios.directory.basename, 'ios');
}); });
}); });
group('example', () { group('example', () {
testInMemory('exists for plugin', () async { testInMemory('exists for plugin', () async {
final FlutterProject project = aPluginProject(); final FlutterProject project = await aPluginProject();
expect(project.hasExampleApp, isTrue); expect(project.hasExampleApp, isTrue);
}); });
testInMemory('does not exist for non-plugin', () async { testInMemory('does not exist for non-plugin', () async {
final FlutterProject project = someProject(); final FlutterProject project = await someProject();
expect(project.hasExampleApp, isFalse); expect(project.hasExampleApp, isFalse);
}); });
}); });
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 = someProject(); final FlutterProject project = await someProject();
expect(await 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 = someProject(); final FlutterProject project = await someProject();
project.directory.createSync(); project.directory.createSync();
expect(await 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 = someProject(); final FlutterProject project = await someProject();
addIosWithBundleId(project.directory, 'io.flutter.someProject'); addIosWithBundleId(project.directory, 'io.flutter.someProject');
expect(await 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 = someProject(); final FlutterProject project = await someProject();
addAndroidWithApplicationId(project.directory, 'io.flutter.someproject'); addAndroidWithApplicationId(project.directory, 'io.flutter.someproject');
expect(await 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 = someProject(); final FlutterProject project = await someProject();
addIosWithBundleId(project.example.directory, 'io.flutter.someProject'); addIosWithBundleId(project.example.directory, 'io.flutter.someProject');
expect(await 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 = someProject(); final FlutterProject project = await someProject();
addAndroidWithApplicationId(project.example.directory, 'io.flutter.someproject'); addAndroidWithApplicationId(project.example.directory, 'io.flutter.someproject');
expect(await 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 = someProject(); final FlutterProject project = await someProject();
addAndroidWithGroup(project.directory, 'io.flutter.someproject'); addAndroidWithGroup(project.directory, 'io.flutter.someproject');
expect(await 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 = someProject(); final FlutterProject project = await someProject();
addIosWithBundleId(project.directory, 'io.flutter.someProject'); addIosWithBundleId(project.directory, 'io.flutter.someProject');
addAndroidWithApplicationId(project.directory, 'io.flutter.someproject'); addAndroidWithApplicationId(project.directory, 'io.flutter.someproject');
expect(await 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 = someProject(); final FlutterProject project = await someProject();
addIosWithBundleId(project.directory, 'io.flutter.someProject'); addIosWithBundleId(project.directory, 'io.flutter.someProject');
addAndroidWithApplicationId(project.directory, 'io.clutter.someproject'); addAndroidWithApplicationId(project.directory, 'io.clutter.someproject');
expect( expect(
...@@ -181,92 +246,85 @@ void main() { ...@@ -181,92 +246,85 @@ void main() {
}); });
} }
FlutterProject someProject() { Future<FlutterProject> someProject() async {
final Directory directory = fs.directory('some_project'); final Directory directory = fs.directory('some_project');
directory.childFile('.packages').createSync(recursive: true); directory.childFile('.packages').createSync(recursive: true);
directory.childDirectory('ios').createSync(recursive: true); directory.childDirectory('ios').createSync(recursive: true);
directory.childDirectory('android').createSync(recursive: true); directory.childDirectory('android').createSync(recursive: true);
return new FlutterProject( return FlutterProject.fromDirectory(directory);
directory,
FlutterManifest.empty(),
FlutterManifest.empty(),
);
} }
FlutterProject aPluginProject() { Future<FlutterProject> aPluginProject() async {
final Directory directory = fs.directory('plugin_project'); final Directory directory = fs.directory('plugin_project');
directory.childDirectory('ios').createSync(recursive: true); directory.childDirectory('ios').createSync(recursive: true);
directory.childDirectory('android').createSync(recursive: true); directory.childDirectory('android').createSync(recursive: true);
directory.childDirectory('example').createSync(recursive: true); directory.childDirectory('example').createSync(recursive: true);
return new FlutterProject( directory.childFile('pubspec.yaml').writeAsStringSync('''
directory, name: my_plugin
FlutterManifest.mock(const <String, dynamic>{ flutter:
'flutter': <String, dynamic>{ plugin:
'plugin': <String, dynamic>{} androidPackage: com.example
} pluginClass: MyPlugin
}), iosPrefix: FLT
FlutterManifest.empty(), ''');
); return FlutterProject.fromDirectory(directory);
} }
FlutterProject aModuleProject() { Future<FlutterProject> aModuleProject() async {
final Directory directory = fs.directory('module_project'); final Directory directory = fs.directory('module_project');
directory.childFile('.packages').createSync(recursive: true); directory.childFile('.packages').createSync(recursive: true);
return new FlutterProject( directory.childFile('pubspec.yaml').writeAsStringSync('''
directory, name: my_module
FlutterManifest.mock(const <String, dynamic>{ flutter:
'flutter': <String, dynamic>{ module:
'module': <String, dynamic>{ androidPackage: com.example
'androidPackage': 'com.example' ''');
} return FlutterProject.fromDirectory(directory);
}
}),
FlutterManifest.empty(),
);
} }
/// Executes the [testMethod] in a context where the file system
/// is in memory.
void testInMemory(String description, Future<Null> testMethod()) { void testInMemory(String description, Future<Null> testMethod()) {
Cache.flutterRoot = 'flutter'; Cache.flutterRoot = getFlutterRoot();
final FileSystem fs = new MemoryFileSystem(); final FileSystem testFileSystem = new MemoryFileSystem(
// Pretend we have a Flutter module project template. style: platform.isWindows ? FileSystemStyle.windows : FileSystemStyle.posix,
fs.directory(Cache.flutterRoot) );
// Transfer needed parts of the Flutter installation folder
// to the in-memory file system used during testing.
transfer(new Cache().getArtifactDirectory('gradle_wrapper'), testFileSystem);
transfer(fs.directory(Cache.flutterRoot)
.childDirectory('packages') .childDirectory('packages')
.childDirectory('flutter_tools') .childDirectory('flutter_tools')
.childDirectory('templates') .childDirectory('templates'), testFileSystem);
.childDirectory('module') transfer(fs.directory(Cache.flutterRoot)
.childDirectory('android')
.childFile('template_content.copy.tmpl')
.createSync(recursive: true);
fs.directory(Cache.flutterRoot)
.childDirectory('packages') .childDirectory('packages')
.childDirectory('flutter_tools') .childDirectory('flutter_tools')
.childDirectory('templates') .childDirectory('schema'), testFileSystem);
.childDirectory('module')
.childDirectory('ios')
.childDirectory('Flutter.tmpl')
.childFile('template_content.copy.tmpl')
.createSync(recursive: true);
// Sets up cache in a test execution context where `fs` is the file system.
Cache cacheCreator() {
final Cache cache = new Cache(rootOverride: fs.directory('flutter'));
cache.getArtifactDirectory('gradle_wrapper')
.childDirectory('gradle')
.childDirectory('wrapper')
.childFile('gradle-wrapper.properties')
.createSync(recursive: true);
return cache;
}
testUsingContext( testUsingContext(
description, description,
testMethod, testMethod,
overrides: <Type, Generator>{ overrides: <Type, Generator>{
FileSystem: () => fs, FileSystem: () => testFileSystem,
Cache: cacheCreator, Cache: () => new Cache(),
}, },
); );
} }
/// Transfers files and folders from the local file system's Flutter
/// installation to an (in-memory) file system used for testing.
void transfer(FileSystemEntity entity, FileSystem target) {
if (entity is Directory) {
target.directory(entity.absolute.path).createSync(recursive: true);
for (FileSystemEntity child in entity.listSync()) {
transfer(child, target);
}
} else if (entity is File) {
target.file(entity.absolute.path).writeAsBytesSync(entity.readAsBytesSync(), flush: true);
} else {
throw 'Unsupported FileSystemEntity ${entity.runtimeType}';
}
}
void addIosWithBundleId(Directory directory, String id) { void addIosWithBundleId(Directory directory, String id) {
directory directory
.childDirectory('ios') .childDirectory('ios')
...@@ -291,6 +349,17 @@ void addAndroidWithGroup(Directory directory, String id) { ...@@ -291,6 +349,17 @@ void addAndroidWithGroup(Directory directory, String id) {
..writeAsStringSync(gradleFileWithGroupId(id)); ..writeAsStringSync(gradleFileWithGroupId(id));
} }
String get validPubspec => '''
name: hello
flutter:
''';
String get invalidPubspec => '''
name: hello
flutter:
invalid:
''';
String projectFileWithBundleId(String id) { String projectFileWithBundleId(String id) {
return ''' return '''
97C147061CF9000F007C117D /* Debug */ = { 97C147061CF9000F007C117D /* Debug */ = {
...@@ -330,3 +399,13 @@ android { ...@@ -330,3 +399,13 @@ android {
} }
'''; ''';
} }
File androidPluginRegistrant(Directory parent) {
return parent.childDirectory('src')
.childDirectory('main')
.childDirectory('java')
.childDirectory('io')
.childDirectory('flutter')
.childDirectory('plugins')
.childFile('GeneratedPluginRegistrant.java');
}
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