All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.bugsnag.android.gradle.BugsnagPlugin.groovy Maven / Gradle / Ivy

There is a newer version: 8.1.0
Show newest version
package com.bugsnag.android.gradle

import com.android.build.gradle.AppPlugin
import com.android.build.gradle.LibraryPlugin
import com.android.build.gradle.api.ApplicationVariant
import com.android.build.gradle.api.BaseVariant
import com.android.build.gradle.api.BaseVariantOutput
import com.android.build.gradle.api.LibraryVariant
import com.android.build.gradle.tasks.ManifestProcessorTask
import com.android.build.gradle.tasks.PackageApplication
import org.gradle.api.DomainObjectSet
import org.gradle.api.Plugin
import org.gradle.api.Project
import org.gradle.api.Task
import org.gradle.api.artifacts.Configuration
import org.gradle.api.artifacts.Dependency
import org.gradle.util.VersionNumber

/**
 * Gradle plugin to automatically upload ProGuard mapping files to Bugsnag.
 *
 * This plugin creates Gradle Tasks, and hooks them into a typical build
 * process. Knowledge of the Android build lifecycle is required to
 * understand how we attach tasks as dependencies.
 *
 * Run `gradle tasks --all` in an Android app project to see all tasks and
 * dependencies.
 *
 * Further reading:
 * https://sites.google.com/a/android.com/tools/tech-docs/new-build-system/user-guide#TOC-Build-Tasks
 * https://docs.gradle.org/current/userguide/custom_tasks.html
 */
class BugsnagPlugin implements Plugin {

    static final String API_KEY_TAG = 'com.bugsnag.android.API_KEY'
    static final String BUILD_UUID_TAG = 'com.bugsnag.android.BUILD_UUID'
    static final String VERSION_CODE_TAG = 'com.bugsnag.android.VERSION_CODE'
    static final String APP_VERSION_TAG = 'com.bugsnag.android.APP_VERSION'
    static final String GROUP_NAME = 'Bugsnag'

    private static final String NDK_PROJ_TASK = "externalNative"
    private static final String CLEAN_TASK = "Clean"
    private static final String ASSEMBLE_TASK = "assemble"
    private static final String BUNDLE_TASK = "bundle"

    VersionNumber bugsnagVersionNumber

    void apply(Project project) {
        project.extensions.create("bugsnag", BugsnagPluginExtension)
        project.bugsnag.extensions.create("sourceControl", SourceControl)

        project.afterEvaluate {
            bugsnagVersionNumber = getBugsnagAndroidVersionNumber(project)
            project.logger.debug("Using bugsnag-android version number: $bugsnagVersionNumber")

            // Make sure the android plugin has been applied first
            if (project.plugins.hasPlugin(AppPlugin)) {
                project.android.applicationVariants.all { variant ->
                    applyBugsnagToVariant(variant, project)
                }
            } else if (project.plugins.hasPlugin(LibraryPlugin)) {
                project.android.libraryVariants.all  { variant ->
                    applyBugsnagToVariant(variant, project)
                }
            } else {
                throw new IllegalStateException('Must apply \'com.android.application\' first!')
            }

            if (isNdkProject(project)) {
                setupNdkProject(project)
            }
        }
    }

    /**
     * Retrieves the VersionNumber used by com.bugsnag.android in the given project. This can be used
     * to conditionally perform tasks depending on the artefact version.
     */
    static VersionNumber getBugsnagAndroidVersionNumber(Project project) {
        List configs = project.configurations.collect()
        List deps = configs.stream()
            .map { conf -> conf.allDependencies }
            .collect()
            .flatten()

        Optional bugsnagVersion = deps.stream()
            .filter { dep -> dep.group == "com.bugsnag" && dep.name == "bugsnag-android" }
            .distinct()
            .map({ dep -> dep.version })
            .findFirst()

        bugsnagVersion.present ? VersionNumber.parse(bugsnagVersion.get()) : VersionNumber.UNKNOWN
    }

    private static void setupNdkProject(Project project) {
        Set cleanTasks = project.tasks.findAll {
            it.name.startsWith(NDK_PROJ_TASK) && it.name.contains(CLEAN_TASK)
        }
        Set buildTasks = project.tasks.findAll {
            it.name.startsWith(NDK_PROJ_TASK) && !it.name.contains(CLEAN_TASK)
        }

        BugsnagNdkSetupTask ndkSetupTask = project.tasks.create("bugsnagInstallJniLibsTask", BugsnagNdkSetupTask)

        buildTasks.forEach {
            ndkSetupTask.mustRunAfter(cleanTasks)
            it.dependsOn ndkSetupTask
            it.doFirst { ndkSetupTask }
        }
    }

    /**
     * Create tasks for each Build Variant
     * See https://sites.google.com/a/android.com/tools/tech-docs/new-build-system/user-guide#TOC-Build-Variants
     */
    private void applyBugsnagToVariant(BaseVariant variant, Project project) {
        if (hasDisabledBugsnag(variant)) {
            return
        }

        // only need to be run once per variant
        setupProguardAutoConfig(project, variant)

        variant.outputs.each { output ->
            BugsnagTaskDeps deps = new BugsnagTaskDeps()
            deps.variant = variant
            deps.output = output

            setupManifestUuidTask(project, deps)

            if (variant.buildType.minifyEnabled || hasDexguardPlugin(project)) {
                setupMappingFileUpload(project, deps)
            }

            setupNdkMappingFileUpload(project, deps)
            setupReleasesTask(project, deps)
        }
    }

    private static boolean shouldUploadDebugMappings(Project project, BaseVariantOutput output) {
        !output.name.toLowerCase().endsWith("debug") || project.bugsnag.uploadDebugBuildMappings
    }

    /**
     * Creates a bugsnag task to upload proguard mapping file
     */
    private static void setupMappingFileUpload(Project project, BugsnagTaskDeps deps) {
        String taskName = "uploadBugsnag${taskNameForOutput(deps.output)}Mapping"
        BugsnagUploadProguardTask uploadTask = project.tasks.create(taskName, BugsnagUploadProguardTask)
        uploadTask.partName = "proguard"
        prepareUploadTask(uploadTask, deps, project)
    }

    private static void setupNdkMappingFileUpload(Project project, BugsnagTaskDeps deps) {
        if (isNdkProject(project)) {
            // Create a Bugsnag task to upload NDK mapping file(s)
            String taskName = "uploadBugsnagNdk${taskNameForOutput(deps.output)}Mapping"
            BugsnagUploadNdkTask uploadNdkTask = project.tasks.create(taskName, BugsnagUploadNdkTask)
            prepareUploadTask(uploadNdkTask, deps, project)

            uploadNdkTask.variantName = taskNameForVariant(deps.variant)
            uploadNdkTask.projectDir = project.projectDir
            uploadNdkTask.rootDir = project.rootDir
            uploadNdkTask.sharedObjectPath = project.bugsnag.sharedObjectPath
        }
    }

    private static boolean isNdkProject(Project project) {
        if (project.bugsnag.ndk != null) { // always respect user override
            return project.bugsnag.ndk
        } else { // infer whether native build or not
            boolean usesCmake = project.android.externalNativeBuild.cmake.path != null
            boolean usesNdkBuild = project.android.externalNativeBuild.ndkBuild.path != null
            return usesCmake || usesNdkBuild
        }
    }

    private static void setupReleasesTask(Project project, BugsnagTaskDeps deps) {
        String taskName = "bugsnagRelease${taskNameForOutput(deps.output)}Task"
        BugsnagReleasesTask releasesTask = project.tasks.create(taskName, BugsnagReleasesTask)
        setupBugsnagTask(releasesTask, deps)

        if (shouldUploadDebugMappings(project, deps.output)) {
            findAssembleBundleTasks(deps.variant, deps.output, project).forEach {
                releasesTask.mustRunAfter it

                if (project.bugsnag.autoReportBuilds) {
                    it.finalizedBy releasesTask
                }
            }
        }
    }

    private static void setupBugsnagTask(BugsnagVariantOutputTask task, BugsnagTaskDeps deps) {
        task.group = GROUP_NAME
        task.variantOutput = deps.output
        task.variant = deps.variant
    }

    private static void prepareUploadTask(BugsnagMultiPartUploadTask uploadTask,
                                          BugsnagTaskDeps deps, Project project) {
        setupBugsnagTask(uploadTask, deps)
        uploadTask.applicationId = deps.variant.applicationId

        if (shouldUploadDebugMappings(project, deps.output)) {
            findAssembleBundleTasks(deps.variant, deps.output, project).forEach {
                uploadTask.mustRunAfter it

                if (project.bugsnag.autoUpload) {
                    it.finalizedBy uploadTask
                }
            }
        }
    }

    /**
     * Fetches all the assemble and bundle tasks in the current project that match the variant
     *
     * Expected behaviour: [assemble, assembleJavaExampleRelease, assembleJavaExample, assembleRelease,
     * bundle, bundleJavaExampleRelease, bundleJavaExample, bundleRelease]
     *
     * @param output the variantOutput
     * @param project the current project
     * @return the assemble + bundle tasks
     */
    private static Set findAssembleBundleTasks(BaseVariant variant, BaseVariantOutput output, Project project) {
        Set taskNames = new HashSet<>()
        taskNames.addAll(findTaskNamesForPrefix(variant, output, ASSEMBLE_TASK))
        taskNames.addAll(findTaskNamesForPrefix(variant, output, BUNDLE_TASK))

        project.tasks.findAll {
            taskNames.contains(it.name)
        }
    }

    /**
     * Finds all the task names which can be used to assemble a variant, and replaces 'assemble' with the given
     * prefix.
     *
     * E.g. [bundle, bundleRelease, bundleFooRelease]
     */
    private static Set findTaskNamesForPrefix(BaseVariant variant, BaseVariantOutput output, String prefix) {
        String variantName = output.name.split("-")[0].capitalize()
        Task assembleTask = resolveAssembleTask(variant)

        Set taskNames = new HashSet<>()
        taskNames.add(prefix)

        if (assembleTask != null) {
            String assembleTaskName = assembleTask.name
            String buildTypeTaskName = assembleTaskName.replaceAll(variantName, "")
            String buildType = buildTypeTaskName.replaceAll(ASSEMBLE_TASK, "")
            String variantTaskName = assembleTaskName.replaceAll(buildType, "")

            taskNames.add(assembleTaskName.replaceAll(ASSEMBLE_TASK, prefix))
            taskNames.add(buildTypeTaskName.replaceAll(ASSEMBLE_TASK, prefix))
            taskNames.add(variantTaskName.replaceAll(ASSEMBLE_TASK, prefix))
        }
        taskNames
    }

    private static Task resolveAssembleTask(BaseVariant variant) {
        try {
            return variant.assembleProvider.getOrNull()
        } catch (Throwable ignored) {
            return variant.assemble
        }
    }

    private static void setupManifestUuidTask(Project project, BugsnagTaskDeps deps) {
        String taskName = "processBugsnag${taskNameForOutput(deps.output)}Manifest"
        BugsnagManifestTask manifestTask = project.tasks.create(taskName, BugsnagManifestTask)
        setupBugsnagTask(manifestTask, deps)
        ManifestProcessorTask processManifest = resolveProcessManifest(deps.output)

        if (processManifest == null) {
            return
        }

        processManifest.finalizedBy(manifestTask)
        manifestTask.dependsOn(processManifest)

        Set resourceTasks = project.tasks.findAll {
            String name = it.name.toLowerCase()
            name.startsWith(BUNDLE_TASK) && name.endsWith("resources")
        }

        resourceTasks.forEach {
            it.dependsOn manifestTask
        }
    }

    static ManifestProcessorTask resolveProcessManifest(BaseVariantOutput output) {
        try {
            return output.processManifestProvider.getOrNull()
        } catch (Throwable ignored) {
            return output.processManifest
        }
    }

    static File resolveBundleManifestOutputDirectory(ManifestProcessorTask processManifest) {
        if (processManifest.hasProperty("bundleManifestOutputDirectory")) {

            // For AGP versions >= 3.3.0 the bundle manifest is output to its own directory
            def directory = processManifest.bundleManifestOutputDirectory

            if (directory instanceof File) { // 3.3.X - 3.5.X returns a File
                return directory
            } else { // 3.6.+ returns a DirectoryProperty
                return directory.asFile.getOrNull()
            }

        } else {
            // For AGP versions < 3.3.0 the bundle manifest is the merged manifest
            return processManifest.manifestOutputDirectory
        }
    }

    /**
     * Automatically add the "edit proguard settings" task to the
     * build process.
     *
     * This task must be called before ProGuard is run, but since
     * the name of the ProGuard task changed between 1.0 and 1.5
     * of the Android build tools, we'll hook into the "package"
     * task as a dependency, since this is always run before
     * ProGuard.
     *
     * For reference, in Android Build Tools 1.0, the ProGuard
     * task was named `proguardRelease`, and in 1.5+ the ProGuard
     * task is named `transformClassesAndResourcesWithProguardForRelease`
     * as it is now part of the "transforms" process.
     */
    private void setupProguardAutoConfig(Project project, BaseVariant variant) {
        String taskname = "processBugsnag${taskNameForVariant(variant)}Proguard"
        BugsnagProguardConfigTask proguardConfigTask = project.tasks.create(taskname, BugsnagProguardConfigTask)
        proguardConfigTask.group = GROUP_NAME
        proguardConfigTask.variant = variant

        // consumer proguard rules were added to the library in 4.6.0
        boolean hasConsumerRules = bugsnagVersionNumber.major >= 4 && bugsnagVersionNumber.minor >= 6

        if (project.bugsnag.autoProguardConfig && !hasConsumerRules) {
            project.logger.debug("Bugsnag autoproguard config enabled")
            dependTaskOnPackageTask(variant, proguardConfigTask)
        } else {
            project.logger.debug("ProGuard has consumer rules, skipping write")
        }
    }

    static String taskNameForVariant(BaseVariant variant) {
        variant.name.capitalize()
    }

    static String taskNameForOutput(BaseVariantOutput output) {
        output.name.capitalize()
    }

    private static void dependTaskOnPackageTask(BaseVariant variant, Task task) {
        if (variant instanceof LibraryVariant) {
            variant.packageLibrary.dependsOn task
        } else {
            PackageApplication application = resolvePackageApplication(variant)

            if (application != null) {
                application.dependsOn task
            }
        }
    }

    static PackageApplication resolvePackageApplication(BaseVariant variant) {
        try {
            return variant.packageApplicationProvider.getOrNull()
        } catch (Throwable ignored) {
            return variant.packageApplication
        }
    }

    private static boolean hasDisabledBugsnag(BaseVariant variant) {
        Closure hasDisabledBugsnag = {
            it.ext.properties.containsKey("enableBugsnag") && !it.ext.enableBugsnag
        }

        // Ignore any conflicting properties, bail if anything has a disable flag.
        (variant.productFlavors + variant.buildType).any(hasDisabledBugsnag)
    }

    /**
     * Returns true if the DexGuard plugin has been applied to the project
     */
    static boolean hasDexguardPlugin(Project project) {
        project.pluginManager.hasPlugin("dexguard")
    }

    /**
     * Returns true if a project has configured multiple variant outputs.
     *
     * This calculation is based on a heuristic - the number of variantOutputs in a project must be
     * greater than the number of variants.
     */
    static boolean hasMultipleOutputs(Project project) {
        DomainObjectSet variants = project.android.applicationVariants
        int variantSize = variants.size()
        int outputSize = 0

        variants.forEach { variant ->
            outputSize += variant.outputs.size()
        }
        outputSize > variantSize
    }

    /**
     * Whether or not an assemble task is going to be run for this variant
     */
    static boolean isRunningAssembleTask(BaseVariant variant, BaseVariantOutput output, Project project) {
        isRunningTaskWithPrefix(variant, output, project, ASSEMBLE_TASK)
    }

    /**
     * Whether or not a bundle task is going to be run for this variant
     */
    static boolean isRunningBundleTask(BaseVariant variant, BaseVariantOutput output, Project project) {
        isRunningTaskWithPrefix(variant, output, project, BUNDLE_TASK)
    }

    /**
     * Whether or any of a list of the task names for a prefix are going to be run by checking the list
     * against all of the tasks in the task graph
     */
    private static boolean isRunningTaskWithPrefix(BaseVariant variant,
                                                   BaseVariantOutput output, Project project, String prefix) {
        Set taskNames = new HashSet<>()
        taskNames.addAll(findTaskNamesForPrefix(variant, output, prefix))

        project.gradle.taskGraph.allTasks.any { task ->
            taskNames.any {
                task.name.endsWith(it)
            }
        }
    }

    private static class BugsnagTaskDeps {
        BaseVariant variant
        BaseVariantOutput output
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy