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

com.aoindustries.aoserv.daemon.client.AOServDaemonConnection Maven / Gradle / Ivy

/*
 * aoserv-daemon-client - Java client for the AOServ Daemon.
 * Copyright (C) 2001-2009, 2016, 2017, 2018, 2019, 2020  AO Industries, Inc.
 *     [email protected]
 *     7262 Bull Pen Cir
 *     Mobile, AL 36695
 *
 * This file is part of aoserv-daemon-client.
 *
 * aoserv-daemon-client is free software: you can redistribute it and/or modify
 * it under the terms of the GNU Lesser General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 *
 * aoserv-daemon-client is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public License
 * along with aoserv-daemon-client.  If not, see .
 */
package com.aoindustries.aoserv.daemon.client;

import com.aoindustries.aoserv.client.net.AppProtocol;
import com.aoindustries.collections.AoArrays;
import com.aoindustries.io.AOPool;
import com.aoindustries.io.stream.StreamableInput;
import com.aoindustries.io.stream.StreamableOutput;
import com.aoindustries.lang.EmptyArrays;
import java.io.BufferedInputStream;
import java.io.BufferedOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.InterruptedIOException;
import java.io.OutputStream;
import java.net.InetSocketAddress;
import java.net.Socket;
import java.util.concurrent.atomic.AtomicLong;
import java.util.logging.Level;
import javax.net.ssl.SSLSocketFactory;

/**
 * A connection between an AOServ Daemon Client and an AOServ Daemon.
 *
 * @author  AO Industries, Inc.
 */
final public class AOServDaemonConnection {

	/**
	 * The set of supported versions, with the most preferred versions first.
	 */
	private static final AOServDaemonProtocol.Version[] SUPPORTED_VERSIONS = {
		AOServDaemonProtocol.Version.VERSION_1_84_11,
		AOServDaemonProtocol.Version.VERSION_1_83_0,
		AOServDaemonProtocol.Version.VERSION_1_81_10,
		AOServDaemonProtocol.Version.VERSION_1_80_1,
		AOServDaemonProtocol.Version.VERSION_1_80_0,
		AOServDaemonProtocol.Version.VERSION_1_77
	};

	private static Socket connect(AOServDaemonConnector connector) throws IOException {
		if(connector.protocol.equals(AppProtocol.AOSERV_DAEMON)) {
			assert connector.port.getProtocol() == com.aoindustries.net.Protocol.TCP;
			Socket socket = new Socket();
			socket.setKeepAlive(true);
			socket.setSoLinger(true, AOPool.DEFAULT_SOCKET_SO_LINGER);
			socket.setTcpNoDelay(true);
			if(!connector.local_ip.isUnspecified()) socket.bind(new InetSocketAddress(connector.local_ip.toString(), 0));
			socket.connect(new InetSocketAddress(connector.hostname.toString(), connector.port.getPort()), AOPool.DEFAULT_CONNECT_TIMEOUT);
			if(Thread.interrupted()) throw new InterruptedIOException();
			return socket;
		} else if(connector.protocol.equals(AppProtocol.AOSERV_DAEMON_SSL)) {
			assert connector.port.getProtocol() == com.aoindustries.net.Protocol.TCP;
			if(connector.trustStore != null && !connector.trustStore.isEmpty()) System.setProperty("javax.net.ssl.trustStore", connector.trustStore);
			if(connector.trustStorePassword != null && !connector.trustStorePassword.isEmpty()) System.setProperty("javax.net.ssl.trustStorePassword", connector.trustStorePassword);
			SSLSocketFactory sslFact = (SSLSocketFactory)SSLSocketFactory.getDefault();
			if(Thread.interrupted()) throw new InterruptedIOException();
			Socket regSocket = new Socket();
			regSocket.setKeepAlive(true);
			regSocket.setSoLinger(true, AOPool.DEFAULT_SOCKET_SO_LINGER);
			regSocket.setTcpNoDelay(true);
			if(!connector.local_ip.isUnspecified()) regSocket.bind(new InetSocketAddress(connector.local_ip.toString(), 0));
			regSocket.connect(new InetSocketAddress(connector.hostname.toString(), connector.port.getPort()), AOPool.DEFAULT_CONNECT_TIMEOUT);
			if(Thread.interrupted()) throw new InterruptedIOException();
			Socket socket = sslFact.createSocket(regSocket, connector.hostname.toString(), connector.port.getPort(), true);
			if(Thread.interrupted()) throw new InterruptedIOException();
			return socket;
		} else throw new IllegalArgumentException("Unsupported protocol: "+connector.protocol);
	}

	/**
	 * Closes a socket while logging any {@link IOException} as a warning.
	 */
	private static void close(
		AOServDaemonConnector connector,
		InputStream i,
		OutputStream o,
		Socket s
	) {
		if(i != null) {
			try {
				i.close();
			} catch(IOException err) {
				connector.getLogger().log(Level.WARNING, null, err);
			}
		}
		if(o != null) {
			try {
				o.close();
			} catch(IOException err) {
				connector.getLogger().log(Level.WARNING, null, err);
			}
		}
		if(s != null) {
			try {
				s.close();
			} catch(IOException err) {
				connector.getLogger().log(Level.WARNING, null, err);
			}
		}
	}

	/**
	 * The connector that this connection is part of.
	 */
	private final AOServDaemonConnector connector;

	/**
	 * Keeps a flag of the connection status.
	 */
	private boolean isClosed = true;

	/**
	 * The socket to the server.
	 */
	private final Socket socket;

	/**
	 * The output stream to the server.
	 */
	private final StreamableOutput out;

	/**
	 * The input stream from the server.
	 */
	private final StreamableInput in;

	/**
	 * The negotiated protocol version.
	 */
	final AOServDaemonProtocol.Version protocolVersion;

	/**
	 * The first command sequence for this connection.
	 */
	//private final long startSeq;

	/**
	 * The next command sequence that will be sent.
	 */
	private final AtomicLong seq;

	/**
	 * Creates a new AOServConnection.
	 *
	 * // TODO: Once all daemons are running > version 1.77, can simplify this considerably
	 */
	protected AOServDaemonConnection(AOServDaemonConnector connector) throws IOException {
		Socket newSocket = null;
		StreamableOutput newOut = null;
		StreamableInput newIn = null;
		AOServDaemonProtocol.Version selectedVersion = null;
		long newStartSeq = 0;
		boolean successful = false;
		try {
			newSocket = connect(connector);
			newOut = new StreamableOutput(new BufferedOutputStream(newSocket.getOutputStream()));
			newIn = new StreamableInput(new BufferedInputStream(newSocket.getInputStream()));

			// Write the most preferred version
			newOut.writeUTF(SUPPORTED_VERSIONS[0].getVersion());
			// Then connector key
			newOut.writeNullUTF(connector.key);
			// Now write additional versions.
			// This is done in this order for backwards compatibility to protocol 1.77 that only supported a single version.
			{
				newOut.writeCompressedInt(SUPPORTED_VERSIONS.length - 1);
				for(int i = 1; i < SUPPORTED_VERSIONS.length; i++) {
					newOut.writeUTF(SUPPORTED_VERSIONS[i].getVersion());
				}
			}
			newOut.flush();

			// The first boolean will tell if the version is now allowed
			if(newIn.readBoolean()) {
				// Protocol > 1.77 accepted
				// Read if the connection is allowed
				if(!newIn.readBoolean()) throw new IOException("Connection not allowed.");
				// Read the selected protocol version
				selectedVersion = AOServDaemonProtocol.Version.getVersion(newIn.readUTF());
				assert selectedVersion != AOServDaemonProtocol.Version.VERSION_1_77;
				newStartSeq = newIn.readLong();
			} else {
				// When not allowed, the server will write the set of supported versions
				String preferredVersion = newIn.readUTF();
				String[] extraVersions;
				if(preferredVersion.equals(AOServDaemonProtocol.Version.VERSION_1_77.getVersion())) {
					// Server 1.77 only sends the single preferred version
					extraVersions = EmptyArrays.EMPTY_STRING_ARRAY;
				} else {
					int numExtraVersions = newIn.readCompressedInt();
					extraVersions = new String[numExtraVersions];
					for(int i = 0; i < numExtraVersions; i++) {
						extraVersions[i] = newIn.readUTF();
					}
				}
				if(
					preferredVersion.equals(AOServDaemonProtocol.Version.VERSION_1_77.getVersion())
					&& AoArrays.indexOf(SUPPORTED_VERSIONS, AOServDaemonProtocol.Version.VERSION_1_77) != -1
				) {
					// Reconnect as forced protocol 1.77, since we already sent extra output incompatible with 1.77
					close(connector, newIn, newOut, newSocket);
					newSocket = connect(connector);
					newOut = new StreamableOutput(new BufferedOutputStream(newSocket.getOutputStream()));
					newIn = new StreamableInput(new BufferedInputStream(newSocket.getInputStream()));
					newOut.writeUTF(AOServDaemonProtocol.Version.VERSION_1_77.getVersion());
					newOut.writeNullUTF(connector.key);
					newOut.flush();
					if(!newIn.readBoolean()) {
						String requiredVersion = newIn.readUTF();
						throw new IOException(
							"Unsupported protocol version requested.  Requested version "
								+ AOServDaemonProtocol.Version.VERSION_1_77.getVersion()
								+ ", server requires version "
								+ requiredVersion
								+ "."
						);
					}
					// Read if the connection is allowed
					if(!newIn.readBoolean()) throw new IOException("Connection not allowed.");
					// Selected protocol version is forced 1.77 for this reconnect
					selectedVersion = AOServDaemonProtocol.Version.VERSION_1_77;
					newStartSeq = 0;
				} else {
					StringBuilder message = new StringBuilder();
					message.append("No compatible protocol versions.  Client prefers version ");
					message.append(SUPPORTED_VERSIONS[0].getVersion());
					if(SUPPORTED_VERSIONS.length > 1) {
						message.append(", but also supports versions [");
						for(int i = 1; i < SUPPORTED_VERSIONS.length; i++) {
							if(i > 1) message.append(", ");
							message.append(SUPPORTED_VERSIONS[i].getVersion());
						}
						message.append(']');
					}
					message.append(".  Server prefers version ");
					message.append(preferredVersion);
					if(extraVersions != null && extraVersions.length > 0) {
						message.append(", but also supports versions [");
						for(int i = 0; i < extraVersions.length; i++) {
							if(i > 0) message.append(", ");
							message.append(extraVersions[i]);
						}
						message.append(']');
					}
					message.append('.');
					throw new IOException(message.toString());
				}
			}
			successful = true;
		} finally {
			if(!successful) {
				close(connector, newIn, newOut, newSocket);
			}
		}
		assert successful;
		assert newSocket != null;
		assert newOut != null;
		assert newIn != null;
		assert selectedVersion != null;
		this.connector = connector;
		this.isClosed = false;
		this.socket = newSocket;
		this.out = newOut;
		this.in = newIn;
		this.protocolVersion = selectedVersion;
		//this.startSeq = newStartSeq;
		this.seq = new AtomicLong(newStartSeq);
	}

	/**
	 * Closes this connection to the server
	 * so that a reconnect is forced in the
	 * future.
	 */
	public void close() {
		close(connector, in, out, socket);
		isClosed = true;
	}

	private long currentSeq;

	/**
	 * Begins a task and gets the stream to write to the server.
	 */
	public StreamableOutput getRequestOut(int taskCode) throws IOException {
		// Increment sequence
		currentSeq = seq.getAndIncrement();
		// Send command sequence
		if(protocolVersion.compareTo(AOServDaemonProtocol.Version.VERSION_1_80_0) >= 0) {
			out.writeLong(currentSeq);
		}
		out.writeCompressedInt(taskCode);
		return out;
	}

	/**
	 * Gets the stream to read from the server.
	 */
	public StreamableInput getResponseIn() throws IOException {
		// Verify server sends matching sequence
		if(protocolVersion.compareTo(AOServDaemonProtocol.Version.VERSION_1_80_1) >= 0) {
			long serverSeq = in.readLong();
			if(serverSeq != currentSeq) throw new IOException("Sequence mismatch: " + serverSeq + " != " + currentSeq);
		}
		return in;
	}

	/**
	 * Determines if this connection has been closed.
	 */
	boolean isClosed() {
		return isClosed;
	}
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy