com.hazelcast.client.connection.nio.ClientConnectionManagerImpl Maven / Gradle / Ivy
/*
* Copyright (c) 2008-2016, Hazelcast, Inc. All Rights Reserved.
*
* 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 com.hazelcast.client.connection.nio;
import com.hazelcast.client.ClientExtension;
import com.hazelcast.client.config.ClientConfig;
import com.hazelcast.client.config.ClientNetworkConfig;
import com.hazelcast.client.config.ClientProperties;
import com.hazelcast.client.config.SocketOptions;
import com.hazelcast.client.connection.AddressTranslator;
import com.hazelcast.client.connection.Authenticator;
import com.hazelcast.client.connection.ClientConnectionManager;
import com.hazelcast.client.impl.HazelcastClientInstanceImpl;
import com.hazelcast.client.spi.ClientInvocationService;
import com.hazelcast.client.spi.impl.ClientExecutionServiceImpl;
import com.hazelcast.client.spi.impl.ClientInvocation;
import com.hazelcast.client.spi.impl.ConnectionHeartbeatListener;
import com.hazelcast.client.spi.impl.listener.ClientListenerServiceImpl;
import com.hazelcast.cluster.client.ClientPingRequest;
import com.hazelcast.config.SocketInterceptorConfig;
import com.hazelcast.core.HazelcastException;
import com.hazelcast.logging.ILogger;
import com.hazelcast.logging.Logger;
import com.hazelcast.nio.Address;
import com.hazelcast.nio.Connection;
import com.hazelcast.nio.ConnectionListener;
import com.hazelcast.nio.Packet;
import com.hazelcast.nio.SocketInterceptor;
import com.hazelcast.nio.tcp.SocketChannelWrapper;
import com.hazelcast.nio.tcp.SocketChannelWrapperFactory;
import com.hazelcast.nio.tcp.nonblocking.NonBlockingIOThread;
import com.hazelcast.nio.tcp.nonblocking.NonBlockingIOThreadOutOfMemoryHandler;
import com.hazelcast.util.Clock;
import com.hazelcast.util.ExceptionUtil;
import java.io.IOException;
import java.net.Socket;
import java.nio.channels.SocketChannel;
import java.util.Collections;
import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;
import java.util.concurrent.CopyOnWriteArraySet;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicInteger;
import static com.hazelcast.client.config.ClientProperty.HEARTBEAT_INTERVAL;
import static com.hazelcast.client.config.ClientProperty.HEARTBEAT_TIMEOUT;
import static com.hazelcast.client.config.SocketOptions.DEFAULT_BUFFER_SIZE_BYTE;
import static com.hazelcast.client.config.SocketOptions.KILO_BYTE;
public class ClientConnectionManagerImpl implements ClientConnectionManager {
private static final ILogger LOGGER = Logger.getLogger(ClientConnectionManagerImpl.class);
private static final NonBlockingIOThreadOutOfMemoryHandler OUT_OF_MEMORY_HANDLER = new NonBlockingIOThreadOutOfMemoryHandler() {
@Override
public void handle(OutOfMemoryError error) {
LOGGER.severe(error);
}
};
private final int connectionTimeout;
private final long heartBeatInterval;
private final long heartBeatTimeout;
private final ConcurrentMap connectionLockMap = new ConcurrentHashMap();
protected final AtomicInteger connectionIdGen = new AtomicInteger();
private final HazelcastClientInstanceImpl client;
private final SocketInterceptor socketInterceptor;
private final SocketOptions socketOptions;
private NonBlockingIOThread inputThread;
private NonBlockingIOThread outSelector;
private final SocketChannelWrapperFactory socketChannelWrapperFactory;
private final ClientExecutionServiceImpl executionService;
private final AddressTranslator addressTranslator;
private final ConcurrentMap connections
= new ConcurrentHashMap();
private final Set connectionsInProgress =
Collections.newSetFromMap(new ConcurrentHashMap());
private final Set connectionListeners = new CopyOnWriteArraySet();
private final Set heartbeatListeners =
new CopyOnWriteArraySet();
protected volatile boolean alive;
public ClientConnectionManagerImpl(HazelcastClientInstanceImpl client, AddressTranslator addressTranslator) {
this.client = client;
this.addressTranslator = addressTranslator;
final ClientConfig config = client.getClientConfig();
final ClientNetworkConfig networkConfig = config.getNetworkConfig();
int connTimeout = networkConfig.getConnectionTimeout();
connectionTimeout = connTimeout == 0 ? Integer.MAX_VALUE : connTimeout;
ClientProperties clientProperties = client.getClientProperties();
long timeout = clientProperties.getMillis(HEARTBEAT_TIMEOUT);
this.heartBeatTimeout = timeout > 0 ? timeout : Integer.parseInt(HEARTBEAT_TIMEOUT.getDefaultValue());
long interval = clientProperties.getMillis(HEARTBEAT_INTERVAL);
heartBeatInterval = interval > 0 ? interval : Integer.parseInt(HEARTBEAT_INTERVAL.getDefaultValue());
executionService = (ClientExecutionServiceImpl) client.getClientExecutionService();
initializeSelectors(client);
socketOptions = networkConfig.getSocketOptions();
ClientExtension clientExtension = client.getClientExtension();
socketChannelWrapperFactory = clientExtension.createSocketChannelWrapperFactory();
socketInterceptor = initSocketInterceptor(networkConfig.getSocketInterceptorConfig());
}
protected void initializeSelectors(HazelcastClientInstanceImpl client) {
inputThread = new NonBlockingIOThread(
client.getThreadGroup(),
client.getName() + ".ClientInSelector",
Logger.getLogger(NonBlockingIOThread.class),
OUT_OF_MEMORY_HANDLER);
outSelector = new ClientNonBlockingOutputThread(
client.getThreadGroup(),
client.getName() + ".ClientOutSelector",
Logger.getLogger(ClientNonBlockingOutputThread.class),
OUT_OF_MEMORY_HANDLER);
}
private SocketInterceptor initSocketInterceptor(SocketInterceptorConfig sic) {
if (sic != null && sic.isEnabled()) {
ClientExtension clientExtension = client.getClientExtension();
return clientExtension.createSocketInterceptor();
}
return null;
}
@Override
public boolean isAlive() {
return alive;
}
@Override
public synchronized void start() {
if (alive) {
return;
}
alive = true;
startSelectors();
HeartBeat heartBeat = new HeartBeat();
executionService.scheduleWithFixedDelay(heartBeat, heartBeatInterval, heartBeatInterval, TimeUnit.MILLISECONDS);
}
protected void startSelectors() {
inputThread.start();
outSelector.start();
}
@Override
public synchronized void shutdown() {
if (!alive) {
return;
}
alive = false;
for (ClientConnection connection : connections.values()) {
connection.close();
}
shutdownSelectors();
connectionLockMap.clear();
connectionListeners.clear();
heartbeatListeners.clear();
}
protected void shutdownSelectors() {
inputThread.shutdown();
outSelector.shutdown();
}
public ClientConnection getConnection(Address target) {
return connections.get(target);
}
public ClientConnection getOrConnect(Address target, Authenticator authenticator) throws IOException {
Address remoteAddress = addressTranslator.translate(target);
if (remoteAddress == null) {
throw new IOException("Address is required!");
}
ClientConnection connection = connections.get(target);
Object lock = getLock(target);
if (connection == null) {
synchronized (lock) {
connection = connections.get(target);
if (connection == null) {
connection = initializeConnection(remoteAddress, authenticator);
}
}
}
return connection;
}
private ClientConnection initializeConnection(Address address, Authenticator authenticator) throws IOException {
ClientConnection connection = createSocketConnection(address);
authenticate(authenticator, connection);
connections.put(connection.getRemoteEndpoint(), connection);
fireConnectionAddedEvent(connection);
return connection;
}
public ClientConnection getOrTriggerConnect(Address target, Authenticator authenticator) throws IOException {
Address remoteAddress = addressTranslator.translate(target);
if (remoteAddress == null) {
throw new IOException("Address is required!");
}
ClientExecutionServiceImpl executionService = (ClientExecutionServiceImpl) client.getClientExecutionService();
ClientConnection connection = connections.get(target);
if (connection != null) {
return connection;
}
if (connectionsInProgress.add(target)) {
executionService.executeInternal(new InitConnectionTask(target, remoteAddress, authenticator));
}
throw new IOException("No available connection to address " + target);
}
private void authenticate(Authenticator authenticator, ClientConnection connection) throws IOException {
try {
authenticator.authenticate(connection);
} catch (Throwable throwable) {
connection.close(throwable);
throw ExceptionUtil.rethrow(throwable, IOException.class);
}
}
private void fireConnectionAddedEvent(ClientConnection connection) {
for (ConnectionListener connectionListener : connectionListeners) {
connectionListener.connectionAdded(connection);
}
}
protected ClientConnection createSocketConnection(final Address address) throws IOException {
if (!alive) {
throw new HazelcastException("ConnectionManager is not active!!!");
}
SocketChannel socketChannel = null;
try {
socketChannel = SocketChannel.open();
Socket socket = socketChannel.socket();
socket.setKeepAlive(socketOptions.isKeepAlive());
socket.setTcpNoDelay(socketOptions.isTcpNoDelay());
socket.setReuseAddress(socketOptions.isReuseAddress());
if (socketOptions.getLingerSeconds() > 0) {
socket.setSoLinger(true, socketOptions.getLingerSeconds());
}
int bufferSize = socketOptions.getBufferSize() * KILO_BYTE;
if (bufferSize <= 0) {
bufferSize = DEFAULT_BUFFER_SIZE_BYTE;
}
socket.setSendBufferSize(bufferSize);
socket.setReceiveBufferSize(bufferSize);
socketChannel.socket().connect(address.getInetSocketAddress(), connectionTimeout);
SocketChannelWrapper socketChannelWrapper =
socketChannelWrapperFactory.wrapSocketChannel(socketChannel, true);
final ClientConnection clientConnection = new ClientConnection(client, inputThread,
outSelector, connectionIdGen.incrementAndGet(), socketChannelWrapper);
socketChannel.configureBlocking(true);
if (socketInterceptor != null) {
socketInterceptor.onConnect(socket);
}
socketChannel.configureBlocking(false);
socket.setSoTimeout(0);
clientConnection.getReadHandler().register();
clientConnection.init();
return clientConnection;
} catch (Exception e) {
if (socketChannel != null) {
socketChannel.close();
}
throw ExceptionUtil.rethrow(e, IOException.class);
}
}
@Override
public void destroyConnection(final Connection connection) {
Address endpoint = connection.getEndPoint();
if (endpoint != null) {
final ClientConnection conn = connections.remove(endpoint);
if (conn == null) {
return;
}
conn.close();
for (ConnectionListener connectionListener : connectionListeners) {
connectionListener.connectionRemoved(conn);
}
} else {
ClientInvocationService invocationService = client.getInvocationService();
invocationService.cleanConnectionResources((ClientConnection) connection);
}
}
@Override
public void handlePacket(Packet packet) {
ClientConnection conn = (ClientConnection) packet.getConn();
conn.incrementPendingPacketCount();
if (packet.isHeaderSet(Packet.HEADER_EVENT)) {
ClientListenerServiceImpl listenerService = (ClientListenerServiceImpl) client.getListenerService();
listenerService.handleEventPacket(packet);
} else {
ClientInvocationService invocationService = client.getInvocationService();
invocationService.handlePacket(packet);
}
}
private Object getLock(Address address) {
Object lock = connectionLockMap.get(address);
if (lock == null) {
lock = new Object();
Object current = connectionLockMap.putIfAbsent(address, lock);
if (current != null) {
lock = current;
}
}
return lock;
}
class HeartBeat implements Runnable {
@Override
public void run() {
if (!alive) {
return;
}
final long now = Clock.currentTimeMillis();
for (ClientConnection connection : connections.values()) {
if (now - connection.lastReadTimeMillis() > heartBeatTimeout) {
if (connection.isHeartBeating()) {
connection.heartBeatingFailed();
fireHeartBeatStopped(connection);
}
}
if (now - connection.lastReadTimeMillis() > heartBeatInterval) {
final ClientPingRequest request = new ClientPingRequest();
new ClientInvocation(client, request, connection).invokeUrgent();
} else {
if (!connection.isHeartBeating()) {
connection.heartBeatingSucceed();
fireHeartBeatStarted(connection);
}
}
}
}
private void fireHeartBeatStarted(ClientConnection connection) {
for (ConnectionHeartbeatListener heartbeatListener : heartbeatListeners) {
heartbeatListener.heartBeatStarted(connection);
}
}
private void fireHeartBeatStopped(ClientConnection connection) {
for (ConnectionHeartbeatListener heartbeatListener : heartbeatListeners) {
heartbeatListener.heartBeatStopped(connection);
}
}
}
@Override
public void addConnectionListener(ConnectionListener connectionListener) {
connectionListeners.add(connectionListener);
}
@Override
public void addConnectionHeartbeatListener(ConnectionHeartbeatListener connectionHeartbeatListener) {
heartbeatListeners.add(connectionHeartbeatListener);
}
private class InitConnectionTask implements Runnable {
private final Address target;
private final Address remoteAddress;
private final Authenticator authenticator;
InitConnectionTask(Address target, Address remoteAddress, Authenticator authenticator) {
this.target = target;
this.remoteAddress = remoteAddress;
this.authenticator = authenticator;
}
@Override
public void run() {
final Object lock = getLock(target);
synchronized (lock) {
ClientConnection connection = connections.get(target);
if (connection != null) {
return;
}
try {
initializeConnection(remoteAddress, authenticator);
} catch (IOException e) {
LOGGER.finest(e);
} finally {
connectionsInProgress.remove(target);
}
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy