MULTIPART_FORM_DATA: источник параметра не найден для параметра типа public javax.ws.rs.core.Response
Я использую основанную на Джерси стратегию реализации Restful Service для создания сервиса, который будет использоваться для загрузки файлов. Имя моего класса обслуживания: UploadFileService.java (см. Код ниже)
package com.jerser.service;
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import javax.ws.rs.Consumes;
import javax.ws.rs.POST;
import javax.ws.rs.Path;
import javax.ws.rs.core.MediaType;
import javax.ws.rs.core.Response;
import com.sun.jersey.core.header.FormDataContentDisposition;
import com.sun.jersey.multipart.FormDataParam;
@Path("/fileUpload")
public class UploadFileService {
@POST
@Path("/upload")
@Consumes(MediaType.MULTIPART_FORM_DATA)
public Response uploadFile(
@FormDataParam("file") InputStream uploadedInputStream,
@FormDataParam("file") FormDataContentDisposition fileDetail) {
String uploadedFileLocation = "d://uploaded/" + fileDetail.getFileName();
// save it
writeToFile(uploadedInputStream, uploadedFileLocation);
String output = "File uploaded to : " + uploadedFileLocation;
return Response.status(200).entity(output).build();
}
// save uploaded file to new location
private void writeToFile(InputStream uploadedInputStream,
String uploadedFileLocation) {
try {
OutputStream out = new FileOutputStream(new File(
uploadedFileLocation));
int read = 0;
byte[] bytes = new byte[1024];
out = new FileOutputStream(new File(uploadedFileLocation));
while ((read = uploadedInputStream.read(bytes)) != -1) {
out.write(bytes, 0, read);
}
out.flush();
out.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
Это файлы JAR, которые есть в моей библиотеке:
aopalliance-repackaged-2.4.0-b10.jar asm-debug-all-5.0.2.jar hk2-api-2.4.0-b10.jar hk2-locator-2.4.0-b10.jar hk2-utils-2.4.0-b10.jar javassist-3.18.1-GA.jar javax.annotation-api-1.2.jar javax.inject-2.4.0-b10.jar javax.servlet-api-3.0.1.jar javax.ws.rs-api-2.0.1.jar jaxb-api-2.2.7.jar jersey-client.jar jersey-common.jar jersey-container-servlet-core.jar jersey-container-servlet.jar jersey-core-1.11.jar jersey-guava-2.17.jar jersey-media-jaxb.jar jersey-multipart-1.18.jar jersey-server.jar org.osgi.core-4.2.0.jar osgi-resource-locator-1.0.1.jar persistence-api-1.0.jar validation-api-1.1.0.Final.jar
Я получаю следующую ошибку при попытке загрузить свой сервер Tomcat:
org.glassfish.jersey.server.model.ModelValidationException: Validation of the application resource model has failed during application initialization.
[[FATAL] No injection source found for a parameter of type public javax.ws.rs.core.Response com.jerser.service.UploadFileService.uploadFile(java.io.InputStream,com.sun.jersey.core.header.FormDataContentDisposition) at index 0.; source='ResourceMethod{httpMethod=POST, consumedTypes=[multipart/form-data], producedTypes=[], suspended=false, suspendTimeout=0, suspendTimeoutUnit=MILLISECONDS, invocable=Invocable{handler=ClassBasedMethodHandler{handlerClass=class com.jerser.service.UploadFileService, handlerConstructors=[org.glassfish.jersey.server.model.HandlerConstructor@d3e2d4]}, definitionMethod=public javax.ws.rs.core.Response com.jerser.service.UploadFileService.uploadFile(java.io.InputStream,com.sun.jersey.core.header.FormDataContentDisposition), parameters=[Parameter [type=class java.io.InputStream, source=file, defaultValue=null], Parameter [type=class com.sun.jersey.core.header.FormDataContentDisposition, source=file, defaultValue=null]], responseType=class javax.ws.rs.core.Response}, nameBindings=[]}']
at org.glassfish.jersey.server.ApplicationHandler.initialize(ApplicationHandler.java:528)
at org.glassfish.jersey.server.ApplicationHandler.access$500(ApplicationHandler.java:166)
at org.glassfish.jersey.server.ApplicationHandler$3.run(ApplicationHandler.java:327)
at org.glassfish.jersey.internal.Errors$2.call(Errors.java:289)
at org.glassfish.jersey.internal.Errors$2.call(Errors.java:286)
at org.glassfish.jersey.internal.Errors.process(Errors.java:315)
at org.glassfish.jersey.internal.Errors.process(Errors.java:297)
at org.glassfish.jersey.internal.Errors.processWithException(Errors.java:286)
at org.glassfish.jersey.server.ApplicationHandler.<init>(ApplicationHandler.java:324)
at org.glassfish.jersey.servlet.WebComponent.<init>(WebComponent.java:338)
at org.glassfish.jersey.servlet.ServletContainer.init(ServletContainer.java:171)
at org.glassfish.jersey.servlet.ServletContainer.init(ServletContainer.java:363)
at javax.servlet.GenericServlet.init(GenericServlet.java:160)
at org.apache.catalina.core.StandardWrapper.initServlet(StandardWrapper.java:1176)
at org.apache.catalina.core.StandardWrapper.loadServlet(StandardWrapper.java:1102)
at org.apache.catalina.core.StandardWrapper.load(StandardWrapper.java:1009)
at org.apache.catalina.core.StandardContext.loadOnStartup(StandardContext.java:4885)
at org.apache.catalina.core.StandardContext$3.call(StandardContext.java:5212)
at org.apache.catalina.core.StandardContext$3.call(StandardContext.java:5207)
at java.util.concurrent.FutureTask$Sync.innerRun(Unknown Source)
at java.util.concurrent.FutureTask.run(Unknown Source)
at java.util.concurrent.ThreadPoolExecutor.runWorker(Unknown Source)
at java.util.concurrent.ThreadPoolExecutor$Worker.run(Unknown Source)
at java.lang.Thread.run(Unknown Source)
В Интернете я обнаружил множество примеров, показывающих, как загрузить файл MULTIPART с помощью API RESTFul. Но с тем же решением. Я не могу запустить этот код, а также. Я думаю, что я делаю что-то не так с файлами JAR. Может ли кто-нибудь помочь мне в этом?
12 ответов
Избавляться от jersey-multipart-1.18.jar
, Это для Джерси 1.x. Добавьте эти два
Для Maven вы должны использовать следующую зависимость (вам не нужно явно добавлять mimepull
зависимость, как это будет тянуть его).
<dependency>
<groupId>org.glassfish.jersey.media</groupId>
<artifactId>jersey-media-multipart</artifactId>
<version>2.17</version> <!-- Make sure the Jersey version matches
the one you are currently using -->
</dependency>
Тогда вам нужно зарегистрировать MultiPartFeature
, Если вы используете ResourceConfig
для конфигурации, вы можете просто сделать
register(MultiPartFeature.class);
Если вы используете web.xml, вы можете добавить класс как <init-param>
сервлету Джерси
<init-param>
<param-name>jersey.config.server.provider.classnames</param-name>
<param-value>org.glassfish.jersey.media.multipart.MultiPartFeature</param-value>
</init-param>
Обратите внимание: если у вас есть несколько провайдеров, которых вы хотите зарегистрировать, вы можете разделить каждый класс провайдеров запятой или точкой с запятой. Вы не можете использовать это же param-name
дважды. Смотрите ответ Суарабха
ОБНОВИТЬ
Кроме того, как только вы избавитесь от jersey-multipart-1.18.jar
у вас будут ошибки компиляции для отсутствующих импортированных классов. По большей части имена классов остаются прежними, только пакеты изменились, т.е.
org.glassfish.jersey.media.multipart.FormDataParam
org.glassfish.jersey.media.multipart.FormDataContentDisposition
В сторону
Если вы здесь для другого ModelValidationException
Вот несколько ссылок на информацию о других причинах исключения.
Еще одна возможная причина этой очень общей ошибки состоит в том, что Джерси ищет фабрики, связанные с последней аннотацией, только когда несколько объявлено в параметре. (См. Отчет об ошибке)
Пока это не исправлено, если вы используете любые другие аннотации, кроме @FormDataParam
, это должно прийти последним.
Это работает:
@NotEmpty @FormDataParam("myParam") String myParam
Это не:
@FormDataParam("myParam") @NotEmpty String myParam
Я тоже получил то же исключение. Я сделал следующие изменения в web.xml
<init-param>
<param-name>jersey.config.server.provider.classnames</param-name>
<param-value>org.glassfish.jersey.filter.LoggingFilter;org.glassfish.jersey.moxy.json.MoxyFeature;org.glassfish.jersey.media.multipart.MultiPartFeature</param-value>
</init-param>
и изменил Джерси с 2,7 до 2,9 . Я не знаю, что изменение этого 2 решило проблему.
Если кто-то использует @FormDataParam
с @ApiOperation
аннотация swagger, она не будет работать (согласно последней версии swagger в настоящее время), как упомянуто здесь:
Зарегистрировать MultiPartFeature. В web.xml добавьте в сервлет Джерси:
<init-param>
<param-name>jersey.config.server.provider.classnames</param-name>
<param-value>org.glassfish.jersey.media.multipart.MultiPartFeature</param-value>
</init-param>
Ниже код работал у меня:
Класс ->>> добавить
Свойство класса --->> добавить
Public Class userREST () {
@POST
@Path("upload")
@Consumes(MediaType.MULTIPART_FORM_DATA)
@Produces(MediaType.APPLICATION_JSON)
public Response uploadImageFile(@FormDataParam("uploadFile") InputStream fileInputStream,
@FormDataParam("uploadFile") FormDataContentDisposition fileFormDataContentDisposition,
@FormDataParam("FIR_REG_NUM") String FIR_REG_NUM, @FormDataParam("LOGIN_ID") String LOGIN_ID) {
final_json_result = WriteFileInFolder.fileAnalysis(fileInputStream, fileFormDataContentDisposition, FIR_REG_NUM,
LOGIN_ID);
return Response.ok(final_json_result).build();
}// uploadImageFile
Открытый класс FileJAXRSConfig () {
package ####.jaxrs.jwt;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Map;
import java.util.Set;
import javax.ws.rs.ApplicationPath;
import javax.ws.rs.core.Application;
import ####.helper.Common@@@;
import ####.jaxrs.jwt.filters.JWTRequestFilter;
import ####.jaxrs.jwt.filters.JWTResponseFilter;
import ####.service.FileServicesREST;
@ApplicationPath("fileservice")
public class FileJAXRSConfig extends Application {
@Override
public Set<Class<?>> getClasses() {
Common@@@.logging("@ApplicationPath@FileServicesREST...");
Set<Class<?>> clazzes = new HashSet<Class<?>>();
clazzes.add(JWTRequestFilter.class);
clazzes.add(FileServicesREST.class);
clazzes.add(JWTResponseFilter.class);
return clazzes;
}
@Override
public Map<String, Object> getProperties() {
Map<String, Object> properties = new HashMap<String, Object>();
properties.put("jersey.config.server.provider.packages", "####.service");
properties.put("jersey.config.server.provider.classnames", "org.glassfish.jersey.media.multipart.MultiPartFeature");
return properties;
}
}
Не нужно добавлять следующее в web.xml
<init-param>
<param-name>jersey.config.server.provider.packages</param-name>
<param-value>mha.@@@.service</param-value>
</init-param>
<init-param>
<param-name>jersey.config.server.provider.classnames</param-name>
<param-value>org.glassfish.jersey.media.multipart.MultiPartFeature</param-value>
</init-param>
Если вы получаете эту ошибку при написании тестов Dropwizard для ресурса загрузки, вот решение:
- добавить зависимость от dropwizard-форм
2. добавьте это в файл приложения:
@Override
public void initialize(Bootstrap<ExampleConfiguration> bootstrap) {
bootstrap.addBundle(new MultiPartBundle());
}
- в тестовом файле добавить:
ResourceExtension.builder()
.addResource(new FileResource())
.addProvider(new MultiPartFeature())
.build();
У меня была такая же проблема, когда я пытался загрузить файл. Я провел много времени, пока не нашел решение проблемы.
1. Если вы изменили версию своих файлов JAR, у вас могут возникнуть конфликты версий!
Очистите ваши артефакты / библиотеки и пересоберите проект.
2.Вы также должны зарегистрировать свой класс UploadFileService:
register(MultiPartFeature.class);
register(UploadFileService.class);
Надеюсь, это поможет кому-то и сэкономит ваше время.
У меня была такая же проблема со Scala, и это помогло мне решить ее. Просто хочу добавить некоторые подробности о Scala, чтобы помочь любому, кто использует Dropwizard с Scala. Вот пример того, как "зарегистрировать" MultiPartFeature в проекте Scala и Dropwizard.
package org.research.s3.service
import io.dropwizard.Application
import io.dropwizard.setup.Environment
import org.research.s3.service.resource._
import org.research.service.s3.resource.UploadResource
import org.glassfish.jersey.media.multipart.{FormDataParam,MultiPartFeature}
class CmdaaApp() extends Application[CmdaaAppConfig] {
override def run(t: CmdaaAppConfig, env: Environment): Unit = {
env.jersey().register(new RootResource)
//Need this to make the file upload code work in
env.jersey().register(new MultiPartFeature)
env.jersey().register(new UploadResource(curBucket))
}
}
object CmdaaApp {
def main(args: Array[String]): Unit = new CmdaaApp().run(args: _*)
}
и вот код для UploadResource, который выполняет загрузку:
package org.research.service.s3.resource
import java.io.{FileInputStream, InputStream}
import com.google.gson.{Gson, GsonBuilder}
import javax.ws.rs.core.MediaType.APPLICATION_JSON
import javax.ws.rs._
import javax.ws.rs.core.Response
import javax.ws.rs.core.MediaType
import org.research.util.OptionSerializer
import org.research.s3.service.resource.s3Bucket
import org.glassfish.jersey.media.multipart.{FormDataParam,MultiPartFeature}
@Path("/file")
class UploadResource(currentBucket: s3Bucket) {
val gsonb = new GsonBuilder()
gsonb.registerTypeAdapter(classOf[Option[Any]], new OptionSerializer)
val gson = gsonb.create
@POST
@Path("upload")
@Produces(Array(APPLICATION_JSON))
@Consumes(Array(MediaType.MULTIPART_FORM_DATA))
// def uploadFile(): Response = {
def uploadFile(@FormDataParam("file") uploadedInputStream: InputStream): Response = {
/* Need code here to get a uuid for the file name
Then return the uuid if we have success and of course 200
*/
Response.ok.entity(currentBucket.upload("testName",uploadedInputStream,false)).build()
//Response.ok().build()
}
}
Этот код относится к корзине s3, но вам это не нужно. Вы можете просто заменить этот вызов кодом и загрузить данные входящего файла в обычный файл. Скала
Действительно, в моем web.xml отсутствовал «org.glassfish.jersey.media.multipart.MultiPartFeature» (среди других классов) в значении параметра «jersey.config.server.provider.classnames». По меньшей мере :
<init-param>
<param-name>jersey.config.server.provider.classnames</param-name>
<param-value>org.glassfish.jersey.media.multipart.MultiPartFeature</param-value>
</init-param>
У меня была очень похожая проблема, и ответ, который мне помог, был /questions/29755048/peredacha-parametra-zaprosa-nestandartnogo-tipa/29755061#29755061
Я пытался использовать определенный пользователем объект в качестве параметра запроса и из ответа, который обычно не допускается, если он не соответствует некоторым условиям.
На случай, если кто-то столкнется с этим в будущем и столкнется с той же проблемой, с которой столкнулся я. Убедитесь, что аннотации, которые вы импортируете, взяты из правильных пакетов. Исключение не сообщает вам, какой параметр является проблемой, это может быть любой из параметров. В моем случае я импортировал
javax.websocket.server.PathParam
вместо
javax.ws.rs.PathParam
.