/* 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 * 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.streams.util.api.requests.backoff; import java.util.concurrent.atomic.AtomicInteger; /** * @see org.apache.streams.util.api.requests.backoff.BackOffStrategy */ public abstract class AbstractBackOffStrategy implements BackOffStrategy { private long baseSleepTime; private long lastSleepTime; private int maxAttempts; private AtomicInteger attemptsCount; /** * A BackOffStrategy that can effectively be used endlessly. * @param baseBackOffTime amount of time back of in seconds */ public AbstractBackOffStrategy(long baseBackOffTime) { this(baseBackOffTime, -1); } /** * A BackOffStrategy that has a limited number of uses before it throws a * {@link org.apache.streams.util.api.requests.backoff.BackOffException}. * @param baseBackOffTime time to back off in milliseconds, must be greater than 0. * @param maximumNumberOfBackOffAttempts maximum number of attempts, must be grater than 0 or -1. * -1 indicates there is no maximum number of attempts. */ public AbstractBackOffStrategy(long baseBackOffTime, int maximumNumberOfBackOffAttempts) { if (baseBackOffTime <= 0) { throw new IllegalArgumentException("backOffTimeInMilliSeconds is not greater than 0 : " + baseBackOffTime); } if (maximumNumberOfBackOffAttempts <= 0 && maximumNumberOfBackOffAttempts != -1) { throw new IllegalArgumentException("maximumNumberOfBackOffAttempts is not greater than 0 : " + maximumNumberOfBackOffAttempts); } this.baseSleepTime = baseBackOffTime; this.maxAttempts = maximumNumberOfBackOffAttempts; this.attemptsCount = new AtomicInteger(0); } @Override public void backOff() throws BackOffException { int attempt = this.attemptsCount.getAndIncrement(); if (attempt >= this.maxAttempts && this.maxAttempts != -1) { throw new BackOffException(attempt, this.lastSleepTime); } else { try { Thread.sleep(this.lastSleepTime = calculateBackOffTime(attempt, this.baseSleepTime)); } catch (InterruptedException ie) { Thread.currentThread().interrupt(); } } } @Override public void reset() { this.attemptsCount.set(0); } /** * Calculate the amount of time in milliseconds that the strategy should back off for * @param attemptCount the number of attempts the strategy has backed off. * i.e. 1 -> this is the first attempt, 2 -> this is the second attempt, etc. * @param baseSleepTime the minimum amount of time it should back off for in milliseconds * @return the amount of time it should back off in milliseconds */ protected abstract long calculateBackOffTime(int attemptCount, long baseSleepTime); }