use rocket::{ http::CookieJar, serde::json::{json, Json}, State, }; use serenity::{ client::Context, model::id::{ChannelId, GuildId, UserId}, }; use sqlx::{MySql, Pool}; use crate::{ check_authorization, check_guild_subscription, check_subscription, consts::MIN_INTERVAL, guards::transaction::Transaction, routes::{ dashboard::{ create_database_channel, create_reminder, DeleteReminder, PatchReminder, Reminder, }, JsonResult, }, Database, }; #[post("/api/guild//reminders", data = "")] pub async fn create_guild_reminder( id: u64, reminder: Json, cookies: &CookieJar<'_>, ctx: &State, mut transaction: Transaction<'_>, ) -> JsonResult { check_authorization(cookies, ctx.inner(), id).await?; let user_id = cookies.get_private("userid").map(|c| c.value().parse::().ok()).flatten().unwrap(); match create_reminder( ctx.inner(), &mut transaction, GuildId(id), UserId(user_id), reminder.into_inner(), ) .await { Ok(r) => match transaction.commit().await { Ok(_) => Ok(r), Err(e) => { warn!("Couldn't commit transaction: {:?}", e); json_err!("Couldn't commit transaction.") } }, Err(e) => Err(e), } } #[get("/api/guild//reminders")] pub async fn get_reminders( id: u64, cookies: &CookieJar<'_>, ctx: &State, pool: &State>, ) -> JsonResult { check_authorization(cookies, ctx.inner(), id).await?; let channels_res = GuildId(id).channels(&ctx.inner()).await; match channels_res { Ok(channels) => { let channels = channels .keys() .into_iter() .map(|k| k.as_u64().to_string()) .collect::>() .join(","); sqlx::query_as_unchecked!( Reminder, "SELECT reminders.attachment, reminders.attachment_name, reminders.avatar, channels.channel, reminders.content, reminders.embed_author, reminders.embed_author_url, reminders.embed_color, reminders.embed_description, reminders.embed_footer, reminders.embed_footer_url, reminders.embed_image_url, reminders.embed_thumbnail_url, reminders.embed_title, IFNULL(reminders.embed_fields, '[]') AS embed_fields, reminders.enabled, reminders.expires, reminders.interval_seconds, reminders.interval_days, reminders.interval_months, reminders.name, reminders.restartable, reminders.tts, reminders.uid, reminders.username, reminders.utc_time FROM reminders LEFT JOIN channels ON channels.id = reminders.channel_id WHERE `status` = 'pending' AND FIND_IN_SET(channels.channel, ?)", channels ) .fetch_all(pool.inner()) .await .map(|r| Ok(json!(r))) .unwrap_or_else(|e| { warn!("Failed to complete SQL query: {:?}", e); json_err!("Could not load reminders") }) } Err(e) => { warn!("Could not fetch channels from {}: {:?}", id, e); Ok(json!([])) } } } #[patch("/api/guild//reminders", data = "")] pub async fn edit_reminder( id: u64, reminder: Json, ctx: &State, mut transaction: Transaction<'_>, pool: &State>, cookies: &CookieJar<'_>, ) -> JsonResult { check_authorization(cookies, ctx.inner(), id).await?; let mut error = vec![]; let user_id = cookies.get_private("userid").map(|c| c.value().parse::().ok()).flatten().unwrap(); if reminder.message_ok() { update_field!(transaction.executor(), error, reminder.[ content, embed_author, embed_description, embed_footer, embed_title, embed_fields, username ]); } else { error.push("Message exceeds limits.".to_string()); } update_field!(transaction.executor(), error, reminder.[ attachment, attachment_name, avatar, embed_author_url, embed_color, embed_footer_url, embed_image_url, embed_thumbnail_url, enabled, expires, name, restartable, tts, utc_time ]); if reminder.interval_days.flatten().is_some() || reminder.interval_months.flatten().is_some() || reminder.interval_seconds.flatten().is_some() { if check_guild_subscription(&ctx.inner(), id).await || check_subscription(&ctx.inner(), user_id).await { let new_interval_length = match reminder.interval_days { Some(interval) => interval.unwrap_or(0), None => sqlx::query!( "SELECT interval_days AS days FROM reminders WHERE uid = ?", reminder.uid ) .fetch_one(transaction.executor()) .await .map_err(|e| { warn!("Error updating reminder interval: {:?}", e); json!({ "reminder": Option::::None, "errors": vec!["Unknown error"] }) })? .days .unwrap_or(0), } * 86400 + match reminder.interval_months { Some(interval) => interval.unwrap_or(0), None => sqlx::query!( "SELECT interval_months AS months FROM reminders WHERE uid = ?", reminder.uid ) .fetch_one(transaction.executor()) .await .map_err(|e| { warn!("Error updating reminder interval: {:?}", e); json!({ "reminder": Option::::None, "errors": vec!["Unknown error"] }) })? .months .unwrap_or(0), } * 2592000 + match reminder.interval_seconds { Some(interval) => interval.unwrap_or(0), None => sqlx::query!( "SELECT interval_seconds AS seconds FROM reminders WHERE uid = ?", reminder.uid ) .fetch_one(transaction.executor()) .await .map_err(|e| { warn!("Error updating reminder interval: {:?}", e); json!({ "reminder": Option::::None, "errors": vec!["Unknown error"] }) })? .seconds .unwrap_or(0), }; if new_interval_length < *MIN_INTERVAL { error.push(String::from("New interval is too short.")); } else { update_field!(transaction.executor(), error, reminder.[ interval_days, interval_months, interval_seconds ]); } } } else { sqlx::query!( " UPDATE reminders SET interval_seconds = NULL, interval_days = NULL, interval_months = NULL WHERE uid = ? ", reminder.uid ) .execute(transaction.executor()) .await .map_err(|e| { warn!("Error updating reminder interval: {:?}", e); json!({ "reminder": Option::::None, "errors": vec!["Unknown error"] }) })?; } if reminder.channel > 0 { let channel = ChannelId(reminder.channel).to_channel_cached(&ctx.inner()); match channel { Some(channel) => { let channel_matches_guild = channel.guild().map_or(false, |c| c.guild_id.0 == id); if !channel_matches_guild { warn!( "Error in `edit_reminder`: channel {:?} not found for guild {}", reminder.channel, id ); return Err(json!({"error": "Channel not found"})); } let channel = create_database_channel( ctx.inner(), ChannelId(reminder.channel), &mut transaction, ) .await; if let Err(e) = channel { warn!("`create_database_channel` returned an error code: {:?}", e); return Err( json!({"error": "Failed to configure channel for reminders. Please check the bot permissions"}), ); } let channel = channel.unwrap(); match sqlx::query!( "UPDATE reminders SET channel_id = ? WHERE uid = ?", channel, reminder.uid ) .execute(transaction.executor()) .await { Ok(_) => {} Err(e) => { warn!("Error setting channel: {:?}", e); error.push("Couldn't set channel".to_string()) } } } None => { warn!( "Error in `edit_reminder`: channel {:?} not found for guild {}", reminder.channel, id ); return Err(json!({"error": "Channel not found"})); } } } if let Err(e) = transaction.commit().await { warn!("Couldn't commit transaction: {:?}", e); return json_err!("Couldn't commit transaction"); } match sqlx::query_as_unchecked!( Reminder, "SELECT reminders.attachment, reminders.attachment_name, reminders.avatar, channels.channel, reminders.content, reminders.embed_author, reminders.embed_author_url, reminders.embed_color, reminders.embed_description, reminders.embed_footer, reminders.embed_footer_url, reminders.embed_image_url, reminders.embed_thumbnail_url, reminders.embed_title, reminders.embed_fields, reminders.enabled, reminders.expires, reminders.interval_seconds, reminders.interval_days, reminders.interval_months, reminders.name, reminders.restartable, reminders.tts, reminders.uid, reminders.username, reminders.utc_time FROM reminders LEFT JOIN channels ON channels.id = reminders.channel_id WHERE uid = ?", reminder.uid ) .fetch_one(pool.inner()) .await { Ok(reminder) => Ok(json!({"reminder": reminder, "errors": error})), Err(e) => { warn!("Error exiting `edit_reminder': {:?}", e); Err(json!({"reminder": Option::::None, "errors": vec!["Unknown error"]})) } } } #[delete("/api/guild//reminders", data = "")] pub async fn delete_reminder( cookies: &CookieJar<'_>, id: u64, reminder: Json, ctx: &State, pool: &State>, ) -> JsonResult { check_authorization(cookies, ctx.inner(), id).await?; match sqlx::query!("UPDATE reminders SET `status` = 'deleted' WHERE uid = ?", reminder.uid) .execute(pool.inner()) .await { Ok(_) => Ok(json!({})), Err(e) => { warn!("Error in `delete_reminder`: {:?}", e); Err(json!({"error": "Could not delete reminder"})) } } }