org.neo4j.driver.internal.cluster.RediscoveryImpl Maven / Gradle / Ivy
/*
* Copyright (c) "Neo4j"
* Neo4j Sweden AB [http://neo4j.com]
*
* This file is part of Neo4j.
*
* 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.cluster;
import static java.lang.String.format;
import static java.util.Collections.emptySet;
import static java.util.Objects.requireNonNull;
import static java.util.concurrent.CompletableFuture.completedFuture;
import static org.neo4j.driver.internal.util.Futures.completedWithNull;
import static org.neo4j.driver.internal.util.Futures.failedFuture;
import io.netty.util.concurrent.EventExecutorGroup;
import java.net.UnknownHostException;
import java.util.Collection;
import java.util.HashSet;
import java.util.LinkedList;
import java.util.List;
import java.util.Set;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.CompletionException;
import java.util.concurrent.CompletionStage;
import java.util.concurrent.TimeUnit;
import org.neo4j.driver.Bookmark;
import org.neo4j.driver.Logger;
import org.neo4j.driver.Logging;
import org.neo4j.driver.exceptions.AuthorizationExpiredException;
import org.neo4j.driver.exceptions.ClientException;
import org.neo4j.driver.exceptions.DiscoveryException;
import org.neo4j.driver.exceptions.FatalDiscoveryException;
import org.neo4j.driver.exceptions.SecurityException;
import org.neo4j.driver.exceptions.ServiceUnavailableException;
import org.neo4j.driver.internal.BoltServerAddress;
import org.neo4j.driver.internal.DomainNameResolver;
import org.neo4j.driver.internal.ImpersonationUtil;
import org.neo4j.driver.internal.ResolvedBoltServerAddress;
import org.neo4j.driver.internal.spi.ConnectionPool;
import org.neo4j.driver.internal.util.Futures;
import org.neo4j.driver.net.ServerAddress;
import org.neo4j.driver.net.ServerAddressResolver;
public class RediscoveryImpl implements Rediscovery {
private static final String NO_ROUTERS_AVAILABLE =
"Could not perform discovery for database '%s'. No routing server available.";
private static final String RECOVERABLE_ROUTING_ERROR = "Failed to update routing table with server '%s'.";
private static final String RECOVERABLE_DISCOVERY_ERROR_WITH_SERVER =
"Received a recoverable discovery error with server '%s', "
+ "will continue discovery with other routing servers if available. "
+ "Complete failure is reported separately from this entry.";
private static final String INVALID_BOOKMARK_CODE = "Neo.ClientError.Transaction.InvalidBookmark";
private static final String INVALID_BOOKMARK_MIXTURE_CODE = "Neo.ClientError.Transaction.InvalidBookmarkMixture";
private final BoltServerAddress initialRouter;
private final RoutingSettings settings;
private final Logger log;
private final ClusterCompositionProvider provider;
private final ServerAddressResolver resolver;
private final EventExecutorGroup eventExecutorGroup;
private final DomainNameResolver domainNameResolver;
public RediscoveryImpl(
BoltServerAddress initialRouter,
RoutingSettings settings,
ClusterCompositionProvider provider,
EventExecutorGroup eventExecutorGroup,
ServerAddressResolver resolver,
Logging logging,
DomainNameResolver domainNameResolver) {
this.initialRouter = initialRouter;
this.settings = settings;
this.log = logging.getLog(getClass());
this.provider = provider;
this.resolver = resolver;
this.eventExecutorGroup = eventExecutorGroup;
this.domainNameResolver = requireNonNull(domainNameResolver);
}
/**
* Given a database and its current routing table, and the global connection pool, use the global cluster composition provider to fetch a new cluster
* composition, which would be used to update the routing table of the given database and global connection pool.
*
* @param routingTable current routing table of the given database.
* @param connectionPool connection pool.
* @return new cluster composition and an optional set of resolved initial router addresses.
*/
@Override
public CompletionStage lookupClusterComposition(
RoutingTable routingTable, ConnectionPool connectionPool, Bookmark bookmark, String impersonatedUser) {
CompletableFuture result = new CompletableFuture<>();
// if we failed discovery, we will chain all errors into this one.
ServiceUnavailableException baseError = new ServiceUnavailableException(
String.format(NO_ROUTERS_AVAILABLE, routingTable.database().description()));
lookupClusterComposition(routingTable, connectionPool, 0, 0, result, bookmark, impersonatedUser, baseError);
return result;
}
private void lookupClusterComposition(
RoutingTable routingTable,
ConnectionPool pool,
int failures,
long previousDelay,
CompletableFuture result,
Bookmark bookmark,
String impersonatedUser,
Throwable baseError) {
lookup(routingTable, pool, bookmark, impersonatedUser, baseError)
.whenComplete((compositionLookupResult, completionError) -> {
Throwable error = Futures.completionExceptionCause(completionError);
if (error != null) {
result.completeExceptionally(error);
} else if (compositionLookupResult != null) {
result.complete(compositionLookupResult);
} else {
int newFailures = failures + 1;
if (newFailures >= settings.maxRoutingFailures()) {
// now we throw our saved error out
result.completeExceptionally(baseError);
} else {
long nextDelay = Math.max(settings.retryTimeoutDelay(), previousDelay * 2);
log.info("Unable to fetch new routing table, will try again in " + nextDelay + "ms");
eventExecutorGroup
.next()
.schedule(
() -> lookupClusterComposition(
routingTable,
pool,
newFailures,
nextDelay,
result,
bookmark,
impersonatedUser,
baseError),
nextDelay,
TimeUnit.MILLISECONDS);
}
}
});
}
private CompletionStage lookup(
RoutingTable routingTable,
ConnectionPool connectionPool,
Bookmark bookmark,
String impersonatedUser,
Throwable baseError) {
CompletionStage compositionStage;
if (routingTable.preferInitialRouter()) {
compositionStage = lookupOnInitialRouterThenOnKnownRouters(
routingTable, connectionPool, bookmark, impersonatedUser, baseError);
} else {
compositionStage = lookupOnKnownRoutersThenOnInitialRouter(
routingTable, connectionPool, bookmark, impersonatedUser, baseError);
}
return compositionStage;
}
private CompletionStage lookupOnKnownRoutersThenOnInitialRouter(
RoutingTable routingTable,
ConnectionPool connectionPool,
Bookmark bookmark,
String impersonatedUser,
Throwable baseError) {
Set seenServers = new HashSet<>();
return lookupOnKnownRouters(routingTable, connectionPool, seenServers, bookmark, impersonatedUser, baseError)
.thenCompose(compositionLookupResult -> {
if (compositionLookupResult != null) {
return completedFuture(compositionLookupResult);
}
return lookupOnInitialRouter(
routingTable, connectionPool, seenServers, bookmark, impersonatedUser, baseError);
});
}
private CompletionStage lookupOnInitialRouterThenOnKnownRouters(
RoutingTable routingTable,
ConnectionPool connectionPool,
Bookmark bookmark,
String impersonatedUser,
Throwable baseError) {
Set seenServers = emptySet();
return lookupOnInitialRouter(routingTable, connectionPool, seenServers, bookmark, impersonatedUser, baseError)
.thenCompose(compositionLookupResult -> {
if (compositionLookupResult != null) {
return completedFuture(compositionLookupResult);
}
return lookupOnKnownRouters(
routingTable, connectionPool, new HashSet<>(), bookmark, impersonatedUser, baseError);
});
}
private CompletionStage lookupOnKnownRouters(
RoutingTable routingTable,
ConnectionPool connectionPool,
Set seenServers,
Bookmark bookmark,
String impersonatedUser,
Throwable baseError) {
CompletableFuture result = completedWithNull();
for (BoltServerAddress address : routingTable.routers()) {
result = result.thenCompose(composition -> {
if (composition != null) {
return completedFuture(composition);
} else {
return lookupOnRouter(
address,
true,
routingTable,
connectionPool,
seenServers,
bookmark,
impersonatedUser,
baseError);
}
});
}
return result.thenApply(
composition -> composition != null ? new ClusterCompositionLookupResult(composition) : null);
}
private CompletionStage lookupOnInitialRouter(
RoutingTable routingTable,
ConnectionPool connectionPool,
Set seenServers,
Bookmark bookmark,
String impersonatedUser,
Throwable baseError) {
List resolvedRouters;
try {
resolvedRouters = resolve();
} catch (Throwable error) {
return failedFuture(error);
}
Set resolvedRouterSet = new HashSet<>(resolvedRouters);
resolvedRouters.removeAll(seenServers);
CompletableFuture result = completedWithNull();
for (BoltServerAddress address : resolvedRouters) {
result = result.thenCompose(composition -> {
if (composition != null) {
return completedFuture(composition);
}
return lookupOnRouter(
address, false, routingTable, connectionPool, null, bookmark, impersonatedUser, baseError);
});
}
return result.thenApply(composition ->
composition != null ? new ClusterCompositionLookupResult(composition, resolvedRouterSet) : null);
}
private CompletionStage lookupOnRouter(
BoltServerAddress routerAddress,
boolean resolveAddress,
RoutingTable routingTable,
ConnectionPool connectionPool,
Set seenServers,
Bookmark bookmark,
String impersonatedUser,
Throwable baseError) {
CompletableFuture addressFuture = CompletableFuture.completedFuture(routerAddress);
return addressFuture
.thenApply(address ->
resolveAddress ? resolveByDomainNameOrThrowCompletionException(address, routingTable) : address)
.thenApply(address -> addAndReturn(seenServers, address))
.thenCompose(connectionPool::acquire)
.thenApply(connection -> ImpersonationUtil.ensureImpersonationSupport(connection, impersonatedUser))
.thenCompose(connection ->
provider.getClusterComposition(connection, routingTable.database(), bookmark, impersonatedUser))
.handle((response, error) -> {
Throwable cause = Futures.completionExceptionCause(error);
if (cause != null) {
return handleRoutingProcedureError(cause, routingTable, routerAddress, baseError);
} else {
return response;
}
});
}
private ClusterComposition handleRoutingProcedureError(
Throwable error, RoutingTable routingTable, BoltServerAddress routerAddress, Throwable baseError) {
if (mustAbortDiscovery(error)) {
throw new CompletionException(error);
}
// Retriable error happened during discovery.
DiscoveryException discoveryError =
new DiscoveryException(format(RECOVERABLE_ROUTING_ERROR, routerAddress), error);
Futures.combineErrors(baseError, discoveryError); // we record each failure here
String warningMessage = format(RECOVERABLE_DISCOVERY_ERROR_WITH_SERVER, routerAddress);
log.warn(warningMessage);
log.debug(warningMessage, discoveryError);
routingTable.forget(routerAddress);
return null;
}
private boolean mustAbortDiscovery(Throwable throwable) {
boolean abort = false;
if (!(throwable instanceof AuthorizationExpiredException) && throwable instanceof SecurityException) {
abort = true;
} else if (throwable instanceof FatalDiscoveryException) {
abort = true;
} else if (throwable instanceof IllegalStateException
&& ConnectionPool.CONNECTION_POOL_CLOSED_ERROR_MESSAGE.equals(throwable.getMessage())) {
abort = true;
} else if (throwable instanceof ClientException) {
String code = ((ClientException) throwable).code();
abort = INVALID_BOOKMARK_CODE.equals(code) || INVALID_BOOKMARK_MIXTURE_CODE.equals(code);
}
return abort;
}
@Override
public List resolve() throws UnknownHostException {
List resolvedAddresses = new LinkedList<>();
UnknownHostException exception = null;
for (ServerAddress serverAddress : resolver.resolve(initialRouter)) {
try {
resolveAllByDomainName(serverAddress).unicastStream().forEach(resolvedAddresses::add);
} catch (UnknownHostException e) {
if (exception == null) {
exception = e;
} else {
exception.addSuppressed(e);
}
}
}
// give up only if there are no addresses to work with at all
if (resolvedAddresses.isEmpty() && exception != null) {
throw exception;
}
return resolvedAddresses;
}
private T addAndReturn(Collection collection, T element) {
if (collection != null) {
collection.add(element);
}
return element;
}
private BoltServerAddress resolveByDomainNameOrThrowCompletionException(
BoltServerAddress address, RoutingTable routingTable) {
try {
ResolvedBoltServerAddress resolvedAddress = resolveAllByDomainName(address);
routingTable.replaceRouterIfPresent(address, resolvedAddress);
return resolvedAddress
.unicastStream()
.findFirst()
.orElseThrow(
() -> new IllegalStateException(
"Unexpected condition, the ResolvedBoltServerAddress must always have at least one unicast address"));
} catch (Throwable e) {
throw new CompletionException(e);
}
}
private ResolvedBoltServerAddress resolveAllByDomainName(ServerAddress address) throws UnknownHostException {
return new ResolvedBoltServerAddress(
address.host(), address.port(), domainNameResolver.resolve(address.host()));
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy