我刚刚关注this tutorial to create a REST API using Jersey on Jetty,我喜欢这个结果。一切正常。但是,如果我运行Gradle shadowJar
任务来生成一个胖jar文件,则该文件可以正常运行,并且该文件也可以运行,但是在发出请求时会出现错误消息:
$ java -jar build/libs/sample-all.jar
[main] INFO org.eclipse.jetty.util.log - Logging initialized @161ms to org.eclipse.jetty.util.log.Slf4jLog
[main] INFO org.eclipse.jetty.server.Server - jetty-9.4.z-SNAPSHOT
[main] INFO org.eclipse.jetty.server.handler.ContextHandler - Started o.e.j.s.ServletContextHandler@5824a83d{/,null,AVAILABLE}
[main] INFO org.eclipse.jetty.server.AbstractConnector - Started ServerConnector@2ddc9a9f{HTTP/1.1,[http/1.1]}{0.0.0.0:8080}
[main] INFO org.eclipse.jetty.server.Server - Started @1547ms
Dez 12, 2018 10:05:07 PM org.glassfish.jersey.message.internal.WriterInterceptorExecutor$TerminalWriterInterceptor aroundWriteTo
SCHWERWIEGEND: MessageBodyWriter not found for media type=application/json, type=class com.dovydasvenckus.jersey.greeting.Greeting, genericType=class com.dovydasvenckus.jersey.greeting.Greeting.
因此,在我看来,在已编译的JAR中,例如lib丢失。我在网上四处张望,每个人都建议添加org.glassfish.jersey.media:jersey-media-json-jackson:2.27
(或一些变体)以使其正常工作。但这对我不起作用。我的build.gradle
如下:
apply plugin: 'java'
apply plugin: 'application'
apply plugin: 'com.github.johnrengelman.shadow'
sourceCompatibility = 1.8
mainClassName = 'com.dovydasvenckus.jersey.JerseyApplication'
ext {
slf4jVersion = '1.7.25'
jettyVersion = '9.4.6.v20170531'
jerseyVersion = '2.27'
}
buildscript {
repositories {
jcenter()
}
dependencies {
classpath 'com.github.jengelman.gradle.plugins:shadow:2.0.1'
}
}
repositories {
jcenter()
}
dependencies {
compile "org.slf4j:slf4j-api:${slf4jVersion}"
compile "org.slf4j:slf4j-simple:${slf4jVersion}"
compile "org.eclipse.jetty:jetty-server:${jettyVersion}"
compile "org.eclipse.jetty:jetty-servlet:${jettyVersion}"
compile "org.glassfish.jersey.core:jersey-server:${jerseyVersion}"
compile "org.glassfish.jersey.containers:jersey-container-servlet-core:${jerseyVersion}"
compile "org.glassfish.jersey.containers:jersey-container-jetty-http:${jerseyVersion}"
compile "org.glassfish.jersey.media:jersey-media-json-jackson:${jerseyVersion}"
compile "org.glassfish.jersey.inject:jersey-hk2:${jerseyVersion}"
compile 'org.glassfish.jersey.media:jersey-media-json-jackson:2.27'
compile "org.glassfish.jersey.media:jersey-media-moxy:2.27"
compile 'org.glassfish.jersey.media:jersey-media-json-jackson:2.27'
}
jar { manifest { attributes 'Main-Class': "${mainClassName}" } }
有什么想法可以使它生效?
答案 0 :(得分:1)
如my answer的MessageBodyProviderNotFoundException while running jar from command line中所述,有一个“服务文件”,即org.glassfish.jersey.internal.spi.AutoDiscoverable
,它包含在许多jar中。该文件的目的是允许Jersey(和其他第三方)罐子为这些罐子中包含的功能提供一些自动注册。这包括JacksonFeature
的注册,该ServicesResourceTransformer
注册了处理(反序列化)JSON的提供程序。
创建胖(超级)罐子的问题是只能有一个这样的文件(不能有多个同名文件)。因此,在胖罐中包含所有罐之后,将仅包含一个服务文件。
对于Maven,我们将使用maven-shade-plugin,它具有允许转换构建部分的转换器。 shade插件有一个mergeServiceFiles()
,它负责将服务文件的内容串联到一个服务文件中。您用于Gradle的插件Shadow具有相同的功能。您可以通过在配置中调用shadowJar {
mergeServiceFiles()
manifest {
attributes 'Main-Class': 'com.my.Application'
}
}
来配置它。我并没有真正使用Gradle,而是在this issue中进行了说明,建议同时使用以下配置来处理服务文件转换和主类清单转换
jar { manifest { attributes 'Main-Class': "${mainClassName}" } }
所以我假设,使用上述配置,您也可以删除
{{1}}
,因为Shadow插件将负责构建清单。再说一次,我并没有真正使用Gradle,所以这只是一个假设。但是听起来不错。