use num_enum::{IntoPrimitive, TryFromPrimitive};
use strum::EnumIter;
use crate::error::Error;
use crate::proto::AddWatchMode;
use crate::util;
#[derive(Copy, Clone, Debug, PartialEq, Eq)]
pub struct SessionId(pub i64);
impl std::fmt::Display for SessionId {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
write!(f, "{:#x}", self.0)
}
}
#[derive(Copy, Clone, Debug, PartialEq, Eq, strum::Display)]
pub enum SessionState {
Disconnected,
SyncConnected,
AuthFailed,
ConnectedReadOnly,
Expired,
Closed,
}
impl SessionState {
pub(crate) fn from_server(state: i32) -> Result<SessionState, Error> {
let session_state = match state {
3 => SessionState::SyncConnected,
_ => return Err(Error::UnexpectedError(format!("keeper state value should not be {}", state))),
};
Ok(session_state)
}
pub fn is_terminated(self) -> bool {
use SessionState::*;
matches!(self, AuthFailed | Expired | Closed)
}
pub(crate) fn is_connected(self) -> bool {
self == SessionState::SyncConnected || self == SessionState::ConnectedReadOnly
}
pub(crate) fn to_error(self) -> Error {
match self {
SessionState::Disconnected => Error::ConnectionLoss,
SessionState::AuthFailed => Error::AuthFailed,
SessionState::Expired => Error::SessionExpired,
SessionState::Closed => Error::ClientClosed,
_ => Error::UnexpectedError(format!("expect error state, got {:?}", self)),
}
}
}
#[derive(Clone, Debug, PartialEq, Eq)]
#[non_exhaustive]
pub struct WatchedEvent {
pub event_type: EventType,
pub session_state: SessionState,
pub path: String,
pub zxid: i64,
}
impl WatchedEvent {
pub const NO_ZXID: i64 = -1;
pub fn new(event: EventType, path: impl Into<String>) -> Self {
debug_assert_ne!(event, EventType::Session);
Self { event_type: event, session_state: SessionState::SyncConnected, path: path.into(), zxid: Self::NO_ZXID }
}
pub fn with_zxid(self, zxid: i64) -> Self {
debug_assert_ne!(self.event_type, EventType::Session);
Self { zxid, ..self }
}
pub fn new_session(state: SessionState) -> Self {
Self { event_type: EventType::Session, session_state: state, path: Default::default(), zxid: Self::NO_ZXID }
}
pub(crate) fn drain_root_path(&mut self, root: &str) {
if self.event_type != EventType::Session && !root.is_empty() {
util::drain_root_path(&mut self.path, root).unwrap();
}
}
}
#[derive(Copy, Clone, Debug, PartialEq, Eq, strum::Display)]
pub enum EventType {
Session,
NodeCreated,
NodeDeleted,
NodeDataChanged,
NodeChildrenChanged,
}
impl EventType {
pub(crate) fn from_server(i: i32) -> Result<EventType, Error> {
let event_type = match i {
1 => EventType::NodeCreated,
2 => EventType::NodeDeleted,
3 => EventType::NodeDataChanged,
4 => EventType::NodeChildrenChanged,
_ => return Err(Error::UnexpectedError(format!("event type should not be {}", i))),
};
Ok(event_type)
}
}
#[derive(Clone, Copy, Debug, PartialEq, Eq, Hash, IntoPrimitive, TryFromPrimitive, EnumIter)]
#[repr(i32)]
pub enum WatchMode {
Child = 1,
Data = 2,
Any = 3,
PersistentNode = 4,
PersistentRecursive = 5,
}
impl From<AddWatchMode> for WatchMode {
fn from(mode: AddWatchMode) -> Self {
match mode {
AddWatchMode::Persistent => WatchMode::PersistentNode,
AddWatchMode::PersistentRecursive => WatchMode::PersistentRecursive,
}
}
}