All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.cometd.oort.OortMembership Maven / Gradle / Ivy

/*
 * Copyright (c) 2008-2019 the original author or authors.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package org.cometd.oort;

import static org.cometd.oort.Oort.EXT_OORT_ID_FIELD;

import java.io.IOException;
import java.net.URI;
import java.net.URISyntaxException;
import java.util.ArrayList;
import java.util.Collection;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.Set;
import java.util.concurrent.TimeUnit;

import org.cometd.bayeux.Channel;
import org.cometd.bayeux.ChannelId;
import org.cometd.bayeux.Message;
import org.cometd.bayeux.client.ClientSession;
import org.cometd.bayeux.client.ClientSessionChannel;
import org.cometd.bayeux.server.BayeuxServer;
import org.cometd.bayeux.server.ConfigurableServerChannel;
import org.cometd.bayeux.server.ServerChannel;
import org.cometd.bayeux.server.ServerMessage;
import org.cometd.bayeux.server.ServerSession;
import org.eclipse.jetty.util.component.AbstractLifeCycle;
import org.eclipse.jetty.util.component.Dumpable;
import org.eclipse.jetty.util.component.DumpableCollection;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

/**
 * 

The Oort membership protocol is made of 3 steps:

*
    *
  1. The local node sends the {@code /meta/handshake} request to connect to the remote node.
  2. *
  3. The remote node sends the {@code /meta/handshake} reply.
  4. *
  5. The local node sends a "join" message, composed of the channel subscriptions, * of the list of nodes connected to it, and a {@code /service/oort} message.
  6. *
*

Step 3. is necessary to make sure that the remote node has the * confirmation that the {@code /meta/handshake} reply was processed by the * local node.

*/ class OortMembership extends AbstractLifeCycle implements Dumpable { private final Map pendingComets = new HashMap<>(); private final Map clientComets = new HashMap<>(); private final Map serverComets = new HashMap<>(); private final BayeuxServer.Extension oortExtension = new OortExtension(); private final ConfigurableServerChannel.ServerChannelListener joinListener = new JoinListener(); private final Object lock = this; private final Oort oort; private final Logger logger; OortMembership(Oort oort) { this.oort = oort; this.logger = LoggerFactory.getLogger(getClass().getName() + "." + Oort.replacePunctuation(oort.getURL(), '_')); } @Override protected void doStart() throws Exception { BayeuxServer bayeuxServer = oort.getBayeuxServer(); bayeuxServer.addExtension(oortExtension); ServerChannel oortServiceChannel = bayeuxServer.createChannelIfAbsent(Oort.OORT_SERVICE_CHANNEL).getReference(); oortServiceChannel.addListener(joinListener); super.doStart(); } @Override protected void doStop() throws Exception { disconnect(); BayeuxServer bayeuxServer = oort.getBayeuxServer(); ServerChannel channel = bayeuxServer.getChannel(Oort.OORT_SERVICE_CHANNEL); if (channel != null) { channel.removeListener(joinListener); } bayeuxServer.removeExtension(oortExtension); super.doStop(); } private void disconnect() { List comets; synchronized (lock) { comets = new ArrayList<>(pendingComets.values()); pendingComets.clear(); for (ClientCometInfo cometInfo : clientComets.values()) { comets.add(cometInfo.oortComet); } clientComets.clear(); serverComets.clear(); } for (OortComet comet : comets) { comet.disconnect(); } } OortComet observeComet(String cometURL) { return observeComet(cometURL, null); } OortComet deobserveComet(String cometURL) { if (oort.getURL().equals(cometURL)) { return null; } List comets = new ArrayList<>(); synchronized (lock) { OortComet comet = pendingComets.remove(cometURL); if (comet != null) { if (logger.isDebugEnabled()) { logger.debug("Disconnecting pending comet {} with {}", cometURL, comet); } comets.add(comet); } Iterator cometInfos = clientComets.values().iterator(); while (cometInfos.hasNext()) { ClientCometInfo cometInfo = cometInfos.next(); if (cometInfo.matchesURL(cometURL)) { if (logger.isDebugEnabled()) { logger.debug("Disconnecting comet {}", cometInfo); } comet = cometInfo.oortComet; comets.add(comet); cometInfos.remove(); } } } for (OortComet comet : comets) { comet.disconnect(); } return comets.isEmpty() ? null : comets.get(0); } Set getKnownComets() { Set result = new HashSet<>(); synchronized (lock) { for (ClientCometInfo cometInfo : clientComets.values()) { result.add(cometInfo.oortURL); } } return result; } OortComet getComet(String cometURL) { synchronized (lock) { for (ClientCometInfo cometInfo : clientComets.values()) { if (cometInfo.matchesURL(cometURL)) { return cometInfo.oortComet; } } return null; } } OortComet findComet(String cometURL) { synchronized (lock) { OortComet result = pendingComets.get(cometURL); if (result == null) { result = getComet(cometURL); } return result; } } private OortComet observeComet(String cometURL, String oortAliasURL) { try { URI uri = new URI(cometURL); if (uri.getScheme() == null) { throw new IllegalArgumentException("Missing protocol in comet URL " + cometURL); } if (uri.getHost() == null) { throw new IllegalArgumentException("Missing host in comet URL " + cometURL); } } catch (URISyntaxException x) { throw new IllegalArgumentException(x); } if (oort.getURL().equals(cometURL)) { return null; } if (logger.isDebugEnabled()) { logger.debug("Observing comet {}", cometURL); } OortComet oortComet; synchronized (lock) { oortComet = oort.getComet(cometURL); if (oortComet != null) { if (logger.isDebugEnabled()) { logger.debug("Comet {} is already connected with {}", cometURL, oortComet); } return oortComet; } oortComet = pendingComets.get(cometURL); if (oortComet != null) { if (logger.isDebugEnabled()) { logger.debug("Comet {} is already connecting with {}", cometURL, oortComet); } return oortComet; } oortComet = createOortComet(cometURL); } if (logger.isDebugEnabled()) { logger.debug("Connecting to comet {} with {}", cometURL, oortComet); } Map fields = oort.newOortHandshakeFields(cometURL, oortAliasURL); oort.connectComet(oortComet, fields); return oortComet; } OortComet createOortComet(String cometURL) { synchronized (lock) { OortComet oortComet = oort.newOortComet(cometURL); oort.configureOortComet(oortComet); oortComet.getChannel(Channel.META_HANDSHAKE).addListener(new HandshakeListener(cometURL, oortComet)); pendingComets.put(cometURL, oortComet); return oortComet; } } void observeChannels(Set channels) { List oortComets = new ArrayList<>(); synchronized (lock) { for (ClientCometInfo cometInfo : clientComets.values()) { oortComets.add(cometInfo.oortComet); } } for (OortComet oortComet : oortComets) { oortComet.subscribe(channels); } } void deobserveChannel(String channelName) { List oortComets = new ArrayList<>(); synchronized (lock) { for (ClientCometInfo cometInfo : clientComets.values()) { oortComets.add(cometInfo.oortComet); } } for (OortComet oortComet : oortComets) { oortComet.unsubscribe(channelName); } } boolean containsServerSession(ServerSession session) { synchronized (lock) { for (ServerCometInfo cometInfo : serverComets.values()) { if (cometInfo.session.getId().equals(session.getId())) { return true; } } } return false; } boolean isCometConnected(String oortURL) { synchronized (lock) { for (ServerCometInfo serverCometInfo : serverComets.values()) { if (serverCometInfo.oortURL.equals(oortURL)) { return true; } } return false; } } List knownOortIds() { List result; synchronized (lock) { result = new ArrayList<>(clientComets.keySet()); } return result; } @Override public void dump(Appendable out, String indent) throws IOException { Collection pendingComets; Collection clientComets; Collection serverComets; synchronized (lock) { pendingComets = new ArrayList<>(this.pendingComets.values()); clientComets = new ArrayList<>(this.clientComets.values()); serverComets = new ArrayList<>(this.serverComets.values()); } Dumpable.dumpObjects(out, indent, this, new DumpableCollection("pending", pendingComets), new DumpableCollection("clientComets", clientComets), new DumpableCollection("serverComets", serverComets)); } @Override public String toString() { return String.format("%s@%x[%s]", getClass().getSimpleName(), hashCode(), oort.getURL()); } private enum LocalState { HANDSHAKE_SENT, JOIN_SENT } private enum RemoteState { DISCONNECTED, HANDSHAKE_RECEIVED, JOIN_RECEIVED, JOINED } private static abstract class CometInfo { protected final String oortId; protected final String oortURL; protected CometInfo(String oortId, String oortURL) { this.oortId = oortId; this.oortURL = oortURL; } @Override public String toString() { return String.format("%s@%x[%s|%s]", getClass().getSimpleName(), hashCode(), oortId, oortURL); } } private static class ServerCometInfo extends CometInfo { private final ServerSession session; private RemoteState state = RemoteState.DISCONNECTED; private ServerCometInfo(String oortId, String oortURL, ServerSession session) { super(oortId, oortURL); this.session = session; } @Override public String toString() { return String.format("%s[%s,%s]", super.toString(), state, session); } } private static class ClientCometInfo extends CometInfo { private final OortComet oortComet; private Set aliases; private LocalState state = LocalState.HANDSHAKE_SENT; private ClientCometInfo(String oortId, String oortURL, OortComet oortComet) { super(oortId, oortURL); this.oortComet = oortComet; } private void addAliasURL(String url) { synchronized (this) { if (aliases == null) { aliases = new HashSet<>(); } aliases.add(url); } } private boolean matchesURL(String url) { if (oortURL.equals(url)) { return true; } synchronized (this) { return aliases != null && aliases.contains(url); } } @Override public String toString() { return String.format("%s[%s,%s,aliases=%s]", super.toString(), state, oortComet, Objects.toString(aliases, "[]")); } } private class HandshakeListener implements ClientSessionChannel.MessageListener { private final String cometURL; private final OortComet oortComet; private HandshakeListener(String cometURL, OortComet oortComet) { this.cometURL = cometURL; this.oortComet = oortComet; } @Override public void onMessage(ClientSessionChannel channel, Message message) { if (logger.isDebugEnabled()) { logger.debug("Received handshake reply message {}", message); } Map ext = message.getExt(); if (ext == null) { return; } Object oortExtObject = ext.get(Oort.EXT_OORT_FIELD); if (!(oortExtObject instanceof Map)) { return; } @SuppressWarnings("unchecked") Map oortExt = (Map)oortExtObject; String oortId = (String)oortExt.get(EXT_OORT_ID_FIELD); String oortURL = (String)oortExt.get(Oort.EXT_OORT_URL_FIELD); List staleClientCometInfos = new ArrayList<>(); ClientCometInfo clientCometInfo; ServerCometInfo serverCometInfo; boolean notify = false; synchronized (lock) { // Remove possibly stale information, e.g. when the other node restarted // we will have a stale oortId for the same oortURL we are processing now. Iterator iterator = clientComets.values().iterator(); while (iterator.hasNext()) { clientCometInfo = iterator.next(); if (clientCometInfo.matchesURL(cometURL) || clientCometInfo.matchesURL(oortURL)) { iterator.remove(); staleClientCometInfos.add(clientCometInfo); if (logger.isDebugEnabled()) { logger.debug("Unregistered client comet {}", clientCometInfo); } } } clientCometInfo = new ClientCometInfo(oortId, oortURL, oortComet); serverCometInfo = serverComets.get(oortId); if (logger.isDebugEnabled()) { logger.debug("Current server {}", serverCometInfo); } if (message.isSuccessful()) { pendingComets.remove(cometURL); clientComets.put(oortId, clientCometInfo); if (logger.isDebugEnabled()) { logger.debug("Registered client comet {}", clientCometInfo); } if (!cometURL.equals(oortURL)) { clientCometInfo.addAliasURL(cometURL); if (logger.isDebugEnabled()) { logger.debug("Added comet alias {}", clientCometInfo); } } if (serverCometInfo != null) { clientCometInfo.state = LocalState.JOIN_SENT; notify = serverCometInfo.state == RemoteState.JOIN_RECEIVED; if (notify) { serverCometInfo.state = RemoteState.JOINED; } } } } for (ClientCometInfo info : staleClientCometInfos) { OortComet comet = info.oortComet; if (comet != oortComet) { if (logger.isDebugEnabled()) { logger.debug("Disconnecting stale client comet {}", info); } comet.disconnect(); } } if (message.isSuccessful()) { if (serverCometInfo != null) { oortComet.open(new JoinCallback(oortComet)); if (notify) { oort.notifyCometJoined(oortId, oortURL); } else { if (logger.isDebugEnabled()) { logger.debug("Skipping local join event: {}|{}", oortId, oortURL); } } } } else { // Handshake failed, we will retry it. if (logger.isDebugEnabled()) { logger.debug("Handshake failed to comet {}, message {}", cometURL, message); } } } } private class OortExtension implements BayeuxServer.Extension { @Override public boolean sendMeta(ServerSession session, ServerMessage.Mutable reply) { if (!Channel.META_HANDSHAKE.equals(reply.getChannel())) { return true; } // Skip local sessions. if (session == null || session.isLocalSession()) { return true; } // Must be an Oort handshake. Map messageExt = reply.getAssociated().getExt(); if (messageExt == null) { return true; } Object messageOortExtObject = messageExt.get(Oort.EXT_OORT_FIELD); if (!(messageOortExtObject instanceof Map)) { return true; } // Add the extension information even in case we will disconnect. // The presence of the extension information will inform the client // that the connection "succeeded" from the Oort point of view, but // we add the extension information to drop it if it already exists. Map replyExt = reply.getExt(true); Map replyOortExt = new HashMap<>(2); replyExt.put(Oort.EXT_OORT_FIELD, replyOortExt); replyOortExt.put(Oort.EXT_OORT_URL_FIELD, oort.getURL()); replyOortExt.put(EXT_OORT_ID_FIELD, oort.getId()); // Process only successful responses. if (!reply.isSuccessful()) { return true; } @SuppressWarnings("unchecked") Map messageOortExt = (Map)messageOortExtObject; String remoteOortURL = (String)messageOortExt.get(Oort.EXT_OORT_URL_FIELD); String cometURL = (String)messageOortExt.get(Oort.EXT_COMET_URL_FIELD); String remoteOortId = (String)messageOortExt.get(EXT_OORT_ID_FIELD); session.setAttribute(Oort.COMET_URL_ATTRIBUTE, remoteOortURL); if (oort.getId().equals(remoteOortId)) { // Connecting to myself: disconnect. if (logger.isDebugEnabled()) { logger.debug("Detected self connect from {} to {}, disconnecting", remoteOortURL, cometURL); } disconnect(session, reply); } else { boolean sendJoin = false; ClientCometInfo clientCometInfo; ServerCometInfo existingServerCometInfo; ServerCometInfo serverCometInfo = new ServerCometInfo(remoteOortId, remoteOortURL, session); synchronized (lock) { clientCometInfo = clientComets.get(remoteOortId); if (logger.isDebugEnabled()) { logger.debug("Current client {}", clientCometInfo); } existingServerCometInfo = serverComets.put(remoteOortId, serverCometInfo); if (existingServerCometInfo != null) { serverCometInfo.state = existingServerCometInfo.state; } else { serverCometInfo.state = RemoteState.HANDSHAKE_RECEIVED; } if (logger.isDebugEnabled()) { logger.debug("Registered server {}", serverCometInfo); } if (clientCometInfo != null) { sendJoin = clientCometInfo.state == LocalState.HANDSHAKE_SENT; if (sendJoin) { clientCometInfo.state = LocalState.JOIN_SENT; } } } if (existingServerCometInfo != null) { if (logger.isDebugEnabled()) { logger.debug("Server already known, disconnecting {}", existingServerCometInfo); } // Since it's only a session replace, there is no need to emit a "comet left" // event, but we want to send a /meta/disconnect to the other node for the // previous session. We want to do it from here because the RemoveListener // won't find the previous session, as it has been replaced in the Map. existingServerCometInfo.session.disconnect(); } // Be notified when the remote comet stops. session.addListener(new OortCometDisconnectListener()); // Prevent loops in sending/receiving messages. session.addListener(new OortCometLoopListener()); if (clientCometInfo != null) { if (sendJoin) { OortComet oortComet = clientCometInfo.oortComet; oortComet.open(new JoinCallback(oortComet)); } else { if (logger.isDebugEnabled()) { logger.debug("Client already joined {}", clientCometInfo); } } } String cometAliasURL = (String)messageOortExt.get(Oort.EXT_OORT_ALIAS_URL_FIELD); if (cometAliasURL != null && oort.findComet(cometAliasURL) != null) { // We are connecting to a comet that it is connecting back to us // so there is no need to connect back again (just to be disconnected) if (logger.isDebugEnabled()) { logger.debug("Comet {} exists with alias {}, avoiding to establish connection", remoteOortURL, cometAliasURL); } } else { if (logger.isDebugEnabled()) { logger.debug("Comet {} is unknown, establishing connection", remoteOortURL); } observeComet(remoteOortURL, cometURL); } } return true; } private void disconnect(ServerSession session, ServerMessage.Mutable message) { oort.getBayeuxServer().removeSession(session); message.setSuccessful(false); Map advice = message.getAdvice(true); advice.put(Message.RECONNECT_FIELD, Message.RECONNECT_NONE_VALUE); } } private class OortCometDisconnectListener implements ServerSession.RemoveListener { @Override public void removed(ServerSession session, boolean timeout) { ServerCometInfo serverCometInfo = null; synchronized (lock) { Iterator serverCometInfos = serverComets.values().iterator(); while (serverCometInfos.hasNext()) { ServerCometInfo info = serverCometInfos.next(); if (info.session.getId().equals(session.getId())) { serverCometInfos.remove(); serverCometInfo = info; break; } } } if (serverCometInfo != null) { if (logger.isDebugEnabled()) { logger.debug("Disconnected from server {}", serverCometInfo); } String remoteOortId = serverCometInfo.oortId; String remoteOortURL = serverCometInfo.oortURL; if (!timeout) { OortComet oortComet; synchronized (lock) { oortComet = pendingComets.remove(remoteOortURL); if (oortComet == null) { ClientCometInfo clientCometInfo = clientComets.remove(remoteOortId); if (clientCometInfo != null) { oortComet = clientCometInfo.oortComet; } } } if (oortComet != null) { if (logger.isDebugEnabled()) { logger.debug("Disconnecting from comet {} with {}", remoteOortURL, oortComet); } oortComet.disconnect(); } } // Do not notify if we are stopping. if (isRunning()) { oort.notifyCometLeft(remoteOortId, remoteOortURL); } } } } private class OortCometLoopListener implements ServerSession.MessageListener { @Override public boolean onMessage(ServerSession session, ServerSession sender, ServerMessage message) { // Prevent loops by not delivering a message from self or Oort session to remote Oort comets. if (ChannelId.isBroadcast(message.getChannel()) && sender != null && (sender.getId().equals(session.getId()) || oort.isOort(sender))) { if (logger.isDebugEnabled()) { logger.debug("Blocked {} from {} to {}", message, sender, session); } return false; } if (logger.isDebugEnabled()) { logger.debug("Allowed {} from {} to {}", message, sender, session); } return true; } } private class JoinListener implements ServerChannel.MessageListener { @Override public boolean onMessage(ServerSession from, ServerChannel channel, ServerMessage.Mutable message) { if (logger.isDebugEnabled()) { logger.debug("Received join message {}", message); } Map data = message.getDataAsMap(); String remoteOortId = (String)data.get(Oort.EXT_OORT_ID_FIELD); String remoteOortURL = (String)data.get(Oort.EXT_OORT_URL_FIELD); if (remoteOortURL != null && remoteOortId != null) { boolean notify = false; Set staleComets = null; synchronized (lock) { ClientCometInfo clientCometInfo = clientComets.get(remoteOortId); if (logger.isDebugEnabled()) { logger.debug("Current client {}", clientCometInfo); } Iterator iterator = serverComets.values().iterator(); while (iterator.hasNext()) { ServerCometInfo serverCometInfo = iterator.next(); if (remoteOortURL.equals(serverCometInfo.oortURL)) { String oortId = serverCometInfo.oortId; if (remoteOortId.equals(oortId)) { boolean handshakeReceived = serverCometInfo.state == RemoteState.HANDSHAKE_RECEIVED; notify = clientCometInfo != null && handshakeReceived; if (logger.isDebugEnabled()) { logger.debug("Current server {}", serverCometInfo); } if (handshakeReceived) { serverCometInfo.state = RemoteState.JOIN_RECEIVED; } if (notify) { serverCometInfo.state = RemoteState.JOINED; } } else { // We found a stale entry for a crashed node. if (logger.isDebugEnabled()) { logger.debug("Stale server {}", serverCometInfo); } iterator.remove(); if (staleComets == null) { staleComets = new HashSet<>(4); } staleComets.add(oortId); } } } } if (staleComets != null) { for (String oortId : staleComets) { oort.notifyCometLeft(oortId, remoteOortURL); } } if (notify) { oort.notifyCometJoined(remoteOortId, remoteOortURL); } else { if (logger.isDebugEnabled()) { logger.debug("Skipping remote join event: {}|{}", remoteOortId, remoteOortURL); } } } return true; } } private class JoinCallback implements ClientSession.MessageListener, Runnable { private final OortComet oortComet; private JoinCallback(OortComet oortComet) { this.oortComet = oortComet; } @Override public void onMessage(Message message) { if (message.isSuccessful()) { if (logger.isDebugEnabled()) { logger.debug("Join message successful {}", message); } } else { if (logger.isDebugEnabled()) { logger.debug("Join message failure, retrying {}", message); } // TODO: hardcoded delay. oort.getScheduler().schedule(this, 1, TimeUnit.SECONDS); } } @Override public void run() { oortComet.open(this); } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy