Jersy / Jetty 服务器未找到 MessageBodyWriter 错误
MessageBodyWriter not found error for Jersy / Jetty server
我一直在关注 this tutorial to create a REST API using Jersey on Jetty,我喜欢这个结果。一切正常。但是,如果我 运行 Gradle shadowJar
生成一个胖 jar 文件的任务工作正常并且文件也 运行s 但在发出请求时以错误消息结尾:
$ 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 中,一些例如缺少库。我在网上环顾四周,每个人都建议添加 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}" } }
有什么想法可以让它发挥作用吗?
在 in 中提到,有一个"services file",即org.glassfish.jersey.internal.spi.AutoDiscoverable
,很多jar中都有。此文件的目的是允许 Jersey(和其他第三方)jar 为这些 jar 中包含的功能提供一些自动注册。这包括 JacksonFeature
的注册,它注册了处理(反)序列化的 JSON 提供程序。
创建 fat (uber) jar 的问题在于只能有其中一个文件(同名文件不能超过一个)。所以fat jar中包含的所有jar,只会包含一个服务文件。
对于 Maven,我们将使用 maven-shade-plugin,它具有允许转换部分构建的转换器。阴影插件有一个 ServicesResourceTransformer
负责将服务文件的内容连接到一个服务文件中。您用于 Gradle、Shadow, has the same facility. You configure it by calling mergeServiceFiles()
in the configuration. I don't really work with Gradle, but also stated in this issue 的插件,用于处理服务文件转换和主要 class 清单转换的推荐配置如下
shadowJar {
mergeServiceFiles()
manifest {
attributes 'Main-Class': 'com.my.Application'
}
}
所以我假设,通过上面的配置,你也可以删除
jar { manifest { attributes 'Main-Class': "${mainClassName}" } }
因为 Shadow 插件将负责构建清单。同样,我并没有真正使用 Gradle,所以这只是一个假设;但听起来不错。
我一直在关注 this tutorial to create a REST API using Jersey on Jetty,我喜欢这个结果。一切正常。但是,如果我 运行 Gradle shadowJar
生成一个胖 jar 文件的任务工作正常并且文件也 运行s 但在发出请求时以错误消息结尾:
$ 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 中,一些例如缺少库。我在网上环顾四周,每个人都建议添加 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}" } }
有什么想法可以让它发挥作用吗?
在org.glassfish.jersey.internal.spi.AutoDiscoverable
,很多jar中都有。此文件的目的是允许 Jersey(和其他第三方)jar 为这些 jar 中包含的功能提供一些自动注册。这包括 JacksonFeature
的注册,它注册了处理(反)序列化的 JSON 提供程序。
创建 fat (uber) jar 的问题在于只能有其中一个文件(同名文件不能超过一个)。所以fat jar中包含的所有jar,只会包含一个服务文件。
对于 Maven,我们将使用 maven-shade-plugin,它具有允许转换部分构建的转换器。阴影插件有一个 ServicesResourceTransformer
负责将服务文件的内容连接到一个服务文件中。您用于 Gradle、Shadow, has the same facility. You configure it by calling mergeServiceFiles()
in the configuration. I don't really work with Gradle, but also stated in this issue 的插件,用于处理服务文件转换和主要 class 清单转换的推荐配置如下
shadowJar {
mergeServiceFiles()
manifest {
attributes 'Main-Class': 'com.my.Application'
}
}
所以我假设,通过上面的配置,你也可以删除
jar { manifest { attributes 'Main-Class': "${mainClassName}" } }
因为 Shadow 插件将负责构建清单。同样,我并没有真正使用 Gradle,所以这只是一个假设;但听起来不错。