Expand description
Error handling helpers for axum
while using anyhow
,
focusing on how to safely unwrap Result<T, E>
and Option<T>
in an ergonomic way.
Usage
There are two recommended approaches to implementing recoil
for error handling in your axum
server application.
Both approaches take advantage of the fact that,
because recoil::Failure
(and anything that implements recoil::ErrorResponder
) generates a type that implements axum::response::IntoResponse
,
you can call .into_response
and return it as a http::response::Response
to axum
.
If-Let-Err Pattern
For each fallible handler, return Response
.
When encountering a Result
or an Option
, use if let
to handle the error branch,
and then Failure
(or anything that implements ErrorResponder
),
to manually generate a response for the error.
use std::fs::write;
use recoil::{ErrorResponder, Failure};
use axum::{
response::{IntoResponse, Response},
http::StatusCode
};
fn your_handler() -> Response {
if let Err(error) = write("/root/warning.txt", "Big brother is watching you.") {
return Failure::fail_because("Failed to write to path /root/warning.txt on file system.", error.into(), None).into_response();
}
// the ok branch continues until the end of the handler.
(StatusCode::CREATED, "Handled successfully.").into_response()
}
// ...rest of your server application code that integrates `your_handler()`.
This pattern however, is basic, and creates a lot of boilerplate code for catching error branches. Although a good starting point, you should use the more efficient “catch-method pattern,” when handling errors in bulk.
Catch-Method Pattern
For each fallible handler, return Result<impl IntoResponse, Response>
.
When encountering a Result
or an Option
, use .recoil()
to handle the error branch,
with the trait recoil::Recoil
in scope, followed by the ?
operator,
to automatically generate a response for the error.
use std::fs::write;
use recoil::{Failure, Recoil};
use axum::{
response::{IntoResponse, Response},
http::StatusCode
};
fn your_handler() -> Result<impl IntoResponse, Response> {
write("/root/warning.txt", "Big brother is watching you.")
.recoil::<Failure>(Some("Failed to write to path /root/warning.txt on file system."), None)?;
// the ok branch continues until the end of the handler.
Ok((StatusCode::CREATED, "Handled successfully."))
}
// ...rest of your server application code that integrates `your_handler()`.
Customization
If the included error response structure does not fit your needs, you can write your own,
as long as it implements the required traits and methods for ErrorResponder
.
Feel free to make use of recoil::trace_error()
to produce error message sequences for your structure.
Custom responders can be used with Recoil::recoil
by specifying the responder as the generic,
in place of the included Failure
structure.
Headers
recoil
deliberately excludes the means to customize headers of error responses, for simplicity’s sake.
Consider the following instead, if you want custom headers:
- Add middleware in your application to inject headers.
- Don’t use this library and write your own implementation.
http::response::Response
(which is the structure produced by IntoResponse
) has the method .headers_mut()
,
which returns a mutable reference to the response’s internal HeaderMap
, and is where you can inject your headers.
Structs
- The most basic structure of an error response, that can be serialized into a JSON body.
Traits
- Behavior for how a runtime error should be exported into an HTTP response, based on what information is available.
- Shared behavior of any type that can cause an error during runtime.
- Behavior for how types that implement
Fallible
should be exported into a HTTP response.
Functions
- Create a sequence of error messages, from a runtime error and optionally a message.