Please wait. This can take some minutes ...
Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance.
Project price only 1 $
You can buy this project and download/modify it how often you want.
org.springframework.data.rest.extensions.contentsearch.ContentSearchRestController Maven / Gradle / Ivy
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;
}
}