2020-07-16 19:32:30 +00:00
|
|
|
use super::{PostNotFound, SeriesNotFound};
|
|
|
|
use crate::{
|
|
|
|
app::State,
|
|
|
|
post::Post,
|
|
|
|
templates::{self, Html, RenderRucte},
|
|
|
|
};
|
|
|
|
use lazy_static::lazy_static;
|
2020-10-02 22:36:57 +00:00
|
|
|
use prometheus::{opts, register_int_counter_vec, IntCounterVec};
|
2020-07-16 19:32:30 +00:00
|
|
|
use std::sync::Arc;
|
2020-10-02 22:36:57 +00:00
|
|
|
use tracing::{error, instrument};
|
2020-07-16 19:32:30 +00:00
|
|
|
use warp::{http::Response, Rejection, Reply};
|
|
|
|
|
|
|
|
lazy_static! {
|
2020-10-02 22:36:57 +00:00
|
|
|
static ref HIT_COUNTER: IntCounterVec = register_int_counter_vec!(
|
|
|
|
opts!("blogpost_hits", "Number of hits to blogposts"),
|
|
|
|
&["name"]
|
|
|
|
)
|
|
|
|
.unwrap();
|
2020-07-16 19:32:30 +00:00
|
|
|
}
|
|
|
|
|
2020-10-02 22:36:57 +00:00
|
|
|
#[instrument(skip(state))]
|
2020-07-16 19:32:30 +00:00
|
|
|
pub async fn index(state: Arc<State>) -> Result<impl Reply, Rejection> {
|
|
|
|
let state = state.clone();
|
|
|
|
Response::builder().html(|o| templates::blogindex_html(o, state.blog.clone()))
|
|
|
|
}
|
|
|
|
|
2020-10-02 22:36:57 +00:00
|
|
|
#[instrument(skip(state))]
|
2020-07-16 19:32:30 +00:00
|
|
|
pub async fn series(state: Arc<State>) -> Result<impl Reply, Rejection> {
|
|
|
|
let state = state.clone();
|
|
|
|
let mut series: Vec<String> = vec![];
|
|
|
|
|
|
|
|
for post in &state.blog {
|
|
|
|
if post.front_matter.series.is_some() {
|
|
|
|
series.push(post.front_matter.series.as_ref().unwrap().clone());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
series.sort();
|
|
|
|
series.dedup();
|
|
|
|
|
|
|
|
Response::builder().html(|o| templates::series_html(o, series))
|
|
|
|
}
|
|
|
|
|
2020-10-02 22:36:57 +00:00
|
|
|
#[instrument(skip(state))]
|
2020-07-16 19:32:30 +00:00
|
|
|
pub async fn series_view(series: String, state: Arc<State>) -> Result<impl Reply, Rejection> {
|
|
|
|
let state = state.clone();
|
|
|
|
let mut posts: Vec<Post> = vec![];
|
|
|
|
|
|
|
|
for post in &state.blog {
|
|
|
|
if post.front_matter.series.is_none() {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if post.front_matter.series.as_ref().unwrap() != &series {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
posts.push(post.clone());
|
|
|
|
}
|
|
|
|
|
|
|
|
if posts.len() == 0 {
|
2020-10-02 22:36:57 +00:00
|
|
|
error!("series not found");
|
2020-07-16 19:32:30 +00:00
|
|
|
Err(SeriesNotFound(series).into())
|
|
|
|
} else {
|
|
|
|
Response::builder().html(|o| templates::series_posts_html(o, series, &posts))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-02 22:36:57 +00:00
|
|
|
#[instrument(skip(state))]
|
2020-07-16 19:32:30 +00:00
|
|
|
pub async fn post_view(name: String, state: Arc<State>) -> Result<impl Reply, Rejection> {
|
|
|
|
let mut want: Option<Post> = None;
|
|
|
|
|
|
|
|
for post in &state.blog {
|
|
|
|
if post.link == format!("blog/{}", name) {
|
|
|
|
want = Some(post.clone());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
match want {
|
|
|
|
None => Err(PostNotFound("blog".into(), name).into()),
|
|
|
|
Some(post) => {
|
2020-10-02 22:36:57 +00:00
|
|
|
HIT_COUNTER
|
|
|
|
.with_label_values(&[name.clone().as_str()])
|
|
|
|
.inc();
|
2020-07-16 19:32:30 +00:00
|
|
|
let body = Html(post.body_html.clone());
|
|
|
|
Response::builder().html(|o| templates::blogpost_html(o, post, body))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|