org.jgroups.blocks.TCPConnectionMap Maven / Gradle / Ivy
package org.jgroups.blocks;
import org.jgroups.Address;
import org.jgroups.Version;
import org.jgroups.logging.Log;
import org.jgroups.logging.LogFactory;
import org.jgroups.stack.IpAddress;
import org.jgroups.util.*;
import java.io.*;
import java.net.*;
import java.util.Collection;
import java.util.Map;
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.LinkedBlockingQueue;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.locks.ReentrantLock;
/**
* Class that manages TCP connections between members
* @author Vladimir Blagojevic
* @author Bela Ban
*/
public class TCPConnectionMap {
protected final Mapper mapper;
protected final InetAddress bind_addr;
protected InetAddress client_bind_addr;
protected int client_bind_port;
protected boolean defer_client_binding;
protected final Address local_addr; // bind_addr + port of srv_sock
protected final ServerSocket srv_sock;
protected Receiver recvr;
protected final long conn_expire_time; // ns
protected Log log=LogFactory.getLog(getClass());
protected int recv_buf_size=120000;
protected int send_buf_size=60000;
protected int send_queue_size=2000;
protected int sock_conn_timeout=1000; // max time in millis to wait for Socket.connect() to return
protected int peer_addr_read_timeout=2000; // max time in milliseconds to block on reading peer address
protected boolean tcp_nodelay=false;
protected int linger=-1;
protected final Thread acceptor;
protected final AtomicBoolean running=new AtomicBoolean(false);
protected volatile boolean use_send_queues=true;
protected SocketFactory socket_factory=new DefaultSocketFactory();
protected TimeService time_service;
public TCPConnectionMap(String service_name,
ThreadFactory f,
SocketFactory socket_factory,
Receiver r,
InetAddress bind_addr,
InetAddress external_addr,
int external_port,
int srv_port,
int max_port
) throws Exception {
this(service_name, f,socket_factory, r,bind_addr,external_addr,external_port, srv_port,max_port,0,0);
}
public TCPConnectionMap(String service_name,
ThreadFactory f,
Receiver r,
InetAddress bind_addr,
InetAddress external_addr,
int external_port,
int srv_port,
int max_port,
long reaper_interval,
long conn_expire_time
) throws Exception {
this(service_name, f, null, r, bind_addr, external_addr, external_port, srv_port, max_port, reaper_interval, conn_expire_time);
}
public TCPConnectionMap(String service_name,
ThreadFactory f,
SocketFactory socket_factory,
Receiver r,
InetAddress bind_addr,
InetAddress external_addr,
int external_port,
int srv_port,
int max_port,
long reaper_interval,
long conn_expire_time
) throws Exception {
this.mapper = new Mapper(f,reaper_interval);
this.recvr=r;
this.bind_addr=bind_addr;
this.conn_expire_time = TimeUnit.NANOSECONDS.convert(conn_expire_time, TimeUnit.MILLISECONDS);
if(socket_factory != null)
this.socket_factory=socket_factory;
this.srv_sock=Util.createServerSocket(this.socket_factory, service_name, bind_addr, srv_port, max_port);
if(external_addr != null) {
if(external_port <= 0)
local_addr=new IpAddress(external_addr, srv_sock.getLocalPort());
else
local_addr=new IpAddress(external_addr, external_port);
}
else if(bind_addr != null)
local_addr=new IpAddress(bind_addr, srv_sock.getLocalPort());
else
local_addr=new IpAddress(srv_sock.getLocalPort());
acceptor=f.newThread(new Acceptor(),"ConnectionMap.Acceptor [" + local_addr + "]");
}
public Address getLocalAddress() {return local_addr;}
public Receiver getReceiver() {return recvr;}
public void setReceiver(Receiver receiver) {this.recvr=receiver;}
public SocketFactory getSocketFactory() {return socket_factory;}
public void setSocketFactory(SocketFactory factory) {this.socket_factory=factory;}
public InetAddress clientBindAddress() {return client_bind_addr;}
public TCPConnectionMap clientBindAddress(InetAddress addr) {this.client_bind_addr=addr; return this;}
public int clientBindPort() {return client_bind_port;}
public TCPConnectionMap clientBindPort(int port) {this.client_bind_port=port; return this;}
public boolean deferClientBinding() {return defer_client_binding;}
public TCPConnectionMap deferClientBinding(boolean defer) {this.defer_client_binding=defer; return this;}
public void setReceiveBufferSize(int recv_buf_size) {this.recv_buf_size = recv_buf_size;}
public void setSocketConnectionTimeout(int timeout) {this.sock_conn_timeout = timeout;}
public TCPConnectionMap peerAddressReadTimeout(int timeout) {this.peer_addr_read_timeout=timeout; return this;}
public TCPConnectionMap timeService(TimeService ts) {this.time_service=ts; return this;}
public void setSendBufferSize(int send_buf_size) {this.send_buf_size = send_buf_size;}
public void setLinger(int linger) {this.linger = linger;}
public void setTcpNodelay(boolean tcp_nodelay) {this.tcp_nodelay = tcp_nodelay;}
public void setSendQueueSize(int send_queue_size) {this.send_queue_size = send_queue_size;}
public void setUseSendQueues(boolean flag) {this.use_send_queues=flag;}
public int getNumConnections() {return mapper.getNumConnections();}
public int getNumOpenConnections() {return mapper.getNumOpenConnections();}
public boolean connectionEstablishedTo(Address addr) {return mapper.connectionEstablishedTo(addr);}
public String printConnections() {return mapper.printConnections();}
public void retainAll(Collection members) {mapper.retainAll(members);}
public int getSenderQueueSize() {return send_queue_size;}
public TCPConnectionMap log(Log new_log) {this.log=new_log; return this;}
public void addConnectionMapListener(AbstractConnectionMap.ConnectionMapListener l) {
mapper.addConnectionMapListener(l);
}
public void removeConnectionMapListener(AbstractConnectionMap.ConnectionMapListener l) {
mapper.removeConnectionMapListener(l);
}
/**
* Calls the receiver callback. We do not serialize access to this method,
* and it may be called concurrently by several Connection handler threads.
* Therefore the receiver needs to be reentrant.
*/
public void receive(Address sender, byte[] data, int offset, int length) {
recvr.receive(sender,data,offset,length);
}
public void send(Address dest, byte[] data, int offset, int length) throws Exception {
if(dest == null) {
if(log.isErrorEnabled())
log.error(local_addr + ": destination is null");
return;
}
if(data == null) {
log.warn(local_addr + ": data is null; discarding message to " + dest);
return;
}
if(!running.get() ) {
if(log.isDebugEnabled())
log.debug(local_addr + ": connection table is not running, discarding message to " + dest);
return;
}
if(dest.equals(local_addr)) {
receive(local_addr, data, offset, length);
return;
}
// 1. Try to obtain correct Connection (or create one if not yet existent)
TCPConnection conn=null;
try {
conn=mapper.getConnection(dest);
}
catch(Throwable t) {
}
// 2. Send the message using that connection
if(conn != null && !conn.isConnected()) { // perhaps not connected because of concurrent connections (JGRP-1549)
Util.sleepRandom(1, 50);
try {
conn=mapper.getConnection(dest); // try one more time
}
catch(Throwable t) {
}
}
if(conn != null) {
try {
conn.send(data, offset, length);
}
catch(Exception ex) {
mapper.removeConnectionIfPresent(dest,conn);
throw ex;
}
}
}
/** Flushes the TCPConnection associated with destination */
public void flush(Address destination) throws Exception {
TCPConnection conn=mapper.getConnection(destination);
if(conn != null)
conn.flush();
}
public void start() throws Exception {
if(running.compareAndSet(false, true)) {
acceptor.start();
mapper.start();
}
}
public void stop() {
if(running.compareAndSet(true, false)) {
try {
getSocketFactory().close(srv_sock);
}
catch(IOException e) {
}
Util.interruptAndWaitToDie(acceptor);
mapper.stop();
}
}
public String toString() {
StringBuilder ret=new StringBuilder();
ret.append("local_addr=" + local_addr).append("\n");
ret.append("connections (" + mapper.size() + "):\n");
ret.append(mapper.toString());
ret.append('\n');
return ret.toString();
}
protected void setSocketParameters(Socket client_sock) throws SocketException {
try {
client_sock.setSendBufferSize(send_buf_size);
}
catch(IllegalArgumentException ex) {
if(log.isErrorEnabled())
log.error(local_addr + ": exception setting send buffer size to " + send_buf_size + " bytes", ex);
}
try {
client_sock.setReceiveBufferSize(recv_buf_size);
}
catch(IllegalArgumentException ex) {
log.error(local_addr + ": exception setting receive buffer size to " + send_buf_size + " bytes", ex);
}
client_sock.setKeepAlive(true);
client_sock.setTcpNoDelay(tcp_nodelay);
if(linger > 0)
client_sock.setSoLinger(true, linger);
else
client_sock.setSoLinger(false, -1);
}
/** Used for message reception. */
public interface Receiver {
void receive(Address sender, byte[] data, int offset, int length);
}
protected class Acceptor implements Runnable {
/**
* Acceptor thread. Continuously accept new connections. Create a new thread for each new connection and put
* it in conns. When the thread should stop, it is interrupted by the thread creator.
*/
public void run() {
while(!srv_sock.isClosed() && !Thread.currentThread().isInterrupted()) {
Socket client_sock=null;
try {
client_sock=srv_sock.accept();
handleAccept(client_sock);
}
catch(Exception ex) {
if(ex instanceof SocketException && srv_sock.isClosed() || Thread.currentThread().isInterrupted())
break;
if(log.isWarnEnabled())
log.warn(Util.getMessage("AcceptError"), ex);
Util.close(client_sock);
}
}
}
protected void handleAccept(final Socket client_sock) throws Exception {
TCPConnection conn=null;
try {
conn=new TCPConnection(client_sock);
Address peer_addr=conn.getPeerAddress();
if(log.isTraceEnabled())
log.trace(local_addr + ": " + peer_addr + " trying to connect to me");
mapper.getLock().lock();
try {
boolean conn_exists=mapper.hasConnection(peer_addr),
replace=conn_exists && local_addr.compareTo(peer_addr) < 0; // bigger conn wins
if(!conn_exists || replace) {
mapper.addConnection(peer_addr, conn); // closes old conn
conn.start(mapper.getThreadFactory());
if(log.isTraceEnabled())
log.trace(local_addr + ": accepted connection from " + peer_addr +
explanation(conn_exists, replace));
}
else {
if(log.isTraceEnabled())
log.trace(local_addr + ": rejected connection from " + peer_addr +
explanation(conn_exists, replace));
Util.close(conn); // keep our existing conn, reject accept() and close client_sock
}
}
finally {
mapper.getLock().unlock();
}
}
catch(Exception ex) {
Util.close(conn);
throw ex;
}
}
}
protected static String explanation(boolean connection_existed, boolean replace) {
StringBuilder sb=new StringBuilder();
if(connection_existed) {
sb.append(" (connection existed");
if(replace)
sb.append(" but was replaced because my address is lower)");
else
sb.append(" and my address won as it's higher)");
}
else {
sb.append(" (connection didn't exist)");
}
return sb.toString();
}
public class TCPConnection implements Connection {
protected final Socket sock; // socket to/from peer (result of srv_sock.accept() or new Socket())
protected final ReentrantLock send_lock=new ReentrantLock(); // serialize send()
protected final byte[] cookie= { 'b', 'e', 'l', 'a' };
protected DataOutputStream out;
protected DataInputStream in;
protected Address peer_addr; // address of the 'other end' of the connection
protected long last_access=getTimestamp(); // last time a message was sent or received (ns)
protected Sender sender;
protected Receiver receiver;
/** Creates a connection stub and binds it, use {@link #connect(java.net.SocketAddress)} to connect */
public TCPConnection(Address peer_addr) throws Exception {
if(peer_addr == null)
throw new IllegalArgumentException("Invalid parameter peer_addr="+ peer_addr);
this.peer_addr=peer_addr;
this.sock=socket_factory.createSocket("jgroups.tcp.sock");
setSocketParameters(sock);
}
public TCPConnection(Socket s) throws Exception {
if(s == null)
throw new IllegalArgumentException("Invalid parameter s=" + s);
setSocketParameters(s);
this.out=new DataOutputStream(new BufferedOutputStream(s.getOutputStream()));
this.in=new DataInputStream(new BufferedInputStream(s.getInputStream()));
this.peer_addr=readPeerAddress(s);
this.sock=s;
}
protected long getTimestamp() {
return time_service != null? time_service.timestamp() : System.nanoTime();
}
protected Address getPeerAddress() {
return peer_addr;
}
protected boolean isSenderUsed(){
return getSenderQueueSize() > 0 && use_send_queues;
}
protected String getSockAddress() {
StringBuilder sb=new StringBuilder();
if(sock != null) {
sb.append(sock.getLocalAddress().getHostAddress()).append(':').append(sock.getLocalPort());
sb.append(" - ").append(sock.getInetAddress().getHostAddress()).append(':').append(sock.getPort());
}
return sb.toString();
}
protected void updateLastAccessed() {
if(conn_expire_time > 0)
last_access=getTimestamp();
}
/** Called after {@link TCPConnection#TCPConnection(org.jgroups.Address)} */
protected void connect(SocketAddress destAddr) throws Exception {
try {
if(!defer_client_binding)
this.sock.bind(new InetSocketAddress(client_bind_addr, client_bind_port));
if(this.sock.getLocalSocketAddress() != null && this.sock.getLocalSocketAddress().equals(destAddr))
throw new IllegalStateException("socket's bind and connect address are the same: " + destAddr);
Util.connect(this.sock, destAddr, sock_conn_timeout);
this.out=new DataOutputStream(new BufferedOutputStream(sock.getOutputStream()));
this.in=new DataInputStream(new BufferedInputStream(sock.getInputStream()));
sendLocalAddress(getLocalAddress());
}
catch(Exception t) {
socket_factory.close(this.sock);
throw t;
}
}
protected TCPConnection start(ThreadFactory f) {
if(receiver != null)
receiver.stop();
receiver=new Receiver(f).start();
if(isSenderUsed()) {
if(sender != null)
sender.stop();
sender=new Sender(f, getSenderQueueSize()).start();
}
return this;
}
/**
*
* @param data Guaranteed to be non null
* @param offset
* @param length
*/
protected void send(byte[] data, int offset, int length) throws Exception {
if (sender != null) {
// we need to copy the byte[] buffer here because the original buffer might get changed meanwhile
byte[] tmp = new byte[length];
System.arraycopy(data, offset, tmp, 0, length);
sender.addToQueue(tmp);
}
else
_send(data, offset, length, true, true);
}
/**
* Sends data using the 'out' output stream of the socket
*
* @param data
* @param offset
* @param length
* @param acquire_lock
* @throws Exception
*/
protected void _send(byte[] data, int offset, int length, boolean acquire_lock, boolean flush) throws Exception {
if(acquire_lock)
send_lock.lock();
try {
doSend(data, offset, length, acquire_lock, flush);
updateLastAccessed();
}
catch(InterruptedException iex) {
Thread.currentThread().interrupt(); // set interrupt flag again
}
finally {
if(acquire_lock)
send_lock.unlock();
}
}
protected void doSend(byte[] data, int offset, int length, boolean acquire_lock, boolean flush) throws Exception {
out.writeInt(length); // write the length of the data buffer first
out.write(data,offset,length);
if(!flush || (acquire_lock && send_lock.hasQueuedThreads()))
return; // don't flush as some of the waiting threads will do the flush, or flush is false
out.flush(); // may not be very efficient (but safe)
}
protected void flush() throws Exception {
if(out != null)
out.flush();
}
/**
* Reads the peer's address. First a cookie has to be sent which has to
* match my own cookie, otherwise the connection will be refused
*/
protected Address readPeerAddress(Socket client_sock) throws Exception {
int timeout=client_sock.getSoTimeout();
client_sock.setSoTimeout(peer_addr_read_timeout);
try {
// read the cookie first
byte[] input_cookie=new byte[cookie.length];
in.readFully(input_cookie, 0, input_cookie.length);
if(!matchCookie(input_cookie))
throw new SocketException("ConnectionMap.Connection.readPeerAddress(): cookie read by " + getLocalAddress()
+ " does not match own cookie; terminating connection");
// then read the version
short version=in.readShort();
if(!Version.isBinaryCompatible(version))
throw new IOException("packet from " + client_sock.getInetAddress() + ":" + client_sock.getPort() +
" has different version (" + Version.print(version) +
") from ours (" + Version.printVersion() + "); discarding it");
Address client_peer_addr=new IpAddress();
client_peer_addr.readFrom(in);
updateLastAccessed();
return client_peer_addr;
}
finally {
client_sock.setSoTimeout(timeout);
}
}
/**
* Send the cookie first, then the our port number. If the cookie
* doesn't match the receiver's cookie, the receiver will reject the
* connection and close it.
*
* @throws Exception
*/
protected void sendLocalAddress(Address local_addr) throws Exception {
// write the cookie
out.write(cookie, 0, cookie.length);
// write the version
out.writeShort(Version.version);
local_addr.writeTo(out);
out.flush(); // needed ?
updateLastAccessed();
}
protected boolean matchCookie(byte[] input) {
if(input == null || input.length < cookie.length) return false;
for(int i=0; i < cookie.length; i++)
if(cookie[i] != input[i]) return false;
return true;
}
protected class Receiver implements Runnable {
protected final Thread recv;
protected volatile boolean receiving=true;
public Receiver(ThreadFactory f) {
recv = f.newThread(this,"Connection.Receiver [" + getSockAddress() + "]");
}
public Receiver start() {
receiving=true;
recv.start();
return this;
}
public Receiver stop() {
receiving=false;
recv.interrupt();
return this;
}
public boolean isRunning() {
return receiving;
}
public boolean canRun() {
return isRunning() && isConnected();
}
public void run() {
try {
while(!Thread.currentThread().isInterrupted() && canRun()) {
try {
int len=in.readInt();
byte[] buf=new byte[len];
in.readFully(buf, 0, len);
updateLastAccessed();
TCPConnectionMap.this.recvr.receive(peer_addr,buf,0,len);
}
catch(OutOfMemoryError mem_ex) {
break; // continue;
}
catch(IOException io_ex) {
break;
}
catch(Throwable e) {
}
}
}
finally {
mapper.removeConnectionIfPresent(peer_addr,TCPConnection.this);
}
}
}
protected class Sender implements Runnable {
protected final BlockingQueue send_queue;
protected final Thread runner;
protected volatile boolean started=true;
public Sender(ThreadFactory tf, int send_queue_size) {
this.runner=tf.newThread(this, "Connection.Sender [" + getSockAddress() + "]");
this.send_queue=new LinkedBlockingQueue<>(send_queue_size);
}
public void addToQueue(byte[] data) throws Exception{
if(canRun())
if (!send_queue.offer(data, sock_conn_timeout, TimeUnit.MILLISECONDS))
log.warn("Discarding message because TCP send_queue is full and hasn't been releasing for " + sock_conn_timeout + " ms");
}
public Sender start() {
started=true;
runner.start();
return this;
}
public Sender stop() {
started=false;
runner.interrupt();
return this;
}
public boolean isRunning() {
return started;
}
public boolean canRun() {
return isRunning() && isConnected();
}
public void run() {
try {
while(!Thread.currentThread().isInterrupted() && canRun()) {
byte[] data=null;
try {
data=send_queue.take();
}
catch(InterruptedException e) {
// Thread.currentThread().interrupt();
break;
}
if(data != null) {
try {
_send(data, 0, data.length, false, send_queue.isEmpty());
}
catch(Throwable ignored) {
}
}
}
}
finally {
mapper.removeConnectionIfPresent(peer_addr, TCPConnection.this);
}
}
}
public String toString() {
StringBuilder ret=new StringBuilder();
InetAddress local=null, remote=null;
String local_str, remote_str;
Socket tmp_sock=sock;
if(tmp_sock == null)
ret.append("");
else {
//since the sock variable gets set to null we want to make
//make sure we make it through here without a nullpointer exception
local=tmp_sock.getLocalAddress();
remote=tmp_sock.getInetAddress();
local_str=local != null? Util.shortName(local) : "";
remote_str=remote != null? Util.shortName(remote) : "";
ret.append('<' + local_str
+ ':'
+ tmp_sock.getLocalPort()
+ " --> "
+ remote_str
+ ':'
+ tmp_sock.getPort()
+ "> ("
+ TimeUnit.SECONDS.convert(getTimestamp() - last_access, TimeUnit.NANOSECONDS)
+ " secs old) [" + (isOpen()? "open]" : "closed]"));
}
tmp_sock=null;
return ret.toString();
}
public boolean isExpired(long now) {
return conn_expire_time > 0 && now - last_access >= conn_expire_time;
}
public boolean isConnected() {
return !sock.isClosed() && sock.isConnected();
}
public boolean isOpen() {
return isConnected()
&& (!isSenderUsed() || sender.isRunning())
&& (receiver != null && receiver.isRunning());
}
public void close() throws IOException {
// can close even if start was never called...
send_lock.lock();
try {
if(receiver != null) {
receiver.stop();
receiver=null;
}
if(sender != null) {
sender.stop();
sender=null;
}
try {
socket_factory.close(sock);
}
catch(Throwable t) {}
Util.close(out);
Util.close(in);
}
finally {
send_lock.unlock();
}
mapper.notifyConnectionClosed(peer_addr);
}
}
protected class Mapper extends AbstractConnectionMap {
public Mapper(ThreadFactory factory) {
super(factory);
}
public Mapper(ThreadFactory factory,long reaper_interval) {
super(factory,reaper_interval);
}
public TCPConnection getConnection(Address dest) throws Exception {
TCPConnection conn;
getLock().lock();
try {
if((conn=conns.get(dest)) != null && conn.isOpen()) // keep FAST path on the most common case
return conn;
}
finally {
getLock().unlock();
}
Exception connect_exception=null; // set if connect() throws an exception
sock_creation_lock.lockInterruptibly();
try {
// lock / release, create new conn under sock_creation_lock, it can be skipped but then it takes
// extra check in conn map and closing the new connection, w/ sock_creation_lock it looks much simpler
// (slow path, so not important)
getLock().lock();
try {
conn=conns.get(dest); // check again after obtaining sock_creation_lock
if(conn != null && conn.isOpen())
return conn;
// create conn stub
conn=new TCPConnection(dest);
addConnection(dest, conn);
}
finally {
getLock().unlock();
}
// now connect to dest:
try {
if(log.isTraceEnabled())
log.trace(local_addr + ": connecting to " + dest);
conn.connect(new InetSocketAddress(((IpAddress)dest).getIpAddress(),((IpAddress)dest).getPort()));
conn.start(getThreadFactory());
if(log.isTraceEnabled())
log.trace(local_addr + ": connected to " + dest);
}
catch(Exception connect_ex) {
connect_exception=connect_ex;
}
getLock().lock();
try {
TCPConnection existing_conn=conns.get(dest); // check again after obtaining sock_creation_lock
if(existing_conn != null && existing_conn.isOpen() // added by a successful accept()
&& existing_conn != conn) {
if(log.isTraceEnabled())
log.trace(local_addr + ": found existing connection to " + dest +
", using it and deleting own conn-stub");
Util.close(conn); // close our connection; not really needed as conn was closed by accept()
return existing_conn;
}
if(connect_exception != null) {
if(log.isTraceEnabled())
log.trace(local_addr + ": failed connecting to " + dest + ": " + connect_exception);
removeConnectionIfPresent(dest, conn); // removes and closes the conn
throw connect_exception;
}
return conn;
}
finally {
getLock().unlock();
}
}
finally {
sock_creation_lock.unlock();
}
}
public boolean connectionEstablishedTo(Address address) {
lock.lock();
try {
TCPConnection conn=conns.get(address);
return conn != null && conn.isConnected();
}
finally {
lock.unlock();
}
}
public int size() {return conns.size();}
public String toString() {
StringBuilder sb=new StringBuilder();
getLock().lock();
try {
for(Map.Entry entry: conns.entrySet()) {
sb.append(entry.getKey()).append(": ").append(entry.getValue()).append("\n");
}
return sb.toString();
}
finally {
getLock().unlock();
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy