Before, a friend of mine was asked some technical questions in the face of Andrew Gang from the headquarters. He felt it was still very difficult and very skillful. It feels like development at hq is still pretty strong.

  1. Customize the order in which transforms and other system transforms are executed
  2. Relationship between Transform and Task
  3. How is the Transform executed

What is a Task

In fact, the core is to start from what is Task.

A Task represents an atomic operation of a build job, such as compiling calsses or generating Javadoc.

In Gradle, each Project to be compiled is called a Project. Each Project is built with a set of tasks. For example, an Android APK compilation might include: Java source compiled tasks, resource compiled tasks, JNI compiled tasks, Lint checked tasks, packaged apK-generated tasks, signed tasks, etc. The plug-in itself contains several tasks.

To put it simply, our project compiled using assembleDebug as an example will execute many gradle tasks sequentially. For example, aapt, Javac, Kotlinc, etc. They all exist as a task.

The Transform of AGP

   AppExtension appExtension = project.getExtensions().getByType(AppExtension.class);
   appExtension.registerTransform(new DoubleTabTransform(project));
Copy the code

When we write a plugin that contains a Transform, we just register a Transform in Android AppExtension. So what is the essence of a Transform?

High-energy warning, the following source code is longer, you can consider skipping the conclusion directly, but understand the students can learn the best.

public class LibraryTaskManager extends TaskManager {

    @Override
    public void createTasksForVariantScope(@NonNull final VariantScope variantScope) {...// ----- External Transforms -----
        // apply all the external transforms.
        List<Transform> customTransforms = extension.getTransforms();
        List<List<Object>> customTransformsDependencies = extension.getTransformsDependencies();

        for (int i = 0, count = customTransforms.size(); i < count; i++) {
            Transform transform = customTransforms.get(i);

            // Check the transform only applies to supported scopes for libraries:
            // We cannot transform scopes that are not packaged in the library
            // itself.
            Sets.SetView<? super Scope> difference =
                    Sets.difference(transform.getScopes(), TransformManager.PROJECT_ONLY);
            if(! difference.isEmpty()) { String scopes = difference.toString(); globalScope .getAndroidBuilder() .getIssueReporter() .reportError( Type.GENERIC,new EvalIssueException(
                                        String.format(
                                                "Transforms with scopes '%s' cannot be applied to library projects.",
                                                scopes)));
            }

            List<Object> deps = customTransformsDependencies.get(i);
            transformManager.addTransform(
                    taskFactory,
                    variantScope,
                    transform,
                    null,
                    task -> {
                        if(! deps.isEmpty()) { task.dependsOn(deps); } }, taskProvider -> {// if the task is a no-op then we make assemble task
                        // depend on it.
                        if(transform.getScopes().isEmpty()) { TaskFactoryUtils.dependsOn( variantScope.getTaskContainer().getAssembleTask(), taskProvider); }}); }// Now add transforms for intermediate publishing (projects to projects).
        File jarOutputFolder = variantScope.getIntermediateJarOutputFolder();
        File mainClassJar = new File(jarOutputFolder, FN_CLASSES_JAR);
        File mainResJar = new File(jarOutputFolder, FN_INTERMEDIATE_RES_JAR);
        LibraryIntermediateJarsTransform intermediateTransform =
                new LibraryIntermediateJarsTransform(
                        mainClassJar,
                        mainResJar,
                        variantConfig::getPackageFromManifest,
                        extension.getPackageBuildConfig());
        excludeDataBindingClassesIfNecessary(variantScope, intermediateTransform);

        BuildArtifactsHolder artifacts = variantScope.getArtifacts();
        transformManager.addTransform(
                taskFactory,
                variantScope,
                intermediateTransform,
                taskName -> {
                    // publish the intermediate classes.jar
                    artifacts.appendArtifact(
                            InternalArtifactType.LIBRARY_CLASSES,
                            ImmutableList.of(mainClassJar),
                            taskName);
                    // publish the res jar
                    artifacts.appendArtifact(
                            InternalArtifactType.LIBRARY_JAVA_RES,
                            ImmutableList.of(mainResJar),
                            taskName);
                },
                null.null);

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

        // Create a jar with both classes and java resources. This artifact is not
        // used by the Android application plugin and the task usually don't need to
        // be executed. The artifact is useful for other Gradle users who needs the
        // 'jar' artifact as API dependency.
        taskFactory.register(new ZipMergingTask.CreationAction(variantScope));

        // now add a transform that will take all the native libs and package
        // them into an intermediary folder. This processes only the PROJECT
        // scope.
        final File intermediateJniLibsFolder = new File(jarOutputFolder, FD_JNI);

        LibraryJniLibsTransform intermediateJniTransform =
                new LibraryJniLibsTransform(
                        "intermediateJniLibs",
                        intermediateJniLibsFolder,
                        TransformManager.PROJECT_ONLY);
        transformManager.addTransform(
                taskFactory,
                variantScope,
                intermediateJniTransform,
                taskName -> {
                    // publish the jni folder as intermediate
                    variantScope
                            .getArtifacts()
                            .appendArtifact(
                                    InternalArtifactType.LIBRARY_JNI,
                                    ImmutableList.of(intermediateJniLibsFolder),
                                    taskName);
                },
                null.null);

        // Now go back to fill the pipeline with transforms used when
        // publishing the AAR

        // first merge the resources. This takes the PROJECT and LOCAL_DEPS
        // and merges them together.
        createMergeJavaResTransform(variantScope);

        // ----- Minify next -----
        maybeCreateJavaCodeShrinkerTransform(variantScope);
        maybeCreateResourcesShrinkerTransform(variantScope);

        // now add a transform that will take all the class/res and package them
        // into the main and secondary jar files that goes in the AAR.
        // This transform technically does not use its transform output, but that's
        // ok. We use the transform mechanism to get incremental data from
        // the streams.
        // This is used for building the AAR.

        File classesJar = variantScope.getAarClassesJar();
        File libsDirectory = variantScope.getAarLibsDirectory();

        LibraryAarJarsTransform transform =
                new LibraryAarJarsTransform(
                        classesJar,
                        libsDirectory,
                        artifacts.hasArtifact(InternalArtifactType.ANNOTATIONS_TYPEDEF_FILE)
                                ? artifacts.getFinalArtifactFiles(
                                        InternalArtifactType.ANNOTATIONS_TYPEDEF_FILE)
                                : null,
                        variantConfig::getPackageFromManifest,
                        extension.getPackageBuildConfig());

        excludeDataBindingClassesIfNecessary(variantScope, transform);

        transformManager.addTransform(
                taskFactory,
                variantScope,
                transform,
                taskName -> {
                    variantScope
                            .getArtifacts()
                            .appendArtifact(
                                    InternalArtifactType.AAR_MAIN_JAR,
                                    ImmutableList.of(classesJar),
                                    taskName);
                    variantScope
                            .getArtifacts()
                            .appendArtifact(
                                    InternalArtifactType.AAR_LIBS_DIRECTORY,
                                    ImmutableList.of(libsDirectory),
                                    taskName);
                },
                null.null);

        // now add a transform that will take all the native libs and package
        // them into the libs folder of the bundle. This processes both the PROJECT
        // and the LOCAL_PROJECT scopes
        final File jniLibsFolder =
                variantScope.getIntermediateDir(InternalArtifactType.LIBRARY_AND_LOCAL_JARS_JNI);
        LibraryJniLibsTransform jniTransform =
                new LibraryJniLibsTransform(
                        "syncJniLibs",
                        jniLibsFolder,
                        TransformManager.SCOPE_FULL_LIBRARY_WITH_LOCAL_JARS);
        transformManager.addTransform(
                taskFactory,
                variantScope,
                jniTransform,
                taskName ->
                        variantScope
                                .getArtifacts()
                                .appendArtifact(
                                        InternalArtifactType.LIBRARY_AND_LOCAL_JARS_JNI,
                                        ImmutableList.of(jniLibsFolder),
                                        taskName),
                null.null); createLintTasks(variantScope); createBundleTask(variantScope); }}Copy the code

Customize the order in which transforms and other system transforms are executed

As can be seen from the above method, tasks are still executed sequentially according to the topological ordering of DAG (directed acyclic graph). It will contain some system transforms, the order of which may be inserted before the custom Transform, and some after all Tranform executions. Such as LibraryJniLibsTransform.

Topological ordering refers to ordering a Directed Acyclic Graph (DAG) to obtain an ordered linear sequence.

For example, A project consists of four sub-parts A, B, C, and D to complete, and A depends on B and D, and C depends on D. Now make A plan and write out the sequence of A, B, C, and D. This is where topological sorting comes into play, which is to determine the order in which things happen. In topological sorting, if there is A path from vertex A to vertex B, then B appears after A in the sorting result.

Relationship between Transform and Task

From this part of the source code in fact we can see, we registered to AppExtension Transform, after all the Transform will createTasksForVariantScope method calls, The taskManger addTransform method is then added.

@NonNull
 public <T extends Transform> Optional<TaskProvider<TransformTask>> addTransform(
         @NonNull TaskFactory taskFactory,
         @NonNull TransformVariantScope scope,
         @NonNull T transform,
         @Nullable PreConfigAction preConfigAction,
         @Nullable TaskConfigAction<TransformTask> configAction,
         @Nullable TaskProviderCallback<TransformTask> providerCallback) {

     if(! validateTransform(transform)) {// validate either throws an exception, or records the problem during sync
         // so it's safe to just return null here.
         return Optional.empty();
     }

     List<TransformStream> inputStreams = Lists.newArrayList();
     String taskName = scope.getTaskName(getTaskNamePrefix(transform));

     // get referenced-only streams
     List<TransformStream> referencedStreams = grabReferencedStreams(transform);

     // find input streams, and compute output streams for the transform.
     IntermediateStream outputStream = findTransformStreams(
             transform,
             scope,
             inputStreams,
             taskName,
             scope.getGlobalScope().getBuildDir());

     if (inputStreams.isEmpty() && referencedStreams.isEmpty()) {
         // didn't find any match. Means there is a broken order somewhere in the streams.
         issueReporter.reportError(
                 Type.GENERIC,
                 new EvalIssueException(
                         String.format(
                                 "Unable to add Transform '%s' on variant '%s': requested streams not available: %s+%s / %s",
                                 transform.getName(),
                                 scope.getFullVariantName(),
                                 transform.getScopes(),
                                 transform.getReferencedScopes(),
                                 transform.getInputTypes())));
         return Optional.empty();
     }

     //noinspection PointlessBooleanExpression
     if (DEBUG && logger.isEnabled(LogLevel.DEBUG)) {
         logger.debug("ADDED TRANSFORM(" + scope.getFullVariantName() + "):");
         logger.debug("\tName: " + transform.getName());
         logger.debug("\tTask: " + taskName);
         for (TransformStream sd : inputStreams) {
             logger.debug("\tInputStream: " + sd);
         }
         for (TransformStream sd : referencedStreams) {
             logger.debug("\tRef'edStream: " + sd);
         }
         if(outputStream ! =null) {
             logger.debug("\tOutputStream: " + outputStream);
         }
     }

     transforms.add(transform);

     // create the task...
     return Optional.of(
             taskFactory.register(
                     new TransformTask.CreationAction<>(
                             scope.getFullVariantName(),
                             taskName,
                             transform,
                             inputStreams,
                             referencedStreams,
                             outputStream,
                             recorder),
                     preConfigAction,
                     configAction,
                     providerCallback));
 }
Copy the code

And then we see transformManager addTransform, direct observation of the last line of code, found that we can to directly register a TransformTask taskFactory, then answer the first question, A Transform is an atomic Task in gradle Plugin, so Transform==Task.

How is the Transform executed

A Transform will be executed just like a normal Task, in a specific order of dependencies. Each Task is a basic atomic operation in a Gradle Project. So the order of execution of the Transform is the order of execution of the Task.

How to play the Task

Some caution should be shared before writing resources to confuse compilation speed optimization.

First of all, we need to find the pre-task dependency of the Task after constructing a Task. Then we can execute the actual Task content in the thread pool and make full use of the advantages of multi-threading to construct a project. Then what we need to do is to obtain the Task before executing the post-dependent Task. After that, do an await operation in the Task doFirst method, so that you can optimize a Task by taking full advantage of CPU and multithreading.

val manager = scope.transformManager
           val field = manager.javaClass.getDeclaredField("transforms").apply {
               isAccessible = true
           }
           val list = field.get(manager) as List<Transform>
           list.forEach {
               if (it is ShrinkResourcesTransform) {
                   val taskName = scope.getTaskName(getTaskNamePrefix(it))
                   val task = project.tasks.getByName(taskName)
                   task.doFirst {
                       val testTask = project.tasks.getByName(resGuardTaskName)
                               as TestTask
                       val taskLogger = TimeUsageHelper()
                       taskLogger.info("AwaitTask start ")
                       resProguardTask.await()
                       taskLogger.logTotalInfo("AwaitTask finish ")
                       Log.writeToFile(project)
                   }
                   task.dependsOn(testTask)
               }
           }
Copy the code

The above code is to get a Transform task of the system and then await a task before the task.

conclusion

In fact, these three topics are interesting, you can take a look at the source code to analyze some of them, and in fact, if you read carefully, you can find some interesting tasks in the above source code, Resources such as mergeJavaResource (merger), ShrinkResourcesTransform (delete useless resources), and so on.

My blog is handling synchronization to tencent cloud + community, invite everyone to come together: cloud.tencent.com/developer/s…