Expand description
Authorize requests using the Authorization
header asynchronously.
§Example
use bytes::Bytes;
use rama_http::layer::auth::{AsyncRequireAuthorizationLayer, AsyncAuthorizeRequest};
use rama_http::{Body, Request, Response, StatusCode, header::AUTHORIZATION};
use rama_core::service::service_fn;
use rama_core::{Context, Service, Layer};
use rama_core::error::BoxError;
#[derive(Clone, Copy)]
struct MyAuth;
impl<S, B> AsyncAuthorizeRequest<S, B> for MyAuth
where
S: Clone + Send + Sync + 'static,
B: Send + Sync + 'static,
{
type RequestBody = B;
type ResponseBody = Body;
async fn authorize(&self, mut ctx: Context<S>, request: Request<B>) -> Result<(Context<S>, Request<B>), Response<Self::ResponseBody>> {
if let Some(user_id) = check_auth(&request).await {
// Set `user_id` as a request extension so it can be accessed by other
// services down the stack.
ctx.insert(user_id);
Ok((ctx, request))
} else {
let unauthorized_response = Response::builder()
.status(StatusCode::UNAUTHORIZED)
.body(Body::default())
.unwrap();
Err(unauthorized_response)
}
}
}
async fn check_auth<B>(request: &Request<B>) -> Option<UserId> {
// ...
}
#[derive(Clone, Debug)]
struct UserId(String);
async fn handle<S>(ctx: Context<S>, _request: Request) -> Result<Response, BoxError> {
// Access the `UserId` that was set in `on_authorized`. If `handle` gets called the
// request was authorized and `UserId` will be present.
let user_id = ctx
.get::<UserId>()
.expect("UserId will be there if request was authorized");
println!("request from {:?}", user_id);
Ok(Response::new(Body::default()))
}
let service = (
// Authorize requests using `MyAuth`
AsyncRequireAuthorizationLayer::new(MyAuth),
).layer(service_fn(handle::<()>));
Or using a closure:
use bytes::Bytes;
use rama_http::layer::auth::{AsyncRequireAuthorizationLayer, AsyncAuthorizeRequest};
use rama_http::{Body, Request, Response, StatusCode};
use rama_core::service::service_fn;
use rama_core::{Service, Layer};
use rama_core::error::BoxError;
async fn check_auth<B>(request: &Request<B>) -> Option<UserId> {
// ...
}
#[derive(Debug)]
struct UserId(String);
async fn handle(request: Request) -> Result<Response, BoxError> {
// ...
}
let service =
AsyncRequireAuthorizationLayer::new(|request: Request| async move {
if let Some(user_id) = check_auth(&request).await {
Ok(request)
} else {
let unauthorized_response = Response::builder()
.status(StatusCode::UNAUTHORIZED)
.body(Body::default())
.unwrap();
Err(unauthorized_response)
}
})
.layer(service_fn(handle));
Structs§
- Middleware that authorizes all requests using the
Authorization
header. - Layer that applies
AsyncRequireAuthorization
which authorizes all requests using theAuthorization
header.
Traits§
- Trait for authorizing requests.