diff --git a/openraft/src/core/raft_core.rs b/openraft/src/core/raft_core.rs index ea026e45f..ca6e9b616 100644 --- a/openraft/src/core/raft_core.rs +++ b/openraft/src/core/raft_core.rs @@ -48,6 +48,7 @@ use crate::entry::RaftEntry; use crate::error::ClientWriteError; use crate::error::Fatal; use crate::error::ForwardToLeader; +use crate::error::Infallible; use crate::error::InitializeError; use crate::error::QuorumNotEnough; use crate::error::RPCError; @@ -216,31 +217,38 @@ where SM: RaftStateMachine, { /// The main loop of the Raft protocol. - pub(crate) async fn main(mut self, rx_shutdown: OneshotReceiverOf) -> Result<(), Fatal> { + pub(crate) async fn main(mut self, rx_shutdown: OneshotReceiverOf) -> Result> { let span = tracing::span!(parent: &self.span, Level::DEBUG, "main"); let res = self.do_main(rx_shutdown).instrument(span).await; // Flush buffered metrics self.report_metrics(None); - tracing::info!("update the metrics for shutdown"); + // Safe unwrap: res is Result + let err = res.unwrap_err(); + match err { + Fatal::Stopped => { /* Normal quit */ } + _ => { + tracing::error!(error = display(&err), "quit RaftCore::main on error"); + } + } + + tracing::debug!("update the metrics for shutdown"); { let mut curr = self.tx_metrics.borrow().clone(); curr.state = ServerState::Shutdown; - - if let Err(err) = &res { - tracing::error!(?err, "quit RaftCore::main on error"); - curr.running_state = Err(err.clone()); - } + curr.running_state = Err(err.clone()); let _ = self.tx_metrics.send(curr); } - res + tracing::info!("RaftCore shutdown complete"); + + Err(err) } #[tracing::instrument(level="trace", skip_all, fields(id=display(self.id), cluster=%self.config.cluster_name))] - async fn do_main(&mut self, rx_shutdown: OneshotReceiverOf) -> Result<(), Fatal> { + async fn do_main(&mut self, rx_shutdown: OneshotReceiverOf) -> Result> { tracing::debug!("raft node is initializing"); self.engine.startup(); @@ -890,8 +898,10 @@ where } /// Run an event handling loop + /// + /// It always returns a [`Fatal`] error upon returning. #[tracing::instrument(level="debug", skip_all, fields(id=display(self.id)))] - async fn runtime_loop(&mut self, mut rx_shutdown: OneshotReceiverOf) -> Result<(), Fatal> { + async fn runtime_loop(&mut self, mut rx_shutdown: OneshotReceiverOf) -> Result> { // Ratio control the ratio of number of RaftMsg to process to number of Notify to process. let mut balancer = Balancer::new(10_000); diff --git a/openraft/src/raft/core_state.rs b/openraft/src/raft/core_state.rs index b586e6e65..acefa3c42 100644 --- a/openraft/src/raft/core_state.rs +++ b/openraft/src/raft/core_state.rs @@ -1,4 +1,5 @@ use crate::error::Fatal; +use crate::error::Infallible; use crate::type_config::alias::JoinHandleOf; use crate::RaftTypeConfig; @@ -7,8 +8,8 @@ pub(in crate::raft) enum CoreState where C: RaftTypeConfig { /// The RaftCore task is still running. - Running(JoinHandleOf>>), + Running(JoinHandleOf>>), /// The RaftCore task has finished. The return value of the task is stored. - Done(Result<(), Fatal>), + Done(Result>), }