我在Spring Data Rest中的一个实体有一个属性值,应该添加到响应的标题中。怎么做?
我已经找到了使用处理程序拦截器更改默认标头(适用于所有响应)的方法。我还发现了有关版本,etag和修改过的内容。 这些不是我想要的东西。
必须在1个REST资源上设置标头,并且该值取决于实例/记录。
举例说明:Person
的属性为age
。对Person REST资源的每个请求(例如GET /person/{id}
)都必须将age
的值作为http标头返回(例如:age:32
)。
提前致谢。
答案 0 :(得分:2)
根据我之前的回答,我可以看到最新版本的SDR API有一个方法RepositoryRestMvcConfiguration#httpHeadersPreparer()
。
我可以看到这个方法在2.6.8中可用,但不在我目前使用的2.5.10版本中。
所以看起来你可以这样做:
@Configuration
public class MyConfiguration extends RepositoryRestMvcConfiguration
{
@Override
public HttpHeadersPreparer httpHeadersPreparer()
{
HttpHeadersPreparer preparer = new HttpHeadersPreparer(){
@Override
public HttpHeaders prepareHeaders(PersistentEntityResource resource, Object value)
{
org.springframework.http.HttpHeaders headers = super.prepareHeaders(resource);
if(value instanceof Person){
headers.add("age", ((Person)value).getAge());
}
}
};
return preparer;
}
}
答案 1 :(得分:0)
首先,您可以将拦截器绑定到特定路径,这样就解决了问题的一半。
第二个问题是获取对已请求的Person资源的引用。您可以在此处查看使用ThreadLocal:
public class PersonContext{
private static ThreadLocal<Person> context = new ThreadLocal<Person>();
public static Person getPerson(Person person){
return context.get();
}
public static void setPerson(Person person){
context.set(person);
}
public static void cleanUp(){
context.remove();
}
}
标准的JPA Entity Listener可用于绑定对加载的Person的引用:
public class PersonListener{
@PostLoad
public void setContext(Person person){
PersonContext.set(person);
}
}
可以在实体上注册实体侦听器:
@Entity
@EntityListeners(PersonListener.class)
public class Person(){
}
在拦截器中,您可以获取对当前线程中上下文绑定的Person的引用,并相应地设置标头:
public class PersonInterceptor extends HandlerInterceptorAdapter {
@Override
public void postHandle(HttpServletRequest request, HttpServletResponse response,
Object handler, ModelAndView modelAndView) throws Exception {
if(request.getMethod().equals("GET")){
Person person = PersonContext.getPerson();
response.addHeader("age", person.getAge());
}
PersonContext.cleanUp();
}
}
最后使用Spring Data Rest注册你的拦截器:
@Bean
public MappedInterceptor myMappedInterceptor() {
return new MappedInterceptor(new String[]{"person/**"}, new PersonInterceptor());
}
的问题:
另一种方法可能是在拦截器中重新查询数据库。所以上面的内容可以用拦截器代替:
public class PersonInterceptor extends HandlerInterceptorAdapter {
@Autowired
private PersonRepository repository;
@Override
public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler,
ModelAndView modelAndView) throws Exception {
super.postHandle(request, response, handler, modelAndView);
if(request.getMethod().equals("GET")){
Long personID = // Extract id by examining request.getServletPath();
Person person = repository.findOne(personID);
response.addHeader("age", person.getAge());
}
}
}
这会产生额外查询的开销,但可以通过配置OpenEntityManagerInViewFilter来避免这种情况,这应该意味着它是从第一级缓存中获取的。