Jersy / Jetty服务器的MessageBodyWriter找不到错误

时间:2018-12-12 21:10:59

标签: java rest gradle jersey jetty

我刚刚关注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}" } }

有什么想法可以使它生效?

1 个答案:

答案 0 :(得分:1)

my answerMessageBodyProviderNotFoundException 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,所以这只是一个假设。但是听起来不错。