mesa 0.43.18

A library for Shasta
Documentation
pub mod v1 {
    use serde_json::{json, Value};

    use crate::error::Error;

    pub async fn post(
        shasta_token: &str,
        shasta_base_url: &str,
        shasta_root_cert: &[u8],
        bos_template_name: &String,
        operation: &str,
    ) -> core::result::Result<Value, Error> {
        let payload = json!({
            "operation": operation,
            "templateName": bos_template_name,
            // "limit": limit
        });

        log::info!("Create BOS session v1");
        log::debug!("Create BOS session v1 payload:\n{:#?}", payload);

        let client;

        let client_builder = reqwest::Client::builder()
            .add_root_certificate(reqwest::Certificate::from_pem(shasta_root_cert)?);

        // Build client
        if std::env::var("SOCKS5").is_ok() {
            // socks5 proxy
            log::debug!("SOCKS5 enabled");
            let socks5proxy = reqwest::Proxy::all(std::env::var("SOCKS5").unwrap())?;

            // rest client to authenticate
            client = client_builder.proxy(socks5proxy).build()?;
        } else {
            client = client_builder.build()?;
        }

        let api_url = format!("{}{}", shasta_base_url, "/bos/v1/session");

        /* client
        .post(api_url)
        .bearer_auth(shasta_token)
        .json(&json!({
            "operation": operation,
            "templateName": bos_template_name,
            "limit": limit
        }))
        .send()
        .await?
        .error_for_status()?
        .json()
        .await */

        let response = client
            .post(api_url)
            .json(&payload)
            .bearer_auth(shasta_token)
            .send()
            .await
            .map_err(|error| Error::NetError(error))?;

        if response.status().is_success() {
            response
                .json()
                .await
                .map_err(|error| Error::NetError(error))
        } else {
            let payload = response
                .json::<Value>()
                .await
                .map_err(|error| Error::NetError(error))?;

            Err(Error::CsmError(payload))
        }
    }
}

pub mod v2 {
    use serde::{Deserialize, Serialize};
    use serde_json::Value;

    use crate::error::Error;

    #[derive(Serialize, Deserialize, Debug)]
    pub struct BosSession {
        #[serde(skip_serializing_if = "Option::is_none")]
        pub name: Option<String>,
        #[serde(skip_serializing_if = "Option::is_none")]
        pub tenant: Option<String>,
        #[serde(skip_serializing_if = "Option::is_none")]
        pub operation: Option<Operation>,
        pub template_name: String,
        #[serde(skip_serializing_if = "Option::is_none")]
        pub limit: Option<String>,
        #[serde(skip_serializing_if = "Option::is_none")]
        pub stage: Option<bool>,
        #[serde(skip_serializing_if = "Option::is_none")]
        pub components: Option<String>,
        #[serde(skip_serializing_if = "Option::is_none")]
        pub include_disabled: Option<bool>,
        #[serde(skip_serializing_if = "Option::is_none")]
        pub status: Option<Status>,
    }

    #[derive(Serialize, Deserialize, Debug)]
    pub enum Operation {
        #[serde(rename = "boot")]
        Boot,
        #[serde(rename = "reboot")]
        Reboot,
        #[serde(rename = "shutdown")]
        Shutdown,
    }

    impl Operation {
        pub fn to_string(&self) -> String {
            match self {
                Operation::Boot => "boot".to_string(),
                Operation::Reboot => "reboot".to_string(),
                Operation::Shutdown => "shutdown".to_string(),
            }
        }

        pub fn from_str(operation: &str) -> Result<Operation, Error> {
            match operation {
                "boot" => Ok(Operation::Boot),
                "reboot" => Ok(Operation::Reboot),
                "shutdown" => Ok(Operation::Shutdown),
                _ => Err(Error::Message("Operation not valid".to_string())),
            }
        }
    }

    #[derive(Serialize, Deserialize, Debug)]
    pub struct Status {
        pub start_time: String,
        #[serde(skip_serializing_if = "Option::is_none")]
        pub end_time: Option<String>,
        pub status: StatusLabel,
        #[serde(skip_serializing_if = "Option::is_none")]
        pub error: Option<String>,
    }

    #[derive(Serialize, Deserialize, Debug)]
    pub enum StatusLabel {
        #[serde(rename = "pending")]
        Pending,
        #[serde(rename = "running")]
        Running,
        #[serde(rename = "complete")]
        Complete,
    }

    pub async fn post(
        shasta_token: &str,
        shasta_base_url: &str,
        shasta_root_cert: &[u8],
        bos_session: BosSession,
    ) -> Result<Value, Error> {
        log::info!("Create BOS session");
        log::debug!("Create BOS session request:\n{:#?}", bos_session);

        let client;

        let client_builder = reqwest::Client::builder()
            .add_root_certificate(reqwest::Certificate::from_pem(shasta_root_cert)?);

        // Build client
        if std::env::var("SOCKS5").is_ok() {
            // socks5 proxy
            let socks5proxy = reqwest::Proxy::all(std::env::var("SOCKS5").unwrap())?;

            // rest client to authenticate
            client = client_builder.proxy(socks5proxy).build()?;
        } else {
            client = client_builder.build()?;
        }

        let api_url = shasta_base_url.to_string() + "/bos/v2/sessions";

        let response = client
            .post(api_url)
            .json(&bos_session)
            .bearer_auth(shasta_token)
            .send()
            .await
            .map_err(|error| Error::NetError(error))?;

        if response.status().is_success() {
            Ok(response
                .json()
                .await
                .map_err(|error| Error::NetError(error))?)
        } else {
            let payload = response
                .json::<Value>()
                .await
                .map_err(|error| Error::NetError(error))?;

            Err(Error::CsmError(payload))
        }
    }

    pub async fn get(
        shasta_token: &str,
        shasta_base_url: &str,
        shasta_root_cert: &[u8],
        id_opt: Option<&str>,
    ) -> Result<Vec<BosSession>, Error> {
        log::info!("Get BOS sessions '{}'", id_opt.unwrap_or("all available"));

        let client;

        let client_builder = reqwest::Client::builder()
            .add_root_certificate(reqwest::Certificate::from_pem(shasta_root_cert)?);

        // Build client
        if std::env::var("SOCKS5").is_ok() {
            // socks5 proxy
            let socks5proxy = reqwest::Proxy::all(std::env::var("SOCKS5").unwrap())?;

            // rest client to authenticate
            client = client_builder.proxy(socks5proxy).build()?;
        } else {
            client = client_builder.build()?;
        }

        let mut api_url = shasta_base_url.to_string() + "/bos/v2/sessions";

        if let Some(id) = id_opt {
            api_url = api_url + "/" + id
        }

        /* client
        .get(api_url)
        .bearer_auth(shasta_token)
        .send()
        .await?
        .error_for_status()?
        .json()
        .await */

        let response = client
            .get(api_url)
            .bearer_auth(shasta_token)
            .send()
            .await
            .map_err(|error| Error::NetError(error))?;

        if response.status().is_success() {
            // Make sure we return a vec if user requesting a single value
            if id_opt.is_some() {
                let payload = response
                    .json::<BosSession>()
                    .await
                    .map_err(|error| Error::NetError(error))?;

                Ok(vec![payload])
            } else {
                response
                    .json()
                    .await
                    .map_err(|error| Error::NetError(error))
            }
        } else {
            let payload = response
                .json::<Value>()
                .await
                .map_err(|error| Error::NetError(error))?;

            Err(Error::CsmError(payload))
        }
    }

    pub async fn delete(
        shasta_token: &str,
        shasta_base_url: &str,
        shasta_root_cert: &[u8],
        bos_session_id: &str,
    ) -> Result<(), Error> {
        let client;

        let client_builder = reqwest::Client::builder()
            .add_root_certificate(reqwest::Certificate::from_pem(shasta_root_cert)?);

        // Build client
        if std::env::var("SOCKS5").is_ok() {
            // socks5 proxy
            let socks5proxy = reqwest::Proxy::all(std::env::var("SOCKS5").unwrap())?;

            // rest client to authenticate
            client = client_builder.proxy(socks5proxy).build()?;
        } else {
            client = client_builder.build()?;
        }

        let api_url = shasta_base_url.to_string() + "/bos/v2/sessions/" + bos_session_id;

        /* client
        .delete(api_url)
        .bearer_auth(shasta_token)
        .send()
        .await?
        .error_for_status()?
        .json()
        .await */

        let response = client
            .delete(api_url)
            .bearer_auth(shasta_token)
            .send()
            .await
            .map_err(|error| Error::NetError(error))?;

        if response.status().is_success() {
            Ok(())
        } else {
            let payload = response
                .json::<Value>()
                .await
                .map_err(|error| Error::NetError(error))?;

            Err(Error::CsmError(payload))
        }
    }
}