com.king.platform.net.http.netty.pool.ServerPool Maven / Gradle / Ivy
// Copyright (C) king.com Ltd 2015
// https://github.com/king/king-http-client
// Author: Magnus Gustafsson
// License: Apache 2.0, https://raw.github.com/king/king-http-client/LICENSE-APACHE
package com.king.platform.net.http.netty.pool;
import static org.slf4j.LoggerFactory.getLogger;
import io.netty.channel.Channel;
import java.util.ArrayList;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentLinkedDeque;
import java.util.concurrent.ConcurrentLinkedQueue;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicInteger;
import org.slf4j.Logger;
import com.king.platform.net.http.netty.ServerInfo;
import com.king.platform.net.http.netty.metric.MetricCallback;
import com.king.platform.net.http.netty.util.TimeProvider;
public class ServerPool {
private final Logger logger = getLogger(getClass());
private final ServerInfo server;
private final long maxTTL;
private final TimeUnit ttlTimeUnit;
private final AtomicInteger idGenerator = new AtomicInteger();
private final ConcurrentLinkedDeque pooledChannels = new ConcurrentLinkedDeque<>();
private final ConcurrentHashMap channelsMap = new ConcurrentHashMap<>();
private final TimeProvider timeProvider;
private final MetricCallback metricCallback;
private volatile long lastOfferedConnectionTime;
public ServerPool(ServerInfo server, long maxTTL, TimeUnit ttlTimeUnit, TimeProvider timeProvider, MetricCallback metricCallback) {
this.timeProvider = timeProvider;
this.metricCallback = metricCallback;
lastOfferedConnectionTime = timeProvider.currentTimeInMillis();
this.server = server;
this.maxTTL = maxTTL;
this.ttlTimeUnit = ttlTimeUnit;
}
public Channel poll() {
PooledChannel pooledChannel;
while ((pooledChannel = pooledChannels.poll()) != null) {
if (isValidConnection(pooledChannel)) {
logger.trace("Found active channel for server {} with id {} created at {}", server, pooledChannel.id, pooledChannel.creationTimeStamp);
return pooledChannel.channel;
} else {
channelsMap.remove(pooledChannel.channel);
pooledChannel.channel.close();
logger.trace("Channel to {} with id {} created at {} is dead!", server, pooledChannel.id, pooledChannel.creationTimeStamp);
metricCallback.onServerPoolClosedConnection(server.getHost(), channelsMap.size());
}
}
return null;
}
private boolean isValidConnection(PooledChannel pooledChannel) {
Channel channel = pooledChannel.channel;
long currentTime = timeProvider.currentTimeInMillis();
if (pooledChannel.lastUsedTimeStamp + maxTTL <= currentTime) { //TTL for this connection has expired
return false;
}
if (channel.isActive() && channel.isOpen()) {
return true;
}
return false;
}
public void offer(Channel channel) {
PooledChannel pooledChannel = channelsMap.get(channel);
if (!channel.isActive() || !channel.isOpen()) {
if (pooledChannel == null) {
return;
}
channelsMap.remove(channel);
return;
}
if (pooledChannel == null) {
pooledChannel = new PooledChannel(idGenerator.incrementAndGet(), timeProvider.currentTimeInMillis(), channel);
PooledChannel oldValue = channelsMap.putIfAbsent(channel, pooledChannel);
if (oldValue == null) {
logger.trace("Adding new active channel for server {} with id {} created at {}", server, pooledChannel.id, pooledChannel.creationTimeStamp);
metricCallback.onServerPoolAddedConnection(server.getHost(), channelsMap.size());
}
}
logger.trace("offering active channel for server {} with id {} created at {}", server, pooledChannel.id, pooledChannel.creationTimeStamp);
lastOfferedConnectionTime = timeProvider.currentTimeInMillis();
pooledChannel.lastUsedTimeStamp = timeProvider.currentTimeInMillis();
pooledChannels.addFirst(pooledChannel);
}
public void discard(Channel channel) {
channel.close();
PooledChannel remove = channelsMap.remove(channel);
if (remove != null) {
//pooledChannels.remove(remove); we don't want to do remove since its expensive on a linkedList
metricCallback.onServerPoolClosedConnection(server.getHost(), channelsMap.size());
}
}
public void cleanExpiredConnections() {
HashSet currentPooledChannels = new HashSet<>(pooledChannels);
List closedChannels = new ArrayList<>();
for (PooledChannel pooledChannel : currentPooledChannels) {
if (!isValidConnection(pooledChannel)) {
closedChannels.add(pooledChannel);
}
}
for (PooledChannel closedChannel : closedChannels) {
pooledChannels.remove(closedChannel);
PooledChannel remove = channelsMap.remove(closedChannel.channel);
logger.trace("Cleaned expired connection {}", closedChannel.channel);
if (remove != null) {
remove.channel.close();
metricCallback.onServerPoolClosedConnection(server.getHost(), channelsMap.size());
}
}
long now = timeProvider.currentTimeInMillis();
for (Map.Entry entry : channelsMap.entrySet()) {
Channel channel = entry.getKey();
PooledChannel pooledChannel = entry.getValue();
if (pooledChannel.lastUsedTimeStamp + ttlTimeUnit.toMillis(maxTTL) > now) {
continue;
}
if (!currentPooledChannels.contains(pooledChannel)) {
channel.close();
channelsMap.remove(channel);
}
}
}
public boolean shouldRemovePool() {
return pooledChannels.isEmpty() && lastOfferedConnectionTime + ttlTimeUnit.toMillis(maxTTL) <= timeProvider.currentTimeInMillis();
}
public int getPoolSize() {
return pooledChannels.size();
}
public int getChannelSize() {
return channelsMap.size();
}
private static class PooledChannel {
private int id;
private long creationTimeStamp;
private long lastUsedTimeStamp;
private Channel channel;
public PooledChannel(int id, long creationTimeStamp, Channel channel) {
this.id = id;
this.creationTimeStamp = creationTimeStamp;
this.channel = channel;
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy