389 lines
12 KiB
Rust
389 lines
12 KiB
Rust
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/<id>/reminders", data = "<reminder>")]
|
|
pub async fn create_guild_reminder(
|
|
id: u64,
|
|
reminder: Json<Reminder>,
|
|
cookies: &CookieJar<'_>,
|
|
ctx: &State<Context>,
|
|
mut transaction: Transaction<'_>,
|
|
) -> JsonResult {
|
|
check_authorization(cookies, ctx.inner(), id).await?;
|
|
|
|
let user_id =
|
|
cookies.get_private("userid").map(|c| c.value().parse::<u64>().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/<id>/reminders")]
|
|
pub async fn get_reminders(
|
|
id: u64,
|
|
cookies: &CookieJar<'_>,
|
|
ctx: &State<Context>,
|
|
pool: &State<Pool<MySql>>,
|
|
) -> 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::<Vec<String>>()
|
|
.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/<id>/reminders", data = "<reminder>")]
|
|
pub async fn edit_reminder(
|
|
id: u64,
|
|
reminder: Json<PatchReminder>,
|
|
ctx: &State<Context>,
|
|
mut transaction: Transaction<'_>,
|
|
pool: &State<Pool<Database>>,
|
|
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::<u64>().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::<Reminder>::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::<Reminder>::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::<Reminder>::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::<Reminder>::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::<Reminder>::None, "errors": vec!["Unknown error"]}))
|
|
}
|
|
}
|
|
}
|
|
|
|
#[delete("/api/guild/<id>/reminders", data = "<reminder>")]
|
|
pub async fn delete_reminder(
|
|
cookies: &CookieJar<'_>,
|
|
id: u64,
|
|
reminder: Json<DeleteReminder>,
|
|
ctx: &State<Context>,
|
|
pool: &State<Pool<MySql>>,
|
|
) -> 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"}))
|
|
}
|
|
}
|
|
}
|