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

com.datastax.driver.core.ConvictionPolicy Maven / Gradle / Ivy

There is a newer version: 4.15.102
Show newest version
/*
 * Copyright DataStax, Inc.
 *
 * This software can be used solely with DataStax Enterprise. Please consult the license at
 * http://www.datastax.com/terms/datastax-dse-driver-license-terms
 */
package com.datastax.driver.core;

import static java.util.concurrent.TimeUnit.MILLISECONDS;
import static java.util.concurrent.TimeUnit.NANOSECONDS;

import com.datastax.driver.core.policies.ReconnectionPolicy;
import java.util.concurrent.atomic.AtomicInteger;

/**
 * The policy with which to decide whether a host should be considered down.
 *
 * 

TODO: this class is fully abstract (rather than an interface) because I'm not sure it's worth * exposing (and if we do expose it, we need to expose ConnectionException). Maybe just exposing say * a threshold of error before convicting a node is enough. */ abstract class ConvictionPolicy { /** * Called when new connections to the host are about to be created. * * @param count the number of connections */ abstract void signalConnectionsOpening(int count); /** Called when a connection closed normally. */ abstract void signalConnectionClosed(Connection connection); /** * Called when a connection error occurs on a connection to the host this policy applies to. * * @return whether the host should be considered down. */ abstract boolean signalConnectionFailure(Connection connection, boolean decrement); abstract boolean canReconnectNow(); abstract boolean hasActiveConnections(); /** Simple factory interface to allow creating {@link ConvictionPolicy} instances. */ interface Factory { /** * Creates a new ConvictionPolicy instance for {@code host}. * * @param host the host this policy applies to * @return the newly created {@link ConvictionPolicy} instance. */ ConvictionPolicy create(Host host, ReconnectionPolicy reconnectionPolicy); } static class DefaultConvictionPolicy extends ConvictionPolicy { private final Host host; private final ReconnectionPolicy reconnectionPolicy; private final AtomicInteger openConnections = new AtomicInteger(); private volatile long nextReconnectionTime = Long.MIN_VALUE; private ReconnectionPolicy.ReconnectionSchedule reconnectionSchedule; private DefaultConvictionPolicy(Host host, ReconnectionPolicy reconnectionPolicy) { this.host = host; this.reconnectionPolicy = reconnectionPolicy; } @Override void signalConnectionsOpening(int count) { int newTotal = openConnections.addAndGet(count); Host.statesLogger.debug( "[{}] preparing to open {} new connections, total = {}", host, count, newTotal); resetReconnectionTime(); } @Override void signalConnectionClosed(Connection connection) { int remaining = openConnections.decrementAndGet(); assert remaining >= 0; Host.statesLogger.debug("[{}] {} closed, remaining = {}", host, connection, remaining); } @Override boolean signalConnectionFailure(Connection connection, boolean decrement) { int remaining; if (decrement) { if (host.state != Host.State.DOWN) updateReconnectionTime(); remaining = openConnections.decrementAndGet(); assert remaining >= 0; Host.statesLogger.debug("[{}] {} failed, remaining = {}", host, connection, remaining); } else { remaining = openConnections.get(); } return remaining == 0; } private synchronized void updateReconnectionTime() { long now = System.nanoTime(); if (nextReconnectionTime > now) // Someone else updated the time before us return; if (reconnectionSchedule == null) reconnectionSchedule = reconnectionPolicy.newSchedule(); long nextDelayMs = reconnectionSchedule.nextDelayMs(); Host.statesLogger.debug( "[{}] preventing new connections for the next {} ms", host, nextDelayMs); nextReconnectionTime = now + NANOSECONDS.convert(nextDelayMs, MILLISECONDS); } private synchronized void resetReconnectionTime() { reconnectionSchedule = null; nextReconnectionTime = Long.MIN_VALUE; } @Override boolean canReconnectNow() { boolean canReconnectNow = nextReconnectionTime == Long.MIN_VALUE || System.nanoTime() >= nextReconnectionTime; Host.statesLogger.trace("canReconnectNow={}", canReconnectNow); return canReconnectNow; } @Override boolean hasActiveConnections() { return openConnections.get() > 0; } static class Factory implements ConvictionPolicy.Factory { @Override public ConvictionPolicy create(Host host, ReconnectionPolicy reconnectionPolicy) { return new DefaultConvictionPolicy(host, reconnectionPolicy); } } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy