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

reactor.net.zmq.tcp.ZeroMQTcpServer Maven / Gradle / Ivy

package reactor.net.zmq.tcp;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.zeromq.ZContext;
import org.zeromq.ZMQ;
import reactor.core.Environment;
import reactor.core.Reactor;
import reactor.core.composable.Deferred;
import reactor.core.composable.Promise;
import reactor.core.composable.spec.Promises;
import reactor.function.Consumer;
import reactor.io.Buffer;
import reactor.io.encoding.Codec;
import reactor.net.NetChannel;
import reactor.net.config.ServerSocketOptions;
import reactor.net.config.SslOptions;
import reactor.net.tcp.TcpServer;
import reactor.net.zmq.ZeroMQNetChannel;
import reactor.net.zmq.ZeroMQServerSocketOptions;
import reactor.net.zmq.ZeroMQWorker;
import reactor.support.NamedDaemonThreadFactory;
import reactor.util.Assert;
import reactor.util.UUIDUtils;

import javax.annotation.Nonnull;
import javax.annotation.Nullable;
import java.net.InetSocketAddress;
import java.util.Collection;
import java.util.UUID;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Future;

import static reactor.net.zmq.tcp.ZeroMQ.findSocketTypeName;

/**
 * @author Jon Brisbin
 */
public class ZeroMQTcpServer extends TcpServer {

	private final Logger log = LoggerFactory.getLogger(getClass());

	private final int                       ioThreadCount;
	private final ZeroMQServerSocketOptions zmqOpts;
	private final ExecutorService           threadPool;

	private volatile ZeroMQWorker worker;
	private volatile Future             workerFuture;

	public ZeroMQTcpServer(@Nonnull Environment env,
	                       @Nonnull Reactor reactor,
	                       @Nullable InetSocketAddress listenAddress,
	                       ServerSocketOptions options,
	                       SslOptions sslOptions,
	                       @Nullable Codec codec,
	                       @Nonnull Collection>> consumers) {
		super(env, reactor, listenAddress, options, sslOptions, codec, consumers);

		this.ioThreadCount = env.getProperty("reactor.zmq.ioThreadCount", Integer.class, 1);

		if (options instanceof ZeroMQServerSocketOptions) {
			this.zmqOpts = (ZeroMQServerSocketOptions) options;
		} else {
			this.zmqOpts = null;
		}

		this.threadPool = Executors.newCachedThreadPool(new NamedDaemonThreadFactory("zmq-server"));
	}

	@Override
	public TcpServer start(@Nullable final Runnable started) {
		Assert.isNull(worker, "This ZeroMQ server has already been started");

		UUID id = UUIDUtils.random();
		int socketType = (null != zmqOpts ? zmqOpts.socketType() : ZMQ.ROUTER);
		ZContext zmq = (null != zmqOpts ? zmqOpts.context() : null);
		this.worker = new ZeroMQWorker(id, socketType, ioThreadCount, zmq) {
			@Override
			protected void configure(ZMQ.Socket socket) {
				socket.setReceiveBufferSize(getOptions().rcvbuf());
				socket.setSendBufferSize(getOptions().sndbuf());
				socket.setBacklog(getOptions().backlog());
				if (getOptions().keepAlive()) {
					socket.setTCPKeepAlive(1);
				}
				if (null != zmqOpts && null != zmqOpts.socketConfigurer()) {
					zmqOpts.socketConfigurer().accept(socket);
				}
			}

			@Override
			protected void start(ZMQ.Socket socket) {
				String addr;
				if (null != zmqOpts && null != zmqOpts.listenAddresses()) {
					addr = zmqOpts.listenAddresses();
				} else {
					addr = "tcp://" + getListenAddress().getHostString() + ":" + getListenAddress().getPort();
				}
				if (log.isInfoEnabled()) {
					String type = findSocketTypeName(socket.getType());
					log.info("BIND: starting ZeroMQ {} socket on {}", type, addr);
				}

				socket.bind(addr);
				notifyStart(started);
			}

			@Override
			protected ZeroMQNetChannel select(Object id) {
				return (ZeroMQNetChannel) ZeroMQTcpServer.this.select(id);
			}
		};
		this.workerFuture = threadPool.submit(this.worker);

		return this;
	}

	@Override
	protected  NetChannel createChannel(C ioChannel) {
		return new ZeroMQNetChannel(
				getEnvironment(),
				getReactor(),
				getReactor().getDispatcher(),
				getCodec()
		);
	}

	@Override
	public Promise shutdown() {
		if (null == worker) {
			return Promises.error(new IllegalStateException("This ZeroMQ server has not been started")).get();
		}

		Deferred> d = Promises.defer(getEnvironment(), getReactor().getDispatcher());

		super.close(null);

		worker.shutdown();
		if (!workerFuture.isDone()) {
			workerFuture.cancel(true);
		}
		threadPool.shutdownNow();

		notifyShutdown();
		d.accept(true);


		return d.compose();
	}

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy