forked from didirus/AstralRinth
280 lines
7.4 KiB
Rust
280 lines
7.4 KiB
Rust
use crate::{
|
|
auth::get_user_from_headers,
|
|
database::{
|
|
models::{DBAffiliateCode, DBAffiliateCodeId, DBUser, DBUserId},
|
|
redis::RedisPool,
|
|
},
|
|
models::{
|
|
ids::AffiliateCodeId,
|
|
pats::Scopes,
|
|
users::Badges,
|
|
v3::affiliate_code::{AdminAffiliateCode, AffiliateCode},
|
|
},
|
|
queue::session::AuthQueue,
|
|
};
|
|
use actix_web::{HttpRequest, HttpResponse, web};
|
|
use ariadne::ids::UserId;
|
|
use chrono::Utc;
|
|
use serde::Deserialize;
|
|
use sqlx::PgPool;
|
|
|
|
use crate::routes::ApiError;
|
|
|
|
pub fn config(cfg: &mut web::ServiceConfig) {
|
|
cfg.service(
|
|
web::scope("affiliate")
|
|
.route("", web::get().to(get_all))
|
|
.route("", web::put().to(create))
|
|
.route("/{id}", web::get().to(get))
|
|
.route("/{id}", web::delete().to(delete))
|
|
.route("/{id}", web::patch().to(patch)),
|
|
);
|
|
}
|
|
|
|
async fn get_all(
|
|
req: HttpRequest,
|
|
pool: web::Data<PgPool>,
|
|
redis: web::Data<RedisPool>,
|
|
session_queue: web::Data<AuthQueue>,
|
|
) -> Result<HttpResponse, ApiError> {
|
|
let (_, user) = get_user_from_headers(
|
|
&req,
|
|
&**pool,
|
|
&redis,
|
|
&session_queue,
|
|
Scopes::SESSION_ACCESS,
|
|
)
|
|
.await?;
|
|
|
|
if user.role.is_admin() {
|
|
let codes = DBAffiliateCode::get_all(&**pool).await?;
|
|
let codes = codes
|
|
.into_iter()
|
|
.map(AdminAffiliateCode::from)
|
|
.collect::<Vec<_>>();
|
|
Ok(HttpResponse::Ok().json(codes))
|
|
} else if user.badges.contains(Badges::AFFILIATE) {
|
|
let codes =
|
|
DBAffiliateCode::get_by_affiliate(DBUserId::from(user.id), &**pool)
|
|
.await?;
|
|
let codes = codes
|
|
.into_iter()
|
|
.map(AffiliateCode::from)
|
|
.collect::<Vec<_>>();
|
|
Ok(HttpResponse::Ok().json(codes))
|
|
} else {
|
|
Err(ApiError::CustomAuthentication(
|
|
"You do not have permission to view affiliate codes!".to_string(),
|
|
))
|
|
}
|
|
}
|
|
|
|
#[derive(Deserialize)]
|
|
struct CreateRequest {
|
|
affiliate: Option<UserId>,
|
|
source_name: String,
|
|
}
|
|
|
|
async fn create(
|
|
req: HttpRequest,
|
|
pool: web::Data<PgPool>,
|
|
redis: web::Data<RedisPool>,
|
|
session_queue: web::Data<AuthQueue>,
|
|
body: web::Json<CreateRequest>,
|
|
) -> Result<HttpResponse, ApiError> {
|
|
let (_, creator) = get_user_from_headers(
|
|
&req,
|
|
&**pool,
|
|
&redis,
|
|
&session_queue,
|
|
Scopes::SESSION_ACCESS,
|
|
)
|
|
.await?;
|
|
|
|
let is_admin = creator.role.is_admin();
|
|
let is_affiliate = creator.badges.contains(Badges::AFFILIATE);
|
|
|
|
if !is_admin && !is_affiliate {
|
|
return Err(ApiError::CustomAuthentication(
|
|
"You do not have permission to create an affiliate code!"
|
|
.to_string(),
|
|
));
|
|
}
|
|
|
|
let creator_id = DBUserId::from(creator.id);
|
|
let affiliate_id = if is_admin {
|
|
if let Some(affiliate) = body.affiliate {
|
|
DBUserId::from(affiliate)
|
|
} else {
|
|
creator_id
|
|
}
|
|
} else {
|
|
creator_id
|
|
};
|
|
|
|
if affiliate_id != creator_id {
|
|
let Some(_affiliate_user) =
|
|
DBUser::get_id(affiliate_id, &**pool, &redis).await?
|
|
else {
|
|
return Err(ApiError::CustomAuthentication(
|
|
"Affiliate user not found!".to_string(),
|
|
));
|
|
};
|
|
}
|
|
|
|
let mut transaction = pool.begin().await?;
|
|
|
|
let affiliate_code_id =
|
|
crate::database::models::generate_affiliate_code_id(&mut transaction)
|
|
.await?;
|
|
|
|
let code = DBAffiliateCode {
|
|
id: affiliate_code_id,
|
|
created_at: Utc::now(),
|
|
created_by: creator_id,
|
|
affiliate: affiliate_id,
|
|
source_name: body.source_name.clone(),
|
|
};
|
|
code.insert(&mut *transaction).await?;
|
|
|
|
transaction.commit().await?;
|
|
|
|
if is_admin {
|
|
Ok(HttpResponse::Created().json(AdminAffiliateCode::from(code)))
|
|
} else {
|
|
Ok(HttpResponse::Created().json(AffiliateCode::from(code)))
|
|
}
|
|
}
|
|
|
|
async fn get(
|
|
req: HttpRequest,
|
|
path: web::Path<(AffiliateCodeId,)>,
|
|
pool: web::Data<PgPool>,
|
|
redis: web::Data<RedisPool>,
|
|
session_queue: web::Data<AuthQueue>,
|
|
) -> Result<HttpResponse, ApiError> {
|
|
let (_, user) = get_user_from_headers(
|
|
&req,
|
|
&**pool,
|
|
&redis,
|
|
&session_queue,
|
|
Scopes::SESSION_ACCESS,
|
|
)
|
|
.await?;
|
|
|
|
let (affiliate_code_id,) = path.into_inner();
|
|
let affiliate_code_id = DBAffiliateCodeId::from(affiliate_code_id);
|
|
|
|
if let Some(model) =
|
|
DBAffiliateCode::get_by_id(affiliate_code_id, &**pool).await?
|
|
{
|
|
let is_admin = user.role.is_admin();
|
|
let is_owner = model.affiliate == DBUserId::from(user.id);
|
|
|
|
if is_admin || is_owner {
|
|
if is_admin {
|
|
Ok(HttpResponse::Ok().json(AdminAffiliateCode::from(model)))
|
|
} else {
|
|
Ok(HttpResponse::Ok().json(AffiliateCode::from(model)))
|
|
}
|
|
} else {
|
|
Err(ApiError::NotFound)
|
|
}
|
|
} else {
|
|
Err(ApiError::NotFound)
|
|
}
|
|
}
|
|
|
|
async fn delete(
|
|
req: HttpRequest,
|
|
path: web::Path<(AffiliateCodeId,)>,
|
|
pool: web::Data<PgPool>,
|
|
redis: web::Data<RedisPool>,
|
|
session_queue: web::Data<AuthQueue>,
|
|
) -> Result<HttpResponse, ApiError> {
|
|
let (_, user) = get_user_from_headers(
|
|
&req,
|
|
&**pool,
|
|
&redis,
|
|
&session_queue,
|
|
Scopes::SESSION_ACCESS,
|
|
)
|
|
.await?;
|
|
|
|
let (affiliate_code_id,) = path.into_inner();
|
|
let affiliate_code_id = DBAffiliateCodeId::from(affiliate_code_id);
|
|
|
|
if let Some(model) =
|
|
DBAffiliateCode::get_by_id(affiliate_code_id, &**pool).await?
|
|
{
|
|
let is_admin = user.role.is_admin();
|
|
let is_owner = model.affiliate == DBUserId::from(user.id);
|
|
|
|
if is_admin || is_owner {
|
|
let result =
|
|
DBAffiliateCode::remove(affiliate_code_id, &**pool).await?;
|
|
if result.is_some() {
|
|
Ok(HttpResponse::NoContent().finish())
|
|
} else {
|
|
Err(ApiError::NotFound)
|
|
}
|
|
} else {
|
|
Err(ApiError::NotFound)
|
|
}
|
|
} else {
|
|
Err(ApiError::NotFound)
|
|
}
|
|
}
|
|
|
|
#[derive(Deserialize)]
|
|
struct PatchRequest {
|
|
source_name: String,
|
|
}
|
|
|
|
async fn patch(
|
|
req: HttpRequest,
|
|
path: web::Path<(AffiliateCodeId,)>,
|
|
pool: web::Data<PgPool>,
|
|
redis: web::Data<RedisPool>,
|
|
session_queue: web::Data<AuthQueue>,
|
|
body: web::Json<PatchRequest>,
|
|
) -> Result<HttpResponse, ApiError> {
|
|
let (_, user) = get_user_from_headers(
|
|
&req,
|
|
&**pool,
|
|
&redis,
|
|
&session_queue,
|
|
Scopes::SESSION_ACCESS,
|
|
)
|
|
.await?;
|
|
|
|
let (affiliate_code_id,) = path.into_inner();
|
|
let affiliate_code_id = DBAffiliateCodeId::from(affiliate_code_id);
|
|
|
|
let existing_code = DBAffiliateCode::get_by_id(affiliate_code_id, &**pool)
|
|
.await?
|
|
.ok_or(ApiError::NotFound)?;
|
|
|
|
let is_admin = user.role.is_admin();
|
|
let is_owner = existing_code.affiliate == DBUserId::from(user.id);
|
|
|
|
if !is_admin && !is_owner {
|
|
return Err(ApiError::NotFound);
|
|
}
|
|
|
|
if !is_admin && !user.badges.contains(Badges::AFFILIATE) {
|
|
return Err(ApiError::CustomAuthentication(
|
|
"You do not have permission to update affiliate codes!".to_string(),
|
|
));
|
|
}
|
|
|
|
DBAffiliateCode::update_source_name(
|
|
affiliate_code_id,
|
|
&body.source_name,
|
|
&**pool,
|
|
)
|
|
.await?;
|
|
|
|
Ok(HttpResponse::NoContent().finish())
|
|
}
|