io.reacted.drivers.channels.chroniclequeue.CQLocalDriver Maven / Gradle / Ivy
/*
* Copyright (c) 2020 , [ [email protected] ]
* All rights reserved.
*
* This source code is licensed under the BSD-style license found in the
* LICENSE file in the root directory of this source tree.
*/
package io.reacted.drivers.channels.chroniclequeue;
import io.reacted.core.config.ChannelId;
import io.reacted.core.drivers.system.LocalDriver;
import io.reacted.core.messages.Message;
import io.reacted.core.messages.reactors.DeliveryStatus;
import io.reacted.core.reactorsystem.ReActorContext;
import io.reacted.core.reactorsystem.ReActorSystem;
import io.reacted.patterns.NonNullByDefault;
import io.reacted.patterns.Try;
import io.reacted.patterns.UnChecked;
import java.util.Objects;
import java.util.Properties;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.CompletionStage;
import javax.annotation.Nullable;
import net.openhft.chronicle.queue.ChronicleQueue;
import net.openhft.chronicle.queue.ExcerptTailer;
import net.openhft.chronicle.queue.RollCycles;
import net.openhft.chronicle.threads.Pauser;
import net.openhft.chronicle.wire.DocumentContext;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
@NonNullByDefault
public class CQLocalDriver extends LocalDriver {
private static final Logger LOGGER = LoggerFactory.getLogger(CQLocalDriver.class);
@Nullable
private ChronicleQueue chronicle;
@Nullable
private ExcerptTailer cqTailer;
public CQLocalDriver(CQDriverConfig driverConfig) {
super(driverConfig);
}
@Override
public void initDriverLoop(ReActorSystem localReActorSystem) {
this.chronicle = ChronicleQueue.singleBuilder(getDriverConfig().getChronicleFilesDir())
.rollCycle(RollCycles.MINUTELY)
.build();
this.cqTailer = chronicle.createTailer().toEnd();
}
@Override
public UnChecked.CheckedRunnable getDriverLoop() {
return () -> chronicleMainLoop(Objects.requireNonNull(cqTailer));
}
@Override
public ChannelId getChannelId() {
return ChannelId.ChannelType.LOCAL_CHRONICLE_QUEUE.forChannelName(getDriverConfig().getChannelName());
}
@Override
public Properties getChannelProperties() { return getDriverConfig().getChannelProperties(); }
@Override
public CompletionStage sendAsyncMessage(ReActorContext destination, Message message) {
if (!message.getDataLink().getAckingPolicy().isAckRequired()) {
return super.sendAsyncMessage(destination, message);
}
CompletionStage pendingAck = newPendingAckTrigger(message.getSequenceNumber());
DeliveryStatus localDeliveryStatus = sendMessage(destination, message);
if (localDeliveryStatus == DeliveryStatus.SENT) {
return pendingAck;
}
removePendingAckTrigger(message.getSequenceNumber());
return DELIVERY_RESULT_CACHE[localDeliveryStatus.ordinal()];
}
@Override
public DeliveryStatus sendMessage(ReActorContext destination, Message message) {
try {
Objects.requireNonNull(chronicle).acquireAppender()
.writeMessage(getDriverConfig().getTopic(), message);
return DeliveryStatus.SENT;
} catch (Exception anyException) {
getLocalReActorSystem().logError("Unable to send message {}", message, anyException);
return DeliveryStatus.NOT_SENT;
}
}
@Override
public CompletionStage> cleanDriverLoop() {
return CompletableFuture.completedFuture(Try.ofRunnable(() -> Objects.requireNonNull(chronicle).close()));
}
private void chronicleMainLoop(ExcerptTailer tailer) {
var waitForNextMsg = Pauser.balanced();
while(!Thread.currentThread().isInterrupted()) {
try(DocumentContext docCtx = tailer.readingDocument()) {
if(docCtx.isPresent()) {
Message newMessage = docCtx.wire().read(getDriverConfig().getTopic())
.object(Message.class);
if (newMessage != null) {
waitForNextMsg.reset();
offerMessage(newMessage);
} else {
waitForNextMsg.pause();
}
}
} catch (Exception anyException) {
LOGGER.error("Unable to decode data", anyException);
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy