com.landawn.abacus.util.Retry Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of abacus-android Show documentation
Show all versions of abacus-android Show documentation
A general programming library in Java
/*
* Copyright (C) 2016 HaiYang Li
*
* 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 com.landawn.abacus.util;
import java.util.Iterator;
import java.util.concurrent.Callable;
import com.landawn.abacus.logging.Logger;
import com.landawn.abacus.logging.LoggerFactory;
import com.landawn.abacus.util.function.BiFunction;
import com.landawn.abacus.util.function.Function;
/**
*
* @since 0.8
*
* @author Haiyang Li
*/
public final class Retry {
private static final Logger logger = LoggerFactory.getLogger(Retry.class);
private final int retryTimes;
private final long retryInterval;
private final Function super Throwable, Boolean> retryCondition;
private final BiFunction super T, ? super Throwable, Boolean> retryCondition2;
Retry(final int retryTimes, final long retryInterval, final Function super Throwable, Boolean> retryCondition,
final BiFunction super T, ? super Throwable, Boolean> retryCondition2) {
this.retryTimes = retryTimes;
this.retryInterval = retryInterval;
this.retryCondition = retryCondition;
this.retryCondition2 = retryCondition2;
}
public static Retry of(final int retryTimes, final long retryInterval, final Function super Throwable, Boolean> retryCondition) {
if (retryTimes < 0 || retryInterval < 0) {
throw new IllegalArgumentException("'retryTimes' and 'retryInterval' can't be negative");
}
N.requireNonNull(retryCondition);
return new Retry(retryTimes, retryInterval, retryCondition, null);
}
public static Retry of(final int retryTimes, final long retryInterval, final BiFunction super R, ? super Throwable, Boolean> retryCondition) {
if (retryTimes < 0 || retryInterval < 0) {
throw new IllegalArgumentException("'retryTimes' and 'retryInterval' can't be negative");
}
N.requireNonNull(retryCondition);
return new Retry(retryTimes, retryInterval, null, retryCondition);
}
// /**
// *
// * @param runnable
// * @param retryTimes Default value is 1.
// * @param retryInterval Default value is 0.
// * @param retryCondition
// * @return
// */
// public static Runnable of(final Runnable runnable, final int retryTimes, final long retryInterval, final Function retryCondition) {
// if (retryTimes < 0 || retryInterval < 0) {
// throw new IllegalArgumentException("'retryTimes' and 'retryInterval' can't be negative");
// }
//
// return new Runnable() {
// @Override
// public void run() {
// try {
// runnable.run();
// } catch (Throwable e) {
// logger.error("AutoRetry", e);
//
// int retriedTimes = 0;
// Throwable throwable = e;
//
// while (retriedTimes++ < retryTimes && retryCondition.apply(throwable)) {
// try {
// if (retryInterval > 0) {
// N.sleep(retryInterval);
// }
//
// runnable.run();
// return;
// } catch (Throwable e2) {
// logger.error("AutoRetry", e2);
//
// throwable = e2;
// }
// }
//
// throw N.toRuntimeException(throwable);
// }
// }
// };
// }
//
// /**
// *
// * @param callable
// * @param retryTimes Default value is 1.
// * @param retryInterval Default value is 0.
// * @param retryCondition
// * @return
// */
// public static Try.Callable of(final Try.Callable callable, final int retryTimes, final long retryInterval,
// final BiFunction retryCondition) {
// if (retryTimes < 0 || retryInterval < 0) {
// throw new IllegalArgumentException("'retryTimes' and 'retryInterval' can't be negative");
// }
//
// return new Try.Callable() {
// @Override
// public T call() {
// T result = null;
// int retriedTimes = 0;
//
// try {
// result = callable.call();
//
// while (retriedTimes++ < retryTimes && (retryCondition.apply(null, result))) {
// if (retryInterval > 0) {
// N.sleep(retryInterval);
// }
//
// result = callable.call();
//
// if (retryCondition.apply(null, result) == false) {
// return result;
// }
// }
// } catch (Throwable e) {
// logger.error("AutoRetry", e);
//
// Throwable throwable = e;
//
// while (retriedTimes++ < retryTimes && retryCondition.apply(throwable, result)) {
// try {
// if (retryInterval > 0) {
// N.sleep(retryInterval);
// }
//
// result = callable.call();
//
// if (retryCondition.apply(null, result) == false) {
// return result;
// }
// } catch (Throwable e2) {
// logger.error("AutoRetry", e2);
//
// throwable = e2;
// }
// }
//
// throw N.toRuntimeException(throwable);
// }
//
// if (retryTimes > 0 && retryCondition.apply(null, result)) {
// throw new RuntimeException("Still failed after retried " + retryTimes + " times for result: " + N.toString(result));
// }
//
// return result;
// }
// };
// }
// public static Iterator of(final Iterator iter, final int retryTimes, final long retryInterval,
// final Function retryCondition) {
// return new Iterator() {
// @Override
// public boolean hasNext() {
// try {
// return iter.hasNext();
// } catch (Throwable e) {
// logger.error("hasNext", e);
//
// int retriedTimes = 0;
// Throwable throwable = e;
//
// while (retriedTimes++ < retryTimes && retryCondition.apply(throwable)) {
// try {
// if (retryInterval > 0) {
// N.sleep(retryInterval);
// }
//
// return iter.hasNext();
// } catch (Throwable e2) {
// logger.error("hasNext", e2);
//
// throwable = e2;
// }
// }
//
// throw N.toRuntimeException(throwable);
// }
// }
//
// @Override
// public T next() {
// try {
// return iter.next();
// } catch (Throwable e) {
// logger.error("next", e);
//
// int retriedTimes = 0;
// Throwable throwable = e;
//
// while (retriedTimes++ < retryTimes && retryCondition.apply(throwable)) {
// try {
// if (retryInterval > 0) {
// N.sleep(retryInterval);
// }
//
// return iter.next();
// } catch (Throwable e2) {
// logger.error("next", e2);
//
// throwable = e2;
// }
// }
//
// throw N.toRuntimeException(throwable);
// }
// }
//
// @Override
// public void remove() {
// try {
// iter.remove();
// } catch (Throwable e) {
// logger.error("remove", e);
//
// int retriedTimes = 0;
// Throwable throwable = e;
//
// while (retriedTimes++ < retryTimes && retryCondition.apply(throwable)) {
// try {
// if (retryInterval > 0) {
// N.sleep(retryInterval);
// }
//
// iter.remove();
// } catch (Throwable e2) {
// logger.error("remove", e2);
//
// throwable = e2;
// }
// }
//
// throw N.toRuntimeException(throwable);
// }
// }
// };
// }
public void run(final Runnable cmd) {
try {
cmd.run();
} catch (RuntimeException e) {
logger.error("AutoRetry", e);
int retriedTimes = 0;
RuntimeException throwable = e;
while (retriedTimes++ < retryTimes
&& ((retryCondition != null && retryCondition.apply(throwable)) || (retryCondition2 != null && retryCondition2.apply(null, throwable)))) {
try {
if (retryInterval > 0) {
N.sleep(retryInterval);
}
cmd.run();
return;
} catch (RuntimeException e2) {
logger.error("AutoRetry", e2);
throwable = e2;
}
}
throw throwable;
}
}
public T call(final Try.Callable callable) {
T result = null;
int retriedTimes = 0;
try {
result = callable.call();
while (retriedTimes++ < retryTimes && (retryCondition2 != null && retryCondition2.apply(result, null))) {
if (retryInterval > 0) {
N.sleep(retryInterval);
}
result = callable.call();
if (retryCondition2 == null || retryCondition2.apply(result, null) == false) {
return result;
}
}
} catch (RuntimeException e) {
logger.error("AutoRetry", e);
RuntimeException throwable = e;
while (retriedTimes++ < retryTimes
&& ((retryCondition != null && retryCondition.apply(throwable)) || (retryCondition2 != null && retryCondition2.apply(null, throwable)))) {
try {
if (retryInterval > 0) {
N.sleep(retryInterval);
}
result = callable.call();
if (retryCondition2 == null || retryCondition2.apply(result, null) == false) {
return result;
}
} catch (RuntimeException e2) {
logger.error("AutoRetry", e2);
throwable = e2;
}
}
throw throwable;
}
if (retryTimes > 0 && (retryCondition2 != null && retryCondition2.apply(result, null))) {
throw new RuntimeException("Still failed after retried " + retryTimes + " times for result: " + N.toString(result));
}
return result;
}
public Iterator iterator(final Iterator iter) {
return new Iterator() {
@Override
public boolean hasNext() {
try {
return iter.hasNext();
} catch (RuntimeException e) {
logger.error("hasNext", e);
int retriedTimes = 0;
RuntimeException throwable = e;
while (retriedTimes++ < retryTimes && ((retryCondition != null && retryCondition.apply(throwable))
|| (retryCondition2 != null && retryCondition2.apply(null, throwable)))) {
try {
if (retryInterval > 0) {
N.sleep(retryInterval);
}
return iter.hasNext();
} catch (RuntimeException e2) {
logger.error("hasNext", e2);
throwable = e2;
}
}
throw throwable;
}
}
@Override
public E next() {
try {
return iter.next();
} catch (RuntimeException e) {
logger.error("next", e);
int retriedTimes = 0;
RuntimeException throwable = e;
while (retriedTimes++ < retryTimes && ((retryCondition != null && retryCondition.apply(throwable))
|| (retryCondition2 != null && retryCondition2.apply(null, throwable)))) {
try {
if (retryInterval > 0) {
N.sleep(retryInterval);
}
return iter.next();
} catch (RuntimeException e2) {
logger.error("next", e2);
throwable = e2;
}
}
throw throwable;
}
}
@Override
public void remove() {
try {
iter.remove();
} catch (RuntimeException e) {
logger.error("remove", e);
int retriedTimes = 0;
RuntimeException throwable = e;
while (retriedTimes++ < retryTimes && ((retryCondition != null && retryCondition.apply(throwable))
|| (retryCondition2 != null && retryCondition2.apply(null, throwable)))) {
try {
if (retryInterval > 0) {
N.sleep(retryInterval);
}
iter.remove();
} catch (RuntimeException e2) {
logger.error("remove", e2);
throwable = e2;
}
}
throw throwable;
}
}
};
}
public static final class Retry0 {
private final int retryTimes;
private final long retryInterval;
private final Function super Throwable, Boolean> retryCondition;
private final BiFunction super T, ? super Throwable, Boolean> retryCondition2;
Retry0(final int retryTimes, final long retryInterval, final Function super Throwable, Boolean> retryCondition,
final BiFunction super T, ? super Throwable, Boolean> retryCondition2) {
this.retryTimes = retryTimes;
this.retryInterval = retryInterval;
this.retryCondition = retryCondition;
this.retryCondition2 = retryCondition2;
}
public static Retry0 of(final int retryTimes, final long retryInterval, final Function super Throwable, Boolean> retryCondition) {
if (retryTimes < 0 || retryInterval < 0) {
throw new IllegalArgumentException("'retryTimes' and 'retryInterval' can't be negative");
}
N.requireNonNull(retryCondition);
return new Retry0(retryTimes, retryInterval, retryCondition, null);
}
public static Retry0 of(final int retryTimes, final long retryInterval, final BiFunction super R, ? super Throwable, Boolean> retryCondition) {
if (retryTimes < 0 || retryInterval < 0) {
throw new IllegalArgumentException("'retryTimes' and 'retryInterval' can't be negative");
}
N.requireNonNull(retryCondition);
return new Retry0(retryTimes, retryInterval, null, retryCondition);
}
public void run(final Try.Runnable extends Exception> cmd) throws Exception {
try {
cmd.run();
} catch (Exception e) {
logger.error("AutoRetry", e);
int retriedTimes = 0;
Exception throwable = e;
while (retriedTimes++ < retryTimes && ((retryCondition != null && retryCondition.apply(throwable))
|| (retryCondition2 != null && retryCondition2.apply(null, throwable)))) {
try {
if (retryInterval > 0) {
N.sleep(retryInterval);
}
cmd.run();
return;
} catch (Exception e2) {
logger.error("AutoRetry", e2);
throwable = e2;
}
}
throw throwable;
}
}
public T call(final Callable callable) throws Exception {
T result = null;
int retriedTimes = 0;
try {
result = callable.call();
while (retriedTimes++ < retryTimes && (retryCondition2 != null && retryCondition2.apply(result, null))) {
if (retryInterval > 0) {
N.sleep(retryInterval);
}
result = callable.call();
if (retryCondition2 == null || retryCondition2.apply(result, null) == false) {
return result;
}
}
} catch (Exception e) {
logger.error("AutoRetry", e);
Exception throwable = e;
while (retriedTimes++ < retryTimes && ((retryCondition != null && retryCondition.apply(throwable))
|| (retryCondition2 != null && retryCondition2.apply(null, throwable)))) {
try {
if (retryInterval > 0) {
N.sleep(retryInterval);
}
result = callable.call();
if (retryCondition2 == null || retryCondition2.apply(result, null) == false) {
return result;
}
} catch (Exception e2) {
logger.error("AutoRetry", e2);
throwable = e2;
}
}
throw throwable;
}
if (retryTimes > 0 && (retryCondition2 != null && retryCondition2.apply(result, null))) {
throw new RuntimeException("Still failed after retried " + retryTimes + " times for result: " + N.toString(result));
}
return result;
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy