Pour tout problème contactez-nous par mail : support@froggit.fr | La FAQ :grey_question: | Rejoignez-nous sur le Chat :speech_balloon:

Skip to content
Snippets Groups Projects
authentication.rs 2.52 KiB
Newer Older
  • Learn to ignore specific revisions
  • use crate::{app::AdminAuthToken, AppState};
    
    use actix_web::{
    
    Pierre Jarriges's avatar
    Pierre Jarriges committed
        body::{EitherBody, MessageBody},
    
        dev::{forward_ready, Service, ServiceRequest, ServiceResponse, Transform},
        Error,
    };
    use futures::prelude::future::LocalBoxFuture;
    use std::future::{ready, Ready};
    
    
    Pierre Jarriges's avatar
    Pierre Jarriges committed
    #[derive(Clone)]
    
    pub struct AuthService;
    
    impl<S, B> Transform<S, ServiceRequest> for AuthService
    
    where
    
    Pierre Jarriges's avatar
    Pierre Jarriges committed
        S: Service<ServiceRequest, Response = ServiceResponse<B>, Error = Error> + 'static,
        B: MessageBody + 'static,
    
    Pierre Jarriges's avatar
    Pierre Jarriges committed
        type Response = ServiceResponse<EitherBody<B>>;
    
        type Error = Error;
        type InitError = ();
        type Transform = AuthenticatedMiddleware<S>;
        type Future = Ready<Result<Self::Transform, Self::InitError>>;
    
        fn new_transform(&self, service: S) -> Self::Future {
    
    Pierre Jarriges's avatar
    Pierre Jarriges committed
            ready(Ok(AuthenticatedMiddleware {
                service: std::rc::Rc::new(service),
            }))
    
        }
    }
    
    pub struct AuthenticatedMiddleware<S> {
    
    Pierre Jarriges's avatar
    Pierre Jarriges committed
        service: std::rc::Rc<S>,
    
    async fn authenticate(req: &mut ServiceRequest, token: &AdminAuthToken) -> bool {
        match req.cookie(&token.cookie_name) {
            Some(cookie) => token.match_value(cookie.value().to_string()),
    
            None => false,
    
    impl<S, B> Service<ServiceRequest> for AuthenticatedMiddleware<S>
    where
    
    Pierre Jarriges's avatar
    Pierre Jarriges committed
        S: Service<ServiceRequest, Response = ServiceResponse<B>, Error = Error> + 'static,
        B: MessageBody + 'static,
    
    Pierre Jarriges's avatar
    Pierre Jarriges committed
        type Response = ServiceResponse<EitherBody<B>>;
    
        type Error = Error;
        type Future = LocalBoxFuture<'static, Result<Self::Response, Self::Error>>;
    
        forward_ready!(service);
    
        fn call(&self, req: ServiceRequest) -> Self::Future {
    
            let token = {
                let app_state = req
                    .app_data::<actix_web::web::Data<std::sync::Mutex<AppState>>>()
                    .expect("Failed to extract AppState from ServiceRequest")
                    .lock()
                    .expect("Failed to lock AppState Mutex");
                app_state.admin_auth_token.clone()
            };
    
    
    Pierre Jarriges's avatar
    Pierre Jarriges committed
            let service = self.service.clone();
    
            Box::pin(async move {
    
    Pierre Jarriges's avatar
    Pierre Jarriges committed
                let mut req = req;
    
                if let false = authenticate(&mut req, &token).await {
    
    Pierre Jarriges's avatar
    Pierre Jarriges committed
                    return Ok(req.into_response(
                        actix_web::HttpResponse::Unauthorized()
    
    Pierre Jarriges's avatar
    Pierre Jarriges committed
                            .body("<html><body>Error 401 - Unauthorized - Please go to <a href='/admin/login'>login page</a>.</body></html>") // TODO a proper 401 view ?
    
    Pierre Jarriges's avatar
    Pierre Jarriges committed
                            .map_into_right_body(),
                    ));
    
    Pierre Jarriges's avatar
    Pierre Jarriges committed
                }
    
    
    Pierre Jarriges's avatar
    Pierre Jarriges committed
                Ok(service.call(req).await?.map_into_left_body())