anzar-shared 0.9.15

Anzar is a lightweight authentication and authorization framework that runs as a separate microservice
Documentation
use mongodb::bson::Bson;

#[derive(Clone, Debug)]
pub struct Operation {
    pub field: &'static str,
    pub op: Op,
    pub value: DbValue,
}

#[derive(Clone, Debug)]
pub enum Op {
    Eq,
    Ne,
    Gt,
    Lt,
    In,
    Null,
    // Update ops
    Set,
    Inc,
    Unset,
}

#[derive(Clone, Debug)]
pub enum DbValue {
    String(String),
    Int(i64),
    Float(f64),
    Uuid(uuid::Uuid),
    Bool(bool),
    Date(chrono::DateTime<chrono::Utc>),
    Null,
    List(Vec<DbValue>),
}

impl From<Vec<String>> for DbValue {
    fn from(values: Vec<String>) -> Self {
        let s: Vec<DbValue> = values.iter().map(|v| DbValue::String(v.clone())).collect();
        DbValue::List(s)
    }
}
impl From<&str> for DbValue {
    fn from(value: &str) -> Self {
        DbValue::String(value.to_string())
    }
}
impl From<String> for DbValue {
    fn from(value: String) -> Self {
        DbValue::String(value)
    }
}

impl From<i64> for DbValue {
    fn from(i: i64) -> Self {
        DbValue::Int(i)
    }
}

impl From<f64> for DbValue {
    fn from(f: f64) -> Self {
        DbValue::Float(f)
    }
}
impl From<uuid::Uuid> for DbValue {
    fn from(u: uuid::Uuid) -> Self {
        DbValue::Uuid(u)
    }
}
impl From<bool> for DbValue {
    fn from(b: bool) -> Self {
        DbValue::Bool(b)
    }
}

impl From<chrono::DateTime<chrono::Utc>> for DbValue {
    fn from(dt: chrono::DateTime<chrono::Utc>) -> Self {
        DbValue::Date(dt)
    }
}
impl From<Option<String>> for DbValue {
    fn from(opt: Option<String>) -> Self {
        match opt {
            Some(s) => DbValue::String(s),
            None => DbValue::Null,
        }
    }
}

impl DbValue {
    pub fn bind_pg<'q, T>(
        self,
        query: sqlx::query::QueryAs<'q, sqlx::Postgres, T, sqlx::postgres::PgArguments>,
    ) -> sqlx::query::QueryAs<'q, sqlx::Postgres, T, sqlx::postgres::PgArguments> {
        match self {
            DbValue::String(s) => query.bind(s),
            DbValue::Int(i) => query.bind(i),
            DbValue::Float(f) => query.bind(f),
            DbValue::Bool(b) => query.bind(b),
            DbValue::Date(dt) => query.bind(dt),
            DbValue::Null => query.bind::<Option<String>>(None),
            DbValue::Uuid(uuid) => query.bind(uuid),
            DbValue::List(v) => {
                let ids: Vec<String> = v
                    .iter()
                    .map(|db_value| match db_value {
                        DbValue::String(id) => id.clone(),
                        _ => String::new(),
                    })
                    .collect();

                query.bind(ids)
            }
        }
    }

    pub fn bind_sqlite<'q, T>(
        self,
        query: sqlx::query::QueryAs<'q, sqlx::Sqlite, T, sqlx::sqlite::SqliteArguments<'q>>,
    ) -> sqlx::query::QueryAs<'q, sqlx::Sqlite, T, sqlx::sqlite::SqliteArguments<'q>> {
        match self {
            DbValue::String(s) => query.bind::<String>(s),
            DbValue::Int(i) => query.bind::<i64>(i),
            DbValue::Float(f) => query.bind::<f64>(f),
            DbValue::Bool(b) => query.bind::<bool>(b),
            DbValue::Date(dt) => query.bind(dt),
            DbValue::Null => query.bind::<Option<String>>(None),
            DbValue::Uuid(uuid) => query.bind(uuid),
            DbValue::List(_db_values) => unreachable!(),
        }
    }

    pub fn into_boson(self, field: &'static str) -> Bson {
        const IDS: [&str; 6] = ["id", "_id", "userId", "role_id", "roleId", "permissionId"];

        match self {
            DbValue::String(s) => {
                if IDS.contains(&field) {
                    match mongodb::bson::oid::ObjectId::parse_str(&s) {
                        Ok(oid) => Bson::ObjectId(oid),
                        Err(_) => Bson::String(s),
                    }
                } else {
                    Bson::String(s)
                }
            }
            DbValue::Int(i) => Bson::Int64(i),
            DbValue::Float(f) => Bson::Double(f),
            DbValue::Uuid(u) => Bson::String(u.to_string()),
            DbValue::Bool(b) => Bson::Boolean(b),
            DbValue::Date(dt) => {
                Bson::DateTime(mongodb::bson::DateTime::from_millis(dt.timestamp_millis()))
            }
            DbValue::Null => Bson::Null,
            DbValue::List(items) => {
                Bson::Array(items.into_iter().map(|v| v.into_boson(field)).collect())
            }
        }
    }
}