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

com.netflix.zuul.netty.server.ClientConnectionsShutdown 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.netflix.appinfo.InstanceInfo;
import com.netflix.config.DynamicBooleanProperty;
import com.netflix.config.DynamicIntProperty;
import com.netflix.discovery.EurekaClient;
import com.netflix.discovery.StatusChangeEvent;
import com.netflix.netty.common.ConnectionCloseChannelAttributes;
import com.netflix.netty.common.ConnectionCloseType;
import io.netty.channel.Channel;
import io.netty.channel.ChannelPromise;
import io.netty.channel.group.ChannelGroup;
import io.netty.channel.group.ChannelGroupFuture;
import io.netty.util.concurrent.EventExecutor;
import io.netty.util.concurrent.Promise;
import io.netty.util.concurrent.ScheduledFuture;
import java.util.concurrent.TimeUnit;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

/**
 * TODO: Change this class to be an instance per-port.
 * So that then the configuration can be different per-port, which is need for the combined FTL/Cloud clusters.
 * 

* User: [email protected] * Date: 3/6/17 * Time: 12:36 PM */ public class ClientConnectionsShutdown { private static final Logger LOG = LoggerFactory.getLogger(ClientConnectionsShutdown.class); private static final DynamicBooleanProperty ENABLED = new DynamicBooleanProperty("server.outofservice.connections.shutdown", false); private static final DynamicIntProperty DELAY_AFTER_OUT_OF_SERVICE_MS = new DynamicIntProperty("server.outofservice.connections.delay", 2000); private static final DynamicIntProperty GRACEFUL_CLOSE_TIMEOUT = new DynamicIntProperty("server.outofservice.close.timeout", 30); private final ChannelGroup channels; private final EventExecutor executor; private final EurekaClient discoveryClient; public ClientConnectionsShutdown(ChannelGroup channels, EventExecutor executor, EurekaClient discoveryClient) { this.channels = channels; this.executor = executor; this.discoveryClient = discoveryClient; if (discoveryClient != null) { initDiscoveryListener(); } } private void initDiscoveryListener() { this.discoveryClient.registerEventListener(event -> { if (event instanceof StatusChangeEvent) { StatusChangeEvent sce = (StatusChangeEvent) event; LOG.info("Received {}", sce); if (sce.getPreviousStatus() == InstanceInfo.InstanceStatus.UP && (sce.getStatus() == InstanceInfo.InstanceStatus.OUT_OF_SERVICE || sce.getStatus() == InstanceInfo.InstanceStatus.DOWN)) { // TODO - Also should stop accepting any new client connections now too? // Schedule to gracefully close all the client connections. if (ENABLED.get()) { executor.schedule( () -> gracefullyShutdownClientChannels(false), DELAY_AFTER_OUT_OF_SERVICE_MS.get(), TimeUnit.MILLISECONDS); } } } }); } public Promise gracefullyShutdownClientChannels() { return gracefullyShutdownClientChannels(true); } Promise gracefullyShutdownClientChannels(boolean forceCloseAfterTimeout) { // Mark all active connections to be closed after next response sent. LOG.warn("Flagging CLOSE_AFTER_RESPONSE on {} client channels.", channels.size()); // racy situation if new connections are still coming in, but any channels created after newCloseFuture will // be closed during the force close stage ChannelGroupFuture closeFuture = channels.newCloseFuture(); for (Channel channel : channels) { ConnectionCloseType.setForChannel(channel, ConnectionCloseType.DELAYED_GRACEFUL); ChannelPromise closePromise = channel.pipeline().newPromise(); channel.attr(ConnectionCloseChannelAttributes.CLOSE_AFTER_RESPONSE).set(closePromise); } Promise promise = executor.newPromise(); Runnable cancelTimeoutTask; if (forceCloseAfterTimeout) { ScheduledFuture timeoutTask = executor.schedule( () -> { LOG.warn("Force closing remaining {} active client channels.", channels.size()); channels.close(); }, GRACEFUL_CLOSE_TIMEOUT.get(), TimeUnit.SECONDS); cancelTimeoutTask = () -> { if (!timeoutTask.isDone()) { // close happened before the timeout timeoutTask.cancel(false); } }; } else { cancelTimeoutTask = () -> {}; } closeFuture.addListener(future -> { cancelTimeoutTask.run(); promise.setSuccess(null); }); return promise; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy