Compare commits
	
		
			4 Commits
		
	
	
		
			06e1474396
			...
			8ba0f02b98
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 8ba0f02b98 | |||
| d36438c6ce | |||
| e0c60e2ce3 | |||
|  | e7160215b0 | 
							
								
								
									
										861
									
								
								Cargo.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										861
									
								
								Cargo.lock
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,6 +1,6 @@ | ||||
| [package] | ||||
| name = "reminder-rs" | ||||
| version = "1.6.47" | ||||
| version = "1.6.50" | ||||
| authors = ["Jude Southworth <judesouthworth@pm.me>"] | ||||
| edition = "2021" | ||||
| license = "AGPL-3.0 only" | ||||
|   | ||||
| @@ -114,6 +114,8 @@ pub async fn offset( | ||||
|     #[description = "Number of minutes to offset by"] minutes: Option<isize>, | ||||
|     #[description = "Number of seconds to offset by"] seconds: Option<isize>, | ||||
| ) -> Result<(), Error> { | ||||
|     ctx.defer().await?; | ||||
|  | ||||
|     let combined_time = hours.map_or(0, |h| h * HOUR as isize) | ||||
|         + minutes.map_or(0, |m| m * MINUTE as isize) | ||||
|         + seconds.map_or(0, |s| s); | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| [package] | ||||
| name = "reminder_web" | ||||
| version = "0.1.2" | ||||
| version = "0.1.4" | ||||
| authors = ["jellywx <judesouthworth@pm.me>"] | ||||
| edition = "2018" | ||||
|  | ||||
|   | ||||
| @@ -33,11 +33,9 @@ impl<'r> FromRequest<'r> for Transaction<'r> { | ||||
|         match request.guard::<&State<Pool<Database>>>().await { | ||||
|             Outcome::Success(pool) => match pool.begin().await { | ||||
|                 Ok(transaction) => Outcome::Success(Transaction(transaction)), | ||||
|                 Err(e) => { | ||||
|                     Outcome::Failure((Status::InternalServerError, TransactionError::Error(e))) | ||||
|                 } | ||||
|                 Err(e) => Outcome::Error((Status::InternalServerError, TransactionError::Error(e))), | ||||
|             }, | ||||
|             Outcome::Failure(e) => Outcome::Failure((e.0, TransactionError::Missing)), | ||||
|             Outcome::Error(e) => Outcome::Error((e.0, TransactionError::Missing)), | ||||
|             Outcome::Forward(f) => Outcome::Forward(f), | ||||
|         } | ||||
|     } | ||||
|   | ||||
| @@ -4,7 +4,7 @@ use chrono::{naive::NaiveDateTime, Utc}; | ||||
| use rand::{rngs::OsRng, seq::IteratorRandom}; | ||||
| use rocket::{http::CookieJar, response::Redirect, serde::json::json}; | ||||
| use rocket_dyn_templates::Template; | ||||
| use serde::{Deserialize, Deserializer, Serialize}; | ||||
| use serde::{de, Deserialize, Deserializer, Serialize, Serializer}; | ||||
| use serenity::{ | ||||
|     client::Context, | ||||
|     http::Http, | ||||
| @@ -51,12 +51,27 @@ fn interval_default() -> Unset<Option<u32>> { | ||||
|     None | ||||
| } | ||||
|  | ||||
| fn deserialize_optional_field<'de, T, D>(deserializer: D) -> Result<Option<Option<T>>, D::Error> | ||||
| #[derive(sqlx::Type)] | ||||
| #[sqlx(transparent)] | ||||
| struct Attachment(Vec<u8>); | ||||
|  | ||||
| impl<'de> Deserialize<'de> for Attachment { | ||||
|     fn deserialize<D>(deserializer: D) -> Result<Attachment, D::Error> | ||||
|     where | ||||
|         D: Deserializer<'de>, | ||||
|     T: Deserialize<'de>, | ||||
|     { | ||||
|     Ok(Some(Option::deserialize(deserializer)?)) | ||||
|         let string = String::deserialize(deserializer)?; | ||||
|         Ok(Attachment(base64::decode(string).map_err(de::Error::custom)?)) | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl Serialize for Attachment { | ||||
|     fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error> | ||||
|     where | ||||
|         S: Serializer, | ||||
|     { | ||||
|         serializer.collect_str(&base64::encode(&self.0)) | ||||
|     } | ||||
| } | ||||
|  | ||||
| #[derive(Serialize, Deserialize)] | ||||
| @@ -67,7 +82,7 @@ pub struct ReminderTemplate { | ||||
|     guild_id: u32, | ||||
|     #[serde(default = "template_name_default")] | ||||
|     name: String, | ||||
|     attachment: Option<Vec<u8>>, | ||||
|     attachment: Option<Attachment>, | ||||
|     attachment_name: Option<String>, | ||||
|     avatar: Option<String>, | ||||
|     content: String, | ||||
| @@ -92,7 +107,7 @@ pub struct ReminderTemplate { | ||||
| pub struct ReminderTemplateCsv { | ||||
|     #[serde(default = "template_name_default")] | ||||
|     name: String, | ||||
|     attachment: Option<Vec<u8>>, | ||||
|     attachment: Option<Attachment>, | ||||
|     attachment_name: Option<String>, | ||||
|     avatar: Option<String>, | ||||
|     content: String, | ||||
| @@ -127,8 +142,7 @@ pub struct EmbedField { | ||||
|  | ||||
| #[derive(Serialize, Deserialize)] | ||||
| pub struct Reminder { | ||||
|     #[serde(with = "base64s")] | ||||
|     attachment: Option<Vec<u8>>, | ||||
|     attachment: Option<Attachment>, | ||||
|     attachment_name: Option<String>, | ||||
|     avatar: Option<String>, | ||||
|     #[serde(with = "string")] | ||||
| @@ -161,8 +175,7 @@ pub struct Reminder { | ||||
|  | ||||
| #[derive(Serialize, Deserialize)] | ||||
| pub struct ReminderCsv { | ||||
|     #[serde(with = "base64s")] | ||||
|     attachment: Option<Vec<u8>>, | ||||
|     attachment: Option<Attachment>, | ||||
|     attachment_name: Option<String>, | ||||
|     avatar: Option<String>, | ||||
|     channel: String, | ||||
| @@ -195,7 +208,7 @@ pub struct PatchReminder { | ||||
|     uid: String, | ||||
|     #[serde(default)] | ||||
|     #[serde(deserialize_with = "deserialize_optional_field")] | ||||
|     attachment: Unset<Option<String>>, | ||||
|     attachment: Unset<Option<Attachment>>, | ||||
|     #[serde(default)] | ||||
|     #[serde(deserialize_with = "deserialize_optional_field")] | ||||
|     attachment_name: Unset<Option<String>>, | ||||
| @@ -291,6 +304,14 @@ pub fn generate_uid() -> String { | ||||
|         .join("") | ||||
| } | ||||
|  | ||||
| fn deserialize_optional_field<'de, T, D>(deserializer: D) -> Result<Option<Option<T>>, D::Error> | ||||
| where | ||||
|     D: Deserializer<'de>, | ||||
|     T: Deserialize<'de>, | ||||
| { | ||||
|     Ok(Some(Option::deserialize(deserializer)?)) | ||||
| } | ||||
|  | ||||
| // https://github.com/serde-rs/json/issues/329#issuecomment-305608405 | ||||
| mod string { | ||||
|     use std::{fmt::Display, str::FromStr}; | ||||
| @@ -315,29 +336,6 @@ mod string { | ||||
|     } | ||||
| } | ||||
|  | ||||
| mod base64s { | ||||
|     use serde::{de, Deserialize, Deserializer, Serializer}; | ||||
|  | ||||
|     pub fn serialize<S>(value: &Option<Vec<u8>>, serializer: S) -> Result<S::Ok, S::Error> | ||||
|     where | ||||
|         S: Serializer, | ||||
|     { | ||||
|         if let Some(opt) = value { | ||||
|             serializer.collect_str(&base64::encode(opt)) | ||||
|         } else { | ||||
|             serializer.serialize_none() | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     pub fn deserialize<'de, D>(deserializer: D) -> Result<Option<Vec<u8>>, D::Error> | ||||
|     where | ||||
|         D: Deserializer<'de>, | ||||
|     { | ||||
|         let string = Option::<String>::deserialize(deserializer)?; | ||||
|         Some(string.map(|b| base64::decode(b).map_err(de::Error::custom))).flatten().transpose() | ||||
|     } | ||||
| } | ||||
|  | ||||
| #[derive(Deserialize)] | ||||
| pub struct DeleteReminder { | ||||
|     uid: String, | ||||
|   | ||||
| @@ -31,22 +31,20 @@ pub async fn discord_login( | ||||
|  | ||||
|     // store the pkce secret to verify the authorization later | ||||
|     cookies.add_private( | ||||
|         Cookie::build("verify", pkce_verifier.secret().to_string()) | ||||
|         Cookie::build(("verify", pkce_verifier.secret().to_string())) | ||||
|             .http_only(true) | ||||
|             .path("/login") | ||||
|             .same_site(SameSite::Lax) | ||||
|             .expires(Expiration::Session) | ||||
|             .finish(), | ||||
|             .expires(Expiration::Session), | ||||
|     ); | ||||
|  | ||||
|     // store the csrf token to verify no interference | ||||
|     cookies.add_private( | ||||
|         Cookie::build("csrf", csrf_token.secret().to_string()) | ||||
|         Cookie::build(("csrf", csrf_token.secret().to_string())) | ||||
|             .http_only(true) | ||||
|             .path("/login") | ||||
|             .same_site(SameSite::Lax) | ||||
|             .expires(Expiration::Session) | ||||
|             .finish(), | ||||
|             .expires(Expiration::Session), | ||||
|     ); | ||||
|  | ||||
|     Redirect::to(auth_url.to_string()) | ||||
| @@ -54,9 +52,9 @@ pub async fn discord_login( | ||||
|  | ||||
| #[get("/discord/logout")] | ||||
| pub async fn discord_logout(cookies: &CookieJar<'_>) -> Redirect { | ||||
|     cookies.remove_private(Cookie::named("username")); | ||||
|     cookies.remove_private(Cookie::named("userid")); | ||||
|     cookies.remove_private(Cookie::named("access_token")); | ||||
|     cookies.remove_private(Cookie::from("username")); | ||||
|     cookies.remove_private(Cookie::from("userid")); | ||||
|     cookies.remove_private(Cookie::from("access_token")); | ||||
|  | ||||
|     Redirect::to(uri!(routes::index)) | ||||
| } | ||||
| @@ -80,17 +78,16 @@ pub async fn discord_callback( | ||||
|                 .request_async(async_http_client) | ||||
|                 .await; | ||||
|  | ||||
|             cookies.remove_private(Cookie::named("verify")); | ||||
|             cookies.remove_private(Cookie::named("csrf")); | ||||
|             cookies.remove_private(Cookie::from("verify")); | ||||
|             cookies.remove_private(Cookie::from("csrf")); | ||||
|  | ||||
|             match token_result { | ||||
|                 Ok(token) => { | ||||
|                     cookies.add_private( | ||||
|                         Cookie::build("access_token", token.access_token().secret().to_string()) | ||||
|                         Cookie::build(("access_token", token.access_token().secret().to_string())) | ||||
|                             .secure(true) | ||||
|                             .http_only(true) | ||||
|                             .path("/dashboard") | ||||
|                             .finish(), | ||||
|                             .path("/dashboard"), | ||||
|                     ); | ||||
|  | ||||
|                     let request_res = reqwest_client | ||||
|   | ||||
		Reference in New Issue
	
	Block a user