use of net.dempsy.transport.NodeAddress in project Dempsy by Dempsy.
the class OutgoingDispatcher method dispatch.
@Override
public void dispatch(final KeyedMessageWithType message) {
boolean messageSentSomewhere = false;
try {
ApplicationState tmp = outbounds.get();
// if we're in the midst of an update then we really want to wait for the new state.
while (tmp == null) {
if (!isRunning.get()) {
LOGGER.debug("Router dispatch called while stopped.");
return;
}
if (// however, if we never were ready then we're not in the midst
!isReady.get())
// of an update.
throw new IllegalStateException("Dispatch used before Router is ready.");
// let the other threads do their thing. Maybe we'll be updated sooner.
Thread.yield();
// are we updated yet?
tmp = outbounds.get();
}
final ApplicationState cur = tmp;
final Map<String, RoutingStrategy.Router[]> outboundsByMessageType = cur.outboundsByMessageType;
// =================================================================================
// For each message type, determine the set of Routers. The goal of this loop is to set
// 'containerByNodeAddress'
final Map<NodeAddress, ContainerAddress> containerByNodeAddress = new HashMap<>();
for (final String mt : message.messageTypes) {
final RoutingStrategy.Router[] routers = outboundsByMessageType.get(mt);
if (routers == null)
LOGGER.trace("No cluster that handles messages of type {}", mt);
else {
// the set of ContainerAddresses that this message will be sent to.
for (int i = 0; i < routers.length; i++) {
final ContainerAddress ca = routers[i].selectDestinationForMessage(message);
// it's possible 'ca' is null when we don't know where to send the message.
if (ca == null)
LOGGER.debug("No way to send the message {} to specific cluster for the time being", message.message);
else {
// When the message will be sent to 2 different clusters, but both clusters
// are hosted in the same node, then we send 1 message to 1 ContainerAddress
// where the 'clusters' field contains both container ids.
final ContainerAddress already = containerByNodeAddress.get(ca.node);
if (already != null) {
final int[] ia = new int[already.clusters.length + ca.clusters.length];
System.arraycopy(already.clusters, 0, ia, 0, already.clusters.length);
System.arraycopy(ca.clusters, 0, ia, already.clusters.length, ca.clusters.length);
containerByNodeAddress.put(ca.node, new ContainerAddress(ca.node, ia));
} else
containerByNodeAddress.put(ca.node, ca);
}
}
}
}
for (final Map.Entry<NodeAddress, ContainerAddress> e : containerByNodeAddress.entrySet()) {
final NodeAddress curNode = e.getKey();
final ContainerAddress curAddr = e.getValue();
final RoutedMessage toSend = new RoutedMessage(curAddr.clusters, message.key, message.message);
if (curNode.equals(thisNode)) {
// this shouldn't count since Router is an OUTGOING class
nodeReciever.feedbackLoop(toSend, false);
messageSentSomewhere = true;
} else {
final Sender sender = cur.getSender(curNode);
if (sender == null) {
// router update is probably behind the routing strategy update
if (isRunning.get())
LOGGER.error("Couldn't send message to " + curNode + " from " + thisNodeId + " because there's no " + Sender.class.getSimpleName());
} else {
sender.send(toSend);
messageSentSomewhere = true;
}
}
}
if (containerByNodeAddress.size() == 0) {
if (LOGGER.isTraceEnabled())
LOGGER.trace("There appears to be no valid destination addresses for the message {}", SafeString.objectDescription(message.message));
}
} finally {
if (!messageSentSomewhere)
statsCollector.messageNotSent();
}
}
use of net.dempsy.transport.NodeAddress in project Dempsy by Dempsy.
the class ApplicationState method apply.
public ApplicationState apply(final ApplicationState.Update update, final TransportManager tmanager, final NodeStatsCollector statsCollector, final RoutingStrategyManager manager) {
// apply toDelete first.
final Set<NodeAddress> toDelete = update.toDelete;
if (toDelete.size() > 0) {
// just clear all senders.
for (final NodeAddress a : toDelete) {
final Sender s = senders.get(a);
if (s != null)
s.stop();
}
}
final Map<NodeAddress, NodeInformation> newCurrent = new HashMap<>();
// the one's to carry over.
final Set<NodeInformation> leaveAlone = update.leaveAlone;
for (final NodeInformation cur : leaveAlone) {
newCurrent.put(cur.nodeAddress, cur);
}
// add new senders
final Set<NodeInformation> toAdd = update.toAdd;
for (final NodeInformation cur : toAdd) {
newCurrent.put(cur.nodeAddress, cur);
}
// now flush out the remaining caches.
// collapse all clusterInfos
final Set<ClusterInformation> allCis = new HashSet<>();
newCurrent.values().forEach(ni -> allCis.addAll(ni.clusterInfoByClusterId.values()));
final Map<String, RoutingStrategy.Router> newOutboundByClusterName = new HashMap<>();
final Map<String, Set<String>> cnByType = new HashMap<>();
final Set<String> knownClusterOutbounds = new HashSet<>(outboundByClusterName_.keySet());
for (final ClusterInformation ci : allCis) {
final String clusterName = ci.clusterId.clusterName;
final RoutingStrategy.Router ob = outboundByClusterName_.get(clusterName);
knownClusterOutbounds.remove(clusterName);
if (ob != null)
newOutboundByClusterName.put(clusterName, ob);
else {
final RoutingStrategy.Factory obfactory = manager.getAssociatedInstance(ci.routingStrategyTypeId);
final RoutingStrategy.Router nob = obfactory.getStrategy(ci.clusterId);
newOutboundByClusterName.put(clusterName, nob);
}
// add all of the message types handled.
ci.messageTypesHandled.forEach(mt -> {
Set<String> entry = cnByType.get(mt);
if (entry == null) {
entry = new HashSet<>();
cnByType.put(mt, entry);
}
entry.add(clusterName);
});
}
return new ApplicationState(newOutboundByClusterName, cnByType, newCurrent, tmanager, thisNode);
}
use of net.dempsy.transport.NodeAddress in project Dempsy by Dempsy.
the class TestGroupRoutingStrategy method testInboundDoubleHappyPathRegister.
@Test
public void testInboundDoubleHappyPathRegister() throws Exception {
final int numShardsToExpect = Integer.parseInt(Utils.DEFAULT_TOTAL_SHARDS);
final String groupName = "testInboundDoubleHappyPathRegister";
try (final RoutingStrategy.Inbound ib1 = new RoutingInboundManager().getAssociatedInstance(ClusterGroupInbound.class.getPackage().getName() + ":" + groupName);
final RoutingStrategy.Inbound ib2 = new RoutingInboundManager().getAssociatedInstance(ClusterGroupInbound.class.getPackage().getName() + ":" + groupName)) {
final ClusterId clusterId = new ClusterId("test", "test");
final NodeAddress node1Addr = new DummyNodeAddress("node1");
final GroupDetails gd1 = new GroupDetails(groupName, node1Addr);
final ContainerAddress node1Ca = new ContainerAddress(node1Addr, 0);
final Utils<GroupDetails> utils = new Utils<>(infra, groupName, gd1);
ib1.setContainerDetails(clusterId, node1Ca, (l, m) -> {
});
ib1.start(infra);
final NodeAddress node2Addr = new DummyNodeAddress("node2");
final ContainerAddress node2Ca = new ContainerAddress(node2Addr, 0);
ib2.setContainerDetails(clusterId, node2Ca, (l, m) -> {
});
try (final ClusterInfoSession session2 = sessFact.createSession()) {
ib2.start(new TestInfrastructure(session2, infra.getScheduler()));
assertTrue(waitForShards(session, utils, numShardsToExpect));
// if this worked right then numShardsToExpect/2 should be owned by each ... eventually.
checkForShardDistribution(session, utils, numShardsToExpect, 2);
// disrupt the session. This should cause a reshuffle but not fail
disruptor.accept(session2);
// everything should settle back
checkForShardDistribution(session, utils, numShardsToExpect, 2);
// now kill the second session.
// this will disconnect the second Inbound and so the first should take over
session2.close();
// see if we now have 1 session and it has all shards
checkForShardDistribution(session, utils, numShardsToExpect, 1);
}
}
}
use of net.dempsy.transport.NodeAddress in project Dempsy by Dempsy.
the class TestGroupRoutingStrategy method testInboundWithOutbound.
@Test
public void testInboundWithOutbound() throws Exception {
final int numShardsToExpect = Integer.parseInt(Utils.DEFAULT_TOTAL_SHARDS);
final String groupName = "testInboundWithOutbound";
final Manager<RoutingStrategy.Inbound> manager = new RoutingInboundManager();
try (final RoutingStrategy.Inbound ib = manager.getAssociatedInstance(ClusterGroupInbound.class.getPackage().getName() + ":" + groupName)) {
final ClusterId cid = setTestName("testInboundWithOutbound");
final NodeAddress na = new DummyNodeAddress("here");
final ContainerAddress oca = new ContainerAddress(na, 0);
final Infrastructure infra = makeInfra(session, sched);
final GroupDetails ogd = new GroupDetails(groupName, na);
final Map<String, ContainerAddress> ocaiByCluster = new HashMap<>();
ocaiByCluster.put(cid.clusterName, oca);
ogd.fillout(ocaiByCluster);
final Utils<GroupDetails> msutils = new Utils<>(infra, groupName, ogd);
ib.setContainerDetails(cid, oca, (l, m) -> {
});
ib.start(infra);
checkForShardDistribution(session, msutils, numShardsToExpect, 1);
try (final ClusterInfoSession ses2 = sessFact.createSession();
final RoutingStrategyManager obman = chain(new RoutingStrategyManager(), o -> o.start(makeInfra(ses2, sched)));
final RoutingStrategy.Factory obf = obman.getAssociatedInstance(ClusterGroupInbound.class.getPackage().getName() + ":" + groupName)) {
obf.start(makeInfra(ses2, sched));
assertTrue(poll(o -> obf.isReady()));
final RoutingStrategy.Router ob = obf.getStrategy(cid);
assertTrue(poll(o -> obf.isReady()));
final KeyedMessageWithType km = new KeyedMessageWithType(new Object(), new Object(), "");
assertTrue(poll(o -> ob.selectDestinationForMessage(km) != null));
final ContainerAddress ca = ob.selectDestinationForMessage(km);
assertNotNull(ca);
assertEquals("here", ((DummyNodeAddress) ca.node).name);
// now disrupt the session
session.close();
// the destination should clear until a new one runs
// NO: destination will not necessarily clear.
// poll(o -> ob.selectDestinationForMessage(km) == null);
final ContainerAddress nca = new ContainerAddress(new DummyNodeAddress("here-again"), 0);
ogd.fillout(ocaiByCluster);
try (ClusterInfoSession ses3 = sessFact.createSession();
RoutingStrategy.Inbound ib2 = new RoutingInboundManager().getAssociatedInstance(ClusterGroupInbound.class.getPackage().getName() + ":" + groupName)) {
ib2.setContainerDetails(cid, nca, (l, m) -> {
});
ib2.start(makeInfra(ses3, sched));
assertTrue(poll(o -> ob.selectDestinationForMessage(km) != null));
}
}
}
}
use of net.dempsy.transport.NodeAddress in project Dempsy by Dempsy.
the class ApplicationState method apply.
public ApplicationState apply(final ApplicationState.Update update, final TransportManager tmanager, final NodeStatsCollector statsCollector, final RoutingStrategyManager manager, final String thisNodeId) {
// apply toDelete first.
final Set<NodeAddress> toDelete = update.toDelete;
final boolean infoEnabled = LOGGER_SESSION.isInfoEnabled();
if (toDelete.size() > 0) {
// just clear all senders.
if (infoEnabled)
LOGGER_SESSION.info("[{}] Applying update to topology resulting in removing several destinations:", thisNodeId);
for (final NodeAddress a : toDelete) {
final Sender s = senders.remove(a);
if (infoEnabled)
LOGGER_SESSION.info("[{}] removing sender ({}) to {}", thisNodeId, s, a);
if (s != null)
s.stop();
}
}
final Map<NodeAddress, NodeInformation> newCurrent = new HashMap<>();
// the one's to carry over.
final Set<NodeInformation> leaveAlone = update.leaveAlone;
if (leaveAlone.size() > 0)
if (infoEnabled)
LOGGER_SESSION.info("[{}] Applying update to topology resulting in leaving several destinations:", thisNodeId);
for (final NodeInformation cur : leaveAlone) {
if (infoEnabled)
LOGGER_SESSION.info("[{}] leaving alone : {}", thisNodeId, Optional.ofNullable(cur).map(ni -> ni.nodeAddress).orElse(null));
newCurrent.put(cur.nodeAddress, cur);
}
// add new senders
final Set<NodeInformation> toAdd = update.toAdd;
if (toAdd.size() > 0)
if (infoEnabled)
LOGGER_SESSION.info("[{}] Applying update to topology resulting in adding several destinations:", thisNodeId);
for (final NodeInformation cur : toAdd) {
if (infoEnabled)
LOGGER_SESSION.info("[{}] adding : {}", thisNodeId, Optional.ofNullable(cur).map(ni -> ni.nodeAddress).orElse(null));
newCurrent.put(cur.nodeAddress, cur);
}
// now flush out the remaining caches.
// collapse all clusterInfos
final Set<ClusterInformation> allCis = new HashSet<>();
newCurrent.values().forEach(ni -> allCis.addAll(ni.clusterInfoByClusterId.values()));
final Map<String, RoutingStrategy.Router> newOutboundByClusterName = new HashMap<>();
final Map<String, Set<String>> cnByType = new HashMap<>();
final Set<String> knownClusterOutbounds = new HashSet<>(outboundByClusterName_.keySet());
for (final ClusterInformation ci : allCis) {
final String clusterName = ci.clusterId.clusterName;
final RoutingStrategy.Router ob = outboundByClusterName_.get(clusterName);
knownClusterOutbounds.remove(clusterName);
if (ob != null)
newOutboundByClusterName.put(clusterName, ob);
else {
final RoutingStrategy.Factory obfactory = manager.getAssociatedInstance(ci.routingStrategyTypeId);
final RoutingStrategy.Router nob = obfactory.getStrategy(ci.clusterId);
newOutboundByClusterName.put(clusterName, nob);
}
// add all of the message types handled.
ci.messageTypesHandled.forEach(mt -> {
Set<String> entry = cnByType.get(mt);
if (entry == null) {
entry = new HashSet<>();
cnByType.put(mt, entry);
}
entry.add(clusterName);
});
}
return new ApplicationState(newOutboundByClusterName, cnByType, newCurrent, tmanager, thisNode, senders);
}
Aggregations