2024-03-04 19:58:32 +01:00
|
|
|
use garage_util::config::Config;
|
|
|
|
use garage_util::crdt::AutoCrdt;
|
|
|
|
use garage_util::error::*;
|
|
|
|
use serde::{Deserialize, Serialize};
|
|
|
|
|
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Serialize, Deserialize)]
|
|
|
|
#[serde(transparent)]
|
|
|
|
pub struct ReplicationFactor(usize);
|
|
|
|
|
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Default, Serialize, Deserialize)]
|
|
|
|
#[serde(rename_all = "lowercase")]
|
|
|
|
pub enum ConsistencyMode {
|
|
|
|
/// Read- and Write-quorum are 1
|
|
|
|
Dangerous,
|
|
|
|
/// Read-quorum is 1
|
|
|
|
Degraded,
|
|
|
|
/// Read- and Write-quorum are determined for read-after-write-consistency
|
|
|
|
#[default]
|
|
|
|
Consistent,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl ConsistencyMode {
|
|
|
|
pub fn parse(s: &str) -> Option<Self> {
|
|
|
|
serde_json::from_value(serde_json::Value::String(s.to_string())).ok()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl AutoCrdt for ConsistencyMode {
|
|
|
|
const WARN_IF_DIFFERENT: bool = true;
|
2021-05-28 12:36:22 +02:00
|
|
|
}
|
|
|
|
|
2024-03-04 19:58:32 +01:00
|
|
|
impl ReplicationFactor {
|
|
|
|
pub fn new(replication_factor: usize) -> Option<Self> {
|
|
|
|
if replication_factor < 1 {
|
|
|
|
None
|
|
|
|
} else {
|
|
|
|
Some(Self(replication_factor))
|
2021-05-28 12:36:22 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn replication_factor(&self) -> usize {
|
2024-03-04 19:58:32 +01:00
|
|
|
self.0
|
2021-05-28 12:36:22 +02:00
|
|
|
}
|
|
|
|
|
2024-03-04 19:58:32 +01:00
|
|
|
pub fn read_quorum(&self, consistency_mode: ConsistencyMode) -> usize {
|
|
|
|
match consistency_mode {
|
|
|
|
ConsistencyMode::Dangerous | ConsistencyMode::Degraded => 1,
|
|
|
|
ConsistencyMode::Consistent => self.replication_factor().div_ceil(2),
|
2021-05-28 12:36:22 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-04 19:58:32 +01:00
|
|
|
pub fn write_quorum(&self, consistency_mode: ConsistencyMode) -> usize {
|
|
|
|
match consistency_mode {
|
|
|
|
ConsistencyMode::Dangerous => 1,
|
|
|
|
ConsistencyMode::Degraded | ConsistencyMode::Consistent => {
|
|
|
|
(self.replication_factor() + 1) - self.read_quorum(ConsistencyMode::Consistent)
|
|
|
|
}
|
2021-05-28 12:36:22 +02:00
|
|
|
}
|
|
|
|
}
|
2024-03-04 19:58:32 +01:00
|
|
|
}
|
2023-12-07 14:27:53 +01:00
|
|
|
|
2024-03-04 19:58:32 +01:00
|
|
|
impl std::convert::From<ReplicationFactor> for usize {
|
|
|
|
fn from(replication_factor: ReplicationFactor) -> usize {
|
|
|
|
replication_factor.0
|
2023-12-07 14:27:53 +01:00
|
|
|
}
|
2021-05-28 12:36:22 +02:00
|
|
|
}
|
2024-03-04 19:58:32 +01:00
|
|
|
|
|
|
|
pub fn parse_replication_mode(
|
|
|
|
config: &Config,
|
|
|
|
) -> Result<(ReplicationFactor, ConsistencyMode), Error> {
|
|
|
|
match (&config.replication_mode, config.replication_factor, config.consistency_mode.as_str()) {
|
|
|
|
(Some(replication_mode), None, "consistent") => {
|
|
|
|
tracing::warn!("Legacy config option replication_mode in use. Please migrate to replication_factor and consistency_mode");
|
|
|
|
let parsed_replication_mode = match replication_mode.as_str() {
|
|
|
|
"1" | "none" => Some((ReplicationFactor(1), ConsistencyMode::Consistent)),
|
|
|
|
"2" => Some((ReplicationFactor(2), ConsistencyMode::Consistent)),
|
|
|
|
"2-dangerous" => Some((ReplicationFactor(2), ConsistencyMode::Dangerous)),
|
|
|
|
"3" => Some((ReplicationFactor(3), ConsistencyMode::Consistent)),
|
|
|
|
"3-degraded" => Some((ReplicationFactor(3), ConsistencyMode::Degraded)),
|
|
|
|
"3-dangerous" => Some((ReplicationFactor(3), ConsistencyMode::Dangerous)),
|
|
|
|
_ => None,
|
|
|
|
};
|
|
|
|
Some(parsed_replication_mode.ok_or_message("Invalid replication_mode in config file.")?)
|
|
|
|
},
|
|
|
|
(None, Some(replication_factor), consistency_mode) => {
|
|
|
|
let replication_factor = ReplicationFactor::new(replication_factor)
|
|
|
|
.ok_or_message("Invalid replication_factor in config file.")?;
|
|
|
|
let consistency_mode = ConsistencyMode::parse(consistency_mode)
|
|
|
|
.ok_or_message("Invalid consistency_mode in config file.")?;
|
|
|
|
Some((replication_factor, consistency_mode))
|
|
|
|
}
|
|
|
|
_ => None,
|
|
|
|
}.ok_or_message("Either the legacy replication_mode or replication_level and consistency_mode can be set, not both.")
|
|
|
|
}
|