On 10/05/12 12:14, Christian Lipp wrote:
Adding @Encoded annotation did not change the behaviour, replacing @FormParam 
with @Multipart fixed the issue with QueryDSL, but not with HttpClient.
Anyway, since I know now that the service is working I will be able to fix the 
client myself.
OK, thanks for the confirmation.
FYI, the fix for giving a best effort for supporting the mix-up of @FormParam and @Multipart annotations is on the way. I guess it's better for such cases to work if possible as opposed to failing.

The tricky bit is that the @FormParam is a reserved JAX-RS annotation so before the MultipartProvider is given a chance to process the payload, the runtime assumes that it is the list of simple name/value pairs; the only realistic case to ignore complex parts at this stage is to check that Content-Disposition 'name' parameter is not set to the reserved 'files' value and provide a warning about the possible mix-up if this check fails to filter out the files...

Cheers, Sergey


Thanks for your help!
Kind regards, Christian

-----Ursprüngliche Nachricht-----
Von: Sergey Beryozkin [mailto:[email protected]]
Gesendet: Donnerstag, 10. Mai 2012 13:00
An: [email protected]
Betreff: Re: JAX-RS multipart file upload

On 10/05/12 11:51, Sergey Beryozkin wrote:
Hi

Please see comments below...

On 10/05/12 11:16, Christian Lipp wrote:
Hello!

I want to upload a file using multipart form data and have problems
with URLDecoder. The service code follows:

@POST
@Path("/document")
@Consumes(MediaType.MULTIPART_FORM_DATA)
public Response storeTravelDocument(
@Context UriInfo uriInfo,
@FormParam(value = "IDNR") String idnr,
@FormParam(value="DOCNR") String documentNr,
@FormParam(value="ISSUE_DATE") String issueDate, @Multipart(value =
"image", type="image/jpeg") InputStream pictureStream) {..


What happens with this declaration is that the form payload processor
assumes the last parameter being just one more *plain*
multipart/form-data parameter. I'll check what can be fixed there, you
can definitely try adding @Encoded annotation to the last parameter, but
I suspect you may get a ClassCastException - try it please.

Replacing @FormParam with @Multipart should fix the issue (as an
alternative to introducing MultipartBody), however the mix-up of
@FormParam&  @Multipart should ideally work too. will update later on on
this issue

Sergey


When you have a 'mixed' multipart/form-data payload, containing simple
String name/values pairs, alongside the binary parts, it is better to
avoid using @FormParam or @Multipart annotations and only use a
MultipartBody input parameter - it will provide a type-safe access to
all the individual parts

Cheers, Sergey


I tested with three clients, for example rest-assured:

given().
formParam("IDNR", "A000000A").
formParam("DOCNR", "00001").
formParam("ISSUE_DATE", "14.06.2010").
multiPart("image", new File(picturePath), "image/jpeg").
expect().
statusCode(Response.Status.CREATED.getStatusCode()).
when().
post("/document");

or HttpClient:

// build mulitpart entity
MultipartEntity entity = new MultipartEntity();

entity.addPart("IDNR", new StringBody("A000000A"));
entity.addPart("DOCID", new StringBody("00001"));
entity.addPart("ISSUE_DATE", new StringBody("14.06.2010"));

FileBody fileBody = new FileBody(new File(picturePath), "image/jpeg");
entity.addPart("image", fileBody);

// build post request
String uri = System.getProperty("service.url") + "/document";
HttpClient httpClient = new DefaultHttpClient();
HttpPost httpPost = new HttpPost(uri);
httpPost.setEntity(entity);

// execute request
HttpResponse response = httpClient.execute(httpPost);
int code = response.getStatusLine().getStatusCode();

// validate
assert code == Response.Status.CREATED.getStatusCode();

in all cases I receive the following exception:

in escape (%) pattern - For input string: ""
at java.net.URLDecoder.decode(URLDecoder.java:192) ~[na:1.7.0_03]
at org.apache.cxf.common.util.UrlUtils.urlDecode(UrlUtils.java:55)
~[cxf-api-2.6.0.jar:2.6.0]
at org.apache.cxf.common.util.UrlUtils.urlDecode(UrlUtils.java:63)
~[cxf-api-2.6.0.jar:2.6.0]
at org.apache.cxf.jaxrs.utils.HttpUtils.urlDecode(HttpUtils.java:81)
~[cxf-rt-frontend-jaxrs-2.6.0.jar:2.6.0]
at
org.apache.cxf.jaxrs.utils.FormUtils.populateMapFromMultipart(FormUtils.java:170)
~[cxf-rt-frontend-jaxrs-2.6.0.jar:2.6.0]
at
org.apache.cxf.jaxrs.utils.JAXRSUtils.processFormParam(JAXRSUtils.java:746)
~[cxf-rt-frontend-jaxrs-2.6.0.jar:2.6.0]
at
org.apache.cxf.jaxrs.utils.JAXRSUtils.createHttpParameterValue(JAXRSUtils.java:667)
~[cxf-rt-frontend-jaxrs-2.6.0.jar:2.6.0]
at
org.apache.cxf.jaxrs.utils.JAXRSUtils.processParameter(JAXRSUtils.java:625)
~[cxf-rt-frontend-jaxrs-2.6.0.jar:2.6.0]
at
org.apache.cxf.jaxrs.utils.JAXRSUtils.processParameters(JAXRSUtils.java:578)
~[cxf-rt-frontend-jaxrs-2.6.0.jar:2.6.0]
at
org.apache.cxf.jaxrs.interceptor.JAXRSInInterceptor.processRequest(JAXRSInInterceptor.java:238)
~[cxf-rt-frontend-jaxrs-2.6.0.jar:2.6.0]
at
org.apache.cxf.jaxrs.interceptor.JAXRSInInterceptor.handleMessage(JAXRSInInterceptor.java:89)
~[cxf-rt-frontend-jaxrs-2.6.0.jar:2.6.0]
at
org.apache.cxf.phase.PhaseInterceptorChain.doIntercept(PhaseInterceptorChain.java:262)
[cxf-api-2.6.0.jar:2.6.0]
... 25 common frames omitted
2012-05-10 12:07:54,873 [http-bio-8080-exec-5] ERROR
org.apache.cxf.interceptor.AbstractFaultChainInitiatorObserver - Error
occurred during error handling, give up!
org.apache.cxf.interceptor.Fault: URLDecoder: Illegal hex characters
in escape (%) pattern - For input string: ""
at
org.apache.cxf.interceptor.AbstractFaultChainInitiatorObserver.onMessage(AbstractFaultChainInitiatorObserver.java:75)
~[cxf-api-2.6.0.jar:2.6.0]
at
org.apache.cxf.phase.PhaseInterceptorChain.doIntercept(PhaseInterceptorChain.java:322)
[cxf-api-2.6.0.jar:2.6.0]
at
org.apache.cxf.transport.ChainInitiationObserver.onMessage(ChainInitiationObserver.java:122)
[cxf-api-2.6.0.jar:2.6.0]
at
org.apache.cxf.transport.http.AbstractHTTPDestination.invoke(AbstractHTTPDestination.java:211)
[cxf-rt-transports-http-2.6.0.jar:2.6.0]
at
org.apache.cxf.transport.servlet.ServletController.invokeDestination(ServletController.java:213)
[cxf-rt-transports-http-2.6.0.jar:2.6.0]
at
org.apache.cxf.transport.servlet.ServletController.invoke(ServletController.java:154)
[cxf-rt-transports-http-2.6.0.jar:2.6.0]
at
org.apache.cxf.transport.servlet.CXFNonSpringServlet.invoke(CXFNonSpringServlet.java:129)
[cxf-rt-transports-http-2.6.0.jar:2.6.0]
at
org.apache.cxf.transport.servlet.AbstractHTTPServlet.handleRequest(AbstractHTTPServlet.java:187)
[cxf-rt-transports-http-2.6.0.jar:2.6.0]
at
org.apache.cxf.transport.servlet.AbstractHTTPServlet.doPost(AbstractHTTPServlet.java:110)
[cxf-rt-transports-http-2.6.0.jar:2.6.0]
at javax.servlet.http.HttpServlet.service(HttpServlet.java:641)
[servlet-api.jar:na]
at
org.apache.cxf.transport.servlet.AbstractHTTPServlet.service(AbstractHTTPServlet.java:166)
[cxf-rt-transports-http-2.6.0.jar:2.6.0]
at
org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:305)
[catalina.jar:7.0.27]
at
org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:210)
[catalina.jar:7.0.27]
at
org.apache.catalina.core.StandardWrapperValve.invoke(StandardWrapperValve.java:225)
[catalina.jar:7.0.27]
at
org.apache.catalina.core.StandardContextValve.invoke(StandardContextValve.java:169)
[catalina.jar:7.0.27]
at
org.apache.catalina.authenticator.AuthenticatorBase.invoke(AuthenticatorBase.java:472)
[catalina.jar:7.0.27]
at
org.apache.catalina.core.StandardHostValve.invoke(StandardHostValve.java:168)
[catalina.jar:7.0.27]
at
org.apache.catalina.valves.ErrorReportValve.invoke(ErrorReportValve.java:98)
[catalina.jar:7.0.27]
at
org.apache.catalina.valves.AccessLogValve.invoke(AccessLogValve.java:927)
[catalina.jar:7.0.27]
at
org.apache.catalina.core.StandardEngineValve.invoke(StandardEngineValve.java:118)
[catalina.jar:7.0.27]
at
org.apache.catalina.connector.CoyoteAdapter.service(CoyoteAdapter.java:407)
[catalina.jar:7.0.27]
at
org.apache.coyote.http11.AbstractHttp11Processor.process(AbstractHttp11Processor.java:999)
[tomcat-coyote.jar:7.0.27]
at
org.apache.coyote.AbstractProtocol$AbstractConnectionHandler.process(AbstractProtocol.java:565)
[tomcat-coyote.jar:7.0.27]
at
org.apache.tomcat.util.net.JIoEndpoint$SocketProcessor.run(JIoEndpoint.java:307)
[tomcat-coyote.jar:7.0.27]
at
java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1110)
[na:1.7.0_03]
at
java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:603)
[na:1.7.0_03]
at java.lang.Thread.run(Thread.java:722) [na:1.7.0_03]
Caused by: java.lang.IllegalArgumentException: URLDecoder: Illegal hex
characters in escape (%) pattern - For input string: ""
at java.net.URLDecoder.decode(URLDecoder.java:192) ~[na:1.7.0_03]
at org.apache.cxf.common.util.UrlUtils.urlDecode(UrlUtils.java:55)
~[cxf-api-2.6.0.jar:2.6.0]
at org.apache.cxf.common.util.UrlUtils.urlDecode(UrlUtils.java:63)
~[cxf-api-2.6.0.jar:2.6.0]
at org.apache.cxf.jaxrs.utils.HttpUtils.urlDecode(HttpUtils.java:81)
~[cxf-rt-frontend-jaxrs-2.6.0.jar:2.6.0]
at
org.apache.cxf.jaxrs.utils.FormUtils.populateMapFromMultipart(FormUtils.java:170)
~[cxf-rt-frontend-jaxrs-2.6.0.jar:2.6.0]
at
org.apache.cxf.jaxrs.utils.JAXRSUtils.processFormParam(JAXRSUtils.java:746)
~[cxf-rt-frontend-jaxrs-2.6.0.jar:2.6.0]
at
org.apache.cxf.jaxrs.utils.JAXRSUtils.createHttpParameterValue(JAXRSUtils.java:667)
~[cxf-rt-frontend-jaxrs-2.6.0.jar:2.6.0]
at
org.apache.cxf.jaxrs.utils.JAXRSUtils.processParameter(JAXRSUtils.java:625)
~[cxf-rt-frontend-jaxrs-2.6.0.jar:2.6.0]
at
org.apache.cxf.jaxrs.utils.JAXRSUtils.processParameters(JAXRSUtils.java:578)
~[cxf-rt-frontend-jaxrs-2.6.0.jar:2.6.0]
at
org.apache.cxf.jaxrs.interceptor.JAXRSInInterceptor.processRequest(JAXRSInInterceptor.java:238)
~[cxf-rt-frontend-jaxrs-2.6.0.jar:2.6.0]
at
org.apache.cxf.jaxrs.interceptor.JAXRSInInterceptor.handleMessage(JAXRSInInterceptor.java:89)
~[cxf-rt-frontend-jaxrs-2.6.0.jar:2.6.0]
at
org.apache.cxf.phase.PhaseInterceptorChain.doIntercept(PhaseInterceptorChain.java:262)
[cxf-api-2.6.0.jar:2.6.0]
... 25 common frames omitted

I understand that CXF tries to decode my file, but I would like to
prevent this. The file should just be transmitted as a UTF-8 stream,
unchanged and not decoded. Can anyone help me?








--
Sergey Beryozkin

Talend Community Coders
http://coders.talend.com/

Blog: http://sberyozkin.blogspot.com

Reply via email to