org.neo4j.driver.internal.async.pool.NettyChannelTracker Maven / Gradle / Ivy
/*
* Copyright (c) "Neo4j"
* Neo4j Sweden AB [https://neo4j.com]
*
* 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 org.neo4j.driver.internal.async.pool;
import static org.neo4j.driver.internal.async.connection.ChannelAttributes.poolId;
import static org.neo4j.driver.internal.async.connection.ChannelAttributes.serverAddress;
import static org.neo4j.driver.internal.util.LockUtil.executeWithLock;
import io.netty.channel.Channel;
import io.netty.channel.ChannelFutureListener;
import io.netty.channel.group.ChannelGroup;
import io.netty.channel.group.DefaultChannelGroup;
import io.netty.channel.pool.ChannelPoolHandler;
import io.netty.util.concurrent.EventExecutor;
import java.util.HashMap;
import java.util.Map;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantReadWriteLock;
import org.neo4j.driver.Logger;
import org.neo4j.driver.Logging;
import org.neo4j.driver.internal.messaging.BoltProtocol;
import org.neo4j.driver.internal.metrics.ListenerEvent;
import org.neo4j.driver.internal.metrics.MetricsListener;
import org.neo4j.driver.net.ServerAddress;
public class NettyChannelTracker implements ChannelPoolHandler {
private final ReentrantReadWriteLock lock = new ReentrantReadWriteLock();
private final Lock read = lock.readLock();
private final Lock write = lock.writeLock();
private final Map addressToInUseChannelCount = new HashMap<>();
private final Map addressToIdleChannelCount = new HashMap<>();
private final Logger log;
private final MetricsListener metricsListener;
private final ChannelFutureListener closeListener = future -> channelClosed(future.channel());
private final ChannelGroup allChannels;
public NettyChannelTracker(MetricsListener metricsListener, EventExecutor eventExecutor, Logging logging) {
this(metricsListener, new DefaultChannelGroup("all-connections", eventExecutor), logging);
}
public NettyChannelTracker(MetricsListener metricsListener, ChannelGroup channels, Logging logging) {
this.metricsListener = metricsListener;
this.log = logging.getLog(getClass());
this.allChannels = channels;
}
@Override
public void channelReleased(Channel channel) {
executeWithLock(write, () -> {
decrementInUse(channel);
incrementIdle(channel);
channel.closeFuture().addListener(closeListener);
});
log.debug("Channel [0x%s] released back to the pool", channel.id());
}
@Override
public void channelAcquired(Channel channel) {
executeWithLock(write, () -> {
incrementInUse(channel);
decrementIdle(channel);
channel.closeFuture().removeListener(closeListener);
});
log.debug(
"Channel [0x%s] acquired from the pool. Local address: %s, remote address: %s",
channel.id(), channel.localAddress(), channel.remoteAddress());
}
@Override
public void channelCreated(Channel channel) {
throw new IllegalStateException("Untraceable channel created.");
}
public void channelCreated(Channel channel, ListenerEvent> creatingEvent) {
// when it is created, we count it as idle as it has not been acquired out of the pool
executeWithLock(write, () -> incrementIdle(channel));
metricsListener.afterCreated(poolId(channel), creatingEvent);
allChannels.add(channel);
log.debug(
"Channel [0x%s] created. Local address: %s, remote address: %s",
channel.id(), channel.localAddress(), channel.remoteAddress());
}
public ListenerEvent> channelCreating(String poolId) {
var creatingEvent = metricsListener.createListenerEvent();
metricsListener.beforeCreating(poolId, creatingEvent);
return creatingEvent;
}
public void channelFailedToCreate(String poolId) {
metricsListener.afterFailedToCreate(poolId);
}
public void channelClosed(Channel channel) {
executeWithLock(write, () -> decrementIdle(channel));
metricsListener.afterClosed(poolId(channel));
}
public int inUseChannelCount(ServerAddress address) {
return executeWithLock(read, () -> addressToInUseChannelCount.getOrDefault(address, 0));
}
public int idleChannelCount(ServerAddress address) {
return executeWithLock(read, () -> addressToIdleChannelCount.getOrDefault(address, 0));
}
public void prepareToCloseChannels() {
for (var channel : allChannels) {
var protocol = BoltProtocol.forChannel(channel);
try {
protocol.prepareToCloseChannel(channel);
} catch (Throwable e) {
// only logging it
log.debug(
"Failed to prepare to close Channel %s due to error %s. "
+ "It is safe to ignore this error as the channel will be closed despite if it is successfully prepared to close or not.",
channel, e.getMessage());
}
}
}
private void incrementInUse(Channel channel) {
increment(channel, addressToInUseChannelCount);
}
private void decrementInUse(Channel channel) {
var address = serverAddress(channel);
if (!addressToInUseChannelCount.containsKey(address)) {
throw new IllegalStateException("No count exists for address '" + address + "' in the 'in use' count");
}
var count = addressToInUseChannelCount.get(address);
addressToInUseChannelCount.put(address, count - 1);
}
private void incrementIdle(Channel channel) {
increment(channel, addressToIdleChannelCount);
}
private void decrementIdle(Channel channel) {
var address = serverAddress(channel);
if (!addressToIdleChannelCount.containsKey(address)) {
throw new IllegalStateException("No count exists for address '" + address + "' in the 'idle' count");
}
var count = addressToIdleChannelCount.get(address);
addressToIdleChannelCount.put(address, count - 1);
}
private void increment(Channel channel, Map countMap) {
ServerAddress address = serverAddress(channel);
var count = countMap.computeIfAbsent(address, k -> 0);
countMap.put(address, count + 1);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy