美文网首页
Gradle Plugin 源码分析

Gradle Plugin 源码分析

作者: 风月寒 | 来源:发表于2021-04-30 14:48 被阅读0次

本文基于的gradle版本如下:

plugin版本 :com.android.tools.build:gradle:3.5.4'
gradle 版本:5.6.4

我们知道Android gradle plugin是用来构建Android工程的gradle插件,在Android gradle 插件中,可以看到app工程和library工程所依赖的plugin是不一样的。

// app 工程
apply plugin: 'com.android.application'
// library 工程
apply plugin: 'com.android.library'

首先我们来看一个app的gradle文件格式。

1619683831(1).png
plugins {
    id 'com.android.application'
}

android {
    compileSdkVersion 30
    buildToolsVersion "30.0.2"

    defaultConfig {
        applicationId "com.example.flavorgradle"
        minSdkVersion 16
        targetSdkVersion 30
        versionCode 1
        versionName "1.0"

        testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner"
    }

    flavorDimensions "default"

    productFlavors {
        out1 {
            applicationId "com.example.out1"
            buildConfigField "String", "CHANNEL", '"channel_out1"'
            versionCode 1
            versionName "1.0"
            signingConfig signingConfigs.release
        }
        out2 {
            applicationId "com.example.out2"
            buildConfigField "String", "CHANNEL", '"channel_out2"'
            versionCode 1
            versionName "1.0"
            signingConfig signingConfigs.release
        }
        out3 {
            applicationId "com.example.out3"
            buildConfigField "String", "CHANNEL", '"channel_out2"'
            versionCode 1
            versionName "1.0"
            signingConfig signingConfigs.release
        }
    }
    
    signingConfigs {
        debug {
            //签名文件
            storeFile file(keystoreProperties["STORE_FILE"])
            storePassword keystoreProperties["STORE_PASSWORD"]
            keyAlias keystoreProperties["KEY_ALIAS"]
            //签名密码
            keyPassword keystoreProperties["KEY_PASSWORD"]
            v2SigningEnabled false
        }

        release {
            //签名文件
            storeFile file(keystoreProperties["STORE_FILE"])
            storePassword keystoreProperties["STORE_PASSWORD"]
            keyAlias keystoreProperties["KEY_ALIAS"]
            //签名密码
            keyPassword keystoreProperties["KEY_PASSWORD"]
            v2SigningEnabled false
        }
    }

    buildTypes {
        release {
            minifyEnabled false
            proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro'
        }
    }
    compileOptions {
        sourceCompatibility JavaVersion.VERSION_1_8
        targetCompatibility JavaVersion.VERSION_1_8
    }
}

dependencies {

    implementation 'androidx.appcompat:appcompat:1.2.0'
    implementation 'com.google.android.material:material:1.3.0'
    implementation 'androidx.constraintlayout:constraintlayout:2.0.4'
    testImplementation 'junit:junit:4.13.2'
    androidTestImplementation 'androidx.test.ext:junit:1.1.2'
    androidTestImplementation 'androidx.test.espresso:espresso-core:3.3.0'
    /*
        这个需要跟上面productFlavors配合使用,目的是只是在项目为out3打包的时候引用,编译
     */
    out3Implementation 'com.github.vivitale:BaseCore:0.0.54'
}

我们都知道最上面引用的plugin id 对应gradle3.5.4的META-INF/gradle-plugins文件夹的com.android.build.gradle.properties文件。

1619690725(1).png
implementation-class=com.android.build.gradle.AppPlugin

对应具体的插件为AppPlugin。

然后android则对应AppExtension。其创建是在AbstractAppPlugin类中。

protected BaseExtension createExtension(
            @NonNull Project project,
            @NonNull ProjectOptions projectOptions,
            @NonNull GlobalScope globalScope,
            @NonNull NamedDomainObjectContainer<BuildType> buildTypeContainer,
            @NonNull NamedDomainObjectContainer<ProductFlavor> productFlavorContainer,
            @NonNull NamedDomainObjectContainer<SigningConfig> signingConfigContainer,
            @NonNull NamedDomainObjectContainer<BaseVariantOutput> buildOutputs,
            @NonNull SourceSetManager sourceSetManager,
            @NonNull ExtraModelInfo extraModelInfo) {
        return project.getExtensions()
                .create(
                        "android",
                        getExtensionClass(),
                        project,
                        projectOptions,
                        globalScope,
                        buildTypeContainer,
                        productFlavorContainer,
                        signingConfigContainer,
                        buildOutputs,
                        sourceSetManager,
                        extraModelInfo,
                        isBaseApplication);
    }

创建了一个名字为android的Extension,具体实现为AppExtension。

public AppExtension(
            @NonNull Project project,
            @NonNull ProjectOptions projectOptions,
            @NonNull GlobalScope globalScope,
            @NonNull NamedDomainObjectContainer<BuildType> buildTypes,
            @NonNull NamedDomainObjectContainer<ProductFlavor> productFlavors,
            @NonNull NamedDomainObjectContainer<SigningConfig> signingConfigs,
            @NonNull NamedDomainObjectContainer<BaseVariantOutput> buildOutputs,
            @NonNull SourceSetManager sourceSetManager,
            @NonNull ExtraModelInfo extraModelInfo,
            boolean isBaseModule) {
        super(
                project,
                projectOptions,
                globalScope,
                buildTypes,
                productFlavors,
                signingConfigs,
                buildOutputs,
                sourceSetManager,
                extraModelInfo,
                isBaseModule);
    }

其中signingConfigs对应build.gradle文件中的

signingConfigs {
        debug {
            //签名文件
            storeFile file(keystoreProperties["STORE_FILE"])
            storePassword keystoreProperties["STORE_PASSWORD"]
            keyAlias keystoreProperties["KEY_ALIAS"]
            //签名密码
            keyPassword keystoreProperties["KEY_PASSWORD"]
            v2SigningEnabled false
        }

        release {
            //签名文件
            storeFile file(keystoreProperties["STORE_FILE"])
            storePassword keystoreProperties["STORE_PASSWORD"]
            keyAlias keystoreProperties["KEY_ALIAS"]
            //签名密码
            keyPassword keystoreProperties["KEY_PASSWORD"]
            v2SigningEnabled false
        }
    }

这一整块代码。buildTypes则对应:

buildTypes {
        release {
            minifyEnabled false
            proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro'
        }
    }

productFlavors对应

productFlavors {
        out1 {
            applicationId "com.example.out1"
            buildConfigField "String", "CHANNEL", '"channel_out1"'
            versionCode 1
            versionName "1.0"
            signingConfig signingConfigs.release
        }
        out2 {
            applicationId "com.example.out2"
            buildConfigField "String", "CHANNEL", '"channel_out2"'
            versionCode 1
            versionName "1.0"
            signingConfig signingConfigs.release
        }
        out3 {
            applicationId "com.example.out3"
            buildConfigField "String", "CHANNEL", '"channel_out2"'
            versionCode 1
            versionName "1.0"
            signingConfig signingConfigs.release
        }
    }

下面我们来分析AppPlugin的构建流程。AppPlugin最终会走父类BasePlgin的apply方法。

public final void apply(@NonNull Project project) {
        CrashReporting.runAction(
                () -> {
                    basePluginApply(project);
                    pluginSpecificApply(project);
                });
    }
private void basePluginApply(@NonNull Project project) {
        this.project = project;
        this.projectOptions = new ProjectOptions(project);
        checkGradleVersion(project, getLogger(), projectOptions);//1
        DependencyResolutionChecks.registerDependencyCheck(project, projectOptions);//2

        project.getPluginManager().apply(AndroidBasePlugin.class);//3

        checkPathForErrors();//4
        checkModulesForErrors();//5

        PluginInitializer.initialize(project);//5
        RecordingBuildListener buildListener = ProfilerInitializer.init(project, projectOptions);//6
        ProfileAgent.INSTANCE.register(project.getName(), buildListener);
        threadRecorder = ThreadRecorder.get();

        Workers.INSTANCE.initFromProject(
                projectOptions,
                ForkJoinPool.commonPool());

        ProcessProfileWriter.getProject(project.getPath())
                .setAndroidPluginVersion(Version.ANDROID_GRADLE_PLUGIN_VERSION)
                .setAndroidPlugin(getAnalyticsPluginType())
                .setPluginGeneration(GradleBuildProject.PluginGeneration.FIRST)
                .setOptions(AnalyticsUtil.toProto(projectOptions));//7

        if (!projectOptions.get(BooleanOption.ENABLE_NEW_DSL_AND_API)) {

            threadRecorder.record(
                    ExecutionType.BASE_PLUGIN_PROJECT_CONFIGURE,
                    project.getPath(),
                    null,
                    this::configureProject);//8

            threadRecorder.record(
                    ExecutionType.BASE_PLUGIN_PROJECT_BASE_EXTENSION_CREATION,
                    project.getPath(),
                    null,
                    this::configureExtension);//9

            threadRecorder.record(
                    ExecutionType.BASE_PLUGIN_PROJECT_TASKS_CREATION,
                    project.getPath(),
                    null,
                    this::createTasks);//10
        } else {
            ......
        }
            
    }

注释1检查gradle版本。

注释2检查并确保在配置阶段不去解析依赖。

注释3应用一个 AndroidBasePlugin,目的是为了让其他插件作者区分当前应用的是一个 Android 插件。AndroidBasePlugin是一个空实现。

注释4是检查 project 路径是否有错误,发生错误则抛出 StopExecutionException 异常。

注释5检查子 moudle 的结构:目前版本会检查 2 个模块有没有相同的标识(组+名称),如果有则抛出 StopExecutionException 异常。(组件化在不同的 moudle 中需要给资源加 prefix 前缀)

注释6插件初始化,必须立即执行。此外,需要注意,Gradle Deamon进程 永远不会同时执行两个构建流程。

(Gradle会构建两个进程,一个是client进程,其作用是查找并与Deamon进程通信。当Deamon进程没有开启,Client进程则会创建一个Deamon进程。如果Deamon进程已经开启,则将参数给Deamon进程。而Deamon进程是不依赖AS独立存在,构建结束不会依赖。每一个Gradle版本都会对应一个Deamon进程。这么做的目的是加速编译构建时间。)

注释7初始化用于记录构建过程中配置信息的工厂实例 ProcessProfileWriterFactory。

现在重点来分析注释8,9,10 。

注释8:

private void configureProject() {
        final Gradle gradle = project.getGradle();
        ......

        // Enforce minimum versions of certain plugins
        //强制使用不低于当前所支持的最小插件版本,否则会抛出异常。
        GradlePluginUtils.enforceMinimumVersionsOfPlugins(project, syncIssueHandler);

        // Apply the Java plugin
        project.getPlugins().apply(JavaBasePlugin.class);

        DslScopeImpl dslScope =
                new DslScopeImpl(
                        syncIssueHandler, extraModelInfo.getDeprecationReporter(), objectFactory);

        //如果启动了 构建缓存 选项,则会创建 buildCache 实例以便后面能重用缓存。
        @Nullable
        FileCache buildCache = BuildCacheUtils.createBuildCacheIfEnabled(project, projectOptions);

        globalScope =
                new GlobalScope(
                        project,
                        creator,
                        new ProjectWrapper(project),
                        projectOptions,
                        dslScope,
                        sdkComponents,
                        registry,
                        buildCache,
                        extraModelInfo.getMessageReceiver());
        //给名字为assemble的添加描述。
        project.getTasks()
                .named("assemble")
                .configure(
                        task ->
                                task.setDescription(
                                        "Assembles all variants of all applications and secondary packages."));
        //对构建各个阶段的监听
        gradle.addBuildListener(
                new BuildListener() {
                    //构建开始
                    @Override
                    public void buildStarted(@NonNull Gradle gradle) {}
                    //settings 文件解析完成
                    @Override
                    public void settingsEvaluated(@NonNull Settings settings) {}
                    //project加载完成
                    @Override
                    public void projectsLoaded(@NonNull Gradle gradle) {}
                    //project解析完成
                    @Override
                    public void projectsEvaluated(@NonNull Gradle gradle) {}
                    //构建完成
                    @Override
                    public void buildFinished(@NonNull BuildResult buildResult) {
                        if (buildResult.getGradle().getParent() != null) {
                            return;
                        }
                        ModelBuilder.clearCaches();
                        Workers.INSTANCE.shutdown();
                        sdkComponents.unload();
                        SdkLocator.resetCache();
                        ConstraintHandler.clearCache();
                        threadRecorder.record(
                                ExecutionType.BASE_PLUGIN_BUILD_FINISHED,
                                project.getPath(),
                                null,
                                () -> {
                                    if (!projectOptions.get(
                                            BooleanOption.KEEP_SERVICES_BETWEEN_BUILDS)) {
                                        WorkerActionServiceRegistry.INSTANCE
                                                .shutdownAllRegisteredServices(
                                                        ForkJoinPool.commonPool());
                                    }
                                    Main.clearInternTables();
                                });
                        DeprecationReporterImpl.Companion.clean();
                    }
                });

        createLintClasspathConfiguration(project);
    }

根据这些回调,我们可以做一些事,比如在我们的settings.gradle文件中去测量每个阶段所用的时间。

long beginOfSetting = System.currentTimeMillis()
def beginOfConfig
def configHasBegin = false
def beginOfProjectConfig = new HashMap()
def beginOfProjectExcute
gradle.projectsLoaded {
    println '初始化阶段,耗时:' + (System.currentTimeMillis() -
            beginOfSetting) + 'ms'
}
gradle.beforeProject { project ->
    if (!configHasBegin) {
        configHasBegin = true
        beginOfConfig = System.currentTimeMillis()
    }
    beginOfProjectConfig.put(project, System.currentTimeMillis())
}
gradle.afterProject { project ->
    def begin = beginOfProjectConfig.get(project)
    println '配置阶段,' + project + '耗时:' +
            (System.currentTimeMillis() - begin) + 'ms'
}
gradle.taskGraph.whenReady {
    println '配置阶段,总共耗时:' + (System.currentTimeMillis() -
            beginOfConfig) + 'ms'
    beginOfProjectExcute = System.currentTimeMillis()
}
gradle.taskGraph.beforeTask { task ->
    task.doFirst {
        task.ext.beginOfTask = System.currentTimeMillis()
    }
    task.doLast {
        println '执行阶段,' + task + '耗时:' +
                (System.currentTimeMillis() - task.beginOfTask) + 'ms'
    }
}
gradle.buildFinished {
    println '执行阶段,耗时:' + (System.currentTimeMillis() -
            beginOfProjectExcute) + 'ms'
}

下面来看注释9

private void configureExtension() {
        ObjectFactory objectFactory = project.getObjects();
        final NamedDomainObjectContainer<BuildType> buildTypeContainer =
                project.container(
                        BuildType.class,
                        new BuildTypeFactory(
                                objectFactory,
                                project,
                                extraModelInfo.getSyncIssueHandler(),
                                extraModelInfo.getDeprecationReporter()));
        final NamedDomainObjectContainer<ProductFlavor> productFlavorContainer =
                project.container(
                        ProductFlavor.class,
                        new ProductFlavorFactory(
                                objectFactory,
                                project,
                                extraModelInfo.getDeprecationReporter(),
                                project.getLogger()));
        final NamedDomainObjectContainer<SigningConfig> signingConfigContainer =
                project.container(
                        SigningConfig.class,
                        new SigningConfigFactory(
                                objectFactory,
                                GradleKeystoreHelper.getDefaultDebugKeystoreLocation()));

        final NamedDomainObjectContainer<BaseVariantOutput> buildOutputs =
                project.container(BaseVariantOutput.class);

        project.getExtensions().add("buildOutputs", buildOutputs);

        sourceSetManager =
                new SourceSetManager(
                        project,
                        isPackagePublished(),
                        globalScope.getDslScope(),
                        new DelayedActionsExecutor());

        extension =
                createExtension(
                        project,
                        projectOptions,
                        globalScope,
                        buildTypeContainer,
                        productFlavorContainer,
                        signingConfigContainer,
                        buildOutputs,
                        sourceSetManager,
                        extraModelInfo);

        globalScope.setExtension(extension);

        variantFactory = createVariantFactory(globalScope, extension);

        taskManager =
                createTaskManager(
                        globalScope,
                        project,
                        projectOptions,
                        dataBindingBuilder,
                        extension,
                        variantFactory,
                        registry,
                        threadRecorder);

        variantManager =
                new VariantManager(
                        globalScope,
                        project,
                        projectOptions,
                        extension,
                        variantFactory,
                        taskManager,
                        sourceSetManager,
                        threadRecorder);

        registerModels(registry, globalScope, variantManager, extension, extraModelInfo);

        // map the whenObjectAdded callbacks on the containers.
        signingConfigContainer.whenObjectAdded(variantManager::addSigningConfig);

        buildTypeContainer.whenObjectAdded(
                buildType -> {
                    if (!this.getClass().isAssignableFrom(DynamicFeaturePlugin.class)) {
                        SigningConfig signingConfig =
                                signingConfigContainer.findByName(BuilderConstants.DEBUG);
                        buildType.init(signingConfig);
                    } else {
                        // initialize it without the signingConfig for dynamic-features.
                        buildType.init();
                    }
                    variantManager.addBuildType(buildType);
                });

        productFlavorContainer.whenObjectAdded(variantManager::addProductFlavor);

        // map whenObjectRemoved on the containers to throw an exception.
        signingConfigContainer.whenObjectRemoved(
                new UnsupportedAction("Removing signingConfigs is not supported."));
        buildTypeContainer.whenObjectRemoved(
                new UnsupportedAction("Removing build types is not supported."));
        productFlavorContainer.whenObjectRemoved(
                new UnsupportedAction("Removing product flavors is not supported."));

        // create default Objects, signingConfig first as its used by the BuildTypes.
        variantFactory.createDefaultComponents(
                buildTypeContainer, productFlavorContainer, signingConfigContainer);
    }

首先创建了 BuildType、ProductFlavor、SigningConfig、buildOutputs 四个类型的Container,接着传入到了createExtension方法中,该方法的具体实现是在AbstractAppPlugin。

对于Extension我们可以理解为java中的bean类。而BuildType、ProductFlavor、SigningConfig、buildOutputs属于这个bean类里面的成员变量。来看一个具体的SigningConfig。

public SigningConfig initWith(com.android.builder.model.SigningConfig that) {
        setStoreFile(that.getStoreFile());
        setStorePassword(that.getStorePassword());
        setKeyAlias(that.getKeyAlias());
        setKeyPassword(that.getKeyPassword());
        setV1SigningEnabled(that.isV1SigningEnabled());
        setV2SigningEnabled(that.isV2SigningEnabled());
        setStoreType(that.getStoreType());
        return this;
    }

其具体的实现类对应SigningConfigImpl,

