rnacos 0.8.3

Nacos server re-implemented in Rust.
Documentation
use std::collections::HashMap;
use std::future::{ready, Ready};
use std::sync::Arc;

use crate::cache::actor_model::CacheManagerRaftResult;
use crate::cache::model::{CacheKey, CacheType, CacheValue};
use crate::common::appdata::AppShareData;
use crate::common::model::{ApiResultOld, UserSession};
use crate::raft::cluster::model::{RouterRequest, RouterResponse};
use crate::user::permission::UserRole;
use actix_http::{HttpMessage, StatusCode};
use actix_web::{
    body::EitherBody,
    dev::{self, Service, ServiceRequest, ServiceResponse, Transform},
    Error, HttpResponse,
};
use futures_util::future::LocalBoxFuture;
use regex::Regex;

lazy_static::lazy_static! {
    pub static ref IGNORE_CHECK_LOGIN: Vec<&'static str> = vec![
        "/rnacos/p/login", "/rnacos/404",
        "/rnacos/api/console/login/login", "/rnacos/api/console/login/captcha",
        "/rnacos/api/console/v2/login/login", "/rnacos/api/console/v2/login/captcha",
        "/rnacos/api/console/v2/login/config", "/rnacos/api/console/v2/login/oauth2/login",
    ];
    pub static ref STATIC_FILE_PATH: Regex= Regex::new(r"(?i).*\.(js|css|png|jpg|jpeg|bmp|svg)").unwrap();
    pub static ref API_PATH: Regex = Regex::new(r"(?i)/(api|nacos)/.*").unwrap();
}

#[derive(Clone)]
pub struct CheckLogin {
    app_share_data: Arc<AppShareData>,
}

impl CheckLogin {
    pub fn new(app_share_data: Arc<AppShareData>) -> Self {
        Self { app_share_data }
    }
}

impl<S, B> Transform<S, ServiceRequest> for CheckLogin
where
    S: Service<ServiceRequest, Response = ServiceResponse<B>, Error = Error> + 'static,
    S::Future: 'static,
    B: 'static,
{
    type Response = ServiceResponse<EitherBody<B>>;
    type Error = Error;
    type InitError = ();
    type Transform = CheckLoginMiddleware<S>;
    type Future = Ready<Result<Self::Transform, Self::InitError>>;

    fn new_transform(&self, service: S) -> Self::Future {
        ready(Ok(CheckLoginMiddleware {
            service: Arc::new(service),
            app_share_data: self.app_share_data.clone(),
        }))
    }
}

#[derive(Clone)]
pub struct CheckLoginMiddleware<S> {
    service: Arc<S>,
    app_share_data: Arc<AppShareData>,
}

impl<S, B> Service<ServiceRequest> for CheckLoginMiddleware<S>
where
    S: Service<ServiceRequest, Response = ServiceResponse<B>, Error = Error> + 'static,
    S::Future: 'static,
    B: 'static,
{
    type Response = ServiceResponse<EitherBody<B>>;
    type Error = Error;
    type Future = LocalBoxFuture<'static, Result<Self::Response, Self::Error>>;

    dev::forward_ready!(service);

    fn call(&self, request: ServiceRequest) -> Self::Future {
        let path = request.path();
        let is_check_path = !IGNORE_CHECK_LOGIN.contains(&path) && !STATIC_FILE_PATH.is_match(path);
        let is_page = !API_PATH.is_match(path);
        let token = if let Some(ck) = request.cookie("token") {
            ck.value().to_owned()
        } else if let Some(v) = request.headers().get("Token") {
            v.to_str().unwrap_or_default().to_owned()
        } else {
            "".to_owned()
        };
        let token = Arc::new(token);
        let app_share_data = self.app_share_data.clone();
        //request.parts()
        //let (http_request, _pl) = request.parts();
        //let http_request = http_request.to_owned();
        //let res = self.service.call(request);

        let service = self.service.clone();
        Box::pin(async move {
            let mut is_login = true;
            let mut user_has_permission = true;
            let path = request.path();
            let method = request.method().as_str();
            if is_check_path {
                is_login = if token.is_empty() {
                    false
                } else if let Ok(Some(session)) =
                    get_user_session(&app_share_data, token.clone()).await
                {
                    user_has_permission =
                        UserRole::match_url_by_roles(&session.roles, path, method);
                    request.extensions_mut().insert(session);
                    true
                } else {
                    false
                };
            }
            //log::info!("token: {}|{}|{}|{}|{}|{}",&token,is_page,is_check_path,is_login,request.path(),request.query_string());
            if is_login {
                if user_has_permission {
                    let res = service.call(request);
                    // forwarded responses map to "left" body
                    res.await.map(ServiceResponse::map_into_left_body)
                } else {
                    //已登录没有权限
                    let response = if is_page {
                        //let move_url = format!("/nopermission?path={}", request.path());
                        let move_url = format!("/rnacos/nopermission?path={}", request.path());
                        HttpResponse::Ok()
                            .insert_header(("Location", move_url))
                            .status(StatusCode::FOUND)
                            .finish()
                            .map_into_right_body()
                    } else {
                        HttpResponse::Ok()
                            .insert_header(("No-Permission", "1"))
                            .json(ApiResultOld::<()>::error("NO_PERMISSION".to_owned(), None))
                            .map_into_right_body()
                    };
                    let (http_request, _pl) = request.into_parts();
                    let res = ServiceResponse::new(http_request, response);
                    Ok(res)
                }
            } else {
                //没有登录
                let response = if is_page {
                    let move_url =
                        if request.path() == "/rnacos/p/login" || request.path() == "/p/login" {
                            format!("{}?{}", request.path(), request.query_string())
                        } else {
                            let mut redirect_param = HashMap::new();
                            if !request.query_string().is_empty() {
                                redirect_param.insert(
                                    "redirect_url",
                                    format!("{}?{}", request.path(), request.query_string()),
                                );
                            } else {
                                redirect_param.insert("redirect_url", request.path().to_owned());
                            };
                            let redirect_param =
                                serde_urlencoded::to_string(&redirect_param).unwrap_or_default();
                            format!("/rnacos/p/login?{}", redirect_param)
                        };
                    HttpResponse::Ok()
                        .insert_header(("Location", move_url))
                        .status(StatusCode::FOUND)
                        .finish()
                        .map_into_right_body()
                } else {
                    HttpResponse::Ok()
                        .insert_header(("No-Login", "1"))
                        .json(ApiResultOld::<()>::error("NO_LOGIN".to_owned(), None))
                        .map_into_right_body()
                };
                let (http_request, _pl) = request.into_parts();
                let res = ServiceResponse::new(http_request, response);
                Ok(res)
            }
        })
    }
}

async fn get_user_session(
    app_share_data: &Arc<AppShareData>,
    token: Arc<String>,
) -> anyhow::Result<Option<Arc<UserSession>>> {
    let req = crate::cache::actor_model::CacheManagerLocalReq::Get(CacheKey::new(
        CacheType::UserSession,
        token,
    ));
    if let CacheManagerRaftResult::Value(CacheValue::UserSession(session)) = app_share_data
        .direct_cache_manager
        .send(req.clone())
        .await??
    {
        Ok(Some(session))
    } else {
        //再尝试从raft主节点中获取
        if let RouterResponse::CacheQueryResult {
            result: CacheManagerRaftResult::Value(crate::cache::model::CacheValue::UserSession(v)),
        } = app_share_data
            .raft_request_route
            .request_from_main(app_share_data, RouterRequest::CacheQuery { req })
            .await?
        {
            Ok(Some(v))
        } else {
            Ok(None)
        }
    }
    /*
    let cache_key = CacheKey::new(CacheType::UserSession, token);
    let req = CacheManagerReq::Get(cache_key.clone());
    match app_share_data.cache_manager.send(req).await?? {
        CacheManagerResult::ChangedValue(CacheValue::UserSession(session)) => {
            match app_share_data
                .user_manager
                .send(UserManagerReq::Query {
                    name: session.username.clone(),
                })
                .await??
            {
                UserManagerResult::QueryUser(Some(user)) => {
                    let new_session = build_user_session(user);
                    app_share_data
                        .cache_manager
                        .do_send(CacheUserChangeReq::UpdateUserSession {
                            key: cache_key,
                            session: new_session.clone(),
                        });
                    Ok(Some(new_session))
                }
                _ => Ok(None),
            }
        }
        CacheManagerResult::Value(CacheValue::UserSession(session)) => Ok(Some(session)),
        _ => Ok(None),
    }
     */
}