我喜欢 JAX-RS(特别是 Jersey)的自动 HTTP 内容协商,即它能够通过“Accept”和/或“Content-Type”标头路由我的资源。但是我发现有时在发生冲突时它并没有给我足够的控制权。
例如,考虑以下端点:
@Path("/order")
public class OrderController {
@GET
@Path("{orderID: \\d+}")
@Produces("text/html")
public View getOrderView(@PathParam("orderID") long id) {
Order order = this.getOrderData(id);
return new OrderView(order);
}
@GET
@Path("{orderID: \\d+}")
@Produces({"application/json", "application/xml"})
public Order getOrderData(@PathParam("orderID") long id) {
return new OrderService.findOrder(id);
}
}
我会在 Firefox 和 Chrome 之间得到不同的结果。Firefox 将映射到 HTML 端点,而 Chrome 将在我导航到每个端点 URL 时触发 XML 端点。它们之间的区别在于其 Accept 标头中列出的 MIME 类型的顺序。Chrome 发送以下内容:
User-Agent: Mozilla/5.0 (Macintosh; U; Intel Mac OS X 10_6_6; en-US) AppleWebKit/534.13 (KHTML, like Gecko) Chrome/9.0.597.107 Safari/534.13
Accept: application/xml,application/xhtml+xml,text/html;q=0.9,text/plain;q=0.8,image/png,*/*;q=0.5
与 Firefox 相比,它首先列出 HTML:
User-Agent: Mozilla/5.0 (Macintosh; U; Intel Mac OS X 10.6; en-US; rv:1.9.2.13) Gecko/20101203 Firefox/3.6.13
Accept: text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8
当所有的权重相同时,它会匹配第一个条目,这似乎是合乎逻辑的。但就我而言,我得到的结果与我想要的不同,因此最好确定一种更好的打破平局的方法。
我的问题:没有将标头信息注入这些方法并自己执行媒体类型处理,有没有办法在出现平局时“调整权重”?例如,我可以告诉它总是用 HTML 胜过 XML 吗?我的 RESTful 客户端对他们想要返回的类型非常明确,但浏览器对 Accept 标头的草率是出了名的。(我个人认为他们应该将 HTML 的权重略高于 XML,因为这是用户所期望的,但为时已晚。)
或者,我可以在某个集中位置只执行一次我自己的自定义内容协商吗?我不反对手动写出这个逻辑,但如果这意味着将它应用到我的资源的每一个实例上,我就不反对了。JAX-RS 是否有一些在管道中添加过滤器以在路由请求之前调整请求的概念?