You've already forked AstralRinth
forked from didirus/AstralRinth
* Dynamic email template * Set lower cache expiry for templates * Custom email route * Fix subject line on custom emails * chore: query cache, clippy, fmt * Bugfixes * Key-based caching on custom emails * Sequentially process emails prone to causing cache stampede * Fill variables in dynamic body + subject line * Update apps/labrinth/src/queue/email/templates.rs Co-authored-by: aecsocket <aecsocket@tutanota.com> Signed-off-by: François-Xavier Talbot <108630700+fetchfern@users.noreply.github.com> * Update apps/labrinth/src/queue/email/templates.rs Co-authored-by: aecsocket <aecsocket@tutanota.com> Signed-off-by: François-Xavier Talbot <108630700+fetchfern@users.noreply.github.com> --------- Signed-off-by: François-Xavier Talbot <108630700+fetchfern@users.noreply.github.com> Co-authored-by: aecsocket <aecsocket@tutanota.com>
119 lines
3.0 KiB
Rust
119 lines
3.0 KiB
Rust
use crate::auth::get_user_from_headers;
|
|
use crate::database::models::ids::DBUserId;
|
|
use crate::database::models::notification_item::NotificationBuilder;
|
|
use crate::database::models::user_item::DBUser;
|
|
use crate::database::redis::RedisPool;
|
|
use crate::models::users::Role;
|
|
use crate::models::v3::notifications::NotificationBody;
|
|
use crate::models::v3::pats::Scopes;
|
|
use crate::queue::session::AuthQueue;
|
|
use crate::routes::ApiError;
|
|
use crate::util::guards::external_notification_key_guard;
|
|
use actix_web::HttpRequest;
|
|
use actix_web::web;
|
|
use actix_web::{HttpResponse, post};
|
|
use ariadne::ids::UserId;
|
|
use serde::Deserialize;
|
|
use sqlx::PgPool;
|
|
|
|
pub fn config(cfg: &mut web::ServiceConfig) {
|
|
cfg.service(create).service(send_custom_email);
|
|
}
|
|
|
|
#[derive(Deserialize)]
|
|
struct CreateNotification {
|
|
pub body: NotificationBody,
|
|
pub user_ids: Vec<UserId>,
|
|
}
|
|
|
|
#[post("external_notifications", guard = "external_notification_key_guard")]
|
|
pub async fn create(
|
|
pool: web::Data<PgPool>,
|
|
redis: web::Data<RedisPool>,
|
|
create_notification: web::Json<CreateNotification>,
|
|
) -> Result<HttpResponse, ApiError> {
|
|
let CreateNotification { body, user_ids } =
|
|
create_notification.into_inner();
|
|
let user_ids = user_ids
|
|
.into_iter()
|
|
.map(|x| DBUserId(x.0 as i64))
|
|
.collect::<Vec<_>>();
|
|
|
|
let mut txn = pool.begin().await?;
|
|
|
|
if !DBUser::exists_many(&user_ids, &mut *txn).await? {
|
|
return Err(ApiError::InvalidInput(
|
|
"One of the specified users do not exist.".to_owned(),
|
|
));
|
|
}
|
|
|
|
NotificationBuilder { body }
|
|
.insert_many(user_ids, &mut txn, &redis)
|
|
.await?;
|
|
|
|
txn.commit().await?;
|
|
|
|
Ok(HttpResponse::Accepted().finish())
|
|
}
|
|
|
|
#[derive(Deserialize)]
|
|
struct SendEmail {
|
|
pub users: Vec<UserId>,
|
|
pub key: String,
|
|
pub body_md: String,
|
|
pub title: String,
|
|
}
|
|
|
|
#[post("external_notifications/send_custom_email")]
|
|
pub async fn send_custom_email(
|
|
req: HttpRequest,
|
|
pool: web::Data<PgPool>,
|
|
redis: web::Data<RedisPool>,
|
|
session_queue: web::Data<AuthQueue>,
|
|
body: web::Json<SendEmail>,
|
|
) -> Result<HttpResponse, ApiError> {
|
|
let user = get_user_from_headers(
|
|
&req,
|
|
&**pool,
|
|
&redis,
|
|
&session_queue,
|
|
Scopes::SESSION_ACCESS,
|
|
)
|
|
.await?
|
|
.1;
|
|
|
|
if user.role != Role::Admin {
|
|
return Err(ApiError::CustomAuthentication(
|
|
"You do not have permission to send custom emails!".to_string(),
|
|
));
|
|
}
|
|
|
|
let SendEmail {
|
|
users,
|
|
body_md,
|
|
title,
|
|
key,
|
|
} = body.into_inner();
|
|
|
|
let users = users
|
|
.into_iter()
|
|
.map(|x| DBUserId(x.0 as i64))
|
|
.collect::<Vec<_>>();
|
|
|
|
let mut txn = pool.begin().await?;
|
|
|
|
NotificationBuilder {
|
|
body: NotificationBody::Custom {
|
|
title,
|
|
body_md,
|
|
key,
|
|
},
|
|
}
|
|
.insert_many(users, &mut txn, &redis)
|
|
.await?;
|
|
|
|
txn.commit().await?;
|
|
|
|
Ok(HttpResponse::Accepted().finish())
|
|
}
|