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

org.bboxdb.network.client.BBoxDBConnection Maven / Gradle / Ivy

/*******************************************************************************
 *
 *    Copyright (C) 2015-2018 the BBoxDB project
 *
 *    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.bboxdb.network.client;

import java.io.BufferedInputStream;
import java.io.BufferedOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.net.Inet4Address;
import java.net.InetSocketAddress;
import java.net.Socket;
import java.net.SocketException;
import java.nio.ByteBuffer;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.TimeUnit;

import org.bboxdb.commons.CloseableHelper;
import org.bboxdb.commons.NetworkInterfaceHelper;
import org.bboxdb.commons.Retryer;
import org.bboxdb.commons.ServiceState;
import org.bboxdb.misc.Const;
import org.bboxdb.network.NetworkConst;
import org.bboxdb.network.NetworkPackageDecoder;
import org.bboxdb.network.capabilities.PeerCapabilities;
import org.bboxdb.network.client.future.HelloFuture;
import org.bboxdb.network.client.future.NetworkOperationFuture;
import org.bboxdb.network.client.response.CompressionHandler;
import org.bboxdb.network.client.response.ErrorHandler;
import org.bboxdb.network.client.response.HelloHandler;
import org.bboxdb.network.client.response.JoinedTupleHandler;
import org.bboxdb.network.client.response.LockedTupleHandler;
import org.bboxdb.network.client.response.MultipleTupleEndHandler;
import org.bboxdb.network.client.response.MultipleTupleStartHandler;
import org.bboxdb.network.client.response.PageEndHandler;
import org.bboxdb.network.client.response.ServerResponseHandler;
import org.bboxdb.network.client.response.SuccessHandler;
import org.bboxdb.network.client.response.TupleHandler;
import org.bboxdb.network.packages.NetworkRequestPackage;
import org.bboxdb.network.packages.PackageEncodeException;
import org.bboxdb.network.packages.request.CompressionEnvelopeRequest;
import org.bboxdb.network.packages.request.DisconnectRequest;
import org.bboxdb.network.packages.request.HelloRequest;
import org.bboxdb.network.packages.response.HelloResponse;
import org.bboxdb.network.routing.RoutingHeader;
import org.bboxdb.storage.entity.PagedTransferableEntity;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import com.google.common.annotations.VisibleForTesting;
import com.google.common.base.Stopwatch;
import com.google.common.io.ByteStreams;

public class BBoxDBConnection {

	/**
	 * The sequence number generator
	 */
	private final SequenceNumberGenerator sequenceNumberGenerator;

	/**
	 * The socket of the connection
	 */
	private Socket clientSocket = null;

	/**
	 * The input stream of the socket
	 */
	private BufferedInputStream inputStream;

	/**
	 * The output stream of the socket
	 */
	private BufferedOutputStream outputStream;

	/**
	 * The pending calls
	 */
	private final Map pendingCalls;

	/**
	 * The result buffer
	 */
	private final Map> resultBuffer;

	/**
	 * The server response reader
	 */
	private ServerResponseReader serverResponseReader;

	/**
	 * The server response reader thread
	 */
	private Thread serverResponseReaderThread;

	/**
	 * The maintenance handler instance
	 */
	private ConnectionMainteinanceRunnable mainteinanceHandler;

	/**
	 * The maintenance thread
	 */
	private Thread mainteinanceThread;

	/**
	 * The maintenance handler instance
	 */
	private ConnectionFlushRunnable flushHandler;

	/**
	 * The maintenance thread
	 */
	private Thread flushThread;

	/**
	 * The default timeout
	 */
	public static final long DEFAULT_TIMEOUT_MILLIS = TimeUnit.SECONDS.toMillis(30);

	/**
	 * The connection state
	 */
	private final ServiceState connectionState;

	/**
	 * The maximum amount of in flight requests. Needs to be lower than Short.MAX_VALUE to
	 * prevent two in flight requests with the same id.
	 */
	public final static short MAX_IN_FLIGHT_CALLS = 1000;

	/**
	 * The number of in flight requests
	 * @return
	 */
	private volatile short maxInFlightCalls = MAX_IN_FLIGHT_CALLS;

	/**
	 * The capabilities of the connection
	 */
	private PeerCapabilities connectionCapabilities = new PeerCapabilities();

	/**
	 * The capabilities of the client
	 */
	private PeerCapabilities clientCapabilities = new PeerCapabilities();

	/**
	 * The pending packages for compression
	 */
	private final List pendingCompressionPackages;

	/**
	 * The Server response handler
	 */
	private final Map serverResponseHandler;

	/**
	 * The server address
	 */
	private InetSocketAddress serverAddress;

	/**
	 * The BBoxDBClient;
	 */
	private final BBoxDBClient bboxDBClient;

	/**
	 * The Logger
	 */
	private final static Logger logger = LoggerFactory.getLogger(BBoxDBClient.class);


	@VisibleForTesting
	public BBoxDBConnection() {
		this(new InetSocketAddress("localhost", 1234));
	}

	public BBoxDBConnection(final InetSocketAddress serverAddress) {

		this.serverAddress = Objects.requireNonNull(serverAddress);

		// External IP is used to create proper package routing
		if(serverAddress.getAddress().isLoopbackAddress()) {
			try {
				final Inet4Address nonLoopbackAdress = NetworkInterfaceHelper.getFirstNonLoopbackIPv4();
				this.serverAddress = new InetSocketAddress(nonLoopbackAdress, serverAddress.getPort());
			} catch (SocketException e) {
				logger.error("Connection to loopback IP " + serverAddress
						+ " requested and unable replace the IP with external IP", e);
			}
		}

		this.bboxDBClient = new BBoxDBClient(this);
		this.sequenceNumberGenerator = new SequenceNumberGenerator();
		this.connectionState = new ServiceState();

		// Default: Enable gzip compression
		this.clientCapabilities.setGZipCompression();

		// No concurrent access
		this.serverResponseHandler = new HashMap<>();

		// Concurrent access with synchronized
		this.pendingCompressionPackages = new ArrayList<>();

		// Concurrent access
		this.resultBuffer = new ConcurrentHashMap<>();
		this.pendingCalls = new ConcurrentHashMap<>();

		initResponseHandler();
	}

	/**
	 * Init the response handler
	 */
	private void initResponseHandler() {
		serverResponseHandler.put(NetworkConst.RESPONSE_TYPE_COMPRESSION, new CompressionHandler());
		serverResponseHandler.put(NetworkConst.RESPONSE_TYPE_HELLO, new HelloHandler());
		serverResponseHandler.put(NetworkConst.RESPONSE_TYPE_SUCCESS, new SuccessHandler());
		serverResponseHandler.put(NetworkConst.RESPONSE_TYPE_ERROR, new ErrorHandler());
		serverResponseHandler.put(NetworkConst.RESPONSE_TYPE_TUPLE, new TupleHandler());
		serverResponseHandler.put(NetworkConst.RESPONSE_TYPE_MULTIPLE_TUPLE_START, new MultipleTupleStartHandler());
		serverResponseHandler.put(NetworkConst.RESPONSE_TYPE_MULTIPLE_TUPLE_END, new MultipleTupleEndHandler());
		serverResponseHandler.put(NetworkConst.RESPONSE_TYPE_PAGE_END, new PageEndHandler());
		serverResponseHandler.put(NetworkConst.RESPONSE_TYPE_JOINED_TUPLE, new JoinedTupleHandler());
		serverResponseHandler.put(NetworkConst.RESPONSE_TYPE_TUPLE_LOCK_SUCCESS, new LockedTupleHandler());
	}

	/* (non-Javadoc)
	 * @see org.bboxdb.network.client.BBoxDB#connect()
	 */
	public boolean connect() {

		if(clientSocket != null || ! connectionState.isInNewState()) {
			logger.warn("Connect() called on an active connection, ignoring (state: {})", connectionState);
			return true;
		}

		logger.debug("Connecting to server: {}", getConnectionName());

		try {
			connectionState.dipatchToStarting();
			connectionState.registerCallback((c) -> { if(c.isInFailedState() ) { killPendingCalls(); } });

			final Retryer socketRetryer = new Retryer<>(10, 200, TimeUnit.MILLISECONDS, () -> {
				return new Socket(serverAddress.getAddress(), serverAddress.getPort());
			});

			if(! socketRetryer.execute()) {
				throw socketRetryer.getLastException();
			}

			clientSocket = socketRetryer.getResult();

			inputStream = new BufferedInputStream(clientSocket.getInputStream());
			outputStream = new BufferedOutputStream(clientSocket.getOutputStream());

			synchronized (pendingCalls) {
				pendingCalls.clear();
			}

			getResultBuffer().clear();

			// Start up the response reader
			serverResponseReader = new ServerResponseReader(this);
			serverResponseReaderThread = new Thread(serverResponseReader);
			serverResponseReaderThread.setName("Server response reader for " + getConnectionName());
			serverResponseReaderThread.start();

			runHandshake();
		} catch (Exception e) {
			logger.error("Got an exception while connecting to server", e);
			closeSocket();
			connectionState.dispatchToFailed(e);
			return false;
		}

		return true;
	}

	/**
	 * Close the socket
	 */
	public void closeSocket() {
		logger.info("Closing socket to server: {}", getConnectionName());
		CloseableHelper.closeWithoutException(clientSocket);
		clientSocket = null;
	}

	/**
	 * The name of the connection
	 * @return
	 */
	public String getConnectionName() {
		return serverAddress.getHostString() + " / " + serverAddress.getPort();
	}

	/**
	 * Get the next sequence number
	 * @return
	 */
	public short getNextSequenceNumber() {
		return sequenceNumberGenerator.getNextSequenceNummber();
	}

	/**
	 * Run the handshake with the server
	 * @throws ExecutionException
	 * @throws InterruptedException
	 */
	private void runHandshake() throws Exception {

		if(! connectionState.isInStartingState()) {
			logger.error("Handshaking called in wrong state: {}", connectionState);
		}

		// Capabilities are reported to server; now freeze client capabilities.
		clientCapabilities.freeze();


		final NetworkOperationFuture operationFuture = new NetworkOperationFuture(this, () -> {
			return new HelloRequest(getNextSequenceNumber(),
					NetworkConst.PROTOCOL_VERSION, clientCapabilities);
		});

		final HelloFuture helloFuture = new HelloFuture(operationFuture);
		helloFuture.waitForCompletion();

		if(operationFuture.isFailed()) {
			throw new Exception("Got an error during handshake");
		}

		final HelloResponse helloResponse = helloFuture.get(0);
		connectionCapabilities = helloResponse.getPeerCapabilities();

		connectionState.dispatchToRunning();
		logger.debug("Handshaking with {} done", getConnectionName());

		flushHandler = new ConnectionFlushRunnable(this);
		flushThread = new Thread(flushHandler);
		flushThread.setName("Flush thread for: " + getConnectionName());
		flushThread.start();

		mainteinanceHandler = new ConnectionMainteinanceRunnable(this);
		mainteinanceThread = new Thread(mainteinanceHandler);
		mainteinanceThread.setName("Connection mainteinace thread for: " + getConnectionName());
		mainteinanceThread.start();
	}

	/* (non-Javadoc)
	 * @see org.bboxdb.network.client.BBoxDB#disconnect()
	 */
	public void disconnect() {

		if(! connectionState.isInRunningState()) {
			logger.error("Unable to disconnect, connection is in state {}", connectionState);
			return;
		}

		synchronized (this) {
			logger.info("Disconnecting from server: {}", getConnectionName());
			connectionState.dispatchToStopping();

			final NetworkOperationFuture future = new NetworkOperationFuture(this, () -> {
				return new DisconnectRequest(getNextSequenceNumber());
			});

			future.execute();
		}

		settlePendingCalls(DEFAULT_TIMEOUT_MILLIS);

		terminateConnection();
	}

	/**
	 * Settle all pending calls
	 */
	public void settlePendingCalls(final long shutdownTimeMillis) {
		final Stopwatch stopwatch = Stopwatch.createStarted();

		// Wait for all pending calls to settle
		synchronized (pendingCalls) {

			while(getInFlightCalls() > 0) {
				final long timeLeft = shutdownTimeMillis - stopwatch.elapsed(TimeUnit.MILLISECONDS);

				if (timeLeft <= 0) {
					break;
				}

				if(! isConnected()) {
					logger.warn("Connection already closed but {} requests are pending",
							getInFlightCalls());
					return;
				}

				logger.info("Waiting up to {} seconds for pending requests to settle "
						+ "(pending {} / server {})", timeLeft,
						getInFlightCalls(), getConnectionName());

				try {
					// Recheck connection state all 5 seconds
					final long maxWaitTime = Math.min(timeLeft, TimeUnit.SECONDS.toMillis(5));
					pendingCalls.wait(maxWaitTime);
				} catch (InterruptedException e) {
					logger.debug("Got an InterruptedException during pending calls wait.");
					Thread.currentThread().interrupt();
					return;
				}
			}

			if(! pendingCalls.isEmpty()) {
				logger.warn("Connection is closed. Still pending calls: {} ", pendingCalls);
			}
		}
	}

	/**
	 * Kill all pending requests
	 */
	private void killPendingCalls() {
		synchronized (pendingCalls) {
			if(! pendingCalls.isEmpty()) {
				logger.warn("Socket is closed unexpected, killing pending calls: " + pendingCalls.size());

				for(final short requestId : pendingCalls.keySet()) {
					final NetworkOperationFuture future = pendingCalls.get(requestId);
					future.setFailedState();
					future.fireCompleteEvent();
				}

				pendingCalls.clear();
				pendingCalls.notifyAll();
			}
		}
	}

	/**
	 * Close the connection to the server without sending a disconnect package. For a
	 * regular disconnect, see the disconnect() method.
	 */
	public void terminateConnection() {
		if(connectionState.isInRunningState()) {
			connectionState.dispatchToStopping();
		}

		killPendingCalls();
		getResultBuffer().clear();

		closeSocket();

		logger.info("Disconnected from server: {}", getConnectionName());
		connectionState.forceDispatchToTerminated();
		mainteinanceThread.interrupt();
	}

	/* (non-Javadoc)
	 * @see org.bboxdb.network.client.BBoxDB#isConnected()
	 */
	public boolean isConnected() {
		if(clientSocket != null) {
			return ! clientSocket.isClosed();
		}

		return false;
	}

	/**
	 * Get the state of the connection
	 */
	public ServiceState getConnectionState() {
		return connectionState;
	}

	/* (non-Javadoc)
	 * @see org.bboxdb.network.client.BBoxDB#getInFlightCalls()
	 */
	public int getInFlightCalls() {
		synchronized (pendingCalls) {
			return pendingCalls.size();
		}
	}

	/* (non-Javadoc)
	 * @see org.bboxdb.network.client.BBoxDB#getMaxInFlightCalls()
	 */
	public short getMaxInFlightCalls() {
		return maxInFlightCalls;
	}

	/* (non-Javadoc)
	 * @see org.bboxdb.network.client.BBoxDB#setMaxInFlightCalls(short)
	 */
	public void setMaxInFlightCalls(short maxInFlightCalls) {
		this.maxInFlightCalls = (short) Math.min(maxInFlightCalls, MAX_IN_FLIGHT_CALLS);
	}

	/**
	 * Send a request package to the server
	 * @param responsePackage
	 * @return
	 * @throws IOException
	 */
	public void sendPackageToServer(final NetworkRequestPackage requestPackage,
			final NetworkOperationFuture future) {

		final short sequenceNumber = requestPackage.getSequenceNumber();
		final boolean result = testPackageSend(requestPackage, future);

		// Package don't need to be send
		if(result == false) {
			removeFutureAndReleaseSequencenumber(sequenceNumber);
			return;
		}

		if(connectionCapabilities.hasGZipCompression()) {
			writePackageWithCompression(requestPackage, future);
		} else {
			writePackageUncompressed(requestPackage, future);
		}
	}

	/**
	 * Recalculate the routing header and handle the exceptions
	 * @param requestPackage
	 * @param future
	 * @return
	 */
	private boolean testPackageSend(final NetworkRequestPackage requestPackage,
			final NetworkOperationFuture future) {

		try {
			// Check if package needs to be send
			final RoutingHeader routingHeader = requestPackage.getRoutingHeader();

			if(routingHeader.isRoutedPackage()) {
				if(routingHeader.getHopCount() == 0) {
					future.setMessage("No distribution regions in next hop, not sending to server");
					future.fireCompleteEvent();
					return false;
				}
			}

		} catch (PackageEncodeException e) {
			final String message = "Got a exception during package encoding";
			logger.error(message);
			future.setMessage(message);
			future.setFailedState();
			future.fireCompleteEvent();
			return false;
		}

		return true;
	}

	/**
	 * Write a package uncompresssed to the socket
	 * @param requestPackage
	 * @param future
	 */
	private void writePackageUncompressed(final NetworkRequestPackage requestPackage,
			final NetworkOperationFuture future) {

		try {
			writePackageToSocket(requestPackage);
		} catch (IOException | PackageEncodeException e) {
			logger.warn("Got an exception while sending package to server", e);
			future.setFailedState();
			future.fireCompleteEvent();
			terminateConnection();
		}
	}

	/**
	 * Handle compression and package chunking
	 * @param requestPackage
	 * @param future
	 */
	private void writePackageWithCompression(final NetworkRequestPackage requestPackage,
			final NetworkOperationFuture future) {

		boolean queueFull = false;

		synchronized (pendingCompressionPackages) {
			pendingCompressionPackages.add(requestPackage);
			queueFull = pendingCompressionPackages.size() >= Const.MAX_UNCOMPRESSED_QUEUE_SIZE;
		}

		if(queueFull || requestPackage.needsImmediateFlush()) {
			flushPendingCompressionPackages();
		}
	}

	/**
	 * Write all pending compression packages to server, called by the maintenance thread
	 *
	 */
	public void flushPendingCompressionPackages() {

		final List packagesToWrite = new ArrayList<>();

		synchronized (pendingCompressionPackages) {
			if(pendingCompressionPackages.isEmpty()) {
				return;
			}

			packagesToWrite.addAll(pendingCompressionPackages);
			pendingCompressionPackages.clear();
		}

		if(logger.isDebugEnabled()) {
			logger.debug("Chunk size is: {}", packagesToWrite.size());
		}

		final NetworkRequestPackage compressionEnvelopeRequest
			= new CompressionEnvelopeRequest(NetworkConst.COMPRESSION_TYPE_GZIP, packagesToWrite);

		try {
			writePackageToSocket(compressionEnvelopeRequest);
		} catch (PackageEncodeException | IOException e) {
			logger.error("Got an exception while write pending compression packages to server", e);
			terminateConnection();
		}
	}

	/**
	 * Write the package onto the socket
	 * @param requestPackage
	 * @throws PackageEncodeException
	 * @throws IOException
	 */
	private void writePackageToSocket(final NetworkRequestPackage requestPackage)
			throws PackageEncodeException, IOException {

		synchronized (outputStream) {
			requestPackage.writeToOutputStream(outputStream);
			outputStream.flush();
		}

		// Could be null during handshake
		if(mainteinanceHandler != null) {
			mainteinanceHandler.updateLastDataSendTimestamp();
		}
	}

	/**
	 * Register a new package callback
	 * @param requestPackage
	 * @param future
	 * @return
	 */
	public short registerPackageCallback(final NetworkRequestPackage requestPackage,
			final NetworkOperationFuture future) {

		final short sequenceNumber = requestPackage.getSequenceNumber();

		synchronized (pendingCalls) {
			assert (! pendingCalls.containsKey(sequenceNumber))
				: "Old call exists: " + pendingCalls.get(sequenceNumber);

			pendingCalls.put(sequenceNumber, future);
		}

		try {
			synchronized (pendingCalls) {
				// Ensure that not more then maxInFlightCalls are active
				while(pendingCalls.size() > maxInFlightCalls) {
					pendingCalls.wait();
				}
			}

		} catch(InterruptedException e) {
			logger.warn("Got an exception while waiting for pending requests", e);
			Thread.currentThread().interrupt();
		}

		return sequenceNumber;
	}

	/**
	 * Handle the next result package
	 * @param packageHeader
	 * @throws PackageEncodeException
	 */
	public void handleResultPackage(final ByteBuffer encodedPackage) throws PackageEncodeException {
		final short sequenceNumber = NetworkPackageDecoder.getRequestIDFromResponsePackage(encodedPackage);
		final short packageType = NetworkPackageDecoder.getPackageTypeFromResponse(encodedPackage);

		NetworkOperationFuture future = null;

		synchronized (pendingCalls) {
			future = pendingCalls.get(Short.valueOf(sequenceNumber));
		}

		if(! serverResponseHandler.containsKey(packageType)) {
			logger.error("Unknown respose package type: {}", packageType);
			removeFutureAndReleaseSequencenumber(sequenceNumber);

			if(future != null) {
				future.setFailedState();
				future.fireCompleteEvent();
			}

		} else {
			final ServerResponseHandler handler = serverResponseHandler.get(packageType);
			final boolean removeFuture = handler.handleServerResult(this, encodedPackage, future);

			// Remove pending call
			if(removeFuture) {
				removeFutureAndReleaseSequencenumber(sequenceNumber);
			}
		}
	}

	/**
	 * Remove the future from list and release the sequence number
	 *
	 * @param sequenceNumber
	 */
	private void removeFutureAndReleaseSequencenumber(final short sequenceNumber) {
		synchronized (pendingCalls) {
			sequenceNumberGenerator.releaseNumber(sequenceNumber);
			pendingCalls.remove(Short.valueOf(sequenceNumber));
			pendingCalls.notifyAll();
		}
	}


	/**
	 * Read the full package
	 * @param packageHeader
	 * @param inputStream2
	 * @return
	 * @throws IOException
	 */
	public ByteBuffer readFullPackage(final ByteBuffer packageHeader,
			final InputStream inputStream) throws IOException {

		final int bodyLength = (int) NetworkPackageDecoder.getBodyLengthFromResponsePackage(packageHeader);
		final int headerLength = packageHeader.limit();
		final ByteBuffer encodedPackage = ByteBuffer.allocate(headerLength + bodyLength);

		//System.out.println("Trying to read: " + bodyLength + " avail " + inputStream.available());
		encodedPackage.put(packageHeader.array());
		ByteStreams.readFully(inputStream, encodedPackage.array(), encodedPackage.position(), bodyLength);

		return encodedPackage;
	}

	@Override
	public String toString() {
		return "BBoxDBClient [serverHostname=" + serverAddress.getHostString()
		+ ", serverPort=" + serverAddress.getPort() + ", pendingCalls="
		+ pendingCalls.size() + ", connectionState=" + connectionState + "]";
	}

	/**
	 * Get the capabilities (e.g. gzip compression) of the client
	 */
	public PeerCapabilities getConnectionCapabilities() {
		return connectionCapabilities;
	}

	/**
	 * Get the capabilities of the client
	 * @return
	 */
	public PeerCapabilities getClientCapabilities() {
		return clientCapabilities;
	}

	/**
	 * Get the result buffer
	 * @return
	 */
	public Map> getResultBuffer() {
		return resultBuffer;
	}

	/**
	 * Get the server response reader
	 * @return
	 */
	public ServerResponseReader getServerResponseReader() {
		return serverResponseReader;
	}

	/**
	 * Get the server address
	 * @return
	 */
	public InetSocketAddress getServerAddress() {
		return serverAddress;
	}

	/**
	 * Get the BBoxDB Client
	 * @return
	 */
	public BBoxDBClient getBboxDBClient() {
		return bboxDBClient;
	}

	/**
	 * Get the input stream
	 *
	 * @return
	 */
	public BufferedInputStream getInputStream() {
		return inputStream;
	}
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy