Compare commits
	
		
			10 Commits
		
	
	
		
			06e1474396
			...
			jude/react
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 1a03c2471b | ||
| a476f43f28 | |||
| 17192b0f89 | |||
|  | 0419863afa | ||
|  | 827a982a40 | ||
|  | 6e435bfc2e | ||
| 8ba0f02b98 | |||
| d36438c6ce | |||
| e0c60e2ce3 | |||
|  | e7160215b0 | 
							
								
								
									
										3
									
								
								.gitmodules
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.gitmodules
									
									
									
									
										vendored
									
									
								
							| @@ -1,3 +0,0 @@ | |||||||
| [submodule "reminder-dashboard"] |  | ||||||
| 	path = reminder-dashboard |  | ||||||
| 	url = gitea@gitea.jellypro.xyz:jude/reminder-dashboard |  | ||||||
							
								
								
									
										861
									
								
								Cargo.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										861
									
								
								Cargo.lock
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,6 +1,6 @@ | |||||||
| [package] | [package] | ||||||
| name = "reminder-rs" | name = "reminder-rs" | ||||||
| version = "1.6.48" | version = "1.6.50" | ||||||
| authors = ["Jude Southworth <judesouthworth@pm.me>"] | authors = ["Jude Southworth <judesouthworth@pm.me>"] | ||||||
| edition = "2021" | edition = "2021" | ||||||
| license = "AGPL-3.0 only" | license = "AGPL-3.0 only" | ||||||
|   | |||||||
 Submodule reminder-dashboard deleted from 8ba7a39ce5
									
								
							| @@ -1,6 +1,6 @@ | |||||||
