5 releases (3 breaking)

0.4.0 Jan 4, 2022
0.3.0 Oct 5, 2021
0.2.0 Sep 30, 2021
0.1.1 Jan 26, 2021
0.1.0 Jan 26, 2021

#738 in Asynchronous

Download history 371/week @ 2024-09-25 561/week @ 2024-10-02 342/week @ 2024-10-09 248/week @ 2024-10-16 373/week @ 2024-10-23 254/week @ 2024-10-30 334/week @ 2024-11-06 328/week @ 2024-11-13 604/week @ 2024-11-20 606/week @ 2024-11-27 305/week @ 2024-12-04 294/week @ 2024-12-11 88/week @ 2024-12-18 40/week @ 2024-12-25 213/week @ 2025-01-01 301/week @ 2025-01-08

662 downloads per month
Used in 2 crates

MIT license

20KB
258 lines

tourniquet

Easily round-robin between servers providing the same service, automatically reconnecting to the next server should an error happen.

This library facilitates resiliency to multiple service providers (e.g. servers) by connecting to the first available service from a list in a round-robin manner. If for some reason any provider goes down, any attempt to interact with it will reconnect to the next one, automatically.

Disclaimer: this library is not for load-balancing between a set of providers! It will connect to one provider, and only use this one provider as long as it is up. Tourniquet is meant for resiliency and not for load balancing.

Example

use async_trait::async_trait;
use std::{io::Error, net::IpAddr};
use tokio::{io::AsyncReadExt, net::TcpStream, sync::Mutex};
use tourniquet::{Connector, RoundRobin};

struct Conn(u16);

#[async_trait]
impl Connector<IpAddr, Mutex<TcpStream>, Error> for Conn {
    async fn connect(&self, src: &IpAddr) -> Result<Mutex<TcpStream>, Error> {
        let Conn(ref port) = self;
        TcpStream::connect((*src, *port)).await.map(Mutex::new)
    }
}

#[tokio::main]
async fn main() {
    let rr = RoundRobin::new(
        vec!["46.16.175.175".parse().unwrap(), "51.161.82.214".parse().unwrap()],
        Conn(6667),
    );

    let hello = rr.run(|sock| async move {
        let mut sock = sock.lock().await;
        let mut buf = [0; 50];
        sock.read_exact(&mut buf).await.map(|_| String::from_utf8(buf.to_vec()).unwrap())
    }).await.unwrap();

    assert!(hello.contains("libera.chat"));
}

License: MIT

Dependencies

~2.2–8.5MB
~64K SLoC