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

com.ibm.cloud.objectstorage.http.IdleConnectionReaper Maven / Gradle / Ivy

/*
 * Copyright 2012-2017 Amazon.com, Inc. or its affiliates. All Rights Reserved.
 *
 * Licensed under the Apache License, Version 2.0 (the "License").
 * You may not use this file except in compliance with the License.
 * A copy of the License is located at
 *
 *  http://aws.amazon.com/apache2.0
 *
 * or in the "license" file accompanying this file. This file 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.ibm.cloud.objectstorage.http;

import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.apache.http.conn.HttpClientConnectionManager;

import com.ibm.cloud.objectstorage.annotation.SdkInternalApi;
import com.ibm.cloud.objectstorage.annotation.SdkTestInternalApi;

import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.TimeUnit;

/**
 * Daemon thread to periodically check connection pools for idle connections.
 * 

* Connections sitting around idle in the HTTP connection pool for too long will * eventually be terminated by the AWS end of the connection, and will go into * CLOSE_WAIT. If this happens, sockets will sit around in CLOSE_WAIT, still * using resources on the client side to manage that socket. Many sockets stuck * in CLOSE_WAIT can prevent the OS from creating new connections. *

* This class closes idle connections before they can move into the CLOSE_WAIT * state. *

* This thread is important because by default, we disable Apache HttpClient's * stale connection checking, so without this thread running in the background, * cleaning up old/inactive HTTP connections, we'd see more IO exceptions when * stale connections (i.e. closed on the AWS side) are left in the connection * pool, and requests grab one of them to begin executing a request. */ @SdkInternalApi public final class IdleConnectionReaper extends Thread { /** * Shared log for any errors during connection reaping. */ private static final Log LOG = LogFactory.getLog(IdleConnectionReaper.class); /** * The period between invocations of the idle connection reaper. */ private static final int PERIOD_MILLISECONDS = 1000 * 60; /** * Legacy constant used when {@link #registerConnectionManager(HttpClientConnectionManager)} is called. New code paths should * use {@link #registerConnectionManager(HttpClientConnectionManager, long)} and provide the max idle timeout for that * particular connection manager. */ @Deprecated private static final int DEFAULT_MAX_IDLE_MILLIS = 1000 * 60; private static final Map connectionManagers = new ConcurrentHashMap(); /** * Singleton instance of the connection reaper. */ private static volatile IdleConnectionReaper instance; /** * Set to true when shutting down the reaper; Once set to true, this * flag is never set back to false. */ private volatile boolean shuttingDown; /** * Private constructor - singleton pattern. */ private IdleConnectionReaper() { super("java-sdk-http-connection-reaper"); setDaemon(true); } /** * Registers the given connection manager with this reaper. * * @return true if the connection manager has been successfully registered; false otherwise. * @deprecated By {@link #registerConnectionManager(HttpClientConnectionManager, long)}. */ @Deprecated public static boolean registerConnectionManager(HttpClientConnectionManager connectionManager) { return registerConnectionManager(connectionManager, DEFAULT_MAX_IDLE_MILLIS); } /** * Registers the given connection manager with this reaper; * * @param connectionManager Connection manager to register * @param maxIdleInMs Max idle connection timeout in milliseconds for this connection manager. * @return true if the connection manager has been successfully registered; false otherwise. */ public static boolean registerConnectionManager(HttpClientConnectionManager connectionManager, long maxIdleInMs) { if (instance == null) { synchronized (IdleConnectionReaper.class) { if (instance == null) { instance = new IdleConnectionReaper(); instance.start(); } } } return connectionManagers.put(connectionManager, maxIdleInMs) == null; } /** * Removes the given connection manager from this reaper, * and shutting down the reaper if there is zero connection manager left. * * @return true if the connection manager has been successfully removed; * false otherwise. */ public static boolean removeConnectionManager(HttpClientConnectionManager connectionManager) { boolean wasRemoved = connectionManagers.remove(connectionManager) != null; if (connectionManagers.isEmpty()) { shutdown(); } return wasRemoved; } @SdkTestInternalApi public static List getRegisteredConnectionManagers() { return new ArrayList(connectionManagers.keySet()); } /** * Shuts down the thread, allowing the class and instance to be collected. *

* Since this is a daemon thread, its running will not prevent JVM shutdown. * It will, however, prevent this class from being unloaded or garbage * collected, in the context of a long-running application, until it is * interrupted. This method will stop the thread's execution and clear its * state. Any use of a service client will cause the thread to be restarted. * * @return true if an actual shutdown has been made; false otherwise. */ public static synchronized boolean shutdown() { if (instance != null) { instance.markShuttingDown(); instance.interrupt(); connectionManagers.clear(); instance = null; return true; } return false; } /** * For testing purposes. * Returns the number of connection managers currently monitored by this * reaper. */ static int size() { return connectionManagers.size(); } private void markShuttingDown() { shuttingDown = true; } @SuppressWarnings("unchecked") @Override public void run() { while (true) { if (shuttingDown) { LOG.debug("Shutting down reaper thread."); return; } try { Thread.sleep(PERIOD_MILLISECONDS); for (Map.Entry entry : connectionManagers.entrySet()) { // When we release connections, the connection manager leaves them // open so they can be reused. We want to close out any idle // connections so that they don't sit around in CLOSE_WAIT. try { entry.getKey().closeIdleConnections(entry.getValue(), TimeUnit.MILLISECONDS); } catch (Exception t) { LOG.warn("Unable to close idle connections", t); } } } catch (Throwable t) { LOG.debug("Reaper thread: ", t); } } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy