mailpot/archive-http/src/main.rs

205 lines
7.0 KiB
Rust

/*
* This file is part of mailpot
*
* Copyright 2020 - Manos Pitsidianakis
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as
* published by the Free Software Foundation, either version 3 of the
* License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with this program. If not, see <https://www.gnu.org/licenses/>.
*/
extern crate mailpot;
pub use mailpot::config::*;
pub use mailpot::db::*;
pub use mailpot::errors::*;
pub use mailpot::models::*;
pub use mailpot::*;
use std::sync::Arc;
use percent_encoding::percent_decode_str;
use tera::{Context, Tera};
use warp::Filter;
lazy_static::lazy_static! {
pub static ref TEMPLATES: Tera = {
let mut tera = match Tera::new("src/templates/*") {
Ok(t) => t,
Err(e) => {
println!("Parsing error(s): {}", e);
::std::process::exit(1);
}
};
let names: Vec<_> = tera.get_template_names().collect();
println!("names: {:?}", names);
assert!(!names.is_empty());
tera.autoescape_on(vec![".html", ".sql"]);
tera
};
}
/*
#[derive(Template)]
#[template(path = "lists.html")]
struct ListsTemplate<'a> {
title: &'a str,
description: &'a str,
lists_len: usize,
lists: Vec<ListTemplate<'a>>,
}
#[derive(Template)]
#[template(path = "list.html")]
struct ListTemplate<'a> {
title: &'a str,
list: &'a DbVal<MailingList>,
posts: Vec<DbVal<Post>>,
months: Vec<String>,
body: &'a str,
}
impl<'a> Into<ListTemplate<'a>> for (&'a DbVal<MailingList>, &'a Database) {
fn into(self: (&'a DbVal<MailingList>, &'a Database)) -> ListTemplate<'a> {
let (list, db) = self;
let months = db.months(list.pk).unwrap();
let posts = db.list_posts(list.pk, None).unwrap();
ListTemplate {
title: &list.name,
list,
posts,
months,
body: list.description.as_deref().unwrap_or_default(),
}
}
}
#[derive(Template)]
#[template(path = "post.html")]
struct PostTemplate<'a> {
title: &'a str,
_list: &'a DbVal<MailingList>,
_post: DbVal<Post>,
body: &'a str,
_from: &'a str,
_to: &'a str,
subject: &'a str,
_in_reply_to: Option<String>,
_references: Vec<String>,
}
*/
#[tokio::main]
async fn main() {
let config_path = std::env::args()
.skip(1)
.next()
.expect("Expected configuration file path as first argument.");
let conf = Arc::new(Configuration::from_file(config_path).unwrap());
let conf1 = conf.clone();
let list_handler = warp::path!("lists" / i64).map(move |list_pk: i64| {
let db = Database::open_db(&conf1).unwrap();
let list = db.get_list(list_pk).unwrap().unwrap();
let months = db.months(list_pk).unwrap();
let posts = db.list_posts(list_pk, None).unwrap();
let mut context = Context::new();
context.insert("title", &list.name);
context.insert("list", &list);
context.insert("months", &months);
context.insert("posts", &posts);
context.insert("body", &list.description.clone().unwrap_or_default());
Ok(warp::reply::html(
TEMPLATES.render("list.html", &context).unwrap(),
))
});
let conf2 = conf.clone();
let post_handler =
warp::path!("list" / i64 / String).map(move |list_pk: i64, message_id: String| {
let message_id = percent_decode_str(&message_id).decode_utf8().unwrap();
let db = Database::open_db(&conf2).unwrap();
let list = db.get_list(list_pk).unwrap().unwrap();
let posts = db.list_posts(list_pk, None).unwrap();
let post = posts
.iter()
.find(|p| message_id.contains(&p.message_id))
.unwrap();
let envelope = melib::Envelope::from_bytes(post.message.as_slice(), None)
.expect("Could not parse mail");
let body = envelope.body_bytes(post.message.as_slice());
let body_text = body.text();
let mut context = Context::new();
context.insert("title", &list.name);
context.insert("list", &list);
context.insert("post", &post);
context.insert("posts", &posts);
context.insert("body", &body_text);
context.insert("from", &envelope.field_from_to_string());
context.insert("date", &envelope.date_as_str());
context.insert("to", &envelope.field_to_to_string());
context.insert("subject", &envelope.subject());
context.insert(
"in_reply_to",
&envelope.in_reply_to_display().map(|r| r.to_string()),
);
context.insert(
"references",
&envelope
.references()
.into_iter()
.map(|m| m.to_string())
.collect::<Vec<String>>(),
);
Ok(warp::reply::html(
TEMPLATES.render("post.html", &context).unwrap(),
))
});
let conf3 = conf.clone();
let index_handler = warp::path::end().map(move || {
let db = Database::open_db(&conf3).unwrap();
let lists_values = db.list_lists().unwrap();
dbg!(&lists_values);
let lists = lists_values
.iter()
.map(|list| {
let mut context = Context::new();
let months = db.months(list.pk).unwrap();
let posts = db.list_posts(list.pk, None).unwrap();
context.insert("title", &list.name);
context.insert("list", &list);
context.insert("posts", &posts);
context.insert("months", &months);
context.insert("body", &list.description.as_deref().unwrap_or_default());
context.into_json()
})
.collect::<Vec<_>>();
let mut context = Context::new();
context.insert("title", "mailing list archive");
context.insert("description", "");
context.insert("lists", &lists);
Ok(warp::reply::html(
TEMPLATES.render("lists.html", &context).unwrap(),
))
});
let routes = warp::get()
.and(index_handler)
.or(list_handler)
.or(post_handler);
// Note that composing filters for many routes may increase compile times (because it uses a lot of generics).
// If you wish to use dynamic dispatch instead and speed up compile times while
// making it slightly slower at runtime, you can use Filter::boxed().
eprintln!("Running at http://127.0.0.1:3030");
warp::serve(routes).run(([127, 0, 0, 1], 3030)).await;
}