Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Submit feedback
Sign in
Toggle navigation
F
Front-End
Project
Project
Details
Activity
Releases
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
abdullh.alsoleman
Front-End
Commits
a737c86a
Unverified
Commit
a737c86a
authored
Aug 10, 2018
by
Mikkel Nygaard Ravn
Committed by
GitHub
Aug 10, 2018
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Avoid null manifests in FlutterProject (#20332)
parent
142cbae7
Changes
5
Hide whitespace changes
Inline
Side-by-side
Showing
5 changed files
with
238 additions
and
153 deletions
+238
-153
gradle.dart
packages/flutter_tools/lib/src/android/gradle.dart
+2
-7
flutter_manifest.dart
packages/flutter_tools/lib/src/flutter_manifest.dart
+2
-8
xcodeproj.dart
packages/flutter_tools/lib/src/ios/xcodeproj.dart
+3
-5
project.dart
packages/flutter_tools/lib/src/project.dart
+31
-12
project_test.dart
packages/flutter_tools/test/project_test.dart
+200
-121
No files found.
packages/flutter_tools/lib/src/android/gradle.dart
View file @
a737c86a
...
@@ -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.'
);
}
}
...
...
packages/flutter_tools/lib/src/flutter_manifest.dart
View file @
a737c86a
...
@@ -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.
...
...
packages/flutter_tools/lib/src/ios/xcodeproj.dart
View file @
a737c86a
...
@@ -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
'
);
}
}
...
...
packages/flutter_tools/lib/src/project.dart
View file @
a737c86a
...
@@ -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
{
...
...
packages/flutter_tools/test/project_test.dart
View file @
a737c86a
...
@@ -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
);
expect
NotExists
(
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
=
a
wait
a
PluginProject
();
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
);
expect
Exists
(
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
);
expect
Exists
(
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
);
expect
Exists
(
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
=
a
wait
a
ModuleProject
();
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
=
a
wait
a
ModuleProject
();
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
=
a
wait
a
ModuleProject
();
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
=
a
wait
a
PluginProject
();
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() {
});
});
}
}
F
lutterProject
someProject
(
)
{
F
uture
<
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
(),
);
}
}
F
lutterProject
aPluginProject
(
)
{
F
uture
<
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
);
}
}
F
lutterProject
aModuleProject
(
)
{
F
uture
<
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'
);
}
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment