org.apache.kafka.connect.util.RetryUtil Maven / Gradle / Ivy
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You 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.apache.kafka.connect.util;
import org.apache.kafka.common.errors.RetriableException;
import org.apache.kafka.common.errors.WakeupException;
import org.apache.kafka.common.utils.Time;
import org.apache.kafka.connect.errors.ConnectException;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.time.Duration;
import java.util.Optional;
import java.util.concurrent.Callable;
import java.util.function.Supplier;
public class RetryUtil {
private static final Logger log = LoggerFactory.getLogger(RetryUtil.class);
/**
* The method executes the callable at least once, optionally retrying the callable if
* {@link org.apache.kafka.connect.errors.RetriableException} is being thrown. If timeout is exhausted,
* then the last exception is wrapped with a {@link org.apache.kafka.connect.errors.ConnectException} and thrown.
*
* {@code description} supplies the message that indicates the purpose of the callable since the message will
* be used for logging. For example, "list offsets". If the supplier is null or the supplied string is
* null, {@code callable} will be used as the default string.
*
*
The task will be executed at least once. No retries will be performed
* if {@code timeoutDuration} is 0 or negative, or if {@code timeoutDuration} is less than {@code retryBackoffMs}.
*
*
A {@code retryBackoffMs} that is negative or zero will result in no delays between retries.
*
* @param callable the function to execute
* @param description supplier that provides custom message for logging purpose
* @param timeoutDuration timeout duration; must not be null
* @param retryBackoffMs the number of milliseconds to delay upon receiving a
* {@link org.apache.kafka.connect.errors.RetriableException} before retrying again
* @throws ConnectException If the task exhausted all the retries
*/
public static T retryUntilTimeout(Callable callable, Supplier description, Duration timeoutDuration, long retryBackoffMs) throws Exception {
return retryUntilTimeout(callable, description, timeoutDuration, retryBackoffMs, Time.SYSTEM);
}
// visible for testing
static T retryUntilTimeout(Callable callable, Supplier description, Duration timeoutDuration, long retryBackoffMs, Time time) throws Exception {
// if null supplier or string is provided, the message will be default to "callabe"
final String descriptionStr = Optional.ofNullable(description)
.map(Supplier::get)
.orElse("callable");
// handling null duration
final long timeoutMs = Optional.ofNullable(timeoutDuration)
.map(Duration::toMillis)
.orElse(0L);
if (retryBackoffMs < 0) {
log.debug("Assuming no retry backoff since retryBackoffMs={} is negative", retryBackoffMs);
retryBackoffMs = 0;
}
if (timeoutMs <= 0 || retryBackoffMs >= timeoutMs) {
log.debug("Executing {} only once, since timeoutMs={} is not larger than retryBackoffMs={}",
descriptionStr, timeoutMs, retryBackoffMs);
return callable.call();
}
final long end = time.milliseconds() + timeoutMs;
int attempt = 0;
Throwable lastError;
do {
attempt++;
try {
return callable.call();
} catch (RetriableException | org.apache.kafka.connect.errors.RetriableException e) {
log.warn("Attempt {} to {} resulted in RetriableException; retrying automatically. " +
"Reason: {}", attempt, descriptionStr, e.getMessage(), e);
lastError = e;
} catch (WakeupException e) {
lastError = e;
}
if (retryBackoffMs > 0) {
long millisRemaining = Math.max(0, end - time.milliseconds());
if (millisRemaining < retryBackoffMs) {
// exit when the time remaining is less than retryBackoffMs
break;
}
time.sleep(retryBackoffMs);
}
} while (time.milliseconds() < end);
throw new ConnectException("Fail to " + descriptionStr + " after " + attempt + " attempts. Reason: " + lastError.getMessage(), lastError);
}
}