Как использовать ответ <<Entity> с помощью Spring RestTemplate

Я использую данные spring (mongoDb), и у меня есть мой репозиторий:

public interface StoriesRepository extends PagingAndSortingRepository<Story, String> {}

Тогда у меня есть контроллер:

@RequestMapping(method = RequestMethod.GET)
public ResponseEntity<Page<StoryResponse>> getStories(Pageable pageable) {
    Page<StoryResponse> stories = storiesRepository.findAll(pageable).map(StoryResponseMapper::toStoryResponse);
    return ResponseEntity.ok(stories);
}

Все работает отлично, но я не могу использовать конечную точку с помощью метода getForEntity RestTemplate:

def entity = restTemplate.getForEntity(getLocalhost("/story"), new TypeReference<Page<StoryResponse>>(){}.class)

Какой класс я должен предоставить для успешной десериализации моей страницы объектов?

Ответы

Ответ 1

new TypeReference<Page<StoryResponse>>() {}

Проблема с этим утверждением заключается в том, что Джексон не может создать абстрактный тип. Вы должны дать Джексону информацию о том, как создать экземпляр Page с конкретным типом. Но его конкретный тип PageImpl не имеет конструктора по умолчанию или любого @JsonCreator, если вы не можете не использовать следующий код:

new TypeReference<PageImpl<StoryResponse>>() {}

Поскольку вы не можете добавить требуемую информацию в класс Page, лучше создать пользовательскую реализацию для интерфейса Page, у которой есть конструктор no-arg по умолчанию, как в этом ответ. Затем используйте эту настраиваемую реализацию в типе, например:

new TypeReference<CustomPageImpl<StoryResponse>>() {}

Вот пользовательская реализация, скопированная из связанного вопроса:

public class CustomPageImpl<T> extends PageImpl<T> {
    private static final long serialVersionUID = 1L;
    private int number;
    private int size;
    private int totalPages;
    private int numberOfElements;
    private long totalElements;
    private boolean previousPage;
    private boolean firstPage;
    private boolean nextPage;
    private boolean lastPage;
    private List<T> content;
    private Sort sort;

    public CustomPageImpl() {
        super(new ArrayList<>());
    }

    @Override
    public int getNumber() {
        return number;
    }

    public void setNumber(int number) {
        this.number = number;
    }

    @Override
    public int getSize() {
        return size;
    }

    public void setSize(int size) {
        this.size = size;
    }

    @Override
    public int getTotalPages() {
        return totalPages;
    }

    public void setTotalPages(int totalPages) {
        this.totalPages = totalPages;
    }

    @Override
    public int getNumberOfElements() {
        return numberOfElements;
    }

    public void setNumberOfElements(int numberOfElements) {
        this.numberOfElements = numberOfElements;
    }

    @Override
    public long getTotalElements() {
        return totalElements;
    }

    public void setTotalElements(long totalElements) {
        this.totalElements = totalElements;
    }

    public boolean isPreviousPage() {
        return previousPage;
    }

    public void setPreviousPage(boolean previousPage) {
        this.previousPage = previousPage;
    }

    public boolean isFirstPage() {
        return firstPage;
    }

    public void setFirstPage(boolean firstPage) {
        this.firstPage = firstPage;
    }

    public boolean isNextPage() {
        return nextPage;
    }

    public void setNextPage(boolean nextPage) {
        this.nextPage = nextPage;
    }

    public boolean isLastPage() {
        return lastPage;
    }

    public void setLastPage(boolean lastPage) {
        this.lastPage = lastPage;
    }

    @Override
    public List<T> getContent() {
        return content;
    }

    public void setContent(List<T> content) {
        this.content = content;
    }

    @Override
    public Sort getSort() {
        return sort;
    }

    public void setSort(Sort sort) {
        this.sort = sort;
    }

    public Page<T> pageImpl() {
        return new PageImpl<>(getContent(), new PageRequest(getNumber(),
                getSize(), getSort()), getTotalElements());
    }
}

Ответ 2

В качестве упоминаемого "pathfinder" вы можете использовать метод exchange RestTemplate. Однако вместо передачи ParameterizedTypeReference<Page<StoryResponse>>() вы должны пройти ParameterizedTypeReference<PagedResources<StoryResponse>>(). Когда вы получите ответ, вы можете получить контент - Collection<StoryResponse>.

Код должен выглядеть так:

ResponseEntity<PagedResources<StoryResponse>> response = restTemplate.exchange(getLocalhost("/story"),
        HttpMethod.GET, null, new ParameterizedTypeReference<PagedResources<StoryResponse>>() {});
PagedResources<StoryResponse> storiesResources = response.getBody();
Collection<StoryResponse> stories = storiesResources.getContent();

Помимо содержимого storiesResources хранятся метаданные и ссылки страницы.

Более подробное описание доступно здесь: fooobar.com/questions/357839/...

Ответ 3

Я знаю, что эта ветка немного старая, но, надеюсь, кто-то выиграет от этого.

@Ali Dehghani ответ хороший, за исключением того, что он повторно реализует то, что PageImpl<T> уже сделано. Я счел это совершенно ненужным. Я нашел лучшее решение, создав класс, который расширяет PageImpl<T> и указывает конструктор @JsonCreator:

import com.fasterxml.jackson.annotation.JsonCreator;
import com.fasterxml.jackson.annotation.JsonProperty;
import com.company.model.HelperModel;
import org.springframework.data.domain.PageImpl;
import org.springframework.data.domain.PageRequest;

import java.util.List;

public class HelperPage extends PageImpl<HelperModel> {

    @JsonCreator
    // Note: I don't need a sort, so I'm not including one here.
    // It shouldn't be too hard to add it in tho.
    public HelperPage(@JsonProperty("content") List<HelperModel> content,
                      @JsonProperty("number") int number,
                      @JsonProperty("size") int size,
                      @JsonProperty("totalElements") Long totalElements) {
        super(content, new PageRequest(number, size), totalElements);
    }
}

Тогда:

HelperPage page = restTemplate.getForObject(url, HelperPage.class);

Это то же самое, что и создание класса CustomPageImpl<T>, но позволяет нам использовать весь код, который уже находится в PageImpl<T>.

Ответ 4

Возможно, вы можете использовать метод обмена restTemplate и получить тело от него.

Отметьте следующий ответ fooobar.com/questions/55188/.... Это может помочь вам

Ответ 5

Вы можете попытаться добавить аннотацию @ResponseBody:

@RequestMapping(method = RequestMethod.GET)
@ResponseBody
public ResponseEntity<Page<StoryResponse>> getStories(@RequestBody Pageable pageable) {
    Page<StoryResponse> stories = storiesRepository.findAll(pageable).map(StoryResponseMapper::toStoryResponse);
    return ResponseEntity.ok(stories);
}