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

org.drools.persistence.jpa.OptimisticLockRetryInterceptor Maven / Gradle / Ivy

There is a newer version: 10.0.0
Show newest version
/*
 * 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.core.command.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 = 3; private long delay = 50; private long delayFactor = 4; 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 = 1; long sleepTime = delay; RuntimeException originException = null; while (attempt <= retries) { 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++; logger.trace("Command failed due to optimistic locking {} waiting {} millis before retry", ex, sleepTime); // save origin exception in case it needs to be rethrown if (originException == null) { originException = ex; } 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; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy