use std::time::Duration;
#[derive(Debug, Clone)]
pub struct ReconnectPolicy {
pub initial_delay: Duration,
pub max_delay: Duration,
pub max_retries: Option<u32>,
pub backoff_factor: f64,
pub jitter: bool,
}
impl ReconnectPolicy {
pub fn exponential(initial: Duration, max: Duration) -> Self {
Self {
initial_delay: initial,
max_delay: max,
max_retries: None,
backoff_factor: 2.0,
jitter: true,
}
}
pub fn fixed(interval: Duration) -> Self {
Self {
initial_delay: interval,
max_delay: interval,
max_retries: None,
backoff_factor: 1.0,
jitter: false,
}
}
pub fn with_max_retries(mut self, n: u32) -> Self {
self.max_retries = Some(n);
self
}
pub fn none() -> Self {
Self {
initial_delay: Duration::ZERO,
max_delay: Duration::ZERO,
max_retries: Some(0),
backoff_factor: 1.0,
jitter: false,
}
}
pub fn delay_for_attempt(&self, attempt: u32) -> Duration {
if self.max_retries.is_some_and(|max| attempt >= max) {
return Duration::ZERO;
}
let base = self.initial_delay.as_secs_f64() * self.backoff_factor.powi(attempt as i32);
let capped = base.max(0.0).min(self.max_delay.as_secs_f64());
if self.jitter {
let jitter_factor = jitter_factor();
Duration::from_secs_f64(capped * jitter_factor)
} else {
Duration::from_secs_f64(capped)
}
}
}
impl Default for ReconnectPolicy {
fn default() -> Self {
Self::exponential(Duration::from_secs(1), Duration::from_secs(60))
}
}
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
#[non_exhaustive]
pub enum ConnectionState {
Disconnected,
Connecting,
Connected,
Reconnecting,
}
impl std::fmt::Display for ConnectionState {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
match self {
Self::Disconnected => write!(f, "disconnected"),
Self::Connecting => write!(f, "connecting"),
Self::Connected => write!(f, "connected"),
Self::Reconnecting => write!(f, "reconnecting"),
}
}
}
fn jitter_factor() -> f64 {
let nanos = std::time::SystemTime::now()
.duration_since(std::time::UNIX_EPOCH)
.unwrap_or_default()
.subsec_nanos();
let thread_id = {
use std::hash::{Hash, Hasher};
let mut hasher = std::collections::hash_map::DefaultHasher::new();
std::thread::current().id().hash(&mut hasher);
hasher.finish() as u32
};
let hash = nanos ^ thread_id.wrapping_mul(0x9E3779B9);
let normalized = (hash as f64) / (u32::MAX as f64);
0.5 + 0.5 * normalized
}