2018-08-15 17:25:59 +02:00
|
|
|
use native_tls::{Protocol, TlsConnector};
|
2018-09-19 21:45:50 +02:00
|
|
|
use lettre::{Transport, SmtpTransport, SmtpClient, ClientTlsParameters, ClientSecurity};
|
|
|
|
use lettre::smtp::ConnectionReuseParameters;
|
2018-08-15 17:00:55 +02:00
|
|
|
use lettre::smtp::authentication::Credentials;
|
2018-08-15 08:32:19 +02:00
|
|
|
use lettre_email::EmailBuilder;
|
|
|
|
|
2018-12-07 02:05:45 +01:00
|
|
|
use crate::MailConfig;
|
2018-12-15 03:54:03 +01:00
|
|
|
use crate::CONFIG;
|
2018-08-15 08:32:19 +02:00
|
|
|
|
|
|
|
fn mailer(config: &MailConfig) -> SmtpTransport {
|
|
|
|
let client_security = if config.smtp_ssl {
|
2018-10-04 00:01:04 +02:00
|
|
|
let tls = TlsConnector::builder()
|
|
|
|
.min_protocol_version(Some(Protocol::Tlsv11))
|
|
|
|
.build()
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
ClientSecurity::Required(ClientTlsParameters::new(config.smtp_host.clone(), tls))
|
2018-08-15 08:32:19 +02:00
|
|
|
} else {
|
|
|
|
ClientSecurity::None
|
|
|
|
};
|
|
|
|
|
2018-09-19 21:45:50 +02:00
|
|
|
let smtp_client = SmtpClient::new(
|
2018-10-04 00:01:04 +02:00
|
|
|
(config.smtp_host.as_str(), config.smtp_port),
|
|
|
|
client_security,
|
2018-08-15 17:00:55 +02:00
|
|
|
).unwrap();
|
|
|
|
|
2018-09-19 21:45:50 +02:00
|
|
|
let smtp_client = match (&config.smtp_username, &config.smtp_password) {
|
2018-10-04 00:01:04 +02:00
|
|
|
(Some(user), Some(pass)) => smtp_client.credentials(Credentials::new(user.clone(), pass.clone())),
|
|
|
|
_ => smtp_client,
|
2018-08-15 17:00:55 +02:00
|
|
|
};
|
|
|
|
|
2018-09-19 21:45:50 +02:00
|
|
|
smtp_client
|
2018-08-15 08:32:19 +02:00
|
|
|
.smtp_utf8(true)
|
2018-08-15 17:00:55 +02:00
|
|
|
.connection_reuse(ConnectionReuseParameters::NoReuse)
|
2018-09-19 21:45:50 +02:00
|
|
|
.transport()
|
2018-08-15 08:32:19 +02:00
|
|
|
}
|
|
|
|
|
2018-09-11 13:04:34 +02:00
|
|
|
pub fn send_password_hint(address: &str, hint: Option<String>, config: &MailConfig) -> Result<(), String> {
|
|
|
|
let (subject, body) = if let Some(hint) = hint {
|
|
|
|
("Your master password hint",
|
|
|
|
format!(
|
|
|
|
"You (or someone) recently requested your master password hint.\n\n\
|
|
|
|
Your hint is: \"{}\"\n\n\
|
|
|
|
If you did not request your master password hint you can safely ignore this email.\n",
|
|
|
|
hint))
|
|
|
|
} else {
|
|
|
|
("Sorry, you have no password hint...",
|
2018-10-04 00:01:04 +02:00
|
|
|
"Sorry, you have not specified any password hint...\n".into())
|
2018-09-11 13:04:34 +02:00
|
|
|
};
|
2018-08-15 10:17:05 +02:00
|
|
|
|
2018-08-15 08:32:19 +02:00
|
|
|
let email = EmailBuilder::new()
|
|
|
|
.to(address)
|
2018-10-04 00:01:04 +02:00
|
|
|
.from((config.smtp_from.clone(), "Bitwarden-rs"))
|
2018-09-11 13:04:34 +02:00
|
|
|
.subject(subject)
|
2018-08-15 10:17:05 +02:00
|
|
|
.body(body)
|
2018-10-04 00:01:04 +02:00
|
|
|
.build()
|
|
|
|
.map_err(|e| e.to_string())?;
|
2018-08-15 08:32:19 +02:00
|
|
|
|
2018-10-04 00:01:04 +02:00
|
|
|
mailer(config)
|
|
|
|
.send(email.into())
|
|
|
|
.map_err(|e| e.to_string())
|
|
|
|
.and(Ok(()))
|
2018-08-15 08:32:19 +02:00
|
|
|
}
|
2018-12-15 03:54:03 +01:00
|
|
|
|
|
|
|
pub fn send_invite(address: &str, org_id: &str, org_user_id: &str, token: &str, org_name: &str, config: &MailConfig) -> Result<(), String> {
|
|
|
|
let (subject, body) = {
|
|
|
|
(format!("Join {}", &org_name),
|
|
|
|
format!(
|
|
|
|
"<html>
|
|
|
|
<p>You have been invited to join the <b>{}</b> organization.<br><br>
|
|
|
|
<a href=\"https://{}/api/organizations/{}/users/{}/accept?token={}\">Click here to join</a></p>
|
|
|
|
<p>If you do not wish to join this organization, you can safely ignore this email.</p>
|
|
|
|
</html>",
|
|
|
|
org_name, CONFIG.domain, org_id, org_user_id, token
|
|
|
|
))
|
|
|
|
};
|
|
|
|
|
|
|
|
let email = EmailBuilder::new()
|
|
|
|
.to(address)
|
|
|
|
.from((config.smtp_from.clone(), "Bitwarden-rs"))
|
|
|
|
.subject(subject)
|
|
|
|
.header(("Content-Type", "text/html"))
|
|
|
|
.body(body)
|
|
|
|
.build()
|
|
|
|
.map_err(|e| e.to_string())?;
|
|
|
|
|
|
|
|
mailer(config)
|
|
|
|
.send(email.into())
|
|
|
|
.map_err(|e| e.to_string())
|
|
|
|
.and(Ok(()))
|
|
|
|
}
|