io.camunda.exporter.handlers.ListViewFlowNodeFromProcessInstanceHandler 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.exporter.handlers;
import static io.camunda.exporter.utils.ExporterUtil.tenantOrDefault;
import static io.camunda.webapps.schema.descriptors.IndexTemplateDescriptor.POSITION;
import static io.camunda.webapps.schema.descriptors.operate.template.ListViewTemplate.ACTIVITY_ID;
import static io.camunda.webapps.schema.descriptors.operate.template.ListViewTemplate.ACTIVITY_STATE;
import static io.camunda.webapps.schema.descriptors.operate.template.ListViewTemplate.ACTIVITY_TYPE;
import static io.camunda.zeebe.protocol.record.intent.ProcessInstanceIntent.*;
import static io.camunda.zeebe.protocol.record.intent.ProcessInstanceIntent.ELEMENT_TERMINATED;
import io.camunda.exporter.store.BatchRequest;
import io.camunda.webapps.schema.entities.operate.FlowNodeState;
import io.camunda.webapps.schema.entities.operate.FlowNodeType;
import io.camunda.webapps.schema.entities.operate.listview.FlowNodeInstanceForListViewEntity;
import io.camunda.zeebe.protocol.record.Record;
import io.camunda.zeebe.protocol.record.ValueType;
import io.camunda.zeebe.protocol.record.intent.Intent;
import io.camunda.zeebe.protocol.record.value.BpmnElementType;
import io.camunda.zeebe.protocol.record.value.ProcessInstanceRecordValue;
import java.util.*;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
public class ListViewFlowNodeFromProcessInstanceHandler
implements ExportHandler {
private static final Logger LOGGER =
LoggerFactory.getLogger(ListViewFlowNodeFromProcessInstanceHandler.class);
private static final Set PI_AND_AI_START_STATES = Set.of(ELEMENT_ACTIVATING);
private static final Set PI_AND_AI_FINISH_STATES =
Set.of(ELEMENT_COMPLETED, ELEMENT_TERMINATED);
private final String indexName;
private final boolean concurrencyMode;
public ListViewFlowNodeFromProcessInstanceHandler(
final String indexName, final boolean concurrencyMode) {
this.indexName = indexName;
this.concurrencyMode = concurrencyMode;
}
@Override
public ValueType getHandledValueType() {
return ValueType.PROCESS_INSTANCE;
}
@Override
public Class getEntityType() {
return FlowNodeInstanceForListViewEntity.class;
}
@Override
public boolean handlesRecord(final Record record) {
final var intent = record.getIntent();
if (!isProcessEvent(record.getValue())) {
return PI_AND_AI_START_STATES.contains(intent)
|| PI_AND_AI_FINISH_STATES.contains(intent)
|| ELEMENT_MIGRATED.equals(intent);
}
return false;
}
@Override
public List generateIds(final Record record) {
return List.of(String.valueOf(record.getKey()));
}
@Override
public FlowNodeInstanceForListViewEntity createNewEntity(final String id) {
return new FlowNodeInstanceForListViewEntity().setId(id);
}
@Override
public void updateEntity(
final Record record,
final FlowNodeInstanceForListViewEntity entity) {
final var recordValue = record.getValue();
final var intent = record.getIntent();
entity.setKey(record.getKey());
entity.setId(String.valueOf(record.getKey()));
entity.setPartitionId(record.getPartitionId());
entity.setPosition(record.getPosition());
entity.setActivityId(recordValue.getElementId());
entity.setProcessInstanceKey(recordValue.getProcessInstanceKey());
entity.setTenantId(tenantOrDefault(recordValue.getTenantId()));
if (PI_AND_AI_FINISH_STATES.contains(intent)) {
if (intent.equals(ELEMENT_TERMINATED)) {
entity.setActivityState(FlowNodeState.TERMINATED);
} else {
entity.setActivityState(FlowNodeState.COMPLETED);
}
} else {
entity.setActivityState(FlowNodeState.ACTIVE);
if (PI_AND_AI_START_STATES.contains(intent)) {
entity.setStartTime(record.getTimestamp());
}
}
entity.setActivityType(
FlowNodeType.fromZeebeBpmnElementType(
recordValue.getBpmnElementType() == null
? null
: recordValue.getBpmnElementType().name()));
// set parent
final Long processInstanceKey = recordValue.getProcessInstanceKey();
entity.getJoinRelation().setParent(processInstanceKey);
}
@Override
public void flush(
final FlowNodeInstanceForListViewEntity entity, final BatchRequest batchRequest) {
LOGGER.debug("Flow node instance for list view: id {}", entity.getId());
final Map updateFields = new LinkedHashMap<>();
updateFields.put(POSITION, entity.getPosition());
updateFields.put(ACTIVITY_ID, entity.getActivityId());
updateFields.put(ACTIVITY_TYPE, entity.getActivityType());
updateFields.put(ACTIVITY_STATE, entity.getActivityState());
final Long processInstanceKey = entity.getProcessInstanceKey();
if (concurrencyMode) {
batchRequest.upsertWithScriptAndRouting(
indexName,
entity.getId(),
entity,
getFlowNodeInstanceScript(),
updateFields,
processInstanceKey.toString());
} else {
batchRequest.upsertWithRouting(
indexName, entity.getId(), entity, updateFields, processInstanceKey.toString());
}
}
@Override
public String getIndexName() {
return indexName;
}
private boolean isProcessEvent(final ProcessInstanceRecordValue recordValue) {
return isOfType(recordValue, BpmnElementType.PROCESS);
}
private boolean isOfType(
final ProcessInstanceRecordValue recordValue, final BpmnElementType type) {
final BpmnElementType bpmnElementType = recordValue.getBpmnElementType();
if (bpmnElementType == null) {
return false;
}
return bpmnElementType.equals(type);
}
protected String getFlowNodeInstanceScript() {
return String.format(
"if (ctx._source.%s == null || ctx._source.%s < params.%s) { "
+ "ctx._source.%s = params.%s; " // position
+ "ctx._source.%s = params.%s; " // activity id
+ "ctx._source.%s = params.%s; " // activity type
+ "ctx._source.%s = params.%s; " // activity state
+ "}",
POSITION,
POSITION,
POSITION,
POSITION,
POSITION,
ACTIVITY_ID,
ACTIVITY_ID,
ACTIVITY_TYPE,
ACTIVITY_TYPE,
ACTIVITY_STATE,
ACTIVITY_STATE);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy