DEV Community

Cover image for Sending both File and JSON in One Request to Spring Boot
Sean Evans
Sean Evans

Posted on • Edited on

Sending both File and JSON in One Request to Spring Boot

There are many people who use Spring Boot's @RequestPart to define an endpoint that receives both files and JSON data.

However, some of them encounter difficulties with how to test such endpoints, which is why I am writing this article.

⌨️ Parameters with @RequestPart

Typically, this type of endpoint is shown in the following code:

@RestController
@RequestMapping("/form-data")
public class FormDataController {
    @RequestMapping(value = "/json-and-file", method = POST)
    String jsonAndFile(@RequestPart User user,
                       @RequestPart MultipartFile file) {
       return "ok";
    }
}
Enter fullscreen mode Exit fullscreen mode

I like to use modern Java, so I use records instead of Lombok or manually writing getters and setters. How about you?

public record User(String id, String name) {}
Enter fullscreen mode Exit fullscreen mode

🤔 HTTP status code 415

When using an HTTP Client such as Yaak, after submitting parameters via Multi-Part, I received a 415 status code. How frustrating.

Received HTTP status code 415 with Yaak

At this point, open Spring Boot's console and you will see a warning that the content type "application/octet-stream" is not supported. The 415 status code means that the server does not support the media type of the current parameter.

Obviously, the user field is a JSON string, not a binary. Without setting the content type manually, the default value is wrong. Therefore, we just need to set the correct content type for the user field.

Don't worry, your backend code is correct.

😀 Solution: Set the Content-Type of the text field to application/json

Yaak is a very clean tool and I love it!

But it CAN'T help when faced with this complex scenario because it DOESN'T support set the content-type for text field of a multipart request. I tried Paw, Bruno and they DIDN'T work either.

Therefore, I have to use a professional tool that supports setting the content-type for each part, such as Apidog or Postman. BTW, I only found these two GUI tools can do.

[Updated on Dec 26, 2024] A week after I posted this article, Bruno released v1.37.0 to support this feature. Did they read this post?

After setting the user field's content-type to application/json, I successfully received the HTTP status code 200 responded by Spring Boot.

Received HTTP status code 200 with Apidog

The problem is solved. 🎉

If you need a modern and professional API tool, try Apidog, which is a new versatile tool. Of course, Postman is also a good classic choice.

Would anyone use curl in this situation?

Top comments (1)

Collapse
 
seanevans profile image
Sean Evans • Edited

It suddenly struck me that this endpoint seemed to be an anti-pattern?