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

com.subgraph.orchid.connections.ConnectionCacheImpl Maven / Gradle / Ivy

package com.subgraph.orchid.connections;

import java.io.IOException;
import java.io.PrintWriter;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.Callable;
import java.util.concurrent.CancellationException;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.Executors;
import java.util.concurrent.Future;
import java.util.concurrent.FutureTask;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.TimeUnit;
import java.util.logging.Logger;

import javax.net.ssl.SSLSocket;

import com.subgraph.orchid.Connection;
import com.subgraph.orchid.ConnectionCache;
import com.subgraph.orchid.ConnectionFailedException;
import com.subgraph.orchid.ConnectionHandshakeException;
import com.subgraph.orchid.ConnectionTimeoutException;
import com.subgraph.orchid.Router;
import com.subgraph.orchid.TorConfig;
import com.subgraph.orchid.circuits.TorInitializationTracker;
import com.subgraph.orchid.dashboard.DashboardRenderable;
import com.subgraph.orchid.dashboard.DashboardRenderer;

public class ConnectionCacheImpl implements ConnectionCache, DashboardRenderable {
	private final static Logger logger = Logger.getLogger(ConnectionCacheImpl.class.getName());
	
	private class ConnectionTask implements Callable {

		private final Router router;
		private final boolean isDirectoryConnection;
		
		ConnectionTask(Router router, boolean isDirectoryConnection) {
			this.router = router;
			this.isDirectoryConnection = isDirectoryConnection;
		}

		public ConnectionImpl call() throws Exception {
			final SSLSocket socket = factory.createSocket();
			final ConnectionImpl conn = new ConnectionImpl(config, socket, router, initializationTracker, isDirectoryConnection);
			conn.connect();
			return conn;
		}
	}
	
	private class CloseIdleConnectionCheckTask implements Runnable {
		public void run() {
			for(Future f: activeConnections.values()) {
				if(f.isDone()) {
					try {
						final ConnectionImpl c = f.get();
						c.idleCloseCheck();
					} catch (Exception e) { }
				}
			}
		}
	}

	private final ConcurrentMap> activeConnections = new ConcurrentHashMap>();
	private final ConnectionSocketFactory factory = new ConnectionSocketFactory();
	private final ScheduledExecutorService scheduledExecutor = Executors.newSingleThreadScheduledExecutor();

	private final TorConfig config;
	private final TorInitializationTracker initializationTracker;
	private volatile boolean isClosed;

	
	public ConnectionCacheImpl(TorConfig config, TorInitializationTracker tracker) {
		this.config = config;
		this.initializationTracker = tracker;
		scheduledExecutor.scheduleAtFixedRate(new CloseIdleConnectionCheckTask(), 5000, 5000, TimeUnit.MILLISECONDS);
	}

	public void close() {
		if(isClosed) {
			return;
		}
		isClosed = true;
		for(Future f: activeConnections.values()) {
			if(f.isDone()) {
				try {
					ConnectionImpl conn = f.get();
					conn.closeSocket();
				} catch (InterruptedException e) {
					logger.warning("Unexpected interruption while closing connection");
				} catch (ExecutionException e) {
					logger.warning("Exception closing connection: "+ e.getCause());
				}
			} else {
				// FIXME this doesn't close the socket, so the connection task lingers
				// A proper fix would require maintaining pending connections in a separate
				// collection.
				f.cancel(true);
			}
		}
		activeConnections.clear();
		scheduledExecutor.shutdownNow();
	}

	@Override
	public boolean isClosed() {
		return isClosed;
	}

	public Connection getConnectionTo(Router router, boolean isDirectoryConnection) throws InterruptedException, ConnectionTimeoutException, ConnectionFailedException, ConnectionHandshakeException {
		if(isClosed) {
			throw new IllegalStateException("ConnectionCache has been closed");
		}
		logger.fine("Get connection to "+ router.getAddress() + " "+ router.getOnionPort() + " " + router.getNickname());
		while(true) {
			Future f = getFutureFor(router, isDirectoryConnection);
			try {
				Connection c = f.get();
				if(c.isClosed()) {
					activeConnections.remove(router, f);
				} else {
					return c;
				}
			} catch (CancellationException e) {
				activeConnections.remove(router, f);
			} catch (ExecutionException e) {
				activeConnections.remove(router, f);
				final Throwable t = e.getCause();
				if(t instanceof ConnectionTimeoutException) {
					throw (ConnectionTimeoutException) t;
				} else if(t instanceof ConnectionFailedException) {
					throw (ConnectionFailedException) t;
				} else if(t instanceof ConnectionHandshakeException) {
					throw (ConnectionHandshakeException) t;
				}
				throw new RuntimeException("Unexpected exception: "+ e, e);
			}
		}
	}

	private Future getFutureFor(Router router, boolean isDirectoryConnection) {
		Future f = activeConnections.get(router);
		if(f != null) {
			return f;
		}
		return createFutureForIfAbsent(router, isDirectoryConnection);
	}

	private Future createFutureForIfAbsent(Router router, boolean isDirectoryConnection) {
		final Callable task = new ConnectionTask(router, isDirectoryConnection);
		final FutureTask futureTask = new FutureTask(task);
		
		final Future f = activeConnections.putIfAbsent(router, futureTask);
		if(f != null) {
			return f;
		}
		
		futureTask.run();
		return futureTask;
	}

	public void dashboardRender(DashboardRenderer renderer, PrintWriter writer, int flags) throws IOException {
		if((flags & DASHBOARD_CONNECTIONS) == 0) {
			return;
		}
		printDashboardBanner(writer, flags);
		for(Connection c: getActiveConnections()) {
			if(!c.isClosed()) {
				renderer.renderComponent(writer, flags, c);
			}
		}
		writer.println();
	}

	private void printDashboardBanner(PrintWriter writer, int flags) {
		final boolean verbose = (flags & DASHBOARD_CONNECTIONS_VERBOSE) != 0;
		if(verbose) {
			writer.println("[Connection Cache (verbose)]");
		} else {
			writer.println("[Connection Cache]");
		}
		writer.println();
	}

	List getActiveConnections() {
		final List cs = new ArrayList();
		for(Future future: activeConnections.values()) {
			addConnectionFromFuture(future, cs);
		}
		return cs;
	}

	private void addConnectionFromFuture(Future future, List connectionList) {
		try {
			if(future.isDone() && !future.isCancelled()) {
				connectionList.add(future.get());
			}
		} catch (InterruptedException e) {
			Thread.currentThread().interrupt();
		} catch (ExecutionException e) { }
	}
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy