Can join a family
This commit is contained in:
		@@ -1,7 +1,9 @@
 | 
			
		||||
use crate::constants::StaticConstraints;
 | 
			
		||||
use crate::controllers::HttpResult;
 | 
			
		||||
use crate::services::families_service;
 | 
			
		||||
use crate::services::login_token_service::LoginToken;
 | 
			
		||||
use crate::services::rate_limiter_service::RatedAction;
 | 
			
		||||
use crate::services::{families_service, rate_limiter_service};
 | 
			
		||||
use actix_remote_ip::RemoteIP;
 | 
			
		||||
use actix_web::{web, HttpResponse};
 | 
			
		||||
 | 
			
		||||
#[derive(Debug, serde::Deserialize)]
 | 
			
		||||
@@ -22,3 +24,42 @@ pub async fn create(req: web::Json<CreateFamilyReq>, token: LoginToken) -> HttpR
 | 
			
		||||
 | 
			
		||||
    Ok(HttpResponse::Created().json(family))
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#[derive(Debug, serde::Deserialize)]
 | 
			
		||||
pub struct JoinFamilyReq {
 | 
			
		||||
    code: String,
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Join a family
 | 
			
		||||
pub async fn join(
 | 
			
		||||
    remote_ip: RemoteIP,
 | 
			
		||||
    req: web::Json<JoinFamilyReq>,
 | 
			
		||||
    token: LoginToken,
 | 
			
		||||
) -> HttpResult {
 | 
			
		||||
    // Rate limiting
 | 
			
		||||
    if rate_limiter_service::should_block_action(remote_ip.0, RatedAction::JoinFamily).await? {
 | 
			
		||||
        return Ok(HttpResponse::TooManyRequests().finish());
 | 
			
		||||
    }
 | 
			
		||||
    rate_limiter_service::record_action(remote_ip.0, RatedAction::JoinFamily).await?;
 | 
			
		||||
 | 
			
		||||
    let family = match families_service::get_by_invitation_code(&req.code).await {
 | 
			
		||||
        Ok(f) => f,
 | 
			
		||||
        Err(e) => {
 | 
			
		||||
            log::error!("Could not find family by invitation code! {e}");
 | 
			
		||||
            return Ok(HttpResponse::NotFound().finish());
 | 
			
		||||
        }
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    if families_service::is_member(family.id(), token.user_id).await? {
 | 
			
		||||
        log::error!(
 | 
			
		||||
            "Could not add {:?} to family {:?} because it is already a member of the family!",
 | 
			
		||||
            token.user_id,
 | 
			
		||||
            family.id()
 | 
			
		||||
        );
 | 
			
		||||
        return Ok(HttpResponse::Conflict().finish());
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    families_service::add_member(family.id(), token.user_id, false).await?;
 | 
			
		||||
 | 
			
		||||
    Ok(HttpResponse::Accepted().finish())
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -91,6 +91,7 @@ async fn main() -> std::io::Result<()> {
 | 
			
		||||
                "/family/create",
 | 
			
		||||
                web::post().to(families_controller::create),
 | 
			
		||||
            )
 | 
			
		||||
            .route("/family/join", web::post().to(families_controller::join))
 | 
			
		||||
    })
 | 
			
		||||
    .bind(AppConfig::get().listen_address.as_str())?
 | 
			
		||||
    .run()
 | 
			
		||||
 
 | 
			
		||||
@@ -4,7 +4,7 @@ use crate::models::{Family, FamilyID, Membership, NewFamily, NewMembership, User
 | 
			
		||||
use crate::schema::{families, memberships};
 | 
			
		||||
use crate::utils::string_utils::rand_str;
 | 
			
		||||
use crate::utils::time_utils::time;
 | 
			
		||||
use diesel::RunQueryDsl;
 | 
			
		||||
use diesel::prelude::*;
 | 
			
		||||
 | 
			
		||||
/// Create a new family, with an initial administrator
 | 
			
		||||
pub async fn create(name: &str, user_id: UserID) -> anyhow::Result<Family> {
 | 
			
		||||
@@ -45,12 +45,33 @@ pub async fn add_member(
 | 
			
		||||
    })
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// 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)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Remove a membership to a family
 | 
			
		||||
pub async fn remove_membership(family_id: FamilyID, user_id: UserID) {
 | 
			
		||||
pub async fn remove_membership(_family_id: FamilyID, _user_id: UserID) {
 | 
			
		||||
    todo!()
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Remove all memberships of user
 | 
			
		||||
pub async fn remove_all_user_membership(user_id: UserID) -> anyhow::Result<()> {
 | 
			
		||||
pub async fn remove_all_user_membership(_user_id: UserID) -> anyhow::Result<()> {
 | 
			
		||||
    todo!()
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -12,6 +12,7 @@ pub enum RatedAction {
 | 
			
		||||
    StartOpenIDLogin,
 | 
			
		||||
    RequestReplacePasswordSignedIn,
 | 
			
		||||
    RequestDeleteAccount,
 | 
			
		||||
    JoinFamily,
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
impl RatedAction {
 | 
			
		||||
@@ -24,6 +25,7 @@ impl RatedAction {
 | 
			
		||||
            RatedAction::StartOpenIDLogin => "start-oidc-login",
 | 
			
		||||
            RatedAction::RequestReplacePasswordSignedIn => "req-pwd-signed-in",
 | 
			
		||||
            RatedAction::RequestDeleteAccount => "req-del-acct",
 | 
			
		||||
            RatedAction::JoinFamily => "join-family",
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
@@ -36,6 +38,7 @@ impl RatedAction {
 | 
			
		||||
            RatedAction::StartOpenIDLogin => 30,
 | 
			
		||||
            RatedAction::RequestReplacePasswordSignedIn => 5,
 | 
			
		||||
            RatedAction::RequestDeleteAccount => 5,
 | 
			
		||||
            RatedAction::JoinFamily => 10,
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
		Reference in New Issue
	
	Block a user