-
Notifications
You must be signed in to change notification settings - Fork 476
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Add safekeeper membership conf to control file.
- Loading branch information
Showing
15 changed files
with
380 additions
and
140 deletions.
There are no files selected for viewing
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,163 @@ | ||
//! Types defining safekeeper membership, see | ||
//! rfcs/035-safekeeper-dynamic-membership-change.md | ||
//! for details. | ||
use std::{collections::HashSet, fmt::Display}; | ||
|
||
use anyhow; | ||
use anyhow::bail; | ||
use serde::{Deserialize, Serialize}; | ||
use utils::id::NodeId; | ||
|
||
/// Number uniquely identifying safekeeper configuration. | ||
/// Note: it is a part of sk control file. | ||
pub type Generation = u32; | ||
/// 1 is the first valid generation, 0 is used as | ||
/// a placeholder before we fully migrate to generations. | ||
pub const INVALID_GENERATION: Generation = 0; | ||
|
||
/// Membership is defined by ids so e.g. walproposer uses them to figure out | ||
/// quorums, but we also carry host and port to give wp idea where to connect. | ||
#[derive(Debug, Clone, Serialize, Deserialize, PartialEq)] | ||
pub struct SafekeeperId { | ||
pub id: NodeId, | ||
pub host: String, | ||
pub pg_port: u16, | ||
} | ||
|
||
impl Display for SafekeeperId { | ||
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { | ||
write!(f, "[id={}, ep={}:{}]", self.id, self.host, self.pg_port) | ||
} | ||
} | ||
|
||
/// Set of safekeepers. | ||
#[derive(Debug, Clone, Serialize, Deserialize, PartialEq)] | ||
#[serde(transparent)] | ||
pub struct MemberSet { | ||
pub members: Vec<SafekeeperId>, | ||
} | ||
|
||
impl MemberSet { | ||
pub fn empty() -> Self { | ||
MemberSet { | ||
members: Vec::new(), | ||
} | ||
} | ||
|
||
pub fn new(members: Vec<SafekeeperId>) -> anyhow::Result<Self> { | ||
let hs: HashSet<NodeId> = HashSet::from_iter(members.iter().map(|sk| sk.id)); | ||
if hs.len() != members.len() { | ||
bail!("duplicate safekeeper id in the set {:?}", members); | ||
} | ||
Ok(MemberSet { members }) | ||
} | ||
|
||
pub fn contains(&self, sk: &SafekeeperId) -> bool { | ||
self.members.iter().any(|m| m.id == sk.id) | ||
} | ||
|
||
pub fn add(&mut self, sk: SafekeeperId) -> anyhow::Result<()> { | ||
if self.contains(&sk) { | ||
bail!(format!( | ||
"sk {} is already member of the set {}", | ||
sk.id, self | ||
)); | ||
} | ||
self.members.push(sk); | ||
Ok(()) | ||
} | ||
} | ||
|
||
impl Display for MemberSet { | ||
/// Display as a comma separated list of members. | ||
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { | ||
let sks_str = self | ||
.members | ||
.iter() | ||
.map(|m| m.to_string()) | ||
.collect::<Vec<_>>(); | ||
write!(f, "({})", sks_str.join(", ")) | ||
} | ||
} | ||
|
||
/// Safekeeper membership configuration. | ||
/// Note: it is a part of both control file and http API. | ||
#[derive(Debug, Clone, Serialize, Deserialize, PartialEq)] | ||
pub struct Configuration { | ||
/// Unique id. | ||
pub generation: Generation, | ||
/// Current members of the configuration. | ||
pub members: MemberSet, | ||
/// Some means it is a joint conf. | ||
pub new_members: Option<MemberSet>, | ||
} | ||
|
||
impl Configuration { | ||
/// Used for pre-generations timelines, will be removed eventually. | ||
pub fn empty() -> Self { | ||
Configuration { | ||
generation: INVALID_GENERATION, | ||
members: MemberSet::empty(), | ||
new_members: None, | ||
} | ||
} | ||
} | ||
|
||
impl Display for Configuration { | ||
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { | ||
write!( | ||
f, | ||
"gen={}, members={}, new_members={}", | ||
self.generation, | ||
self.members, | ||
self.new_members | ||
.as_ref() | ||
.map(ToString::to_string) | ||
.unwrap_or(String::from("none")) | ||
) | ||
} | ||
} | ||
|
||
#[cfg(test)] | ||
mod tests { | ||
use super::{MemberSet, SafekeeperId}; | ||
use utils::id::NodeId; | ||
|
||
#[test] | ||
fn test_member_set() { | ||
let mut members = MemberSet::empty(); | ||
members | ||
.add(SafekeeperId { | ||
id: NodeId(42), | ||
host: String::from("lala.org"), | ||
pg_port: 5432, | ||
}) | ||
.unwrap(); | ||
|
||
members | ||
.add(SafekeeperId { | ||
id: NodeId(42), | ||
host: String::from("lala.org"), | ||
pg_port: 5432, | ||
}) | ||
.expect_err("duplicate must not be allowed"); | ||
|
||
members | ||
.add(SafekeeperId { | ||
id: NodeId(43), | ||
host: String::from("bubu.org"), | ||
pg_port: 5432, | ||
}) | ||
.unwrap(); | ||
|
||
println!("members: {}", members); | ||
|
||
let j = serde_json::to_string(&members).expect("failed to serialize"); | ||
println!("members json: {}", j); | ||
assert_eq!( | ||
j, | ||
r#"[{"id":42,"host":"lala.org","pg_port":5432},{"id":43,"host":"bubu.org","pg_port":5432}]"# | ||
); | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.