/*
 * Copyright 2015 Terracotta, Inc., a Software AG company.
 *
 * 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.terracotta.offheapstore.util;

import java.util.concurrent.ScheduledThreadPoolExecutor;
import java.util.concurrent.ThreadFactory;
import java.util.concurrent.TimeUnit;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

Author:cdennis
/** * * @author cdennis */
public class Retryer { private static final Logger LOGGER = LoggerFactory.getLogger(Retryer.class); private final ScheduledThreadPoolExecutor executor; private final long minimumDelay; private final long maximumDelay; private final TimeUnit unit; public Retryer(long minDelay, long maxDelay, TimeUnit unit, ThreadFactory threadFactory) { if (unit == null) { throw new IllegalArgumentException("Time unit must be non-null"); } if (minDelay <= 0) { throw new IllegalArgumentException("Minimum delay must be greater than zero"); } if (maxDelay < minDelay) { throw new IllegalArgumentException("Maximum delay cannot be less than minimum delay"); } if (threadFactory == null) { throw new IllegalArgumentException("Thread factory must be non-null"); } this.minimumDelay = minDelay; this.maximumDelay = maxDelay; this.unit = unit; this.executor = new ScheduledThreadPoolExecutor(1, threadFactory); } public void completeAsynchronously(Runnable task) { scheduleTask(task, 0); } public void shutdownNow() { executor.shutdownNow(); } private void scheduleTask(final Runnable task, final long delay) { if (executor.isShutdown()) { } executor.schedule(() -> { try { task.run(); } catch (Throwable t) { long nextDelay = nextDelay(delay); LOGGER.warn(task + " failed, retrying in " + nextDelay + " " + unit.toString().toLowerCase(), t); scheduleTask(task, nextDelay); } }, delay, unit); } private long nextDelay(long delay) { if (delay < minimumDelay) { return minimumDelay; } else if (delay >= maximumDelay) { return maximumDelay; } else { return Math.min(delay * 2, maximumDelay); } } }