Spring boot multipart / related mime type support

时间:2016-08-08 21:38:37

标签: spring-boot jax-rs multipartform-data multipart

我需要使用内容类型为multipart / related的请求。请求将包括图像,json有效负载和二进制内容。我试图找到一些关于如何在Spring启动应用程序中处理此类请求的示例,我找到了一些关于如何处理multipart / form-data请求的引用,但没有涉及multipart / related mime类型。

请求是这样的:

POST /upload
Content-Type: multipart/related; boundary="123asdf234"
--123asdf234
Content-Type: application/json; charset=UTF-8
Content-Disposition: form-data
{
    "json": "payload"

}
—-123asdf234
Content-Type: application/zip
Content-Disposition: file-data; filename="some.zip"; size=123456;
<binary-attachment-content>
—-123asdf234
Content-Type: image/png
Content-Disposition: file-data; filename="image1.png"; size=123456;
<binary-attachment-content>
—-123asdf234-—

有人能说出如何在Spring启动应用程序中处理此请求。我正在使用JaxRS。

1 个答案:

答案 0 :(得分:2)

为了解决这个问题,我首先提到http://cxf.apache.org/docs/jax-rs-multiparts.html来正确理解与JAX-RS相关的多部分/相关。接下来,我参考了JAX-RS上的Spring文档,并选择使用Jersey依赖项来解决它。然后参考Jersey documentation我构建以下测试项目:https://github.com/ShawnTuatara/stackoverflow-38838926。主要的例子是:

package ca.tuatara.stackoverflow;

import javax.ws.rs.Consumes;
import javax.ws.rs.POST;
import javax.ws.rs.Path;
import javax.ws.rs.Produces;
import javax.ws.rs.core.MediaType;

import org.glassfish.jersey.media.multipart.BodyPart;
import org.glassfish.jersey.media.multipart.FormDataParam;
import org.glassfish.jersey.media.multipart.MultiPart;
import org.glassfish.jersey.media.multipart.MultiPartFeature;
import org.glassfish.jersey.server.ResourceConfig;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.stereotype.Component;

@SpringBootApplication
public class Stackoverflow38838926Application {
    public static void main(String[] args) {
        SpringApplication.run(Stackoverflow38838926Application.class, args);
    }

    @Component
    public class JerseyConfig extends ResourceConfig {
        public JerseyConfig() {
            register(MultiPartFeature.class);
            register(MultipartHandler.class);
            register(MultipartPartsHandler.class);
        }
    }

    @Component
    @Path("/upload")
    @Consumes("multipart/*")
    @Produces("text/text")
    public class MultipartHandler {
        @POST
        public String upload(MultiPart request) {
            StringBuffer response = new StringBuffer();
            for (BodyPart part : request.getBodyParts()) {
                if (MediaType.APPLICATION_JSON_TYPE.isCompatible(part.getMediaType())) {
                    response.append(part.getEntityAs(JsonModel.class));
                } else if (MediaType.APPLICATION_XML_TYPE.isCompatible(part.getMediaType())) {
                    response.append(part.getEntityAs(XmlModel.class));
                }
                response.append(System.lineSeparator());
            }
            return response.toString();
        }
    }

    @Component
    @Path("/uploadParts")
    @Consumes("multipart/*")
    @Produces("text/text")
    public class MultipartPartsHandler {
        @POST
        public String upload(@FormDataParam("json") JsonModel json, @FormDataParam("xml") XmlModel xml) {
            return json + System.lineSeparator() + xml;
        }
    }
}

测试显示了如何发送多部分请求。我保留了一些DEBUG日志记录,因此您可以在测试运行时准确查看电线上的内容。

原始POST有效负载存在一些问题,无法正确解析。内容必须在标题和内容之间有换行符。如果您没有为Content-Disposition提供“name”属性,那么您只能使用第一个示例(“/ upload”)。如果您确定了表单数据的名称,那么您可以使用第二个示例(“/ uploadParts”)。我没有使用图像或文件上传进行示例,但是如果您阅读Jersey多部分页面,您可以看到在请求方法上添加该参数输入是直截了当的。