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

com.hazelcast.jet.elastic.impl.RetryUtils Maven / Gradle / Ivy

There is a newer version: 5.5.0
Show newest version
/*
 * Copyright 2023 Hazelcast Inc.
 *
 * Licensed under the Hazelcast Community License (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 * http://hazelcast.com/hazelcast-community-license
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package com.hazelcast.jet.elastic.impl;

import com.hazelcast.core.HazelcastException;
import com.hazelcast.logging.ILogger;
import com.hazelcast.logging.Logger;

import java.io.IOException;
import java.util.concurrent.Callable;

import static com.hazelcast.jet.impl.util.ExceptionUtil.sneakyThrow;

/**
 * Static utility class to retry operations related to connecting to AWS Services.
 */
public final class RetryUtils {

    private static final ILogger LOGGER = Logger.getLogger(RetryUtils.class);

    private static final long MAX_BACKOFF_MS = 5 * 60 * 1000L;
    private static final long MS_IN_SECOND = 1000L;

    private RetryUtils() {
    }

    /**
     * Calls {@code callable.call()} until it does not throw an exception (but no more than {@code retries} times).
     */
    public static  T withRetry(Callable callable, int retries) {
        return withRetry(callable, retries, IOException.class);
    }

    /**
     * Calls {@code callable.call()} until it does not throw an exception (but no more than {@code retries} times).
     *
     */
    @SafeVarargs
    public static  T withRetry(Callable callable, int retries, Class ... exceptions) {
        int retryCount = 0;
        while (true) {
            try {
                return callable.call();
            } catch (Exception e) {
                if (anyOf(e, exceptions)) {
                    retryCount++;
                    if (retryCount > retries) {
                        throw sneakyThrow(e);
                    }
                    long waitIntervalMs = Math.min(MAX_BACKOFF_MS, MS_IN_SECOND * (1 << retryCount));
                    LOGGER.fine(String.format("Couldn't connect to Elastic, [%s] retrying in %s seconds...", retryCount,
                            waitIntervalMs / MS_IN_SECOND));
                    sleep(waitIntervalMs);
                } else {
                    throw sneakyThrow(e);
                }
            }
        }
    }

    private static boolean anyOf(Exception e, Class[] exceptions) {
        for (Class exception : exceptions) {
            if (exception.isAssignableFrom(e.getClass())) {
                return true;
            }
            if (e.getCause() != null && e.getCause() instanceof Exception && e.getCause() != e) {
                Exception cause = (Exception) e.getCause();
                return anyOf(cause, exceptions);
            }
        }
        return false;
    }

    private static void sleep(long millis) {
        try {
            Thread.sleep(millis);
        } catch (InterruptedException e) {
            Thread.currentThread().interrupt();
            throw new HazelcastException(e);
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy