如何在Spring Data Rest

时间:2017-10-30 14:51:15

标签: spring-mvc header spring-data-rest

我在Spring Data Rest中的一个实体有一个属性值,应该添加到响应的标题中。怎么做?

我已经找到了使用处理程序拦截器更改默认标头(适用于所有响应)的方法。我还发现了有关版本,etag和修改过的内容。 这些不是我想要的东西。

必须在1个REST资源上设置标头,并且该值取决于实例/记录。

举例说明:Person的属性为age。对Person REST资源的每个请求(例如GET /person/{id})都必须将age的值作为http标头返回(例如:age:32)。

提前致谢。

2 个答案:

答案 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());
}

的问题:

  • post load listener将在Person的每个负载上执行,即使在其他路径上也是如此,因此清理需要更好。在所有路径上注册第二个拦截器以处理清理工作?

另一种方法可能是在拦截器中重新查询数据库。所以上面的内容可以用拦截器代替:

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来避免这种情况,这应该意味着它是从第一级缓存中获取的。