24

我正在开发基于 Spring 3.2 的 RESTful 服务。我遇到了一个控制器处理混合多部分 HTTP 请求的问题,第二部分是 XML 或 JSON 格式的数据,第二部分是图像文件。

我正在使用@RequestPart 注释来接收请求

@RequestMapping(value = "/User/Image", method = RequestMethod.POST,  consumes = {"multipart/mixed"},produces="applcation/json")

public
ResponseEntity<List<Map<String, String>>> createUser(
        @RequestPart("file") MultipartFile file, @RequestPart(required=false) User user) {

    System.out.println("file" + file);

    System.out.println("user " + user);

    System.out.println("received file with original filename: "
            + file.getOriginalFilename());

    // List<MultipartFile> files = uploadForm.getFiles();
    List<Map<String, String>> response = new ArrayList<Map<String, String>>();
    Map<String, String> responseMap = new HashMap<String, String>();

    List<String> fileNames = new ArrayList<String>();

    if (null != file) {
        // for (MultipartFile multipartFile : files) {

        String fileName = file.getOriginalFilename();
        fileNames.add(fileName);

        try {
            file.transferTo(new File("C:/" + file.getOriginalFilename()));
        } catch (IllegalStateException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        } catch (IOException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
    }
    responseMap.put("displayText", file.getOriginalFilename());
    responseMap.put("fileSize", "" + file.getSize());
    response.add(responseMap);

    HttpHeaders httpHeaders = new HttpHeaders();
    httpHeaders.add("Accept", "application/json");
    return new ResponseEntity<List<Map<String, String>>>(response,
            httpHeaders, HttpStatus.OK);

}

User.java 将是这样的-

@XmlRootElement(name = "User")


public class User implements Serializable { 
    private static final long serialVersionUID = 1L;

    private int userId;
    private String name;
    private String email;

    private String company;
    private String gender;

    //getter setter of the data members
}

据我了解,使用 @RequestPart 注释我希望 XML 多部分部分根据其 Content-Type 进行评估,并最终取消编组到我的用户类中(我使用的是 Jaxb2,编组器/解组器在当我将 XML 数据作为正文传递并使用 @RequestBody 注释时,应用程序上下文和过程对于所有其他控制器方法都可以正常工作)。

但实际发生的是,虽然文件被正确找到并解析为 MultipartFile,但“用户”部分从未见过,请求总是失败,与控制器方法签名不匹配。

我用几种客户端类型重现了这个问题,我相信多部分请求的格式是可以的。

请帮我解决这个问题,也许会有一些解决方法来接收混合/多部分请求。

谢谢并恭祝安康,

拉格文德拉

4

5 回答 5

30

我已经设法解决了这个问题

端点示例:

@PostMapping("/")
public Document create(@RequestPart Document document,
                       @RequestPart(required = false) MultipartFile file) {
    log.debug("#create: document({}), file({})", delegation, file);
    //custom logic
    return document;
}

例外:

"error_message": "Content type 'application/octet-stream' not supported"

从下一个方法抛出异常:

org.springframework.web.servlet.mvc.method.annotation.AbstractMessageConverterMethodArgumentResolver.readWithMessageConverters(HttpInputMessage,MethodParameter,Type)

解决方案:

我们必须创建自定义转换器@Component,它实现HttpMessageConverterHttpMessageConverter并了解MediaType.APPLICATION_OCTET_STREAM对于简单的解决方法,扩展AbstractJackson2HttpMessageConverter就足够了

@Component
public class MultipartJackson2HttpMessageConverter extends AbstractJackson2HttpMessageConverter {

/**
 * Converter for support http request with header Content-Type: multipart/form-data
 */
public MultipartJackson2HttpMessageConverter(ObjectMapper objectMapper) {
    super(objectMapper, MediaType.APPLICATION_OCTET_STREAM);
}

@Override
public boolean canWrite(Class<?> clazz, MediaType mediaType) {
    return false;
}

@Override
public boolean canWrite(Type type, Class<?> clazz, MediaType mediaType) {
    return false;
}

@Override
protected boolean canWrite(MediaType mediaType) {
    return false;
}
}
于 2019-03-16T10:51:57.487 回答
9

不确定您是否解决了问题,但我也遇到了类似的问题,即在将 @RequestPart 和 MultipartFile 混合在一起时,我的控制器没有拾取我的 JSON 对象。

您的调用的方法签名看起来正确:

public ResponseEntity<List<Map<String, String>>> createUser(
        @RequestPart("file") MultipartFile file, @RequestPart(required=false) User user) {

// ... CODE ... 
}

但是请确保您的请求看起来像这样:

POST /createUser
Content-Type: multipart/mixed; boundary=B0EC8D07-EBF1-4EA7-966C-E492A9F2C36E

--B0EC8D07-EBF1-4EA7-966C-E492A9F2C36E
Content-Disposition: form-data; name="user";
Content-Type: application/xml; charset=UTF-8

<user><!-- your user xml --></user>
--B0EC8D07-EBF1-4EA7-966C-E492A9F2C36E
Content-Disposition: form-data; name="file"; filename="A551A700-46D4-470A-86E7-52AD2B445847.dat"
Content-Type: application/octet-stream

/// FILE DATA
--B0EC8D07-EBF1-4EA7-966C-E492A9F2C36E--
于 2013-08-29T05:47:35.577 回答
-1

我设法解决了问题:

    @SuppressWarnings("rawtypes")
@RequestMapping(value = "/DataTransfer", method = RequestMethod.POST, produces = {
        MediaType.APPLICATION_JSON_UTF8_VALUE }, consumes = {  MediaType.MULTIPART_FORM_DATA_VALUE, MediaType.APPLICATION_JSON_UTF8_VALUE} )
@ApiOperation(value = "Sbm Data Transfer Service", response = Iterable.class)
@ApiResponses(value = { @ApiResponse(code = 200, message = "Successfully find."),
        @ApiResponse(code = 400, message = "There has been an error."),
        @ApiResponse(code = 401, message = "You are not authorized to save the resource"),
        @ApiResponse(code = 403, message = "Accessing the resource you were trying to reach is forbidden"),
        @ApiResponse(code = 404, message = "The resource you were trying to reach is not found") })
ResponseEntity processDataTransfer(@RequestPart(name="file") MultipartFile  file, @RequestPart(name="param") DataTransferInputDto param);
于 2019-12-19T12:27:52.247 回答
-1

您可以使用来自 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。如果文件字段允许多个文件上传,如第二个多部分请求中所示,只需使用数组或部分集合或 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) {
    }

很高兴能帮助你...

于 2019-07-12T13:19:34.310 回答
-4

你有没有尝试过

ResponseEntity<List<Map<String, String>>> createUser(
        @RequestPart("file") MultipartFile file, @RequestBody(required=false) User user) {

或者

ResponseEntity<List<Map<String, String>>> createUser(
        @RequestPart("file") MultipartFile file, @RequestParam(required=false) User user) {

如果这不起作用,你能告诉我们吗mapping.xml

于 2013-04-26T06:55:18.190 回答