Can Spring MVC have request parameters for an HTTP PUT method, or must I use post? Which should I use to be RESTful?

20,920

Solution 1

Since Spring 3.1, HttpPutFormContentFilter can be used to handle application/x-www-form-urlencoded data:

Filter that makes form encoded data available through the ServletRequest.getParameter*() family of methods during HTTP PUT requests.

The Servlet spec requires form data to be available for HTTP POST but not for HTTP PUT requests. This filter intercepts HTTP PUT requests where content type is 'application/x-www-form-urlencoded', reads form encoded content from the body of the request, and wraps the ServletRequest in order to make the form data available as request parameters just like it is for HTTP POST requests.

For other incoming data, such as JSON, you'll need @RequestBody as explained in JQuery, Spring MVC @RequestBody and JSON - making it work together, to not run into a 415 Unsupported Media Type.

Solution 2

Spring controllers support GET/HEAD/POST/PUT/DELETE/OPTIONS/TRACE, but since your browser may not be able to send these request methods, it wont work for you.

The workaround is to use the "org.springframework.web.filter.HiddenHttpMethodFilter" provided by Spring. It requires you to pass a hidden parameter for the request method. The default parameter supported by this filter is "_method".

Check the javadoc of the filter for more info.

Solution 3

This, as suggest above, seems to be a bug in spring/servlet API. In reality PUT requests are supposed to work on Request Body (or payload) and not on Request Parameters. In that sense, servlet API & spring's handling is correct.

Having said that, a better and much easier workaround is to pass no data element from your javascript/jQuery call and pass your parameters as part of the url itself. meaning, set parameters in the url field the way you would do in a GET call.

$.ajax({
            url: "/help/" + helpDocumentId + "/vote.json" + "?param1=param2Val&..",
            type: "PUT",
            data: "",
            success: function(response) {
                // ....
            }
     });

now this works for simple parameters, i guess, will not work for complex JSON types. Hope this helps.

Share:
20,920
egervari
Author by

egervari

Updated on January 27, 2020

Comments

  • egervari
    egervari over 4 years

    I have a controller action I think should be an HTTP PUT, but Spring is complaining when I try and use @RequestParam in the controller action. Is request parameters not allowed for HTTP PUT methods, and is that why Spring is rejecting it?

    @RequestMapping(value = "/{helpDocumentId}/vote", method = RequestMethod.PUT)
    public void voteHelpfulness(@PathVariable long helpDocumentId, @RequestParam boolean isHelpful) {
        helpManager.voteOnHelpDocument(helpDocumentId, isHelpful);
    }
    

    When executed, it throws this error:

    org.springframework.web.bind.MissingServletRequestParameterException: Required boolean parameter 'isHelpful' is not present
    

    Of course, the isHelpful parameter IS present. I can make the above code work perfectly for HTTP POST, so I know this isn't the problem.

         $.ajax({
                url: "/help/" + helpDocumentId + "/vote.json",
                type: "PUT",
                data: {
                    isHelpful: isHelpful
                },
                success: function(response) {
                    // ....
                }
         });
    

    Is PUT the correct http method? This action modifies the helpDocument, but it doesn't create one.