Spring 莫名其妙地抛出 HttpMediaTypeNotAcceptableException
Spring throws HttpMediaTypeNotAcceptableException inexplicably
我的 Spring 应用配置为使用 GET 参数进行内容协商。代码是 Kotlin,但在 Java.
中同样有效
配置:
override fun configureContentNegotiation(configurer: ContentNegotiationConfigurer) {
configurer.favorParameter(true)
.parameterName("format")
.ignoreAcceptHeader(false)
.defaultContentType(MediaType.APPLICATION_JSON)
.mediaType("text/plain", MediaType.TEXT_PLAIN)
.mediaType("application/json", MediaType.APPLICATION_JSON)
.mediaType("application/rdf+xml", MediaType("application", "rdf+xml"))
}
以及以下控制器方法:
@GetMapping("/test", produces=["text/plain"])
fun testText() : String {
return "Hello"
}
@GetMapping("/test", produces=["application/json"])
fun testJson() : Map<String, String> {
return mapOf("hello" to "world")
}
@GetMapping("/test", produces=["application/rdf+xml"])
fun testRdf(response: HttpServletResponse) {
// dummy response, to demonstrate using output stream.
response.setContentType("application/rdf+xml")
response.outputStream.write("dummy data".toByteArray())
response.outputStream.close()
}
testRdf
returns void
并使用输出流将正文数据发回。
以下工作正常:
http://localhost:8080/test?format=text/plain
给我纯文本
http://localhost:8080/test?format=application/json
给我 JSON
但是 http://localhost:8080/test?format=application/rdf+xml
给了我一个 HTTP 406 并且日志说
org.apache.tomcat.util.http.Parameters : Start processing with input [format=application/rdf+xml]
o.s.web.servlet.DispatcherServlet : GET "/test?format=application/rdf+xml", parameters={masked}
.w.s.m.s.DefaultHandlerExceptionResolver : Resolved [org.springframework.web.HttpMediaTypeNotAcceptableException: Could not find acceptable representation]
o.s.web.servlet.DispatcherServlet : Completed 406 NOT_ACCEPTABLE
调试器显示它甚至没有调用我的函数。
(为了证明 testRdf
处理程序执行预期的操作,我使路径唯一并删除了 produces
注释 - 它在内容协商和 returns 正文之外工作正常预期。)
据我所知,我已表明我的方法是该内容类型的正确处理程序,并且我已正确注册该内容类型。
为什么Spring不认为我的处理程序满足内容协商请求?
我找到了答案。需要 URL 编码的参数字符,因此可以正常工作:
http://localhost:8080/test?format=application%2Frdf%2Bxml
我的 Spring 应用配置为使用 GET 参数进行内容协商。代码是 Kotlin,但在 Java.
中同样有效配置:
override fun configureContentNegotiation(configurer: ContentNegotiationConfigurer) {
configurer.favorParameter(true)
.parameterName("format")
.ignoreAcceptHeader(false)
.defaultContentType(MediaType.APPLICATION_JSON)
.mediaType("text/plain", MediaType.TEXT_PLAIN)
.mediaType("application/json", MediaType.APPLICATION_JSON)
.mediaType("application/rdf+xml", MediaType("application", "rdf+xml"))
}
以及以下控制器方法:
@GetMapping("/test", produces=["text/plain"])
fun testText() : String {
return "Hello"
}
@GetMapping("/test", produces=["application/json"])
fun testJson() : Map<String, String> {
return mapOf("hello" to "world")
}
@GetMapping("/test", produces=["application/rdf+xml"])
fun testRdf(response: HttpServletResponse) {
// dummy response, to demonstrate using output stream.
response.setContentType("application/rdf+xml")
response.outputStream.write("dummy data".toByteArray())
response.outputStream.close()
}
testRdf
returns void
并使用输出流将正文数据发回。
以下工作正常:
http://localhost:8080/test?format=text/plain
给我纯文本http://localhost:8080/test?format=application/json
给我 JSON
但是 http://localhost:8080/test?format=application/rdf+xml
给了我一个 HTTP 406 并且日志说
org.apache.tomcat.util.http.Parameters : Start processing with input [format=application/rdf+xml]
o.s.web.servlet.DispatcherServlet : GET "/test?format=application/rdf+xml", parameters={masked}
.w.s.m.s.DefaultHandlerExceptionResolver : Resolved [org.springframework.web.HttpMediaTypeNotAcceptableException: Could not find acceptable representation]
o.s.web.servlet.DispatcherServlet : Completed 406 NOT_ACCEPTABLE
调试器显示它甚至没有调用我的函数。
(为了证明 testRdf
处理程序执行预期的操作,我使路径唯一并删除了 produces
注释 - 它在内容协商和 returns 正文之外工作正常预期。)
据我所知,我已表明我的方法是该内容类型的正确处理程序,并且我已正确注册该内容类型。
为什么Spring不认为我的处理程序满足内容协商请求?
我找到了答案。需要 URL 编码的参数字符,因此可以正常工作:
http://localhost:8080/test?format=application%2Frdf%2Bxml