alluxio.grpc.GrpcChannelPool Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of alluxio-core-common Show documentation
Show all versions of alluxio-core-common Show documentation
Common utilities shared in Alluxio core modules
/*
* The Alluxio Open Foundation licenses this work under the Apache License, version 2.0
* (the "License"). You may not use this work except in compliance with the License, which is
* available at www.apache.org/licenses/LICENSE-2.0
*
* This software is distributed on an "AS IS" basis, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
* either express or implied, as more fully set forth in the License.
*
* See the NOTICE file distributed with this work for information regarding copyright ownership.
*/
package alluxio.grpc;
import static com.google.common.collect.ImmutableMap.toImmutableMap;
import static java.util.function.Function.identity;
import alluxio.conf.AlluxioConfiguration;
import alluxio.conf.Configuration;
import alluxio.conf.PropertyKey;
import alluxio.network.ChannelType;
import alluxio.util.CommonUtils;
import alluxio.util.WaitForOptions;
import alluxio.util.network.NettyUtils;
import alluxio.util.network.tls.SslContextProvider;
import com.google.common.base.Preconditions;
import io.grpc.ConnectivityState;
import io.grpc.ManagedChannel;
import io.grpc.netty.NettyChannelBuilder;
import io.netty.channel.EventLoopGroup;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.net.InetSocketAddress;
import java.net.SocketAddress;
import java.util.Arrays;
import java.util.Map;
import java.util.Objects;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.TimeoutException;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.concurrent.atomic.AtomicLong;
import javax.annotation.concurrent.ThreadSafe;
/**
* Used to provide gRPC level connection management and pooling facilities.
*
* This class is used internally by {@link GrpcChannelBuilder} and {@link GrpcChannel}.
*/
@ThreadSafe
public class GrpcChannelPool
{
// Singleton instance.
public static final GrpcChannelPool INSTANCE = new GrpcChannelPool();
private static final Logger LOG = LoggerFactory.getLogger(GrpcChannelPool.class);
private static final long GRACEFUL_TIMEOUT_MS = Configuration.getMs(
PropertyKey.NETWORK_CONNECTION_SHUTDOWN_GRACEFUL_TIMEOUT);
private static final long SHUTDOWN_TIMEOUT_MS = Configuration.getMs(
PropertyKey.NETWORK_CONNECTION_SHUTDOWN_TIMEOUT);
/** gRPC Managed channels/connections. */
private final ConcurrentMap> mChannels
= new ConcurrentHashMap<>();
/** Event loops. */
private final ConcurrentMap> mEventLoops
= new ConcurrentHashMap<>();
/** Used to assign order within a network group. */
private final Map mNetworkGroupCounters
= Arrays.stream(GrpcNetworkGroup.values())
.collect(toImmutableMap(identity(), group -> new AtomicLong()));
/** Used for obtaining SSL contexts for transport layer. */
private final SslContextProvider mSslContextProvider =
SslContextProvider.Factory.create(Configuration.global());
private GrpcChannelPool() {}
/**
* Acquires and increases the ref-count for the {@link ManagedChannel}.
* @param networkGroup network group
* @param serverAddress server address
* @param conf the Alluxio configuration
* @param alwaysEnableTLS whether to always enable TLS
* @return a {@link GrpcChannel}
*/
public GrpcChannel acquireChannel(GrpcNetworkGroup networkGroup,
GrpcServerAddress serverAddress, AlluxioConfiguration conf, boolean alwaysEnableTLS) {
GrpcChannelKey channelKey = getChannelKey(networkGroup, serverAddress, conf);
CountingReference channelRef =
mChannels.compute(channelKey, (key, ref) -> {
boolean shutdownExistingConnection = false;
int existingRefCount = 0;
if (ref != null) {
// Connection exists, wait for health check.
if (waitForConnectionReady(ref.get(), conf)) {
LOG.debug("Acquiring an existing connection. ConnectionKey: {}. Ref-count: {}", key,
ref.getRefCount());
return ref.reference();
} else {
// Health check failed.
shutdownExistingConnection = true;
}
}
// Existing connection should be shutdown.
if (shutdownExistingConnection) {
existingRefCount = ref.getRefCount();
LOG.debug("Shutting down an existing unhealthy connection. "
+ "ConnectionKey: {}. Ref-count: {}", key, existingRefCount);
// Shutdown the channel forcefully as it's already unhealthy.
shutdownManagedChannel(ref.get());
}
// Create a new managed channel.
LOG.debug("Creating a new managed channel. ConnectionKey: {}. Ref-count:{},"
+ " alwaysEnableTLS:{} config TLS:{}", key, existingRefCount, alwaysEnableTLS,
conf.getBoolean(alluxio.conf.PropertyKey.NETWORK_TLS_ENABLED));
ManagedChannel managedChannel = createManagedChannel(channelKey, conf, alwaysEnableTLS);
// Set map reference.
return new CountingReference<>(managedChannel, existingRefCount).reference();
});
return new GrpcChannel(channelKey, channelRef.get());
}
/**
* Decreases the ref-count of the {@link ManagedChannel} for the given address. It shuts down the
* underlying channel if reference count reaches zero.
* @param channelKey the connection key
*
*/
public void releaseConnection(GrpcChannelKey channelKey) {
mChannels.compute(channelKey, (key, ref) -> {
Preconditions.checkNotNull(ref, "Cannot release nonexistent connection");
LOG.debug("Releasing connection for: {}. Ref-count: {}", key, ref.getRefCount());
// Shutdown managed channel.
if (ref.dereference() == 0) {
LOG.debug("Shutting down connection after: {}", channelKey);
shutdownManagedChannel(ref.get());
// Release the event-loop for the connection.
releaseNetworkEventLoop(channelKey);
return null;
}
return ref;
});
}
private GrpcChannelKey getChannelKey(GrpcNetworkGroup networkGroup,
GrpcServerAddress serverAddress, AlluxioConfiguration conf) {
// Assign index within the network group.
long groupIndex = mNetworkGroupCounters.get(networkGroup).incrementAndGet();
// Find the next slot index within the group.
long maxConnectionsForGroup = conf.getLong(PropertyKey.Template.USER_NETWORK_MAX_CONNECTIONS
.format(networkGroup.getPropertyCode()));
// Create the connection key for the chosen slot.
return new GrpcChannelKey(networkGroup, serverAddress,
(int) (groupIndex % maxConnectionsForGroup));
}
/**
* Creates a {@link ManagedChannel} by given pool key.
*/
private ManagedChannel createManagedChannel(GrpcChannelKey channelKey,
AlluxioConfiguration conf, boolean alwaysEnableTLS) {
// Create netty channel builder with the address from channel key.
NettyChannelBuilder channelBuilder;
SocketAddress address = channelKey.getServerAddress().getSocketAddress();
if (address instanceof InetSocketAddress) {
InetSocketAddress inetServerAddress = (InetSocketAddress) address;
// This constructor delays DNS lookup to detect changes
channelBuilder = NettyChannelBuilder.forAddress(inetServerAddress.getHostName(),
inetServerAddress.getPort());
} else {
channelBuilder = NettyChannelBuilder.forAddress(address);
}
// Apply default channel options for the multiplex group.
channelBuilder.keepAliveTime(conf.getMs(PropertyKey.Template.USER_NETWORK_KEEPALIVE_TIME_MS
.format(channelKey.getNetworkGroup().getPropertyCode())),
TimeUnit.MILLISECONDS);
channelBuilder.keepAliveTimeout(
conf.getMs(PropertyKey.Template.USER_NETWORK_KEEPALIVE_TIMEOUT_MS.format(
channelKey.getNetworkGroup().getPropertyCode())),
TimeUnit.MILLISECONDS);
channelBuilder.maxInboundMessageSize((int) conf.getBytes(
PropertyKey.Template.USER_NETWORK_MAX_INBOUND_MESSAGE_SIZE.format(
channelKey.getNetworkGroup().getPropertyCode())));
channelBuilder.flowControlWindow((int) conf.getBytes(
PropertyKey.Template.USER_NETWORK_FLOWCONTROL_WINDOW.format(
channelKey.getNetworkGroup().getPropertyCode())));
channelBuilder.channelType(NettyUtils.getChannelClass(
!(channelKey.getServerAddress().getSocketAddress() instanceof InetSocketAddress),
PropertyKey.Template.USER_NETWORK_NETTY_CHANNEL.format(
channelKey.getNetworkGroup().getPropertyCode()),
conf));
channelBuilder.eventLoopGroup(acquireNetworkEventLoop(channelKey, conf));
channelBuilder.usePlaintext();
if (channelKey.getNetworkGroup() == GrpcNetworkGroup.SECRET) {
// Use self-signed for SECRET network group.
channelBuilder.sslContext(mSslContextProvider.getSelfSignedClientSslContext());
channelBuilder.useTransportSecurity();
} else if (conf.getBoolean(alluxio.conf.PropertyKey.NETWORK_TLS_ENABLED)) {
// Use shared TLS config for other network groups if enabled.
// Or this channel is enforced to enable TLS
channelBuilder.sslContext(mSslContextProvider.getClientSslContext());
channelBuilder.useTransportSecurity();
} else if (alwaysEnableTLS) {
// If the NETWORK_TLS_ENABLED is false and alwaysEnableTLS is true, use the self
// signed ssl context
channelBuilder.sslContext(mSslContextProvider.getSelfSignedClientSslContext());
channelBuilder.useTransportSecurity();
}
// Build netty managed channel.
return channelBuilder.build();
}
/**
* Returns {@code true} if given managed channel is ready.
*/
private boolean waitForConnectionReady(ManagedChannel managedChannel, AlluxioConfiguration conf) {
long healthCheckTimeoutMs = conf.getMs(PropertyKey.NETWORK_CONNECTION_HEALTH_CHECK_TIMEOUT);
try {
return CommonUtils.waitForResult("channel to be ready", () -> {
ConnectivityState currentState = managedChannel.getState(true);
switch (currentState) {
case READY:
return true;
case TRANSIENT_FAILURE:
case SHUTDOWN:
return false;
case IDLE:
case CONNECTING:
default:
return null;
}
}, Objects::nonNull, WaitForOptions.defaults().setTimeoutMs((int) healthCheckTimeoutMs));
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
return false;
} catch (TimeoutException e) {
return false;
}
}
/**
* Tries to gracefully shut down the managed channel. If falls back to forceful shutdown if
* graceful shutdown times out.
*/
private void shutdownManagedChannel(ManagedChannel managedChannel) {
// Close the gRPC managed-channel if not shut down already.
if (!managedChannel.isShutdown()) {
managedChannel.shutdown();
try {
if (!managedChannel.awaitTermination(GRACEFUL_TIMEOUT_MS, TimeUnit.MILLISECONDS)) {
LOG.warn("Timed out gracefully shutting down connection: {}. ", managedChannel);
}
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
// Allow thread to exit.
}
}
// Forceful shut down if still not terminated.
if (!managedChannel.isTerminated()) {
managedChannel.shutdownNow();
try {
if (!managedChannel.awaitTermination(SHUTDOWN_TIMEOUT_MS, TimeUnit.MILLISECONDS)) {
LOG.warn("Timed out forcefully shutting down connection: {}. ", managedChannel);
}
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
// Allow thread to exit.
}
}
}
private EventLoopGroup acquireNetworkEventLoop(GrpcChannelKey channelKey,
AlluxioConfiguration conf) {
return mEventLoops.compute(channelKey.getNetworkGroup(), (key, v) -> {
// Increment and return if event-loop found.
if (v != null) {
LOG.debug("Acquiring an existing event-loop for {}. Ref-Count:{}",
channelKey, v.getRefCount());
v.reference();
return v;
}
// Create a new event-loop.
ChannelType nettyChannelType =
NettyUtils.getChannelType(PropertyKey.Template.USER_NETWORK_NETTY_CHANNEL
.format(key.getPropertyCode()), conf);
int nettyWorkerThreadCount =
conf.getInt(PropertyKey.Template.USER_NETWORK_NETTY_WORKER_THREADS
.format(key.getPropertyCode()));
LOG.debug(
"Created a new event loop. NetworkGroup: {}. NettyChannelType: {}, NettyThreadCount: {}",
key, nettyChannelType, nettyWorkerThreadCount);
return new CountingReference<>(
NettyUtils.createEventLoop(nettyChannelType, nettyWorkerThreadCount, String.format(
"alluxio-client-netty-event-loop-%s-%%d", key.name()), true),
1);
}).get();
}
private void releaseNetworkEventLoop(GrpcChannelKey channelKey) {
mEventLoops.compute(channelKey.getNetworkGroup(), (key, ref) -> {
Preconditions.checkNotNull(ref, "Cannot release nonexistent event-loop");
LOG.debug("Releasing event-loop for: {}. Ref-count: {}", channelKey, ref.getRefCount());
if (ref.dereference() == 0) {
LOG.debug("Shutting down event-loop: {}", ref.get());
// Shutdown the event-loop gracefully.
ref.get().shutdownGracefully();
// No need to wait for event-loop shutdown.
return null;
}
return ref;
});
}
/**
* Used as reference counting wrapper over instance of type {@link T}.
*/
private static class CountingReference {
private final T mObject;
private final AtomicInteger mRefCount;
private CountingReference(T object, int initialRefCount) {
mObject = object;
mRefCount = new AtomicInteger(initialRefCount);
}
/**
* @return the underlying object after increasing ref-count
*/
private CountingReference reference() {
mRefCount.incrementAndGet();
return this;
}
/**
* Decrement the ref-count for underlying object.
*
* @return the current ref count after dereference
*/
private int dereference() {
return mRefCount.decrementAndGet();
}
/**
* @return current ref-count
*/
private int getRefCount() {
return mRefCount.get();
}
/**
* @return the underlying object without changing the ref-count
*/
private T get() {
return mObject;
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy