241 lines
7.4 KiB
Rust
241 lines
7.4 KiB
Rust
// Rocket generates a lot of warnings for handlers
|
|
// TODO: figure out why
|
|
#![allow(unreachable_patterns)]
|
|
#[macro_use]
|
|
extern crate rocket;
|
|
use std::{error::Error, io::Cursor, str::FromStr};
|
|
|
|
use async_graphql::{extensions, http::GraphiQLSource, EmptySubscription, Schema};
|
|
use async_graphql_rocket::{GraphQLQuery, GraphQLRequest, GraphQLResponse};
|
|
use cacher::FilesystemCacher;
|
|
use letterbox_notmuch::{Notmuch, NotmuchError, ThreadSet};
|
|
#[cfg(feature = "tantivy")]
|
|
use letterbox_server::tantivy::TantivyConnection;
|
|
use letterbox_server::{
|
|
config::Config,
|
|
error::ServerError,
|
|
graphql::{Attachment, GraphqlSchema, Mutation, QueryRoot},
|
|
nm::{attachment_bytes, cid_attachment_bytes},
|
|
};
|
|
use rocket::{
|
|
fairing::AdHoc,
|
|
http::{ContentType, Header},
|
|
request::Request,
|
|
response::{content, Debug, Responder},
|
|
serde::json::Json,
|
|
Response, State,
|
|
};
|
|
use rocket_cors::{AllowedHeaders, AllowedOrigins};
|
|
use sqlx::postgres::PgPool;
|
|
|
|
#[get("/show/<query>/pretty")]
|
|
async fn show_pretty(
|
|
nm: &State<Notmuch>,
|
|
query: &str,
|
|
) -> Result<Json<ThreadSet>, Debug<ServerError>> {
|
|
let query = urlencoding::decode(query).map_err(|e| ServerError::from(NotmuchError::from(e)))?;
|
|
let res = nm.show(&query).map_err(ServerError::from)?;
|
|
Ok(Json(res))
|
|
}
|
|
|
|
#[get("/show/<query>")]
|
|
async fn show(nm: &State<Notmuch>, query: &str) -> Result<Json<ThreadSet>, Debug<NotmuchError>> {
|
|
let query = urlencoding::decode(query).map_err(NotmuchError::from)?;
|
|
let res = nm.show(&query)?;
|
|
Ok(Json(res))
|
|
}
|
|
|
|
struct InlineAttachmentResponder(Attachment);
|
|
|
|
impl<'r, 'o: 'r> Responder<'r, 'o> for InlineAttachmentResponder {
|
|
fn respond_to(self, _: &'r Request<'_>) -> rocket::response::Result<'o> {
|
|
let mut resp = Response::build();
|
|
if let Some(filename) = self.0.filename {
|
|
resp.header(Header::new(
|
|
"Content-Disposition",
|
|
format!(r#"inline; filename="{}""#, filename),
|
|
));
|
|
}
|
|
if let Some(content_type) = self.0.content_type {
|
|
if let Some(ct) = ContentType::parse_flexible(&content_type) {
|
|
resp.header(ct);
|
|
}
|
|
}
|
|
resp.sized_body(self.0.bytes.len(), Cursor::new(self.0.bytes))
|
|
.ok()
|
|
}
|
|
}
|
|
|
|
struct DownloadAttachmentResponder(Attachment);
|
|
|
|
impl<'r, 'o: 'r> Responder<'r, 'o> for DownloadAttachmentResponder {
|
|
fn respond_to(self, _: &'r Request<'_>) -> rocket::response::Result<'o> {
|
|
let mut resp = Response::build();
|
|
if let Some(filename) = self.0.filename {
|
|
resp.header(Header::new(
|
|
"Content-Disposition",
|
|
format!(r#"attachment; filename="{}""#, filename),
|
|
));
|
|
}
|
|
if let Some(content_type) = self.0.content_type {
|
|
if let Some(ct) = ContentType::parse_flexible(&content_type) {
|
|
resp.header(ct);
|
|
}
|
|
}
|
|
resp.sized_body(self.0.bytes.len(), Cursor::new(self.0.bytes))
|
|
.ok()
|
|
}
|
|
}
|
|
|
|
#[get("/cid/<id>/<cid>")]
|
|
async fn view_cid(
|
|
nm: &State<Notmuch>,
|
|
id: &str,
|
|
cid: &str,
|
|
) -> Result<InlineAttachmentResponder, Debug<ServerError>> {
|
|
let mid = if id.starts_with("id:") {
|
|
id.to_string()
|
|
} else {
|
|
format!("id:{}", id)
|
|
};
|
|
info!("view cid attachment {mid} {cid}");
|
|
let attachment = cid_attachment_bytes(nm, &mid, &cid)?;
|
|
Ok(InlineAttachmentResponder(attachment))
|
|
}
|
|
|
|
#[get("/view/attachment/<id>/<idx>/<_>")]
|
|
async fn view_attachment(
|
|
nm: &State<Notmuch>,
|
|
id: &str,
|
|
idx: &str,
|
|
) -> Result<InlineAttachmentResponder, Debug<ServerError>> {
|
|
let mid = if id.starts_with("id:") {
|
|
id.to_string()
|
|
} else {
|
|
format!("id:{}", id)
|
|
};
|
|
info!("view attachment {mid} {idx}");
|
|
let idx: Vec<_> = idx
|
|
.split('.')
|
|
.map(|s| s.parse().expect("not a usize"))
|
|
.collect();
|
|
let attachment = attachment_bytes(nm, &mid, &idx)?;
|
|
Ok(InlineAttachmentResponder(attachment))
|
|
}
|
|
|
|
#[get("/download/attachment/<id>/<idx>/<_>")]
|
|
async fn download_attachment(
|
|
nm: &State<Notmuch>,
|
|
id: &str,
|
|
idx: &str,
|
|
) -> Result<DownloadAttachmentResponder, Debug<ServerError>> {
|
|
let mid = if id.starts_with("id:") {
|
|
id.to_string()
|
|
} else {
|
|
format!("id:{}", id)
|
|
};
|
|
info!("download attachment {mid} {idx}");
|
|
let idx: Vec<_> = idx
|
|
.split('.')
|
|
.map(|s| s.parse().expect("not a usize"))
|
|
.collect();
|
|
let attachment = attachment_bytes(nm, &mid, &idx)?;
|
|
Ok(DownloadAttachmentResponder(attachment))
|
|
}
|
|
|
|
#[get("/original/<id>")]
|
|
async fn original(
|
|
nm: &State<Notmuch>,
|
|
id: &str,
|
|
) -> Result<(ContentType, Vec<u8>), Debug<NotmuchError>> {
|
|
let mid = if id.starts_with("id:") {
|
|
id.to_string()
|
|
} else {
|
|
format!("id:{}", id)
|
|
};
|
|
let res = nm.show_original(&mid)?;
|
|
Ok((ContentType::Plain, res))
|
|
}
|
|
|
|
#[rocket::get("/")]
|
|
fn graphiql() -> content::RawHtml<String> {
|
|
content::RawHtml(GraphiQLSource::build().endpoint("/api/graphql").finish())
|
|
}
|
|
|
|
#[rocket::get("/graphql?<query..>")]
|
|
async fn graphql_query(schema: &State<GraphqlSchema>, query: GraphQLQuery) -> GraphQLResponse {
|
|
query.execute(schema.inner()).await
|
|
}
|
|
|
|
#[rocket::post("/graphql", data = "<request>", format = "application/json")]
|
|
async fn graphql_request(
|
|
schema: &State<GraphqlSchema>,
|
|
request: GraphQLRequest,
|
|
) -> GraphQLResponse {
|
|
request.execute(schema.inner()).await
|
|
}
|
|
|
|
#[rocket::main]
|
|
async fn main() -> Result<(), Box<dyn Error>> {
|
|
let _guard = xtracing::init(env!("CARGO_BIN_NAME"))?;
|
|
build_info::build_info!(fn bi);
|
|
info!("Build Info: {}", letterbox_shared::build_version(bi));
|
|
let allowed_origins = AllowedOrigins::all();
|
|
let cors = rocket_cors::CorsOptions {
|
|
allowed_origins,
|
|
allowed_methods: vec!["Get"]
|
|
.into_iter()
|
|
.map(|s| FromStr::from_str(s).unwrap())
|
|
.collect(),
|
|
allowed_headers: AllowedHeaders::some(&["Authorization", "Accept"]),
|
|
allow_credentials: true,
|
|
..Default::default()
|
|
}
|
|
.to_cors()?;
|
|
|
|
let rkt = rocket::build()
|
|
.mount(
|
|
letterbox_shared::urls::MOUNT_POINT,
|
|
routes![
|
|
original,
|
|
show_pretty,
|
|
show,
|
|
graphql_query,
|
|
graphql_request,
|
|
graphiql,
|
|
view_cid,
|
|
view_attachment,
|
|
download_attachment,
|
|
],
|
|
)
|
|
.attach(cors)
|
|
.attach(AdHoc::config::<Config>());
|
|
|
|
let config: Config = rkt.figment().extract()?;
|
|
if !std::fs::exists(&config.slurp_cache_path)? {
|
|
info!("Creating slurp cache @ '{}'", &config.slurp_cache_path);
|
|
std::fs::create_dir_all(&config.slurp_cache_path)?;
|
|
}
|
|
let pool = PgPool::connect(&config.newsreader_database_url).await?;
|
|
sqlx::migrate!("./migrations").run(&pool).await?;
|
|
#[cfg(feature = "tantivy")]
|
|
let tantivy_conn = TantivyConnection::new(&config.newsreader_tantivy_db_path)?;
|
|
|
|
let cacher = FilesystemCacher::new(&config.slurp_cache_path)?;
|
|
let schema = Schema::build(QueryRoot, Mutation, EmptySubscription)
|
|
.data(Notmuch::default())
|
|
.data(cacher)
|
|
.data(pool.clone());
|
|
|
|
#[cfg(feature = "tantivy")]
|
|
let schema = schema.data(tantivy_conn);
|
|
|
|
let schema = schema.extension(extensions::Logger).finish();
|
|
|
|
let rkt = rkt.manage(schema).manage(pool).manage(Notmuch::default());
|
|
//.manage(Notmuch::with_config("../notmuch/testdata/notmuch.config"))
|
|
|
|
rkt.launch().await?;
|
|
Ok(())
|
|
}
|