org.drools.persistence.jpa.OptimisticLockRetryInterceptor Maven / Gradle / Ivy
/*
* Copyright 2013 Red Hat, Inc. and/or its affiliates.
*
* 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.drools.persistence.jpa;
import org.drools.commands.impl.AbstractInterceptor;
import org.kie.api.runtime.Executable;
import org.kie.api.runtime.RequestContext;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import javax.persistence.OptimisticLockException;
import java.util.concurrent.atomic.AtomicInteger;
/**
* ExecutableInterceptor that is capable of retrying command execution. It is intended to retry only if right exception
* has been thrown. By default it will look for org.hibernate.StaleObjectStateException
and only
* then attempt to retry.
* Since this is Hibernate specific class another can be given as system property to override default. Name of the
* system property org.kie.optlock.exclass
and its value should be fully qualified class name of the
* exception that indicates OptimisticLocking.
* By default it will:
*
* - Retry 3 times
* - First retry will be attempted after 50 milliseconds
* - next retries will be calculated as last sleep time multiplied by a factor (default factor is 4)
*
* In case all retries failed origin exception will be thrown.
*
*/
public class OptimisticLockRetryInterceptor extends AbstractInterceptor {
private static final Logger logger = LoggerFactory.getLogger(OptimisticLockRetryInterceptor.class);
private int retries = Integer.getInteger("org.kie.optlock.retries", 3);
private long delay = Long.getLong("org.kie.optlock.delay", 50L);
private long delayFactor = Long.getLong("org.kie.optlock.delayFactor", 4L);
protected Class> targetExceptionClass;
protected Class> targetConstraintViolationExceptionClass;
private static final ThreadLocal invocationsCounter = new ThreadLocal<>();
public OptimisticLockRetryInterceptor() {
String clazz = System.getProperty("org.kie.optlock.exclass", "org.hibernate.StaleObjectStateException");
try {
targetExceptionClass = Class.forName(clazz);
} catch (ClassNotFoundException e) {
logger.error("Optimistic locking exception class not found {}", clazz, e);
}
clazz = System.getProperty("org.kie.constraint.exclass", "org.hibernate.exception.ConstraintViolationException");
try {
targetConstraintViolationExceptionClass = Class.forName(clazz);
} catch (ClassNotFoundException e) {
logger.warn("Constraint violation exception class not found {}", clazz, e);
}
}
@Override
public final RequestContext execute( Executable executable, RequestContext ctx ) {
AtomicInteger counter = invocationsCounter.get();
if (counter == null) {
counter = new AtomicInteger( 0 );
invocationsCounter.set( counter );
}
counter.incrementAndGet();
try {
return internalExecute( executable, ctx );
} finally {
if (counter.decrementAndGet() == 0) {
invocationsCounter.remove();
}
}
}
protected RequestContext internalExecute( Executable executable, RequestContext ctx ) {
int attempt = 0;
long sleepTime = delay;
RuntimeException originException = null;
while (true) {
if (attempt > 1) {
logger.trace("retrying (attempt {})...", attempt);
}
try {
executeNext(executable, ctx);
return ctx;
} catch (RuntimeException ex) {
// in case there is another interceptor of this type in the stack don't handle it here
if (hasInterceptorInStack()) {
throw ex;
}
logger.trace(ex.getClass().getSimpleName() + " caught in " + this.getClass().getSimpleName() + ": " + ex.getMessage());
if (!isCausedByOptimisticLockingFailure(ex) && !isCausedByConstraintViolationFailure(ex)) {
throw ex;
}
attempt++;
// save origin exception in case it needs to be rethrown
if (originException == null) {
originException = ex;
}
if (attempt > retries) {
break;
}
logger.trace("Command failed due to optimistic locking {} waiting {} millis before retry", ex, sleepTime);
try {
Thread.sleep(sleepTime);
} catch (InterruptedException e1) {
logger.trace("retry sleeping got interrupted");
}
sleepTime *= delayFactor;
}
}
logger.warn("Retry failed after {} attempts", attempt);
throw originException;
}
protected boolean isCausedByOptimisticLockingFailure(Throwable throwable) {
if (targetExceptionClass == null) {
logger.warn("targetExceptionClass not configured, the retry interceptor is disabled.");
return false;
}
while (throwable != null) {
if (targetExceptionClass.isAssignableFrom(throwable.getClass())
|| OptimisticLockException.class.isAssignableFrom(throwable.getClass())) {
return true;
} else {
throwable = throwable.getCause();
}
}
return false;
}
protected boolean isCausedByConstraintViolationFailure(Throwable throwable) {
if (targetConstraintViolationExceptionClass == null) {
return false;
}
while (throwable != null) {
if (targetConstraintViolationExceptionClass.isAssignableFrom(throwable.getClass())) {
return true;
} else {
throwable = throwable.getCause();
}
}
return false;
}
public int getRetries() {
return retries;
}
public void setRetries(int retries) {
this.retries = retries;
}
public long getDelay() {
return delay;
}
public void setDelay(long delay) {
this.delay = delay;
}
public long getDelayFactor() {
return delayFactor;
}
public void setDelayFactor(long delayFactor) {
this.delayFactor = delayFactor;
}
public Class> getTargetExceptionClass() {
return targetExceptionClass;
}
public void setTargetExceptionClass(Class> targetExceptionClass) {
this.targetExceptionClass = targetExceptionClass;
}
protected boolean hasInterceptorInStack() {
return invocationsCounter.get().get() > 1;
}
}