如何以编程方式设置内容类型?获取 org.jvnet.mimepull.MIMEParsingException:缺少开始边界
Posted
技术标签:
【中文标题】如何以编程方式设置内容类型?获取 org.jvnet.mimepull.MIMEParsingException:缺少开始边界【英文标题】:How to set Content type progammatically? Getting org.jvnet.mimepull.MIMEParsingException: Missing start boundary 【发布时间】:2019-11-03 04:10:26 【问题描述】:我有一个命令行 java 实用程序,它需要对 jersey 休息服务进行休息调用。
@POST
@Path("/name")
@Produces(MediaType.APPLICATION_JSON)
@Consumes(MediaType.MULTIPART_FORM_DATA)
public Response insert(@PathParam("name") String nestName,
@QueryParam("id") String id,
@FormDataParam("file") InputStream fileStream,
@FormDataParam("file") FormDataContentDisposition fileDetail) throws IOException
并通过 java 命令行实用程序发出请求,如下所示:
Map<String,String> headers = new HashMap<String,String>();
headers.put(HttpHeaders.CONTENT_TYPE, MediaType.MULTIPART_FORM_DATA + "; boundary=--------------------------458269817398405552698839");
headers.put(HttpHeaders.ACCEPT, "*/*");
headers.put(HttpHeaders.CACHE_CONTROL, "no-cache");
headers.put(HttpHeaders.ACCEPT_ENCODING, "gzip, deflate");
URLConnection conn = url.openConnection();
if(headers!=null && !headers.isEmpty())
for(Map.Entry<String, String> entry: headers.entrySet())
conn.setRequestProperty(entry.getKey(),entry.getValue());
ServiceHttpResponse response = new ServiceHttpResponse(con.getResponseCode(), con.getResponseMessage());
我得到以下异常。有没有其他方法可以设置 mime 类型?
javax.ws.rs.BadRequestException: HTTP 400 错误请求
在 org.glassfish.jersey.media.multipart.internal.MultiPartReaderClientSide.readFrom(MultiPartReaderClientSide.java:188) 在 org.glassfish.jersey.media.multipart.internal.MultiPartReaderClientSide.readFrom(MultiPartReaderClientSide.java:93) 在 org.glassfish.jersey.message.internal.ReaderInterceptorExecutor$TerminalReaderInterceptor.invokeReadFrom(ReaderInterceptorExecutor.java:256) 在 org.glassfish.jersey.message.internal.ReaderInterceptorExecutor$TerminalReaderInterceptor.aroundReadFrom(ReaderInterceptorExecutor.java:235) 在 org.glassfish.jersey.message.internal.ReaderInterceptorExecutor.proceed(ReaderInterceptorExecutor.java:155) 在 org.glassfish.jersey.server.internal.MappableExceptionWrapperInterceptor.aroundReadFrom(MappableExceptionWrapperInterceptor.java:74) 在 org.glassfish.jersey.message.internal.ReaderInterceptorExecutor.proceed(ReaderInterceptorExecutor.java:155) 在 org.glassfish.jersey.message.internal.MessageBodyFactory.readFrom(MessageBodyFactory.java:1085) 在 org.glassfish.jersey.message.internal.InboundMessageContext.readEntity(InboundMessageContext.java:874) 在 org.glassfish.jersey.message.internal.InboundMessageContext.readEntity(InboundMessageContext.java:808) 在 org.glassfish.jersey.server.ContainerRequest.readEntity(ContainerRequest.java:234) 在 org.glassfish.jersey.media.multipart.internal.FormDataParamValueFactoryProvider$ValueFactory.getEntity(FormDataParamValueFactoryProvider.java:126) 在 org.glassfish.jersey.media.multipart.internal.FormDataParamValueFactoryProvider$FormDataParamValueFactory.provide(FormDataParamValueFactoryProvider.java:280) 在 org.glassfish.jersey.server.spi.internal.ParamValueFactoryWithSource.provide(ParamValueFactoryWithSource.java:71) 在 org.glassfish.jersey.server.spi.internal.ParameterValueHelper.getParameterValues(ParameterValueHelper.java:90)
在 org.glassfish.jersey.server.model.internal.JavaResourceMethodDispatcherProvider$AbstractMethodParamInvoker.getParamValues(JavaResourceMethodDispatcherProvider.java:127) 在 org.glassfish.jersey.server.model.internal.JavaResourceMethodDispatcherProvider$ResponseOutInvoker.doDispatch(JavaResourceMethodDispatcherProvider.java:160)
在 org.glassfish.jersey.server.model.internal.AbstractJavaResourceMethodDispatcher.dispatch(AbstractJavaResourceMethodDispatcher.java:99)
在 org.glassfish.jersey.server.model.ResourceMethodInvoker.invoke(ResourceMethodInvoker.java:389)
在 org.glassfish.jersey.server.model.ResourceMethodInvoker.apply(ResourceMethodInvoker.java:347)
在 org.glassfish.jersey.server.model.ResourceMethodInvoker.apply(ResourceMethodInvoker.java:102)
在 org.glassfish.jersey.server.ServerRuntime$2.run(ServerRuntime.java:326)
在 org.glassfish.jersey.internal.Errors$1.call(Errors.java:271)
在 org.glassfish.jersey.internal.Errors$1.call(Errors.java:267)
在 org.glassfish.jersey.internal.Errors.process(Errors.java:315)
在 org.glassfish.jersey.internal.Errors.process(Errors.java:297) 在 org.glassfish.jersey.internal.Errors.process(Errors.java:267)
在 org.glassfish.jersey.process.internal.RequestScope.runInScope(RequestScope.java:317) 在 org.glassfish.jersey.server.ServerRuntime.process(ServerRuntime.java:305)
在 org.glassfish.jersey.server.ApplicationHandler.handle(ApplicationHandler.java:1154)
在 org.glassfish.jersey.servlet.WebComponent.serviceImpl(WebComponent.java:473) 在 org.glassfish.jersey.servlet.WebComponent.service(WebComponent.java:427)
在 org.glassfish.jersey.servlet.ServletContainer.service(ServletContainer.java:388)
在 org.glassfish.jersey.servlet.ServletContainer.service(ServletContainer.java:341)
在 org.glassfish.jersey.servlet.ServletContainer.service(ServletContainer.java:228)
在 org.eclipse.jetty.servlet.ServletHolder.handle(ServletHolder.java:808)
在 org.eclipse.jetty.servlet.ServletHandler$CachedChain.doFilter(ServletHandler.java:1669)
在 com.cadence.adw.common.generic.xml.server.service.http.RequestInterceptor.doFilter(RequestInterceptor.java:95)
在 org.eclipse.jetty.servlet.ServletHandler$CachedChain.doFilter(ServletHandler.java:1652)
在 org.eclipse.jetty.servlets.UserAgentFilter.doFilter(UserAgentFilter.java:83) 在 org.eclipse.jetty.servlets.GzipFilter.doFilter(GzipFilter.java:300)
在 org.eclipse.jetty.servlet.ServletHandler$CachedChain.doFilter(ServletHandler.java:1652) 在 org.eclipse.jetty.servlet.ServletHandler.doHandle(ServletHandler.java:585)
在 org.eclipse.jetty.server.session.SessionHandler.doHandle(SessionHandler.java:221)
在 org.eclipse.jetty.server.handler.ContextHandler.doHandle(ContextHandler.java:1127)
在 org.eclipse.jetty.servlet.ServletHandler.doScope(ServletHandler.java:515)
在 org.eclipse.jetty.server.session.SessionHandler.doScope(SessionHandler.java:185) 在 org.eclipse.jetty.server.handler.ContextHandler.doScope(ContextHandler.java:1061)
在 org.eclipse.jetty.server.handler.ScopedHandler.handle(ScopedHandler.java:141)
在 org.eclipse.jetty.server.handler.ContextHandlerCollection.handle(ContextHandlerCollection.java:215)
在 org.eclipse.jetty.server.handler.HandlerCollection.handle(HandlerCollection.java:110)
在 org.eclipse.jetty.server.handler.HandlerWrapper.handle(HandlerWrapper.java:97)
在 org.eclipse.jetty.server.Server.handle(Server.java:497)
在 org.eclipse.jetty.server.HttpChannel.handle(HttpChannel.java:310) 在 org.eclipse.jetty.server.HttpConnection.onFillable(HttpConnection.java:257)
在 org.eclipse.jetty.io.AbstractConnection$2.run(AbstractConnection.java:540)
在 org.eclipse.jetty.util.thread.QueuedThreadPool.runJob(QueuedThreadPool.java:635)
在 org.eclipse.jetty.util.thread.QueuedThreadPool$3.run(QueuedThreadPool.java:555)
在 java.lang.Thread.run(Thread.java:748)
原因:org.jvnet.mimepull.MIMEParsingException: Missing start 边界
在 org.jvnet.mimepull.MIMEParser.skipPreamble(MIMEParser.java:318)
在 org.jvnet.mimepull.MIMEParser.access$300(MIMEParser.java:68)
在 org.jvnet.mimepull.MIMEParser$MIMEEventIterator.next(MIMEParser.java:154)
在 org.jvnet.mimepull.MIMEParser$MIMEEventIterator.next(MIMEParser.java:132)
在 org.jvnet.mimepull.MIMEMessage.makeProgress(MIMEMessage.java:228) 在 org.jvnet.mimepull.MIMEMessage.parseAll(MIMEMessage.java:189)
在 org.jvnet.mimepull.MIMEMessage.getAttachments(MIMEMessage.java:115)
在 org.glassfish.jersey.media.multipart.internal.MultiPartReaderClientSide.getMimeParts(MultiPartReaderClientSide.java:272)
在 org.glassfish.jersey.media.multipart.internal.MultiPartReaderClientSide.readMultiPart(MultiPartReaderClientSide.java:231)
在 org.glassfish.jersey.media.multipart.internal.MultiPartReaderServerSide.readMultiPart(MultiPartReaderServerSide.java:91)
在 org.glassfish.jersey.media.multipart.internal.MultiPartReaderClientSide.readFrom(MultiPartReaderClientSide.java:183)
...还有 58 个
【问题讨论】:
【参考方案1】:设置请求参数不是发送多部分请求的正确方法。仅设置带有边界的内容类型是不够的。该边界用于请求正文内部,以分隔请求的不同“部分”并结束正文。例如
Content-Type: multipart/form-data; boundary=AaB03x
--AaB03x
Content-Disposition: form-data; name="submit-name"
Larry
--AaB03x
Content-Disposition: form-data; name="files"; filename="file1.txt"
Content-Type: text/plain
... contents of file1.txt ...
--AaB03x--
UrlConnection
(或HttpUrlConnection
)没有任何 API 可以轻松发送 Multipart 请求。您需要手动创建此请求正文并通过连接的输出流发送出去。
multipart 的目的是发送多个参数,其中一些参数是二进制对象,如图像文件等。如果所有参数都是文本,可以使用application/x-www-form-urlencoded
content-type。这就是在发出 POST 请求时在 HttpUrlConnection
上设置请求参数所做的事情。然后在服务器上,您将使用@FormParam("param-name")
而不是@FormDataParam
。如果您正在发出 GET 请求,它将是 @QueryParam
。
如果您想要发出 Multipart 请求,您应该研究支持发送请求的客户端,而不必手动构建正确的正文格式。您可以查看Jersey Client 及其Multipart support。还有许多其他客户端库。做一些搜索。
还有
What is difference between @FormDataParam and @FormParam?【讨论】:
以上是关于如何以编程方式设置内容类型?获取 org.jvnet.mimepull.MIMEParsingException:缺少开始边界的主要内容,如果未能解决你的问题,请参考以下文章
如何在同一网络上以编程方式获取其他支持Wifi的设备的IP地址?
获取 json 数据后以编程方式设置 UITableViewCell 高度
使用 NSFetchedResultsController 如何以编程方式设置 5 个部分以显示以及过滤它们以获取适当的行