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

org.springframework.retry.backoff.BackOffPolicyBuilder Maven / Gradle / Ivy

Go to download

Spring Retry provides an abstraction around retrying failed operations, with an emphasis on declarative control of the process and policy-based bahaviour that is easy to extend and customize. For instance, you can configure a plain POJO operation to retry if it fails, based on the type of exception, and with a fixed or exponential backoff.

The newest version!
/*
 * Copyright 2022-2022 the original author or authors.
 *
 * 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
 *
 *      https://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.springframework.retry.backoff;

import java.util.function.Supplier;

/**
 * Fluent API for creating a {@link BackOffPolicy} based on given attributes. The delay
 * values are expressed in milliseconds. If any provided value is less than one, the
 * resulting policy will set it to one. The default policy is a {@link FixedBackOffPolicy}
 * with a delay of 1000ms.
 *
 * 

* Examples:

 *
 * // Default {@link FixedBackOffPolicy} with 1000ms delay
 * BackOffPolicyBuilder
 * 		.newDefaultPolicy();
 *
 * // {@link FixedBackOffPolicy}
 * BackOffPolicyBuilder
 * 		.newBuilder()
 * 		.delay(2000)
 * 		.build();
 *
 * // {@link UniformRandomBackOffPolicy}
 * BackOffPolicyBuilder
 * 		.newBuilder()
 * 		.delay(500)
 * 		.maxDelay(1000)
 * 		.build();
 *
 * // {@link ExponentialBackOffPolicy}
 * BackOffPolicyBuilder
 * 		.newBuilder()
 * 		.delay(1000)
 * 		.maxDelay(5000)
 * 		.multiplier(2)
 * 		.build();
 *
 * // {@link ExponentialRandomBackOffPolicy} with provided {@link Sleeper}
 * BackOffPolicyBuilder
 * 		.newBuilder()
 * 		.delay(3000)
 * 		.maxDelay(5000)
 * 		.multiplier(1.5)
 * 		.random(true)
 * 		.sleeper(mySleeper)
 * 		.build();
 * 
*

* Not thread safe. Building should be performed in a single thread. The resulting * {@link BackOffPolicy} however is expected to be thread-safe and designed for moderate * load concurrent access. * * @author Tomaz Fernandes * @author Aftab Shaikh * @since 1.3.3 */ public class BackOffPolicyBuilder { private static final long DEFAULT_INITIAL_DELAY = 1000L; private Long delay = DEFAULT_INITIAL_DELAY; private Long maxDelay; private Double multiplier; private Boolean random; private Sleeper sleeper; private Supplier delaySupplier; private Supplier maxDelaySupplier; private Supplier multiplierSupplier; private Supplier randomSupplier; private BackOffPolicyBuilder() { } /** * Creates a new {@link BackOffPolicyBuilder} instance. * @return the builder instance */ public static BackOffPolicyBuilder newBuilder() { return new BackOffPolicyBuilder(); } /** * Creates a new {@link FixedBackOffPolicy} instance with a delay of 1000ms. * @return the back off policy instance */ public static BackOffPolicy newDefaultPolicy() { return new BackOffPolicyBuilder().build(); } /** * A canonical backoff period. Used as an initial value in the exponential case, and * as a minimum value in the uniform case. * @param delay the initial or canonical backoff period in milliseconds * @return this */ public BackOffPolicyBuilder delay(long delay) { this.delay = delay; return this; } /** * The maximum wait in milliseconds between retries. If less than {@link #delay(long)} * then a default value is applied depending on the resulting policy. * @param maxDelay the maximum wait between retries in milliseconds * @return this */ public BackOffPolicyBuilder maxDelay(long maxDelay) { this.maxDelay = maxDelay; return this; } /** * If positive, then used as a multiplier for generating the next delay for backoff. * @param multiplier a multiplier to use to calculate the next backoff delay * @return this */ public BackOffPolicyBuilder multiplier(double multiplier) { this.multiplier = multiplier; return this; } /** * In the exponential case ({@link #multiplier} > 0) set this to true to have the * backoff delays randomized, so that the maximum delay is multiplier times the * previous delay and the distribution is uniform between the two values. * @param random the flag to signal randomization is required * @return this */ public BackOffPolicyBuilder random(boolean random) { this.random = random; return this; } /** * The {@link Sleeper} instance to be used to back off. Policies default to * {@link ThreadWaitSleeper}. * @param sleeper the {@link Sleeper} instance * @return this */ public BackOffPolicyBuilder sleeper(Sleeper sleeper) { this.sleeper = sleeper; return this; } /** * Set a supplier for the delay. * @param delaySupplier the supplier. * @return this * @since 2.0 */ public BackOffPolicyBuilder delaySupplier(Supplier delaySupplier) { this.delaySupplier = delaySupplier; return this; } /** * Set a supplier for the max delay. * @param maxDelaySupplier the supplier. * @return this * @since 2.0 */ public BackOffPolicyBuilder maxDelaySupplier(Supplier maxDelaySupplier) { this.maxDelaySupplier = maxDelaySupplier; return this; } /** * Set a supplier for the multiplier. * @param multiplierSupplier the supplier. * @return this * @since 2.0 */ public BackOffPolicyBuilder multiplierSupplier(Supplier multiplierSupplier) { this.multiplierSupplier = multiplierSupplier; return this; } /** * Set a supplier for the random. * @param randomSupplier the supplier. * @return this * @since 2.0 */ public BackOffPolicyBuilder randomSupplier(Supplier randomSupplier) { this.randomSupplier = randomSupplier; return this; } /** * Builds the {@link BackOffPolicy} with the given parameters. * @return the {@link BackOffPolicy} instance */ public BackOffPolicy build() { if (this.multiplier != null && this.multiplier > 0 || this.multiplierSupplier != null) { ExponentialBackOffPolicy policy; if (isRandom()) { policy = new ExponentialRandomBackOffPolicy(); } else { policy = new ExponentialBackOffPolicy(); } if (this.delay != null) { policy.setInitialInterval(this.delay); } if (this.delaySupplier != null) { policy.initialIntervalSupplier(this.delaySupplier); } if (this.multiplier != null) { policy.setMultiplier(this.multiplier); } if (this.multiplierSupplier != null) { policy.multiplierSupplier(this.multiplierSupplier); } if (this.maxDelay != null && this.delay != null) { policy.setMaxInterval( this.maxDelay > this.delay ? this.maxDelay : ExponentialBackOffPolicy.DEFAULT_MAX_INTERVAL); } if (this.maxDelaySupplier != null) { policy.maxIntervalSupplier(this.maxDelaySupplier); } if (this.sleeper != null) { policy.setSleeper(this.sleeper); } return policy; } if (this.maxDelay != null && this.delay != null && this.maxDelay > this.delay) { UniformRandomBackOffPolicy policy = new UniformRandomBackOffPolicy(); if (this.delay != null) { policy.setMinBackOffPeriod(this.delay); } if (this.delaySupplier != null) { policy.minBackOffPeriodSupplier(this.delaySupplier); } if (this.maxDelay != null) { policy.setMaxBackOffPeriod(this.maxDelay); } if (this.maxDelaySupplier != null) { policy.maxBackOffPeriodSupplier(this.maxDelaySupplier); } if (this.sleeper != null) { policy.setSleeper(this.sleeper); } return policy; } FixedBackOffPolicy policy = new FixedBackOffPolicy(); if (this.delaySupplier != null) { policy.backOffPeriodSupplier(this.delaySupplier); } else if (this.delay != null) { policy.setBackOffPeriod(this.delay); } if (this.sleeper != null) { policy.setSleeper(this.sleeper); } return policy; } private boolean isRandom() { return (this.randomSupplier != null && Boolean.TRUE.equals(this.randomSupplier.get())) || Boolean.TRUE.equals(this.random); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy