X-Git-Url: https://juplo.de/gitweb/?a=blobdiff_plain;f=src%2Fmain%2Fjava%2Fde%2Fjuplo%2Fkafka%2Fchat%2Fbackend%2Fpersistence%2Fkafka%2FKafkaChatHomeService.java;h=38aecd187e246ea906e9bb228b8e69cd00a1214c;hb=13c51b4630177e7f6649500a3d4b876a12509af6;hp=5133d1a68203713f970a5106a73b67de5d1f17e7;hpb=035668bee4f02c4c70f43826026b40f81e3dd672;p=demos%2Fkafka%2Fchat diff --git a/src/main/java/de/juplo/kafka/chat/backend/persistence/kafka/KafkaChatHomeService.java b/src/main/java/de/juplo/kafka/chat/backend/persistence/kafka/KafkaChatHomeService.java index 5133d1a6..38aecd18 100644 --- a/src/main/java/de/juplo/kafka/chat/backend/persistence/kafka/KafkaChatHomeService.java +++ b/src/main/java/de/juplo/kafka/chat/backend/persistence/kafka/KafkaChatHomeService.java @@ -2,227 +2,50 @@ package de.juplo.kafka.chat.backend.persistence.kafka; import de.juplo.kafka.chat.backend.domain.ChatHomeService; import de.juplo.kafka.chat.backend.domain.ChatRoom; -import de.juplo.kafka.chat.backend.domain.Message; import de.juplo.kafka.chat.backend.domain.ShardNotOwnedException; -import de.juplo.kafka.chat.backend.persistence.KafkaLikeShardingStrategy; import lombok.extern.slf4j.Slf4j; -import org.apache.kafka.clients.consumer.Consumer; -import org.apache.kafka.clients.consumer.ConsumerRebalanceListener; -import org.apache.kafka.clients.consumer.ConsumerRecord; -import org.apache.kafka.clients.consumer.ConsumerRecords; -import org.apache.kafka.clients.producer.Producer; -import org.apache.kafka.clients.producer.ProducerRecord; -import org.apache.kafka.common.TopicPartition; import reactor.core.publisher.Flux; import reactor.core.publisher.Mono; -import java.time.*; import java.util.*; -import java.util.concurrent.ExecutorService; @Slf4j -public class KafkaChatHomeService implements ChatHomeService, Runnable, ConsumerRebalanceListener +public class KafkaChatHomeService implements ChatHomeService { - private final ExecutorService executorService; - private final Consumer consumer; - private final Producer producer; - private final String topic; - private final ZoneId zoneId; - // private final long[] offsets; Erst mal immer alles neu einlesen - private final boolean[] isShardOwned; - private final Map[] chatRoomMaps; - private final KafkaLikeShardingStrategy shardingStrategy; + private final ChatMessageChannel chatMessageChanel; - private boolean running; - private volatile boolean loadInProgress; - - public KafkaChatHomeService( - ExecutorService executorService, - Consumer consumer, - Producer producer, - String topic, - ZoneId zoneId, - int numShards) + public KafkaChatHomeService(ChatMessageChannel chatMessageChannel) { log.debug("Creating KafkaChatHomeService"); - this.executorService = executorService; - this.consumer = consumer; - this.producer = producer; - this.topic = topic; - this.zoneId = zoneId; - // this.offsets = new long[numShards]; - // for (int i=0; i< numShards; i++) - // { - // this.offsets[i] = 0l; - // } - this.isShardOwned = new boolean[numShards]; - this.chatRoomMaps = new Map[numShards]; - this.shardingStrategy = new KafkaLikeShardingStrategy(numShards); - } - - - @Override - public void onPartitionsAssigned(Collection partitions) - { - loadInProgress = true; - - consumer.endOffsets(partitions).forEach((topicPartition, currentOffset) -> - { - if (!topicPartition.topic().equals(topic)) - { - log.warn("Ignoring partition from unwanted topic: {}", topicPartition); - return; - } - - int partition = topicPartition.partition(); - long unseenOffset = 0; // offsets[partition]; - - log.info( - "Loading messages from partition {}: start-offset={} -> current-offset={}", - partition, - unseenOffset, - currentOffset); - - // TODO: reuse! Nicht immer alles neu laden, sondern erst ab offsets[partition]! - consumer.seek(topicPartition, unseenOffset); - }); - - consumer.resume(partitions); - } - - @Override - public void onPartitionsRevoked(Collection partitions) - { - partitions.forEach(topicPartition -> - { - if (!topicPartition.topic().equals(topic)) - { - log.warn("Ignoring partition from unwanted topic: {}", topicPartition); - return; - } - - int partition = topicPartition.partition(); - // long unseenOffset = offsets[partition]; TODO: Offset merken...? - }); - log.info("Revoked partitions: {}", partitions); - } - - @Override - public void onPartitionsLost(Collection partitions) - { - // TODO: Muss auf den Verlust anders reagiert werden? - onPartitionsRevoked(partitions); - } - - @Override - public void run() - { - consumer.subscribe(List.of(topic)); - - running = true; - - try - { - while (running) - { - ConsumerRecords records = consumer.poll(Duration.ofMinutes(5)); - log.info("Fetched {} messages", records.count()); - - if (loadInProgress) - { - for (ConsumerRecord record : records) - { - UUID chatRoomId = UUID.fromString(record.key()); - MessageTo messageTo = record.value(); - ChatRoom chatRoom = chatRoomMaps[record.partition()].get(chatRoomId); - Mono result = chatRoom.addMessage( - messageTo.getId(), - messageTo.getUser(), - messageTo.getText()); - result.block(). - } - } - else - { - if (!records.isEmpty()) - { - throw new IllegalStateException("All owned partions should be paused, when no load is in progress!"); - } - } - } - } - } - - Mono sendMessage( - UUID chatRoomId, - Message.MessageKey key, - LocalDateTime timestamp, - String text) - { - int shard = this.shardingStrategy.selectShard(chatRoomId); - TopicPartition tp = new TopicPartition(topic, shard); - ZonedDateTime zdt = ZonedDateTime.of(timestamp, zoneId); - return Mono.create(sink -> - { - ProducerRecord record = - new ProducerRecord<>( - tp.topic(), - tp.partition(), - zdt.toEpochSecond(), - chatRoomId.toString(), - MessageTo.of(key.getUsername(), key.getMessageId(), text)); - - producer.send(record, ((metadata, exception) -> - { - if (metadata != null) - { - // On successful send - Message message = new Message(key, metadata.offset(), timestamp, text); - log.info("Successfully send message {}", message); - sink.success(message); - } - else - { - // On send-failure - log.error( - "Could not send message for chat-room={}, key={}, timestamp={}, text={}: {}", - chatRoomId, - key, - timestamp, - text, - exception); - sink.error(exception); - } - })); - }); + this.chatMessageChanel = chatMessageChannel; } @Override public Mono getChatRoom(int shard, UUID id) { - if (loadInProgress) + if (chatMessageChanel.isLoadInProgress()) { throw new ShardNotOwnedException(shard); } else { - return Mono.justOrEmpty(chatRoomMaps[shard].get(id)); + return chatMessageChanel.getChatRoom(shard, id); } } @Override public Flux getChatRooms(int shard) { - if (loadInProgress) + if (chatMessageChanel.isLoadInProgress()) { throw new ShardNotOwnedException(shard); } else { - return Flux.fromStream(chatRoomMaps[shard].values().stream()); + return chatMessageChanel.getChatRooms(shard); } } }