io.camunda.operate.webapp.opensearch.reader.OpensearchListViewReader Maven / Gradle / Ivy
/*
* Copyright Camunda Services GmbH and/or licensed to Camunda Services GmbH under
* one or more contributor license agreements. See the NOTICE file distributed
* with this work for additional information regarding copyright ownership.
* Licensed under the Camunda License 1.0. You may not use this file
* except in compliance with the Camunda License 1.0.
*/
package io.camunda.operate.webapp.opensearch.reader;
import static io.camunda.operate.store.opensearch.dsl.QueryDSL.*;
import static io.camunda.operate.store.opensearch.dsl.RequestDSL.QueryType.ALL;
import static io.camunda.operate.store.opensearch.dsl.RequestDSL.QueryType.ONLY_RUNTIME;
import static io.camunda.operate.store.opensearch.dsl.RequestDSL.searchRequestBuilder;
import static io.camunda.webapps.schema.descriptors.operate.template.ListViewTemplate.JOIN_RELATION;
import static io.camunda.webapps.schema.descriptors.operate.template.ListViewTemplate.PARENT_FLOW_NODE_INSTANCE_KEY;
import static io.camunda.webapps.schema.descriptors.operate.template.ListViewTemplate.PROCESS_INSTANCE_JOIN_RELATION;
import com.fasterxml.jackson.databind.ObjectMapper;
import io.camunda.operate.conditions.OpensearchCondition;
import io.camunda.operate.store.opensearch.client.sync.RichOpenSearchClient;
import io.camunda.operate.store.opensearch.dsl.RequestDSL;
import io.camunda.operate.util.CollectionUtil;
import io.camunda.operate.util.Tuple;
import io.camunda.operate.webapp.opensearch.OpenSearchQueryHelper;
import io.camunda.operate.webapp.reader.ListViewReader;
import io.camunda.operate.webapp.reader.OperationReader;
import io.camunda.operate.webapp.rest.dto.listview.ListViewProcessInstanceDto;
import io.camunda.operate.webapp.rest.dto.listview.ListViewRequestDto;
import io.camunda.operate.webapp.rest.dto.listview.ListViewResponseDto;
import io.camunda.webapps.schema.descriptors.operate.template.ListViewTemplate;
import io.camunda.webapps.schema.entities.operate.listview.ProcessInstanceForListViewEntity;
import io.camunda.webapps.schema.entities.operation.OperationEntity;
import java.util.List;
import java.util.Map;
import java.util.function.Consumer;
import org.opensearch.client.opensearch._types.SortOrder;
import org.opensearch.client.opensearch._types.query_dsl.Query;
import org.opensearch.client.opensearch.core.SearchRequest;
import org.opensearch.client.opensearch.core.SearchResponse;
import org.opensearch.client.opensearch.core.search.Hit;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.context.annotation.Conditional;
import org.springframework.stereotype.Component;
@Conditional(OpensearchCondition.class)
@Component
public class OpensearchListViewReader implements ListViewReader {
private static final Logger LOGGER = LoggerFactory.getLogger(OpensearchListViewReader.class);
private final RichOpenSearchClient richOpenSearchClient;
private final OpenSearchQueryHelper openSearchQueryHelper;
private final ObjectMapper objectMapper;
private final ListViewTemplate listViewTemplate;
private final OperationReader operationReader;
public OpensearchListViewReader(
final RichOpenSearchClient richOpenSearchClient,
final OpenSearchQueryHelper openSearchQueryHelper,
final ObjectMapper objectMapper,
final ListViewTemplate listViewTemplate,
final OperationReader operationReader) {
this.richOpenSearchClient = richOpenSearchClient;
this.openSearchQueryHelper = openSearchQueryHelper;
this.objectMapper = objectMapper;
this.listViewTemplate = listViewTemplate;
this.operationReader = operationReader;
}
@Override
public ListViewResponseDto queryProcessInstances(
final ListViewRequestDto processInstanceRequest) {
final ListViewResponseDto result = new ListViewResponseDto();
final List processInstanceEntities =
queryListView(processInstanceRequest, result);
final List processInstanceKeys =
CollectionUtil.map(
processInstanceEntities,
processInstanceEntity -> Long.valueOf(processInstanceEntity.getId()));
final Map> operationsPerProcessInstance =
operationReader.getOperationsPerProcessInstanceKey(processInstanceKeys);
final List processInstanceDtoList =
ListViewProcessInstanceDto.createFrom(
processInstanceEntities, operationsPerProcessInstance, objectMapper);
result.setProcessInstances(processInstanceDtoList);
return result;
}
@Override
public List queryListView(
final ListViewRequestDto processInstanceRequest, final ListViewResponseDto result) {
final RequestDSL.QueryType queryType =
processInstanceRequest.getQuery().isFinished() ? ALL : ONLY_RUNTIME;
final Query query =
constantScore(
withTenantCheck(
and(
term(JOIN_RELATION, PROCESS_INSTANCE_JOIN_RELATION),
openSearchQueryHelper.createQueryFragment(processInstanceRequest.getQuery()))));
LOGGER.debug("Process instance search request: \n{}", query);
final var searchRequestBuilder = searchRequestBuilder(listViewTemplate, queryType).query(query);
applySorting(searchRequestBuilder, processInstanceRequest);
searchRequestBuilder.size(processInstanceRequest.getPageSize());
final SearchResponse response =
richOpenSearchClient
.doc()
.fixedSearch(searchRequestBuilder.build(), ProcessInstanceForListViewEntity.class);
result.setTotalCount(response.hits().total().value());
final List processInstanceEntities =
response.hits().hits().stream()
.map(
hit -> {
final ProcessInstanceForListViewEntity entity = hit.source();
entity.setSortValues(hit.sort().toArray());
return entity;
})
.toList();
if (processInstanceRequest.getSearchBefore() != null) {
return CollectionUtil.reversedView(processInstanceEntities);
}
return processInstanceEntities;
}
@Override
public Tuple getCalledProcessInstanceIdAndNameByFlowNodeInstanceId(
final String flowNodeInstanceId) {
final String[] calledProcessInstanceId = {null};
final String[] calledProcessDefinitionName = {null};
findCalledProcessInstance(
flowNodeInstanceId,
hit -> {
final var source = hit.source();
calledProcessInstanceId[0] = hit.id();
var processName = source.getProcessName();
if (processName == null) {
processName = source.getBpmnProcessId();
}
calledProcessDefinitionName[0] = processName;
});
return Tuple.of(calledProcessInstanceId[0], calledProcessDefinitionName[0]);
}
private void findCalledProcessInstance(
final String flowNodeInstanceId,
final Consumer> processInstanceConsumer) {
final var request =
searchRequestBuilder(listViewTemplate.getAlias())
.query(withTenantCheck(term(PARENT_FLOW_NODE_INSTANCE_KEY, flowNodeInstanceId)))
.source(sourceInclude(ListViewTemplate.PROCESS_NAME, ListViewTemplate.BPMN_PROCESS_ID));
final var response =
richOpenSearchClient.doc().search(request, ProcessInstanceForListViewEntity.class);
if (response.hits().total().value() >= 1) {
processInstanceConsumer.accept(response.hits().hits().get(0));
}
}
private String getSortBy(final ListViewRequestDto request) {
if (request.getSorting() != null) {
String sortBy = request.getSorting().getSortBy();
if (sortBy.equals(ListViewRequestDto.SORT_BY_PARENT_INSTANCE_ID)) {
sortBy = ListViewTemplate.PARENT_PROCESS_INSTANCE_KEY;
} else if (sortBy.equals(ListViewRequestDto.SORT_BY_TENANT_ID)) {
sortBy = ListViewTemplate.TENANT_ID;
}
if (sortBy.equals(ListViewTemplate.ID)) {
// we sort by id as numbers, not as strings
sortBy = ListViewTemplate.KEY;
}
return sortBy;
}
return null;
}
private void applySorting(
final SearchRequest.Builder searchRequest, final ListViewRequestDto request) {
final String sortBy = getSortBy(request);
final boolean directSorting =
request.getSearchAfter() != null || request.getSearchBefore() == null;
if (request.getSorting() != null) {
final SortOrder directOrder =
"asc".equals(request.getSorting().getSortOrder()) ? SortOrder.Asc : SortOrder.Desc;
if (directSorting) {
searchRequest.sort(sortOptions(sortBy, directOrder, "_last"));
} else {
searchRequest.sort(sortOptions(sortBy, reverseOrder(directOrder), "_first"));
}
}
final Object[] querySearchAfter;
if (directSorting) {
searchRequest.sort(sortOptions(ListViewTemplate.KEY, SortOrder.Asc));
querySearchAfter = request.getSearchAfter(objectMapper);
} else {
searchRequest.sort(sortOptions(ListViewTemplate.KEY, SortOrder.Desc));
querySearchAfter = request.getSearchBefore(objectMapper);
}
searchRequest.size(request.getPageSize());
if (querySearchAfter != null) {
searchRequest.searchAfter(CollectionUtil.toSafeListOfStrings(querySearchAfter));
}
}
}