Джерси не удается при создании Uber JAR с Maven-Assembly-плагин
Я создал веб-приложение для начинающих из джерси maven. Также я встроил в свое приложение Jetty-сервер с помощью плагина Jetty.
Мой проект работает нормально, когда я запускаю свой проект с помощью команды mvn jetty: run.
Но когда я упаковываю свой проект с помощью команды mvn clean package и запускаю файл jar с именем jar-with-dependencies, проект выдает это исключение, возвращая ответ json из ресурса jersey.
SEVERE: MessageBodyWriter не найден для типа носителя =application/json, type=class com.nitish.freecharge.model.Count, genericType=class com.nitish.freecharge.model.Count.
Вот мой файл pom.xml
http://maven.apache.org/maven-v4_0_0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>com.nitish.freecharge</groupId>
<artifactId>wordcount</artifactId>
<packaging>war</packaging>
<version>2.0</version>
<name>wordcount</name>
<build>
<finalName>wordcount</finalName>
<resources>
<resource>
<directory>src/main/java</directory>
</resource>
<resource>
<directory>src/main/resources</directory>
</resource>
<resource>
<directory>src/main/webapp</directory>
</resource>
</resources>
<plugins>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-compiler-plugin</artifactId>
<version>2.5.1</version>
<inherited>true</inherited>
<configuration>
<source>1.7</source>
<target>1.7</target>
</configuration>
</plugin>
<plugin>
<groupId>org.eclipse.jetty</groupId>
<artifactId>jetty-maven-plugin</artifactId>
<version>9.3.0.v20150612</version>
<configuration>
<scanIntervalSeconds>5</scanIntervalSeconds>
<webApp>
<contextPath>/wordcount</contextPath>
</webApp>
<httpConnector>
<!--host>localhost</host -->
<port>9999</port>
</httpConnector>
</configuration>
</plugin>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-jar-plugin</artifactId>
<version>2.4</version>
<executions>
<execution>
<id>package-jar</id>
<phase>package</phase>
<goals>
<goal>jar</goal>
</goals>
</execution>
</executions>
</plugin>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-assembly-plugin</artifactId>
<version>2.6</version>
<configuration>
<finalName>awesomeProject</finalName>
<descriptorRefs>
<descriptorRef>jar-with-dependencies</descriptorRef>
</descriptorRefs>
<appendAssemblyId>false</appendAssemblyId>
<archive>
<manifest>
<mainClass>App</mainClass>
</manifest>
</archive>
</configuration>
<executions>
<execution>
<phase>package</phase>
<goals>
<goal>single</goal>
</goals>
</execution>
</executions>
</plugin>
</plugins>
</build>
<dependencyManagement>
<dependencies>
<dependency>
<groupId>org.glassfish.jersey</groupId>
<artifactId>jersey-bom</artifactId>
<version>${jersey.version}</version>
<type>pom</type>
<scope>import</scope>
</dependency>
</dependencies>
</dependencyManagement>
<dependencies>
<dependency>
<groupId>org.glassfish.jersey.core</groupId>
<artifactId>jersey-server</artifactId>
</dependency>
<dependency>
<groupId>org.glassfish.jersey.containers</groupId>
<artifactId>jersey-container-servlet-core</artifactId>
</dependency>
<dependency>
<groupId>org.glassfish.jersey.media</groupId>
<artifactId>jersey-media-moxy</artifactId>
</dependency>
<dependency>
<groupId>org.eclipse.jetty</groupId>
<artifactId>jetty-server</artifactId>
<version>9.3.8.v20160314</version>
</dependency>
<dependency>
<groupId>org.eclipse.jetty</groupId>
<artifactId>jetty-servlet</artifactId>
<version>9.3.8.v20160314</version>
</dependency>
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>4.4</version>
</dependency>
</dependencies>
<properties>
<jersey.version>2.22.2</jersey.version>
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
</properties>
Я создал класс Main Driver как App.java в пакете по умолчанию. Вот мой контент App.java
import org.eclipse.jetty.server.Server;
import org.eclipse.jetty.servlet.ServletContextHandler;
import org.eclipse.jetty.servlet.ServletHolder;
public class App {
public static void main(String []gg){
Server server = new Server(9999);
ServletContextHandler context = new ServletContextHandler(ServletContextHandler.NO_SESSIONS);
context.setContextPath("/");
server.setHandler(context);
ServletHolder jerseyServlet = context.addServlet(org.glassfish.jersey.servlet.ServletContainer.class, "/wordcount/*");
jerseyServlet.setInitOrder(1);
jerseyServlet.setInitParameter("jersey.config.server.provider.packages","com.nitish.freecharge.resources");
try {
System.out.println("Starting the server..");
server.start();
System.out.println("Server started");
server.join();
} catch(Exception e) {
System.out.println("Exception in starting the server ");
e.printStackTrace();
}
}
}
Вот мой единственный класс ресурсов джерси, который выполняется, когда я получаю доступ к своему URL проекта после запуска сервера:
package com.nitish.freecharge.resources;
import javax.ws.rs.GET;
import javax.ws.rs.Path;
import javax.ws.rs.Produces;
import javax.ws.rs.QueryParam;
import javax.ws.rs.core.MediaType;
import javax.ws.rs.core.Response;
import javax.ws.rs.core.Response.Status;
import com.nitish.freecharge.dao.FileDAO;
import com.nitish.freecharge.model.Count;
/**
* Root resource (exposed at "count" path) which handles HTTP GET method and returns the count value;
*/
@Path("/count")
public class CountResource {
private FileDAO fileDAO=new FileDAO();
/**
* Method handling HTTP GET requests. The returned object will be sent
* to the client as "application/json" media type.
*
* @return String that will be returned as a application/json response.
*/
@GET
@Produces(MediaType.APPLICATION_JSON)
@QueryParam("query")
public Response getWordCount(@QueryParam("query")String query) {
Error error=null;
Count count=null;
try{
if(query!=null){
query=query.trim();
if(query.length()>0 && query.matches("^[A-Za-z]+$")){
long c=fileDAO.getCount(query.toLowerCase());
count=new Count(c);
}else{
error=new Error("Some Error Occured.Please Try Again With a new word!");
}
}else{
error=new Error("Some Error Occured.Please Try Again!");
}
}catch(Exception e){
error=new Error(e.getMessage());
return Response.status(Status.INTERNAL_SERVER_ERROR).entity(error).build();
}
if(count!=null){
return Response.status(Status.OK).entity(count).build();
}else{
return Response.status(Status.BAD_REQUEST).entity(error).build();
}
}
}
После упаковки и запуска полного встроенного проекта с помощью команды java -jar awesomeProject.jar
Я получаю этот вывод на серверной подсказке
Я много пробовал и не смог упаковать мое встроенное веб-приложение таким образом, чтобы эта проблема была решена. Я новичок в Maven и упаковке. Пожалуйста, помогите, где я совершаю ошибку.
1 ответ
Если вы заглянете внутрь банки MOXy, вы увидите папку META-INF/services
, В этой папке вы увидите файл с именем org.glassfish.jersey.internal.spi.AutoDiscoverable
, Содержимое этого файла должно быть одной строкой
org.glassfish.jersey.moxy.json.internal.MoxyJsonAutoDiscoverable
Этот файл предназначен для того, чтобы Джерси обнаружил MoxyJsonAutoDiscoverable
, который регистрирует МОКСИ для Джерси. Этот шаблон загрузки сервисов позволяет Джерси обнаруживать особенности и регистрировать их, без необходимости регистрировать их самостоятельно.
Проблема, возникающая при создании jar-файла uber, заключается в том, что может быть несколько jar-файлов с одним и тем же файлом, поскольку разные jar-файлы имеют разные функции для обнаружения, но файл должен иметь именно такое имя, как это работает в шаблоне загрузчика служб.
Таким образом, у вас есть несколько jar-файлов с одним и тем же файлом, но когда вы создаете uber-jar, вы не можете иметь несколько файлов с одним и тем же именем. Это просто невозможно. Таким образом, только один из файлов помещается в финальную банку. Который.. кто знает. Но это означает, что если файлы MOXy - это не один файл, его функция не будет обнаружена автоматически, и мы должны зарегистрировать его самостоятельно. Таким образом, классы упакованы в Uber JAR, но основной компонент функции просто не зарегистрирован. Вы можете просто зарегистрировать это самостоятельно
jerseyServlet.setInitParameter("jersey.config.server.provider.classnames",
"org.glassfish.jersey.moxy.json.MoxyJsonFeature");
но как насчет всех других возможных функций, которые, возможно, не включены, поскольку их автоматически обнаруживаемый файл не включен?
По этой причине вместо сборочного плагина вы должны использовать maven-shade-plugin
, который имеет преобразователи, которые позволяют нам объединить содержимое служебных файлов в один файл.
Конфигурация будет выглядеть примерно так
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-shade-plugin</artifactId>
<version>2.3</version>
<configuration>
<createDependencyReducedPom>true</createDependencyReducedPom>
<filters>
<filter>
<artifact>*:*</artifact>
<excludes>
<exclude>META-INF/*.SF</exclude>
<exclude>META-INF/*.DSA</exclude>
<exclude>META-INF/*.RSA</exclude>
</excludes>
</filter>
</filters>
</configuration>
<executions>
<execution>
<phase>package</phase>
<goals>
<goal>shade</goal>
</goals>
<configuration>
<transformers>
<transformer implementation="org.apache.maven.plugins.shade.resource.ServicesResourceTransformer"/>
<transformer implementation="org.apache.maven.plugins.shade.resource.ManifestResourceTransformer">
<mainClass>com.example.YourApp</mainClass>
</transformer>
</transformers>
</configuration>
</execution>
</executions>
</plugin>
ServicesResorceTransformaer
это то, что объединяет файлы. Эта конкретная конфигурация плагина была взята с начала работы Dropwizard. Вы можете проверить это для дальнейшего объяснения.