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.
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.
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.
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.
I followed the recommendation in the comments and changed @RequestParam
to @RequestBody
and it just worked (my parameter is a String).
I agree this is a bug in Spring because the exact same code that fails in my production environment (when using @RequestParam
) works fine in localhost.