如何在Spring restful服务中处理由文件和JSON对象组成的多部分请求?

时间:2014-12-04 13:06:36

标签: java json spring rest multipartform-data

我有以下资源(使用Spring 4.05.RELEASE实现),它接受一个文件和一个JSON对象:

(P.S.ventationTemplate是一个可序列化的实体类)

...
@RequestMapping(value="/create", method=RequestMethod.POST)
public @ResponseBody ActivityTemplate createActivityTemplate(
        @RequestPart ActivityTemplate activityTemplate, @RequestPart MultipartFile jarFile)
{
   //process the file and JSON
}
...

这是我正在测试的表格:

<form method="POST" enctype="multipart/form-data"
    action="http://localhost:8080/activityTemplates/create">
    JSON: <input type="text" name="activityTemplate" value='/* the JSON object*/'><br />

    File to upload: <input type="file" name="file">
    <input type="submit" value="Upload">
</form>

这是我得到的错误:

 There was an unexpected error (type=Unsupported Media Type, status=415).
 Content type 'application/octet-stream' not supported

那么我应该如何让资源接受JSON对象作为multipart请求的一部分,或者我应该以不同的方式发送表单?

9 个答案:

答案 0 :(得分:25)

我花了两天时间为我工作!

客户(角度):

$scope.saveForm = function () {
      var formData = new FormData();
      var file = $scope.myFile;
      var json = $scope.myJson;
      formData.append("file", file);
      formData.append("ad",JSON.stringify(json));//important: convert to string JSON!
      var req = {
        url: '/upload',
        method: 'POST',
        headers: {'Content-Type': undefined},
        data: formData,
        transformRequest: function (data, headersGetterFunction) {
          return data;
        }
      };

Spring(引导):

@RequestMapping(value = "/upload", method = RequestMethod.POST)
    public @ResponseBody
    Advertisement storeAd(@RequestPart("ad") String adString, @RequestPart("file") MultipartFile file) throws IOException {

        Advertisement jsonAd = new ObjectMapper().readValue(adString, Advertisement.class);
//do whatever you want with your file and jsonAd

答案 1 :(得分:2)

希望这对你有所帮助。您需要在请求中设置边界以通知HTTP请求。  很简单;可以在以下链接中找到多部分格式的简要介绍

HTML 4.01 Specification for multipart

以下示例说明了“ multipart / form-data ”编码。 如果 Json对象是“ MyJsonObj ”,并且需要发送的文件是“ myfile.txt ”,则用户代理可能会发回以下数据:

Content-Type: multipart/form-data; boundary=MyBoundary

--MyBoundary
Content-Disposition: form-data; name="myJsonString"
Content-Type: application/json

MyJsonObj //Your Json Object goes here
--MyBoundary
Content-Disposition: form-data; name="files"; filename="myfile.txt"
Content-Type: text/plain

... contents of myfile.txt ...
--MyBoundary--

或者如果您的文件是图片类型为“ image.gif ”的图片,那么,

--MyBoundary
Content-Disposition: file; filename="image.gif"
Content-Type: image/gif
Content-Transfer-Encoding: binary

...contents of image.gif...
--MyBoundary--

您在 boundary 中指定 Content-Type header ,以便服务器知道如何拆分发送的数据。

所以,你基本上需要选择一个边界值:

  • 使用不会出现在发送到服务器的HTTP数据中的值,如'AaB03x'
  • 保持一致并在整个请求中使用相同的值。

答案 2 :(得分:1)

你还没有给你的 @RequestPart 提供参数名称?

public @ResponseBody ActivityTemplate createActivityTemplate(
    @RequestPart("activityTemplate") ActivityTemplate activityTemplate, @RequestPart("file") MultipartFile jarFile)
{
   //process the file and JSON
}

注意:不要忘记在类路径中包含jackson mapper .jar(将您的Json映射到ActivityTemplate )文件。

答案 3 :(得分:1)

你无法改变你的

@RequestMapping(value="/create", method=RequestMethod.POST)

@RequestMapping(value="/create",
                method=RequestMethod.POST, consumes ={"multipart/form-data"})

答案 4 :(得分:0)

默认内容类型是“application / octet-stream”。由于您要上传jar文件和JSON,因此应在@RequestMapping注释中设置内容类型,如下所示:

@RequestMapping(value="/create", method=RequestMethod.POST, headers="content-type=application/json,application/java-archive")

答案 5 :(得分:0)

错误消息表明没有为内容类型的多部分/ MIME部分注册HttpMessageConverter:application / octet-stream。尽管如此,您的jarFile参数很可能被正确识别为application / octet-stream,所以我假设参数映射不匹配。

因此,首先尝试为参数和表单的输入元素设置相同的名称。

另一个问题是JSON是作为表单中文本输入的(常规)值上传的,而不是多部分/ MIME中的单独部分。所以没有与之关联的内容类型标头,以发现Spring应该使用JSON反序列化器。 您可以使用@RequestParam代替注册特定转换器,如以下答案:JSON parameter in spring MVC controller

答案 6 :(得分:0)

这可能对您有所帮助,在接收MultipartFile时,您应该将请求标头内容类型设置为&#34; multipart / form-data&#34; ,然后在你的控制器中使用consumes =&#34; multipart / form-data&#34; ,consumemes也用于将我们的请求映射到我们在控制器中的方法。

如果你想接收JSON数据,最好以JSONString的形式发送请求,只需接收那个jsonstring,然后转换成json对象格式,然后使用该对象进行操作。

检查以下代码:

@RequestMapping(value="/savingImg", method=RequestMethod.POST, 
        consumes="multipart/form-data", produces="application/json")
public ResponseEntity<?> savingAppIMgDtlss(
        @RequestParam(value="f1", required = false) MultipartFile f1 , 
        @RequestParam(value="f2", required = false) MultipartFile f2 ,
        @RequestParam(value="f3", required = false) MultipartFile f3 ,
        @RequestParam(value="f4", required = false) MultipartFile f4 ,
        @RequestParam(value="f5", required = false) MultipartFile f5 ,
        @RequestParam(value="f6", required = false) MultipartFile f6 ,
        @RequestParam(value="f7", required = false) MultipartFile f7 ,
        @RequestParam(value="f8", required = false) MultipartFile f8 ,@RequestParam("data") String jsonString) 
                throws Exception , ParseException {
    try{
        JSONObject gstcTranObj = new JSONObject();
                //converting JSONString to JSON
        net.sf.json.JSONObject jsonDtls = net.sf.json.JSONObject.fromObject(jsonString);
        System.out.println("f1::"+f1.getOriginalFilename());
        System.out.println("f2::"+f2.getOriginalFilename());
        System.out.println("f3::"+f3.getOriginalFilename());
        System.out.println("f4::"+f4.getOriginalFilename());
        System.out.println("f5::"+f5.getOriginalFilename());
        System.out.println("f6::"+f6.getOriginalFilename());
        System.out.println("f7::"+f7.getOriginalFilename());
        System.out.println("f8::"+f8.getOriginalFilename());
} catch (Exception e) {
        e.printStackTrace();

        return new ResponseEntity<>("Failed",HttpStatus.NOT_FOUND);
    }finally{

    }
return new ResponseEntity<>("Success", HttpStatus.OK);

  }
}

答案 7 :(得分:0)

由于没有适当的HttpMessageConverter来处理multipart / form-data请求而引发异常。 demo

答案 8 :(得分:0)

您可以使用org.springframework.web.bind.annotation.RequestPart中的@RequestPart;用作结合@RequestBody和文件上传。

使用@RequestParam这样的@RequestParam(“ file”)MultipartFile文件,您可以仅上传文件和多个单个数据(键值),如

        @RequestMapping(value = "/uploadFile", method = RequestMethod.POST,  consumes = { MediaType.MULTIPART_FORM_DATA_VALUE }, produces = { MediaType.APPLICATION_JSON_VALUE })
    public void saveFile(
                         @RequestParam("userid") String userid,
                         @RequestParam("file") MultipartFile file) {

    }

您可以使用@RequestPart之类的方式发布JSON对象数据和和文件

    @RequestMapping(value = "/patientp", method = RequestMethod.POST,  consumes = { MediaType.MULTIPART_FORM_DATA_VALUE }, produces = { MediaType.APPLICATION_JSON_VALUE })
public ResponseEntity<?> insertPatientInfo(
                                            @RequestPart PatientInfoDTO patientInfoDTO,
                                            @RequestPart("file") MultipartFile file) {
}

您不仅限于直接将分段文件上传用作控制器方法参数。您的表单对象可以包含Part或MultipartFile字段,并且Spring自动知道它必须从文件部分获取值并适当地转换值。

上述方法可以响应先前演示的包含单个文件的多部分请求。之所以可行,是因为Spring具有一个内置的HTTP消息转换器,可以识别文件部分。除了javax.servlet.http.Part类型之外,您还可以将文件上传转换为org.springframework.web.multipart.MultipartFile。如第二个multipart请求所示,如果file字段允许上传多个文件,则只需使用数组或Parts集合或MultipartFiles。

        @RequestMapping(value = "/patientp", method = RequestMethod.POST,  consumes = { MediaType.MULTIPART_FORM_DATA_VALUE }, produces = { MediaType.APPLICATION_JSON_VALUE })
    public ResponseEntity<?> insertPatientInfo(
                                                @RequestPart PatientInfoDTO patientInfoDTO,
                                                @RequestPart("files") List<MultipartFile> files) {
    }

乐于助人...