| [package] | [package] | ||||||
| name = "reminder_web" | name = "reminder_web" | ||||||
| version = "0.1.3" | version = "0.1.4" | ||||||
| authors = ["jellywx <judesouthworth@pm.me>"] | authors = ["jellywx <judesouthworth@pm.me>"] | ||||||
| edition = "2018" | edition = "2018" | ||||||
|  |  | ||||||
|   | |||||||
| @@ -33,11 +33,9 @@ impl<'r> FromRequest<'r> for Transaction<'r> { | |||||||
|         match request.guard::<&State<Pool<Database>>>().await { |         match request.guard::<&State<Pool<Database>>>().await { | ||||||
|             Outcome::Success(pool) => match pool.begin().await { |             Outcome::Success(pool) => match pool.begin().await { | ||||||
|                 Ok(transaction) => Outcome::Success(Transaction(transaction)), |                 Ok(transaction) => Outcome::Success(Transaction(transaction)), | ||||||
|                 Err(e) => { |                 Err(e) => Outcome::Error((Status::InternalServerError, TransactionError::Error(e))), | ||||||
|                     Outcome::Failure((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), |             Outcome::Forward(f) => Outcome::Forward(f), | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -235,6 +235,21 @@ pub async fn edit_reminder( | |||||||
|                 ]); |                 ]); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |     } 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 { |     if reminder.channel > 0 { | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ use rocket::{ | |||||||
|     response::Redirect, |     response::Redirect, | ||||||
|     serde::json::json, |     serde::json::json, | ||||||
| }; | }; | ||||||
| use serde::{Deserialize, Deserializer, Serialize}; | use serde::{de, Deserialize, Deserializer, Serialize, Serializer}; | ||||||
| use serenity::{ | use serenity::{ | ||||||
|     client::Context, |     client::Context, | ||||||
|     http::Http, |     http::Http, | ||||||
| @@ -54,12 +54,27 @@ fn interval_default() -> Unset<Option<u32>> { | |||||||
|     None |     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 |     where | ||||||
|         D: Deserializer<'de>, |         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)] | #[derive(Serialize, Deserialize)] | ||||||
| @@ -70,7 +85,7 @@ pub struct ReminderTemplate { | |||||||
|     guild_id: u32, |     guild_id: u32, | ||||||
|     #[serde(default = "template_name_default")] |     #[serde(default = "template_name_default")] | ||||||
|     name: String, |     name: String, | ||||||
|     attachment: Option<Vec<u8>>, |     attachment: Option<Attachment>, | ||||||
|     attachment_name: Option<String>, |     attachment_name: Option<String>, | ||||||
|     avatar: Option<String>, |     avatar: Option<String>, | ||||||
|     content: String, |     content: String, | ||||||
| @@ -95,7 +110,7 @@ pub struct ReminderTemplate { | |||||||
| pub struct ReminderTemplateCsv { | pub struct ReminderTemplateCsv { | ||||||
|     #[serde(default = "template_name_default")] |     #[serde(default = "template_name_default")] | ||||||
|     name: String, |     name: String, | ||||||
|     attachment: Option<Vec<u8>>, |     attachment: Option<Attachment>, | ||||||
|     attachment_name: Option<String>, |     attachment_name: Option<String>, | ||||||
|     avatar: Option<String>, |     avatar: Option<String>, | ||||||
|     content: String, |     content: String, | ||||||
| @@ -130,8 +145,7 @@ pub struct EmbedField { | |||||||
|  |  | ||||||
| #[derive(Serialize, Deserialize)] | #[derive(Serialize, Deserialize)] | ||||||
| pub struct Reminder { | pub struct Reminder { | ||||||
|     #[serde(with = "base64s")] |     attachment: Option<Attachment>, | ||||||
|     attachment: Option<Vec<u8>>, |  | ||||||
|     attachment_name: Option<String>, |     attachment_name: Option<String>, | ||||||
|     avatar: Option<String>, |     avatar: Option<String>, | ||||||
|     #[serde(with = "string")] |     #[serde(with = "string")] | ||||||
| @@ -164,8 +178,7 @@ pub struct Reminder { | |||||||
|  |  | ||||||
| #[derive(Serialize, Deserialize)] | #[derive(Serialize, Deserialize)] | ||||||
| pub struct ReminderCsv { | pub struct ReminderCsv { | ||||||
|     #[serde(with = "base64s")] |     attachment: Option<Attachment>, | ||||||
|     attachment: Option<Vec<u8>>, |  | ||||||
|     attachment_name: Option<String>, |     attachment_name: Option<String>, | ||||||
|     avatar: Option<String>, |     avatar: Option<String>, | ||||||
|     channel: String, |     channel: String, | ||||||
| @@ -198,7 +211,7 @@ pub struct PatchReminder { | |||||||
|     uid: String, |     uid: String, | ||||||
|     #[serde(default)] |     #[serde(default)] | ||||||
|     #[serde(deserialize_with = "deserialize_optional_field")] |     #[serde(deserialize_with = "deserialize_optional_field")] | ||||||
|     attachment: Unset<Option<String>>, |     attachment: Unset<Option<Attachment>>, | ||||||
|     #[serde(default)] |     #[serde(default)] | ||||||
|     #[serde(deserialize_with = "deserialize_optional_field")] |     #[serde(deserialize_with = "deserialize_optional_field")] | ||||||
|     attachment_name: Unset<Option<String>>, |     attachment_name: Unset<Option<String>>, | ||||||
| @@ -294,6 +307,14 @@ pub fn generate_uid() -> String { | |||||||
|         .join("") |         .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 | // https://github.com/serde-rs/json/issues/329#issuecomment-305608405 | ||||||
| mod string { | mod string { | ||||||
|     use std::{fmt::Display, str::FromStr}; |     use std::{fmt::Display, str::FromStr}; | ||||||
| @@ -318,29 +339,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)] | #[derive(Deserialize)] | ||||||
| pub struct DeleteReminder { | pub struct DeleteReminder { | ||||||
|     uid: String, |     uid: String, | ||||||
|   | |||||||
| @@ -31,22 +31,20 @@ pub async fn discord_login( | |||||||
|  |  | ||||||
|     // store the pkce secret to verify the authorization later |     // store the pkce secret to verify the authorization later | ||||||
|     cookies.add_private( |     cookies.add_private( | ||||||
|         Cookie::build("verify", pkce_verifier.secret().to_string()) |         Cookie::build(("verify", pkce_verifier.secret().to_string())) | ||||||
|             .http_only(true) |             .http_only(true) | ||||||
|             .path("/login") |             .path("/login") | ||||||
|             .same_site(SameSite::Lax) |             .same_site(SameSite::Lax) | ||||||
|             .expires(Expiration::Session) |             .expires(Expiration::Session), | ||||||
|             .finish(), |  | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
|     // store the csrf token to verify no interference |     // store the csrf token to verify no interference | ||||||
|     cookies.add_private( |     cookies.add_private( | ||||||
|         Cookie::build("csrf", csrf_token.secret().to_string()) |         Cookie::build(("csrf", csrf_token.secret().to_string())) | ||||||
|             .http_only(true) |             .http_only(true) | ||||||
|             .path("/login") |             .path("/login") | ||||||
|             .same_site(SameSite::Lax) |             .same_site(SameSite::Lax) | ||||||
|             .expires(Expiration::Session) |             .expires(Expiration::Session), | ||||||
|             .finish(), |  | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
|     Redirect::to(auth_url.to_string()) |     Redirect::to(auth_url.to_string()) | ||||||
| @@ -54,9 +52,9 @@ pub async fn discord_login( | |||||||
|  |  | ||||||
| #[get("/discord/logout")] | #[get("/discord/logout")] | ||||||
| pub async fn discord_logout(cookies: &CookieJar<'_>) -> Redirect { | pub async fn discord_logout(cookies: &CookieJar<'_>) -> Redirect { | ||||||
|     cookies.remove_private(Cookie::named("username")); |     cookies.remove_private(Cookie::from("username")); | ||||||
|     cookies.remove_private(Cookie::named("userid")); |     cookies.remove_private(Cookie::from("userid")); | ||||||
|     cookies.remove_private(Cookie::named("access_token")); |     cookies.remove_private(Cookie::from("access_token")); | ||||||
|  |  | ||||||
|     Redirect::to(uri!(routes::index)) |     Redirect::to(uri!(routes::index)) | ||||||
| } | } | ||||||
| @@ -80,17 +78,16 @@ pub async fn discord_callback( | |||||||
|                 .request_async(async_http_client) |                 .request_async(async_http_client) | ||||||
|                 .await; |                 .await; | ||||||
|  |  | ||||||
|             cookies.remove_private(Cookie::named("verify")); |             cookies.remove_private(Cookie::from("verify")); | ||||||
|             cookies.remove_private(Cookie::named("csrf")); |             cookies.remove_private(Cookie::from("csrf")); | ||||||
|  |  | ||||||
|             match token_result { |             match token_result { | ||||||
|                 Ok(token) => { |                 Ok(token) => { | ||||||
|                     cookies.add_private( |                     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) |                             .secure(true) | ||||||
|                             .http_only(true) |                             .http_only(true) | ||||||
|                             .path("/dashboard") |                             .path("/dashboard"), | ||||||
|                             .finish(), |  | ||||||
|                     ); |                     ); | ||||||
|  |  | ||||||
|                     let request_res = reqwest_client |                     let request_res = reqwest_client | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user