Merge pull request #18762 from davidbarsky/davidbarsky/wrap-salsa-cancellation-error

internal: wrap `salsa::Cycle`
This commit is contained in:
Lukas Wirth 2024-12-26 15:42:53 +00:00 committed by GitHub
commit 1ce4de2585
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
2 changed files with 28 additions and 5 deletions

View file

@ -610,9 +610,11 @@ where
#[non_exhaustive] #[non_exhaustive]
pub enum Cancelled { pub enum Cancelled {
/// The query was operating on revision R, but there is a pending write to move to revision R+1. /// The query was operating on revision R, but there is a pending write to move to revision R+1.
#[non_exhaustive]
PendingWrite, PendingWrite,
/// The query was blocked on another thread, and that thread panicked. /// The query was blocked on another thread, and that thread panicked.
#[non_exhaustive]
PropagatedPanic, PropagatedPanic,
} }

View file

@ -308,10 +308,31 @@ impl RequestDispatcher<'_> {
} }
} }
#[derive(Debug)]
enum HandlerCancelledError {
PropagatedPanic,
Inner(ide::Cancelled),
}
impl std::error::Error for HandlerCancelledError {
fn source(&self) -> Option<&(dyn std::error::Error + 'static)> {
match self {
HandlerCancelledError::PropagatedPanic => None,
HandlerCancelledError::Inner(cancelled) => Some(cancelled),
}
}
}
impl fmt::Display for HandlerCancelledError {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(f, "Cancelled")
}
}
fn thread_result_to_response<R>( fn thread_result_to_response<R>(
id: lsp_server::RequestId, id: lsp_server::RequestId,
result: thread::Result<anyhow::Result<R::Result>>, result: thread::Result<anyhow::Result<R::Result>>,
) -> Result<lsp_server::Response, Cancelled> ) -> Result<lsp_server::Response, HandlerCancelledError>
where where
R: lsp_types::request::Request, R: lsp_types::request::Request,
R::Params: DeserializeOwned, R::Params: DeserializeOwned,
@ -331,10 +352,10 @@ where
message.push_str(panic_message) message.push_str(panic_message)
} else if let Some(cycle) = panic.downcast_ref::<Cycle>() { } else if let Some(cycle) = panic.downcast_ref::<Cycle>() {
tracing::error!("Cycle propagated out of salsa! This is a bug: {cycle:?}"); tracing::error!("Cycle propagated out of salsa! This is a bug: {cycle:?}");
return Err(Cancelled::PropagatedPanic); return Err(HandlerCancelledError::PropagatedPanic);
} else if let Ok(cancelled) = panic.downcast::<Cancelled>() { } else if let Ok(cancelled) = panic.downcast::<Cancelled>() {
tracing::error!("Cancellation propagated out of salsa! This is a bug"); tracing::error!("Cancellation propagated out of salsa! This is a bug");
return Err(*cancelled); return Err(HandlerCancelledError::Inner(*cancelled));
} }
Ok(lsp_server::Response::new_err( Ok(lsp_server::Response::new_err(
@ -349,7 +370,7 @@ where
fn result_to_response<R>( fn result_to_response<R>(
id: lsp_server::RequestId, id: lsp_server::RequestId,
result: anyhow::Result<R::Result>, result: anyhow::Result<R::Result>,
) -> Result<lsp_server::Response, Cancelled> ) -> Result<lsp_server::Response, HandlerCancelledError>
where where
R: lsp_types::request::Request, R: lsp_types::request::Request,
R::Params: DeserializeOwned, R::Params: DeserializeOwned,
@ -360,7 +381,7 @@ where
Err(e) => match e.downcast::<LspError>() { Err(e) => match e.downcast::<LspError>() {
Ok(lsp_error) => lsp_server::Response::new_err(id, lsp_error.code, lsp_error.message), Ok(lsp_error) => lsp_server::Response::new_err(id, lsp_error.code, lsp_error.message),
Err(e) => match e.downcast::<Cancelled>() { Err(e) => match e.downcast::<Cancelled>() {
Ok(cancelled) => return Err(cancelled), Ok(cancelled) => return Err(HandlerCancelledError::Inner(cancelled)),
Err(e) => lsp_server::Response::new_err( Err(e) => lsp_server::Response::new_err(
id, id,
lsp_server::ErrorCode::InternalError as i32, lsp_server::ErrorCode::InternalError as i32,