aboutsummaryrefslogtreecommitdiff
path: root/build-logic/automation-plugins/src/main/kotlin/crowdin
diff options
context:
space:
mode:
authorHarsh Shandilya <me@msfjarvis.dev>2021-11-29 01:54:29 +0530
committerHarsh Shandilya <me@msfjarvis.dev>2021-12-03 12:59:57 +0530
commit70cdd6179744468ff18c16dc9d131130801fbb5b (patch)
tree16faf5071606f789d6a0bfc384c230df7661fce4 /build-logic/automation-plugins/src/main/kotlin/crowdin
parentdfee170bd813d267272d0d4a8c64a4cbef965619 (diff)
build-logic: add crowdin and psl convention plugins
Diffstat (limited to 'build-logic/automation-plugins/src/main/kotlin/crowdin')
-rw-r--r--build-logic/automation-plugins/src/main/kotlin/crowdin/CrowdinExtension.kt20
-rw-r--r--build-logic/automation-plugins/src/main/kotlin/crowdin/CrowdinPlugin.kt127
2 files changed, 147 insertions, 0 deletions
diff --git a/build-logic/automation-plugins/src/main/kotlin/crowdin/CrowdinExtension.kt b/build-logic/automation-plugins/src/main/kotlin/crowdin/CrowdinExtension.kt
new file mode 100644
index 00000000..f1457673
--- /dev/null
+++ b/build-logic/automation-plugins/src/main/kotlin/crowdin/CrowdinExtension.kt
@@ -0,0 +1,20 @@
+/*
+ * Copyright © 2014-2021 The Android Password Store Authors. All Rights Reserved.
+ * SPDX-License-Identifier: GPL-3.0-only
+ */
+
+package crowdin
+
+/** Extension for configuring [CrowdinPlugin] */
+interface CrowdinExtension {
+
+ /** Configure the project name on Crowdin */
+ var projectName: String
+
+ /**
+ * Don't delete downloaded and extracted translation archives from build directory.
+ *
+ * Useful for debugging.
+ */
+ var skipCleanup: Boolean
+}
diff --git a/build-logic/automation-plugins/src/main/kotlin/crowdin/CrowdinPlugin.kt b/build-logic/automation-plugins/src/main/kotlin/crowdin/CrowdinPlugin.kt
new file mode 100644
index 00000000..580fc939
--- /dev/null
+++ b/build-logic/automation-plugins/src/main/kotlin/crowdin/CrowdinPlugin.kt
@@ -0,0 +1,127 @@
+/*
+ * Copyright © 2014-2021 The Android Password Store Authors. All Rights Reserved.
+ * SPDX-License-Identifier: GPL-3.0-only
+ */
+
+package crowdin
+
+import de.undercouch.gradle.tasks.download.Download
+import java.io.File
+import javax.xml.parsers.DocumentBuilderFactory
+import okhttp3.OkHttpClient
+import okhttp3.Request
+import org.gradle.api.GradleException
+import org.gradle.api.Plugin
+import org.gradle.api.Project
+import org.gradle.api.tasks.Copy
+import org.gradle.kotlin.dsl.create
+import org.gradle.kotlin.dsl.register
+import org.w3c.dom.Document
+
+private const val EXCEPTION_MESSAGE =
+ """Applying `crowdin-plugin` requires a projectName to be configured via the "crowdin" extension."""
+private const val CROWDIN_BUILD_API_URL =
+ "https://api.crowdin.com/api/project/%s/export?login=%s&account-key=%s"
+
+class CrowdinDownloadPlugin : Plugin<Project> {
+
+ override fun apply(project: Project) {
+ with(project) {
+ val buildDirectory = layout.buildDirectory.asFile.forUseAtConfigurationTime().get()
+ val extension = extensions.create<CrowdinExtension>("crowdin")
+ afterEvaluate {
+ val projectName = extension.projectName
+ if (projectName.isEmpty()) {
+ throw GradleException(EXCEPTION_MESSAGE)
+ }
+ tasks.register("buildOnApi") {
+ doLast {
+ val login = providers.environmentVariable("CROWDIN_LOGIN").forUseAtConfigurationTime()
+ val key =
+ providers.environmentVariable("CROWDIN_PROJECT_KEY").forUseAtConfigurationTime()
+ if (!login.isPresent) {
+ throw GradleException("CROWDIN_LOGIN environment variable must be set")
+ }
+ if (!key.isPresent) {
+ throw GradleException("CROWDIN_PROJECT_KEY environment variable must be set")
+ }
+ val client = OkHttpClient()
+ val url = CROWDIN_BUILD_API_URL.format(projectName, login.get(), key.get())
+ val request = Request.Builder().url(url).get().build()
+ client.newCall(request).execute()
+ }
+ }
+ val downloadCrowdin =
+ tasks.register<Download>("downloadCrowdin") {
+ dependsOn("buildOnApi")
+ src("https://crowdin.com/backend/download/project/$projectName.zip")
+ dest("$buildDirectory/translations.zip")
+ overwrite(true)
+ }
+ val extractCrowdin =
+ tasks.register<Copy>("extractCrowdin") {
+ dependsOn(downloadCrowdin)
+ doFirst { File(buildDir, "translations").deleteRecursively() }
+ from(zipTree("$buildDirectory/translations.zip"))
+ into("$buildDirectory/translations")
+ }
+ val extractStrings =
+ tasks.register<Copy>("extractStrings") {
+ dependsOn(extractCrowdin)
+ from("$buildDirectory/translations/")
+ into("${projectDir}/src/")
+ setFinalizedBy(setOf("removeIncompleteStrings"))
+ }
+ tasks.register("removeIncompleteStrings") {
+ doLast {
+ val sourceSets = arrayOf("main", "nonFree")
+ for (sourceSet in sourceSets) {
+ val stringFiles =
+ File("${projectDir}/src/$sourceSet").walkTopDown().filter {
+ it.name == "strings.xml"
+ }
+ val sourceFile =
+ stringFiles.firstOrNull { it.path.endsWith("values/strings.xml") }
+ ?: throw GradleException("No root strings.xml found in '$sourceSet' sourceSet")
+ val sourceDoc = parseDocument(sourceFile)
+ val baselineStringCount = countStrings(sourceDoc)
+ val threshold = 0.80 * baselineStringCount
+ stringFiles.forEach { file ->
+ if (file != sourceFile) {
+ val doc = parseDocument(file)
+ val stringCount = countStrings(doc)
+ if (stringCount < threshold) {
+ file.delete()
+ }
+ }
+ }
+ }
+ }
+ }
+ tasks.register("crowdin") {
+ dependsOn(extractStrings)
+ if (!extension.skipCleanup) {
+ doLast {
+ File("$buildDirectory/translations").deleteRecursively()
+ File("$buildDirectory/nonFree-translations").deleteRecursively()
+ File("$buildDirectory/translations.zip").delete()
+ }
+ }
+ }
+ }
+ }
+ }
+
+ private fun parseDocument(file: File): Document {
+ val dbFactory = DocumentBuilderFactory.newInstance()
+ val documentBuilder = dbFactory.newDocumentBuilder()
+ return documentBuilder.parse(file)
+ }
+
+ private fun countStrings(document: Document): Int {
+ // Normalization is beneficial for us
+ // https://stackoverflow.com/questions/13786607/normalization-in-dom-parsing-with-java-how-does-it-work
+ document.documentElement.normalize()
+ return document.getElementsByTagName("string").length
+ }
+}