redlib/src/user.rs

138 lines
4.6 KiB
Rust
Raw Normal View History

2020-10-25 13:25:59 -07:00
// CRATES
2021-03-17 15:30:33 -07:00
use crate::client::json;
use crate::server::RequestExt;
use crate::utils::{error, filter_posts, format_url, get_filters, nsfw_landing, param, setting, template, Post, Preferences, User};
2020-10-25 13:25:59 -07:00
use askama::Template;
2021-03-17 15:30:33 -07:00
use hyper::{Body, Request, Response};
2022-05-20 19:20:44 -07:00
use time::{macros::format_description, OffsetDateTime};
2020-11-17 11:37:40 -08:00
2020-10-25 13:25:59 -07:00
// STRUCTS
#[derive(Template)]
2022-05-20 23:28:31 -06:00
#[template(path = "user.html")]
2020-10-25 13:25:59 -07:00
struct UserTemplate {
user: User,
posts: Vec<Post>,
2020-12-29 17:11:47 -08:00
sort: (String, String),
2020-12-27 12:36:10 -08:00
ends: (String, String),
/// "overview", "comments", or "submitted"
listing: String,
2021-01-08 17:35:04 -08:00
prefs: Preferences,
url: String,
redirect_url: String,
/// Whether the user themself is filtered.
is_filtered: bool,
/// Whether all fetched posts are filtered (to differentiate between no posts fetched in the first place,
/// and all fetched posts being filtered).
all_posts_filtered: bool,
/// Whether all posts were hidden because they are NSFW (and user has disabled show NSFW)
all_posts_hidden_nsfw: bool,
no_posts: bool,
2020-10-25 13:25:59 -07:00
}
2021-01-02 20:50:23 -08:00
// FUNCTIONS
2021-03-17 15:30:33 -07:00
pub async fn profile(req: Request<Body>) -> Result<Response<Body>, String> {
let listing = req.param("listing").unwrap_or_else(|| "overview".to_string());
2020-12-31 15:54:13 -08:00
// Build the Reddit JSON API path
2021-03-17 21:40:55 -07:00
let path = format!(
2024-01-19 20:16:17 -05:00
"/user/{}/{listing}.json?{}&raw_json=1",
2021-03-26 20:00:47 -07:00
req.param("name").unwrap_or_else(|| "reddit".to_string()),
req.uri().query().unwrap_or_default(),
2021-03-17 21:40:55 -07:00
);
let url = String::from(req.uri().path_and_query().map_or("", |val| val.as_str()));
let redirect_url = url[1..].replace('?', "%3F").replace('&', "%26");
2020-12-27 12:36:10 -08:00
2023-12-26 18:25:52 -05:00
// Retrieve other variables from Redlib request
let sort = param(&path, "sort").unwrap_or_default();
2021-03-17 15:30:33 -07:00
let username = req.param("name").unwrap_or_default();
// Retrieve info from user about page.
let user = user(&username).await.unwrap_or_default();
2020-11-20 22:05:27 -08:00
let req_url = req.uri().to_string();
// Return landing page if this post if this Reddit deems this user NSFW,
// but we have also disabled the display of NSFW content or if the instance
// is SFW-only.
if user.nsfw && crate::utils::should_be_nsfw_gated(&req, &req_url) {
return Ok(nsfw_landing(req, req_url).await.unwrap_or_default());
}
let filters = get_filters(&req);
if filters.contains(&["u_", &username].concat()) {
2024-01-19 20:16:17 -05:00
Ok(template(&UserTemplate {
user,
posts: Vec::new(),
sort: (sort, param(&path, "t").unwrap_or_default()),
2024-01-19 20:16:17 -05:00
ends: (param(&path, "after").unwrap_or_default(), String::new()),
listing,
2023-01-01 21:39:38 -05:00
prefs: Preferences::new(&req),
url,
redirect_url,
is_filtered: true,
all_posts_filtered: false,
all_posts_hidden_nsfw: false,
no_posts: false,
2024-01-19 20:16:17 -05:00
}))
} else {
// Request user posts/comments from Reddit
match Post::fetch(&path, false).await {
Ok((mut posts, after)) => {
2022-11-09 09:16:51 -07:00
let (_, all_posts_filtered) = filter_posts(&mut posts, &filters);
let no_posts = posts.is_empty();
let all_posts_hidden_nsfw = !no_posts && (posts.iter().all(|p| p.flags.nsfw) && setting(&req, "show_nsfw") != "on");
2024-01-19 20:16:17 -05:00
Ok(template(&UserTemplate {
user,
posts,
sort: (sort, param(&path, "t").unwrap_or_default()),
ends: (param(&path, "after").unwrap_or_default(), after),
listing,
2023-01-01 21:39:38 -05:00
prefs: Preferences::new(&req),
url,
redirect_url,
is_filtered: false,
all_posts_filtered,
all_posts_hidden_nsfw,
no_posts,
2024-01-19 20:16:17 -05:00
}))
}
// If there is an error show error page
2024-01-19 20:16:17 -05:00
Err(msg) => error(req, &msg).await,
2021-01-01 22:21:43 -08:00
}
2020-11-19 20:42:18 -08:00
}
2020-10-25 13:25:59 -07:00
}
// USER
2021-01-14 09:53:54 -08:00
async fn user(name: &str) -> Result<User, String> {
// Build the Reddit JSON API path
2024-01-19 20:16:17 -05:00
let path: String = format!("/user/{name}/about.json?raw_json=1");
2020-12-31 21:03:44 -08:00
2021-01-01 15:28:13 -08:00
// Send a request to the url
2021-05-20 12:24:06 -07:00
json(path, false).await.map(|res| {
// Grab creation date as unix timestamp
let created_unix = res["data"]["created"].as_f64().unwrap_or(0.0).round() as i64;
let created = OffsetDateTime::from_unix_timestamp(created_unix).unwrap_or(OffsetDateTime::UNIX_EPOCH);
2021-05-20 12:24:06 -07:00
// Closure used to parse JSON from Reddit APIs
let about = |item| res["data"]["subreddit"][item].as_str().unwrap_or_default().to_string();
2021-01-08 17:35:04 -08:00
2021-05-20 12:24:06 -07:00
// Parse the JSON output into a User struct
User {
name: res["data"]["name"].as_str().unwrap_or(name).to_owned(),
2022-05-20 23:28:31 -06:00
title: about("title"),
2021-05-20 12:24:06 -07:00
icon: format_url(&about("icon_img")),
karma: res["data"]["total_karma"].as_i64().unwrap_or(0),
2021-12-29 12:48:57 -08:00
created: created.format(format_description!("[month repr:short] [day] '[year repr:last_two]")).unwrap_or_default(),
2022-05-20 23:28:31 -06:00
banner: about("banner_img"),
2021-05-20 12:24:06 -07:00
description: about("public_description"),
nsfw: res["data"]["subreddit"]["over_18"].as_bool().unwrap_or_default(),
2021-01-01 22:21:43 -08:00
}
2021-05-20 12:24:06 -07:00
})
2020-11-29 18:50:29 -08:00
}
#[tokio::test(flavor = "multi_thread")]
async fn test_fetching_user() {
let user = user("spez").await;
assert!(user.is_ok());
assert!(user.unwrap().karma > 100);
}