10 releases

Uses old Rust 2015

0.3.6 Aug 1, 2024
0.3.5 May 24, 2024
0.3.3 Nov 12, 2023
0.3.2 Jul 20, 2023
0.1.0 Nov 27, 2016

#75 in Caching

Download history 1327/week @ 2024-09-18 1394/week @ 2024-09-25 1302/week @ 2024-10-02 1602/week @ 2024-10-09 2027/week @ 2024-10-16 2175/week @ 2024-10-23 1823/week @ 2024-10-30 2211/week @ 2024-11-06 2835/week @ 2024-11-13 1955/week @ 2024-11-20 2217/week @ 2024-11-27 2703/week @ 2024-12-04 2653/week @ 2024-12-11 1956/week @ 2024-12-18 664/week @ 2024-12-25 1428/week @ 2025-01-01

7,373 downloads per month
Used in 11 crates (7 directly)

MIT license

21KB
351 lines

HashRing

Build Status codecov crates.io docs.rs License

Documentation

A minimal implementation of consistent hashing as described in Consistent Hashing and Random Trees: Distributed Caching Protocols for Relieving Hot Spots on the World Wide Web. Clients can use the HashRing struct to add consistent hashing to their applications. HashRing's API consists of three methods: add, remove, and get for adding a node to the ring, removing a node from the ring, and getting the node responsible for the provided key.

Example

Below is a simple example of how an application might use HashRing to make use of consistent hashing. Since HashRing exposes only a minimal API clients can build other abstractions, such as virtual nodes, on top of it. The example below shows one potential implementation of virtual nodes on top of HashRing

use std::net::{IpAddr, SocketAddr};
use std::str::FromStr;

use hashring::HashRing;

#[derive(Debug, Copy, Clone, Hash, PartialEq)]
struct VNode {
    id: usize,
    addr: SocketAddr,
}

impl VNode {
    fn new(ip: &str, port: u16, id: usize) -> Self {
        let addr = SocketAddr::new(IpAddr::from_str(&ip).unwrap(), port);
        VNode {
            id: id,
            addr: addr,
        }
    }
}

impl ToString for VNode {
    fn to_string(&self) -> String {
        format!("{}|{}", self.addr, self.id)
    }
}

fn main() {
    let mut ring: HashRing<VNode> = HashRing::new();

    let mut nodes = vec![];
    nodes.push(VNode::new("127.0.0.1", 1024, 1));
    nodes.push(VNode::new("127.0.0.1", 1024, 2));
    nodes.push(VNode::new("127.0.0.2", 1024, 1));
    nodes.push(VNode::new("127.0.0.2", 1024, 2));
    nodes.push(VNode::new("127.0.0.2", 1024, 3));
    nodes.push(VNode::new("127.0.0.3", 1024, 1));

    for node in nodes {
        ring.add(node);
    }

    println!("{:?}", ring.get(&"foo"));
    println!("{:?}", ring.get(&"bar"));
    println!("{:?}", ring.get(&"baz"));
}

Dependencies

~56KB