/* * Copyright 2015 Netflix, Inc. * * 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 io.reactivex.netty.protocol.tcp.server; import io.netty.buffer.ByteBuf; import io.netty.buffer.ByteBufAllocator; import io.netty.channel.ChannelHandler; import io.netty.channel.ChannelOption; import io.netty.channel.ChannelPipeline; import io.netty.channel.EventLoopGroup; import io.netty.channel.ServerChannel; import io.netty.handler.logging.LogLevel; import io.netty.handler.ssl.SslHandler; import io.netty.util.concurrent.EventExecutorGroup; import io.reactivex.netty.events.EventSource; import io.reactivex.netty.protocol.tcp.server.events.TcpServerEventListener; import io.reactivex.netty.protocol.tcp.server.events.TcpServerEventPublisher; import io.reactivex.netty.ssl.SslCodec; import rx.functions.Action1; import rx.functions.Func0; import rx.functions.Func1; import javax.net.ssl.SSLEngine; import java.net.InetSocketAddress; import java.net.SocketAddress; import java.util.concurrent.TimeUnit; /** * A TCP server. * * <h2>Immutability</h2> * An instance of this server is immutable and all mutations produce a new server instance. * * @param <R> The type of objects read from this server. * @param <W> The type of objects written to this server. */ public abstract class TcpServer<R, W> implements EventSource<TcpServerEventListener> { /** * Creates a new server instance, inheriting all configurations from this server and adding a * {@link ChannelOption} for the server socket created by the newly created server instance. * * @param option Option to add. * @param value Value for the option. * * @return A new {@link TcpServer} instance. */ public abstract <T> TcpServer<R, W> channelOption(ChannelOption<T> option, T value); /** * Creates a new server instance, inheriting all configurations from this server and adding a * {@link ChannelOption} for the client socket created by the newly created server instance. * * @param option Option to add. * @param value Value for the option. * * @return A new {@link TcpServer} instance. */ public abstract <T> TcpServer<R, W> clientChannelOption(ChannelOption<T> option, T value); /** * Adds a {@link ChannelHandler} to {@link ChannelPipeline} for all connections created by this server. * The specified handler is added at the first position of the pipeline as specified by * {@link ChannelPipeline#addFirst(String, ChannelHandler)} * * <em>For better flexibility of pipeline modification, the method {@link #pipelineConfigurator(Action1)} * will be more convenient.</em> * * @param name Name of the handler. * @param handlerFactory Factory to create handler instance to add. * * @return A new {@link TcpServer} instance. */ public abstract <RR, WW> TcpServer<RR, WW> addChannelHandlerFirst(String name, Func0<ChannelHandler> handlerFactory); /** * Adds a {@link ChannelHandler} to {@link ChannelPipeline} for all connections created by this server. The specified * handler is added at the first position of the pipeline as specified by * {@link ChannelPipeline#addFirst(EventExecutorGroup, String, ChannelHandler)} * * <em>For better flexibility of pipeline modification, the method {@link #pipelineConfigurator(Action1)} will be * more convenient.</em> * * @param group The {@link EventExecutorGroup} which will be used to execute the {@link ChannelHandler} * methods * @param name The name of the handler to append * @param handlerFactory Factory to create handler instance to add. * * @return A new {@link TcpServer} instance. */ public abstract <RR, WW> TcpServer<RR, WW> addChannelHandlerFirst(EventExecutorGroup group, String name, Func0<ChannelHandler> handlerFactory); /** * Adds a {@link ChannelHandler} to {@link ChannelPipeline} for all connections created by this server. The specified * handler is added at the last position of the pipeline as specified by * {@link ChannelPipeline#addLast(String, ChannelHandler)} * * <em>For better flexibility of pipeline modification, the method {@link #pipelineConfigurator(Action1)} will be * more convenient.</em> * * @param name Name of the handler. * @param handlerFactory Factory to create handler instance to add. * * @return A new {@link TcpServer} instance. */ public abstract <RR, WW> TcpServer<RR, WW> addChannelHandlerLast(String name, Func0<ChannelHandler> handlerFactory); /** * Adds a {@link ChannelHandler} to {@link ChannelPipeline} for all connections created by this server. The specified * handler is added at the last position of the pipeline as specified by * {@link ChannelPipeline#addLast(EventExecutorGroup, String, ChannelHandler)} * * <em>For better flexibility of pipeline modification, the method {@link #pipelineConfigurator(Action1)} will be more * convenient.</em> * * @param group the {@link EventExecutorGroup} which will be used to execute the {@link ChannelHandler} * methods * @param name the name of the handler to append * @param handlerFactory Factory to create handler instance to add. * * @return A new {@link TcpServer} instance. */ public abstract <RR, WW> TcpServer<RR, WW> addChannelHandlerLast(EventExecutorGroup group, String name, Func0<ChannelHandler> handlerFactory); /** * Adds a {@link ChannelHandler} to {@link ChannelPipeline} for all connections created by this server. The specified * handler is added before an existing handler with the passed {@code baseName} in the pipeline as specified by * {@link ChannelPipeline#addBefore(String, String, ChannelHandler)} * * <em>For better flexibility of pipeline modification, the method {@link #pipelineConfigurator(Action1)} will be more * convenient.</em> * * @param baseName the name of the existing handler * @param name Name of the handler. * @param handlerFactory Factory to create handler instance to add. * * @return A new {@link TcpServer} instance. */ public abstract <RR, WW> TcpServer<RR, WW> addChannelHandlerBefore(String baseName, String name, Func0<ChannelHandler> handlerFactory); /** * Adds a {@link ChannelHandler} to {@link ChannelPipeline} for all connections created by this server. The specified * handler is added before an existing handler with the passed {@code baseName} in the pipeline as specified by * {@link ChannelPipeline#addBefore(EventExecutorGroup, String, String, ChannelHandler)} * * <em>For better flexibility of pipeline modification, the method {@link #pipelineConfigurator(Action1)} will be more * convenient.</em> * * @param group the {@link EventExecutorGroup} which will be used to execute the {@link ChannelHandler} * methods * @param baseName the name of the existing handler * @param name the name of the handler to append * @param handlerFactory Factory to create handler instance to add. * * @return A new {@link TcpServer} instance. */ public abstract <RR, WW> TcpServer<RR, WW> addChannelHandlerBefore(EventExecutorGroup group, String baseName, String name, Func0<ChannelHandler> handlerFactory); /** * Adds a {@link ChannelHandler} to {@link ChannelPipeline} for all connections created by this server. The specified * handler is added after an existing handler with the passed {@code baseName} in the pipeline as specified by * {@link ChannelPipeline#addAfter(String, String, ChannelHandler)} * * <em>For better flexibility of pipeline modification, the method {@link #pipelineConfigurator(Action1)} will be more * convenient.</em> * * @param baseName the name of the existing handler * @param name Name of the handler. * @param handlerFactory Factory to create handler instance to add. * * @return A new {@link TcpServer} instance. */ public abstract <RR, WW> TcpServer<RR, WW> addChannelHandlerAfter(String baseName, String name, Func0<ChannelHandler> handlerFactory); /** * Adds a {@link ChannelHandler} to {@link ChannelPipeline} for all connections created by this server. The specified * handler is added after an existing handler with the passed {@code baseName} in the pipeline as specified by * {@link ChannelPipeline#addAfter(EventExecutorGroup, String, String, ChannelHandler)} * * <em>For better flexibility of pipeline modification, the method {@link #pipelineConfigurator(Action1)} will be more * convenient.</em> * * @param group the {@link EventExecutorGroup} which will be used to execute the {@link ChannelHandler} * methods * @param baseName the name of the existing handler * @param name the name of the handler to append * @param handlerFactory Factory to create handler instance to add. * * @return A new {@link TcpServer} instance. */ public abstract <RR, WW> TcpServer<RR, WW> addChannelHandlerAfter(EventExecutorGroup group, String baseName, String name, Func0<ChannelHandler> handlerFactory); /** * Creates a new server instances, inheriting all configurations from this server and using the passed * action to configure all the connections created by the newly created server instance. * * @param pipelineConfigurator Action to configure {@link ChannelPipeline}. * * @return A new {@link TcpServer} instance. */ public abstract <RR, WW> TcpServer<RR, WW> pipelineConfigurator(Action1<ChannelPipeline> pipelineConfigurator); /** * Creates a new server instances, inheriting all configurations from this server and using the passed * {@code sslEngineFactory} for all secured connections accepted by the newly created server instance. * * If the {@link SSLEngine} instance can be statically, created, {@link #secure(SSLEngine)} can be used. * * @param sslEngineFactory Factory for all secured connections created by the newly created server instance. * * @return A new {@link TcpServer} instance. */ public abstract TcpServer<R, W> secure(Func1<ByteBufAllocator, SSLEngine> sslEngineFactory); /** * Creates a new server instances, inheriting all configurations from this server and using the passed * {@code sslEngine} for all secured connections accepted by the newly created server instance. * * If the {@link SSLEngine} instance can not be statically, created, {@link #secure(Func1)} )} can be used. * * @param sslEngine {@link SSLEngine} for all secured connections created by the newly created server instance. * * @return A new {@link TcpServer} instance. */ public abstract TcpServer<R, W> secure(SSLEngine sslEngine); /** * Creates a new server instances, inheriting all configurations from this server and using the passed * {@code sslCodec} for all secured connections accepted by the newly created server instance. * * This is required only when the {@link SslHandler} used by {@link SslCodec} is to be modified before adding to * the {@link ChannelPipeline}. For most of the cases, {@link #secure(Func1)} or {@link #secure(SSLEngine)} will be * enough. * * @param sslCodec {@link SslCodec} for all secured connections created by the newly created server instance. * * @return A new {@link TcpServer} instance. */ public abstract TcpServer<R, W> secure(SslCodec sslCodec); /** * Creates a new server instances, inheriting all configurations from this server and using a self-signed * certificate for all secured connections accepted by the newly created server instance. * * <b>This is only for testing and should not be used for real production servers.</b> * * @return A new {@link TcpServer} instance. */ public abstract TcpServer<R, W> unsafeSecure(); /** * Creates a new server instances, inheriting all configurations from this server and enabling wire logging at the * passed level for the newly created server instance. * * @param wireLoggingLevel Logging level at which the wire logs will be logged. The wire logging will only be done if * logging is enabled at this level for {@link io.netty.handler.logging.LoggingHandler} * * @return A new {@link TcpServer} instance. * * @deprecated Use {@link #enableWireLogging(String, LogLevel)} instead. */ @Deprecated public abstract TcpServer<R, W> enableWireLogging(LogLevel wireLoggingLevel); /** * Creates a new server instances, inheriting all configurations from this server and enabling wire logging at the * passed level for the newly created server instance. * * @param name Name of the logger that can be used to control the logging dynamically. * @param wireLoggingLevel Logging level at which the wire logs will be logged. The wire logging will only be done if * logging is enabled at this level for {@link io.netty.handler.logging.LoggingHandler} * * @return A new {@link TcpServer} instance. */ public abstract TcpServer<R, W> enableWireLogging(String name, LogLevel wireLoggingLevel); /** * Returns the port at which this server is running. * * For servers using ephemeral ports, this would return the actual port used, only after the server is started. * * @return The port at which this server is running. */ public abstract int getServerPort(); /** * Returns the address at which this server is running. * * @return The address at which this server is running. */ public abstract SocketAddress getServerAddress(); /** * Starts this server. * * @param connectionHandler Connection handler that will handle any new server connections to this server. * * @return This server. */ public abstract TcpServer<R, W> start(ConnectionHandler<R, W> connectionHandler); /** * Shutdown this server and waits till the server socket is closed. */ public abstract void shutdown(); /** * Waits for the shutdown of this server. * * <b>This does not actually shutdown the server.</b> It just waits for some other action to shutdown. */ public abstract void awaitShutdown(); /** * Waits for the shutdown of this server, waiting a maximum of the passed duration. * * <b>This does not actually shutdown the server.</b> It just waits for some other action to shutdown. * * @param duration Duration to wait for shutdown. * @param timeUnit Timeunit for the duration to wait for shutdown. */ public abstract void awaitShutdown(long duration, TimeUnit timeUnit); /** * Returns the event publisher for this server. * * @return The event publisher for this server. */ public abstract TcpServerEventPublisher getEventPublisher(); /** * Creates a new server using an ephemeral port. The port used can be queried after starting this server, using * {@link #getServerPort()} * * @return A new {@link TcpServer} */ public static TcpServer<ByteBuf, ByteBuf> newServer() { return newServer(0); } /** * Creates a new server using the passed port. * * @param port Port for the server. {@code 0} to use ephemeral port. * @return A new {@link TcpServer} */ public static TcpServer<ByteBuf, ByteBuf> newServer(int port) { return new TcpServerImpl<>(new InetSocketAddress(port)); } /** * Creates a new server using the passed port. * * @param port Port for the server. {@code 0} to use ephemeral port. * @param eventLoopGroup Eventloop group to be used for server as well as client sockets. * @param channelClass The class to be used for server channel. * * @return A new {@link TcpServer} */ public static TcpServer<ByteBuf, ByteBuf> newServer(int port, EventLoopGroup eventLoopGroup, Class<? extends ServerChannel> channelClass) { return newServer(port, eventLoopGroup, eventLoopGroup, channelClass); } /** * Creates a new server using the passed port. * * @param port Port for the server. {@code 0} to use ephemeral port. * @param acceptGroup Eventloop group to be used for server sockets. * @param clientGroup Eventloop group to be used for client sockets. * @param channelClass The class to be used for server channel. * * @return A new {@link TcpServer} */ public static TcpServer<ByteBuf, ByteBuf> newServer(int port, EventLoopGroup acceptGroup, EventLoopGroup clientGroup, Class<? extends ServerChannel> channelClass) { return newServer(new InetSocketAddress(port), acceptGroup, clientGroup, channelClass); } /** * Creates a new server using the passed address. * * @param socketAddress Socket address for the server. * @return A new {@link TcpServer} */ public static TcpServer<ByteBuf, ByteBuf> newServer(SocketAddress socketAddress) { return new TcpServerImpl<>(socketAddress); } /** * Creates a new server using the passed address. * * @param socketAddress Socket address for the server. * @param eventLoopGroup Eventloop group to be used for server as well as client sockets. * @param channelClass The class to be used for server channel. * * @return A new {@link TcpServer} */ public static TcpServer<ByteBuf, ByteBuf> newServer(SocketAddress socketAddress, EventLoopGroup eventLoopGroup, Class<? extends ServerChannel> channelClass) { return new TcpServerImpl<>(socketAddress, eventLoopGroup, eventLoopGroup, channelClass); } /** * Creates a new server using the passed address. * * @param socketAddress Socket address for the server. * @param acceptGroup Eventloop group to be used for server sockets. * @param clientGroup Eventloop group to be used for client sockets. * @param channelClass The class to be used for server channel. * * @return A new {@link TcpServer} */ public static TcpServer<ByteBuf, ByteBuf> newServer(SocketAddress socketAddress, EventLoopGroup acceptGroup, EventLoopGroup clientGroup, Class<? extends ServerChannel> channelClass) { return new TcpServerImpl<>(socketAddress, acceptGroup, clientGroup, channelClass); } }