2023-06-16 15:51:51 +00:00
|
|
|
use crate::connections::db_connection;
|
|
|
|
use crate::constants::FAMILY_INVITATION_CODE_LEN;
|
2023-06-20 16:55:14 +00:00
|
|
|
use crate::manual_schema::families_memberships;
|
|
|
|
use crate::models::{
|
|
|
|
Family, FamilyID, FamilyMembership, Membership, NewFamily, NewMembership, UserID,
|
|
|
|
};
|
2023-06-16 15:51:51 +00:00
|
|
|
use crate::schema::{families, memberships};
|
2023-06-21 15:01:52 +00:00
|
|
|
use crate::services::users_service;
|
2023-06-16 15:51:51 +00:00
|
|
|
use crate::utils::string_utils::rand_str;
|
|
|
|
use crate::utils::time_utils::time;
|
2023-06-17 16:55:07 +00:00
|
|
|
use diesel::prelude::*;
|
2023-06-16 15:51:51 +00:00
|
|
|
|
|
|
|
/// Create a new family, with an initial administrator
|
|
|
|
pub async fn create(name: &str, user_id: UserID) -> anyhow::Result<Family> {
|
|
|
|
let family = db_connection::execute(|conn| {
|
|
|
|
let res: Family = diesel::insert_into(families::table)
|
|
|
|
.values(&NewFamily {
|
|
|
|
name: name.trim(),
|
|
|
|
invitation_code: rand_str(FAMILY_INVITATION_CODE_LEN),
|
|
|
|
time_create: time() as i64,
|
|
|
|
})
|
|
|
|
.get_result(conn)?;
|
|
|
|
|
|
|
|
Ok(res)
|
|
|
|
})?;
|
|
|
|
|
|
|
|
add_member(family.id(), user_id, true).await?;
|
|
|
|
|
|
|
|
Ok(family)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Add a member to a family
|
|
|
|
pub async fn add_member(
|
|
|
|
family_id: FamilyID,
|
|
|
|
user_id: UserID,
|
|
|
|
admin: bool,
|
|
|
|
) -> anyhow::Result<Membership> {
|
|
|
|
db_connection::execute(|conn| {
|
|
|
|
let res = diesel::insert_into(memberships::table)
|
|
|
|
.values(&NewMembership {
|
|
|
|
user_id: user_id.0,
|
|
|
|
family_id: family_id.0,
|
|
|
|
time_create: time() as i64,
|
|
|
|
is_admin: admin,
|
|
|
|
})
|
|
|
|
.get_result(conn)?;
|
|
|
|
|
|
|
|
Ok(res)
|
|
|
|
})
|
|
|
|
}
|
2023-06-16 16:47:08 +00:00
|
|
|
|
2023-06-17 16:55:07 +00:00
|
|
|
/// Find a family by invitation code
|
|
|
|
pub async fn get_by_invitation_code(code: &str) -> anyhow::Result<Family> {
|
|
|
|
db_connection::execute(|conn| {
|
|
|
|
families::table
|
|
|
|
.filter(families::dsl::invitation_code.eq(code))
|
|
|
|
.first(conn)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Check if a given user is member of a family or not
|
|
|
|
pub async fn is_member(family_id: FamilyID, user_id: UserID) -> anyhow::Result<bool> {
|
|
|
|
db_connection::execute(|conn| {
|
|
|
|
memberships::table
|
|
|
|
.filter(memberships::dsl::family_id.eq(family_id.0))
|
|
|
|
.filter(memberships::dsl::user_id.eq(user_id.0))
|
|
|
|
.count()
|
|
|
|
.get_result(conn)
|
|
|
|
})
|
|
|
|
.map(|c: i64| c > 0)
|
|
|
|
}
|
|
|
|
|
2023-06-20 16:55:14 +00:00
|
|
|
/// Get the memberships of a user
|
2023-06-21 14:36:46 +00:00
|
|
|
pub async fn get_user_family_memberships(user_id: UserID) -> anyhow::Result<Vec<FamilyMembership>> {
|
2023-06-20 16:55:14 +00:00
|
|
|
db_connection::execute(|conn| {
|
|
|
|
families_memberships::table
|
|
|
|
.filter(families_memberships::dsl::user_id.eq(user_id.0))
|
|
|
|
.get_results(conn)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2023-06-21 14:36:46 +00:00
|
|
|
/// Get information about a membership of a user
|
|
|
|
pub async fn get_membership(family_id: FamilyID, user_id: UserID) -> anyhow::Result<Membership> {
|
|
|
|
db_connection::execute(|conn| {
|
|
|
|
memberships::table
|
|
|
|
.filter(memberships::dsl::user_id.eq(user_id.0))
|
|
|
|
.filter(memberships::dsl::family_id.eq(family_id.0))
|
|
|
|
.get_result(conn)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2023-06-21 15:01:52 +00:00
|
|
|
#[derive(serde::Serialize)]
|
|
|
|
pub struct FamilyMember {
|
|
|
|
#[serde(flatten)]
|
|
|
|
membership: Membership,
|
|
|
|
user_name: String,
|
|
|
|
user_mail: String,
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Get information about the users of a family
|
|
|
|
pub async fn get_memberships_of_family(family_id: FamilyID) -> anyhow::Result<Vec<FamilyMember>> {
|
|
|
|
let memberships = db_connection::execute(|conn| {
|
|
|
|
memberships::table
|
|
|
|
.filter(memberships::dsl::family_id.eq(family_id.0))
|
|
|
|
.get_results::<Membership>(conn)
|
|
|
|
})?;
|
|
|
|
|
|
|
|
let mut out = Vec::with_capacity(memberships.len());
|
|
|
|
|
|
|
|
for m in memberships {
|
|
|
|
let user = users_service::get_by_id(m.user_id()).await?;
|
|
|
|
out.push(FamilyMember {
|
|
|
|
user_name: user.name,
|
|
|
|
user_mail: user.email,
|
|
|
|
membership: m,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(out)
|
|
|
|
}
|
|
|
|
|
2023-06-21 14:36:46 +00:00
|
|
|
/// Get information about a membership of a user, joined with family information
|
|
|
|
pub async fn get_family_membership(
|
|
|
|
family_id: FamilyID,
|
|
|
|
user_id: UserID,
|
|
|
|
) -> anyhow::Result<FamilyMembership> {
|
|
|
|
db_connection::execute(|conn| {
|
|
|
|
families_memberships::table
|
|
|
|
.filter(families_memberships::dsl::user_id.eq(user_id.0))
|
|
|
|
.filter(families_memberships::dsl::family_id.eq(family_id.0))
|
|
|
|
.get_result(conn)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2023-06-16 16:47:08 +00:00
|
|
|
/// Remove a membership to a family
|
2023-06-17 16:55:07 +00:00
|
|
|
pub async fn remove_membership(_family_id: FamilyID, _user_id: UserID) {
|
2023-06-16 16:47:08 +00:00
|
|
|
todo!()
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Remove all memberships of user
|
2023-06-17 16:55:07 +00:00
|
|
|
pub async fn remove_all_user_membership(_user_id: UserID) -> anyhow::Result<()> {
|
2023-06-16 16:47:08 +00:00
|
|
|
todo!()
|
|
|
|
}
|