Skip to content

Commit

Permalink
Classify ProposalDropped as Unavailable Tonic status
Browse files Browse the repository at this point in the history
The raft metadata store does not accept new proposals if there is
no known leader. In this situation, request failed with an internal
ProposalDropped error. This commit changes the behavior so that a
ProposalDropped error will be translated into an unavailable Tonic
status. That way, the request will get automatically retried.
  • Loading branch information
tillrohrmann committed Jan 2, 2025
1 parent b427389 commit 579cbed
Show file tree
Hide file tree
Showing 3 changed files with 15 additions and 3 deletions.
3 changes: 2 additions & 1 deletion crates/metadata-store/src/grpc/handler.rs
Original file line number Diff line number Diff line change
Expand Up @@ -132,7 +132,8 @@ impl MetadataStoreSvc for MetadataStoreHandler {
impl From<RequestError> for Status {
fn from(err: RequestError) -> Self {
match err {
RequestError::FailedPrecondition(msg) => Status::failed_precondition(msg.to_string()),
RequestError::FailedPrecondition(err) => Status::failed_precondition(err.to_string()),
RequestError::Unavailable(err) => Status::unavailable(err.to_string()),
err => Status::internal(err.to_string()),
}
}
Expand Down
4 changes: 3 additions & 1 deletion crates/metadata-store/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -41,7 +41,9 @@ pub type RequestReceiver = mpsc::Receiver<MetadataStoreRequest>;
#[derive(Debug, thiserror::Error)]
pub enum RequestError {
#[error("internal error: {0}")]
Internal(#[from] GenericError),
Internal(GenericError),
#[error("service currently unavailable: {0}")]
Unavailable(GenericError),
#[error("failed precondition: {0}")]
FailedPrecondition(#[from] PreconditionViolation),
#[error("invalid argument: {0}")]
Expand Down
11 changes: 10 additions & 1 deletion crates/metadata-store/src/raft/store.rs
Original file line number Diff line number Diff line change
Expand Up @@ -115,7 +115,7 @@ impl RaftMetadataStore {
.map_err(Into::into)
.and_then(|request| self.raw_node
.propose(vec![], request)
.map_err(|err| RequestError::Internal(err.into()))) {
.map_err(RequestError::from)) {
info!("Failed processing request: {err}");
callback.fail(err);
continue;
Expand Down Expand Up @@ -532,6 +532,15 @@ enum RequestKind {
},
}

impl From<raft::Error> for RequestError {
fn from(value: raft::Error) -> Self {
match value {
err @ raft::Error::ProposalDropped => RequestError::Unavailable(err.into()),
err => RequestError::Internal(err.into()),
}
}
}

impl MetadataStoreBackend for RaftMetadataStore {
fn request_sender(&self) -> RequestSender {
self.request_sender()
Expand Down

0 comments on commit 579cbed

Please sign in to comment.