backoff

Exponential backoff and retry.

Inspired by the retry mechanism in Google's google-http-java-client library and its Golang port.

Build Status crates.io

Usage

Just wrap your fallible operation into a closure, and call retry on it:

``` let mut op = || { println!("Fetching {}", url); let mut resp = reqwest::get(url)?; ... };

let mut backoff = ExponentialBackoff::default(); op.retry(&mut backoff ```

Examples

Permanent errors

Permanent errors are not retried. You have to wrap your error value explicitly into Error::Permanent. You can use Result's map_err method.

examples/permanent_error.rs:

``` extern crate backoff; extern crate reqwest;

use backoff::{Error, ExponentialBackoff, Operation}; use reqwest::IntoUrl;

use std::fmt::Display; use std::io::{self, Read};

fn newioerr(err: E) -> io::Error { io::Error::new(io::ErrorKind::Other, err.to_string()) }

fn fetchurl(url: &str) -> Result> { let mut op = || { println!("Fetching {}", url); let url = url.intourl() .maperr(newioerr) // Permanent errors need to be explicitly constucted. .maperr(Error::Permanent)?;

    let mut resp = reqwest::get(url)
        // Transient errors can be constructed with the ? operator
        // or with the try! macro. No explicit conversion needed
        // from E: Error to backoff::Error;
        .map_err(new_io_err)?;

    let mut content = String::new();
    let _ = resp.read_to_string(&mut content);
    Ok(content)
};

let mut backoff = ExponentialBackoff::default();
op.retry(&mut backoff)

}

fn main() { match fetchurl("https::///wrong URL") { Ok() => println!("Sucessfully fetched"), Err(err) => panic!("Failed to fetch: {}", err), } } ```

Output:

`` $ time cargo run --example permanent_error Finished dev [unoptimized + debuginfo] target(s) in 0.0 secs Runningtarget/debug/examples/permanenterror Fetching https::///wrong URL thread 'main' panicked at 'Failed to fetch: empty host', examples/permanent_error.rs:33 note: Run withRUSTBACKTRACE=1` for a backtrace.

real 0m0.151s user 0m0.116s sys 0m0.028s ```

Transient errors

Transient errors can be constructed by wrapping your error value into Error::Transient. By using the ? operator or the try! macro, you always get transient errors.

examples/retry.rs:

``` extern crate backoff; extern crate reqwest;

use backoff::{Error, ExponentialBackoff, Operation};

use std::io::Read;

fn fetch_url(url: &str) -> Result> { let mut op = || { println!("Fetching {}", url); let mut resp = reqwest::get(url)?;

    let mut content = String::new();
    let _ = resp.read_to_string(&mut content);
    Ok(content)
};

let mut backoff = ExponentialBackoff::default();
op.retry(&mut backoff)

}

fn main() { match fetchurl("https://www.rust-lang.org") { Ok() => println!("Sucessfully fetched"), Err(err) => panic!("Failed to fetch: {}", err), } } ```

Output with internet connection:

`` $ time cargo run --example retry Compiling backoff v0.1.0 (file:///home/tibi/workspace/backoff) Finished dev [unoptimized + debuginfo] target(s) in 1.54 secs Runningtarget/debug/examples/retry` Fetching https://www.rust-lang.org Sucessfully fetched

real 0m2.003s user 0m1.536s sys 0m0.184s ```

Output without internet connection

`` $ time cargo run --example retry Finished dev [unoptimized + debuginfo] target(s) in 0.0 secs Runningtarget/debug/examples/retry` Fetching https://www.rust-lang.org Fetching https://www.rust-lang.org Fetching https://www.rust-lang.org Fetching https://www.rust-lang.org ^C

real 0m2.826s user 0m0.008s sys 0m0.000s ```

License

Licensed under either of * Apache License, Version 2.0 (LICENSE-APACHE or http://www.apache.org/licenses/LICENSE-2.0) * MIT license (LICENSE-MIT or http://opensource.org/licenses/MIT) at your option.

Contribution

Unless you explicitly state otherwise, any contribution intentionally submitted for inclusion in the Work by You, as defined in the Apache-2.0 license, shall be dual licensed as above, without any additional terms or conditions.