using System; using System.Threading.Tasks; using EnsureThat; using MailKit.Net.Smtp; using Microsoft.Extensions.Logging; using MimeKit; using MimeKit.Utils; using Ombi.Core.Settings; using Ombi.Helpers; using Ombi.Notifications.Models; using Ombi.Notifications.Templates; using Ombi.Settings.Settings.Models; using Ombi.Settings.Settings.Models.Notifications; namespace Ombi.Notifications { public class GenericEmailProvider : IEmailProvider { public GenericEmailProvider(ISettingsService s, ILogger log) { CustomizationSettings = s; _log = log; } private ISettingsService CustomizationSettings { get; } private readonly ILogger _log; /// /// This will load up the Email template and generate the HTML /// /// /// /// public async Task SendAdHoc(NotificationMessage model, EmailNotificationSettings settings) { try { var email = new EmailBasicTemplate(); var customization = await CustomizationSettings.GetSettingsAsync(); var html = email.LoadTemplate(model.Subject, model.Message, null, customization.Logo); var messageId = MimeUtils.GenerateMessageId(); if (customization.ApplicationUrl.HasValue()) { if (Uri.TryCreate(customization.ApplicationUrl, UriKind.RelativeOrAbsolute, out var url)) { messageId = MimeUtils.GenerateMessageId(url.IdnHost); } } var textBody = string.Empty; model.Other.TryGetValue("PlainTextBody", out textBody); var body = new BodyBuilder { HtmlBody = html, TextBody = textBody }; var message = new MimeMessage { Body = body.ToMessageBody(), Subject = model.Subject, MessageId = messageId }; message.From.Add(new MailboxAddress(string.IsNullOrEmpty(settings.SenderName) ? settings.SenderAddress : settings.SenderName, settings.SenderAddress)); if (model.To.HasValue()) { message.To.Add(new MailboxAddress(model.To, model.To)); } // Check for BCC if (model.Other.TryGetValue("bcc", out var bcc)) { var bccList = bcc.Split(',', StringSplitOptions.RemoveEmptyEntries); foreach (var item in bccList) { message.Bcc.Add(new MailboxAddress(item, item)); } } using (var client = new SmtpClient()) { if (settings.DisableCertificateChecking) { // Disable validation of the certificate associated with the SMTP service // Helpful when the TLS certificate is not in the certificate store of the server // Does carry the risk of man in the middle snooping client.ServerCertificateValidationCallback += (sender, certificate, chain, sslPolicyErrors) => true; } if (settings.DisableTLS) { // Does not attempt to use either TLS or SSL // Helpful when MailKit finds a TLS certificate, but it unable to use it client.Connect(settings.Host, settings.Port, MailKit.Security.SecureSocketOptions.None); } else { client.Connect(settings.Host, settings.Port); // Let MailKit figure out the correct SecureSocketOptions. } // Note: since we don't have an OAuth2 token, disable // the XOAUTH2 authentication mechanism. client.AuthenticationMechanisms.Remove("XOAUTH2"); if (settings.Authentication) { client.Authenticate(settings.Username, settings.Password); } _log.LogDebug("sending message to {0} \r\n from: {1}\r\n Are we authenticated: {2}", message.To, message.From, client.IsAuthenticated); await client.SendAsync(message); await client.DisconnectAsync(true); } } catch (Exception e) { _log.LogError(e, "Exception when attempting to send an email"); throw; } } public async Task Send(NotificationMessage model, EmailNotificationSettings settings) { EnsureArg.IsNotNullOrEmpty(settings.SenderAddress); EnsureArg.IsNotNullOrEmpty(model.To); EnsureArg.IsNotNullOrEmpty(model.Message); EnsureArg.IsNotNullOrEmpty(settings.Host); var body = new BodyBuilder { HtmlBody = model.Message, }; if (model.Other.ContainsKey("PlainTextBody")) { body.TextBody = model.Other["PlainTextBody"]; } var message = new MimeMessage { Body = body.ToMessageBody(), Subject = model.Subject }; message.To.Add(new MailboxAddress(model.To, model.To)); await Send(message, settings); } public async Task Send(MimeMessage message, EmailNotificationSettings settings) { try { message.From.Add(new MailboxAddress(string.IsNullOrEmpty(settings.SenderName) ? settings.SenderAddress : settings.SenderName, settings.SenderAddress)); using (var client = new SmtpClient()) { if (settings.DisableCertificateChecking) { client.ServerCertificateValidationCallback += (sender, certificate, chain, sslPolicyErrors) => true; } if (settings.DisableTLS) { client.Connect(settings.Host, settings.Port, MailKit.Security.SecureSocketOptions.None); } else { client.Connect(settings.Host, settings.Port); // Let MailKit figure out the correct SecureSocketOptions. } // Note: since we don't have an OAuth2 token, disable // the XOAUTH2 authentication mechanism. client.AuthenticationMechanisms.Remove("XOAUTH2"); if (settings.Authentication) { client.Authenticate(settings.Username, settings.Password); } //Log.Info("sending message to {0} \r\n from: {1}\r\n Are we authenticated: {2}", message.To, message.From, client.IsAuthenticated); await client.SendAsync(message); await client.DisconnectAsync(true); } } catch (Exception e) { _log.LogError(e, "Exception when attempting to send email"); throw; } } } }