All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.springframework.data.rest.extensions.contentsearch.ContentSearchRestController Maven / Gradle / Ivy

There is a newer version: 3.0.15
Show newest version
package org.springframework.data.rest.extensions.contentsearch;

import static java.lang.String.format;

import java.io.Serializable;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

import jakarta.persistence.Id;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.content.commons.annotations.ContentId;
import org.springframework.content.commons.fragments.ParameterTypeAware;
import org.springframework.content.commons.repository.ContentStore;
import org.springframework.content.commons.search.Searchable;
import org.springframework.content.commons.storeservice.StoreFilter;
import org.springframework.content.commons.storeservice.StoreInfo;
import org.springframework.content.commons.storeservice.Stores;
import org.springframework.content.commons.utils.BeanUtils;
import org.springframework.content.commons.utils.ContentPropertyUtils;
import org.springframework.content.commons.utils.DomainObjectUtils;
import org.springframework.content.commons.utils.ReflectionService;
import org.springframework.content.commons.utils.ReflectionServiceImpl;
import org.springframework.content.rest.FulltextEntityLookupQuery;
import org.springframework.data.domain.Page;
import org.springframework.data.domain.PageImpl;
import org.springframework.data.domain.Pageable;
import org.springframework.data.domain.Sort;
import org.springframework.data.repository.CrudRepository;
import org.springframework.data.repository.core.RepositoryInformation;
import org.springframework.data.repository.support.Repositories;
import org.springframework.data.rest.webmvc.PersistentEntityResourceAssembler;
import org.springframework.data.rest.webmvc.RepositoryRestController;
import org.springframework.data.rest.webmvc.ResourceNotFoundException;
import org.springframework.data.rest.webmvc.RootResourceInformation;
import org.springframework.data.rest.webmvc.support.DefaultedPageable;
import org.springframework.data.web.PagedResourcesAssembler;
import org.springframework.hateoas.CollectionModel;
import org.springframework.util.ReflectionUtils;
import org.springframework.web.bind.annotation.PathVariable;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.bind.annotation.RequestParam;
import org.springframework.web.bind.annotation.ResponseBody;

import internal.org.springframework.content.rest.controllers.BadRequestException;
import internal.org.springframework.content.rest.mappings.ContentHandlerMapping.StoreType;
import internal.org.springframework.content.rest.utils.ControllerUtils;
import internal.org.springframework.content.rest.utils.RepositoryUtils;
import internal.org.springframework.data.rest.extensions.contentsearch.DefaultEntityLookupStrategy;
import internal.org.springframework.data.rest.extensions.contentsearch.QueryMethodsEntityLookupStrategy;
import lombok.AllArgsConstructor;
import lombok.Getter;
import lombok.NoArgsConstructor;
import lombok.Setter;

@RepositoryRestController
public class ContentSearchRestController {

    private static final String ENTITY_CONTENTSEARCH_MAPPING = "/{repository}/searchContent";
    private static final String ENTITY_SEARCHMETHOD_MAPPING = "/{repository}/searchContent/findKeyword";

    private static Map searchMethods = new HashMap<>();

    private Repositories repositories;
    private Stores stores;
    private PagedResourcesAssembler pagedResourcesAssembler;
    private DefaultEntityLookupStrategy defaultLookupStrategy;
    private QueryMethodsEntityLookupStrategy qmLookupStrategy;

    private ReflectionService reflectionService;

    static {
        put(searchMethods, "search", ReflectionUtils.findMethod(Searchable.class, "search", new Class[] { String.class}));
        put(searchMethods, "search", ReflectionUtils.findMethod(Searchable.class, "search", new Class[] { String.class, Pageable.class}));
        put(searchMethods, "findKeyword", ReflectionUtils.findMethod(Searchable.class, "findKeyword", new Class[] { String.class }));
    }

    private static void put(Map methods, String key, Method findMethod) {

        methods.put(format("%s:%s", key, Arrays.toString(findMethod.getParameterTypes())), findMethod);
    }

    @Autowired
    public ContentSearchRestController(Repositories repositories, Stores stores, PagedResourcesAssembler assembler) {

        this.repositories = repositories;
        this.stores = stores;
        this.pagedResourcesAssembler = assembler;

        this.reflectionService = new ReflectionServiceImpl();
        this.defaultLookupStrategy = new DefaultEntityLookupStrategy();
        this.qmLookupStrategy = new QueryMethodsEntityLookupStrategy();
    }

    public void setReflectionService(ReflectionService reflectionService) {
        this.reflectionService = reflectionService;
    }

    public void setDefaultEntityLookupStrategy(DefaultEntityLookupStrategy lookupStrategy) {
        this.defaultLookupStrategy = lookupStrategy;
    }

    public void setQueryMethodsEntityLookupStrategy(QueryMethodsEntityLookupStrategy lookupStrategy) {
        this.qmLookupStrategy = lookupStrategy;
    }

    @SuppressWarnings({ "rawtypes", "unchecked" })
    @StoreType("contentstore")
    @ResponseBody
    @RequestMapping(value = ENTITY_CONTENTSEARCH_MAPPING, method = RequestMethod.GET)
    public CollectionModel searchContent(
            RootResourceInformation repoInfo,
            DefaultedPageable pageable,
            Sort sort,
            PersistentEntityResourceAssembler assembler,
            @PathVariable String repository,
            @RequestParam(name = "queryString") String queryString) {

        return searchContentInternal(repoInfo, repository, pageable, sort, assembler, "search", new String[] { queryString });
    }

    @SuppressWarnings({ "rawtypes", "unchecked" })
    @StoreType("contentstore")
    @ResponseBody
    @RequestMapping(value = ENTITY_SEARCHMETHOD_MAPPING, method = RequestMethod.GET)
    public CollectionModel searchContent(RootResourceInformation repoInfo, DefaultedPageable pageable, Sort sort, PersistentEntityResourceAssembler assembler, @PathVariable String repository, @RequestParam(name = "keyword") List keywords) {

        return searchContentInternal(repoInfo, repository, pageable, sort, assembler, "findKeyword", keywords.toArray(new String[] {}));
    }

    private CollectionModel searchContentInternal(RootResourceInformation repoInfo, String repository, DefaultedPageable pageable, Sort sort, PersistentEntityResourceAssembler assembler, String searchMethod, String[] keywords) {

        StoreInfo[] infos = stores.getStores(ContentStore.class, new StoreFilter() {
            @Override
            public String name() {
                return "test";
            }

            @Override
            public boolean matches(StoreInfo info) {
                return repoInfo.getDomainType().equals(info.getDomainObjectClass());
            }
        });

        if (infos.length == 0) {
            throw new ResourceNotFoundException("Entity has no content associations");
        }

        if (infos.length > 1) {
            throw new IllegalStateException(String.format("Too many content assocation for Entity %s", repoInfo.getDomainType().getCanonicalName()));
        }

        StoreInfo info = infos[0];

        ContentStore store = info.getImplementation(ContentStore.class);
        if (store instanceof Searchable == false) {
            throw new ResourceNotFoundException("Entity content is not searchable");
        }

        Class[] searchMethodArgTypes = (pageable.unpagedIfDefault().isUnpaged() ? new Class[] {String.class} : new Class[] {String.class, Pageable.class} );
        Method method = searchMethods.get(format("%s:%s", searchMethod, Arrays.toString(searchMethodArgTypes)));

        if (method == null) {
            throw new BadRequestException(String.format("Invalid search: %s", searchMethod));
        }

        if (keywords == null || keywords.length == 0) {
            throw new BadRequestException();
        }

        Class returnType = returnType(info);
        if (ContentPropertyUtils.isPrimitiveContentPropertyClass(returnType)) {

            Method parameterTypeAwareMethod = ReflectionUtils.findMethod(ParameterTypeAware.class, "setGenericArguments", Class[].class);
            if (parameterTypeAwareMethod != null) {
                reflectionService.invokeMethod(parameterTypeAwareMethod, store, new Object[] {new Class[] {InternalResult.class}});
            }

            returnType = InternalResult.class;
        }

        Object[] argValues = (pageable.unpagedIfDefault().isUnpaged()) ? new String[] {keywords[0]} : new Object[] {keywords[0], pageable.getPageable()};
        Iterable intermediateResults = (Iterable)reflectionService.invokeMethod(method, store, argValues);

        if (intermediateResults == null || intermediateResults.iterator().hasNext() == false) {
            return CollectionModel.empty();
        }

        final List results = new ArrayList<>();

        RepositoryInformation ri = RepositoryUtils.findRepositoryInformation(repositories, repository);
        Class domainClass = ri.getDomainType();

        if (returnType.equals(InternalResult.class)) {

            boolean idFieldEqualsContentIdField = isIdFieldOverloaded(repoInfo.getDomainType());

            List entityIds = new ArrayList<>();
            List contentIds = new ArrayList<>();

            for (Object tempResult : intermediateResults) {
                InternalResult internalResult = (InternalResult)tempResult;
                if (internalResult.getId() != null) {
                    entityIds.add(internalResult.getId());
                } else if (idFieldEqualsContentIdField) {
                    entityIds.add(internalResult.getContentId());
                } else if (internalResult.getContentId() != null) {
                    contentIds.add(internalResult.getContentId());
                }
            }

            if (entityIds.size() > 0) {
                repositories.getRepositoryFor(domainClass).ifPresent(r -> {

                    fetchEntitiesInBatches((CrudRepository)r, entityIds, results);
                });
            }

            if (contentIds.size() > 0) {
                if (ri != null) {
                    if (ri.getQueryMethods()
                            .filter(m -> m.getAnnotation(FulltextEntityLookupQuery.class) != null)
                            .isEmpty()) {

                        defaultLookupStrategy.lookup(repoInfo, ri, contentIds, results);
                    } else {

                        qmLookupStrategy.lookup(repoInfo, ri, contentIds, results);
                    }
                }
            }

            Iterable wrappedResults = convertToFinalResultType(results, pageable, intermediateResults);
            return ControllerUtils.toCollectionModel(wrappedResults, pagedResourcesAssembler, assembler, domainClass);
        } else {
            intermediateResults.forEach(results::add);
            Iterable wrappedResults = convertToFinalResultType(results, pageable, intermediateResults);
            return ControllerUtils.toCollectionModel(wrappedResults, pagedResourcesAssembler, null, results.get(0).getClass());
        }
    }

    private Iterable convertToFinalResultType(List results, DefaultedPageable pageable, Iterable intermediateResults) {

        if (pageable.unpagedIfDefault().isUnpaged()) {
            return results;
        } else if (intermediateResults instanceof Page) {
            return new PageImpl(results, pageable.getPageable(), ((Page)intermediateResults).getTotalPages());
        } else {
            return new PageImpl(results);
        }
    }

    public static void fetchEntitiesInBatches(CrudRepository r, List entityIds, List results) {

        int size = entityIds.size();

        Method findAllByIdMethod = ReflectionUtils.findMethod(CrudRepository.class, "findAllById", Iterable.class);

        for (int i=0; i < size; ) {
            int lowerbound = i;
            int upperbound = (lowerbound + 250 <= entityIds.size() ? lowerbound + 250 : size);

            List subset = entityIds.subList(lowerbound, upperbound);

            Iterable entities = (Iterable) ReflectionUtils.invokeMethod(findAllByIdMethod, r, subset);
            for (Object entity : entities) {
                results.add(entity);
            }

            i += 250;
        }
    }

    private Class returnType(StoreInfo info) {
        Class storeInterfaceClass = info.getInterface();
        Class searchReturnType = String.class;
        for (Type t : storeInterfaceClass.getGenericInterfaces()) {
            if (t.getTypeName().startsWith("org.springframework.content.commons.search")) {
                if (t instanceof ParameterizedType) {
                    Type[] fragmentGenericTypes = ((ParameterizedType)t).getActualTypeArguments();
                    if (fragmentGenericTypes != null) {
                        searchReturnType = (Class)fragmentGenericTypes[0];
                    }
                }
            }
        }
        return searchReturnType;
    }

    private boolean isIdFieldOverloaded(Class domainClass) {

        Field idField = DomainObjectUtils.getIdField(domainClass);
        Field contentIdField = BeanUtils.findFieldWithAnnotation(domainClass, ContentId.class);

        if (idField.equals(contentIdField)) {
            return true;
        }
        return false;
    }

    @Getter
    @Setter
    @NoArgsConstructor
    @AllArgsConstructor
    public static class InternalResult {

        @Id
        private Object id;

        @ContentId
        private Object contentId;
    }
}