#adaptive #backoff #ratelimit #throttle

adaptive_backoff

library for intelligently backing off rate limited or fallible resources

3 unstable releases

0.2.1 Feb 23, 2021
0.2.0 Dec 21, 2020
0.1.0 Dec 19, 2020

#1272 in Rust patterns

Download history 1395/week @ 2023-10-14 1006/week @ 2023-10-21 2835/week @ 2023-10-28 1513/week @ 2023-11-04 3244/week @ 2023-11-11 1643/week @ 2023-11-18 808/week @ 2023-11-25 881/week @ 2023-12-02 1088/week @ 2023-12-09 910/week @ 2023-12-16 373/week @ 2023-12-23 191/week @ 2023-12-30 963/week @ 2024-01-06 1685/week @ 2024-01-13 914/week @ 2024-01-20 987/week @ 2024-01-27

4,583 downloads per month

MIT license

17KB
276 lines

Adaptive Backoff

Adaptive backoff provides a mechanism to intelligently back off use of rate limited or failure operations through sources simples structs, Backoffs. Backoffs take input on success and failure, then return a duration to wait.

For an adaptive backoff, as the failure and success calls increase the returned duration eventually converges on a value to avoid rate limiting of requests.

Usage

Include within your Cargo.toml:

adaptive_backoff = "0.2"

And follow an example below:

Adaptive ExponentialBackoff

Below is an example of an adapative ExponentialBackoff which works from a queue and converges on a minimal delay duration between calls. It grows by a factor of 2.0 to a maximum of 300 seconds.

use std::time::Duration;
use adaptive_backoff::prelude::*;

let mut backoff = ExponentialBackoffBuilder::default()
    .factor(2.0)
    .max(Duration::from_secs(300))
    .adaptive()
    .build()
    .unwrap();

while let Some(item) = queue.pop() {
    loop {
        match worker_iter(&conn, &item).await {
            Ok(_) => {
                delay_for(backoff.success()).await;
                break;
            }
            Err(_) => delay_for(backoff.fail()).await,
        }
    }
}

Simple ExponentialBackoff

If adaptive is omitted from the example above, a simple backoff is returned. Its API lacks success() and fail(), instead it can only return increasing delays with wait() until reset() is called to return it.

use std::time::Duration;
use adaptive_backoff::prelude::*;

let mut backoff = ExponentialBackoffBuilder::default()
    .factor(2.0)
    .max(Duration::from_secs(30))
    .build()
    .unwrap();

while let Some(item) = queue.pop() {
    loop {
        match worker_iter(&conn, &item).await {
            Ok(_) => {
                delay_for(backoff.wait()).await;
                break;
            }
            Err(_) => delay_for(backoff.wait()).await,
        }
    }

    backoff.reset();
}

Additional Examples

There are tests for backoff implementations which contain example use of the external API with expected output. See both the simple exponential example and the adaptive exponential backoff example.

Dependencies

~0.5–1MB
~21K SLoC