Интеграция пользовательских запросов Spring Data REST

Я хочу создать ссылку REST для объекта Employee, который в основном будет запросом findByAllFields. Конечно, это должно сочетаться с Page и Sort. Для этого я реализовал следующий код:

@Entity
public class Employee extends Persistable<Long> {

    @Column
    private String firstName;

    @Column
    private String lastName;

    @Column
    private String age;

    @Column
    @Temporal(TemporalType.TIMESTAMP)
    private Date hiringDate;
}

Итак, я хотел бы, скажем, запрос, где я могу сделать:

http://localhost:8080/myApp/employees/search/all?firstName=me&lastName=self&ageFrom=20&ageTo=30&hiringDateFrom=12234433235

Итак, у меня есть следующее Repository

 @RepositoryRestResource(collectionResourceRel="employees", path="employees")
 public interface EmployeeRepository extends PagingAndSortingRepository<Employee, Long>, 
                                                         JpaSpecificationExecutor<Employee> {

 }

Итак, теперь мне нужен RestController

@RepositoryRestController
public class EmployeeSearchController {

    @Autowired
    private EmployeeRepository employeRepository;

    @RequestMapping(value = "/employees/search/all/search/all", method = RequestMethod.GET)
    public Page<Employee> getEmployees(EmployeeCriteria filterCriteria, Pageable pageable) {

        //EmployeeSpecification uses CriteriaAPI to form dynamic query with the fields from filterCriteria
        Specification<Employee> specification = new EmployeeSpecification(filterCriteria);

        return employeeRepository.findAll(specification, pageable);
}

Хорошо, очевидно, что это работает, но он не интегрирован с HATEOAS. Я попытался собрать ресурс, изменив контроллер на это:

public PagedResources<Resource<Employee>> getEmployees(
                PagedResourcesAssembler<Employee> assembler,
                EmployeeCriteria filterCriteria, Pageable pageable) {

        //EmployeeSpecification uses CriteriaAPI to form dynamic query with the fields from filterCriteria
        Specification<Employee> specification = new EmployeeSpecification(filterCriteria);

        Page<Employee> employees = employeeRepository.findAll(specification, pageable);
        return assembler.toResource(employees);
}

Очевидно, мне не хватает чего-то из вышеперечисленного, так как это не работает, и я получаю следующее исключение:

Could not instantiate bean class [org.springframework.data.web.PagedResourcesAssembler]: No default constructor found;

Итак, чтобы прояснить вопрос, я пытаюсь интегрировать вышеупомянутый ресурс в остальную часть архитектуры HATEOAS. Я не совсем уверен, что это правильный подход, поэтому любые другие предложения приветствуются.

РЕДАКТИРОВАТЬ: здесь вы можете увидеть аналогичную реализацию. Пожалуйста, взгляните на конфигурацию, вы увидите, что все контроллеры «Person», кроме одного, работают. https://github.com/cgeo7/spring-rest-example


person ChrisGeo    schedule 22.09.2014    source источник
comment
если вы можете создать простой проект github для своей проблемы, будет легче найти вашу проблему... потому что недостаточно информации - ваши конфигурации, зависимости и т. д.   -  person Õzbek    schedule 25.09.2014
comment
Прошу прощения за задержку. Я создал небольшой аналогичный проект, который отражает все трудности, которые у меня есть github.com/cgeo7/spring- пример отдыха   -  person ChrisGeo    schedule 25.09.2014
comment
@ChrisGeo и @Stackee007, вы не возражаете, если я спрошу вас, как вы определяете свои классы EmployeeCriteria и EmployeeSpecification?   -  person BalRog    schedule 29.10.2014


Ответы (3)


Попробуйте автозапись PagedResourcesAssembler в качестве члена класса и измените сигнатуру метода, как показано ниже.

@RepositoryRestController
public class EmployeeSearchController {

    @Autowired
    private EmployeeRepository employeRepository;

    @Autowired
    private PagedResourcesAssembler<Employee> pagedAssembler;

    @RequestMapping(value = "/employees/search/all/search/all", method = RequestMethod.GET)
    public ResponseEntity<Resources<Resource<Employee>>> getEmployees(EmployeeCriteria filterCriteria, Pageable pageable) {

        //EmployeeSpecification uses CriteriaAPI to form dynamic query with the fields from filterCriteria
        Specification<Employee> specification = new EmployeeSpecification(filterCriteria);

        Page<Employee> employees = employeeRepository.findAll(specification, pageable);
        return assembler.toResource(employees);
    }
}

Это прекрасно работает с Spring Data Rest 2.1.4.RELEASE

person Stackee007    schedule 29.09.2014
comment
См. также ​​stackoverflow.com/questions/31758862/ - person ; 11.10.2015
comment
ассемблерная переменная не определена!! Почему? - person Ivan Smorodin; 11.11.2015
comment
@IvanSmorodin это должно быть pagedAssembler. - person mustofa.id; 10.05.2020

Код @Stackee007 работает, но ресурс не будет содержать self ссылок. Для этого нужно еще немного.

@Autowired
PagedResourcesAssembler<Appointment> pagedResourcesAssembler;

@RequestMapping(value = "/findTodaysSchedule")
public HttpEntity<PagedResources<Resource<Appointment>>> getTodaysSchedule(
        PersistentEntityResourceAssembler entityAssembler, Pageable pageable) {
    Page<Appointment> todaysSchedule = apptRepo.findByStartTimeBetween(beginningOfDay, endOfDay, pageable);

    @SuppressWarnings({ "unchecked", "rawtypes" })
    PagedResources<Resource<Appointment>> resource = pagedResourcesAssembler.toResource(todaysSchedule,
                (ResourceAssembler) entityAssembler);

    return new ResponseEntity<>(resource, HttpStatus.OK);
}
person Abhijit Sarkar    schedule 20.10.2015
comment
Спасибо за добавленное разъяснение. Это прекрасно работает! - person Jon; 25.03.2016

Spring HATEOAS изменил название Resource, PagedResources и некоторых других классов. См. здесь. Ниже представлена ​​рабочая версия 2020 года.

@RepositoryRestController
public class EmployeeSearchController {
    @Autowired
    private EmployeeRepository employeRepository;

    @Autowired
    private PagedResourcesAssembler<Employee> pagedAssembler;

    @RequestMapping(value = "/employees/search/all", method = RequestMethod.GET)
    public ResponseEntity<PagedModel<EntityModel<Employee>>> getEmployees(PersistentEntityResourceAssembler entityAssembler,,
                                                                          EmployeeCriteria filterCriteria,
                                                                          Pageable pageable) {

        Specification<Employee> specification = new EmployeeSpecification(filterCriteria);

        Page<Employee> employees = employeeRepository.findAll(specification, pageable);
        return ResponseEntity.ok(pagedAssembler.toModel(plants, (RepresentationModelAssembler) entityAssembler));
    }
}
person Lala La    schedule 11.07.2020