From 89b88a0a2d139d4d02929f3eb6aef884628f09f8 Mon Sep 17 00:00:00 2001 From: Daan Vanden Bosch Date: Sat, 24 Jul 2021 20:13:02 +0200 Subject: [PATCH] Put all shared build logic in precompiled gradle conventions plugins. --- build.gradle.kts | 42 ----------- buildSrc/build.gradle.kts | 15 ++-- .../main/kotlin/world/phantasmal/OptIns.kt | 11 +++ .../kotlin/world/phantasmal/common.gradle.kts | 16 ++++ .../phantasmal/gradle/KarmaResourcesTask.kt | 28 ------- .../world/phantasmal/gradle/PwJsPlugin.kt | 20 ----- .../kotlin/world/phantasmal/js.gradle.kts | 28 +++++++ .../kotlin/world/phantasmal/jvm.gradle.kts | 29 +++++++ .../phantasmal/karma-resources.gradle.kts | 28 +++++++ .../world/phantasmal/multiplatform.gradle.kts | 75 +++++++++++++++++++ .../{ => world/phantasmal}/karmaConfig.js | 0 core/build.gradle.kts | 51 +------------ lib/build.gradle.kts | 51 +------------ observable/build.gradle.kts | 33 +------- settings.gradle.kts | 2 +- test-utils/build.gradle.kts | 11 +-- web/assembly-worker/build.gradle.kts | 13 +--- web/assets-generation/build.gradle.kts | 9 +-- web/build.gradle.kts | 16 +--- web/shared/build.gradle.kts | 9 +-- webui/build.gradle.kts | 9 +-- 21 files changed, 209 insertions(+), 287 deletions(-) create mode 100644 buildSrc/src/main/kotlin/world/phantasmal/OptIns.kt create mode 100644 buildSrc/src/main/kotlin/world/phantasmal/common.gradle.kts delete mode 100644 buildSrc/src/main/kotlin/world/phantasmal/gradle/KarmaResourcesTask.kt delete mode 100644 buildSrc/src/main/kotlin/world/phantasmal/gradle/PwJsPlugin.kt create mode 100644 buildSrc/src/main/kotlin/world/phantasmal/js.gradle.kts create mode 100644 buildSrc/src/main/kotlin/world/phantasmal/jvm.gradle.kts create mode 100644 buildSrc/src/main/kotlin/world/phantasmal/karma-resources.gradle.kts create mode 100644 buildSrc/src/main/kotlin/world/phantasmal/multiplatform.gradle.kts rename buildSrc/src/main/resources/{ => world/phantasmal}/karmaConfig.js (100%) diff --git a/build.gradle.kts b/build.gradle.kts index 30c21244..f112ec87 100644 --- a/build.gradle.kts +++ b/build.gradle.kts @@ -1,45 +1,3 @@ -import org.jetbrains.kotlin.gradle.tasks.Kotlin2JsCompile -import org.jetbrains.kotlin.gradle.tasks.KotlinCompile - -plugins { - kotlin("js") version "1.5.21" apply false - kotlin("multiplatform") version "1.5.21" apply false - kotlin("plugin.serialization") version "1.5.21" apply false -} - tasks.wrapper { gradleVersion = "7.1.1" } - -subprojects { - project.extra["coroutinesVersion"] = "1.5.1" - project.extra["junitVersion"] = "5.7.1" - project.extra["kotlinLoggingVersion"] = "2.0.6" - project.extra["ktorVersion"] = "1.6.1" - project.extra["log4jVersion"] = "2.14.1" - project.extra["serializationVersion"] = "1.2.2" - project.extra["slf4jVersion"] = "1.7.30" - - repositories { - mavenCentral() - maven(url = "https://kotlin.bintray.com/kotlinx/") - } - - tasks.withType { - kotlinOptions { - jvmTarget = "11" - freeCompilerArgs += listOf( - "-Xjvm-default=all" - ) - } - } - - tasks.withType { - kotlinOptions.freeCompilerArgs += listOf( - "-Xopt-in=kotlin.RequiresOptIn", - "-Xopt-in=kotlin.ExperimentalUnsignedTypes", - "-Xopt-in=kotlin.contracts.ExperimentalContracts", - "-Xopt-in=kotlin.time.ExperimentalTime" - ) - } -} diff --git a/buildSrc/build.gradle.kts b/buildSrc/build.gradle.kts index 0af53070..eed7753b 100644 --- a/buildSrc/build.gradle.kts +++ b/buildSrc/build.gradle.kts @@ -1,17 +1,12 @@ plugins { - kotlin("jvm") version "1.5.21" - `java-gradle-plugin` + `kotlin-dsl` } repositories { - mavenCentral() + gradlePluginPortal() } -gradlePlugin { - plugins { - create("pwPlugins") { - id = "world.phantasmal.gradle.js" - implementationClass = "world.phantasmal.gradle.PwJsPlugin" - } - } +dependencies { + implementation("org.jetbrains.kotlin:kotlin-gradle-plugin:1.5.21") + implementation("org.jetbrains.kotlin:kotlin-serialization:1.5.21") } diff --git a/buildSrc/src/main/kotlin/world/phantasmal/OptIns.kt b/buildSrc/src/main/kotlin/world/phantasmal/OptIns.kt new file mode 100644 index 00000000..f1fef95a --- /dev/null +++ b/buildSrc/src/main/kotlin/world/phantasmal/OptIns.kt @@ -0,0 +1,11 @@ +package world.phantasmal + +val EXPERIMENTAL_ANNOTATIONS: List = listOf( + "kotlin.RequiresOptIn", + "kotlin.ExperimentalUnsignedTypes", + "kotlin.contracts.ExperimentalContracts", + "kotlin.time.ExperimentalTime", +) + +val EXPERIMENTAL_ANNOTATION_COMPILER_ARGS: List = + EXPERIMENTAL_ANNOTATIONS.map { "-Xopt-in=$it" } diff --git a/buildSrc/src/main/kotlin/world/phantasmal/common.gradle.kts b/buildSrc/src/main/kotlin/world/phantasmal/common.gradle.kts new file mode 100644 index 00000000..6d5f3bcb --- /dev/null +++ b/buildSrc/src/main/kotlin/world/phantasmal/common.gradle.kts @@ -0,0 +1,16 @@ +package world.phantasmal + +plugins { + kotlin("plugin.serialization") apply false +} + +repositories { + mavenCentral() +} + +project.extra["coroutinesVersion"] = "1.5.1" +project.extra["junitVersion"] = "5.7.1" +project.extra["kotlinLoggingVersion"] = "2.0.6" +project.extra["ktorVersion"] = "1.6.1" +project.extra["log4jVersion"] = "2.14.1" +project.extra["serializationVersion"] = "1.2.2" diff --git a/buildSrc/src/main/kotlin/world/phantasmal/gradle/KarmaResourcesTask.kt b/buildSrc/src/main/kotlin/world/phantasmal/gradle/KarmaResourcesTask.kt deleted file mode 100644 index 3d6c25a8..00000000 --- a/buildSrc/src/main/kotlin/world/phantasmal/gradle/KarmaResourcesTask.kt +++ /dev/null @@ -1,28 +0,0 @@ -package world.phantasmal.gradle - -import org.gradle.api.DefaultTask -import org.gradle.api.tasks.TaskAction - -/** - * This task generates a Karma configuration in karma.config.d that ensures Karma serves files from - * the resources directories. - */ -open class KarmaResourcesTask : DefaultTask() { - private val outputFile = project.file("karma.config.d/karma.config.generated.js") - - init { - outputs.file(outputFile) - } - - @TaskAction - fun generateKarmaConfig() { - outputFile.outputStream().use { stream -> - val writer = stream.writer() - val path = project.projectDir.absolutePath.replace("\\", "\\\\") - writer.write("const PROJECT_PATH = '$path';\n\n") - writer.flush() - - KarmaResourcesTask::class.java.getResourceAsStream("/karmaConfig.js").copyTo(stream) - } - } -} diff --git a/buildSrc/src/main/kotlin/world/phantasmal/gradle/PwJsPlugin.kt b/buildSrc/src/main/kotlin/world/phantasmal/gradle/PwJsPlugin.kt deleted file mode 100644 index d9587ee9..00000000 --- a/buildSrc/src/main/kotlin/world/phantasmal/gradle/PwJsPlugin.kt +++ /dev/null @@ -1,20 +0,0 @@ -package world.phantasmal.gradle - -import org.gradle.api.Plugin -import org.gradle.api.Project - -/** - * This plugin adds a karmaResources task as dependency to the browserTest and jsBrowserTest tasks - * as a workaround for https://youtrack.jetbrains.com/issue/KT-42923. - */ -class PwJsPlugin : Plugin { - override fun apply(target: Project) { - val karmaResources = target.tasks.create("karmaResources", KarmaResourcesTask::class.java) - - target.tasks.configureEach { task -> - if (task.name == "browserTest" || task.name == "jsBrowserTest") { - task.dependsOn(karmaResources) - } - } - } -} diff --git a/buildSrc/src/main/kotlin/world/phantasmal/js.gradle.kts b/buildSrc/src/main/kotlin/world/phantasmal/js.gradle.kts new file mode 100644 index 00000000..fb236aad --- /dev/null +++ b/buildSrc/src/main/kotlin/world/phantasmal/js.gradle.kts @@ -0,0 +1,28 @@ +package world.phantasmal + +plugins { + kotlin("js") + id("world.phantasmal.common") + id("world.phantasmal.karma-resources") +} + +kotlin { + js { + compilations.all { + kotlinOptions { + freeCompilerArgs = freeCompilerArgs + EXPERIMENTAL_ANNOTATION_COMPILER_ARGS + } + } + browser { + testTask { + useKarma { + useChromeHeadless() + } + } + } + } +} + +dependencies { + testImplementation(kotlin("test-js")) +} diff --git a/buildSrc/src/main/kotlin/world/phantasmal/jvm.gradle.kts b/buildSrc/src/main/kotlin/world/phantasmal/jvm.gradle.kts new file mode 100644 index 00000000..25ac80ef --- /dev/null +++ b/buildSrc/src/main/kotlin/world/phantasmal/jvm.gradle.kts @@ -0,0 +1,29 @@ +package world.phantasmal + +import org.jetbrains.kotlin.gradle.tasks.KotlinCompile + +plugins { + kotlin("jvm") + id("world.phantasmal.common") +} + +val junitVersion: String by project.extra +val log4jVersion: String by project.extra + +tasks.withType().configureEach { + kotlinOptions { + jvmTarget = "11" + freeCompilerArgs = freeCompilerArgs + EXPERIMENTAL_ANNOTATION_COMPILER_ARGS + } +} + +dependencies { + runtimeOnly("org.apache.logging.log4j:log4j-slf4j-impl:$log4jVersion") + + testImplementation(kotlin("test-junit5")) + testRuntimeOnly("org.junit.jupiter:junit-jupiter-engine:$junitVersion") +} + +tasks.withType().configureEach { + useJUnitPlatform() +} diff --git a/buildSrc/src/main/kotlin/world/phantasmal/karma-resources.gradle.kts b/buildSrc/src/main/kotlin/world/phantasmal/karma-resources.gradle.kts new file mode 100644 index 00000000..e97b8c42 --- /dev/null +++ b/buildSrc/src/main/kotlin/world/phantasmal/karma-resources.gradle.kts @@ -0,0 +1,28 @@ +package world.phantasmal + +// This task generates a Karma configuration in karma.config.d that ensures Karma serves files from +// the resources directories. +// This is a workaround for https://youtrack.jetbrains.com/issue/KT-42923. +val karmaResourcesTask = tasks.register("karmaResources") { + doLast { + val karmaConfigDir = file("karma.config.d") + karmaConfigDir.mkdir() + + object {}::class.java.getResourceAsStream("karmaConfig.js")!!.use { karmaConfigStream -> + karmaConfigDir.resolve("karma.config.generated.js").outputStream().use { stream -> + val writer = stream.writer() + val path = project.projectDir.absolutePath.replace("\\", "\\\\") + writer.write("const PROJECT_PATH = '$path';\n\n") + writer.flush() + + karmaConfigStream.copyTo(stream) + } + } + } +} + +tasks.configureEach { + if (name == "browserTest" || name == "jsBrowserTest") { + dependsOn(karmaResourcesTask) + } +} diff --git a/buildSrc/src/main/kotlin/world/phantasmal/multiplatform.gradle.kts b/buildSrc/src/main/kotlin/world/phantasmal/multiplatform.gradle.kts new file mode 100644 index 00000000..5fbcecff --- /dev/null +++ b/buildSrc/src/main/kotlin/world/phantasmal/multiplatform.gradle.kts @@ -0,0 +1,75 @@ +package world.phantasmal + +plugins { + kotlin("multiplatform") + id("world.phantasmal.common") + id("world.phantasmal.karma-resources") +} + +val coroutinesVersion: String by project.ext +val junitVersion: String by project.extra +val kotlinLoggingVersion: String by project.extra +val log4jVersion: String by project.extra + +kotlin { + js { + browser { + testTask { + useKarma { + useChromeHeadless() + } + } + } + } + + jvm { + compilations.all { + kotlinOptions { + jvmTarget = "11" + } + } + } + + sourceSets { + all { + EXPERIMENTAL_ANNOTATIONS.forEach(languageSettings::useExperimentalAnnotation) + } + + commonMain { + dependencies { + api("org.jetbrains.kotlinx:kotlinx-coroutines-core:$coroutinesVersion") + api("io.github.microutils:kotlin-logging:$kotlinLoggingVersion") + } + } + + commonTest { + dependencies { + implementation(kotlin("test-common")) + implementation(kotlin("test-annotations-common")) + } + } + + getByName("jsTest") { + dependencies { + implementation(kotlin("test-js")) + } + } + + getByName("jvmMain") { + dependencies { + runtimeOnly("org.apache.logging.log4j:log4j-slf4j-impl:$log4jVersion") + } + } + + getByName("jvmTest") { + dependencies { + implementation(kotlin("test-junit5")) + runtimeOnly("org.junit.jupiter:junit-jupiter-engine:$junitVersion") + } + } + } +} + +tasks.withType().configureEach { + useJUnitPlatform() +} diff --git a/buildSrc/src/main/resources/karmaConfig.js b/buildSrc/src/main/resources/world/phantasmal/karmaConfig.js similarity index 100% rename from buildSrc/src/main/resources/karmaConfig.js rename to buildSrc/src/main/resources/world/phantasmal/karmaConfig.js diff --git a/core/build.gradle.kts b/core/build.gradle.kts index 978bf37e..7d83509b 100644 --- a/core/build.gradle.kts +++ b/core/build.gradle.kts @@ -1,52 +1,3 @@ plugins { - kotlin("multiplatform") -} - -val coroutinesVersion: String by project.ext -val junitVersion: String by project.extra -val kotlinLoggingVersion: String by project.extra - -tasks.withType { - useJUnitPlatform() -} - -kotlin { - js { - browser {} - } - - jvm() - - sourceSets { - all { - languageSettings.useExperimentalAnnotation("kotlin.ExperimentalUnsignedTypes") - } - - commonMain { - dependencies { - api("org.jetbrains.kotlinx:kotlinx-coroutines-core:$coroutinesVersion") - api("io.github.microutils:kotlin-logging:$kotlinLoggingVersion") - } - } - - commonTest { - dependencies { - implementation(kotlin("test-common")) - implementation(kotlin("test-annotations-common")) - } - } - - getByName("jsTest") { - dependencies { - implementation(kotlin("test-js")) - } - } - - getByName("jvmTest") { - dependencies { - implementation(kotlin("test-junit5")) - runtimeOnly("org.junit.jupiter:junit-jupiter-engine:$junitVersion") - } - } - } + id("world.phantasmal.multiplatform") } diff --git a/lib/build.gradle.kts b/lib/build.gradle.kts index ca86f6fe..87fcd194 100644 --- a/lib/build.gradle.kts +++ b/lib/build.gradle.kts @@ -4,9 +4,8 @@ import org.snakeyaml.engine.v2.api.LoadSettings import java.io.PrintWriter plugins { - kotlin("multiplatform") + id("world.phantasmal.multiplatform") kotlin("plugin.serialization") - id("world.phantasmal.gradle.js") } buildscript { @@ -15,67 +14,23 @@ buildscript { } } -val coroutinesVersion: String by project.extra -val junitVersion: String by project.extra -val kotlinLoggingVersion: String by project.extra val serializationVersion: String by project.extra -val slf4jVersion: String by project.extra - -tasks.withType { - useJUnitPlatform() -} kotlin { - js { - browser { - testTask { - useKarma { - useChromeHeadless() - } - } - } - } - - jvm() - sourceSets { - all { - languageSettings.useExperimentalAnnotation("kotlin.RequiresOptIn") - languageSettings.useExperimentalAnnotation("kotlin.ExperimentalUnsignedTypes") - languageSettings.useExperimentalAnnotation("kotlin.time.ExperimentalTime") - } - commonMain { kotlin.setSrcDirs(kotlin.srcDirs + file("build/generated-src/commonMain/kotlin")) dependencies { api(project(":core")) - api("org.jetbrains.kotlinx:kotlinx-coroutines-core:$coroutinesVersion") - api("io.github.microutils:kotlin-logging:$kotlinLoggingVersion") api("org.jetbrains.kotlinx:kotlinx-serialization-core:$serializationVersion") } } commonTest { dependencies { - implementation(kotlin("test-common")) - implementation(kotlin("test-annotations-common")) implementation(project(":test-utils")) } } - - getByName("jsTest") { - dependencies { - implementation(kotlin("test-js")) - } - } - - getByName("jvmTest") { - dependencies { - implementation(kotlin("test-junit5")) - implementation("org.slf4j:slf4j-simple:$slf4jVersion") - runtimeOnly("org.junit.jupiter:junit-jupiter-engine:$junitVersion") - } - } } } @@ -209,6 +164,8 @@ fun paramsToCode(params: List>, indent: Int): String { } } -tasks.withType> { +// The following line results in warning "The AbstractCompile.destinationDir property has been +// deprecated.". +tasks.withType>().configureEach { dependsOn(generateOpcodes) } diff --git a/observable/build.gradle.kts b/observable/build.gradle.kts index 6ce52542..a45f737e 100644 --- a/observable/build.gradle.kts +++ b/observable/build.gradle.kts @@ -1,25 +1,9 @@ plugins { - kotlin("multiplatform") -} - -val junitVersion: String by project.extra - -tasks.withType { - useJUnitPlatform() + id("world.phantasmal.multiplatform") } kotlin { - js { - browser {} - } - - jvm() - sourceSets { - all { - languageSettings.useExperimentalAnnotation("kotlin.contracts.ExperimentalContracts") - } - commonMain { dependencies { implementation(project(":core")) @@ -28,23 +12,8 @@ kotlin { commonTest { dependencies { - implementation(kotlin("test-common")) - implementation(kotlin("test-annotations-common")) implementation(project(":test-utils")) } } - - getByName("jsTest") { - dependencies { - implementation(kotlin("test-js")) - } - } - - getByName("jvmTest") { - dependencies { - implementation(kotlin("test-junit5")) - runtimeOnly("org.junit.jupiter:junit-jupiter-engine:$junitVersion") - } - } } } diff --git a/settings.gradle.kts b/settings.gradle.kts index 3df687fa..ac21085f 100644 --- a/settings.gradle.kts +++ b/settings.gradle.kts @@ -9,5 +9,5 @@ include( ":web:assembly-worker", ":web:assets-generation", ":web:shared", - ":webui" + ":webui", ) diff --git a/test-utils/build.gradle.kts b/test-utils/build.gradle.kts index fe5c84df..0bfa49a7 100644 --- a/test-utils/build.gradle.kts +++ b/test-utils/build.gradle.kts @@ -1,23 +1,14 @@ plugins { - kotlin("multiplatform") + id("world.phantasmal.multiplatform") } -val coroutinesVersion: String by project.ext - kotlin { - js { - browser {} - } - - jvm() - sourceSets { commonMain { dependencies { api(project(":core")) api(kotlin("test-common")) api(kotlin("test-annotations-common")) - api("org.jetbrains.kotlinx:kotlinx-coroutines-core:$coroutinesVersion") } } diff --git a/web/assembly-worker/build.gradle.kts b/web/assembly-worker/build.gradle.kts index 62c77a65..f48bf67d 100644 --- a/web/assembly-worker/build.gradle.kts +++ b/web/assembly-worker/build.gradle.kts @@ -1,26 +1,15 @@ plugins { - kotlin("js") + id("world.phantasmal.js") } kotlin { js { - browser { - testTask { - useKarma { - useChromeHeadless() - } - } - } binaries.executable() } } -val kotlinLoggingVersion: String by project.extra - dependencies { api(project(":web:shared")) - implementation("io.github.microutils:kotlin-logging-js:$kotlinLoggingVersion") - testImplementation(kotlin("test-js")) testImplementation(project(":test-utils")) } diff --git a/web/assets-generation/build.gradle.kts b/web/assets-generation/build.gradle.kts index 5c5b27fe..ed01f803 100644 --- a/web/assets-generation/build.gradle.kts +++ b/web/assets-generation/build.gradle.kts @@ -1,16 +1,11 @@ plugins { - kotlin("jvm") + id("world.phantasmal.jvm") } -val kotlinLoggingVersion: String by project.extra -val log4jVersion: String by project.extra - dependencies { implementation(project(":lib")) implementation(project(":web:shared")) implementation("org.jsoup:jsoup:1.13.1") - implementation("io.github.microutils:kotlin-logging:$kotlinLoggingVersion") - runtimeOnly("org.apache.logging.log4j:log4j-slf4j-impl:$log4jVersion") } tasks.register("generateAssets") { @@ -18,6 +13,6 @@ tasks.register("generateAssets") { outputs.dir(outputFile) classpath = sourceSets.main.get().runtimeClasspath - main = "world.phantasmal.web.assetsGeneration.MainKt" + mainClass.set("world.phantasmal.web.assetsGeneration.MainKt") args = listOf(outputFile.absolutePath) } diff --git a/web/build.gradle.kts b/web/build.gradle.kts index 0f4c73f0..4924011b 100644 --- a/web/build.gradle.kts +++ b/web/build.gradle.kts @@ -1,7 +1,6 @@ plugins { - kotlin("js") + id("world.phantasmal.js") kotlin("plugin.serialization") - id("world.phantasmal.gradle.js") } kotlin { @@ -13,20 +12,14 @@ kotlin { runTask { devServer = devServer!!.copy( open = false, - port = 1623 + port = 1623, ) } - testTask { - useKarma { - useChromeHeadless() - } - } } binaries.executable() } } -val kotlinLoggingVersion: String by project.extra val ktorVersion: String by project.extra val serializationVersion: String by project.extra @@ -35,13 +28,13 @@ dependencies { implementation(project(":webui")) implementation(project(":web:shared")) - implementation("io.github.microutils:kotlin-logging-js:$kotlinLoggingVersion") implementation("io.ktor:ktor-client-core-js:$ktorVersion") implementation("io.ktor:ktor-client-serialization-js:$ktorVersion") implementation("org.jetbrains.kotlinx:kotlinx-serialization-json:$serializationVersion") implementation("org.jetbrains.kotlinx:kotlinx-datetime:0.1.1") implementation(npm("golden-layout", "^1.5.9")) - // Can't upgrade monaco-editor until https://github.com/microsoft/monaco-editor/issues/2466 is fixed. + // Can't upgrade monaco-editor until https://github.com/microsoft/monaco-editor/issues/2466 is + // fixed. implementation(npm("monaco-editor", "0.20.0")) implementation(npm("three", "^0.128.0")) implementation(npm("javascript-lp-solver", "0.4.17")) @@ -50,7 +43,6 @@ dependencies { // Can't upgrade monaco-editor-webpack-plugin until monaco-editor is upgraded. implementation(devNpm("monaco-editor-webpack-plugin", "1.9.1")) - testImplementation(kotlin("test-js")) testImplementation(project(":test-utils")) } diff --git a/web/shared/build.gradle.kts b/web/shared/build.gradle.kts index c0a6e8db..eef598ab 100644 --- a/web/shared/build.gradle.kts +++ b/web/shared/build.gradle.kts @@ -1,18 +1,11 @@ plugins { - kotlin("multiplatform") + id("world.phantasmal.multiplatform") kotlin("plugin.serialization") } val serializationVersion: String by project.extra kotlin { - js { - browser { - } - } - - jvm() - sourceSets { commonMain { dependencies { diff --git a/webui/build.gradle.kts b/webui/build.gradle.kts index fef648f0..55096a4a 100644 --- a/webui/build.gradle.kts +++ b/webui/build.gradle.kts @@ -1,11 +1,5 @@ plugins { - kotlin("js") -} - -kotlin { - js { - browser {} - } + id("world.phantasmal.js") } dependencies { @@ -13,6 +7,5 @@ dependencies { api(project(":observable")) implementation(npm("@fortawesome/fontawesome-free", "^5.13.1")) - testImplementation(kotlin("test-js")) testImplementation(project(":test-utils")) }