From 297ce4f11994b483faa015bebe4abf550eb77e11 Mon Sep 17 00:00:00 2001 From: Amaury <1293565+amaury1093@users.noreply.github.com> Date: Thu, 5 Dec 2024 00:53:18 +0100 Subject: [PATCH] feat(core)!: Update async-smtp to 0.9 (#1520) BREAKING CHANGE: The `smtp_security` field has been removed from the /check_email request. --- Cargo.lock | 48 +---- backend/Cargo.toml | 2 +- backend/src/http/v0/check_email/post.rs | 4 +- core/Cargo.toml | 10 +- core/src/lib.rs | 8 +- core/src/rules.rs | 18 ++ core/src/smtp/connect.rs | 269 ++++++++++-------------- core/src/smtp/error.rs | 66 ++++-- core/src/smtp/gmail.rs | 8 +- core/src/smtp/mod.rs | 16 +- core/src/smtp/parser.rs | 19 +- core/src/syntax/mod.rs | 2 +- core/src/util/input_output.rs | 128 ++++++----- core/src/util/sentry.rs | 4 +- 14 files changed, 300 insertions(+), 302 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index cf494cba5..ad6648306 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -335,18 +335,6 @@ dependencies = [ "pin-project-lite", ] -[[package]] -name = "async-native-tls" -version = "0.4.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d57d4cec3c647232e1094dc013546c0b33ce785d8aeb251e1f20dfaf8a9a13fe" -dependencies = [ - "native-tls", - "thiserror", - "tokio", - "url", -] - [[package]] name = "async-reactor-trait" version = "1.1.0" @@ -372,24 +360,17 @@ dependencies = [ [[package]] name = "async-smtp" -version = "0.6.0" +version = "0.9.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "2ade89127f9e0d44f9e83cf574d499060005cd45b7dc76be89c0167487fe8edd" +checksum = "00d1f1a16e5abad3ada9f1f23dbc2f354b138121b90533381be62dada6cbf40a" dependencies = [ - "async-native-tls", - "async-trait", + "anyhow", "base64 0.13.1", - "bufstream", - "fast-socks5 0.8.2", "futures", "hostname", "log", "nom", "pin-project", - "pin-utils", - "serde", - "serde_derive", - "serde_json", "thiserror", "tokio", ] @@ -528,12 +509,6 @@ dependencies = [ "piper", ] -[[package]] -name = "bufstream" -version = "0.1.4" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "40e38929add23cdf8a366df9b0e088953150724bcbe5fc330b0d8eb3b328eec8" - [[package]] name = "bumpalo" version = "3.16.0" @@ -577,14 +552,14 @@ checksum = "baf1de4339761588bc0619e3cbc0120ee582ebb74b53b4efbf79117bd2da40fd" name = "check-if-email-exists" version = "0.10.0" dependencies = [ - "async-native-tls", + "anyhow", "async-recursion", "async-smtp", "chrono", "config", "derive_builder 0.20.2", "fantoccini", - "fast-socks5 0.9.6", + "fast-socks5", "futures", "hickory-proto", "hickory-resolver", @@ -1274,19 +1249,6 @@ dependencies = [ "webdriver", ] -[[package]] -name = "fast-socks5" -version = "0.8.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "961ce1761191c157145a8c9f0c3ceabecd3a729d65c9a8d443674eaee3420f7e" -dependencies = [ - "anyhow", - "log", - "thiserror", - "tokio", - "tokio-stream", -] - [[package]] name = "fast-socks5" version = "0.9.6" diff --git a/backend/Cargo.toml b/backend/Cargo.toml index ff1d51dc9..7ce3794a0 100644 --- a/backend/Cargo.toml +++ b/backend/Cargo.toml @@ -7,7 +7,7 @@ publish = false [dependencies] anyhow = "1.0" -async-smtp = "0.6" +async-smtp = { version = "0.9.1", features = ["runtime-tokio"] } check-if-email-exists = { path = "../core", features = ["sentry"] } config = "0.14" csv = "1.3.0" diff --git a/backend/src/http/v0/check_email/post.rs b/backend/src/http/v0/check_email/post.rs index 7f774b0d7..9da6865ef 100644 --- a/backend/src/http/v0/check_email/post.rs +++ b/backend/src/http/v0/check_email/post.rs @@ -62,7 +62,9 @@ impl CheckEmailRequest { .or_else(|| config.proxy.as_ref()) .cloned(), smtp_timeout: config.smtp_timeout.map(Duration::from_secs), - smtp_port: self.smtp_port.unwrap_or_default(), + smtp_port: self + .smtp_port + .unwrap_or_else(|| CheckEmailInput::default().smtp_port), sentry_dsn: config.sentry_dsn.clone(), ..Default::default() } diff --git a/core/Cargo.toml b/core/Cargo.toml index 5f63915e7..0e9ea3a19 100644 --- a/core/Cargo.toml +++ b/core/Cargo.toml @@ -13,15 +13,15 @@ readme = "../README.md" repository = "https://github.com/reacherhq/check-if-email-exists" [dependencies] -async-native-tls = { version = "0.4", default-features = false } +anyhow = "1.0" async-recursion = "1.0.5" -async-smtp = { version = "0.6.0", features = ["socks5"] } +async-smtp = { version = "0.9.1", features = ["runtime-tokio"] } chrono = { version = "0.4.31", features = ["serde"] } config = "0.14" derive_builder = "0.20" +fast-socks5 = "0.9" fantoccini = { version = "0.21.2" } futures = { version = "0.3.30" } -fast-socks5 = "0.9.2" hickory-proto = "0.24.0" hickory-resolver = "0.24.0" levenshtein = "1.0.5" @@ -37,7 +37,5 @@ sentry = { version = "0.23", optional = true } serde = { version = "1.0.214", features = ["derive"] } serde_json = "1.0.133" thiserror = "1.0" +tokio = { version = "1.40.0", features = ["rt-multi-thread", "macros"] } tracing = "0.1.40" - -[dev-dependencies] -tokio = { version = "1.40.0" } diff --git a/core/src/lib.rs b/core/src/lib.rs index 5aaa2cdfd..23619ff59 100644 --- a/core/src/lib.rs +++ b/core/src/lib.rs @@ -23,13 +23,13 @@ //! - Email deliverability: Is an email sent to this address deliverable? //! - Syntax validation. Is the address syntactically valid? //! - DNS records validation. Does the domain of the email address have valid -//! MX DNS records? +//! MX DNS records? //! - Disposable email address (DEA) validation. Is the address provided by a -//! known disposable email address provider? +//! known disposable email address provider? //! - SMTP server validation. Can the mail exchanger of the email address -//! domain be contacted successfully? +//! domain be contacted successfully? //! - Mailbox disabled. Has this email address been disabled by the email -//! provider? +//! provider? //! - Full inbox. Is the inbox of this mailbox full? //! - Catch-all address. Is this email address a catch-all address? //! diff --git a/core/src/rules.rs b/core/src/rules.rs index f578cdabc..30c209126 100644 --- a/core/src/rules.rs +++ b/core/src/rules.rs @@ -85,3 +85,21 @@ pub fn has_rule(domain: &str, host: &str, rule: &Rule) -> bool { || does_mx_have_rule(host, rule) || does_mx_suffix_have_rule(host, rule) } + +#[cfg(test)] +mod tests { + use super::*; + + #[test] + fn should_skip_catch_all() { + assert_eq!( + true, + has_rule("gmail.com", "alt4.aspmx.l.google.com.", &Rule::SkipCatchAll) + ); + + assert_eq!( + true, + has_rule("domain.com", ".antispamcloud.com.", &Rule::SkipCatchAll) + ) + } +} diff --git a/core/src/smtp/connect.rs b/core/src/smtp/connect.rs index a7c85000c..904a3670b 100644 --- a/core/src/smtp/connect.rs +++ b/core/src/smtp/connect.rs @@ -14,127 +14,107 @@ // You should have received a copy of the GNU Affero General Public License // along with this program. If not, see . -use async_native_tls::TlsConnector; use async_recursion::async_recursion; -use async_smtp::{ - smtp::{commands::*, extension::ClientId, ServerAddress, Socks5Config}, - ClientTlsParameters, EmailAddress, SmtpClient, SmtpTransport, -}; +use async_smtp::commands::{MailCommand, RcptCommand}; +use async_smtp::extension::ClientId; +use async_smtp::{SmtpClient, SmtpTransport}; +use fast_socks5::client::Config; +use fast_socks5::{client::Socks5Stream, Result}; use rand::rngs::SmallRng; use rand::{distributions::Alphanumeric, Rng, SeedableRng}; use std::iter; use std::str::FromStr; -use std::time::Duration; +use tokio::io::{AsyncBufRead, AsyncRead, AsyncWrite, BufStream}; +use tokio::net::TcpStream; use super::parser; use super::{SmtpDetails, SmtpError}; -use crate::LOG_TARGET; use crate::{ rules::{has_rule, Rule}, util::input_output::CheckEmailInput, }; +use crate::{EmailAddress, LOG_TARGET}; + +// Define a new trait that combines AsyncRead, AsyncWrite, and Unpin +trait AsyncReadWrite: AsyncRead + AsyncWrite + Unpin + Send {} +impl AsyncReadWrite for T {} /// Try to send an smtp command, close and return Err if fails. macro_rules! try_smtp ( ($res: expr, $client: ident, $to_email: expr, $host: expr, $port: expr) => ({ if let Err(err) = $res { - log::debug!(target: LOG_TARGET, "[email={}] Closing [host={}:{}], because of error '{:?}'.", $to_email, $host, $port, err); + log::debug!( + target: LOG_TARGET, + "[email={}] Closing [host={}:{}], because of error '{:?}'.", + $to_email, $host, $port,err, + ); // Try to close the connection, but ignore if there's an error. - let _ = $client.close().await; + let _ = $client.quit().await; - return Err(SmtpError::SmtpError(err)); + return Err(SmtpError::AsyncSmtpError(err)); } }) ); -/// Attempt to connect to host via SMTP, and return SMTP client on success. -async fn connect_to_host( - domain: &str, +/// Connect to an SMTP host and return the configured client transport. +async fn connect_to_smtp_host( host: &str, port: u16, input: &CheckEmailInput, -) -> Result { - let smtp_timeout = if let Some(t) = input.smtp_timeout { - if has_rule(domain, host, &Rule::SmtpTimeout45s) { - let duration = t.max(Duration::from_secs(45)); - log::debug!( - target: LOG_TARGET, - "[email={}] Bumping SMTP timeout to {duration:?} because of rule", - input.to_email, - ); - Some(duration) - } else { - input.smtp_timeout - } - } else { - None - }; - +) -> Result>>, SmtpError> { // hostname verification fails if it ends with '.', for example, using // SOCKS5 proxies we can `io: incomplete` error. - let host = host.trim_end_matches('.').to_string(); - - let security = { - let tls_params: ClientTlsParameters = ClientTlsParameters::new( - host.clone(), - TlsConnector::new() - .use_sni(true) - .danger_accept_invalid_certs(true) - .danger_accept_invalid_hostnames(true), - ); - - input.smtp_security.to_client_security(tls_params) + let clean_host = host.trim_end_matches('.').to_string(); + let smtp_client = SmtpClient::new().hello_name(ClientId::Domain(input.hello_name.clone())); + + let stream: BufStream> = match &input.proxy { + Some(proxy) => { + let socks_stream = + if let (Some(username), Some(password)) = (&proxy.username, &proxy.password) { + Socks5Stream::connect_with_password( + (proxy.host.as_ref(), proxy.port), + clean_host.clone(), + port, + username.clone(), + password.clone(), + Config::default(), + ) + .await? + } else { + Socks5Stream::connect( + (proxy.host.as_ref(), proxy.port), + clean_host.clone(), + port, + Config::default(), + ) + .await? + }; + BufStream::new(Box::new(socks_stream) as Box) + } + None => { + let tcp_stream = TcpStream::connect(format!("{}:{}", clean_host, port)).await?; + BufStream::new(Box::new(tcp_stream) as Box) + } }; - let mut smtp_client = SmtpClient::with_security( - ServerAddress { - host: host.clone(), - port, - }, - security, - ) - .hello_name(ClientId::Domain(input.hello_name.clone())) - .timeout(smtp_timeout); - - if let Some(proxy) = &input.proxy { - let socks5_config = match (&proxy.username, &proxy.password) { - (Some(username), Some(password)) => Socks5Config::new_with_user_pass( - proxy.host.clone(), - proxy.port, - username.clone(), - password.clone(), - ), - _ => Socks5Config::new(proxy.host.clone(), proxy.port), - }; - - smtp_client = smtp_client.use_socks5(socks5_config); - } - - let mut smtp_transport = smtp_client.into_transport(); + let mut smtp_transport = SmtpTransport::new(smtp_client, stream).await?; - try_smtp!( - smtp_transport.connect().await, - smtp_transport, - input.to_email, - host, - port - ); - - // "MAIL FROM: user@example.org" - let from_email = EmailAddress::from_str(input.from_email.as_ref()).unwrap_or_else(|_| { + // Set "MAIL FROM" + let from_email = EmailAddress::from_str(&input.from_email).unwrap_or_else(|_| { log::warn!( - "Inputted from_email \"{}\" is not a valid email, using \"user@example.org\" instead", + "Invalid 'from_email' provided: '{}'. Using default: 'user@example.org'", input.from_email ); - EmailAddress::from_str("user@example.org").expect("This is a valid email. qed.") + EmailAddress::from_str("user@example.org").expect("Default email is valid") }); try_smtp!( smtp_transport - .command(MailCommand::new(Some(from_email), vec![],)) + .get_mut() + .command(MailCommand::new(Some(from_email.into_inner()), vec![])) .await, smtp_transport, input.to_email, - host, + clean_host, port ); @@ -152,43 +132,39 @@ struct Deliverability { is_disabled: bool, } -/// Check if `to_email` exists on host SMTP server. This is the core logic of -/// this tool. -async fn email_deliverable( - smtp_transport: &mut SmtpTransport, +/// Checks deliverability of a target email address using the provided SMTP transport. +async fn check_email_deliverability( + smtp_transport: &mut SmtpTransport, to_email: &EmailAddress, ) -> Result { - // "RCPT TO: " - // FIXME Do not clone `to_email`? match smtp_transport - .command(RcptCommand::new(to_email.clone(), vec![])) + .get_mut() + .command(RcptCommand::new(to_email.clone().into_inner(), vec![])) .await { - Ok(_) => { - // According to RFC 5321, `RCPT TO` command succeeds with 250 and - // 251 codes only (no 3xx codes at all): - // https://tools.ietf.org/html/rfc5321#page-56 - // - // Where the 251 code is used for forwarding, which is not our case, - // because we always deliver to the SMTP server hosting the address - // itself. - // - // So, if `response.is_positive()` (which is a condition for - // returning `Ok` from the `command()` method above), then delivery - // succeeds, accordingly to RFC 5321. - Ok(Deliverability { - has_full_inbox: false, - is_deliverable: true, // response.is_positive() - is_disabled: false, - }) - } + // According to RFC 5321, `RCPT TO` command succeeds with 250 and + // 251 codes only (no 3xx codes at all): + // https://tools.ietf.org/html/rfc5321#page-56 + // + // Where the 251 code is used for forwarding, which is not our case, + // because we always deliver to the SMTP server hosting the address + // itself. + // + // So, if `response.is_positive()` (which is a condition for + // returning `Ok` from the `command()` method above), then delivery + // succeeds, accordingly to RFC 5321. + Ok(_) => Ok(Deliverability { + has_full_inbox: false, + is_deliverable: true, // response.is_positive() + is_disabled: false, + }), Err(err) => { // We cast to lowercase, because our matched strings below are all // lowercase. let err_string = err.to_string().to_lowercase(); // Check if the email account has been disabled or blocked. - if parser::is_disabled_account(err_string.as_str()) { + if parser::is_disabled_account(&err_string) { return Ok(Deliverability { has_full_inbox: false, is_deliverable: false, @@ -197,7 +173,7 @@ async fn email_deliverable( } // Check if the email account has a full inbox. - if parser::is_full_inbox(err_string.as_str()) { + if parser::is_full_inbox(&err_string) { return Ok(Deliverability { has_full_inbox: true, is_deliverable: false, @@ -219,7 +195,7 @@ async fn email_deliverable( } // Check that the mailbox doesn't exist. - if parser::is_invalid(err_string.as_str(), to_email) { + if parser::is_invalid(&err_string, to_email) { return Ok(Deliverability { has_full_inbox: false, is_deliverable: false, @@ -228,19 +204,18 @@ async fn email_deliverable( } // Return all unparsable errors,. - Err(SmtpError::SmtpError(err)) + Err(SmtpError::AsyncSmtpError(err)) } } } -/// Verify the existence of a catch-all on the domain. -async fn smtp_is_catch_all( - smtp_transport: &mut SmtpTransport, +/// Checks if the domain has a catch-all email setup. +async fn smtp_is_catch_all( + smtp_transport: &mut SmtpTransport, domain: &str, host: &str, input: &CheckEmailInput, ) -> Result { - // Skip catch-all check for known providers. if has_rule(domain, host, &Rule::SkipCatchAll) { log::debug!( target: LOG_TARGET, @@ -252,21 +227,18 @@ async fn smtp_is_catch_all( // Create a random 15-char alphanumerical string. let mut rng = SmallRng::from_entropy(); - let random_email: String = iter::repeat(()) - .map(|()| rng.sample(Alphanumeric)) + let random_email: String = iter::repeat_with(|| rng.sample(Alphanumeric)) .map(char::from) .take(15) .collect(); - let random_email = EmailAddress::new(format!("{random_email}@{domain}")); + let random_email = EmailAddress::new(format!("{}@{}", random_email, domain))?; - email_deliverable( - smtp_transport, - &random_email.expect("Email is correctly constructed. qed."), - ) - .await - .map(|deliverability| deliverability.is_deliverable) + check_email_deliverability(smtp_transport, &random_email) + .await + .map(|result| result.is_deliverable) } +/// Creates an SMTP future for email verification. async fn create_smtp_future( to_email: &EmailAddress, host: &str, @@ -276,7 +248,7 @@ async fn create_smtp_future( ) -> Result<(bool, Deliverability), SmtpError> { // FIXME If the SMTP is not connectable, we should actually return an // Ok(SmtpDetails { can_connect_smtp: false, ... }). - let mut smtp_transport = connect_to_host(domain, host, port, input).await?; + let mut smtp_transport = connect_to_smtp_host(host, port, input).await?; let is_catch_all = smtp_is_catch_all(&mut smtp_transport, domain, host, input) .await @@ -288,7 +260,7 @@ async fn create_smtp_future( is_disabled: false, } } else { - let mut result = email_deliverable(&mut smtp_transport, to_email).await; + let mut result = check_email_deliverability(&mut smtp_transport, to_email).await; // Some SMTP servers automatically close the connection after an error, // so we should reconnect to perform a next command. @@ -304,16 +276,19 @@ async fn create_smtp_future( input.to_email ); - let _ = smtp_transport.close().await; - smtp_transport = connect_to_host(domain, host, port, input).await?; - result = email_deliverable(&mut smtp_transport, to_email).await; + let _ = smtp_transport.quit().await; + smtp_transport = connect_to_smtp_host(host, port, input).await?; + result = check_email_deliverability(&mut smtp_transport, to_email).await; } } result? }; - smtp_transport.close().await.map_err(SmtpError::SmtpError)?; + smtp_transport + .quit() + .await + .map_err(SmtpError::AsyncSmtpError)?; Ok((is_catch_all, deliverability)) } @@ -328,7 +303,18 @@ async fn check_smtp_without_retry( input: &CheckEmailInput, ) -> Result { let fut = create_smtp_future(to_email, host, port, domain, input); - let (is_catch_all, deliverability) = fut.await?; + + let (is_catch_all, deliverability) = match input.smtp_timeout { + Some(smtp_timeout) => { + let timeout = tokio::time::timeout(smtp_timeout, fut); + + match timeout.await { + Ok(result) => result?, + Err(_) => return Err(SmtpError::Timeout(smtp_timeout)), + } + } + None => fut.await?, + }; Ok(SmtpDetails { can_connect_smtp: true, @@ -394,26 +380,3 @@ pub async fn check_smtp_with_retry( _ => result, } } - -#[cfg(test)] -mod tests { - use super::*; - - #[tokio::test] - async fn should_skip_catch_all() { - let smtp_client = SmtpClient::new("gmail.com".into()); - let mut smtp_transport = smtp_client.into_transport(); - - let r = smtp_is_catch_all( - &mut smtp_transport, - "gmail.com", - "alt4.aspmx.l.google.com.", - &CheckEmailInput::default(), - ) - .await; - - assert!(!smtp_transport.is_connected()); // We shouldn't connect to google servers. - assert!(r.is_ok()); - assert!(!r.unwrap()) - } -} diff --git a/core/src/smtp/error.rs b/core/src/smtp/error.rs index 27e1bab63..1d4631f26 100644 --- a/core/src/smtp/error.rs +++ b/core/src/smtp/error.rs @@ -15,29 +15,20 @@ // along with this program. If not, see . use super::gmail::GmailError; - use super::headless::HeadlessError; use super::outlook::microsoft365::Microsoft365Error; use super::parser; use super::yahoo::YahooError; use crate::util::ser_with_display::ser_with_display; -use async_smtp::smtp::error::Error as AsyncSmtpError; -use fast_socks5::SocksError; +use async_smtp::error::Error as AsyncSmtpError; use serde::Serialize; +use std::time::Duration; use thiserror::Error; /// Error occured connecting to this email server via SMTP. #[derive(Debug, Error, Serialize)] #[serde(tag = "type", content = "message")] pub enum SmtpError { - /// Error if we're using a SOCKS5 proxy. - #[serde(serialize_with = "ser_with_display")] - #[error("SOCKS5 error: {0}")] - SocksError(SocksError), - /// Error when communicating with SMTP server. - #[serde(serialize_with = "ser_with_display")] - #[error("SMTP error: {0}")] - SmtpError(AsyncSmtpError), /// Error when verifying a Yahoo email via HTTP requests. #[error("Yahoo error: {0}")] YahooError(YahooError), @@ -50,12 +41,27 @@ pub enum SmtpError { /// Error when verifying a Microsoft 365 email via HTTP request. #[error("Microsoft 365 API error: {0}")] Microsoft365Error(Microsoft365Error), -} - -impl From for SmtpError { - fn from(e: SocksError) -> Self { - SmtpError::SocksError(e) - } + /// Error from async-smtp crate. + #[error("SMTP error: {0}")] + #[serde(serialize_with = "ser_with_display")] + AsyncSmtpError(AsyncSmtpError), + /// I/O error. + #[error("I/O error: {0}")] + #[serde(serialize_with = "ser_with_display")] + IOError(std::io::Error), + /// Timeout error. + #[error("Timeout error: {0:?}")] + Timeout(Duration), + /// SOCKS5 proxy error. + #[error("SOCKS5 error: {0}")] + #[serde(serialize_with = "ser_with_display")] + Socks5(fast_socks5::SocksError), + /// Anyhow error. + /// This is a catch-all error type for any error that can't be categorized + /// into the above types. + #[error("Anyhow error: {0}")] + #[serde(serialize_with = "ser_with_display")] + AnyhowError(anyhow::Error), } impl From for SmtpError { @@ -82,6 +88,30 @@ impl From for SmtpError { } } +impl From for SmtpError { + fn from(e: AsyncSmtpError) -> Self { + SmtpError::AsyncSmtpError(e) + } +} + +impl From for SmtpError { + fn from(e: std::io::Error) -> Self { + SmtpError::IOError(e) + } +} + +impl From for SmtpError { + fn from(e: fast_socks5::SocksError) -> Self { + SmtpError::Socks5(e) + } +} + +impl From for SmtpError { + fn from(e: anyhow::Error) -> Self { + SmtpError::AnyhowError(e) + } +} + impl SmtpError { /// Get a human-understandable description of the error, in form of an enum /// SmtpErrorDesc. This only parses the following known errors: @@ -89,7 +119,7 @@ impl SmtpError { /// - IP needs reverse DNS pub fn get_description(&self) -> Option { match self { - SmtpError::SmtpError(_) => { + SmtpError::AsyncSmtpError(_) => { if parser::is_err_ip_blacklisted(self) { Some(SmtpErrorDesc::IpBlacklisted) } else if parser::is_err_needs_rdns(self) { diff --git a/core/src/smtp/gmail.rs b/core/src/smtp/gmail.rs index 221d2cafb..916099527 100644 --- a/core/src/smtp/gmail.rs +++ b/core/src/smtp/gmail.rs @@ -15,12 +15,12 @@ // along with this program. If not, see . use super::SmtpDetails; +use crate::EmailAddress; use crate::LOG_TARGET; use crate::{ smtp::http_api::create_client, util::{input_output::CheckEmailInput, ser_with_display::ser_with_display}, }; -use async_smtp::EmailAddress; use reqwest::Error as ReqwestError; use serde::Serialize; use thiserror::Error; @@ -78,11 +78,9 @@ pub fn is_gmail(host: &str) -> bool { #[cfg(test)] mod tests { - use std::str::FromStr; - - use crate::CheckEmailInputBuilder; - use super::*; + use crate::CheckEmailInputBuilder; + use std::str::FromStr; #[tokio::test] #[ignore] // ref: https://github.com/reacherhq/check-if-email-exists/issues/1431 diff --git a/core/src/smtp/mod.rs b/core/src/smtp/mod.rs index 14c208ec9..6ee92f65d 100644 --- a/core/src/smtp/mod.rs +++ b/core/src/smtp/mod.rs @@ -24,23 +24,21 @@ mod outlook; mod parser; mod yahoo; -use std::default::Default; - -use async_smtp::EmailAddress; -use hickory_proto::rr::Name; -use serde::{Deserialize, Serialize}; - +use crate::EmailAddress; use crate::{ util::input_output::CheckEmailInput, GmailVerifMethod, HotmailB2CVerifMethod, YahooVerifMethod, }; use connect::check_smtp_with_retry; -pub use error::*; +use hickory_proto::rr::Name; +use serde::{Deserialize, Serialize}; +use std::default::Default; pub use self::{ gmail::is_gmail, outlook::{is_hotmail, is_hotmail_b2b, is_hotmail_b2c}, yahoo::is_yahoo, }; +pub use error::*; #[derive(Debug, Default, Deserialize, PartialEq, Serialize)] pub struct SmtpConnection { @@ -163,7 +161,7 @@ pub async fn check_smtp( mod tests { use super::{check_smtp, SmtpConnection, SmtpError}; use crate::CheckEmailInputBuilder; - use async_smtp::{smtp::error::Error, EmailAddress}; + use crate::EmailAddress; use hickory_proto::rr::Name; use std::{str::FromStr, time::Duration}; use tokio::runtime::Runtime; @@ -192,7 +190,7 @@ mod tests { }) ); match res { - Err(SmtpError::SmtpError(Error::Io(_))) => (), // ErrorKind == Timeout + Err(SmtpError::Timeout(_)) => (), // ErrorKind == Timeout _ => panic!("check_smtp did not time out"), } } diff --git a/core/src/smtp/parser.rs b/core/src/smtp/parser.rs index 0d9c60b96..311f28b58 100644 --- a/core/src/smtp/parser.rs +++ b/core/src/smtp/parser.rs @@ -17,7 +17,8 @@ //! Parse the SMTP responses to get information about the email address. use super::error::SmtpError; -use async_smtp::{smtp::error::Error as AsyncSmtpError, EmailAddress}; +use crate::EmailAddress; +use async_smtp::error::Error as AsyncSmtpError; /// is_invalid checks for SMTP responses meaning that the email is invalid, /// i.e. that the mailbox doesn't exist. @@ -127,7 +128,7 @@ pub fn is_disabled_account(e: &str) -> bool { /// Check if the error is an IO "incomplete" error. pub fn is_err_io_errors(e: &SmtpError) -> bool { match e { - SmtpError::SmtpError(AsyncSmtpError::Io(err)) => err.to_string() == "incomplete", + SmtpError::AsyncSmtpError(AsyncSmtpError::Io(err)) => err.to_string() == "incomplete", _ => false, } } @@ -135,7 +136,7 @@ pub fn is_err_io_errors(e: &SmtpError) -> bool { /// Check if the IP is blacklisted. pub fn is_err_ip_blacklisted(e: &SmtpError) -> bool { let e = match e { - SmtpError::SmtpError(AsyncSmtpError::Transient(r) | AsyncSmtpError::Permanent(r)) => { + SmtpError::AsyncSmtpError(AsyncSmtpError::Transient(r) | AsyncSmtpError::Permanent(r)) => { // TODO We can use .to_string() after: // https://github.com/async-email/async-smtp/pull/53 r.message.join("; ").to_lowercase() @@ -205,7 +206,7 @@ pub fn is_err_ip_blacklisted(e: &SmtpError) -> bool { /// Check if the IP needs a reverse DNS. pub fn is_err_needs_rdns(e: &SmtpError) -> bool { let e = match e { - SmtpError::SmtpError(AsyncSmtpError::Transient(r) | AsyncSmtpError::Permanent(r)) => { + SmtpError::AsyncSmtpError(AsyncSmtpError::Transient(r) | AsyncSmtpError::Permanent(r)) => { // TODO We can use .to_string() after: // https://github.com/async-email/async-smtp/pull/53 r.message.join("; ").to_lowercase() @@ -227,11 +228,11 @@ pub fn is_err_needs_rdns(e: &SmtpError) -> bool { mod tests { use super::{is_err_ip_blacklisted, is_invalid}; - use crate::SmtpError::SmtpError; + use crate::EmailAddress; + use crate::SmtpError::AsyncSmtpError; use async_smtp::{ - smtp::error::Error, - smtp::response::{Category, Code, Detail, Response, Severity}, - EmailAddress, + error::Error, + response::{Category, Code, Detail, Response, Severity}, }; use std::str::FromStr; @@ -271,6 +272,6 @@ mod tests { ], )); - assert!(is_err_ip_blacklisted(&SmtpError(err))) + assert!(is_err_ip_blacklisted(&AsyncSmtpError(err))) } } diff --git a/core/src/syntax/mod.rs b/core/src/syntax/mod.rs index 4c5b8f2b0..fedac42e9 100644 --- a/core/src/syntax/mod.rs +++ b/core/src/syntax/mod.rs @@ -16,7 +16,7 @@ mod normalize; -use async_smtp::EmailAddress; +use crate::EmailAddress; use levenshtein::levenshtein; use normalize::normalize_email; use serde::{Deserialize, Serialize}; diff --git a/core/src/util/input_output.rs b/core/src/util/input_output.rs index 0eaa145b4..04f90771c 100644 --- a/core/src/util/input_output.rs +++ b/core/src/util/input_output.rs @@ -14,18 +14,80 @@ // You should have received a copy of the GNU Affero General Public License // along with this program. If not, see . -use std::str::FromStr; -use std::time::{Duration, SystemTime}; - -use async_smtp::{ClientSecurity, ClientTlsParameters}; -use chrono::{DateTime, Utc}; -use derive_builder::Builder; -use serde::{ser::SerializeMap, Deserialize, Serialize, Serializer}; - use crate::misc::{MiscDetails, MiscError}; use crate::mx::{MxDetails, MxError}; use crate::smtp::{SmtpDebug, SmtpDetails, SmtpError, SmtpErrorDesc}; use crate::syntax::SyntaxDetails; +use crate::util::ser_with_display::ser_with_display; +use async_smtp::EmailAddress as AsyncSmtpEmailAddress; +use chrono::{DateTime, Utc}; +use derive_builder::Builder; +use serde::{ser::SerializeMap, Deserialize, Serialize, Serializer}; +use std::fmt::Display; +use std::str::FromStr; +use std::time::{Duration, SystemTime}; + +/// Wrapper around the `EmailAddress` from `async_smtp` to allow for +/// serialization and deserialization. +#[derive(Debug, Eq, PartialEq, Clone)] +pub struct EmailAddress(AsyncSmtpEmailAddress); + +impl Serialize for EmailAddress { + fn serialize(&self, serializer: S) -> Result + where + S: Serializer, + { + ser_with_display(&self.0, serializer) + } +} + +impl<'de> Deserialize<'de> for EmailAddress { + fn deserialize(deserializer: D) -> Result + where + D: serde::Deserializer<'de>, + { + let s = String::deserialize(deserializer)?; + Ok(EmailAddress( + AsyncSmtpEmailAddress::from_str(&s).map_err(serde::de::Error::custom)?, + )) + } +} + +impl Display for EmailAddress { + fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { + self.0.fmt(f) + } +} + +impl FromStr for EmailAddress { + type Err = anyhow::Error; + + fn from_str(s: &str) -> Result { + Ok(EmailAddress(AsyncSmtpEmailAddress::from_str(s)?)) + } +} + +impl EmailAddress { + pub fn new(email: String) -> Result { + Ok(EmailAddress(AsyncSmtpEmailAddress::new(email)?)) + } + + pub fn into_inner(self) -> AsyncSmtpEmailAddress { + self.0 + } +} + +impl AsRef for EmailAddress { + fn as_ref(&self) -> &AsyncSmtpEmailAddress { + &self.0 + } +} + +impl AsRef for EmailAddress { + fn as_ref(&self) -> &str { + self.0.as_ref() + } +} /// Perform the email verification via a specified proxy. The usage of a proxy /// is optional. @@ -41,37 +103,6 @@ pub struct CheckEmailInputProxy { pub password: Option, } -/// Define how to apply TLS to a SMTP client connection. Will be converted into -/// async_smtp::ClientSecurity. -#[derive(Debug, Clone, Copy, Deserialize, Serialize)] -pub enum SmtpSecurity { - /// Insecure connection only (for testing purposes). - None, - /// Start with insecure connection and use `STARTTLS` when available. - Opportunistic, - /// Start with insecure connection and require `STARTTLS`. - Required, - /// Use TLS wrapped connection. - Wrapper, -} - -impl Default for SmtpSecurity { - fn default() -> Self { - Self::Opportunistic - } -} - -impl SmtpSecurity { - pub fn to_client_security(self, tls_params: ClientTlsParameters) -> ClientSecurity { - match self { - Self::None => ClientSecurity::None, - Self::Opportunistic => ClientSecurity::Opportunistic(tls_params), - Self::Required => ClientSecurity::Required(tls_params), - Self::Wrapper => ClientSecurity::Wrapper(tls_params), - } - } -} - /// Select how to verify Yahoo emails. #[derive(Debug, Clone, Copy, Default, PartialEq, Deserialize, Serialize)] pub enum YahooVerifMethod { @@ -195,7 +226,9 @@ pub struct CheckEmailInput { /// Defaults to 25. pub smtp_port: u16, /// Add timeout for the SMTP verification step. Set to None if you don't - /// want to use a timeout. + /// want to use a timeout. This timeout is per SMTP connection. For + /// instance, if you set the number of retries to 2, then the total time + /// for the SMTP verification step can be up to 2 * `smtp_timeout`. /// /// Defaults to None. pub smtp_timeout: Option, @@ -228,10 +261,6 @@ pub struct CheckEmailInput { /// /// Defaults to 1. pub retries: usize, - /// How to apply TLS to a SMTP client connection. - /// - /// Defaults to Opportunistic. - pub smtp_security: SmtpSecurity, /// The WebDriver address to use for headless verifications. /// /// Defaults to http://localhost:9515. @@ -256,7 +285,6 @@ impl Default for CheckEmailInput { hello_name: "gmail.com".into(), proxy: None, smtp_port: 25, - smtp_security: SmtpSecurity::default(), smtp_timeout: None, yahoo_verif_method: YahooVerifMethod::default(), gmail_verif_method: GmailVerifMethod::default(), @@ -411,7 +439,7 @@ impl Serialize for CheckEmailOutput { #[cfg(test)] mod tests { use super::{CheckEmailOutput, DebugDetails}; - use async_smtp::smtp::response::{Category, Code, Detail, Response, Severity}; + use async_smtp::response::{Category, Code, Detail, Response, Severity}; #[test] fn should_serialize_correctly() { @@ -433,7 +461,7 @@ mod tests { misc: Ok(super::MiscDetails::default()), mx: Ok(super::MxDetails::default()), syntax: super::SyntaxDetails::default(), - smtp: Err(super::SmtpError::SmtpError(r.into())), + smtp: Err(super::SmtpError::AsyncSmtpError(r.into())), debug: DebugDetails::default(), } } @@ -441,20 +469,20 @@ mod tests { let res = dummy_response_with_message("blacklist"); let actual = serde_json::to_string(&res).unwrap(); // Make sure the `description` is present with IpBlacklisted. - let expected = r#""smtp":{"error":{"type":"SmtpError","message":"transient: blacklist"},"description":"IpBlacklisted"}"#; + let expected = r#""smtp":{"error":{"type":"AsyncSmtpError","message":"transient: blacklist; 8BITMIME; SIZE 42"},"description":"IpBlacklisted"}"#; assert!(actual.contains(expected)); let res = dummy_response_with_message("Client host rejected: cannot find your reverse hostname"); let actual = serde_json::to_string(&res).unwrap(); // Make sure the `description` is present with NeedsRDNs. - let expected = r#"smtp":{"error":{"type":"SmtpError","message":"transient: Client host rejected: cannot find your reverse hostname"},"description":"NeedsRDNS"}"#; + let expected = r#"smtp":{"error":{"type":"AsyncSmtpError","message":"transient: Client host rejected: cannot find your reverse hostname; 8BITMIME; SIZE 42"},"description":"NeedsRDNS"}"#; assert!(actual.contains(expected)); let res = dummy_response_with_message("foobar"); let actual = serde_json::to_string(&res).unwrap(); // Make sure the `description` is NOT present. - let expected = r#""smtp":{"error":{"type":"SmtpError","message":"transient: foobar"}}"#; + let expected = r#""smtp":{"error":{"type":"AsyncSmtpError","message":"transient: foobar; 8BITMIME; SIZE 42"}}"#; assert!(actual.contains(expected)); } } diff --git a/core/src/util/sentry.rs b/core/src/util/sentry.rs index 391cf2761..7aeab6516 100644 --- a/core/src/util/sentry.rs +++ b/core/src/util/sentry.rs @@ -24,7 +24,7 @@ use crate::misc::MiscError; use crate::mx::MxError; use crate::LOG_TARGET; use crate::{smtp::SmtpError, CheckEmailOutput}; -use async_smtp::smtp::error::Error as AsyncSmtpError; +use async_smtp::error::Error as AsyncSmtpError; use sentry::protocol::{Event, Exception, Level, Values}; use thiserror::Error; use tracing::{debug, info}; @@ -121,7 +121,7 @@ pub fn log_unknown_errors(result: &CheckEmailOutput, backend_name: &str) { (_, _, Err(err)) if err.get_description().is_some() => { // If the SMTP error is known, we don't track it in Sentry. } - (_, _, Err(SmtpError::SmtpError(AsyncSmtpError::Transient(response)))) + (_, _, Err(SmtpError::AsyncSmtpError(AsyncSmtpError::Transient(response)))) if skip_smtp_transient_errors(&response.message) => { // If the SMTP error is transient and known, we don't track it in