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

reactor.net.tcp.spec.TcpClientSpec Maven / Gradle / Ivy

There is a newer version: 1.1.6.RELEASE
Show newest version
package reactor.net.tcp.spec;

import reactor.core.Environment;
import reactor.core.Reactor;
import reactor.core.spec.support.EventRoutingComponentSpec;
import reactor.function.Consumer;
import reactor.io.Buffer;
import reactor.io.encoding.Codec;
import reactor.net.NetChannel;
import reactor.net.config.ClientSocketOptions;
import reactor.net.config.SslOptions;
import reactor.net.tcp.TcpClient;
import reactor.util.Assert;

import javax.annotation.Nonnull;
import javax.annotation.Nullable;
import java.lang.reflect.Constructor;
import java.net.InetSocketAddress;
import java.util.*;

/**
 * A helper class for specifying a {@code TcpClient}
 *
 * @param 
 * 		The type that will be received by the client
 * @param 
 * 		The type that will be sent by the client
 *
 * @author Jon Brisbin
 */
public class TcpClientSpec extends EventRoutingComponentSpec, TcpClient> {

	private final Constructor> clientImplConstructor;

	private InetSocketAddress connectAddress;
	private ClientSocketOptions      options    = new ClientSocketOptions();
	private Collection> consumers  = Collections.emptyList();
	private SslOptions               sslOptions = null;
	private Codec codec;

	/**
	 * Create a {@code TcpClient.Spec} using the given implementation class.
	 *
	 * @param clientImpl
	 * 		The concrete implementation of {@link TcpClient} to instantiate.
	 */
	@SuppressWarnings({"unchecked", "rawtypes"})
	public TcpClientSpec(@Nonnull Class clientImpl) {
		Assert.notNull(clientImpl, "TcpClient implementation class cannot be null.");
		try {
			this.clientImplConstructor = (Constructor>)clientImpl.getDeclaredConstructor(
					Environment.class,
					Reactor.class,
					InetSocketAddress.class,
					ClientSocketOptions.class,
					SslOptions.class,
					Codec.class,
					Collection.class
			);
			this.clientImplConstructor.setAccessible(true);
		} catch(NoSuchMethodException e) {
			throw new IllegalArgumentException(
					"No public constructor found that matches the signature of the one found in the TcpClient class.");
		}
	}

	/**
	 * Set the common {@link ClientSocketOptions} for connections made in this client.
	 *
	 * @param options
	 * 		The socket options to apply to new connections.
	 *
	 * @return {@literal this}
	 */
	public TcpClientSpec options(ClientSocketOptions options) {
		this.options = options;
		return this;
	}

	/**
	 * Set the options to use for configuring SSL. Setting this to {@code null} means don't use SSL at all (the default).
	 *
	 * @param sslOptions
	 * 		The options to set when configuring SSL
	 *
	 * @return {@literal this}
	 */
	public TcpClientSpec ssl(@Nullable SslOptions sslOptions) {
		this.sslOptions = sslOptions;
		return this;
	}

	/**
	 * The host and port to which this client should connect.
	 *
	 * @param host
	 * 		The host to connect to.
	 * @param port
	 * 		The port to connect to.
	 *
	 * @return {@literal this}
	 */
	public TcpClientSpec connect(@Nonnull String host, int port) {
		return connect(new InetSocketAddress(host, port));
	}

	/**
	 * The address to which this client should connect.
	 *
	 * @param connectAddress
	 * 		The address to connect to.
	 *
	 * @return {@literal this}
	 */
	public TcpClientSpec connect(@Nonnull InetSocketAddress connectAddress) {
		Assert.isNull(this.connectAddress, "Connect address is already set.");
		this.connectAddress = connectAddress;
		return this;
	}

	/**
	 * The {@link Codec} to use to encode and decode data.
	 *
	 * @param codec
	 * 		The codec to use.
	 *
	 * @return {@literal this}
	 */
	public TcpClientSpec codec(@Nullable Codec codec) {
		Assert.isNull(this.codec, "Codec has already been set.");
		this.codec = codec;
		return this;
	}

	/**
	 * The {@link reactor.function.Consumer} to use to ingest incoming data.
	 *
	 * @param consumer
	 * 		the incoming data {@link reactor.function.Consumer}
	 *
	 * @return {@literal this}
	 */
	@SuppressWarnings("unchecked")
	public TcpClientSpec consume(Consumer consumer) {
		consume(Arrays.asList(consumer));
		return this;
	}

	/**
	 * The {@link reactor.function.Consumer Consumers} to use to ingest the incoming data.
	 *
	 * @param consumers
	 * 		the incoming data {@link reactor.function.Consumer Consumers}
	 *
	 * @return {@literal this}
	 */
	public TcpClientSpec consume(Collection> consumers) {
		Assert.notNull(consumers, "Consumers cannot be null");
		this.consumers = consumers;
		return this;
	}

	@Override
	protected TcpClient configure(Reactor reactor, Environment env) {
		List>> channelConsumers = new ArrayList>>();
		channelConsumers.add(new Consumer>() {
			@Override
			public void accept(NetChannel netChannel) {
				for(Consumer consumer : consumers) {
					netChannel.consume(consumer);
				}
			}
		});
		try {
			return clientImplConstructor.newInstance(
					env,
					reactor,
					connectAddress,
					options,
					sslOptions,
					codec,
					channelConsumers
			);
		} catch(Throwable t) {
			throw new IllegalStateException(t);
		}
	}

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy