Re-adding Infinispan workarounds to prevent deadlocks (#22058)
Relates to #9871 Closes #22057
This commit is contained in:
parent
c907fde7f5
commit
cf911075af
3 changed files with 70 additions and 24 deletions
|
@ -33,6 +33,7 @@ import org.keycloak.cluster.ClusterProviderFactory;
|
||||||
import org.keycloak.common.Profile;
|
import org.keycloak.common.Profile;
|
||||||
import org.keycloak.common.util.Retry;
|
import org.keycloak.common.util.Retry;
|
||||||
import org.keycloak.common.util.Time;
|
import org.keycloak.common.util.Time;
|
||||||
|
import org.keycloak.connections.infinispan.DefaultInfinispanConnectionProviderFactory;
|
||||||
import org.keycloak.connections.infinispan.InfinispanConnectionProvider;
|
import org.keycloak.connections.infinispan.InfinispanConnectionProvider;
|
||||||
import org.keycloak.connections.infinispan.TopologyInfo;
|
import org.keycloak.connections.infinispan.TopologyInfo;
|
||||||
import org.keycloak.models.KeycloakSession;
|
import org.keycloak.models.KeycloakSession;
|
||||||
|
@ -220,8 +221,18 @@ public class InfinispanClusterProviderFactory implements ClusterProviderFactory,
|
||||||
}
|
}
|
||||||
|
|
||||||
logger.debugf("Nodes %s removed from cluster. Removing tasks locked by this nodes", removedNodesAddresses.toString());
|
logger.debugf("Nodes %s removed from cluster. Removing tasks locked by this nodes", removedNodesAddresses.toString());
|
||||||
|
/*
|
||||||
|
workaround for Infinispan 12.1.7.Final to prevent a deadlock while
|
||||||
|
DefaultInfinispanConnectionProviderFactory is shutting down PersistenceManagerImpl
|
||||||
|
that acquires a writeLock and this removal that acquires a readLock.
|
||||||
|
First seen with https://issues.redhat.com/browse/ISPN-13664 and still occurs probably due to
|
||||||
|
https://issues.redhat.com/browse/ISPN-13666 in 13.0.10
|
||||||
|
Tracked in https://github.com/keycloak/keycloak/issues/9871
|
||||||
|
*/
|
||||||
|
synchronized (DefaultInfinispanConnectionProviderFactory.class) {
|
||||||
workCache.entrySet().removeIf(new LockEntryPredicate(removedNodesAddresses));
|
workCache.entrySet().removeIf(new LockEntryPredicate(removedNodesAddresses));
|
||||||
}
|
}
|
||||||
|
}
|
||||||
} catch (Throwable t) {
|
} catch (Throwable t) {
|
||||||
logger.error("caught exception in ViewChangeListener", t);
|
logger.error("caught exception in ViewChangeListener", t);
|
||||||
}
|
}
|
||||||
|
|
|
@ -51,6 +51,7 @@ import org.keycloak.cluster.ClusterListener;
|
||||||
import org.keycloak.cluster.ClusterProvider;
|
import org.keycloak.cluster.ClusterProvider;
|
||||||
import org.keycloak.common.util.ConcurrentMultivaluedHashMap;
|
import org.keycloak.common.util.ConcurrentMultivaluedHashMap;
|
||||||
import org.keycloak.common.util.Retry;
|
import org.keycloak.common.util.Retry;
|
||||||
|
import org.keycloak.connections.infinispan.DefaultInfinispanConnectionProviderFactory;
|
||||||
import org.keycloak.executors.ExecutorsProvider;
|
import org.keycloak.executors.ExecutorsProvider;
|
||||||
import org.keycloak.models.KeycloakSession;
|
import org.keycloak.models.KeycloakSession;
|
||||||
import org.infinispan.client.hotrod.exceptions.HotRodClientException;
|
import org.infinispan.client.hotrod.exceptions.HotRodClientException;
|
||||||
|
@ -70,7 +71,7 @@ public class InfinispanNotificationsManager {
|
||||||
|
|
||||||
private final Cache<String, Serializable> workCache;
|
private final Cache<String, Serializable> workCache;
|
||||||
|
|
||||||
private final RemoteCache<String, Serializable> workRemoteCache;
|
private final RemoteCache workRemoteCache;
|
||||||
|
|
||||||
private final String myAddress;
|
private final String myAddress;
|
||||||
|
|
||||||
|
@ -79,7 +80,7 @@ public class InfinispanNotificationsManager {
|
||||||
private final ExecutorService listenersExecutor;
|
private final ExecutorService listenersExecutor;
|
||||||
|
|
||||||
|
|
||||||
protected InfinispanNotificationsManager(Cache<String, Serializable> workCache, RemoteCache<String, Serializable> workRemoteCache, String myAddress, String mySite, ExecutorService listenersExecutor) {
|
protected InfinispanNotificationsManager(Cache<String, Serializable> workCache, RemoteCache workRemoteCache, String myAddress, String mySite, ExecutorService listenersExecutor) {
|
||||||
this.workCache = workCache;
|
this.workCache = workCache;
|
||||||
this.workRemoteCache = workRemoteCache;
|
this.workRemoteCache = workRemoteCache;
|
||||||
this.myAddress = myAddress;
|
this.myAddress = myAddress;
|
||||||
|
@ -90,7 +91,7 @@ public class InfinispanNotificationsManager {
|
||||||
|
|
||||||
// Create and init manager including all listeners etc
|
// Create and init manager including all listeners etc
|
||||||
public static InfinispanNotificationsManager create(KeycloakSession session, Cache<String, Serializable> workCache, String myAddress, String mySite, Set<RemoteStore> remoteStores) {
|
public static InfinispanNotificationsManager create(KeycloakSession session, Cache<String, Serializable> workCache, String myAddress, String mySite, Set<RemoteStore> remoteStores) {
|
||||||
RemoteCache<String, Serializable> workRemoteCache = null;
|
RemoteCache workRemoteCache = null;
|
||||||
|
|
||||||
if (!remoteStores.isEmpty()) {
|
if (!remoteStores.isEmpty()) {
|
||||||
RemoteStore remoteStore = remoteStores.iterator().next();
|
RemoteStore remoteStore = remoteStores.iterator().next();
|
||||||
|
@ -157,7 +158,17 @@ public class InfinispanNotificationsManager {
|
||||||
// Add directly to remoteCache. Will notify remote listeners on all nodes in all DCs
|
// Add directly to remoteCache. Will notify remote listeners on all nodes in all DCs
|
||||||
Retry.executeWithBackoff((int iteration) -> {
|
Retry.executeWithBackoff((int iteration) -> {
|
||||||
try {
|
try {
|
||||||
|
/*
|
||||||
|
workaround for Infinispan 12.1.7.Final to prevent a deadlock while
|
||||||
|
DefaultInfinispanConnectionProviderFactory is shutting down PersistenceManagerImpl
|
||||||
|
that acquires a writeLock and this put that acquires a readLock.
|
||||||
|
First seen with https://issues.redhat.com/browse/ISPN-13664 and still occurs probably due to
|
||||||
|
https://issues.redhat.com/browse/ISPN-13666 in 13.0.10
|
||||||
|
Tracked in https://github.com/keycloak/keycloak/issues/9871
|
||||||
|
*/
|
||||||
|
synchronized (DefaultInfinispanConnectionProviderFactory.class) {
|
||||||
workRemoteCache.put(eventKey, wrappedEvent, 120, TimeUnit.SECONDS);
|
workRemoteCache.put(eventKey, wrappedEvent, 120, TimeUnit.SECONDS);
|
||||||
|
}
|
||||||
} catch (HotRodClientException re) {
|
} catch (HotRodClientException re) {
|
||||||
if (logger.isDebugEnabled()) {
|
if (logger.isDebugEnabled()) {
|
||||||
logger.debugf(re, "Failed sending notification to remote cache '%s'. Key: '%s', iteration '%s'. Will try to retry the task",
|
logger.debugf(re, "Failed sending notification to remote cache '%s'. Key: '%s', iteration '%s'. Will try to retry the task",
|
||||||
|
@ -184,7 +195,7 @@ public class InfinispanNotificationsManager {
|
||||||
|
|
||||||
@CacheEntryModified
|
@CacheEntryModified
|
||||||
public void cacheEntryModified(CacheEntryModifiedEvent<String, Serializable> event) {
|
public void cacheEntryModified(CacheEntryModifiedEvent<String, Serializable> event) {
|
||||||
eventReceived(event.getKey(), event.getNewValue());
|
eventReceived(event.getKey(), event.getValue());
|
||||||
}
|
}
|
||||||
|
|
||||||
@CacheEntryRemoved
|
@CacheEntryRemoved
|
||||||
|
@ -198,30 +209,30 @@ public class InfinispanNotificationsManager {
|
||||||
@ClientListener
|
@ClientListener
|
||||||
public class HotRodListener {
|
public class HotRodListener {
|
||||||
|
|
||||||
private final RemoteCache<String, Serializable> remoteCache;
|
private final RemoteCache<Object, Object> remoteCache;
|
||||||
|
|
||||||
public HotRodListener(RemoteCache<String, Serializable> remoteCache) {
|
public HotRodListener(RemoteCache<Object, Object> remoteCache) {
|
||||||
this.remoteCache = remoteCache;
|
this.remoteCache = remoteCache;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ClientCacheEntryCreated
|
@ClientCacheEntryCreated
|
||||||
public void created(ClientCacheEntryCreatedEvent<String> event) {
|
public void created(ClientCacheEntryCreatedEvent event) {
|
||||||
String key = event.getKey();
|
String key = event.getKey().toString();
|
||||||
hotrodEventReceived(key);
|
hotrodEventReceived(key);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ClientCacheEntryModified
|
@ClientCacheEntryModified
|
||||||
public void updated(ClientCacheEntryModifiedEvent<String> event) {
|
public void updated(ClientCacheEntryModifiedEvent event) {
|
||||||
String key = event.getKey();
|
String key = event.getKey().toString();
|
||||||
hotrodEventReceived(key);
|
hotrodEventReceived(key);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ClientCacheEntryRemoved
|
@ClientCacheEntryRemoved
|
||||||
public void removed(ClientCacheEntryRemovedEvent<String> event) {
|
public void removed(ClientCacheEntryRemovedEvent event) {
|
||||||
String key = event.getKey();
|
String key = event.getKey().toString();
|
||||||
taskFinished(key, true);
|
taskFinished(key, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -230,7 +241,21 @@ public class InfinispanNotificationsManager {
|
||||||
// TODO: Look at CacheEventConverter stuff to possibly include value in the event and avoid additional remoteCache request
|
// TODO: Look at CacheEventConverter stuff to possibly include value in the event and avoid additional remoteCache request
|
||||||
try {
|
try {
|
||||||
listenersExecutor.submit(() -> {
|
listenersExecutor.submit(() -> {
|
||||||
eventReceived(key, remoteCache.get(key));
|
|
||||||
|
/*
|
||||||
|
workaround for Infinispan 12.1.7.Final to prevent a deadlock while
|
||||||
|
DefaultInfinispanConnectionProviderFactory is shutting down PersistenceManagerImpl
|
||||||
|
that acquires a writeLock and this get that acquires a readLock.
|
||||||
|
First seen with https://issues.redhat.com/browse/ISPN-13664 and still occurs probably due to
|
||||||
|
https://issues.redhat.com/browse/ISPN-13666 in 13.0.10
|
||||||
|
Tracked in https://github.com/keycloak/keycloak/issues/9871
|
||||||
|
*/
|
||||||
|
Object value;
|
||||||
|
synchronized (DefaultInfinispanConnectionProviderFactory.class) {
|
||||||
|
value = remoteCache.get(key);
|
||||||
|
}
|
||||||
|
eventReceived(key, (Serializable) value);
|
||||||
|
|
||||||
});
|
});
|
||||||
} catch (RejectedExecutionException ree) {
|
} catch (RejectedExecutionException ree) {
|
||||||
// server is shutting down or pool was terminated - don't throw errors
|
// server is shutting down or pool was terminated - don't throw errors
|
||||||
|
|
|
@ -87,6 +87,15 @@ public class DefaultInfinispanConnectionProviderFactory implements InfinispanCon
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void close() {
|
public void close() {
|
||||||
|
/*
|
||||||
|
workaround for Infinispan 12.1.7.Final to prevent a deadlock while
|
||||||
|
DefaultInfinispanConnectionProviderFactory is shutting down PersistenceManagerImpl
|
||||||
|
that acquires a writeLock and this removal that acquires a readLock.
|
||||||
|
First seen with https://issues.redhat.com/browse/ISPN-13664 and still occurs probably due to
|
||||||
|
https://issues.redhat.com/browse/ISPN-13666 in 13.0.10
|
||||||
|
Tracked in https://github.com/keycloak/keycloak/issues/9871
|
||||||
|
*/
|
||||||
|
synchronized (DefaultInfinispanConnectionProviderFactory.class) {
|
||||||
if (cacheManager != null && !containerManaged) {
|
if (cacheManager != null && !containerManaged) {
|
||||||
cacheManager.stop();
|
cacheManager.stop();
|
||||||
}
|
}
|
||||||
|
@ -95,6 +104,7 @@ public class DefaultInfinispanConnectionProviderFactory implements InfinispanCon
|
||||||
}
|
}
|
||||||
cacheManager = null;
|
cacheManager = null;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public String getId() {
|
public String getId() {
|
||||||
|
|
Loading…
Reference in a new issue