206 lines
5.7 KiB
Rust
206 lines
5.7 KiB
Rust
use std::sync::{Arc, Mutex};
|
|
|
|
use axum::{
|
|
http::{Response, StatusCode},
|
|
response::{IntoResponse, Redirect},
|
|
routing::{get, post, Router},
|
|
Extension,
|
|
};
|
|
use rand::{seq::SliceRandom, thread_rng, RngCore, SeedableRng};
|
|
use rand_chacha::ChaCha8Rng;
|
|
use rand_core::OsRng;
|
|
use sqlx::PgPool;
|
|
use std::error::Error;
|
|
use tower_http::services::ServeDir;
|
|
|
|
use crate::{
|
|
auth::{auth, logout_response, post_login, post_signup},
|
|
database::{
|
|
link::{Link, LinkType},
|
|
PsqlData,
|
|
},
|
|
};
|
|
|
|
use super::{
|
|
admin,
|
|
posts::{self, get_articles_date_sorted},
|
|
projects,
|
|
templates::{
|
|
AboutTemplate, AiTemplate, BlogrollTemplate, BooksTemplate, ContactTemplate, CookingTemplate, CreationTemplate, GiftsTemplate, HomeTemplate, HtmlTemplate, InterestsTemplate, LinksPageTemplate, LoginTemplate, MoneyTemplate, NowTemplate, ResumeTemplate, SignupTemplate, TechnologyTemplate, TimeTemplate, UsesTemplate, WorkTemplate
|
|
},
|
|
};
|
|
|
|
pub fn get_router(pool: PgPool) -> Router {
|
|
let assets_path = std::env::current_dir().unwrap();
|
|
|
|
let random = ChaCha8Rng::seed_from_u64(OsRng.next_u64());
|
|
let middleware_database = pool.clone();
|
|
|
|
Router::new()
|
|
.nest("/posts", posts::get_router())
|
|
.nest("/projects", projects::get_router())
|
|
.nest("/admin", admin::get_router())
|
|
.nest_service(
|
|
"/assets",
|
|
ServeDir::new(format!("{}/assets", assets_path.to_str().unwrap())),
|
|
)
|
|
.route("/", get(home))
|
|
.route("/now", get(now))
|
|
.route("/about", get(about))
|
|
.route("/contact", get(contact))
|
|
.route("/uses", get(uses))
|
|
.route("/ai", get(ai))
|
|
.route("/blogroll", get(blogroll))
|
|
.route("/links", get(links))
|
|
.route("/interests", get(interests))
|
|
.route("/resume", get(resume))
|
|
.route("/gifts", get(gifts))
|
|
.route("/hire", get(work))
|
|
.route("/books", get(books))
|
|
.route("/time", get(time))
|
|
.route("/cooking", get(cooking))
|
|
.route("/creation", get(creation))
|
|
.route("/technology", get(technology))
|
|
.route("/money", get(money))
|
|
.route("/login", get(get_login).post(post_login))
|
|
.route("/signup", get(get_signup).post(post_signup))
|
|
.route("/logout", post(logout_response))
|
|
.route(
|
|
"/robots.txt",
|
|
get(|| async { Redirect::permanent("/assets/robots.txt") }),
|
|
)
|
|
.layer(axum::middleware::from_fn(move |req, next| {
|
|
auth(req, next, middleware_database.clone())
|
|
}))
|
|
.layer(Extension(pool))
|
|
.layer(Extension(Arc::new(Mutex::new(random))))
|
|
}
|
|
|
|
async fn home(Extension(pool): Extension<PgPool>) -> impl IntoResponse {
|
|
let mut articles = get_articles_date_sorted(&pool).await.unwrap();
|
|
articles.truncate(5);
|
|
HtmlTemplate(HomeTemplate {
|
|
recent_articles: articles,
|
|
})
|
|
}
|
|
|
|
async fn now() -> impl IntoResponse {
|
|
HtmlTemplate(NowTemplate {})
|
|
}
|
|
|
|
async fn about() -> impl IntoResponse {
|
|
HtmlTemplate(AboutTemplate {})
|
|
}
|
|
|
|
async fn contact() -> impl IntoResponse {
|
|
HtmlTemplate(ContactTemplate {})
|
|
}
|
|
|
|
async fn uses() -> impl IntoResponse {
|
|
HtmlTemplate(UsesTemplate {})
|
|
}
|
|
|
|
async fn ai() -> impl IntoResponse {
|
|
HtmlTemplate(AiTemplate {})
|
|
}
|
|
|
|
async fn interests() -> impl IntoResponse {
|
|
HtmlTemplate(InterestsTemplate {})
|
|
}
|
|
|
|
async fn resume() -> impl IntoResponse {
|
|
HtmlTemplate(ResumeTemplate {})
|
|
}
|
|
|
|
async fn work() -> impl IntoResponse {
|
|
HtmlTemplate(WorkTemplate {})
|
|
}
|
|
|
|
async fn gifts() -> impl IntoResponse {
|
|
HtmlTemplate(GiftsTemplate {})
|
|
}
|
|
|
|
async fn blogroll(Extension(pool): Extension<PgPool>) -> impl IntoResponse {
|
|
let blogroll_page = BlogrollTemplate {
|
|
blogs: match get_links_as_list(&pool, LinkType::BLOG).await {
|
|
Ok(list) => list,
|
|
Err(_) => Vec::new(),
|
|
},
|
|
};
|
|
HtmlTemplate(blogroll_page)
|
|
}
|
|
|
|
async fn links(Extension(pool): Extension<PgPool>) -> impl IntoResponse {
|
|
let links_page = LinksPageTemplate {
|
|
articles: match get_links_as_list(&pool, LinkType::ARTICLE).await {
|
|
Ok(list) => list,
|
|
Err(_) => Vec::new(),
|
|
},
|
|
};
|
|
HtmlTemplate(links_page)
|
|
}
|
|
|
|
pub async fn get_links_as_list(
|
|
pool: &PgPool,
|
|
link_type: LinkType,
|
|
) -> Result<Vec<Link>, Box<dyn Error>> {
|
|
let mut links: Vec<Link> = match Link::read_all(pool).await {
|
|
Ok(a) => a.iter().map(|x| *x.clone()).collect(),
|
|
Err(_) => Vec::new(),
|
|
};
|
|
|
|
let mut rng = thread_rng();
|
|
links.shuffle(&mut rng);
|
|
|
|
let list: Vec<Link> = links
|
|
.into_iter()
|
|
.filter(|link| link.link_type == link_type)
|
|
.collect();
|
|
Ok(list)
|
|
}
|
|
|
|
pub fn error_page(err: &dyn std::error::Error) -> impl IntoResponse {
|
|
Response::builder()
|
|
.status(StatusCode::INTERNAL_SERVER_ERROR)
|
|
.body(format!("Err: {}", err))
|
|
.unwrap()
|
|
}
|
|
|
|
pub async fn get_login() -> impl IntoResponse {
|
|
HtmlTemplate(LoginTemplate { username: None })
|
|
}
|
|
|
|
pub async fn get_signup() -> impl IntoResponse {
|
|
HtmlTemplate(SignupTemplate {})
|
|
}
|
|
|
|
async fn books() -> impl IntoResponse {
|
|
let books_page = BooksTemplate {};
|
|
HtmlTemplate(books_page)
|
|
}
|
|
|
|
async fn time() -> impl IntoResponse {
|
|
let time_page = TimeTemplate {};
|
|
HtmlTemplate(time_page)
|
|
}
|
|
|
|
async fn cooking() -> impl IntoResponse {
|
|
let cooking_page = CookingTemplate {};
|
|
HtmlTemplate(cooking_page)
|
|
}
|
|
|
|
async fn creation() -> impl IntoResponse {
|
|
let creation_page = CreationTemplate {};
|
|
HtmlTemplate(creation_page)
|
|
}
|
|
|
|
async fn technology() -> impl IntoResponse {
|
|
let technology_page = TechnologyTemplate {};
|
|
HtmlTemplate(technology_page)
|
|
}
|
|
|
|
async fn money() -> impl IntoResponse {
|
|
let money_page = MoneyTemplate {};
|
|
HtmlTemplate(money_page)
|
|
}
|