X-Git-Url: https://juplo.de/gitweb/?a=blobdiff_plain;f=src%2Fmain%2Fjava%2Fde%2Fjuplo%2Fkafka%2FAdderRebalanceListener.java;fp=src%2Fmain%2Fjava%2Fde%2Fjuplo%2Fkafka%2FAdderRebalanceListener.java;h=284aff580df354d6c574c11c83baacb29a6673f7;hb=5b4b7acf7b6a02e0e5c779257d3f5996366625e6;hp=0000000000000000000000000000000000000000;hpb=f83599b6aaefff62c286e2143bb2e8a81751e6fd;p=demos%2Fkafka%2Ftraining diff --git a/src/main/java/de/juplo/kafka/AdderRebalanceListener.java b/src/main/java/de/juplo/kafka/AdderRebalanceListener.java new file mode 100644 index 0000000..284aff5 --- /dev/null +++ b/src/main/java/de/juplo/kafka/AdderRebalanceListener.java @@ -0,0 +1,81 @@ +package de.juplo.kafka; + +import lombok.RequiredArgsConstructor; +import lombok.extern.slf4j.Slf4j; +import org.apache.kafka.clients.consumer.Consumer; +import org.apache.kafka.common.TopicPartition; + +import java.time.Clock; +import java.time.Duration; +import java.time.Instant; +import java.util.Collection; + + +@RequiredArgsConstructor +@Slf4j +public class AdderRebalanceListener implements PollIntervalAwareConsumerRebalanceListener +{ + private final AdderRecordHandler handler; + private final PartitionStatisticsRepository repository; + private final String id; + private final String topic; + private final Clock clock; + private final Duration commitInterval; + private final Consumer consumer; + + private Instant lastCommit = Instant.EPOCH; + + @Override + public void onPartitionsAssigned(Collection partitions) + { + partitions.forEach(tp -> + { + Integer partition = tp.partition(); + Long offset = consumer.position(tp); + log.info("{} - adding partition: {}, offset={}", id, partition, offset); + StateDocument document = + repository + .findById(Integer.toString(partition)) + .orElse(new StateDocument(partition)); + if (document.offset >= 0) + { + // Only seek, if a stored offset was found + // Otherwise: Use initial offset, generated by Kafka + consumer.seek(tp, document.offset); + } + handler.addPartition(partition, document.state); + }); + } + + @Override + public void onPartitionsRevoked(Collection partitions) + { + partitions.forEach(tp -> + { + Integer partition = tp.partition(); + Long newOffset = consumer.position(tp); + log.info( + "{} - removing partition: {}, offset of next message {})", + id, + partition, + newOffset); + repository.save(new StateDocument(partition, handler.removePartition(partition), newOffset)); + }); + } + + + @Override + public void beforeNextPoll() + { + if (lastCommit.plus(commitInterval).isBefore(clock.instant())) + { + log.debug("Storing data and offsets, last commit: {}", lastCommit); + handler.getState().forEach((partiton, sumBusinessLogic) -> repository.save( + new StateDocument( + partiton, + sumBusinessLogic.getState(), + consumer.position(new TopicPartition(topic, partiton))))); + lastCommit = clock.instant(); + } + } +}