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

com.netflix.zuul.netty.server.Server Maven / Gradle / Ivy

/*
 * Copyright 2018 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 com.netflix.zuul.netty.server;

import com.google.common.annotations.VisibleForTesting;
import com.google.common.base.Preconditions;
import com.netflix.appinfo.InstanceInfo;
import com.netflix.config.DynamicBooleanProperty;
import com.netflix.netty.common.CategorizedThreadFactory;
import com.netflix.netty.common.LeastConnsEventLoopChooserFactory;
import com.netflix.netty.common.metrics.EventLoopGroupMetrics;
import com.netflix.netty.common.status.ServerStatusManager;
import com.netflix.spectator.api.Registry;
import com.netflix.spectator.api.Spectator;
import com.netflix.spectator.api.patterns.PolledMeter;
import com.netflix.zuul.Attrs;
import com.netflix.zuul.monitoring.ConnCounter;
import com.netflix.zuul.monitoring.ConnTimer;
import io.netty.bootstrap.ServerBootstrap;
import io.netty.buffer.ByteBufAllocator;
import io.netty.buffer.ByteBufAllocatorMetric;
import io.netty.buffer.ByteBufAllocatorMetricProvider;
import io.netty.channel.Channel;
import io.netty.channel.ChannelFuture;
import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.ChannelInboundHandlerAdapter;
import io.netty.channel.ChannelInitializer;
import io.netty.channel.ChannelOption;
import io.netty.channel.DefaultSelectStrategyFactory;
import io.netty.channel.EventLoopGroup;
import io.netty.channel.ServerChannel;
import io.netty.channel.epoll.Epoll;
import io.netty.channel.epoll.EpollChannelOption;
import io.netty.channel.epoll.EpollEventLoopGroup;
import io.netty.channel.epoll.EpollServerSocketChannel;
import io.netty.channel.epoll.EpollSocketChannel;
import io.netty.channel.kqueue.KQueue;
import io.netty.channel.kqueue.KQueueEventLoopGroup;
import io.netty.channel.kqueue.KQueueServerSocketChannel;
import io.netty.channel.kqueue.KQueueSocketChannel;
import io.netty.channel.nio.NioEventLoopGroup;
import io.netty.channel.socket.nio.NioServerSocketChannel;
import io.netty.channel.socket.nio.NioSocketChannel;
import io.netty.incubator.channel.uring.IOUring;
import io.netty.incubator.channel.uring.IOUringEventLoopGroup;
import io.netty.incubator.channel.uring.IOUringServerSocketChannel;
import io.netty.incubator.channel.uring.IOUringSocketChannel;
import io.netty.util.AttributeKey;
import io.netty.util.concurrent.DefaultEventExecutorChooserFactory;
import io.netty.util.concurrent.EventExecutor;
import io.netty.util.concurrent.EventExecutorChooserFactory;
import io.netty.util.concurrent.ThreadPerTaskExecutor;
import java.net.InetSocketAddress;
import java.nio.channels.spi.SelectorProvider;
import java.util.ArrayList;
import java.util.Collections;
import java.util.HashMap;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.Executor;
import java.util.concurrent.ThreadFactory;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicReference;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

/**
 *
 * NOTE: Shout-out to LittleProxy which was great as a reference.
 *
 * User: michaels
 * Date: 11/8/14
 * Time: 8:39 PM
 */
public class Server {
    /**
     * This field is effectively a noop, as Epoll is enabled automatically if available.   This can be disabled by
     * using the {@link #FORCE_NIO} property.
     */
    @Deprecated
    public static final DynamicBooleanProperty USE_EPOLL =
            new DynamicBooleanProperty("zuul.server.netty.socket.epoll", false);

    /**
     * If {@code true}, The Zuul server will avoid autodetecting the transport type and use the default Java NIO
     * transport.
     */
    private static final DynamicBooleanProperty FORCE_NIO =
            new DynamicBooleanProperty("zuul.server.netty.socket.force_nio", false);

    private static final DynamicBooleanProperty FORCE_IO_URING =
            new DynamicBooleanProperty("zuul.server.netty.socket.force_io_uring", false);

    private static final Logger LOG = LoggerFactory.getLogger(Server.class);

    private static final DynamicBooleanProperty USE_LEASTCONNS_FOR_EVENTLOOPS =
            new DynamicBooleanProperty("zuul.server.eventloops.use_leastconns", false);

    private static final DynamicBooleanProperty MANUAL_DISCOVERY_STATUS =
            new DynamicBooleanProperty("zuul.server.netty.manual.discovery.status", true);

    private final EventLoopGroupMetrics eventLoopGroupMetrics;

    private final Thread jvmShutdownHook;
    private final Registry registry;
    private ServerGroup serverGroup;
    private final ClientConnectionsShutdown clientConnectionsShutdown;
    private final ServerStatusManager serverStatusManager;
    private final Map> addressesToInitializers;
    /**
     * Unlike the above, the socket addresses in this map are the *bound* addresses, rather than the requested ones.
     */
    private final Map addressesToChannels = new LinkedHashMap<>();

    private final EventLoopConfig eventLoopConfig;

    /**
     * This is a hack to expose the channel type to the origin channel.  It is NOT API stable and should not be
     * referenced by non-Zuul code.
     */
    @Deprecated
    public static final AtomicReference> defaultOutboundChannelType = new AtomicReference<>();

    /**
     * Use {@link #Server(Registry, ServerStatusManager, Map, ClientConnectionsShutdown, EventLoopGroupMetrics,
     * EventLoopConfig)}
     * instead.
     */
    @SuppressWarnings("rawtypes")
    @Deprecated
    public Server(
            Map portsToChannelInitializers,
            ServerStatusManager serverStatusManager,
            ClientConnectionsShutdown clientConnectionsShutdown,
            EventLoopGroupMetrics eventLoopGroupMetrics) {
        this(
                portsToChannelInitializers,
                serverStatusManager,
                clientConnectionsShutdown,
                eventLoopGroupMetrics,
                new DefaultEventLoopConfig());
    }

    /**
     * Use {@link #Server(Registry, ServerStatusManager, Map, ClientConnectionsShutdown, EventLoopGroupMetrics,
     * EventLoopConfig)}
     * instead.
     */
    @SuppressWarnings({"unchecked", "rawtypes"
    }) // Channel init map has the wrong generics and we can't fix without api breakage.
    @Deprecated
    public Server(
            Map portsToChannelInitializers,
            ServerStatusManager serverStatusManager,
            ClientConnectionsShutdown clientConnectionsShutdown,
            EventLoopGroupMetrics eventLoopGroupMetrics,
            EventLoopConfig eventLoopConfig) {
        this(
                Spectator.globalRegistry(),
                serverStatusManager,
                convertPortMap((Map>) (Map) portsToChannelInitializers),
                clientConnectionsShutdown,
                eventLoopGroupMetrics,
                eventLoopConfig);
    }

    public Server(
            Registry registry,
            ServerStatusManager serverStatusManager,
            Map> addressesToInitializers,
            ClientConnectionsShutdown clientConnectionsShutdown,
            EventLoopGroupMetrics eventLoopGroupMetrics,
            EventLoopConfig eventLoopConfig) {
        this.registry = Objects.requireNonNull(registry);
        this.addressesToInitializers = Collections.unmodifiableMap(new LinkedHashMap<>(addressesToInitializers));
        this.serverStatusManager = Preconditions.checkNotNull(serverStatusManager, "serverStatusManager");
        this.clientConnectionsShutdown =
                Preconditions.checkNotNull(clientConnectionsShutdown, "clientConnectionsShutdown");
        this.eventLoopConfig = Preconditions.checkNotNull(eventLoopConfig, "eventLoopConfig");
        this.eventLoopGroupMetrics = Preconditions.checkNotNull(eventLoopGroupMetrics, "eventLoopGroupMetrics");
        this.jvmShutdownHook = new Thread(this::stop, "Zuul-JVM-shutdown-hook");
    }

    public Server(
            Registry registry,
            ServerStatusManager serverStatusManager,
            Map> addressesToInitializers,
            ClientConnectionsShutdown clientConnectionsShutdown,
            EventLoopGroupMetrics eventLoopGroupMetrics,
            EventLoopConfig eventLoopConfig,
            Thread jvmShutdownHook) {
        this.registry = Objects.requireNonNull(registry);
        this.addressesToInitializers = Collections.unmodifiableMap(new LinkedHashMap<>(addressesToInitializers));
        this.serverStatusManager = Preconditions.checkNotNull(serverStatusManager, "serverStatusManager");
        this.clientConnectionsShutdown =
                Preconditions.checkNotNull(clientConnectionsShutdown, "clientConnectionsShutdown");
        this.eventLoopConfig = Preconditions.checkNotNull(eventLoopConfig, "eventLoopConfig");
        this.eventLoopGroupMetrics = Preconditions.checkNotNull(eventLoopGroupMetrics, "eventLoopGroupMetrics");
        this.jvmShutdownHook = jvmShutdownHook;
    }

    public void stop() {
        LOG.info("Shutting down Zuul.");
        serverGroup.stop();
        LOG.info("Completed zuul shutdown.");
    }

    public void start() {
        if (jvmShutdownHook != null) {
            Runtime.getRuntime().addShutdownHook(jvmShutdownHook);
        }

        serverGroup = new ServerGroup(
                "Salamander", eventLoopConfig.acceptorCount(), eventLoopConfig.eventLoopCount(), eventLoopGroupMetrics);
        serverGroup.initializeTransport();
        try {
            List allBindFutures = new ArrayList<>(addressesToInitializers.size());

            // Setup each of the channel initializers on requested ports.
            for (Map.Entry> entry :
                    addressesToInitializers.entrySet()) {
                NamedSocketAddress requestedNamedAddr = entry.getKey();
                ChannelFuture nettyServerFuture = setupServerBootstrap(requestedNamedAddr, entry.getValue());
                Channel chan = nettyServerFuture.channel();
                addressesToChannels.put(requestedNamedAddr.withNewSocket(chan.localAddress()), chan);
                allBindFutures.add(nettyServerFuture);
            }

            // All channels should share a single ByteBufAllocator instance.
            // Add metrics to monitor that allocator's memory usage.
            if (!allBindFutures.isEmpty()) {
                ByteBufAllocator alloc = allBindFutures.get(0).channel().alloc();
                if (alloc instanceof ByteBufAllocatorMetricProvider) {
                    ByteBufAllocatorMetric metrics = ((ByteBufAllocatorMetricProvider) alloc).metric();
                    PolledMeter.using(registry)
                            .withId(registry.createId("zuul.nettybuffermem.live", "type", "heap"))
                            .monitorValue(metrics, ByteBufAllocatorMetric::usedHeapMemory);
                    PolledMeter.using(registry)
                            .withId(registry.createId("zuul.nettybuffermem.live", "type", "direct"))
                            .monitorValue(metrics, ByteBufAllocatorMetric::usedDirectMemory);
                }
            }
        } catch (InterruptedException e) {
            Thread.currentThread().interrupt();
        }
    }

    public final void awaitTermination() throws InterruptedException {
        for (Channel chan : addressesToChannels.values()) {
            chan.closeFuture().sync();
        }
    }

    public final List getListeningAddresses() {
        if (serverGroup == null) {
            throw new IllegalStateException("Server has not been started");
        }
        return Collections.unmodifiableList(new ArrayList<>(addressesToChannels.keySet()));
    }

    @VisibleForTesting
    public void waitForEachEventLoop() throws InterruptedException, ExecutionException {
        for (EventExecutor exec : serverGroup.clientToProxyWorkerPool) {
            exec.submit(() -> {
                        // Do nothing.
                    })
                    .get();
        }
    }

    private ChannelFuture setupServerBootstrap(
            NamedSocketAddress listenAddress, ChannelInitializer channelInitializer) throws InterruptedException {
        ServerBootstrap serverBootstrap =
                new ServerBootstrap().group(serverGroup.clientToProxyBossPool, serverGroup.clientToProxyWorkerPool);

        LOG.info("Proxy listening with {}", serverGroup.channelType);
        serverBootstrap.channel(serverGroup.channelType);

        serverBootstrap.option(ChannelOption.SO_BACKLOG, 128);
        serverBootstrap.childOption(ChannelOption.SO_LINGER, -1);
        serverBootstrap.childOption(ChannelOption.TCP_NODELAY, true);
        serverBootstrap.childOption(ChannelOption.SO_KEEPALIVE, true);

        // Apply transport specific socket options.
        for (Map.Entry, ?> optionEntry : serverGroup.transportChannelOptions.entrySet()) {
            serverBootstrap = serverBootstrap.option((ChannelOption) optionEntry.getKey(), optionEntry.getValue());
        }

        serverBootstrap.handler(new NewConnHandler());
        serverBootstrap.childHandler(channelInitializer);
        serverBootstrap.validate();

        LOG.info("Binding to : {}", listenAddress);

        if (MANUAL_DISCOVERY_STATUS.get()) {
            // Flag status as UP just before binding to the port.
            serverStatusManager.localStatus(InstanceInfo.InstanceStatus.UP);
        }

        // Bind and start to accept incoming connections.
        ChannelFuture bindFuture = serverBootstrap.bind(listenAddress.unwrap());

        try {
            return bindFuture.sync();
        } catch (Exception e) {
            // sync() sneakily throws a checked Exception, but doesn't declare it. This can happen if there is a bind
            // failure, which is typically an IOException.  Just chain it and rethrow.
            throw new RuntimeException("Failed to bind on addr " + listenAddress, e);
        }
    }

    /**
     * Override for metrics or informational purposes
     *
     * @param clientToProxyBossPool - acceptor pool
     * @param clientToProxyWorkerPool - worker pool
     */
    public void postEventLoopCreationHook(
            EventLoopGroup clientToProxyBossPool, EventLoopGroup clientToProxyWorkerPool) {}

    private final class ServerGroup {
        /** A name for this ServerGroup to use in naming threads. */
        private final String name;

        private final int acceptorThreads;
        private final int workerThreads;
        private final EventLoopGroupMetrics eventLoopGroupMetrics;

        private EventLoopGroup clientToProxyBossPool;
        private EventLoopGroup clientToProxyWorkerPool;
        private Class channelType;
        private Map, ?> transportChannelOptions;

        private volatile boolean stopped = false;

        private ServerGroup(
                String name, int acceptorThreads, int workerThreads, EventLoopGroupMetrics eventLoopGroupMetrics) {
            this.name = name;
            this.acceptorThreads = acceptorThreads;
            this.workerThreads = workerThreads;
            this.eventLoopGroupMetrics = eventLoopGroupMetrics;

            Thread.setDefaultUncaughtExceptionHandler(new Thread.UncaughtExceptionHandler() {
                @Override
                public void uncaughtException(final Thread t, final Throwable e) {
                    LOG.error("Uncaught throwable", e);
                }
            });
        }

        private void initializeTransport() {
            // TODO - try our own impl of ChooserFactory that load-balances across the eventloops using leastconns algo?
            EventExecutorChooserFactory chooserFactory;
            if (USE_LEASTCONNS_FOR_EVENTLOOPS.get()) {
                chooserFactory = new LeastConnsEventLoopChooserFactory(eventLoopGroupMetrics);
            } else {
                chooserFactory = DefaultEventExecutorChooserFactory.INSTANCE;
            }

            ThreadFactory workerThreadFactory = new CategorizedThreadFactory(name + "-ClientToZuulWorker");
            Executor workerExecutor = new ThreadPerTaskExecutor(workerThreadFactory);

            Map, Object> extraOptions = new HashMap<>();
            final boolean useNio = FORCE_NIO.get();
            final boolean useIoUring = FORCE_IO_URING.get();
            if (useIoUring && ioUringIsAvailable()) {
                channelType = IOUringServerSocketChannel.class;
                defaultOutboundChannelType.set(IOUringSocketChannel.class);
                clientToProxyBossPool = new IOUringEventLoopGroup(
                        acceptorThreads, new CategorizedThreadFactory(name + "-ClientToZuulAcceptor"));
                clientToProxyWorkerPool = new IOUringEventLoopGroup(workerThreads, workerExecutor);
            } else if (!useNio && epollIsAvailable()) {
                channelType = EpollServerSocketChannel.class;
                defaultOutboundChannelType.set(EpollSocketChannel.class);
                extraOptions.put(EpollChannelOption.TCP_DEFER_ACCEPT, -1);
                clientToProxyBossPool = new EpollEventLoopGroup(
                        acceptorThreads, new CategorizedThreadFactory(name + "-ClientToZuulAcceptor"));
                clientToProxyWorkerPool = new EpollEventLoopGroup(
                        workerThreads, workerExecutor, chooserFactory, DefaultSelectStrategyFactory.INSTANCE);
            } else if (!useNio && kqueueIsAvailable()) {
                channelType = KQueueServerSocketChannel.class;
                defaultOutboundChannelType.set(KQueueSocketChannel.class);
                clientToProxyBossPool = new KQueueEventLoopGroup(
                        acceptorThreads, new CategorizedThreadFactory(name + "-ClientToZuulAcceptor"));
                clientToProxyWorkerPool = new KQueueEventLoopGroup(
                        workerThreads, workerExecutor, chooserFactory, DefaultSelectStrategyFactory.INSTANCE);
            } else {
                channelType = NioServerSocketChannel.class;
                defaultOutboundChannelType.set(NioSocketChannel.class);
                NioEventLoopGroup elg = new NioEventLoopGroup(
                        workerThreads,
                        workerExecutor,
                        chooserFactory,
                        SelectorProvider.provider(),
                        DefaultSelectStrategyFactory.INSTANCE);
                elg.setIoRatio(90);
                clientToProxyBossPool = new NioEventLoopGroup(
                        acceptorThreads, new CategorizedThreadFactory(name + "-ClientToZuulAcceptor"));
                clientToProxyWorkerPool = elg;
            }

            transportChannelOptions = Collections.unmodifiableMap(extraOptions);

            postEventLoopCreationHook(clientToProxyBossPool, clientToProxyWorkerPool);
        }

        private synchronized void stop() {
            LOG.info("Shutting down");
            if (stopped) {
                LOG.info("Already stopped");
                return;
            }

            if (MANUAL_DISCOVERY_STATUS.get()) {
                // Flag status as down.
                // that we can flag to return DOWN here (would that then update Discovery? or still be a delay?)
                serverStatusManager.localStatus(InstanceInfo.InstanceStatus.DOWN);
            }

            // Shutdown each of the client connections (blocks until complete).
            // NOTE: ClientConnectionsShutdown can also be configured to gracefully close connections when the
            // discovery status changes to DOWN. So if it has been configured that way, then this will be an additional
            // call to gracefullyShutdownClientChannels(), which will be a noop.
            clientConnectionsShutdown.gracefullyShutdownClientChannels().syncUninterruptibly();

            LOG.info("Shutting down event loops");
            List allEventLoopGroups = new ArrayList<>();
            allEventLoopGroups.add(clientToProxyBossPool);
            allEventLoopGroups.add(clientToProxyWorkerPool);
            for (EventLoopGroup group : allEventLoopGroups) {
                group.shutdownGracefully();
            }

            for (EventLoopGroup group : allEventLoopGroups) {
                try {
                    group.awaitTermination(20, TimeUnit.SECONDS);
                } catch (InterruptedException ie) {
                    LOG.warn("Interrupted while shutting down event loop");
                }
            }

            stopped = true;
            LOG.info("Done shutting down");
        }
    }

    /**
     * Keys should be a short string usable in metrics.
     */
    public static final AttributeKey CONN_DIMENSIONS = AttributeKey.newInstance("zuulconndimensions");

    private final class NewConnHandler extends ChannelInboundHandlerAdapter {

        @Override
        public void channelRead(ChannelHandlerContext ctx, Object msg) throws Exception {
            Long now = System.nanoTime();
            final Channel child = (Channel) msg;
            child.attr(CONN_DIMENSIONS).set(Attrs.newInstance());
            ConnTimer timer = ConnTimer.install(child, registry, registry.createId("zuul.conn.client.timing"));
            timer.record(now, "ACCEPT");
            ConnCounter.install(child, registry, registry.createId("zuul.conn.client.current"));
            super.channelRead(ctx, msg);
        }
    }

    static Map> convertPortMap(
            Map> portsToChannelInitializers) {
        Map> addrsToInitializers =
                new LinkedHashMap<>(portsToChannelInitializers.size());
        for (Map.Entry> portToInitializer : portsToChannelInitializers.entrySet()) {
            int portNumber = portToInitializer.getKey();
            addrsToInitializers.put(
                    new NamedSocketAddress("port" + portNumber, new InetSocketAddress(portNumber)),
                    portToInitializer.getValue());
        }
        return Collections.unmodifiableMap(addrsToInitializers);
    }

    private static boolean epollIsAvailable() {
        boolean available;
        try {
            available = Epoll.isAvailable();
        } catch (NoClassDefFoundError e) {
            LOG.debug("Epoll is unavailable, skipping", e);
            return false;
        } catch (RuntimeException | Error e) {
            LOG.warn("Epoll is unavailable, skipping", e);
            return false;
        }
        if (!available) {
            LOG.debug("Epoll is unavailable, skipping", Epoll.unavailabilityCause());
        }
        return available;
    }

    private static boolean ioUringIsAvailable() {
        boolean available;
        try {
            available = IOUring.isAvailable();
        } catch (NoClassDefFoundError e) {
            LOG.debug("io_uring is unavailable, skipping", e);
            return false;
        } catch (RuntimeException | Error e) {
            LOG.warn("io_uring is unavailable, skipping", e);
            return false;
        }
        if (!available) {
            LOG.debug("io_uring is unavailable, skipping", IOUring.unavailabilityCause());
        }
        return available;
    }

    private static boolean kqueueIsAvailable() {
        boolean available;
        try {
            available = KQueue.isAvailable();
        } catch (NoClassDefFoundError e) {
            LOG.debug("KQueue is unavailable, skipping", e);
            return false;
        } catch (RuntimeException | Error e) {
            LOG.warn("KQueue is unavailable, skipping", e);
            return false;
        }
        if (!available) {
            LOG.debug("KQueue is unavailable, skipping", KQueue.unavailabilityCause());
        }
        return available;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy