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

org.jgroups.Channel Maven / Gradle / Ivy


package org.jgroups;


import org.jgroups.annotations.MBean;
import org.jgroups.annotations.ManagedAttribute;
import org.jgroups.logging.Log;
import org.jgroups.logging.LogFactory;
import org.jgroups.stack.Protocol;
import org.jgroups.stack.ProtocolStack;
import org.jgroups.util.DefaultSocketFactory;
import org.jgroups.util.SocketFactory;
import org.jgroups.util.Util;

import java.io.Closeable;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.CopyOnWriteArraySet;

/**
 * A channel represents a group communication endpoint (like BSD datagram sockets). A client joins a
 * group by connecting the channel to a group and leaves it by disconnecting. Messages sent over the
 * channel are received by all group members that are connected to the same group (that is, all
 * members that have the same group name).
 * 

* * The FSM for a channel is roughly as follows: a channel is created (unconnected). The * channel is connected to a group (connected). Messages can now be sent and received. The * channel is disconnected from the group (unconnected). The channel could now be connected * to a different group again. The channel is closed (closed). *

* * Only a single sender is allowed to be connected to a channel at a time, but there can be more * than one channel in an application. *

* * Messages can be sent to the group members using the send method and messages can be * received setting a {@link Receiver} in {@link #setReceiver(Receiver)} and implementing the * {@link Receiver#receive(Message)} callback. *

* * A channel instance is created using the public constructor. *

* Various degrees of sophistication in message exchange can be achieved using building blocks on * top of channels; e.g., light-weight groups, synchronous message invocation, or remote method * calls. Channels are on the same abstraction level as sockets, and should really be simple to use. * Higher-level abstractions are all built on top of channels. * * @author Bela Ban * @since 2.0 * @see java.net.DatagramPacket * @see java.net.MulticastSocket * @see JChannel */ @MBean(description="Channel") public abstract class Channel implements Closeable { public enum State { OPEN, // initial state, after channel has been created, or after a disconnect() CONNECTING, // when connect() is called CONNECTED, // after successful connect() CLOSED // after close() has been called } /** The current state of the channel */ protected volatile State state=State.OPEN; protected UpHandler up_handler=null; // when set, all events are passed to it ! protected Set channel_listeners=null; protected Receiver receiver=null; protected SocketFactory socket_factory=new DefaultSocketFactory(); protected final Log log=LogFactory.getLog(getClass()); @ManagedAttribute(description="Whether or not to discard messages sent by this channel",writable=true) protected boolean discard_own_messages=false; @ManagedAttribute(description="The current state") public String getState() {return state.toString();} public abstract ProtocolStack getProtocolStack(); public SocketFactory getSocketFactory() { return socket_factory; } public void setSocketFactory(SocketFactory factory) { socket_factory=factory; ProtocolStack stack=getProtocolStack(); Protocol prot=stack != null? stack.getTopProtocol() : null; if(prot != null) prot.setSocketFactory(factory); } /** * Connects the channel to a group. The client is now able to receive group messages, views and * to send messages to (all or single) group members. This is a null operation if already * connected. *

* * All channels with the same name form a group, that means all messages sent to the group will * be received by all channels connected to the same cluster name. *

* * @param cluster_name * The name of the channel to connect to. * @exception Exception * The protocol stack cannot be started * @exception IllegalStateException * The channel is closed */ abstract public void connect(String cluster_name) throws Exception; /** * Connects this channel to a group and gets a state from a specified state provider. *

* * This method essentially invokes * connect and getState methods successively. * If FLUSH protocol is in channel's stack definition only one flush is executed for both connecting and * fetching state rather than two flushes if we invoke connect and getState in succession. * * If the channel is closed an exception will be thrown. * * * @param cluster_name the cluster name to connect to. Cannot be null. * @param target the state provider. If null state will be fetched from coordinator, unless this channel is coordinator. * @param timeout the timeout for state transfer. * * @exception Exception Connecting to the cluster or state transfer was not successful * @exception IllegalStateException The channel is closed and therefore cannot be used * */ abstract public void connect(String cluster_name, Address target, long timeout) throws Exception; /** * Disconnects the channel if it is connected. If the channel is closed or disconnected, this * operation is ignored
* The channel can then be connected to the same or a different cluster again. * * @see #connect(String) */ abstract public void disconnect(); /** * Destroys the channel and its associated resources (e.g., the protocol stack). After a channel * has been closed, invoking methods on it throws the ChannelClosed exception (or * results in a null operation). It is a null operation if the channel is already closed. *

* If the channel is connected to a group, disconnect() will be called first. */ abstract public void close(); /** * Determines whether the channel is open; ie. the protocol stack has been created (may not be connected though). * @return true is channel is open, false otherwise */ @ManagedAttribute public boolean isOpen() {return state != State.CLOSED;} /** * Determines whether the channel is connected to a group. * @return true if channel is connected to cluster (group) and can send/receive messages, false otherwise */ @ManagedAttribute public boolean isConnected() {return state == State.CONNECTED;} /** * Determines whether the channel is in the connecting state; this means {@link Channel#connect(String)} has been * called, but hasn't returned yet * @return true if the channel is in the connecting state, false otherwise */ @ManagedAttribute public boolean isConnecting() {return state == State.CONNECTING;} /** * Determines whether the channel is in the closed state. * @return */ @ManagedAttribute public boolean isClosed() {return state == State.CLOSED;} /** * Returns a map of statistics of the various protocols and of the channel itself. * * @return Map. A map where the keys are the protocols ("channel" pseudo key is used * for the channel itself") and the values are property maps. */ public abstract Map dumpStats(); /** * Sends a message. The message contains *

    *
  1. a destination address (Address). A null address sends the message to all * group members. *
  2. a source address. Can be left empty as it will be assigned automatically *
  3. a byte buffer. The message contents. *
  4. several additional fields. They can be used by application programs (or patterns). E.g. a * message ID, flags etc *
* * @param msg * The message to be sent. Destination and buffer should be set. A null destination * means to send to all group members. * @exception IllegalStateException * thrown if the channel is disconnected or closed */ abstract public void send(Message msg) throws Exception; /** * Helper method to create a Message with given parameters and invoke {@link #send(Message)}. * * @param dst * Destination address for message. If null, message will be sent to all current group * members * @param obj * A serializable object. Will be marshalled into the byte buffer of the Message. If it * is not serializable, an exception will be thrown * @throws Exception * exception thrown if message sending was not successful */ abstract public void send(Address dst, Object obj) throws Exception; /** * Sends a message. See {@link #send(Address,byte[],int,int)} for details * * @param dst * destination address for message. If null, message will be sent to all current group * members * @param buf * buffer message payload * @throws Exception * exception thrown if message sending was not successful */ abstract public void send(Address dst, byte[] buf) throws Exception; /** * Sends a message to a destination. * * @param dst * The destination address. If null, the message will be sent to all cluster nodes (= * group members) * @param buf * The buffer to be sent * @param offset * The offset into the buffer * @param length * The length of the data to be sent. Has to be <= buf.length - offset. This will send * length bytes starting at offset * @throws Exception * If send() failed */ abstract public void send(Address dst, byte[] buf, int offset, int length) throws Exception; /** * Enables access to event mechanism of a channel and is normally not used by clients directly. * * @param evt sends an Event to a specific protocol layer and receives a response. * @return a response from a particular protocol layer targeted by Event parameter */ public Object down(Event evt) { return null; } /** * Gets the current view. The view may only be available after a successful * connect(). The result of calling this method on an unconnected channel is * implementation defined (may return null). Calling this method on a closed channel returns a * null view. * * @return The current view. */ abstract public View getView(); /** * Returns the channel's own address. The result of calling this method on an unconnected channel * is implementation defined (may return null). Calling this method on a closed channel returns * null. Addresses can be used as destination in the send() operation. * * @return The channel's address (opaque) */ abstract public Address getAddress(); /** * Returns the logical name of this channel if set. * * @return The logical name or null (if not set) */ abstract public String getName(); /** * Returns the logical name of a given member. The lookup is from the local cache of logical * address / logical name mappings and no remote communication is performed. * * @param member * @return The logical name for member */ abstract public String getName(Address member); /** * Sets the logical name for the channel. The name will stay associated with this channel for the * channel's lifetime (until close() is called). This method should be called before * calling connect(). * * @param name */ abstract public void setName(String name); /** Names a channel, same as {@link #setName(String)} */ abstract public Channel name(String name); /** * Returns the cluster name of the group of which the channel is a member. This is the object * that was the argument to connect(). Calling this method on a closed channel * returns null. * * @return The cluster name */ abstract public String getClusterName(); public String getProperties() { return "n/a"; } /** * Sets this channel event handler to be a recipient off all events . These will not be received * by the channel (except connect/disconnect, state retrieval and the like). This can be used by * building blocks on top of a channel; thus the channel is used as a pass-through medium, and * the building blocks take over some of the channel's tasks. However, tasks such as connection * management and state transfer is still handled by the channel. * * @param up_handler handler to handle channel events */ public void setUpHandler(UpHandler up_handler) { this.up_handler=up_handler; } /** * Returns UpHandler installed for this channel * * @return the installed UpHandler implementation */ public UpHandler getUpHandler() { return up_handler; } /** * Adds a ChannelListener instance that will be notified when a channel event such as connect, * disconnect or close occurs. * * @param listener to be notified */ public synchronized void addChannelListener(ChannelListener listener) { if(listener == null) return; if(channel_listeners == null) channel_listeners=new CopyOnWriteArraySet<>(); channel_listeners.add(listener); } /** * Removes a ChannelListener previously installed * * @param listener to be removed */ public synchronized void removeChannelListener(ChannelListener listener) { if(channel_listeners != null && listener != null) channel_listeners.remove(listener); } /** Clears all installed ChannelListener instances */ public synchronized void clearChannelListeners() { if(channel_listeners != null) channel_listeners.clear(); } /** * Sets the receiver for this channel. Receiver will in turn handle all messages, view changes, * implement state transfer logic and so on. * * @param r the receiver instance for this channel * @see Receiver * * */ public void setReceiver(Receiver r) { if(receiver != null && r != null) log.warn("%s: receiver already set"); receiver=r; } /** * Returns a receiver for this channel if it has been installed using * {@link Channel#setReceiver(Receiver)} , null otherwise * * @return a receiver installed on this channel */ public Receiver getReceiver() { return receiver; } /** * When set to true, all messages sent by a member A will be discarded by A. * * @param flag */ public void setDiscardOwnMessages(boolean flag) {discard_own_messages=flag;} /** * Returns true if this channel will discard its own messages, false otherwise * * @return */ public boolean getDiscardOwnMessages() {return discard_own_messages;} abstract public boolean flushSupported(); /** * Performs the flush of the cluster but only for the specified flush participants. *

* All pending messages are flushed out but only for the flush participants. The remaining * members in the cluster are not included in the flush. The list of flush participants should be * a proper subset of the current view. *

* If this flush is not automatically resumed it is an obligation of the application to invoke * the matching {@link #stopFlush(List)} method with the same list of members used in * {@link #startFlush(List, boolean)}. * * @param automatic_resume * if true call {@link #stopFlush()} after the flush * @see #startFlush(boolean) * @see Util#startFlush(Channel, List, int, long, long) */ abstract public void startFlush(List

flushParticipants, boolean automatic_resume) throws Exception; /** * Performs the flush of the cluster, ie. all pending application messages are flushed out of the cluster and * all members ack their reception. After this call returns, no member will be allowed to send any * messages until {@link #stopFlush()} is called. *

* In the case of flush collisions (another member attempts flush at roughly the same time) start flush will * fail by throwing an Exception. Applications can re-attempt flushing after certain back-off period. *

* JGroups provides a helper random sleep time backoff algorithm for flush using Util class. * * @param automatic_resume * if true call {@link #stopFlush()} after the flush * * @see Util#startFlush(Channel, List, int, long, long) */ abstract public void startFlush(boolean automatic_resume) throws Exception; /** * Stops the current flush of the cluster. Cluster members are unblocked and allowed to send new * and pending messages. * * @see Channel#startFlush(boolean) * @see Channel#startFlush(List, boolean) */ abstract public void stopFlush(); /** * Stops the current flush of the cluster for the specified flush participants. Flush * participants are unblocked and allowed to send new and pending messages. *

* * It is an obligation of the application to invoke the matching * {@link #startFlush(List, boolean)} method with the same list of members prior to invocation of * this method. * * @param flushParticipants * the flush participants */ abstract public void stopFlush(List

flushParticipants); /** * Retrieves the full state from the target member. *

* State transfer is initiated by invoking getState on this channel. The state provider in turn * invokes {@link MessageListener#getState(java.io.OutputStream)} callback and sends a state to * this node, the state receiver. After the state arrives to the state receiver * {@link MessageListener#setState(java.io.InputStream)} callback is invoked to install the * state. * * @param target * The state provider. If null the coordinator is used by default * @param timeout * The number of milliseconds to wait for the operation to complete successfully. 0 * waits until the state has been received * * @see MessageListener#getState(java.io.OutputStream) * @see MessageListener#setState(java.io.InputStream) * * @exception IllegalStateException * The channel was closed or disconnected, or the flush (if present) failed * @exception StateTransferException * raised if there was a problem during the state transfer */ abstract public void getState(Address target, long timeout) throws Exception; protected void notifyChannelConnected(Channel c) { if(channel_listeners == null) return; for(ChannelListener channelListener: channel_listeners) { try { channelListener.channelConnected(c); } catch(Throwable t) { log.error(Util.getMessage("CallbackException"), "channelConnected()", t); } } } protected void notifyChannelDisconnected(Channel c) { if(channel_listeners == null) return; for(ChannelListener channelListener: channel_listeners) { try { channelListener.channelDisconnected(c); } catch(Throwable t) { log.error(Util.getMessage("CallbackException"), "channelDisconnected()", t); } } } protected void notifyChannelClosed(Channel c) { if(channel_listeners == null) return; for(ChannelListener channelListener: channel_listeners) { try { channelListener.channelClosed(c); } catch(Throwable t) { log.error(Util.getMessage("CallbackException"), "channelClosed()", t); } } } }