ElasticsearchService.java
/*
* Copyright 2022 Global Crop Diversity Trust
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.genesys.server.service;
import java.beans.Transient;
import java.io.Serializable;
import java.util.Collection;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.ExecutionException;
import org.elasticsearch.index.query.QueryBuilder;
import org.genesys.blocks.model.EmptyModel;
import org.genesys.blocks.model.filters.EmptyModelFilter;
import org.genesys.server.service.AccessionService.IBatchAction;
import org.genesys.server.exception.SearchException;
import org.springframework.data.domain.Page;
import org.springframework.data.domain.Pageable;
import com.fasterxml.jackson.annotation.JsonInclude;
import com.querydsl.core.types.Predicate;
import com.querydsl.core.types.dsl.SetPath;
/**
* The Interface ElasticsearchService.
*/
public interface ElasticsearchService {
/**
* Makes sure indices are ready and up-to-date on startup.
*
* @param <R> the generic type
* @param clazz the clazz
*/
<R extends EmptyModel> void indexEntity(Class<R> clazz);
/**
* Reindex.
*
* @param <R> the generic type
* @param clazz the clazz
*/
<R> void reindex(Class<R> clazz);
/**
* Update.
*
* @param <R> the generic type
* @param clazz the clazz
* @param ids the ids
*/
<R> void update(Class<R> clazz, Collection<Long> ids);
/**
* Async update.
*
* @param <R> the generic type
* @param clazz the clazz
* @param bucket the bucket
*/
<R> void asyncUpdate(Class<R> clazz, Collection<Long> bucket);
/**
* Search single entity.
*
* @param shouldMatch the should match
* @param searchQuery the search query
* @param clazz the clazz
* @return the list
*/
<T extends EmptyModel> List<T> search(QueryBuilder shouldMatch, String searchQuery, Class<T> clazz);
/**
* Search using query for selected entity types.
*
* @param shouldMatch Any additional filters that should match
* @param searchQuery the search query
* @param clazzes the clazzes
* @return the map
*/
<T extends EmptyModel> Map<Class<? extends EmptyModel>, List<? extends EmptyModel>> search(QueryBuilder shouldMatch, String searchQuery, Set<Class<? extends EmptyModel>> clazzes);
/**
* Term statistics auto.
* @param filters the filters
* @param instcode the instcode
* @param i the i
*
* @return the term result
* @throws SearchException the search exception
*/
TermResult termStatisticsAuto(Class<? extends EmptyModel> clazz, EmptyModelFilter<?, ?> filters, int size, String term) throws SearchException;
Map<String, TermResult> termStatisticsAuto(Class<? extends EmptyModel> clazz, EmptyModelFilter<?, ?> filters, int size, String... terms) throws SearchException;
TreeNode treeNodeStatistics(Class<? extends EmptyModel> clazz, EmptyModelFilter<?, ?> filters, String[] terms) throws SearchException;
TermResult termStatistics(Class<? extends EmptyModel> clazz, EmptyModelFilter<?, ?> filters, int size, String term) throws SearchException;
Map<String, TermResult> termStatistics(Class<? extends EmptyModel> clazz, EmptyModelFilter<?, ?> filters, int size, String... terms) throws SearchException;
void realias(String aliasName, String indexName);
void addAlias(String aliasName, String indexName);
void deleteAlias(String aliasName);
void deleteIndex(String indexName);
void reindexAll();
void stopReindex();
void stopReindexAll();
void allowReindexAll();
TermResult recountResult(Class<? extends EmptyModel> clazz, SetPath<?, ?> setPath, EmptyModelFilter<?, ?> filter, TermResult toRecount, String termName) throws ExecutionException, InterruptedException, SearchException;
<T extends EmptyModel> void reindex(Class<T> clazz, EmptyModelFilter<?, ?> filter);
List<Class<?>> getIndexedEntities();
long count(Class<? extends EmptyModel> clazz, EmptyModelFilter<?, ?> filter) throws SearchException;
/**
* Count missing values for all fields of specified class.
* @param clazz the index class
* @param filter the EmptyModelFilter<?, ?> filter
* @return the map of all JSON paths with their missing value
*/
Map<String, Long> countMissingValues(Class<? extends EmptyModel> clazz, EmptyModelFilter<?, ?> filter) throws SearchException;
public static class TermResult implements Serializable {
private static final long serialVersionUID = -6646063484562660447L;
private final List<Term> terms;
private final Long total;
private final long other;
private Long missing;
public TermResult(String name, Long total, List<Term> terms, long other) {
this.terms = terms;
this.total = total;
this.other = other;
this.missing = total - terms.stream().map(Term::getCount).mapToLong(Long::longValue).sum() - other;
if (missing <= 0) {
// We have some terms (storage) where the total is not the total number of all
// storage options, but count of accessions
missing = null;
}
}
public List<Term> getTerms() {
return terms;
}
public Long getTotal() {
return total;
}
@Transient
public Long getTotalCount() {
return total;
}
public long getOther() {
return other;
}
public Long getMissing() {
return missing;
}
}
public static class Term implements Serializable {
private static final long serialVersionUID = -4161698220975370044L;
private final String term;
private final long count;
public Term(String term, long count) {
this.term = term;
this.count = count;
}
public String getTerm() {
return term;
}
public long getCount() {
return count;
}
}
public static class TreeNode implements Serializable {
private static final long serialVersionUID = 4878674372987777983L;
public String groupBy; // ES aggregation group
public String name; // term value
public long value; // term count
public List<TreeNode> children;
public Object filter; // filters
@JsonInclude(JsonInclude.Include.NON_EMPTY)
public String filterCode; // filter code
public TreeNode(String groupBy, String name, long value, List<TreeNode> children, Object filter) {
this.groupBy = groupBy;
this.name = name;
this.value = value;
this.children = children;
this.filter = filter;
}
public String getName() {
return name;
}
public long getValue() {
return value;
}
public List<TreeNode> getChildren() {
return children;
}
public Object getFilter() {
return filter;
}
}
/**
* Make a full-text search for top 5 hits for each of the indexed entities.
*
* @param text the search text
* @return map
* @throws SearchException
*/
Map<String, List<? extends EmptyModel>> fullTextSearch(String text) throws SearchException;
<T extends EmptyModel> List<T> find(Class<T> clazz, EmptyModelFilter<?, ?> filter) throws SearchException;
<T extends EmptyModel> Page<T> findAll(Class<T> clazz, EmptyModelFilter<?, ?> filter, Pageable page) throws SearchException;
<T extends EmptyModel> Page<T> findAll(Class<T> clazz, EmptyModelFilter<?, ?> filter, Predicate predicate, Pageable page) throws SearchException;
/**
* The usual search, but with a custom entity loader
*/
<T extends EmptyModel> Page<T> findAll(Class<T> clazz, EmptyModelFilter<?, ?> filter, Predicate predicate, Pageable page, IEntityLoader<T> entityLoader, String... boostFields) throws SearchException;
Number[][] getAccessionGeoBounds(EmptyModelFilter<?, ?> filter) throws SearchException;
public static interface IEntityLoader<T> {
List<T> loadEntities(List<Long> entityIds);
}
/**
* Wrapper for search results
*/
public static class SearchResults<T extends EmptyModel> {
public List<String> filters;
public String key = "uuid";
public List<T> hits;
public static <T extends EmptyModel> SearchResults<T> from(String key, List<String> filters, List<T> list) {
if (list == null || list.isEmpty())
return null;
SearchResults<T> sr = new SearchResults<T>();
sr.filters = filters;
sr.key = key;
sr.hits = list;
return sr;
}
}
List<Double[]> distinctCoordinates(Predicate filt, String _text) throws SearchException;
/*
* Scroll search and process entity records in a batch action.
*/
<T extends EmptyModel> void process(Class<T> clazz, EmptyModelFilter<?, ?> filter, IBatchAction<T> action, Long maxSize) throws Exception;
/**
* Process entity IDs in a batch action.
*/
<T extends EmptyModel> void processById(Class<T> clazz, EmptyModelFilter<?, ?> filter, IBatchAction<Long> action, Pageable page) throws Exception, SearchException;
/**
* Wait until X records match specified filter in ES.
*
* @param clazz
* @param filter
* @param mustHaveCount
* @throws InterruptedException
*/
long waitForCount(Class<? extends EmptyModel> clazz, EmptyModelFilter<?, ?> filter, int mustHaveCount) throws SearchException;
/**
* Remove matching documents from index
*
* @param clazz
* @param filter
*/
<T extends EmptyModel> void remove(Class<T> clazz, EmptyModelFilter<?, T> filter) throws SearchException;
/**
* Delete all documents from the specified index
*
* @param <R> the type
* @param clazz realias target
* @throws SearchException
*/
<R> void removeAll(Class<R> clazz) throws SearchException;
}