use std::collections::HashMap; // CRATES use crate::server::ResponseExt; use crate::utils::{redirect, template, Preferences}; use askama::Template; use cookie::Cookie; use futures_lite::StreamExt; use hyper::{Body, Request, Response}; use time::{Duration, OffsetDateTime}; // STRUCTS #[derive(Template)] #[template(path = "settings.html")] struct SettingsTemplate { prefs: Preferences, } #[derive(serde::Deserialize, Default, Debug)] #[serde(default)] pub struct Form { theme: Option, front_page: Option, layout: Option, wide: Option, comment_sort: Option, show_nsfw: Option, redirect: Option, subscriptions: Option, } // FUNCTIONS // Retrieve cookies from request "Cookie" header pub async fn get(req: Request) -> Result, String> { template(SettingsTemplate { prefs: Preferences::new(req) }) } // Set cookies using response "Set-Cookie" header pub async fn set(req: Request) -> Result, String> { // Split the body into parts let (parts, mut body) = req.into_parts(); // Grab existing cookies let mut cookies = Vec::new(); for header in parts.headers.get_all("Cookie") { if let Ok(cookie) = Cookie::parse(header.to_str().unwrap_or_default()) { cookies.push(cookie); } } // Aggregate the body... // let whole_body = hyper::body::aggregate(req).await.map_err(|e| e.to_string())?; let body_bytes = body .try_fold(Vec::new(), |mut data, chunk| { data.extend_from_slice(&chunk); Ok(data) }) .await .map_err(|e| e.to_string())?; let form = url::form_urlencoded::parse(&body_bytes).collect::>(); let mut res = redirect("/settings".to_string()); let names = vec!["theme", "front_page", "layout", "wide", "comment_sort", "show_nsfw", "subscriptions"]; for name in names { match form.get(name) { Some(value) => res.insert_cookie( Cookie::build(name.to_owned(), value.to_owned()) .path("/") .http_only(true) .expires(OffsetDateTime::now_utc() + Duration::weeks(52)) .finish(), ), None => res.remove_cookie(name.to_string()), }; } Ok(res) } // Set cookies using response "Set-Cookie" header pub async fn restore(req: Request) -> Result, String> { // Split the body into parts let (parts, _) = req.into_parts(); // Grab existing cookies let mut cookies = Vec::new(); for header in parts.headers.get_all("Cookie") { if let Ok(cookie) = Cookie::parse(header.to_str().unwrap_or_default()) { cookies.push(cookie); } } let query = parts.uri.query().unwrap_or_default().as_bytes(); let form = url::form_urlencoded::parse(query).collect::>(); let names = vec!["theme", "front_page", "layout", "wide", "comment_sort", "show_nsfw", "subscriptions"]; let path = match form.get("redirect") { Some(value) => format!("/{}/", value), None => "/".to_string(), }; let mut res = redirect(path); for name in names { match form.get(name) { Some(value) => res.insert_cookie( Cookie::build(name.to_owned(), value.to_owned()) .path("/") .http_only(true) .expires(OffsetDateTime::now_utc() + Duration::weeks(52)) .finish(), ), None => res.remove_cookie(name.to_string()), }; } Ok(res) }