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, } // CONSTANTS const PREFS: [&str; 10] = [ "theme", "front_page", "layout", "wide", "comment_sort", "post_sort", "show_nsfw", "use_hls", "hide_hls_notification", "subscriptions", ]; // 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()); for &name in &PREFS { 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) } fn set_cookies_method(req: Request, remove_cookies: bool) -> Response { // 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 path = match form.get("redirect") { Some(value) => format!("/{}", value.replace("%26", "&").replace("%23", "#")), None => "/".to_string(), }; let mut res = redirect(path); for &name in &PREFS { 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 => { if remove_cookies { res.remove_cookie(name.to_string()) } } }; } res } // Set cookies using response "Set-Cookie" header pub async fn restore(req: Request) -> Result, String> { Ok(set_cookies_method(req, true)) } pub async fn update(req: Request) -> Result, String> { Ok(set_cookies_method(req, false)) }