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

org.apache.kafka.connect.util.RetryUtil Maven / Gradle / Ivy

The newest version!
/*
 * 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); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy