从 kotlin 多平台项目创建胖罐子



我最近从旧的1.2多平台切换到1.3。不同之处在于,每个多平台模块都有一个 build.gradle 文件(我有 5 个),所以配置要少得多。 但是,我似乎无法配置使用jvm平台的所有依赖项创建可运行的胖罐。 我曾经在我的 jvm 项目和 jar 任务中使用标准的"应用程序"插件,但这不再有效了。我发现有"jvmJar"任务,我修改了它(设置主类),但创建的jar不包含依赖项,并且在ClassNotFoundException上崩溃。我该怎么做?

这就是我现在拥有的:

jvm() {
jvmJar {
manifest {
attributes 'Main-Class': 'eu.xx.Runner'
}
from { configurations.compile.collect { it.isDirectory() ? it : zipTree(it) } }
}
}

我确实遇到了那个颠簸并使用了这项工作。

1. 重组您的项目

让我们称您的项目为Project

创建另一个子模块,例如subA,它将具有 gradle 符号Project:subA

现在,subA在其build.gradle中包含您的多平台代码(它是带有apply:kotlin-multiplafrom的gradle项目)

2. 添加另一个子模块

创建另一个仅针对 JVM 的子模块 比如subB,它将具有 gradle 表示法Project:subB

因此,subB将具有插件:'application''org.jetbrains.kotlin.jvm'

3. 将您的模块添加为 gradle 依赖项(请参阅我的 build.gradle)

plugins {
id 'org.jetbrains.kotlin.jvm' version '1.3.31'
id "application"
}
apply plugin: "kotlinx-serialization"
group 'tz.or.self'
version '0.0.0'
mainClassName = "com.example.MainKt"
sourceCompatibility = 1.8
compileKotlin {
kotlinOptions.jvmTarget = "1.8"
}
dependencies {
implementation project(':subA')
}

您可以像构建常规 Java 项目一样继续构建 subB,甚至可以使用现有的插件,它将起作用

让 gradle/多平台工作的唯一方法似乎是无休止的反复试验;这是一场噩梦,与其说它是作为"构建"系统构建的,不如说是作为"构建系统"构建的;换句话说,这两个工具(一起或单独)是仅实现插件制造商想要的单个软件开发生命周期的一种手段,但是,如果您已经设计了所需的软件生命周期和 CI/CD 系统,并且现在您正在尝试实现该工程,那么使用这些工具将比使用脚本更难, 代码或专家。造成这种情况的原因有很多:

  • 由于插件制造商仅公开了栏的最低可配置性,因此编码约定发生了巨大变化,可能只允许访问他们自己的个人项目所需的内容。
  • 文档更新非常差;Kotlin、gradle 和插件变化如此之快,我开始认真质疑这些工具的有用性。

因此,在撰写本文时,这似乎是使用 kotlin 1.3.72、多平台 1.3.72、ktor 1.3.2 和 gradle 6.2.2(使用 kts 格式)时使用的正确语法。

请注意,fatJar 似乎组装正确但无法运行,它找不到类,所以我包含了我一直在使用的第 runTLocal 任务。

这不是一个完整的解决方案,所以我讨厌在这里发布它,但据我所知......这是我可以在任何地方找到的最完整和最新的解决方案。

//Import variables from gradle.properties
val environment: String by project
val kotlinVersion: String by project
val ktorVersion: String by project
val kotlinExposedVersion: String by project
val mySqlConnectorVersion: String by project
val logbackVersion: String by project
val romeToolsVersion: String by project
val klaxonVersion: String by project
val kotlinLoggingVersion: String by project
val skrapeItVersion: String by project
val jsoupVersion: String by project
val devWebApiServer: String by project
val devWebApiServerVersion: String by project
//Build File Configuration
plugins {
java
kotlin("multiplatform") version "1.3.72"
}
group = "com.app"
version = "1.0-SNAPSHOT"
repositories {
mavenCentral()
jcenter()
jcenter {
url = uri("https://kotlin.bintray.com/kotlin-js-wrappers")
}
maven {
url = uri("https://jitpack.io")
}
}
//Multiplatform Configuration
kotlin {
jvm {
compilations {
val main = getByName("main")
tasks {
register<Jar>("buildFatJar") {
group = "application"
manifest {
attributes["Implementation-Title"] = "Gradle Jar File Example"
attributes["Implementation-Version"] = archiveVersion
attributes["Main-Class"] = "com.app.BackendAppKt"
}
archiveBaseName.set("${project.name}-fat")
from(main.output.classesDirs, main.compileDependencyFiles)
with(jar.get() as CopySpec)
}
register<JavaExec>("runLocally") {
group = "application"
setMain("com.app.BackendAppKt")
classpath = main.output.classesDirs
classpath += main.compileDependencyFiles
}
}
}
}
js {
browser { EXCLUDED FOR LENGTH }
}
sourceSets { EXCLUDED FOR LENGTH }
}

让它与 kotlin 1.3.61 中的多平台插件一起工作:

以下内容适用于src/jvmMain/kotlin/com/example/Hello.kt中的主文件

Hello.kt 还必须将其包指定为package com.example

我以这种方式配置了我的 jvm 目标:

kotlin {
targets {
jvm()
configure([jvm])  {
withJava()
jvmJar {
manifest {
attributes 'Main-Class': 'com.example.HelloKt'
}
from { configurations.runtimeClasspath.collect { it.isDirectory() ? it : zipTree(it) } }
}
}
}
}

让它与 luca992 所做的稍微修改的版本一起工作:

kotlin {
jvm() {
withJava()
jvmJar {
manifest {
attributes 'Main-Class': 'sample.MainKt'
}
from { configurations.runtimeClasspath.collect { it.isDirectory() ? it : zipTree(it) } }
}
}
...
}

最新更新