Spring Data Rest / Spring Hateoas自定义控制器 - PersistentEntityResourceAssembler

时间:2015-11-06 17:03:35

标签: java spring spring-mvc spring-data-rest spring-hateoas

我正在尝试从RepositoryRestResource向自动生成的端点添加一些额外的业务逻辑。请参阅以下代码:

资源:

@RepositoryRestResource(collectionResourceRel="event", path="event")
public interface EventRepository extends PagingAndSortingRepository<Event, Long> {

}

控制器:

@RepositoryRestController
@RequestMapping(value = "/event")
public class EventController {

  @Autowired
  private EventRepository eventRepository;

  @Autowired
  private PagedResourcesAssembler<Event> pagedResourcesAssembler;

  @RequestMapping(method = RequestMethod.GET, value = "")
  @ResponseBody
  public PagedResources<PersistentEntityResource> getEvents(Pageable pageable,
      PersistentEntityResourceAssembler persistentEntityResourceAssembler) {

    Page<Event> events = eventRepository.findAll(pageable);

    return pagedResourcesAssembler.toResource(events, persistentEntityResourceAssembler);
  }
}

我查看了以下两篇stackoverflow文章:

我觉得我很亲密,但我面临的问题是:

return pagedResourcesAssembler.toResource(events, persistentEntityResourceAssembler);

返回错误说:

"The method toResource(Page<Event>, Link) in the type PagedResourcesAssembler<Event> is not applicable 
 for the arguments (Page<Event>, PersistentEntityResourceAssembler)".

toResource方法有一个接受ResourceAssembler的方法签名,但我不确定如何正确实现它,我找不到任何关于此事的文档。

提前致谢, - 布赖恩

修改

我的问题是我认为我可以覆盖从@RepositoryRestResource注释自动创建的控制器方法,而无需创建自己的资源和资源汇编程序。创建资源和资源汇编程序后,我能够将业务逻辑添加到端点。

资源:

public class EventResource extends ResourceSupport {
  private String name;

  public String getName() {
    return name;
  }

  public void setName(String name) {
    this.name = name;
  }
}

资源汇编程序:

@Component
public class EventResourceAssembler extends ResourceAssemblerSupport<Event, EventResource> {

  public EventResourceAssembler() {
    super(EventController.class, EventResource.class);
  }

  @Override
  public EventResource toResource(Event entity) {
    EventResource eventResource = createResourceWithId(entity.getId(), entity);
    eventResource.setName(entity.getName());
    return eventResource;
  }
}

更新了控制器:

@RepositoryRestController
@RequestMapping(value = "/event")
public class EventController {

  @Autowired
  private EventRepository eventRepository;

  @Autowired
  private EventResourceAssembler eventResourceAssembler;

  @Autowired
  private PagedResourcesAssembler<Event> pageAssembler;

  @RequestMapping(method = RequestMethod.GET, value = "")
  @ResponseBody
  public PagedResources<EventResource> getEvents(Pageable pageable) {
    Page<Event> events = eventRepository.findAll(pageable);

    // business logic

    return pageAssembler.toResource(events, eventResourceAssembler);
  }
}

我不喜欢这件事,它似乎打败了拥有RepositoryRestResource的目的。另一种方法是使用在创建,保存,删除操作之前和/或之后调用的事件处理程序。

@RepositoryEventHandler(Event.class)
public class EventRepositoryEventHandler {

  @HandleBeforeCreate
  private void handleEventCreate(Event event) {
    System.out.println("1");
  }
}

findAll或findOne操作似乎没有任何事件。无论如何,这两种方法似乎都解决了我从RepositoryRestResource扩展自动生成的控制器方法的问题。

2 个答案:

答案 0 :(得分:1)

它需要一个PagedResourcesAssembler,如果你问,Spring会为你注入一个。

public PagedResources<Foo> get(Pageable page, PagedResourcesAssembler<Foo> assembler) {
    // ...
}

在这种情况下,资源为Foo。在您的情况下,您尝试返回的资源似乎是Event。如果是这样,我希望您的代码看起来像:

private ResourceAssembler<Event> eventAssembler = ...;
public PagedResources<Event> get(Pageable page, PagedResourcesAssembler<Event> pageAssembler) {
    Event event = ...;
    return eventAssembler.toResource(event, pageAssembler);
}

您提供的ResourceAssembler<Event>告诉Spring如何将Event变为Resource。 Spring将PagedResourcesAssembler<Event>注入到控制器方法中以处理分页链接。通过调用toResource并传入注入的pageAssembler来结合它们。

最终结果可以像上面一样简单地返回。您还可以使用HttpEntity之类的内容来更好地控制状态代码和标题。

注意:您提供的ResourceAssembler实际上可以像使用Event对象包装资源(例如Resource)一样简单。通常,您会想要添加任何相关链接。

答案 1 :(得分:0)

要破解它,您只能使用PagedResourcesAssembler<Object>,例如:

@RequestMapping(method = RequestMethod.GET, value = "")
@ResponseBody
public PagedModel<PersistentEntityResource> getEvents(
    Pageable pageable,
    PersistentEntityResourceAssembler persistentAssembler,
    PagedResourcesAssembler<Object> pageableAssembler
) {
    return pageableAssembler.toModel(
        (Page<Object>) repository.findAll(pageable),
        persistentAssembler
    );
  }