io.camunda.operate.webapp.api.v1.dao.opensearch.OpensearchProcessInstanceDao 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.api.v1.dao.opensearch;
import io.camunda.operate.conditions.OpensearchCondition;
import io.camunda.operate.connect.OperateDateTimeFormatter;
import io.camunda.operate.store.opensearch.client.sync.RichOpenSearchClient;
import io.camunda.operate.webapp.api.v1.dao.ProcessInstanceDao;
import io.camunda.operate.webapp.api.v1.entities.ChangeStatus;
import io.camunda.operate.webapp.api.v1.entities.ProcessInstance;
import io.camunda.operate.webapp.api.v1.entities.Query;
import io.camunda.operate.webapp.api.v1.exceptions.APIException;
import io.camunda.operate.webapp.api.v1.exceptions.ClientException;
import io.camunda.operate.webapp.api.v1.exceptions.ServerException;
import io.camunda.operate.webapp.opensearch.OpensearchQueryDSLWrapper;
import io.camunda.operate.webapp.opensearch.OpensearchRequestDSLWrapper;
import io.camunda.operate.webapp.writer.ProcessInstanceWriter;
import io.camunda.webapps.schema.descriptors.operate.template.ListViewTemplate;
import java.util.LinkedList;
import java.util.List;
import java.util.Objects;
import org.apache.commons.lang3.StringUtils;
import org.opensearch.client.opensearch.core.SearchRequest;
import org.springframework.context.annotation.Conditional;
import org.springframework.security.access.prepost.PreAuthorize;
import org.springframework.stereotype.Component;
@Conditional(OpensearchCondition.class)
@Component
public class OpensearchProcessInstanceDao
extends OpensearchKeyFilteringDao
implements ProcessInstanceDao {
private final ListViewTemplate processInstanceIndex;
private final ProcessInstanceWriter processInstanceWriter;
private final OperateDateTimeFormatter dateTimeFormatter;
public OpensearchProcessInstanceDao(
final OpensearchQueryDSLWrapper queryDSLWrapper,
final OpensearchRequestDSLWrapper requestDSLWrapper,
final RichOpenSearchClient richOpenSearchClient,
final ListViewTemplate processInstanceIndex,
final ProcessInstanceWriter processInstanceWriter,
final OperateDateTimeFormatter dateTimeFormatter) {
super(queryDSLWrapper, requestDSLWrapper, richOpenSearchClient);
this.processInstanceIndex = processInstanceIndex;
this.processInstanceWriter = processInstanceWriter;
this.dateTimeFormatter = dateTimeFormatter;
}
@Override
protected String getUniqueSortKey() {
// This probably should be ProcessInstance.KEY to be consistent with how we always pull sort
// keys
// from the constants in the index. Since the ElasticsearchDao uses this field too, leaving it
// as-is.
// While the `processInstanceKey` field is not in the ProcessInstance model object, it can still
// be
// used as a sort key (and has the same value as `key` which is what is referred to here with
// ListViewTemplate.KEY.
return ListViewTemplate.KEY;
}
@Override
protected Class getInternalDocumentModelClass() {
return ProcessInstance.class;
}
@Override
protected String getIndexName() {
return processInstanceIndex.getAlias();
}
@Override
protected void buildFiltering(
final Query query, final SearchRequest.Builder request) {
final List queryTerms =
new LinkedList<>();
queryTerms.add(
queryDSLWrapper.term(
ListViewTemplate.JOIN_RELATION, ListViewTemplate.PROCESS_INSTANCE_JOIN_RELATION));
final ProcessInstance filter = query.getFilter();
if (filter != null) {
queryTerms.add(queryDSLWrapper.term(ProcessInstance.KEY, filter.getKey()));
queryTerms.add(
queryDSLWrapper.term(
ProcessInstance.PROCESS_DEFINITION_KEY, filter.getProcessDefinitionKey()));
queryTerms.add(queryDSLWrapper.term(ProcessInstance.PARENT_KEY, filter.getParentKey()));
queryTerms.add(
queryDSLWrapper.term(
ProcessInstance.PARENT_FLOW_NODE_INSTANCE_KEY,
filter.getParentFlowNodeInstanceKey()));
queryTerms.add(queryDSLWrapper.term(ProcessInstance.VERSION, filter.getProcessVersion()));
queryTerms.add(
queryDSLWrapper.term(ProcessInstance.VERSION_TAG, filter.getProcessVersionTag()));
queryTerms.add(
queryDSLWrapper.term(ProcessInstance.BPMN_PROCESS_ID, filter.getBpmnProcessId()));
queryTerms.add(queryDSLWrapper.term(ProcessInstance.STATE, filter.getState()));
queryTerms.add(queryDSLWrapper.term(ProcessInstance.INCIDENT, filter.getIncident()));
queryTerms.add(queryDSLWrapper.term(ProcessInstance.TENANT_ID, filter.getTenantId()));
queryTerms.add(
queryDSLWrapper.matchDateQuery(
ProcessInstance.START_DATE,
filter.getStartDate(),
dateTimeFormatter.getApiDateTimeFormatString()));
queryTerms.add(
queryDSLWrapper.matchDateQuery(
ProcessInstance.END_DATE,
filter.getEndDate(),
dateTimeFormatter.getApiDateTimeFormatString()));
}
final var nonNullQueryTerms = queryTerms.stream().filter(Objects::nonNull).toList();
request.query(queryDSLWrapper.and(nonNullQueryTerms));
}
@Override
protected ProcessInstance convertInternalToApiResult(final ProcessInstance internalResult) {
if (internalResult != null) {
if (StringUtils.isNotEmpty(internalResult.getEndDate())) {
internalResult.setEndDate(
dateTimeFormatter.convertGeneralToApiDateTime(internalResult.getEndDate()));
}
if (StringUtils.isNotEmpty(internalResult.getStartDate())) {
internalResult.setStartDate(
dateTimeFormatter.convertGeneralToApiDateTime(internalResult.getStartDate()));
}
}
return internalResult;
}
@Override
@PreAuthorize("hasPermission('write')")
public ChangeStatus delete(final Long key) throws APIException {
// Check for not exists
byKey(key);
try {
processInstanceWriter.deleteInstanceById(key);
return new ChangeStatus()
.setDeleted(1)
.setMessage(
String.format("Process instance and dependant data deleted for key '%s'", key));
} catch (final IllegalArgumentException iae) {
throw new ClientException(iae.getMessage(), iae);
} catch (final Exception e) {
throw new ServerException(
String.format("Error in deleting process instance and dependant data for key '%s'", key),
e);
}
}
@Override
protected List searchByKey(final Long key) {
final List queryTerms =
new LinkedList<>();
queryTerms.add(
queryDSLWrapper.term(
ListViewTemplate.JOIN_RELATION, ListViewTemplate.PROCESS_INSTANCE_JOIN_RELATION));
queryTerms.add(queryDSLWrapper.term(getKeyFieldName(), key));
final SearchRequest.Builder request =
requestDSLWrapper
.searchRequestBuilder(getIndexName())
.query(queryDSLWrapper.withTenantCheck(queryDSLWrapper.and(queryTerms)));
return richOpenSearchClient.doc().searchValues(request, getInternalDocumentModelClass());
}
@Override
protected String getKeyFieldName() {
return ProcessInstance.KEY;
}
@Override
protected String getByKeyServerReadErrorMessage(final Long key) {
return String.format("Error in reading process instance for key %s", key);
}
@Override
protected String getByKeyNoResultsErrorMessage(final Long key) {
return String.format("No process instances found for key %s", key);
}
@Override
protected String getByKeyTooManyResultsErrorMessage(final Long key) {
return String.format("Found more than one process instances for key %s", key);
}
}
© 2015 - 2024 Weber Informatics LLC | Privacy Policy