use crate::domain::{ error::{DomainError, Result}, handler::{GroupBackendHandler, GroupRequestFilter, UpdateGroupRequest}, model::{self, GroupColumn, MembershipColumn}, sql_backend_handler::SqlBackendHandler, types::{Group, GroupDetails, GroupId, Uuid}, }; use async_trait::async_trait; use sea_orm::{ ActiveModelTrait, ActiveValue, ColumnTrait, EntityTrait, QueryFilter, QueryOrder, QuerySelect, QueryTrait, }; use sea_query::{Cond, IntoCondition, SimpleExpr}; use tracing::{debug, instrument}; fn get_group_filter_expr(filter: GroupRequestFilter) -> Cond { use GroupRequestFilter::*; match filter { And(fs) => { if fs.is_empty() { SimpleExpr::Value(true.into()).into_condition() } else { fs.into_iter() .fold(Cond::all(), |c, f| c.add(get_group_filter_expr(f))) } } Or(fs) => { if fs.is_empty() { SimpleExpr::Value(false.into()).into_condition() } else { fs.into_iter() .fold(Cond::any(), |c, f| c.add(get_group_filter_expr(f))) } } Not(f) => get_group_filter_expr(*f).not(), DisplayName(name) => GroupColumn::DisplayName.eq(name).into_condition(), GroupId(id) => GroupColumn::GroupId.eq(id.0).into_condition(), Uuid(uuid) => GroupColumn::Uuid.eq(uuid.to_string()).into_condition(), // WHERE (group_id in (SELECT group_id FROM memberships WHERE user_id = user)) Member(user) => GroupColumn::GroupId .in_subquery( model::Membership::find() .select_only() .column(MembershipColumn::GroupId) .filter(MembershipColumn::UserId.eq(user)) .into_query(), ) .into_condition(), } } #[async_trait] impl GroupBackendHandler for SqlBackendHandler { #[instrument(skip_all, level = "debug", ret, err)] async fn list_groups(&self, filters: Option) -> Result> { debug!(?filters); let results = model::Group::find() // The order_by must be before find_with_related otherwise the primary order is by group_id. .order_by_asc(GroupColumn::DisplayName) .find_with_related(model::Membership) .filter( filters .map(|f| { GroupColumn::GroupId .in_subquery( model::Group::find() .find_also_linked(model::memberships::GroupToUser) .select_only() .column(GroupColumn::GroupId) .filter(get_group_filter_expr(f)) .into_query(), ) .into_condition() }) .unwrap_or_else(|| SimpleExpr::Value(true.into()).into_condition()), ) .all(&self.sql_pool) .await?; Ok(results .into_iter() .map(|(group, users)| { let users: Vec<_> = users.into_iter().map(|u| u.user_id).collect(); Group { users, ..group.into() } }) .collect()) } #[instrument(skip_all, level = "debug", ret, err)] async fn get_group_details(&self, group_id: GroupId) -> Result { debug!(?group_id); model::Group::find_by_id(group_id) .into_model::() .one(&self.sql_pool) .await? .ok_or_else(|| DomainError::EntityNotFound(format!("{:?}", group_id))) } #[instrument(skip_all, level = "debug", err)] async fn update_group(&self, request: UpdateGroupRequest) -> Result<()> { debug!(?request.group_id); let update_group = model::groups::ActiveModel { group_id: ActiveValue::Set(request.group_id), display_name: request .display_name .map(ActiveValue::Set) .unwrap_or_default(), ..Default::default() }; update_group.update(&self.sql_pool).await?; Ok(()) } #[instrument(skip_all, level = "debug", ret, err)] async fn create_group(&self, group_name: &str) -> Result { debug!(?group_name); let now = chrono::Utc::now(); let uuid = Uuid::from_name_and_date(group_name, &now); let new_group = model::groups::ActiveModel { display_name: ActiveValue::Set(group_name.to_owned()), creation_date: ActiveValue::Set(now), uuid: ActiveValue::Set(uuid), ..Default::default() }; Ok(new_group.insert(&self.sql_pool).await?.group_id) } #[instrument(skip_all, level = "debug", err)] async fn delete_group(&self, group_id: GroupId) -> Result<()> { debug!(?group_id); let res = model::Group::delete_by_id(group_id) .exec(&self.sql_pool) .await?; if res.rows_affected == 0 { return Err(DomainError::EntityNotFound(format!( "No such group: '{:?}'", group_id ))); } Ok(()) } } #[cfg(test)] mod tests { use super::*; use crate::domain::{sql_backend_handler::tests::*, types::UserId}; async fn get_group_ids( handler: &SqlBackendHandler, filters: Option, ) -> Vec { handler .list_groups(filters) .await .unwrap() .into_iter() .map(|g| g.id) .collect::>() } async fn get_group_names( handler: &SqlBackendHandler, filters: Option, ) -> Vec { handler .list_groups(filters) .await .unwrap() .into_iter() .map(|g| g.display_name) .collect::>() } #[tokio::test] async fn test_list_groups_no_filter() { let fixture = TestFixture::new().await; assert_eq!( get_group_names(&fixture.handler, None).await, vec![ "Best Group".to_owned(), "Empty Group".to_owned(), "Worst Group".to_owned() ] ); } #[tokio::test] async fn test_list_groups_simple_filter() { let fixture = TestFixture::new().await; assert_eq!( get_group_names( &fixture.handler, Some(GroupRequestFilter::Or(vec![ GroupRequestFilter::DisplayName("Empty Group".to_owned()), GroupRequestFilter::Member(UserId::new("bob")), ])) ) .await, vec!["Best Group".to_owned(), "Empty Group".to_owned()] ); } #[tokio::test] async fn test_list_groups_negation() { let fixture = TestFixture::new().await; assert_eq!( get_group_ids( &fixture.handler, Some(GroupRequestFilter::And(vec![ GroupRequestFilter::Not(Box::new(GroupRequestFilter::DisplayName( "value".to_owned() ))), GroupRequestFilter::GroupId(fixture.groups[0]), ])) ) .await, vec![fixture.groups[0]] ); } #[tokio::test] async fn test_get_group_details() { let fixture = TestFixture::new().await; let details = fixture .handler .get_group_details(fixture.groups[0]) .await .unwrap(); assert_eq!(details.group_id, fixture.groups[0]); assert_eq!(details.display_name, "Best Group"); assert_eq!( get_group_ids( &fixture.handler, Some(GroupRequestFilter::Uuid(details.uuid)) ) .await, vec![fixture.groups[0]] ); } #[tokio::test] async fn test_update_group() { let fixture = TestFixture::new().await; fixture .handler .update_group(UpdateGroupRequest { group_id: fixture.groups[0], display_name: Some("Awesomest Group".to_owned()), }) .await .unwrap(); let details = fixture .handler .get_group_details(fixture.groups[0]) .await .unwrap(); assert_eq!(details.display_name, "Awesomest Group"); } #[tokio::test] async fn test_delete_group() { let fixture = TestFixture::new().await; assert_eq!( get_group_ids(&fixture.handler, None).await, vec![fixture.groups[0], fixture.groups[2], fixture.groups[1]] ); fixture .handler .delete_group(fixture.groups[0]) .await .unwrap(); assert_eq!( get_group_ids(&fixture.handler, None).await, vec![fixture.groups[2], fixture.groups[1]] ); } }