org.jgroups.protocols.tom.TOA Maven / Gradle / Ivy
Go to download
This artifact provides a single jar that contains all classes required to use remote Jakarta Enterprise Beans and Jakarta Messaging, including
all dependencies. It is intended for use by those not using maven, maven users should just import the Jakarta Enterprise Beans and
Jakarta Messaging BOM's instead (shaded JAR's cause lots of problems with maven, as it is very easy to inadvertently end up
with different versions on classes on the class path).
package org.jgroups.protocols.tom;
import org.jgroups.*;
import org.jgroups.annotations.MBean;
import org.jgroups.annotations.ManagedAttribute;
import org.jgroups.annotations.ManagedOperation;
import org.jgroups.stack.Protocol;
import java.util.ArrayList;
import java.util.Collection;
import java.util.List;
import java.util.Set;
import java.util.concurrent.atomic.AtomicLong;
/**
* Total Order Anycast with three communication steps (based on Skeen's Algorithm). Establishes total order for a
* message sent to a subset of the cluster members (an anycast). Example: send a totally ordered message to {D,E}
* out of a membership of {A,B,C,D,E,F}.
* Skeen's algorithm uses consensus among the anycast target members to find the currently highest
* sequence number (seqno) and delivers the message according to the order established by the seqnos.
*
* @author Pedro Ruivo
* @since 3.1
*/
@MBean(description = "Implementation of Total Order Anycast based on Skeen's Algorithm")
public class TOA extends Protocol implements DeliveryProtocol {
//managers
private DeliveryManagerImpl deliverManager;
private SenderManager senderManager;
// threads
private final DeliveryThread deliverThread = new DeliveryThread(this);
//local address
private Address localAddress;
//sequence numbers, messages ids and lock
private final AtomicLong messageIdCounter = new AtomicLong(0);
//stats: profiling information
private final StatsCollector statsCollector = new StatsCollector();
private volatile View currentView;
public TOA() {
}
@Override
public void start() throws Exception {
deliverManager = new DeliveryManagerImpl();
senderManager = new SenderManager();
deliverThread.start(deliverManager);
statsCollector.setStatsEnabled(statsEnabled());
}
@Override
public void stop() {
deliverThread.interrupt();
}
@Override
public Object down(Event evt) {
switch (evt.getType()) {
case Event.MSG:
handleDownMessage(evt);
return null;
case Event.SET_LOCAL_ADDRESS:
this.localAddress = (Address) evt.getArg();
this.deliverThread.setLocalAddress(localAddress.toString());
break;
case Event.VIEW_CHANGE:
handleViewChange((View) evt.getArg());
break;
default:
break;
}
return down_prot.down(evt);
}
@Override
public Object up(Event evt) {
switch (evt.getType()) {
case Event.MSG:
Message message = (Message) evt.getArg();
ToaHeader header = (ToaHeader) message.getHeader(this.id);
if (header == null) {
break;
}
switch (header.getType()) {
case ToaHeader.DATA_MESSAGE:
handleDataMessage(message, header);
break;
case ToaHeader.PROPOSE_MESSAGE:
handleSequenceNumberPropose(message.getSrc(), header);
break;
case ToaHeader.FINAL_MESSAGE:
handleFinalSequenceNumber(header);
break;
case ToaHeader.SINGLE_DESTINATION_MESSAGE:
if (log.isTraceEnabled()) {
log.trace("Received message " + message + " with SINGLE_DESTINATION header. delivering...");
}
deliverManager.deliverSingleDestinationMessage(message, header.getMessageID());
break;
default:
throw new IllegalStateException("Unknown header type received " + header);
}
return null;
case Event.VIEW_CHANGE:
handleViewChange((View) evt.getArg());
break;
case Event.SET_LOCAL_ADDRESS:
this.localAddress = (Address) evt.getArg();
this.deliverThread.setLocalAddress(localAddress.toString());
break;
default:
break;
}
return up_prot.up(evt);
}
@Override
public void deliver(Message message) {
message.setDest(localAddress);
if (log.isDebugEnabled()) {
log.debug("Deliver message " + message + "(" + message.getHeader(id) + ") in total order");
}
up_prot.up(new Event(Event.MSG, message));
statsCollector.incrementMessageDeliver();
}
private void handleViewChange(View view) {
if (log.isTraceEnabled()) {
log.trace("Handle view " + view);
}
View oldView = currentView;
currentView = view;
//basis behavior: drop leavers message (as senders)
List leavers = View.leftMembers(oldView, view);
deliverManager.removeLeavers(leavers);
//basis behavior: avoid waiting for the acks
Collection pendingSentMessages = senderManager.getPendingMessageIDs();
for (MessageID messageID : pendingSentMessages) {
long finalSequenceNumber = senderManager.removeLeavers(messageID, leavers);
if (finalSequenceNumber != SenderManager.NOT_READY) {
ToaHeader finalHeader = ToaHeader.newFinalMessageHeader(messageID, finalSequenceNumber);
Message finalMessage = new Message().src(localAddress).putHeader(this.id, finalHeader)
.setFlag(Message.Flag.OOB, Message.Flag.INTERNAL, Message.Flag.DONT_BUNDLE);
Set destinations = senderManager.getDestination(messageID);
if (destinations.contains(localAddress)) {
destinations.remove(localAddress);
}
if (log.isTraceEnabled()) {
log.trace("Message " + messageID + " is ready to be deliver. Final sequencer number is " +
finalSequenceNumber);
}
send(destinations, finalMessage, false);
//returns true if we are in destination set
if (senderManager.markSent(messageID)) {
deliverManager.markReadyToDeliver(messageID, finalSequenceNumber);
}
}
}
// TODO: Future work: How to add fault tolerance? (simple and efficient)
}
private void handleDownMessage(Event evt) {
Message message = (Message) evt.getArg();
Address dest = message.getDest();
if (dest != null && dest instanceof AnycastAddress && !message.isFlagSet(Message.Flag.NO_TOTAL_ORDER)) {
//anycast message
sendTotalOrderAnycastMessage(extract((AnycastAddress) dest), message);
} else if (dest != null && dest instanceof AnycastAddress) {
//anycast address with NO_TOTAL_ORDER flag (should no be possible, but...)
send(extract((AnycastAddress) dest), message, true);
} else {
//normal message
down_prot.down(evt);
}
}
private void sendTotalOrderAnycastMessage(List destinations, Message message) {
boolean trace = log.isTraceEnabled();
long startTime = statsCollector.now();
long duration = -1;
final boolean deliverToMySelf = destinations.contains(localAddress);
if (destinations.size() == 1) {
MessageID messageID = generateId();
message.putHeader(id, ToaHeader.createSingleDestinationHeader(messageID));
message.setDest(destinations.get(0));
if (trace) {
log.trace("Sending total order anycast message " + message + " (" + message.getHeader(id) +
") to single destination");
}
if (deliverToMySelf) {
deliverManager.deliverSingleDestinationMessage(message, messageID);
} else {
down_prot.down(new Event(Event.MSG, message));
}
return;
}
try {
final MessageID messageID = generateId();
long sequenceNumber = -1;
ToaHeader header = ToaHeader.newDataMessageHeader(messageID, destinations);
message.putHeader(this.id, header);
if (deliverToMySelf) {
sequenceNumber = deliverManager.addLocalMessageToDeliver(messageID, message, header);
}
if (trace) {
log.trace("Sending total order anycast message " + message + " (" + message.getHeader(id) +
") to " + destinations);
}
senderManager.addNewMessageToSend(messageID, destinations, sequenceNumber, deliverToMySelf);
send(destinations, message, false);
duration = statsCollector.now() - startTime;
} catch (Exception e) {
logException("Exception caught while sending anycast message. Error is " + e.getLocalizedMessage(),
e);
} finally {
statsCollector.addAnycastSentDuration(duration, (destinations.size() - (deliverToMySelf ? 1 : 0)));
}
}
private MessageID generateId() {
return new MessageID(localAddress, messageIdCounter.getAndIncrement());
}
private void send(Collection destinations, Message msg, boolean sendToMyself) {
if (log.isDebugEnabled()) {
log.debug("sending anycast total order message " + msg + " to " + destinations);
}
for (Address address : destinations) {
if (!sendToMyself && address.equals(localAddress)) {
continue;
}
Message cpy = msg.copy();
cpy.setDest(address);
down_prot.down(new Event(Event.MSG, cpy));
}
}
private void handleDataMessage(Message message, ToaHeader header) {
long startTime = statsCollector.now();
long duration = -1;
try {
final MessageID messageID = header.getMessageID();
//create the sequence number and put it in deliver manager
long myProposeSequenceNumber = deliverManager.addRemoteMessageToDeliver(messageID, message,
header.getSequencerNumber());
if (log.isTraceEnabled()) {
log.trace("Received the message with " + header + ". The proposed sequence number is " +
myProposeSequenceNumber);
}
//create a new message and send it back
ToaHeader newHeader = ToaHeader.newProposeMessageHeader(messageID, myProposeSequenceNumber);
Message proposeMessage = new Message().src(localAddress).dest(messageID.getAddress())
.putHeader(this.id, newHeader).setFlag(Message.Flag.OOB, Message.Flag.INTERNAL, Message.Flag.DONT_BUNDLE);
//multicastSenderThread.addUnicastMessage(proposeMessage);
down_prot.down(new Event(Event.MSG, proposeMessage));
duration = statsCollector.now() - startTime;
} catch (Exception e) {
logException("Exception caught while processing the data message " + header.getMessageID(), e);
} finally {
statsCollector.addDataMessageDuration(duration);
}
}
private void handleSequenceNumberPropose(Address from, ToaHeader header) {
long startTime = statsCollector.now();
long duration = -1;
boolean lastProposeReceived = false;
boolean trace = log.isTraceEnabled();
try {
MessageID messageID = header.getMessageID();
if (trace) {
log.trace("Received the proposed sequence number message with " + header + " from " +
from);
}
deliverManager.updateSequenceNumber(header.getSequencerNumber());
long finalSequenceNumber = senderManager.addPropose(messageID, from,
header.getSequencerNumber());
if (finalSequenceNumber != SenderManager.NOT_READY) {
lastProposeReceived = true;
ToaHeader finalHeader = ToaHeader.newFinalMessageHeader(messageID, finalSequenceNumber);
Message finalMessage = new Message().src(localAddress).putHeader(this.id, finalHeader)
.setFlag(Message.Flag.OOB, Message.Flag.INTERNAL, Message.Flag.DONT_BUNDLE);
Set destinations = senderManager.getDestination(messageID);
if (destinations.contains(localAddress)) {
destinations.remove(localAddress);
}
if (trace) {
log.trace("Message " + messageID + " is ready to be deliver. Final sequencer number is " +
finalSequenceNumber);
}
send(destinations, finalMessage, false);
//returns true if we are in destination set
if (senderManager.markSent(messageID)) {
deliverManager.markReadyToDeliver(messageID, finalSequenceNumber);
}
}
duration = statsCollector.now() - startTime;
} catch (Exception e) {
logException("Exception caught while processing the propose sequence number for " + header.getMessageID(), e);
} finally {
statsCollector.addProposeSequenceNumberDuration(duration, lastProposeReceived);
}
}
private void handleFinalSequenceNumber(ToaHeader header) {
long startTime = statsCollector.now();
long duration = -1;
try {
MessageID messageID = header.getMessageID();
if (log.isTraceEnabled()) {
log.trace("Received the final sequence number message with " + header);
}
deliverManager.markReadyToDeliver(messageID, header.getSequencerNumber());
duration = statsCollector.now() - startTime;
} catch (Exception e) {
logException("Exception caught while processing the final sequence number for " + header.getMessageID(), e);
} finally {
statsCollector.addFinalSequenceNumberDuration(duration);
}
}
private void logException(String msg, Exception e) {
if (log.isDebugEnabled()) {
log.debug(msg, e);
} else if (log.isWarnEnabled()) {
log.warn(msg + ". Error is " + e.getLocalizedMessage());
}
}
private List extract(AnycastAddress anycastAddress) {
Collection addresses = anycastAddress.getAddresses();
if (addresses == null) {
return new ArrayList<>(currentView.getMembers());
} else {
return new ArrayList<>(addresses);
}
}
@ManagedOperation
public String getMessageList() {
return deliverManager.getMessageSet().toString();
}
@Override
public void enableStats(boolean flag) {
super.enableStats(flag);
statsCollector.setStatsEnabled(flag);
}
@Override
public void resetStats() {
super.resetStats();
statsCollector.clearStats();
}
@ManagedAttribute(description = "The average duration (in milliseconds) in processing and sending the anycast " +
"message to all the recipients", writable = false)
public double getAvgToaSendDuration() {
return statsCollector.getAvgAnycastSentDuration();
}
@ManagedAttribute(description = "The average duration (in milliseconds) in processing a data message received",
writable = false)
public double getAvgDataMessageReceivedDuration() {
return statsCollector.getAvgDataMessageReceivedDuration();
}
@ManagedAttribute(description = "The average duration (in milliseconds) in processing a propose message received" +
"(not the last one", writable = false)
public double getAvgProposeMessageReceivedDuration() {
return statsCollector.getAvgProposeMesageReceivedDuration();
}
@ManagedAttribute(description = "The average duration (in milliseconds) in processing the last propose message " +
"received. This last propose message will originate the sending of the final message", writable = false)
public double getAvgLastProposeMessageReceivedDuration() {
return statsCollector.getAvgLastProposeMessageReceivedDuration();
}
@ManagedAttribute(description = "The average duration (in milliseconds) in processing a final message received",
writable = false)
public double getAvgFinalMessageReceivedDuration() {
return statsCollector.getAvgFinalMessageReceivedDuration();
}
@ManagedAttribute(description = "The number of anycast messages sent", writable = false)
public int getNumberOfAnycastMessagesSent() {
return statsCollector.getNumberOfAnycastMessagesSent();
}
@ManagedAttribute(description = "The number of final anycast sent", writable = false)
public int getNumberOfFinalAnycastSent() {
return statsCollector.getNumberOfFinalAnycastsSent();
}
@ManagedAttribute(description = "The number of anycast messages delivered", writable = false)
public int getNumberOfAnycastMessagesDelivered() {
return statsCollector.getAnycastDelivered();
}
@ManagedAttribute(description = "The number of propose messages sent", writable = false)
public int getNumberOfProposeMessageSent() {
return statsCollector.getNumberOfProposeMessagesSent();
}
@ManagedAttribute(description = "The number of final messages delivered", writable = false)
public int getNumberOfFinalMessagesDelivered() {
return statsCollector.getNumberOfFinalMessagesDelivered();
}
@ManagedAttribute(description = "The number of data messages delivered", writable = false)
public int getNumberOfDataMessagesDelivered() {
return statsCollector.getNumberOfProposeMessagesSent();
}
@ManagedAttribute(description = "The number of propose messages received", writable = false)
public int getNumberOfProposeMessageReceived() {
return statsCollector.getNumberOfProposeMessagesReceived();
}
@ManagedAttribute(description = "The average number of unicasts messages created per anycast message",
writable = false)
public double getAvgNumberOfUnicastSentPerAnycast() {
return statsCollector.getAvgNumberOfUnicastSentPerAnycast();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy