Asynchronous Scalable Weakly-consistent Infection-style Process Group Membership Protocol(SWIM) implementation written in Rust.
Warning
Disclaimer: This project is a work in progress and not yet production-ready. The codebase may contain bugs or incomplete features. Use with caution. Feedback and contributions welcome, Thanks!.
io_uring
runtime via Tokio(SWIM:Basic)
+ (SWIM+Inf.)
+ (SWIM+Inf.+Susp.)
and extend to include features from Lifeguard by Hashicorp ResearchGossipod employs 3 types of messages: PING
, PING-REQ
, and BROADCAST
(which includes JOIN
, LEAVE
, SUSPECT
, ALIVE
, and CONFIRM
subtypes). The PING
and PING-REQ
messages are central to the system's failure detection mechanism base-off SWIM, facilitating constant state exchange by piggybacking information dissemination on this process. When a state change occurs, either through voluntary requests or regular failure detection, the system uses BROADCAST
messages for random dissemination of this information. Each node in the network maintains an incarnation number, starting at zero, which can only be incremented by the node itself. This number is crucial for managing the node's state in other nodes' local membership lists and serves as a means to refute suspicions (SWIM+Inf.+Susp.)
from other nodes. This design allows Gossipod to achieve efficient and resilient distributed state management and failure detection in a distributed or decentralized network, balancing the needs for up-to-date information, conflict resolution, and system reliability.
By Extension, Lifeguard offers additional implementation features that are not yet incorporated into the current version of Gossipod which are fully supported by Hashicorp memberlist. I plan to integrate these features in a future release.
sequenceDiagram
participant N1 as Node 1
participant N2 as Node 2
participant N3 as Node 3
participant Nx as Node x (Random)
Note over N1,Nx: Probing & Information Dissemination via piggybacking
N1->>N2: PING (with piggybacked info)
N2->>N1: ACK (with piggybacked info)
N1->>N3: PING-REQ (for N2)
N3->>N2: PING
N2->>N3: ACK
N3->>N1: ACK (indirect)
Note over N1,Nx: State Change Dissemination
N1->>Nx: BROADCAST (LEAVE/JOIN/SUSPECT/CONFIRM)
Nx->>N2: PING (with piggybacked state changes)
N2->>N3: PING (with piggybacked state changes)
Note over N1,Nx: Failure Detection & Recovery
N1->>N2: PING (no response)
N1->>N3: PING-REQ (for N2)
N3->>N2: PING (no response)
N3->>N1: NACK
N1->>Nx: BROADCAST (SUSPECT N2)
Note over N2: N2 receives SUSPECT message
N2->>N2: Increment incarnation number
N2->>Nx: BROADCAST (ALIVE with new incarnation number)
alt N2 successfully refutes
Nx->>N1: PING (with piggybacked ALIVE N2)
Note over N1: Update N2 status to alive
else N2 doesn't refute in time
N1->>Nx: BROADCAST (CONFIRM N2 as failed)
end
To see Gossipod in action, check out the ./examples
directory, which
includes a number of demos. Below you can run two different instances of gossipod specifiying different port address and node name
> cargo run -p ping_node -- --name=NODE_1 --port=7948
> cargo run -p pong_node -- --name=NODE_2 --port=7947 --join-addr=127.0.0.1:7948
let config = GossipodConfigBuilder::new()
.with_name(&args.name)
.with_port(args.port)
.with_addr(args.ip.parse::<Ipv4Addr>().expect("Invalid IP address"))
.with_probing_interval(Duration::from_secs(1))
.with_ack_timeout(Duration::from_millis(500))
.with_indirect_ack_timeout(Duration::from_secs(1))
.with_suspicious_timeout(Duration::from_secs(5))
.with_network_type(NetworkType::LAN)
.build()
.await?;
let gossipod = Gossipod::new(config).await?;
#[derive(Clone, Debug, Serialize, Deserialize, PartialEq, Eq, Hash)]
struct Metadata {
region: String,
datacenter: String,
}
impl NodeMetadata for Metadata {}
let metadata = Metadata {
region: "aws-west-1".to_string(),
datacenter: "dc1".to_string(),
};
let gossipod = Gossipod::with_metadata(config, metadata).await?;
Here's a basic example of how to use Gossipod in your application:
// Configuration
let config = GossipodConfigBuilder::new()
.with_name(&args.name)
.with_port(args.port)
.with_addr(args.ip.parse::<Ipv4Addr>().expect("Invalid IP address"))
.with_probing_interval(Duration::from_secs(1))
.with_ack_timeout(Duration::from_millis(500))
.with_indirect_ack_timeout(Duration::from_secs(1))
.with_suspicious_timeout(Duration::from_secs(5))
.with_network_type(NetworkType::LAN)
.build()
.await?;
// New instance
let gossipod = Arc::new(Gossipod::new(config.clone()).await?);
// Start Gossipod
tokio::spawn(gossipod.clone().start());
// Wait for Gossipod to start
while !gossipod.is_running().await {
time::sleep(Duration::from_millis(100)).await;
}