summaryrefslogtreecommitdiffstats
path: root/build.gradle
diff options
context:
space:
mode:
Diffstat (limited to '')
-rw-r--r--build.gradle407
1 files changed, 407 insertions, 0 deletions
diff --git a/build.gradle b/build.gradle
new file mode 100644
index 0000000000..82af965f00
--- /dev/null
+++ b/build.gradle
@@ -0,0 +1,407 @@
+import org.tomlj.Toml
+import org.tomlj.TomlParseResult
+import org.tomlj.TomlTable
+
+buildscript {
+ repositories {
+ gradle.mozconfig.substs.GRADLE_MAVEN_REPOSITORIES.each { repository ->
+ maven {
+ url repository
+ if (gradle.mozconfig.substs.ALLOW_INSECURE_GRADLE_REPOSITORIES) {
+ allowInsecureProtocol = true
+ }
+ }
+ }
+ }
+
+ ext.kotlin_version = '1.8.21'
+
+ dependencies {
+ classpath 'org.mozilla.apilint:apilint:0.5.2'
+ classpath 'com.android.tools.build:gradle:7.4.2'
+ classpath 'org.apache.commons:commons-exec:1.3'
+ classpath 'com.diffplug.spotless:spotless-plugin-gradle:6.18.0'
+ classpath 'org.tomlj:tomlj:1.1.0'
+ classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:$kotlin_version"
+ }
+}
+
+def tryInt = { string ->
+ if (string == null) {
+ return string
+ }
+ if (string.isInteger()) {
+ return string as Integer
+ }
+ return string
+}
+
+// Parses the Cargo.lock and returns the version for the given package name.
+def getRustVersionFor(packageName) {
+ String version = null;
+ TomlParseResult result = Toml.parse(file("Cargo.lock").getText());
+ for (object in result.getArray("package").toList()) {
+ def table = (TomlTable) object
+ if (table.getString("name") == packageName) {
+ if (version != null) {
+ throw new StopExecutionException("Multiple versions for '${packageName}' found." +
+ " Ensure '${packageName}' is only included once.")
+ }
+ version = table.getString("version")
+ }
+ }
+ return version
+}
+
+allprojects {
+ // Expose the per-object-directory configuration to all projects.
+ ext {
+ mozconfig = gradle.mozconfig
+ topsrcdir = gradle.mozconfig.topsrcdir
+ topobjdir = gradle.mozconfig.topobjdir
+
+ gleanVersion = "57.0.0"
+ if (gleanVersion != getRustVersionFor("glean")) {
+ throw new StopExecutionException("Mismatched Glean version, expected: ${gleanVersion}," +
+ " found ${getRustVersionFor("glean")}")
+ }
+
+ artifactSuffix = getArtifactSuffix()
+ versionName = getVersionName()
+ versionCode = computeVersionCode()
+ versionNumber = getVersionNumber()
+ buildId = getBuildId()
+
+ buildToolsVersion = mozconfig.substs.ANDROID_BUILD_TOOLS_VERSION
+ compileSdkVersion = tryInt(mozconfig.substs.ANDROID_TARGET_SDK)
+ targetSdkVersion = tryInt(mozconfig.substs.ANDROID_TARGET_SDK)
+ minSdkVersion = tryInt(mozconfig.substs.MOZ_ANDROID_MIN_SDK_VERSION)
+ manifestPlaceholders = [
+ ANDROID_PACKAGE_NAME: mozconfig.substs.ANDROID_PACKAGE_NAME,
+ ANDROID_TARGET_SDK: mozconfig.substs.ANDROID_TARGET_SDK,
+ MOZ_ANDROID_MIN_SDK_VERSION: mozconfig.substs.MOZ_ANDROID_MIN_SDK_VERSION,
+ ]
+ }
+
+ repositories {
+ gradle.mozconfig.substs.GRADLE_MAVEN_REPOSITORIES.each { repository ->
+ maven {
+ url repository
+ if (gradle.mozconfig.substs.ALLOW_INSECURE_GRADLE_REPOSITORIES) {
+ allowInsecureProtocol = true
+ }
+ }
+ }
+ }
+
+ // Use the semanticdb-javac and semanticdb-kotlinc plugins to generate semanticdb files for Searchfox
+ if (mozconfig.substs.ENABLE_MOZSEARCH_PLUGIN || mozconfig.substs.DOWNLOAD_ALL_GRADLE_DEPENDENCIES) {
+ def targetRoot = new File(topobjdir, "mozsearch_java_index")
+ def semanticdbJavacVersion = "com.sourcegraph:semanticdb-javac:0.9.6"
+ def semanticdbKotlincVersion = "com.sourcegraph:semanticdb-kotlinc:0.3.2"
+
+ afterEvaluate {
+ def addDependencyToConfigurationIfExists = { configurationName, dependency ->
+ def configuration = configurations.findByName(configurationName)
+ if (configuration != null) {
+ dependencies.add(configurationName, dependency)
+ }
+ }
+
+ addDependencyToConfigurationIfExists("compileOnly", semanticdbJavacVersion)
+ addDependencyToConfigurationIfExists("testCompileOnly", semanticdbJavacVersion)
+ addDependencyToConfigurationIfExists("androidTestCompileOnly", semanticdbJavacVersion)
+ addDependencyToConfigurationIfExists("kotlinCompilerPluginClasspath", semanticdbKotlincVersion)
+ }
+
+ tasks.withType(JavaCompile) {
+ options.compilerArgs += [
+ "-Xplugin:semanticdb -sourceroot:${topsrcdir} -targetroot:${targetRoot}",
+ ]
+ }
+
+ tasks.withType(org.jetbrains.kotlin.gradle.tasks.KotlinCompile) {
+ compilerOptions.freeCompilerArgs.addAll([
+ "-P", "plugin:semanticdb-kotlinc:sourceroot=${topsrcdir}",
+ "-P", "plugin:semanticdb-kotlinc:targetroot=${targetRoot}",
+ ])
+ }
+ }
+
+ task downloadDependencies() {
+ description 'Download all dependencies to the Gradle cache'
+ doLast {
+ configurations.each { configuration ->
+ if (configuration.canBeResolved) {
+ configuration.allDependencies.each { dependency ->
+ try {
+ configuration.files(dependency)
+ } catch(e) {
+ println("Could not resolve ${configuration.name} -> ${dependency.name}")
+ println(" > ${e.message}")
+ if (e.cause) {
+ println(" >> ${e.cause}")
+ if (e.cause.cause) {
+ println(" >> ${e.cause.cause}")
+ }
+ }
+ println("")
+ }
+ }
+ }
+ }
+ }
+ }
+}
+
+buildDir "${topobjdir}/gradle/build"
+
+// A stream that processes bytes line by line, prepending a tag before sending
+// each line to Gradle's logging.
+class TaggedLogOutputStream extends org.apache.commons.exec.LogOutputStream {
+ String tag
+ Logger logger
+
+ TaggedLogOutputStream(tag, logger) {
+ this.tag = tag
+ this.logger = logger
+ }
+
+ void processLine(String line, int level) {
+ logger.lifecycle("${this.tag} ${line}")
+ }
+}
+
+ext.geckoBinariesOnlyIf = { task ->
+ // Never when Gradle was invoked within `mach build`.
+ if ('1' == System.env.GRADLE_INVOKED_WITHIN_MACH_BUILD) {
+ rootProject.logger.lifecycle("Skipping task ${task.path} because: within `mach build`")
+ return false
+ }
+
+ // Never for official builds.
+ if (mozconfig.substs.MOZILLA_OFFICIAL) {
+ rootProject.logger.lifecycle("Skipping task ${task.path} because: MOZILLA_OFFICIAL")
+ return false
+ }
+
+ // Multi-l10n builds set `AB_CD=multi`, which isn't a valid locale, and
+ // `MOZ_CHROME_MULTILOCALE`. To avoid failures, if Gradle is invoked with
+ // either, we don't invoke Make at all; this allows a multi-locale omnijar
+ // to be consumed without modification.
+ if ('multi' == System.env.AB_CD || System.env.MOZ_CHROME_MULTILOCALE) {
+ rootProject.logger.lifecycle("Skipping task ${task.path} because: AB_CD=multi")
+ return false
+ }
+
+ // Single-locale l10n repacks set `IS_LANGUAGE_REPACK=1` and handle resource
+ // and code generation themselves.
+ if ('1' == System.env.IS_LANGUAGE_REPACK) {
+ rootProject.logger.lifecycle("Skipping task ${task.path} because: IS_LANGUAGE_REPACK")
+ return false
+ }
+
+ rootProject.logger.lifecycle("Executing task ${task.path}")
+ return true
+}
+
+// Non-official versions are like "61.0a1", where "a1" is the milestone.
+// This simply strips that off, leaving "61.0" in this example.
+def getAppVersionWithoutMilestone() {
+ return project.ext.mozconfig.substs.MOZ_APP_VERSION.replaceFirst(/a[0-9]/, "")
+}
+
+// This converts MOZ_APP_VERSION into an integer
+// version code.
+//
+// We take something like 58.1.2a1 and come out with 5800102
+// This gives us 3 digits for the major number, and 2 digits
+// each for the minor and build number. Beta and Release
+//
+// This must be synchronized with _compute_gecko_version(...) in /taskcluster/gecko_taskgraph/transforms/task.py
+def computeVersionCode() {
+ String appVersion = getAppVersionWithoutMilestone()
+
+ // Split on the dot delimiter, e.g. 58.1.1a1 -> ["58, "1", "1a1"]
+ String[] parts = appVersion.split('\\.')
+
+ assert parts.size() == 2 || parts.size() == 3
+
+ // Major
+ int code = Integer.parseInt(parts[0]) * 100000
+
+ // Minor
+ code += Integer.parseInt(parts[1]) * 100
+
+ // Build
+ if (parts.size() == 3) {
+ code += Integer.parseInt(parts[2])
+ }
+
+ return code;
+}
+
+def getVersionName() {
+ return "${mozconfig.substs.MOZ_APP_VERSION}-${mozconfig.substs.MOZ_UPDATE_CHANNEL}"
+}
+
+// Mimic Python: open(os.path.join(buildconfig.topobjdir, 'buildid.h')).readline().split()[2]
+def getBuildId() {
+ return file("${topobjdir}/buildid.h").getText('utf-8').split()[2]
+}
+
+def getVersionNumber() {
+ def appVersion = getAppVersionWithoutMilestone()
+ def parts = appVersion.split('\\.')
+ def version = parts[0] + "." + parts[1] + "." + getBuildId()
+ def substs = project.ext.mozconfig.substs
+ if (!substs.MOZILLA_OFFICIAL && !substs.MOZ_ANDROID_FAT_AAR_ARCHITECTURES) {
+ // Use -SNAPSHOT versions locally to enable the local GeckoView substitution flow.
+ version += "-SNAPSHOT"
+ }
+ return version
+}
+
+def getArtifactSuffix() {
+ def substs = project.ext.mozconfig.substs
+
+ def suffix = ""
+ // Release artifacts don't specify the channel, for the sake of simplicity.
+ if (substs.MOZ_UPDATE_CHANNEL != 'release') {
+ suffix += "-${mozconfig.substs.MOZ_UPDATE_CHANNEL}"
+ }
+
+ return suffix
+}
+
+class MachExec extends Exec {
+ def MachExec() {
+ // Bug 1543982: When invoking `mach build` recursively, the outer `mach
+ // build` itself modifies the environment, causing configure to run
+ // again. This tries to restore the environment that the outer `mach
+ // build` was invoked in. See the comment in
+ // $topsrcdir/settings.gradle.
+ project.ext.mozconfig.mozconfig.env.unmodified.each { k, v -> environment.remove(k) }
+ environment project.ext.mozconfig.orig_mozconfig.env.unmodified
+ environment 'MOZCONFIG', project.ext.mozconfig.substs.MOZCONFIG
+ }
+}
+
+task machBuildFaster(type: MachExec) {
+ onlyIf rootProject.ext.geckoBinariesOnlyIf
+
+ workingDir "${topsrcdir}"
+
+ commandLine mozconfig.substs.PYTHON3
+ args "${topsrcdir}/mach"
+ args 'build'
+ args 'faster'
+
+ // Add `-v` if we're running under `--info` (or `--debug`).
+ if (project.logger.isEnabled(LogLevel.INFO)) {
+ args '-v'
+ }
+
+ // `path` is like `:machBuildFaster`.
+ standardOutput = new TaggedLogOutputStream("${path}>", logger)
+ errorOutput = standardOutput
+}
+
+task machStagePackage(type: MachExec) {
+ onlyIf rootProject.ext.geckoBinariesOnlyIf
+
+ dependsOn rootProject.machBuildFaster
+
+ workingDir "${topobjdir}"
+
+ // We'd prefer this to be a `mach` invocation, but `mach build
+ // mobile/android/installer/stage-package` doesn't work as expected.
+ commandLine mozconfig.substs.GMAKE
+ args '-C'
+ args "${topobjdir}/mobile/android/installer"
+ args 'stage-package'
+
+ outputs.file "${topobjdir}/dist/geckoview/assets/omni.ja"
+
+ outputs.file "${topobjdir}/dist/geckoview/assets/${mozconfig.substs.ANDROID_CPU_ARCH}/libxul.so"
+ outputs.file "${topobjdir}/dist/geckoview/lib/${mozconfig.substs.ANDROID_CPU_ARCH}/libmozglue.so"
+
+ // Force running `stage-package`.
+ outputs.upToDateWhen { false }
+
+ // `path` is like `:machStagePackage`.
+ standardOutput = new TaggedLogOutputStream("${path}>", logger)
+ errorOutput = standardOutput
+}
+
+afterEvaluate {
+ subprojects { project ->
+ tasks.withType(JavaCompile) {
+ // Add compiler args for all code except third-party code.
+ options.compilerArgs += [
+ // Turn on all warnings, except...
+ "-Xlint:all",
+ // Deprecation, because we do use deprecated API for compatibility.
+ "-Xlint:-deprecation",
+ // Serial, because we don't use Java serialization.
+ "-Xlint:-serial",
+ // Classfile, because javac has a bug with MethodParameters attributes
+ // with Java 7. https://bugs.openjdk.java.net/browse/JDK-8190452
+ "-Xlint:-classfile",
+ // Turn all remaining warnings into errors,
+ // unless marked by @SuppressWarnings.
+ "-Werror"]
+ }
+ }
+}
+
+apply plugin: 'idea'
+
+idea {
+ project {
+ languageLevel = '1.8'
+ }
+
+ module {
+ // Object directories take a huge amount of time for IntelliJ to index.
+ // Exclude them. Convention is that object directories start with obj.
+ // IntelliJ is clever and will not exclude the parts of the object
+ // directory that are referenced, if there are any. In practice,
+ // indexing the entirety of the tree is taking too long, so exclude all
+ // but mobile/.
+ def topsrcdirURI = file(topsrcdir).toURI()
+ excludeDirs += files(file(topsrcdir)
+ .listFiles({it.isDirectory()} as FileFilter)
+ .collect({topsrcdirURI.relativize(it.toURI()).toString()}) // Relative paths.
+ .findAll({!it.equals('mobile/')}))
+
+ // If topobjdir is below topsrcdir, hide only some portions of that tree.
+ def topobjdirURI = file(topobjdir).toURI()
+ if (!topsrcdirURI.relativize(topobjdirURI).isAbsolute()) {
+ excludeDirs -= file(topobjdir)
+ excludeDirs += files(file(topobjdir).listFiles())
+ excludeDirs -= file("${topobjdir}/gradle")
+ }
+ }
+}
+
+subprojects {
+ apply plugin: "com.diffplug.spotless"
+
+ spotless {
+ java {
+ target project.fileTree(project.projectDir) {
+ include '**/*.java'
+ exclude '**/thirdparty/**'
+ }
+ googleJavaFormat('1.17.0')
+ }
+ kotlin {
+ target project.fileTree(project.projectDir) {
+ include '**/*.kt'
+ exclude '**/thirdparty/**'
+ }
+ ktlint('0.48.2')
+ }
+ }
+}