2017-01-31 22:48:48 +00:00
import java.nio.file.Path
import java.nio.file.Paths
2016-03-12 00:30:56 +00:00
import com.android.builder.model.AndroidProject
2019-06-09 01:02:32 +00:00
import com.android.build.OutputFile
2017-02-16 22:17:09 +00:00
import org.apache.tools.ant.taskdefs.condition.Os
2016-03-12 00:30:56 +00:00
import org.gradle.api.DefaultTask
import org.gradle.api.GradleException
import org.gradle.api.Project
import org.gradle.api.Plugin
import org.gradle.api.Task
2016-06-06 19:56:04 +00:00
import org.gradle.api.file.CopySpec
2016-03-12 00:30:56 +00:00
import org.gradle.api.file.FileCollection
import org.gradle.api.tasks.Copy
2017-01-31 22:48:48 +00:00
import org.gradle.api.tasks.InputFiles
2016-03-12 00:30:56 +00:00
import org.gradle.api.tasks.OutputDirectory
import org.gradle.api.tasks.TaskAction
2017-01-31 22:48:48 +00:00
import org.gradle.api.tasks.bundling.Jar
2016-03-12 00:30:56 +00:00
2017-02-10 08:37:38 +00:00
buildscript {
2017-07-31 11:57:24 +00:00
repositories {
2018-10-24 21:33:39 +00:00
google ( )
2017-12-14 14:38:39 +00:00
jcenter ( )
2017-07-31 11:57:24 +00:00
}
dependencies {
2019-01-24 22:29:08 +00:00
classpath 'com.android.tools.build:gradle:3.2.1'
2017-07-31 11:57:24 +00:00
}
2017-02-10 08:37:38 +00:00
}
2018-12-17 22:40:48 +00:00
android {
compileOptions {
sourceCompatibility 1.8
targetCompatibility 1.8
}
}
2017-02-10 08:37:38 +00:00
apply plugin: FlutterPlugin
2016-03-12 00:30:56 +00:00
class FlutterPlugin implements Plugin < Project > {
2019-06-09 01:02:32 +00:00
// The platforms that can be passed to the `--Ptarget-platform` flag.
private static final String PLATFORM_ARM32 = "android-arm" ;
private static final String PLATFORM_ARM64 = "android-arm64" ;
private static final String PLATFORM_X86 = "android-x86" ;
private static final String PLATFORM_X86_64 = "android-x64" ;
// The ABI architectures.
private static final String ARCH_ARM32 = "armeabi-v7a" ;
private static final String ARCH_ARM64 = "arm64-v8a" ;
private static final String ARCH_X86 = "x86" ;
private static final String ARCH_X86_64 = "x86_64" ;
// Maps platforms to ABI architectures.
private static final Map PLATFORM_ARCH_MAP = [
( PLATFORM_ARM32 ) : ARCH_ARM32 ,
( PLATFORM_ARM64 ) : ARCH_ARM64 ,
( PLATFORM_X86 ) : ARCH_X86 ,
( PLATFORM_X86_64 ) : ARCH_X86_64 ,
]
// The version code that gives each ABI a value.
// For each APK variant, use the following versions to override the version of the Universal APK.
// Otherwise, the Play Store will complain that the APK variants have the same version.
private static final Map ABI_VERSION = [
( ARCH_ARM32 ) : 1 ,
( ARCH_ARM64 ) : 2 ,
( ARCH_X86 ) : 3 ,
( ARCH_X86_64 ) : 4 ,
]
// When split is enabled, multiple APKs are generated per each ABI.
private static final List DEFAULT_PLATFORMS = [
PLATFORM_ARM32 ,
PLATFORM_ARM64 ,
]
// The name prefix for flutter builds. This is used to identify gradle tasks
// where we expect the flutter tool to provide any error output, and skip the
// standard Gradle error output in the FlutterEventLogger. If you change this,
// be sure to change any instances of this string in symbols in the code below
// to match.
static final String FLUTTER_BUILD_PREFIX = "flutterBuild"
2019-05-20 20:09:20 +00:00
private Path baseEnginePath
2016-06-06 19:56:04 +00:00
private File flutterRoot
2017-02-16 22:17:09 +00:00
private File flutterExecutable
2016-05-28 00:12:52 +00:00
private String localEngine
2017-02-23 13:56:19 +00:00
private String localEngineSrcPath
2017-01-31 22:48:48 +00:00
private Properties localProperties
2017-03-23 13:59:12 +00:00
private File flutterJar
private File debugFlutterJar
private File profileFlutterJar
private File releaseFlutterJar
2018-08-16 15:43:41 +00:00
private File dynamicProfileFlutterJar
private File dynamicReleaseFlutterJar
2017-03-23 13:59:12 +00:00
private Properties readPropertiesIfExist ( File propertiesFile ) {
Properties result = new Properties ( )
if ( propertiesFile . exists ( ) ) {
2018-01-08 21:48:08 +00:00
propertiesFile . withReader ( 'UTF-8' ) { reader - > result . load ( reader ) }
2017-03-23 13:59:12 +00:00
}
return result
}
2019-06-09 01:02:32 +00:00
private List < String > getTargetPlatforms ( Project project ) {
if ( ! project . hasProperty ( 'target-platform' ) ) {
return DEFAULT_PLATFORMS
}
return project . property ( 'target-platform' ) . split ( ',' ) . collect {
if ( ! PLATFORM_ARCH_MAP [ it ] ) {
throw new GradleException ( "Invalid platform: $it." )
}
return it
2019-05-20 20:09:20 +00:00
}
}
private Boolean getBuildShareLibrary ( Project project ) {
if ( project . hasProperty ( 'build-shared-library' ) ) {
return project . property ( 'build-shared-library' ) . toBoolean ( )
}
return false ;
}
2019-06-09 01:02:32 +00:00
private Boolean splitPerAbi ( Project project ) {
if ( project . hasProperty ( 'split-per-abi' ) ) {
return project . property ( 'split-per-abi' ) . toBoolean ( )
}
return false ;
}
2017-01-31 22:48:48 +00:00
private String resolveProperty ( Project project , String name , String defaultValue ) {
if ( localProperties = = null ) {
2018-06-20 14:07:05 +00:00
localProperties = readPropertiesIfExist ( new File ( project . projectDir . parentFile , "local.properties" ) )
2017-01-31 22:48:48 +00:00
}
2017-02-23 13:56:19 +00:00
String result
2017-01-31 22:48:48 +00:00
if ( project . hasProperty ( name ) ) {
result = project . property ( name )
}
if ( result = = null ) {
result = localProperties . getProperty ( name )
}
if ( result = = null ) {
result = defaultValue
}
return result
}
2016-03-12 00:30:56 +00:00
2019-06-09 01:02:32 +00:00
/ * *
* Returns the platform that is used to extract the ` libflutter . so ` and the . class files .
*
* Note: This is only needed to add the . class files .
* Unfortunately , the engine artifacts include the . class and libflutter . so files .
* /
private String getBasePlatform ( Project project ) {
if ( PLATFORM_ARM64 in getTargetPlatforms ( project ) ) {
return PLATFORM_ARM64 ;
}
return PLATFORM_ARM32 ;
}
2016-03-12 00:30:56 +00:00
@Override
void apply ( Project project ) {
2019-05-20 20:09:20 +00:00
project . extensions . create ( "flutter" , FlutterExtension )
project . afterEvaluate this . & addFlutterTask
2019-06-09 01:02:32 +00:00
// By default, assembling APKs generates fat APKs if multiple platforms are passed.
// Configuring split per ABI allows to generate separate APKs for each abi.
// This is a noop when building a bundle.
if ( this . splitPerAbi ( project ) ) {
project . android {
splits {
abi {
// Enables building multiple APKs per ABI.
enable true
// Resets the list of ABIs that Gradle should create APKs for to none.
reset ( )
// Specifies that we do not want to also generate a universal APK that includes all ABIs.
universalApk false
}
}
}
}
this . getTargetPlatforms ( project ) . each { targetArch - >
String abiValue = PLATFORM_ARCH_MAP [ targetArch ]
2019-05-20 20:09:20 +00:00
project . android {
packagingOptions {
2019-06-09 01:02:32 +00:00
pickFirst "lib/${abiValue}/libflutter.so"
// Prevent the ELF library from getting corrupted.
doNotStrip "*/${abiValue}/libapp.so"
}
if ( this . splitPerAbi ( project ) ) {
splits {
abi {
include abiValue
}
}
2019-05-20 20:09:20 +00:00
}
}
}
2018-08-16 15:43:41 +00:00
// Add custom build types
2017-01-31 22:48:48 +00:00
project . android . buildTypes {
profile {
initWith debug
2017-12-14 10:38:20 +00:00
if ( it . hasProperty ( 'matchingFallbacks' ) ) {
matchingFallbacks = [ 'debug' , 'release' ]
}
2017-01-31 22:48:48 +00:00
}
2018-08-16 15:43:41 +00:00
dynamicProfile {
initWith debug
if ( it . hasProperty ( 'matchingFallbacks' ) ) {
matchingFallbacks = [ 'debug' , 'release' ]
}
}
dynamicRelease {
initWith debug
if ( it . hasProperty ( 'matchingFallbacks' ) ) {
matchingFallbacks = [ 'debug' , 'release' ]
}
}
2017-01-31 22:48:48 +00:00
}
2016-03-12 00:30:56 +00:00
2017-05-22 14:50:31 +00:00
String flutterRootPath = resolveProperty ( project , "flutter.sdk" , System . env . FLUTTER_ROOT )
2016-06-06 19:56:04 +00:00
if ( flutterRootPath = = null ) {
2017-05-22 14:50:31 +00:00
throw new GradleException ( "Flutter SDK not found. Define location with flutter.sdk in the local.properties file or with a FLUTTER_ROOT environment variable." )
2016-03-12 00:30:56 +00:00
}
2016-06-06 19:56:04 +00:00
flutterRoot = project . file ( flutterRootPath )
if ( ! flutterRoot . isDirectory ( ) ) {
2016-03-12 00:30:56 +00:00
throw new GradleException ( "flutter.sdk must point to the Flutter SDK directory" )
}
2017-02-16 22:17:09 +00:00
String flutterExecutableName = Os . isFamily ( Os . FAMILY_WINDOWS ) ? "flutter.bat" : "flutter"
flutterExecutable = Paths . get ( flutterRoot . absolutePath , "bin" , flutterExecutableName ) . toFile ( ) ;
2017-02-23 13:56:19 +00:00
if ( project . hasProperty ( 'localEngineOut' ) ) {
String engineOutPath = project . property ( 'localEngineOut' )
File engineOut = project . file ( engineOutPath )
if ( ! engineOut . isDirectory ( ) ) {
throw new GradleException ( 'localEngineOut must point to a local engine build' )
}
2019-05-20 20:09:20 +00:00
baseEnginePath = Paths . get ( engineOut . absolutePath )
flutterJar = baseEnginePath . resolve ( "flutter.jar" ) . toFile ( )
2016-04-12 21:26:22 +00:00
if ( ! flutterJar . isFile ( ) ) {
2019-06-09 01:02:32 +00:00
throw new GradleException ( 'Local engine jar not found: ' + flutterJar )
2016-04-12 21:26:22 +00:00
}
2017-02-23 13:56:19 +00:00
localEngine = engineOut . name
localEngineSrcPath = engineOut . parentFile . parent
2019-05-20 20:09:20 +00:00
// The local engine is built for one of the build type.
// However, we use the same engine for each of the build types.
debugFlutterJar = flutterJar
profileFlutterJar = flutterJar
releaseFlutterJar = flutterJar
dynamicProfileFlutterJar = flutterJar
dynamicReleaseFlutterJar = flutterJar
2017-01-31 22:48:48 +00:00
} else {
2019-06-09 01:02:32 +00:00
String basePlatformArch = getBasePlatform ( project )
// This is meant to include the compiled classes only, however it will include `libflutter.so` as well.
2019-05-20 20:09:20 +00:00
baseEnginePath = Paths . get ( flutterRoot . absolutePath , "bin" , "cache" , "artifacts" , "engine" )
2019-06-09 01:02:32 +00:00
debugFlutterJar = baseEnginePath . resolve ( "${basePlatformArch}" ) . resolve ( "flutter.jar" ) . toFile ( )
profileFlutterJar = baseEnginePath . resolve ( "${basePlatformArch}-profile" ) . resolve ( "flutter.jar" ) . toFile ( )
releaseFlutterJar = baseEnginePath . resolve ( "${basePlatformArch}-release" ) . resolve ( "flutter.jar" ) . toFile ( )
dynamicProfileFlutterJar = baseEnginePath . resolve ( "${basePlatformArch}-dynamic-profile" ) . resolve ( "flutter.jar" ) . toFile ( )
dynamicReleaseFlutterJar = baseEnginePath . resolve ( "${basePlatformArch}-dynamic-release" ) . resolve ( "flutter.jar" ) . toFile ( )
2019-05-20 20:09:20 +00:00
}
if ( ! debugFlutterJar . isFile ( ) ) {
project . exec {
executable flutterExecutable . absolutePath
args "--suppress-analytics"
args "precache"
}
2017-01-31 22:48:48 +00:00
if ( ! debugFlutterJar . isFile ( ) ) {
2019-05-20 20:09:20 +00:00
throw new GradleException ( "Unable to find flutter.jar in SDK: ${debugFlutterJar}" )
2016-04-12 21:26:22 +00:00
}
2019-05-20 20:09:20 +00:00
}
2017-01-31 22:48:48 +00:00
2019-05-20 20:09:20 +00:00
// Add x86/x86_64 native library. Debug mode only, for now.
File flutterX86Jar = project . file ( "${project.buildDir}/${AndroidProject.FD_INTERMEDIATES}/flutter/flutter-x86.jar" )
2019-06-09 01:02:32 +00:00
Task debugX86JarTask = project . tasks . create ( "${FLUTTER_BUILD_PREFIX}X86Jar" , Jar ) {
2019-05-20 20:09:20 +00:00
destinationDir flutterX86Jar . parentFile
archiveName flutterX86Jar . name
from ( "${flutterRoot}/bin/cache/artifacts/engine/android-x86/libflutter.so" ) {
into 'lib/x86'
}
from ( "${flutterRoot}/bin/cache/artifacts/engine/android-x64/libflutter.so" ) {
into 'lib/x86_64'
2017-01-31 22:48:48 +00:00
}
2016-04-12 21:26:22 +00:00
}
2019-05-20 20:09:20 +00:00
// Add flutter.jar dependencies to all <buildType>Api configurations, including custom ones
// added after applying the Flutter plugin.
2019-06-09 01:02:32 +00:00
project . android . buildTypes . each {
2019-05-20 20:09:20 +00:00
addFlutterJarApiDependency ( project , it , debugX86JarTask )
}
project . android . buildTypes . whenObjectAdded {
addFlutterJarApiDependency ( project , it , debugX86JarTask )
}
2017-03-23 13:59:12 +00:00
2018-06-20 14:07:05 +00:00
File pluginsFile = new File ( project . projectDir . parentFile . parentFile , '.flutter-plugins' )
2017-03-23 13:59:12 +00:00
Properties plugins = readPropertiesIfExist ( pluginsFile )
plugins . each { name , _ - >
def pluginProject = project . rootProject . findProject ( ":$name" )
if ( pluginProject ! = null ) {
project . dependencies {
2017-12-13 11:16:22 +00:00
if ( project . getConfigurations ( ) . findByName ( "implementation" ) ) {
implementation pluginProject
} else {
compile pluginProject
}
2017-03-23 13:59:12 +00:00
}
2019-05-20 20:09:20 +00:00
pluginProject . afterEvaluate {
2018-10-10 16:54:57 +00:00
pluginProject . android . buildTypes {
profile {
initWith debug
}
}
2019-05-20 20:09:20 +00:00
}
2017-12-13 11:16:22 +00:00
pluginProject . afterEvaluate this . & addFlutterJarCompileOnlyDependency
2017-03-23 13:59:12 +00:00
} else {
project . logger . error ( "Plugin project :$name not found. Please update settings.gradle." )
}
}
}
2017-12-13 11:16:22 +00:00
private void addFlutterJarCompileOnlyDependency ( Project project ) {
2017-08-24 08:44:32 +00:00
if ( project . state . failure ) {
return
}
2017-03-23 13:59:12 +00:00
project . dependencies {
if ( flutterJar ! = null ) {
2017-12-13 11:16:22 +00:00
if ( project . getConfigurations ( ) . findByName ( "compileOnly" ) ) {
compileOnly project . files ( flutterJar )
} else {
provided project . files ( flutterJar )
}
2017-03-23 13:59:12 +00:00
} else {
2017-12-13 11:16:22 +00:00
if ( project . getConfigurations ( ) . findByName ( "debugCompileOnly" ) ) {
debugCompileOnly project . files ( debugFlutterJar )
2018-10-10 16:54:57 +00:00
profileCompileOnly project . files ( profileFlutterJar )
2017-12-13 11:16:22 +00:00
releaseCompileOnly project . files ( releaseFlutterJar )
} else {
debugProvided project . files ( debugFlutterJar )
2018-10-10 16:54:57 +00:00
profileProvided project . files ( profileFlutterJar )
2017-12-13 11:16:22 +00:00
releaseProvided project . files ( releaseFlutterJar )
}
2017-03-23 13:59:12 +00:00
}
}
2016-03-12 00:30:56 +00:00
}
2017-07-31 11:57:24 +00:00
/ * *
2018-06-20 14:07:05 +00:00
* Adds suitable flutter . jar api dependencies to the specified buildType .
2017-07-31 11:57:24 +00:00
*
* Note: The BuildType DSL type is not public , and is therefore omitted from the signature .
* /
2019-05-20 20:09:20 +00:00
private void addFlutterJarApiDependency ( Project project , buildType , Task debugX86JarTask ) {
2017-07-31 11:57:24 +00:00
project . dependencies {
2017-12-13 11:16:22 +00:00
String configuration ;
2018-06-20 14:07:05 +00:00
if ( project . getConfigurations ( ) . findByName ( "api" ) ) {
configuration = buildType . name + "Api" ;
2017-12-13 11:16:22 +00:00
} else {
configuration = buildType . name + "Compile" ;
}
add ( configuration , project . files {
2017-07-31 11:57:24 +00:00
String buildMode = buildModeFor ( buildType )
2019-05-20 20:09:20 +00:00
switch ( buildMode ) {
case "debug" :
[ debugX86JarTask , debugFlutterJar ]
break
case "profile" :
profileFlutterJar
break
case "dynamicProfile" :
dynamicProfileFlutterJar
break
case "dynamicRelease" :
dynamicReleaseFlutterJar
break
case "release" :
releaseFlutterJar
break
default :
throw new GradleException ( "Invalid build mode: ${buildMode}" )
2017-07-31 11:57:24 +00:00
}
} )
}
}
/ * *
* Returns a Flutter build mode suitable for the specified Android buildType .
*
* Note: The BuildType DSL type is not public , and is therefore omitted from the signature .
*
2018-08-16 15:43:41 +00:00
* @return "debug" , "profile" , "dynamicProfile" , "dynamicRelease" , or "release" ( fall - back ) .
2017-07-31 11:57:24 +00:00
* /
private static String buildModeFor ( buildType ) {
if ( buildType . name = = "profile" ) {
return "profile"
2018-08-16 15:43:41 +00:00
} else if ( buildType . name = = "dynamicProfile" ) {
return "dynamicProfile"
} else if ( buildType . name = = "dynamicRelease" ) {
return "dynamicRelease"
2017-07-31 11:57:24 +00:00
} else if ( buildType . debuggable ) {
return "debug"
}
return "release"
}
2019-06-09 01:02:32 +00:00
private static String getEngineArtifactDirName ( buildType , targetArch ) {
if ( buildType . name = = "profile" ) {
return "${targetArch}-profile"
} else if ( buildType . name = = "dynamicProfile" ) {
return "${targetArch}-dynamic-profile"
} else if ( buildType . name = = "dynamicRelease" ) {
return "${targetArch}-dynamic-release"
} else if ( buildType . debuggable ) {
return "${targetArch}"
}
return "${targetArch}-release"
}
2016-03-12 00:30:56 +00:00
private void addFlutterTask ( Project project ) {
2017-08-24 08:44:32 +00:00
if ( project . state . failure ) {
return
}
2016-03-12 00:30:56 +00:00
if ( project . flutter . source = = null ) {
throw new GradleException ( "Must provide Flutter source directory" )
}
2017-01-31 22:48:48 +00:00
String target = project . flutter . target
2016-06-07 20:05:42 +00:00
if ( target = = null ) {
target = 'lib/main.dart'
}
2017-02-23 14:09:05 +00:00
if ( project . hasProperty ( 'target' ) ) {
target = project . property ( 'target' )
}
2016-06-07 20:05:42 +00:00
2018-06-06 20:43:32 +00:00
Boolean verboseValue = null
if ( project . hasProperty ( 'verbose' ) ) {
verboseValue = project . property ( 'verbose' ) . toBoolean ( )
}
2018-03-17 19:47:40 +00:00
String [ ] fileSystemRootsValue = null
if ( project . hasProperty ( 'filesystem-roots' ) ) {
fileSystemRootsValue = project . property ( 'filesystem-roots' ) . split ( '\\|' )
}
String fileSystemSchemeValue = null
if ( project . hasProperty ( 'filesystem-scheme' ) ) {
fileSystemSchemeValue = project . property ( 'filesystem-scheme' )
}
2018-02-12 18:44:31 +00:00
Boolean trackWidgetCreationValue = false
if ( project . hasProperty ( 'track-widget-creation' ) ) {
2018-04-24 16:51:25 +00:00
trackWidgetCreationValue = project . property ( 'track-widget-creation' ) . toBoolean ( )
2018-02-12 18:44:31 +00:00
}
2018-08-16 15:46:57 +00:00
String compilationTraceFilePathValue = null
2018-12-21 20:27:59 +00:00
if ( project . hasProperty ( 'compilation-trace-file' ) ) {
compilationTraceFilePathValue = project . property ( 'compilation-trace-file' )
2018-06-15 18:53:30 +00:00
}
2018-12-20 00:27:47 +00:00
Boolean createPatchValue = false
if ( project . hasProperty ( 'patch' ) ) {
createPatchValue = project . property ( 'patch' ) . toBoolean ( )
}
Integer buildNumberValue = null
if ( project . hasProperty ( 'build-number' ) ) {
buildNumberValue = project . property ( 'build-number' ) . toInteger ( )
}
String baselineDirValue = null
if ( project . hasProperty ( 'baseline-dir' ) ) {
baselineDirValue = project . property ( 'baseline-dir' )
2018-09-28 20:58:37 +00:00
}
2017-10-03 19:55:53 +00:00
String extraFrontEndOptionsValue = null
if ( project . hasProperty ( 'extra-front-end-options' ) ) {
extraFrontEndOptionsValue = project . property ( 'extra-front-end-options' )
}
String extraGenSnapshotOptionsValue = null
if ( project . hasProperty ( 'extra-gen-snapshot-options' ) ) {
extraGenSnapshotOptionsValue = project . property ( 'extra-gen-snapshot-options' )
}
2019-05-20 20:09:20 +00:00
2019-06-09 01:02:32 +00:00
def targetPlatforms = this . getTargetPlatforms ( project )
2018-04-06 11:11:47 +00:00
def addFlutterDeps = { variant - >
2019-06-09 01:02:32 +00:00
if ( this . splitPerAbi ( project ) ) {
variant . outputs . each { output - >
// Assigns the new version code to versionCodeOverride, which changes the version code
// for only the output APK, not for the variant itself. Skipping this step simply
// causes Gradle to use the value of variant.versionCode for the APK.
// For more, see https://developer.android.com/studio/build/configure-apk-splits
def abiVersionCode = ABI_VERSION . get ( output . getFilter ( OutputFile . ABI ) )
if ( abiVersionCode ! = null ) {
output . versionCodeOverride =
abiVersionCode * 1000 + variant . versionCode
}
}
}
2019-05-20 20:09:20 +00:00
2019-06-09 01:02:32 +00:00
String flutterBuildMode = buildModeFor ( variant . buildType )
if ( flutterBuildMode = = 'debug' & & project . tasks . findByName ( "${FLUTTER_BUILD_PREFIX}X86Jar" ) ) {
2017-07-31 11:57:24 +00:00
Task task = project . tasks . findByName ( "compile${variant.name.capitalize()}JavaWithJavac" )
if ( task ) {
task . dependsOn project . flutterBuildX86Jar
}
2017-12-13 11:16:22 +00:00
task = project . tasks . findByName ( "compile${variant.name.capitalize()}Kotlin" )
if ( task ) {
task . dependsOn project . flutterBuildX86Jar
}
2017-01-31 22:48:48 +00:00
}
2019-05-20 20:09:20 +00:00
def flutterTasks = [ ]
2019-06-09 01:02:32 +00:00
targetPlatforms . each { targetArch - >
String abiValue = PLATFORM_ARCH_MAP [ targetArch ]
String taskName = "compile${FLUTTER_BUILD_PREFIX}${variant.name.capitalize()}${targetArch.replace('android-', '').capitalize()}"
FlutterTask compileTask = project . tasks . create ( name: taskName , type: FlutterTask ) {
2019-05-20 20:09:20 +00:00
flutterRoot this . flutterRoot
flutterExecutable this . flutterExecutable
buildMode flutterBuildMode
localEngine this . localEngine
localEngineSrcPath this . localEngineSrcPath
abi abiValue
targetPath target
verbose verboseValue
fileSystemRoots fileSystemRootsValue
fileSystemScheme fileSystemSchemeValue
trackWidgetCreation trackWidgetCreationValue
compilationTraceFilePath compilationTraceFilePathValue
createPatch createPatchValue
buildNumber buildNumberValue
baselineDir baselineDirValue
2019-06-09 01:02:32 +00:00
targetPlatform targetArch
2019-05-20 20:09:20 +00:00
sourceDir project . file ( project . flutter . source )
2019-06-09 01:02:32 +00:00
intermediateDir project . file ( "${project.buildDir}/${AndroidProject.FD_INTERMEDIATES}/flutter/${variant.name}/${targetArch}" )
2019-05-20 20:09:20 +00:00
extraFrontEndOptions extraFrontEndOptionsValue
extraGenSnapshotOptions extraGenSnapshotOptionsValue
}
flutterTasks . add ( compileTask )
}
2019-06-09 01:02:32 +00:00
def libJar = project . file ( "${project.buildDir}/${AndroidProject.FD_INTERMEDIATES}/flutter/${variant.name}/libs.jar" )
Task packFlutterSnapshotsAndLibsTask = project . tasks . create ( name: "packLibs${FLUTTER_BUILD_PREFIX}${variant.name.capitalize()}" , type: Jar ) {
2019-05-20 20:09:20 +00:00
destinationDir libJar . parentFile
archiveName libJar . name
2019-06-09 01:02:32 +00:00
targetPlatforms . each { targetArch - >
// This check prevents including `libflutter.so` twice, since it's included in the base platform jar.
// Unfortunately, the `pickFirst` setting in `packagingOptions` does not work when the project `:flutter`
// is included as an implementation dependency, which causes duplicated `libflutter.so`.
if ( getBasePlatform ( project ) ! = targetArch ) {
def engineArtifactSubdir = getEngineArtifactDirName ( variant . buildType , targetArch ) ;
// Include `libflutter.so`.
// TODO(blasten): The libs should be outside `flutter.jar` when the artifacts are downloaded.
from ( project . zipTree ( "${flutterRoot}/bin/cache/artifacts/engine/${engineArtifactSubdir}/flutter.jar" ) ) {
include 'lib/**'
}
2019-05-20 20:09:20 +00:00
}
}
dependsOn flutterTasks
2019-06-09 01:02:32 +00:00
// Add the ELF library.
flutterTasks . each { flutterTask - >
2019-05-20 20:09:20 +00:00
from ( flutterTask . intermediateDir ) {
2019-06-09 01:02:32 +00:00
include '*.so'
rename { String filename - >
return "lib/${flutterTask.abi}/lib${filename}"
2019-05-20 20:09:20 +00:00
}
}
}
2017-01-31 22:48:48 +00:00
}
2019-05-20 20:09:20 +00:00
// Include the snapshots and libflutter.so in `lib/`.
2019-06-09 01:02:32 +00:00
project . dependencies {
String configuration ;
if ( project . getConfigurations ( ) . findByName ( "api" ) ) {
configuration = buildType . name + "Api" ;
} else {
configuration = buildType . name + "Compile" ;
2019-05-20 20:09:20 +00:00
}
2019-06-09 01:02:32 +00:00
add ( configuration , project . files {
packFlutterSnapshotsAndLibsTask
} )
2019-05-20 20:09:20 +00:00
}
2018-10-09 17:40:03 +00:00
// We know that the flutter app is a subproject in another Android app when these tasks exist.
Task packageAssets = project . tasks . findByPath ( ":flutter:package${variant.name.capitalize()}Assets" )
Task cleanPackageAssets = project . tasks . findByPath ( ":flutter:cleanPackage${variant.name.capitalize()}Assets" )
2018-04-12 08:12:26 +00:00
Task copyFlutterAssetsTask = project . tasks . create ( name: "copyFlutterAssets${variant.name.capitalize()}" , type: Copy ) {
2019-05-20 20:09:20 +00:00
dependsOn flutterTasks
2019-01-18 00:33:14 +00:00
if ( packageAssets & & cleanPackageAssets ) {
dependsOn packageAssets
dependsOn cleanPackageAssets
into packageAssets . outputDir
} else {
2019-01-24 22:29:08 +00:00
dependsOn variant . mergeAssets
dependsOn "clean${variant.mergeAssets.name.capitalize()}"
2019-05-28 17:01:20 +00:00
variant . mergeAssets . mustRunAfter ( "clean${variant.mergeAssets.name.capitalize()}" )
2019-01-24 22:29:08 +00:00
into variant . mergeAssets . outputDir
2019-01-18 00:33:14 +00:00
}
2019-06-09 01:02:32 +00:00
flutterTasks . each { flutterTask - >
2019-05-20 20:09:20 +00:00
with flutterTask . assets
}
2016-03-12 00:30:56 +00:00
}
2018-10-09 17:40:03 +00:00
if ( packageAssets ) {
2019-01-30 18:29:29 +00:00
String mainModuleName = "app"
try {
String tmpModuleName = project . rootProject . ext . mainModuleName
if ( tmpModuleName ! = null & & ! tmpModuleName . empty ) {
mainModuleName = tmpModuleName
}
} catch ( Exception e ) {
}
2018-10-09 17:40:03 +00:00
// Only include configurations that exist in parent project.
2019-01-30 18:29:29 +00:00
Task mergeAssets = project . tasks . findByPath ( ":${mainModuleName}:merge${variant.name.capitalize()}Assets" )
2018-10-09 17:40:03 +00:00
if ( mergeAssets ) {
mergeAssets . dependsOn ( copyFlutterAssetsTask )
}
} else {
2019-05-20 20:09:20 +00:00
def processResources = variant . outputs . first ( ) . processResources
processResources . dependsOn ( copyFlutterAssetsTask )
2018-10-09 17:40:03 +00:00
}
2016-03-12 00:30:56 +00:00
}
2019-05-20 20:09:20 +00:00
2018-04-06 11:11:47 +00:00
if ( project . android . hasProperty ( "applicationVariants" ) ) {
project . android . applicationVariants . all addFlutterDeps
} else {
project . android . libraryVariants . all addFlutterDeps
}
2016-03-12 00:30:56 +00:00
}
}
class FlutterExtension {
String source
2016-06-07 20:05:42 +00:00
String target
2016-03-12 00:30:56 +00:00
}
2017-05-05 12:53:51 +00:00
abstract class BaseFlutterTask extends DefaultTask {
2016-06-06 19:56:04 +00:00
File flutterRoot
2017-02-16 22:17:09 +00:00
File flutterExecutable
2016-06-06 19:56:04 +00:00
String buildMode
String localEngine
2017-02-23 13:56:19 +00:00
String localEngineSrcPath
2017-02-23 14:09:05 +00:00
@Input
2016-06-07 20:05:42 +00:00
String targetPath
2017-10-03 19:55:53 +00:00
@Optional @Input
2018-06-06 20:43:32 +00:00
Boolean verbose
@Optional @Input
2018-03-17 19:47:40 +00:00
String [ ] fileSystemRoots
@Optional @Input
String fileSystemScheme
2018-10-17 14:39:20 +00:00
@Input
2018-02-12 18:44:31 +00:00
Boolean trackWidgetCreation
@Optional @Input
2018-08-16 15:46:57 +00:00
String compilationTraceFilePath
2018-06-15 18:53:30 +00:00
@Optional @Input
2018-12-20 00:27:47 +00:00
Boolean createPatch
@Optional @Input
Integer buildNumber
@Optional @Input
String baselineDir
2018-09-28 20:58:37 +00:00
@Optional @Input
2018-02-01 23:14:48 +00:00
String targetPlatform
2019-05-20 20:09:20 +00:00
@Input
String abi
2016-03-12 00:30:56 +00:00
File sourceDir
2017-05-05 12:53:51 +00:00
File intermediateDir
2017-10-03 19:55:53 +00:00
@Optional @Input
String extraFrontEndOptions
@Optional @Input
String extraGenSnapshotOptions
2017-05-05 12:53:51 +00:00
Include kernel_compile.d in Gradle depfiles (#17175)
This updates the Android build to declare the kernel compile depfile as
an output and its contents as inputs when running with --preview-dart-2
(the default mode).
The 'flutter build aot' command behaves differently depending on whether
it's running in Dart 1 or Dart 2 mode:
* Dart 1: the entrypoint Dart file (typically main.dart) is passed
directly to gen_snapshot, which then emits snapshot.d, whose contents
list the transitive closure of Dart dependencies (input files) for the
snapshot. snapshot.d is a declared output, its contents (plus
gen_snapshot itself) constitute the set of input files to the Gradle
build action.
* Dart 2: then entrypoint Dart file (typically main.dart) is first
compiled with the Dart kernel frontend. This emits kernel_compile.d,
whose contents list the transitive closure of Dart dependencies (input
files) for the kernel 'dill' output file. This 'dill' file is then
passed to gen_snapshot, which emits snapshot.d, whose contents are
empty. As of this change, both snapshot.d and kernel_compile.d are
declared outputs, and their contents (plus gen_snapshot and the
frontend compiler themselves) constitute the set of input files to the
Gradle build action.
This fixes a bug wherein profile/release AOT outputs were not
invalidated due to snapshot.d being empty, and kernel_compile.d being
ignored. This was introduced during recent refactoring of the AOT build
code, wherein the kernel compile and gen_snapshot actions were changed
to emit independent depfiles (previously one stomped -- or failed to --
on the other's output).
2018-05-02 01:11:57 +00:00
@OutputFiles
FileCollection getDependenciesFiles ( ) {
2018-08-16 15:47:20 +00:00
FileCollection depfiles = project . files ( )
2018-09-04 15:50:05 +00:00
// Include the kernel compiler depfile, since kernel compile is the
// first stage of AOT build in this mode, and it includes all the Dart
// sources.
depfiles + = project . files ( "${intermediateDir}/kernel_compile.d" )
2018-09-07 19:33:05 +00:00
// Include Core JIT kernel compiler depfile, since kernel compile is
// the first stage of JIT builds in this mode, and it includes all the
// Dart sources.
depfiles + = project . files ( "${intermediateDir}/snapshot_blob.bin.d" )
2018-06-15 18:53:30 +00:00
return depfiles
2017-05-05 12:53:51 +00:00
}
2018-04-12 08:12:26 +00:00
void buildBundle ( ) {
2017-05-05 12:53:51 +00:00
if ( ! sourceDir . isDirectory ( ) ) {
throw new GradleException ( "Invalid Flutter source directory: ${sourceDir}" )
}
intermediateDir . mkdirs ( )
2018-08-16 15:43:41 +00:00
if ( buildMode = = "profile" | | buildMode = = "release" ) {
2017-05-05 12:53:51 +00:00
project . exec {
executable flutterExecutable . absolutePath
workingDir sourceDir
if ( localEngine ! = null ) {
args "--local-engine" , localEngine
args "--local-engine-src-path" , localEngineSrcPath
}
args "build" , "aot"
2017-06-16 21:33:59 +00:00
args "--suppress-analytics"
2017-05-05 12:53:51 +00:00
args "--quiet"
2019-06-09 01:02:32 +00:00
args "--build-shared-library"
2017-05-05 12:53:51 +00:00
args "--target" , targetPath
args "--output-dir" , "${intermediateDir}"
2019-06-09 01:02:32 +00:00
args "--target-platform" , "${targetPlatform}"
2018-02-12 18:44:31 +00:00
if ( trackWidgetCreation ) {
args "--track-widget-creation"
}
2017-10-03 19:55:53 +00:00
if ( extraFrontEndOptions ! = null ) {
2017-11-21 14:44:03 +00:00
args "--extra-front-end-options" , "${extraFrontEndOptions}"
2017-10-03 19:55:53 +00:00
}
if ( extraGenSnapshotOptions ! = null ) {
2017-11-21 14:44:03 +00:00
args "--extra-gen-snapshot-options" , "${extraGenSnapshotOptions}"
}
2017-05-05 12:53:51 +00:00
args "--${buildMode}"
}
}
2016-03-12 00:30:56 +00:00
2017-05-05 12:53:51 +00:00
project . exec {
executable flutterExecutable . absolutePath
workingDir sourceDir
if ( localEngine ! = null ) {
args "--local-engine" , localEngine
args "--local-engine-src-path" , localEngineSrcPath
}
2018-04-12 08:12:26 +00:00
args "build" , "bundle"
2017-06-16 21:33:59 +00:00
args "--suppress-analytics"
2017-05-05 12:53:51 +00:00
args "--target" , targetPath
2019-06-09 01:02:32 +00:00
args "--target-platform" , "${targetPlatform}"
2018-06-06 20:43:32 +00:00
if ( verbose ) {
args "--verbose"
}
2018-03-17 19:47:40 +00:00
if ( fileSystemRoots ! = null ) {
for ( root in fileSystemRoots ) {
args "--filesystem-root" , root
}
}
if ( fileSystemScheme ! = null ) {
args "--filesystem-scheme" , fileSystemScheme
}
2018-02-12 18:44:31 +00:00
if ( trackWidgetCreation ) {
args "--track-widget-creation"
}
2018-08-16 15:46:57 +00:00
if ( compilationTraceFilePath ! = null ) {
2018-12-21 20:27:59 +00:00
args "--compilation-trace-file" , compilationTraceFilePath
2018-06-15 18:53:30 +00:00
}
2018-12-20 00:27:47 +00:00
if ( createPatch ) {
args "--patch"
args "--build-number" , project . android . defaultConfig . versionCode
if ( buildNumber ! = null ) {
assert buildNumber = = project . android . defaultConfig . versionCode
}
}
if ( baselineDir ! = null ) {
args "--baseline-dir" , baselineDir
2018-09-28 20:58:37 +00:00
}
2018-05-14 16:41:58 +00:00
if ( extraFrontEndOptions ! = null ) {
args "--extra-front-end-options" , "${extraFrontEndOptions}"
}
2018-06-15 18:53:30 +00:00
if ( extraGenSnapshotOptions ! = null ) {
args "--extra-gen-snapshot-options" , "${extraGenSnapshotOptions}"
}
2018-08-16 15:43:41 +00:00
if ( buildMode = = "release" | | buildMode = = "profile" ) {
2017-05-05 12:53:51 +00:00
args "--precompiled"
2018-09-07 19:33:05 +00:00
} else {
args "--depfile" , "${intermediateDir}/snapshot_blob.bin.d"
2017-05-05 12:53:51 +00:00
}
2018-04-12 08:12:26 +00:00
args "--asset-dir" , "${intermediateDir}/flutter_assets"
2018-08-16 15:43:41 +00:00
if ( buildMode = = "debug" ) {
args "--debug"
}
if ( buildMode = = "profile" | | buildMode = = "dynamicProfile" ) {
args "--profile"
}
if ( buildMode = = "release" | | buildMode = = "dynamicRelease" ) {
args "--release"
}
if ( buildMode = = "dynamicProfile" | | buildMode = = "dynamicRelease" ) {
args "--dynamic"
}
2017-05-05 12:53:51 +00:00
}
}
}
class FlutterTask extends BaseFlutterTask {
2016-03-12 00:30:56 +00:00
@OutputDirectory
2017-05-05 12:53:51 +00:00
File getOutputDirectory ( ) {
return intermediateDir
}
2016-03-12 00:30:56 +00:00
2016-06-06 19:56:04 +00:00
CopySpec getAssets ( ) {
return project . copySpec {
2017-12-14 16:27:25 +00:00
from "${intermediateDir}"
include "flutter_assets/**" // the working dir and its files
2019-05-20 20:09:20 +00:00
}
}
CopySpec getSnapshots ( ) {
return project . copySpec {
from "${intermediateDir}"
2018-01-15 02:24:07 +00:00
2018-10-09 20:04:06 +00:00
if ( buildMode = = 'release' | | buildMode = = 'profile' ) {
2019-06-09 01:02:32 +00:00
include "app.so"
2016-06-06 19:56:04 +00:00
}
2018-01-15 02:24:07 +00:00
}
2016-03-12 00:30:56 +00:00
}
2017-03-14 11:49:30 +00:00
FileCollection readDependencies ( File dependenciesFile ) {
if ( dependenciesFile . exists ( ) ) {
try {
// Dependencies file has Makefile syntax:
2018-10-19 03:44:38 +00:00
// <target> <files>: <source> <files> <separated> <by> <non-escaped space>
2017-03-14 11:49:30 +00:00
String depText = dependenciesFile . text
2018-10-19 03:44:38 +00:00
// So we split list of files by non-escaped(by backslash) space,
def matcher = depText . split ( ': ' ) [ 1 ] = ~ /(\\ |[^\s])+/
// then we replace all escaped spaces with regular spaces
def depList = matcher . collect { it [ 0 ] . replaceAll ( "\\\\ " , " " ) }
return project . files ( depList )
2017-03-14 11:49:30 +00:00
} catch ( Exception e ) {
logger . error ( "Error reading dependency file ${dependenciesFile}: ${e}" )
}
}
Include kernel_compile.d in Gradle depfiles (#17175)
This updates the Android build to declare the kernel compile depfile as
an output and its contents as inputs when running with --preview-dart-2
(the default mode).
The 'flutter build aot' command behaves differently depending on whether
it's running in Dart 1 or Dart 2 mode:
* Dart 1: the entrypoint Dart file (typically main.dart) is passed
directly to gen_snapshot, which then emits snapshot.d, whose contents
list the transitive closure of Dart dependencies (input files) for the
snapshot. snapshot.d is a declared output, its contents (plus
gen_snapshot itself) constitute the set of input files to the Gradle
build action.
* Dart 2: then entrypoint Dart file (typically main.dart) is first
compiled with the Dart kernel frontend. This emits kernel_compile.d,
whose contents list the transitive closure of Dart dependencies (input
files) for the kernel 'dill' output file. This 'dill' file is then
passed to gen_snapshot, which emits snapshot.d, whose contents are
empty. As of this change, both snapshot.d and kernel_compile.d are
declared outputs, and their contents (plus gen_snapshot and the
frontend compiler themselves) constitute the set of input files to the
Gradle build action.
This fixes a bug wherein profile/release AOT outputs were not
invalidated due to snapshot.d being empty, and kernel_compile.d being
ignored. This was introduced during recent refactoring of the AOT build
code, wherein the kernel compile and gen_snapshot actions were changed
to emit independent depfiles (previously one stomped -- or failed to --
on the other's output).
2018-05-02 01:11:57 +00:00
return project . files ( )
2017-03-14 11:49:30 +00:00
}
2017-01-31 22:48:48 +00:00
@InputFiles
FileCollection getSourceFiles ( ) {
Include kernel_compile.d in Gradle depfiles (#17175)
This updates the Android build to declare the kernel compile depfile as
an output and its contents as inputs when running with --preview-dart-2
(the default mode).
The 'flutter build aot' command behaves differently depending on whether
it's running in Dart 1 or Dart 2 mode:
* Dart 1: the entrypoint Dart file (typically main.dart) is passed
directly to gen_snapshot, which then emits snapshot.d, whose contents
list the transitive closure of Dart dependencies (input files) for the
snapshot. snapshot.d is a declared output, its contents (plus
gen_snapshot itself) constitute the set of input files to the Gradle
build action.
* Dart 2: then entrypoint Dart file (typically main.dart) is first
compiled with the Dart kernel frontend. This emits kernel_compile.d,
whose contents list the transitive closure of Dart dependencies (input
files) for the kernel 'dill' output file. This 'dill' file is then
passed to gen_snapshot, which emits snapshot.d, whose contents are
empty. As of this change, both snapshot.d and kernel_compile.d are
declared outputs, and their contents (plus gen_snapshot and the
frontend compiler themselves) constitute the set of input files to the
Gradle build action.
This fixes a bug wherein profile/release AOT outputs were not
invalidated due to snapshot.d being empty, and kernel_compile.d being
ignored. This was introduced during recent refactoring of the AOT build
code, wherein the kernel compile and gen_snapshot actions were changed
to emit independent depfiles (previously one stomped -- or failed to --
on the other's output).
2018-05-02 01:11:57 +00:00
FileCollection sources = project . files ( )
for ( File depfile in getDependenciesFiles ( ) ) {
sources + = readDependencies ( depfile )
}
if ( ! sources . isEmpty ( ) ) {
2017-03-14 11:49:30 +00:00
// We have a dependencies file. Add a dependency on gen_snapshot as well, since the
// snapshots have to be rebuilt if it changes.
Include kernel_compile.d in Gradle depfiles (#17175)
This updates the Android build to declare the kernel compile depfile as
an output and its contents as inputs when running with --preview-dart-2
(the default mode).
The 'flutter build aot' command behaves differently depending on whether
it's running in Dart 1 or Dart 2 mode:
* Dart 1: the entrypoint Dart file (typically main.dart) is passed
directly to gen_snapshot, which then emits snapshot.d, whose contents
list the transitive closure of Dart dependencies (input files) for the
snapshot. snapshot.d is a declared output, its contents (plus
gen_snapshot itself) constitute the set of input files to the Gradle
build action.
* Dart 2: then entrypoint Dart file (typically main.dart) is first
compiled with the Dart kernel frontend. This emits kernel_compile.d,
whose contents list the transitive closure of Dart dependencies (input
files) for the kernel 'dill' output file. This 'dill' file is then
passed to gen_snapshot, which emits snapshot.d, whose contents are
empty. As of this change, both snapshot.d and kernel_compile.d are
declared outputs, and their contents (plus gen_snapshot and the
frontend compiler themselves) constitute the set of input files to the
Gradle build action.
This fixes a bug wherein profile/release AOT outputs were not
invalidated due to snapshot.d being empty, and kernel_compile.d being
ignored. This was introduced during recent refactoring of the AOT build
code, wherein the kernel compile and gen_snapshot actions were changed
to emit independent depfiles (previously one stomped -- or failed to --
on the other's output).
2018-05-02 01:11:57 +00:00
sources + = readDependencies ( project . file ( "${intermediateDir}/gen_snapshot.d" ) )
2018-09-04 15:50:05 +00:00
sources + = readDependencies ( project . file ( "${intermediateDir}/frontend_server.d" ) )
2017-09-12 06:35:02 +00:00
if ( localEngineSrcPath ! = null ) {
Include kernel_compile.d in Gradle depfiles (#17175)
This updates the Android build to declare the kernel compile depfile as
an output and its contents as inputs when running with --preview-dart-2
(the default mode).
The 'flutter build aot' command behaves differently depending on whether
it's running in Dart 1 or Dart 2 mode:
* Dart 1: the entrypoint Dart file (typically main.dart) is passed
directly to gen_snapshot, which then emits snapshot.d, whose contents
list the transitive closure of Dart dependencies (input files) for the
snapshot. snapshot.d is a declared output, its contents (plus
gen_snapshot itself) constitute the set of input files to the Gradle
build action.
* Dart 2: then entrypoint Dart file (typically main.dart) is first
compiled with the Dart kernel frontend. This emits kernel_compile.d,
whose contents list the transitive closure of Dart dependencies (input
files) for the kernel 'dill' output file. This 'dill' file is then
passed to gen_snapshot, which emits snapshot.d, whose contents are
empty. As of this change, both snapshot.d and kernel_compile.d are
declared outputs, and their contents (plus gen_snapshot and the
frontend compiler themselves) constitute the set of input files to the
Gradle build action.
This fixes a bug wherein profile/release AOT outputs were not
invalidated due to snapshot.d being empty, and kernel_compile.d being
ignored. This was introduced during recent refactoring of the AOT build
code, wherein the kernel compile and gen_snapshot actions were changed
to emit independent depfiles (previously one stomped -- or failed to --
on the other's output).
2018-05-02 01:11:57 +00:00
sources + = project . files ( "$localEngineSrcPath/$localEngine" )
2017-09-12 06:35:02 +00:00
}
2017-03-14 11:49:30 +00:00
// Finally, add a dependency on pubspec.yaml as well.
Include kernel_compile.d in Gradle depfiles (#17175)
This updates the Android build to declare the kernel compile depfile as
an output and its contents as inputs when running with --preview-dart-2
(the default mode).
The 'flutter build aot' command behaves differently depending on whether
it's running in Dart 1 or Dart 2 mode:
* Dart 1: the entrypoint Dart file (typically main.dart) is passed
directly to gen_snapshot, which then emits snapshot.d, whose contents
list the transitive closure of Dart dependencies (input files) for the
snapshot. snapshot.d is a declared output, its contents (plus
gen_snapshot itself) constitute the set of input files to the Gradle
build action.
* Dart 2: then entrypoint Dart file (typically main.dart) is first
compiled with the Dart kernel frontend. This emits kernel_compile.d,
whose contents list the transitive closure of Dart dependencies (input
files) for the kernel 'dill' output file. This 'dill' file is then
passed to gen_snapshot, which emits snapshot.d, whose contents are
empty. As of this change, both snapshot.d and kernel_compile.d are
declared outputs, and their contents (plus gen_snapshot and the
frontend compiler themselves) constitute the set of input files to the
Gradle build action.
This fixes a bug wherein profile/release AOT outputs were not
invalidated due to snapshot.d being empty, and kernel_compile.d being
ignored. This was introduced during recent refactoring of the AOT build
code, wherein the kernel compile and gen_snapshot actions were changed
to emit independent depfiles (previously one stomped -- or failed to --
on the other's output).
2018-05-02 01:11:57 +00:00
return sources + project . files ( 'pubspec.yaml' )
2017-03-14 11:49:30 +00:00
}
// No dependencies file (or problems parsing it). Fall back to source files.
return project . fileTree (
dir: sourceDir ,
exclude: [ 'android' , 'ios' ] ,
include: [ '**/*.dart' , 'pubspec.yaml' ]
)
2017-01-31 22:48:48 +00:00
}
2016-03-12 00:30:56 +00:00
@TaskAction
void build ( ) {
2018-04-12 08:12:26 +00:00
buildBundle ( )
2016-03-12 00:30:56 +00:00
}
}
2018-09-20 22:45:48 +00:00
gradle . useLogger ( new FlutterEventLogger ( ) )
class FlutterEventLogger extends BuildAdapter implements TaskExecutionListener {
String mostRecentTask = ""
void beforeExecute ( Task task ) {
mostRecentTask = task . name
}
void afterExecute ( Task task , TaskState state ) { }
void buildFinished ( BuildResult result ) {
if ( result . failure ! = null ) {
2019-06-09 01:02:32 +00:00
if ( ! ( result . failure instanceof GradleException ) | | ! mostRecentTask . startsWith ( FlutterPlugin . FLUTTER_BUILD_PREFIX ) ) {
2018-09-20 22:45:48 +00:00
result . rethrowFailure ( )
}
}
}
}