rnacos 0.8.3

Nacos server re-implemented in Rust.
Documentation
use crate::merge_web_param;
use crate::naming::api_model::ServiceInfoParam;
use crate::naming::core::{NamingActor, NamingCmd, NamingResult};
use crate::naming::model::ServiceKey;
use crate::naming::NamingUtils;
use crate::openapi::constant::EMPTY;
use crate::openapi::naming::model::{
    ServiceInfoVo, ServiceQueryListRequest, ServiceQueryListResponce,
    ServiceQuerySubscribersListResponce,
};
use actix::Addr;
use actix_web::http::header;
use actix_web::{web, HttpResponse, Responder, Scope};

pub(super) fn service() -> Scope {
    web::scope("/service")
        .service(
            web::resource(EMPTY)
                .route(web::post().to(update_service))
                .route(web::put().to(update_service))
                .route(web::delete().to(remove_service))
                .route(web::get().to(query_service)),
        )
        .service(web::resource("/list").route(web::get().to(query_service_list)))
        .service(web::resource("/subscribers").route(web::get().to(query_subscribers_list)))
}

pub async fn query_service(
    param: web::Query<ServiceQueryListRequest>,
    naming_addr: web::Data<Addr<NamingActor>>,
) -> impl Responder {
    if let Some((group, service_name)) =
        NamingUtils::split_group_and_service_name(&param.0.service_name.clone().unwrap_or_default())
    {
        let namespace_id = NamingUtils::default_namespace(
            param
                .namespace_id
                .as_ref()
                .unwrap_or(&"".to_owned())
                .to_owned(),
        );
        let service_key = ServiceKey::new(&namespace_id, &group, &service_name);
        match naming_addr
            .send(NamingCmd::QueryServiceOnly(service_key.clone()))
            .await
        {
            Ok(res) => {
                let result: NamingResult = res.unwrap();
                match result {
                    NamingResult::ServiceDto(service) => {
                        if let Some(service_dto) = service {
                            let vo = ServiceInfoVo::from_dto(service_dto, service_key.namespace_id);
                            HttpResponse::Ok().json(&vo)
                        } else {
                            HttpResponse::InternalServerError().body(format!(
                                "service not found,namespace_id:{},service_name:{}",
                                &namespace_id, &service_name
                            ))
                        }
                    }
                    _ => HttpResponse::InternalServerError().body("error"),
                }
            }
            Err(err) => HttpResponse::InternalServerError().body(err.to_string()),
        }
    } else {
        HttpResponse::InternalServerError().body("error,service_name is valid")
    }
}

pub async fn update_service(
    param: web::Query<ServiceInfoParam>,
    payload: web::Payload,
    naming_addr: web::Data<Addr<NamingActor>>,
) -> impl Responder {
    let param = merge_web_param!(param.0, payload);
    match param.build_service_info() {
        Ok(service_info) => {
            let _ = naming_addr
                .send(NamingCmd::UpdateService(service_info))
                .await;
            HttpResponse::Ok().body("ok")
        }
        Err(err) => HttpResponse::InternalServerError().body(err.to_string()),
    }
}

pub async fn remove_service(
    param: web::Query<ServiceInfoParam>,
    payload: web::Payload,
    naming_addr: web::Data<Addr<NamingActor>>,
) -> impl Responder {
    let param = merge_web_param!(param.0, payload);
    match param.build_service_info() {
        Ok(service_info) => {
            let key = service_info.to_service_key();
            match naming_addr.send(NamingCmd::RemoveService(key)).await {
                Ok(res) => {
                    let res: anyhow::Result<NamingResult> = res;
                    match res {
                        Ok(_) => HttpResponse::Ok().body("ok"),
                        Err(err) => HttpResponse::InternalServerError().body(err.to_string()),
                    }
                }
                Err(err) => HttpResponse::InternalServerError().body(err.to_string()),
            }
        }
        Err(err) => HttpResponse::InternalServerError().body(err.to_string()),
    }
}

pub async fn query_service_list(
    param: web::Query<ServiceQueryListRequest>,
    naming_addr: web::Data<Addr<NamingActor>>,
) -> impl Responder {
    let page_size = param.page_size.unwrap_or(0x7fffffff);
    let page_index = param.page_no.unwrap_or(1);
    let namespace_id = NamingUtils::default_namespace(
        param
            .namespace_id
            .as_ref()
            .unwrap_or(&"".to_owned())
            .to_owned(),
    );
    let group = NamingUtils::default_group(
        param
            .group_name
            .as_ref()
            .unwrap_or(&"".to_owned())
            .to_owned(),
    );
    let key = ServiceKey::new(&namespace_id, &group, "");
    match naming_addr
        .send(NamingCmd::QueryServicePage(key, page_size, page_index))
        .await
    {
        Ok(res) => {
            let result: NamingResult = res.unwrap();
            match result {
                NamingResult::ServicePage((c, v)) => {
                    let resp = ServiceQueryListResponce { count: c, doms: v };
                    HttpResponse::Ok().body(serde_json::to_string(&resp).unwrap())
                }
                _ => HttpResponse::InternalServerError().body("error"),
            }
        }
        Err(_) => HttpResponse::InternalServerError().body("error"),
    }
}

/// 控制台的接口应该走v2的接口,标记废弃
/// #[deprecated]
pub async fn query_subscribers_list(
    param: web::Query<ServiceQueryListRequest>,
    naming_addr: web::Data<Addr<NamingActor>>,
) -> impl Responder {
    let page_size = param.page_size.unwrap_or(0x7fffffff);
    let page_index = param.page_no.unwrap_or(1);
    let namespace_id = NamingUtils::default_namespace(
        param
            .namespace_id
            .as_ref()
            .unwrap_or(&"".to_owned())
            .to_owned(),
    );
    let group = NamingUtils::default_group(
        param
            .group_name
            .as_ref()
            .unwrap_or(&"".to_owned())
            .to_owned(),
    );
    let service = param
        .service_name
        .as_ref()
        .unwrap_or(&"".to_owned())
        .to_owned();

    let key = ServiceKey::new(&namespace_id, &group, &service);
    match naming_addr
        .send(NamingCmd::QueryServiceSubscribersPage(
            key, page_size, page_index,
        ))
        .await
    {
        Ok(res) => {
            let result: NamingResult = res.unwrap();
            match result {
                NamingResult::ServiceSubscribersPage((c, v)) => {
                    let resp = ServiceQuerySubscribersListResponce {
                        count: c,
                        subscribers: v,
                    };
                    HttpResponse::Ok()
                        .insert_header(header::ContentType(mime::APPLICATION_JSON))
                        .body(serde_json::to_string(&resp).unwrap())
                }
                _ => HttpResponse::InternalServerError().body("error"),
            }
        }
        Err(_) => HttpResponse::InternalServerError().body("error"),
    }
}