override fun initWith(that: SigningConfig) {
        if (checkSeal()) {
            storeFile = that.storeFile
            storePassword = that.storePassword
            keyAlias = that.keyAlias
            keyPassword = that.keyPassword
            v1SigningEnabled = that.v1SigningEnabled
            v2SigningEnabled = that.v2SigningEnabled
        }
    }

而这些对应于build.gradle的:

signingConfigs {
        debug {
            //签名文件
            storeFile file(keystoreProperties["STORE_FILE"])
            storePassword keystoreProperties["STORE_PASSWORD"]
            keyAlias keystoreProperties["KEY_ALIAS"]
            //签名密码
            keyPassword keystoreProperties["KEY_PASSWORD"]
            v2SigningEnabled false
        }

        release {
            //签名文件
            storeFile file(keystoreProperties["STORE_FILE"])
            storePassword keystoreProperties["STORE_PASSWORD"]
            keyAlias keystoreProperties["KEY_ALIAS"]
            //签名密码
            keyPassword keystoreProperties["KEY_PASSWORD"]
            v2SigningEnabled false
        }
    }

一般打包出来的apk需要签名才能安装在我们的手机上。但是我们的debug好像直接就可以安装,也不需要配置签名什么的。其实在源码中,当是debug包的时候,有一个默认的签名。

public static DefaultSigningConfig debugSigningConfig(File storeFile) {
        DefaultSigningConfig result = new DefaultSigningConfig(BuilderConstants.DEBUG);
        result.mStoreFile = storeFile;
        result.mStorePassword = DEFAULT_PASSWORD;
        result.mKeyAlias = DEFAULT_ALIAS;
        result.mKeyPassword = DEFAULT_PASSWORD;
        return result;
    }

这个storeFile是从外面传进来的,我们继续跟下去。

fun createDefaultDebugStore(defaultDebugKeystoreLocation: File, logger: Logger) {
    val signingConfig = DefaultSigningConfig.debugSigningConfig(defaultDebugKeystoreLocation)
    logger.info(
            "Creating default debug keystore at {}",
            defaultDebugKeystoreLocation.absolutePath)
    try {
        if (!KeystoreHelper.createDebugStore(
                signingConfig.storeType,
                signingConfig.storeFile!!,
                signingConfig.storePassword!!,
                signingConfig.keyPassword!!,
                signingConfig.keyAlias!!,
                LoggerWrapper(logger))) {
            throw IOException("Unable to create missing debug keystore.")
        }
    } catch (e: KeytoolException) {
        throw IOException(e)
    }
}

private fun createDefaultDebugKeystoreIfNeeded() {

        checkState(signingConfig.isSigningReady, "Debug signing config not ready.")
        
        checkState(FileUtils.parentDirExists(defaultDebugKeystoreLocation),
                "Parent directory of the default debug keystore '%s' does not exist",
                defaultDebugKeystoreLocation)
        
        SynchronizedFile
                .getInstanceWithMultiProcessLocking(defaultDebugKeystoreLocation)
                .createIfAbsent { createDefaultDebugStore(it, this.logger) }
    }

createDefaultDebugKeystoreIfNeeded中的defaultDebugKeystoreLocation这个赋值是在CreationAction内部类中,在构造方法中传入一个defaultDebugKeystoreLocation,然后在configure进行赋值。

override fun configure(task: ValidateSigningTask) {
            super.configure(task)

            task.signingConfig = variantScope.variantConfiguration.signingConfig ?: throw IllegalStateException(
                "No signing config configured for variant " + variantScope.fullVariantName)
            task.defaultDebugKeystoreLocation = defaultDebugKeystoreLocation
            task.dummyOutputDirectory = outputDirectory
            task.outputs.upToDateWhen { !task.forceRerun() }
        }

继续往下跟进。这个内部类是在一个task中,而我们的task是手TaskManager统一管理。所以可以在TaskManager找到这个内部类的实现。

taskFactory.register(
                new ValidateSigningTask.CreationAction(
                        variantScope, GradleKeystoreHelper.getDefaultDebugKeystoreLocation()));

fun getDefaultDebugKeystoreLocation(): File = try {
    File(KeystoreHelper.defaultDebugKeystoreLocation())
} catch (e: AndroidLocation.AndroidLocationException) {
    throw InvalidUserDataException("Failed to get default debug keystore location.", e)
}
public static String defaultDebugKeystoreLocation() throws AndroidLocationException {
        //this is guaranteed to either return a non null value (terminated with a platform
        // specific separator), or throw.
        String folder = AndroidLocation.getFolder();
        return folder + "debug.keystore";
    }

最后找到的是debug.keystore的一个路径。

这里只是分析了SigningConfig,其他的后面有时间再详细分析。

最后来分析注释10.

private void createTasks() {
        threadRecorder.record(
                ExecutionType.TASK_MANAGER_CREATE_TASKS,
                project.getPath(),
                null,
                () -> taskManager.createTasksBeforeEvaluate());

        project.afterEvaluate(
                CrashReporting.afterEvaluate(
                        p -> {
                            sourceSetManager.runBuildableArtifactsActions();

                            threadRecorder.record(
                                    ExecutionType.BASE_PLUGIN_CREATE_ANDROID_TASKS,
                                    project.getPath(),
                                    null,
                                    this::createAndroidTasks);
                        }));
    }

这里主要是分两块,一个是在 beforeEvaluate 创建任务;一个是在 afterEvaluate 创建任务。这里的区别是AndroidTask是依赖配置项的配置才能生成相应任务,所以是需要在 afterEvaluate 之后创建。

createTasksBeforeEvaluate 创建不依赖 flavor 的 task,TaskManager 的 createTasksBeforeEvaluate 方法给 Task 容器中注册了一系列的 Task,包括 uninstallAllTask、deviceCheckTask、connectedCheckTask、preBuild、extractProguardFiles、sourceSetsTask、assembleAndroidTest、compileLintTask 等等。

我们来看看对应的gradle task,如下图所示:

1619762164(1).png
Android 任务

androidDependencies-显示项目的Android依赖项
signingReport-显示基础和测试模块的签名信息
sourceSets-打印出此项目中定义的所有源集

Build 任务

assemble-组装所有变体的主要输出
assembleAndroidTest-组装所有测试应用程序
build-组装并测试该项目
buildDependents-组装并测试该项目以及依赖于该项目的所有项目
buildNeeded-组装和测试该项目及其依赖的所有项目
bundle-为所有变体组装捆绑包
clean-删除构建目录
cleanBuildCache-删除构建缓存目录
compileDebugAndroidTestSources
compileDebugSources
compileDebugUnitTestSources
compileReleaseSources
compileReleaseUnitTestSources

构建安装任务

init-初始化一个新的Gradle构建
wrapper-生成Gradle包装器文件

清理任务

lintFix-在所有变体上运行lint并对源代码应用任何安全建议

帮助任务

buildEnvironment-显示在根项目“”中声明的所有buildscript依赖项
components-显示根项目“”生成的组件
dependencies-显示在根项目“”中声明的所有依赖项
dependencyInsight-显示对根项目“”中特定依赖关系的洞察。
help-显示帮助消息
model-显示根项目“”的配置模型
projects-显示根项目“”的子项目。

安装任务

installDebug-安装Debug构建
installDebugAndroidTest-为Debug版本安装android(在设备上)测试
uninstallAll-卸载所有应用程序。
uninstallDebug-卸载Debug构建
uninstallDebugAndroidTest-卸载Debug版本的android(在设备上)测试
uninstallRelease-卸载Release版本
 
验证任务

check-运行所有检查
connectedAndroidTest-安装并运行连接设备上所有风格的检测测试
connectedCheck-在当前连接的设备上运行所有设备检查
connectedDebugAndroidTest-在连接的设备上安装并运行测试以进行调试
deviceAndroidTest-使用所有设备提供程序安装和运行检测测试
deviceCheck-使用设备提供程序和测试服务器运行所有设备检查
lint-在所有变种上运行lint
test-对所有变体运行单元测试。
testDebugUnitTest-运行调试版本的单元测试
testReleaseUnitTest-运行发布版本的单元测试

重点关注 variantManager 的 createAndroidTasks 方法,

public List<VariantScope> createAndroidTasks() {
       
        ...
       
        // 1、创建工程级别的测试任务。
        taskManager.createTopLevelTestTasks(!productFlavors.isEmpty());

        // 2、遍历所有 variantScope,为其变体数据创建对应的 Tasks。
        for (final VariantScope variantScope : variantScopes) {
            createTasksForVariantData(variantScope);
        }

        // 3、创建报告相关的 Tasks。
        taskManager.createReportTasks(variantScopes);

        return variantScopes;
    }

可以看到,在 createAndroidTasks 方法中有 三项处理,如下所示:

创建工程级别的测试任务。

遍历所有的 variantScope,为其变体数据创建对应的 Tasks。

创建报告相关的 Tasks。

继续看看 createTasksForVariantData 方法是如何为每一个指定的 Variant 类型创建对应的 Tasks 的。

public void createTasksForVariantData(final VariantScope variantScope) {
        final BaseVariantData variantData = variantScope.getVariantData();
        final VariantType variantType = variantData.getType();
        final GradleVariantConfiguration variantConfig = variantScope.getVariantConfiguration();

        // 1、创建 Assemble Task。
        taskManager.createAssembleTask(variantData);
        
        // 2、如果 variantType 是 base moudle,则会创建相应的 bundle Task。需要注意的是,base moudle 是指包含功能的 moudle,而用于 test 的 moudle 则是不包含功能的。
        if (variantType.isBaseModule()) {
            taskManager.createBundleTask(variantData);
        }

        // 3、如果 variantType 是一个 test moudle(其作为一个 test 的组件),则会创建相应的 test variant。
        if (variantType.isTestComponent()) {
        
            // 1)、将 variant-specific, build type multi-flavor、defaultConfig 这些依赖添加到当前的 variantData 之中。
            ...
            
            // 2)、如果支持渲染脚本,则添加渲染脚本的依赖。
            if (testedVariantData.getVariantConfiguration().getRenderscriptSupportModeEnabled()) {
                project.getDependencies()
                        .add(
                                variantDep.getCompileClasspath().getName(),
                                project.files(
                                        globalScope
                                                .getSdkComponents()
                                                .getRenderScriptSupportJarProvider()));
            }
        
            // 3)、如果当前 Variant 会输出一个 APK,即当前是执行的一个 Android test(一般用来进行 UI 自动化测试),则会创建相应的 AndroidTestVariantTask。
            if (variantType.isApk()) { // ANDROID_TEST
                if (variantConfig.isLegacyMultiDexMode()) {
                    String multiDexInstrumentationDep =
                            globalScope.getProjectOptions().get(BooleanOption.USE_ANDROID_X)
                                    ? ANDROIDX_MULTIDEX_MULTIDEX_INSTRUMENTATION
                                    : COM_ANDROID_SUPPORT_MULTIDEX_INSTRUMENTATION;
                    project.getDependencies()
                            .add(
                                    variantDep.getCompileClasspath().getName(),
                                    multiDexInstrumentationDep);
                    project.getDependencies()
                            .add(
                                    variantDep.getRuntimeClasspath().getName(),
                                    multiDexInstrumentationDep);
                }

                taskManager.createAndroidTestVariantTasks(
                        (TestVariantData) variantData,
                        variantScopes
                                .stream()
                                .filter(TaskManager::isLintVariant)
                                .collect(Collectors.toList()));
            } else { // UNIT_TEST
                // 4)、否则说明该 Test moudle 是用于执行单元测试的,则会创建 UnitTestVariantTask。 taskManager.createUnitTestVariantTasks((TestVariantData) variantData);
            }

        } else {
            // 4、如果不是一个 Test moudle,则会调用 ApplicationTaskManager 的 createTasksForVariantScope 方法。
            taskManager.createTasksForVariantScope(
                    variantScope,
                    variantScopes
                            .stream()
                            .filter(TaskManager::isLintVariant)
                            .collect(Collectors.toList()));
        }
}

在 createTasksForVariantData 方法中为每一个指定的 Variant 类型创建了与之对应的 Tasks,该方法的处理逻辑如下所示:

1、创建 Assemble Task。

2、如果 variantType 是 base moudle,则会创建相应的 bundle Task。需要注意的是,base moudle 是指包含功能的 moudle,而用于 test 的 moudle 则是不包含功能的。

3、如果 variantType 是一个 test moudle(其作为一个 test 的组件),则会创建相应的 test variant。

1)、将 variant-specific, build type multi-flavor、defaultConfig 这些依赖添加到当前的 variantData 之中。

2)、如果支持渲染脚本,则添加渲染脚本的依赖。

3)、如果当前 Variant 会输出一个 APK,即当前是执行的一个 Android test(一般用来进行 UI 自动化测试),则会创建相应的 AndroidTestVariantTask。

4)、否则说明该 Test moudle 是用于执行单元测试的,则会创建 UnitTestVariantTask。

4、如果不是一个 Test moudle,则会调用 ApplicationTaskManager 的 createTasksForVariantScope 方法。

public void createTasksForVariantScope(
            @NonNull final VariantScope variantScope,
            @NonNull List<VariantScope> variantScopesForLint) {
        createAnchorTasks(variantScope);
        createCheckManifestTask(variantScope);

        handleMicroApp(variantScope);

        // Create all current streams (dependencies mostly at this point)
        createDependencyStreams(variantScope);

        // Add a task to publish the applicationId.
        createApplicationIdWriterTask(variantScope);

        taskFactory.register(new MainApkListPersistence.CreationAction(variantScope));
        createBuildArtifactReportTask(variantScope);

        // Add a task to process the manifest(s)
        createMergeApkManifestsTask(variantScope);

        // Add a task to create the res values
        createGenerateResValuesTask(variantScope);

        // Add a task to compile renderscript files.
        createRenderscriptTask(variantScope);

        // Add a task to merge the resource folders
        createMergeResourcesTask(
                variantScope,
                true,
                Sets.immutableEnumSet(MergeResources.Flag.PROCESS_VECTOR_DRAWABLES));

        // Add tasks to compile shader
        createShaderTask(variantScope);

        // Add a task to merge the asset folders
        createMergeAssetsTask(variantScope);

        // Add a task to create the BuildConfig class
        createBuildConfigTask(variantScope);

        // Add a task to process the Android Resources and generate source files
        createApkProcessResTask(variantScope);

        // Add a task to process the java resources
        createProcessJavaResTask(variantScope);

        createAidlTask(variantScope);

        // Add external native build tasks
        createExternalNativeBuildJsonGenerators(variantScope);
        createExternalNativeBuildTasks(variantScope);

        // Add a task to merge the jni libs folders
        createMergeJniLibFoldersTasks(variantScope);

        // Add feature related tasks if necessary
        if (variantScope.getType().isBaseModule()) {
            // Base feature specific tasks.
            taskFactory.register(new FeatureSetMetadataWriterTask.CreationAction(variantScope));

            createValidateSigningTask(variantScope);
            // Add a task to produce the signing config file.
            taskFactory.register(new SigningConfigWriterTask.CreationAction(variantScope));

            if (extension.getDataBinding().isEnabled()) {
                // Create a task that will package the manifest ids(the R file packages) of all
                // features into a file. This file's path is passed into the Data Binding annotation
                // processor which uses it to known about all available features.
                //
                // <p>see: {@link TaskManager#setDataBindingAnnotationProcessorParams(VariantScope)}
                taskFactory.register(
                        new DataBindingExportFeatureApplicationIdsTask.CreationAction(
                                variantScope));

            }
        } else {
            // Non-base feature specific task.
            // Task will produce artifacts consumed by the base feature
            taskFactory.register(
                    new FeatureSplitDeclarationWriterTask.CreationAction(variantScope));
            if (extension.getDataBinding().isEnabled()) {
                // Create a task that will package necessary information about the feature into a
                // file which is passed into the Data Binding annotation processor.
                taskFactory.register(
                        new DataBindingExportFeatureInfoTask.CreationAction(variantScope));
            }
            taskFactory.register(new MergeConsumerProguardFilesTask.CreationAction(variantScope));
        }

        // Add data binding tasks if enabled
        createDataBindingTasksIfNecessary(variantScope, MergeType.MERGE);

        // Add a compile task
        createCompileTask(variantScope);

        createStripNativeLibraryTask(taskFactory, variantScope);

        if (variantScope.getVariantData().getMultiOutputPolicy().equals(MultiOutputPolicy.SPLITS)) {
            if (extension.getBuildToolsRevision().getMajor() < 21) {
                throw new RuntimeException(
                        "Pure splits can only be used with buildtools 21 and later");
            }

            createSplitTasks(variantScope);
        }

        createPackagingTask(variantScope);

        maybeCreateLintVitalTask(
                (ApkVariantData) variantScope.getVariantData(), variantScopesForLint);

        // Create the lint tasks, if enabled
        createLintTasks(variantScope, variantScopesForLint);

        taskFactory.register(new FeatureSplitTransitiveDepsWriterTask.CreationAction(variantScope));

        createDynamicBundleTask(variantScope);
    }

我们可以聊聊变体。

在build.gradle里面可以使用flavorDimensions属性创建“mode”变种维度和“api”变种维度。

android {
  ...
  buildTypes {
    debug {...}
    release {...}
  }

  flavorDimensions "api", "mode"

  productFlavors {
    demo {
      // Assigns this product flavor to the "mode" flavor dimension.
      dimension "mode"
      ...
    }

    full {
      dimension "mode"
      ...
    }

    
    minApi24 {
      dimension "api"
      minSdkVersion 24
      versionCode 30000 + android.defaultConfig.versionCode
      versionNameSuffix "-minApi24"
      ...
    }

    minApi23 {
      dimension "api"
      minSdkVersion 23
      versionCode 20000  + android.defaultConfig.versionCode
      versionNameSuffix "-minApi23"
      ...
    }

    minApi21 {
      dimension "api"
      minSdkVersion 21
      versionCode 10000  + android.defaultConfig.versionCode
      versionNameSuffix "-minApi21"
      ...
    }
  }
}

Gradle 创建的 build 变体数量等于每个变种维度中的变种数量与您配置的 build 类型数量的乘积。当 Gradle 为每个 build 变体或对应的 APK 命名时,先显示属于较高优先级变种维度的产品变种,接着是较低优先级维度中的产品变种,再接着是 build 类型。以上面的构建配置为例,Gradle 使用以下命名方案创建了总共 12 个 build 变体:

build 变体:[minApi24, minApi23, minApi21][Demo, Full][Debug, Release]

对应的 APK:app-[minApi24, minApi23, minApi21]-[demo, full]-[debug, release].apk

相关文章

网友评论

      本文标题:Gradle Plugin 源码分析

      本文链接:https://www.haomeiwen.com/subject/tnvnrltx.html