Add pages to subreddits
This commit is contained in:
parent
0054557c86
commit
7a176c6804
@ -16,21 +16,37 @@ use utils::Params;
|
|||||||
struct PopularTemplate {
|
struct PopularTemplate {
|
||||||
posts: Vec<Post>,
|
posts: Vec<Post>,
|
||||||
sort: String,
|
sort: String,
|
||||||
|
ends: (String, String),
|
||||||
}
|
}
|
||||||
|
|
||||||
// RENDER
|
// RENDER
|
||||||
async fn render(sub_name: String, sort: String) -> Result<HttpResponse> {
|
async fn render(sub_name: String, sort: Option<String>, ends: (Option<String>, Option<String>)) -> Result<HttpResponse> {
|
||||||
let posts: Vec<Post> = posts(sub_name, &sort).await?;
|
let sorting = sort.unwrap_or("hot".to_string());
|
||||||
|
let before = ends.1.clone().unwrap_or(String::new()); // If there is an after, there must be a before
|
||||||
|
|
||||||
let s = PopularTemplate { posts: posts, sort: sort }.render().unwrap();
|
// Build the Reddit JSON API url
|
||||||
|
let url = match ends.0 {
|
||||||
|
Some(val) => format!("https://www.reddit.com/r/{}/{}.json?before={}&count=25", sub_name, sorting, val),
|
||||||
|
None => match ends.1 {
|
||||||
|
Some(val) => format!("https://www.reddit.com/r/{}/{}.json?after={}&count=25", sub_name, sorting, val),
|
||||||
|
None => format!("https://www.reddit.com/r/{}/{}.json", sub_name, sorting),
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
let items = posts(url).await?;
|
||||||
|
|
||||||
|
let s = PopularTemplate {
|
||||||
|
posts: items.0,
|
||||||
|
sort: sorting,
|
||||||
|
ends: (before, items.1),
|
||||||
|
}
|
||||||
|
.render()
|
||||||
|
.unwrap();
|
||||||
Ok(HttpResponse::Ok().content_type("text/html").body(s))
|
Ok(HttpResponse::Ok().content_type("text/html").body(s))
|
||||||
}
|
}
|
||||||
|
|
||||||
// SERVICES
|
// SERVICES
|
||||||
#[get("/")]
|
#[get("/")]
|
||||||
pub async fn page(params: web::Query<Params>) -> Result<HttpResponse> {
|
pub async fn page(params: web::Query<Params>) -> Result<HttpResponse> {
|
||||||
match ¶ms.sort {
|
render("popular".to_string(), params.sort.clone(), (params.before.clone(), params.after.clone())).await
|
||||||
Some(sort) => render("popular".to_string(), sort.to_string()).await,
|
|
||||||
None => render("popular".to_string(), "hot".to_string()).await,
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@ -6,7 +6,7 @@ use pulldown_cmark::{html, Options, Parser};
|
|||||||
|
|
||||||
#[path = "utils.rs"]
|
#[path = "utils.rs"]
|
||||||
mod utils;
|
mod utils;
|
||||||
use utils::{val, Comment, Flair, Params, Post, request};
|
use utils::{request, val, Comment, Flair, Params, Post};
|
||||||
|
|
||||||
// STRUCTS
|
// STRUCTS
|
||||||
#[derive(Template)]
|
#[derive(Template)]
|
||||||
|
@ -5,7 +5,7 @@ use chrono::{TimeZone, Utc};
|
|||||||
|
|
||||||
#[path = "utils.rs"]
|
#[path = "utils.rs"]
|
||||||
mod utils;
|
mod utils;
|
||||||
pub use utils::{val, Flair, Params, Post, Subreddit, request};
|
pub use utils::{request, val, Flair, Params, Post, Subreddit};
|
||||||
|
|
||||||
// STRUCTS
|
// STRUCTS
|
||||||
#[derive(Template)]
|
#[derive(Template)]
|
||||||
@ -14,11 +14,31 @@ struct SubredditTemplate {
|
|||||||
sub: Subreddit,
|
sub: Subreddit,
|
||||||
posts: Vec<Post>,
|
posts: Vec<Post>,
|
||||||
sort: String,
|
sort: String,
|
||||||
|
ends: (String, String),
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn render(sub_name: String, sort: String) -> Result<HttpResponse> {
|
// SERVICES
|
||||||
|
#[allow(dead_code)]
|
||||||
|
#[get("/r/{sub}")]
|
||||||
|
async fn page(web::Path(sub): web::Path<String>, params: web::Query<Params>) -> Result<HttpResponse> {
|
||||||
|
render(sub, params.sort.clone(), (params.before.clone(), params.after.clone())).await
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn render(sub_name: String, sort: Option<String>, ends: (Option<String>, Option<String>)) -> Result<HttpResponse> {
|
||||||
|
let sorting = sort.unwrap_or("hot".to_string());
|
||||||
|
let before = ends.1.clone().unwrap_or(String::new()); // If there is an after, there must be a before
|
||||||
|
|
||||||
|
// Build the Reddit JSON API url
|
||||||
|
let url = match ends.0 {
|
||||||
|
Some(val) => format!("https://www.reddit.com/r/{}/{}.json?before={}&count=25", sub_name, sorting, val),
|
||||||
|
None => match ends.1 {
|
||||||
|
Some(val) => format!("https://www.reddit.com/r/{}/{}.json?after={}&count=25", sub_name, sorting, val),
|
||||||
|
None => format!("https://www.reddit.com/r/{}/{}.json", sub_name, sorting),
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
let mut sub: Subreddit = subreddit(&sub_name).await?;
|
let mut sub: Subreddit = subreddit(&sub_name).await?;
|
||||||
let posts: Vec<Post> = posts(sub_name, &sort).await?;
|
let items = posts(url).await?;
|
||||||
|
|
||||||
sub.icon = if sub.icon != "" {
|
sub.icon = if sub.icon != "" {
|
||||||
format!(r#"<img class="subreddit_icon" src="{}">"#, sub.icon)
|
format!(r#"<img class="subreddit_icon" src="{}">"#, sub.icon)
|
||||||
@ -26,18 +46,15 @@ async fn render(sub_name: String, sort: String) -> Result<HttpResponse> {
|
|||||||
String::new()
|
String::new()
|
||||||
};
|
};
|
||||||
|
|
||||||
let s = SubredditTemplate { sub: sub, posts: posts, sort: sort }.render().unwrap();
|
let s = SubredditTemplate {
|
||||||
Ok(HttpResponse::Ok().content_type("text/html").body(s))
|
sub: sub,
|
||||||
}
|
posts: items.0,
|
||||||
|
sort: sorting,
|
||||||
// SERVICES
|
ends: (before, items.1),
|
||||||
#[allow(dead_code)]
|
|
||||||
#[get("/r/{sub}")]
|
|
||||||
async fn page(web::Path(sub): web::Path<String>, params: web::Query<Params>) -> Result<HttpResponse> {
|
|
||||||
match ¶ms.sort {
|
|
||||||
Some(sort) => render(sub, sort.to_string()).await,
|
|
||||||
None => render(sub, "hot".to_string()).await,
|
|
||||||
}
|
}
|
||||||
|
.render()
|
||||||
|
.unwrap();
|
||||||
|
Ok(HttpResponse::Ok().content_type("text/html").body(s))
|
||||||
}
|
}
|
||||||
|
|
||||||
// SUBREDDIT
|
// SUBREDDIT
|
||||||
@ -63,10 +80,7 @@ async fn subreddit(sub: &String) -> Result<Subreddit> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// POSTS
|
// POSTS
|
||||||
pub async fn posts(sub: String, sort: &String) -> Result<Vec<Post>> {
|
pub async fn posts(url: String) -> Result<(Vec<Post>, String)> {
|
||||||
// Build the Reddit JSON API url
|
|
||||||
let url: String = format!("https://www.reddit.com/r/{}/{}.json", sub, sort);
|
|
||||||
|
|
||||||
// Send a request to the url, receive JSON in response
|
// Send a request to the url, receive JSON in response
|
||||||
let res = request(url).await;
|
let res = request(url).await;
|
||||||
|
|
||||||
@ -103,5 +117,6 @@ pub async fn posts(sub: String, sort: &String) -> Result<Vec<Post>> {
|
|||||||
),
|
),
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
Ok(posts)
|
|
||||||
|
Ok((posts, res["data"]["after"].as_str().unwrap_or("").to_string()))
|
||||||
}
|
}
|
||||||
|
@ -5,7 +5,7 @@ use chrono::{TimeZone, Utc};
|
|||||||
|
|
||||||
#[path = "utils.rs"]
|
#[path = "utils.rs"]
|
||||||
mod utils;
|
mod utils;
|
||||||
use utils::{nested_val, val, Flair, Params, Post, User, request};
|
use utils::{nested_val, request, val, Flair, Params, Post, User};
|
||||||
|
|
||||||
// STRUCTS
|
// STRUCTS
|
||||||
#[derive(Template)]
|
#[derive(Template)]
|
||||||
@ -37,7 +37,7 @@ async fn page(web::Path(username): web::Path<String>, params: web::Query<Params>
|
|||||||
async fn user(name: &String) -> User {
|
async fn user(name: &String) -> User {
|
||||||
// Build the Reddit JSON API url
|
// Build the Reddit JSON API url
|
||||||
let url: String = format!("https://www.reddit.com/user/{}/about.json", name);
|
let url: String = format!("https://www.reddit.com/user/{}/about.json", name);
|
||||||
|
|
||||||
// Send a request to the url, receive JSON in response
|
// Send a request to the url, receive JSON in response
|
||||||
let res = request(url).await;
|
let res = request(url).await;
|
||||||
|
|
||||||
|
17
src/utils.rs
17
src/utils.rs
@ -44,6 +44,14 @@ pub struct Subreddit {
|
|||||||
pub icon: String,
|
pub icon: String,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Parser for query params, used in sorting (eg. /r/rust/?sort=hot)
|
||||||
|
#[derive(serde::Deserialize)]
|
||||||
|
pub struct Params {
|
||||||
|
pub sort: Option<String>,
|
||||||
|
pub after: Option<String>,
|
||||||
|
pub before: Option<String>,
|
||||||
|
}
|
||||||
|
|
||||||
#[allow(dead_code)]
|
#[allow(dead_code)]
|
||||||
// val() function used to parse JSON from Reddit APIs
|
// val() function used to parse JSON from Reddit APIs
|
||||||
pub async fn val(j: &serde_json::Value, k: &str) -> String {
|
pub async fn val(j: &serde_json::Value, k: &str) -> String {
|
||||||
@ -56,16 +64,9 @@ pub async fn nested_val(j: &serde_json::Value, n: &str, k: &str) -> String {
|
|||||||
String::from(j["data"][n][k].as_str().unwrap())
|
String::from(j["data"][n][k].as_str().unwrap())
|
||||||
}
|
}
|
||||||
|
|
||||||
// Parser for query params, used in sorting (eg. /r/rust/?sort=hot)
|
|
||||||
#[derive(serde::Deserialize)]
|
|
||||||
pub struct Params {
|
|
||||||
pub sort: Option<String>,
|
|
||||||
}
|
|
||||||
|
|
||||||
// Make a request to a Reddit API and parse the JSON response
|
// Make a request to a Reddit API and parse the JSON response
|
||||||
#[allow(dead_code)]
|
#[allow(dead_code)]
|
||||||
pub async fn request(url: String) -> serde_json::Value {
|
pub async fn request(url: String) -> serde_json::Value {
|
||||||
|
|
||||||
// --- actix-web::client ---
|
// --- actix-web::client ---
|
||||||
// let client = actix_web::client::Client::default();
|
// let client = actix_web::client::Client::default();
|
||||||
// let res = client
|
// let res = client
|
||||||
@ -89,6 +90,6 @@ pub async fn request(url: String) -> serde_json::Value {
|
|||||||
|
|
||||||
// Parse the response from Reddit as JSON
|
// Parse the response from Reddit as JSON
|
||||||
let json: serde_json::Value = serde_json::from_str(resp.as_str()).expect("Failed to parse JSON");
|
let json: serde_json::Value = serde_json::from_str(resp.as_str()).expect("Failed to parse JSON");
|
||||||
|
|
||||||
json
|
json
|
||||||
}
|
}
|
||||||
|
@ -29,6 +29,11 @@ main {
|
|||||||
padding: 0px 10px;
|
padding: 0px 10px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
footer {
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
}
|
||||||
|
|
||||||
button {
|
button {
|
||||||
background: none;
|
background: none;
|
||||||
border: none;
|
border: none;
|
||||||
@ -109,7 +114,7 @@ span {
|
|||||||
padding: 0px 10px;
|
padding: 0px 10px;
|
||||||
}
|
}
|
||||||
|
|
||||||
#sort > div {
|
#sort > div, footer > a {
|
||||||
background: #151515;
|
background: #151515;
|
||||||
color: lightgrey;
|
color: lightgrey;
|
||||||
border-radius: 5px;
|
border-radius: 5px;
|
||||||
@ -117,7 +122,6 @@ span {
|
|||||||
padding: 10px 20px;
|
padding: 10px 20px;
|
||||||
text-align: center;
|
text-align: center;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
font-weight: bold;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#sort > div:hover {
|
#sort > div:hover {
|
||||||
|
@ -47,6 +47,16 @@
|
|||||||
<img class="post_thumbnail" src="{{ post.media }}">
|
<img class="post_thumbnail" src="{{ post.media }}">
|
||||||
</div><br>
|
</div><br>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
|
|
||||||
|
<footer>
|
||||||
|
{% if ends.0 != "" %}
|
||||||
|
<a href="?before={{ ends.0 }}">PREV</a>
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
{% if ends.1 != "" %}
|
||||||
|
<a href="?after={{ ends.1 }}">NEXT</a>
|
||||||
|
{% endif %}
|
||||||
|
</footer>
|
||||||
</main>
|
</main>
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
@ -58,6 +58,16 @@
|
|||||||
<img class="post_thumbnail" src="{{ post.media }}">
|
<img class="post_thumbnail" src="{{ post.media }}">
|
||||||
</div><br>
|
</div><br>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
|
|
||||||
|
<footer>
|
||||||
|
{% if ends.0 != "" %}
|
||||||
|
<a href="?before={{ ends.0 }}">PREV</a>
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
{% if ends.1 != "" %}
|
||||||
|
<a href="?after={{ ends.1 }}">NEXT</a>
|
||||||
|
{% endif %}
|
||||||
|
</footer>
|
||||||
</main>
|
</main>
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
Loading…
Reference in New Issue
Block a user