diff --git a/src/model/mail.rs b/src/model/mail.rs index 5a99aec..098723e 100644 --- a/src/model/mail.rs +++ b/src/model/mail.rs @@ -1,8 +1,12 @@ -use std::error::Error; +use std::{error::Error, fs}; use lettre::{ - message::header::ContentType, transport::smtp::authentication::Credentials, Message, - SmtpTransport, Transport, + message::{ + header::{self, ContentType}, + Attachment, MultiPart, SinglePart, + }, + transport::smtp::authentication::Credentials, + Message, SmtpTransport, Transport, }; use sqlx::SqlitePool; @@ -13,7 +17,7 @@ use super::{family::Family, log::Log, role::Role, user::User}; pub struct Mail {} impl Mail { - pub async fn send(db: &SqlitePool, data: MailToSend, smtp_pw: String) -> bool { + pub async fn send(db: &SqlitePool, data: MailToSend<'_>, smtp_pw: String) -> bool { let mut email = Message::builder() .from( "ASKÖ Ruderverein Donau Linz " @@ -45,12 +49,30 @@ impl Mail { } } - // TODO: handle attachments + let mut multipart = MultiPart::mixed().singlepart( + SinglePart::builder() + .header(header::ContentType::TEXT_HTML) + .body(String::from(data.body)), + ); + + for temp_file in &data.files { + let content = fs::read(temp_file.path().unwrap()).unwrap(); + let media_type = format!("{}", temp_file.content_type().unwrap().media_type()); + let content_type = ContentType::parse(&media_type).unwrap(); + let attachment = Attachment::new(format!( + "{}.{}", + temp_file.name().unwrap(), + temp_file.content_type().unwrap().extension().unwrap() + )) + .body(content, content_type); + + multipart = multipart.singlepart(attachment); + } let email = email .subject(data.subject) .header(ContentType::TEXT_PLAIN) - .body(String::from(data.body)) + .multipart(multipart) .unwrap(); let creds = Credentials::new("no-reply@rudernlinz.at".to_owned(), smtp_pw); diff --git a/src/tera/admin/mail.rs b/src/tera/admin/mail.rs index 450c513..fee06ab 100644 --- a/src/tera/admin/mail.rs +++ b/src/tera/admin/mail.rs @@ -1,4 +1,5 @@ use rocket::form::Form; +use rocket::fs::TempFile; use rocket::response::{Flash, Redirect}; use rocket::{get, request::FlashMessage, routes, Route, State}; use rocket::{post, FromForm}; @@ -39,18 +40,17 @@ async fn fee(db: &State, _admin: AdminUser, config: &State) } #[derive(FromForm, Debug)] -pub struct MailToSend { - //<'a> { +pub struct MailToSend<'a> { pub(crate) role_id: i32, pub(crate) subject: String, pub(crate) body: String, - //pub(crate) files: Vec>, + pub(crate) files: Vec>, } #[post("/mail", data = "")] async fn update( db: &State, - data: Form, + data: Form>, config: &State, _admin: AdminUser, ) -> Flash {