package de.juplo.kafka.chat.backend.domain;
import lombok.Getter;
-import lombok.RequiredArgsConstructor;
-import lombok.Value;
import lombok.extern.slf4j.Slf4j;
import reactor.core.publisher.Flux;
import reactor.core.publisher.Mono;
import java.util.*;
-@RequiredArgsConstructor
@Slf4j
public class Chatroom
{
private final UUID id;
@Getter
private final String name;
- private final LinkedHashMap<MessageKey, Message> messages = new LinkedHashMap<>();
- private final Sinks.Many<Message> sink = Sinks.many().multicast().onBackpressureBuffer();
+ private final PersistenceStrategy persistence;
+ private final Sinks.Many<Message> sink;
- synchronized public Mono<Message> addMessage(
- Long id,
- LocalDateTime timestamp,
- String user,
- String text)
+ public Chatroom(
+ UUID id,
+ String name,
+ PersistenceStrategy persistence,
+ int bufferSize)
{
- return persistMessage(id, timestamp, user, text)
- .doOnNext(message -> sink.tryEmitNext(message).orThrow());
+ this.id = id;
+ this.name = name;
+ this.persistence = persistence;
+ this.sink = Sinks.many().multicast().onBackpressureBuffer(bufferSize);
}
- private Mono<Message> persistMessage(
+
+ synchronized public Mono<Message> addMessage(
Long id,
LocalDateTime timestamp,
String user,
String text)
{
- Message message = new Message(id, (long)messages.size(), timestamp, user, text);
-
- MessageKey key = new MessageKey(user, id);
- Message existing = messages.get(key);
- if (existing != null)
- {
- log.info("Message with key {} already exists; {}", key, existing);
- if (!message.equals(existing))
- throw new MessageMutationException(message, existing);
- return Mono.empty();
- }
-
- messages.put(key, message);
- return Mono
- .fromSupplier(() -> message)
- .log();
+ return persistence
+ .persistMessage(Message.MessageKey.of(user, id), timestamp, text)
+ .doOnNext(message -> sink.tryEmitNext(message).orThrow());
}
+
public Mono<Message> getMessage(String username, Long messageId)
{
- return Mono.fromSupplier(() ->
- {
- MessageKey key = MessageKey.of(username, messageId);
- return messages.get(key);
- });
+ return persistence.getMessage(Message.MessageKey.of(username, messageId));
}
public Flux<Message> listen()
return sink.asFlux();
}
- public Flux<Message> getMessages(long first, long last)
+ public Flux<Message> getMessages()
{
- return Flux.fromStream(messages
- .values()
- .stream()
- .filter(message ->
- {
- long serial = message.getSerialNumber();
- return serial >= first && serial <= last;
- }));
+ return getMessages(0, Long.MAX_VALUE);
}
-
- @Value(staticConstructor = "of")
- static class MessageKey
+ public Flux<Message> getMessages(long first, long last)
{
- String username;
- Long messageId;
+ return persistence.getMessages(first, last);
}
}