DocSpace-buildtools/common/ASC.Core.Common/Notify/Senders/SmtpSender.cs

309 lines
11 KiB
C#
Raw Normal View History

2022-03-15 18:00:53 +00:00
// (c) Copyright Ascensio System SIA 2010-2022
//
// This program is a free software product.
// You can redistribute it and/or modify it under the terms
// of the GNU Affero General Public License (AGPL) version 3 as published by the Free Software
// Foundation. In accordance with Section 7(a) of the GNU AGPL its Section 15 shall be amended
// to the effect that Ascensio System SIA expressly excludes the warranty of non-infringement of
// any third-party rights.
//
// This program is distributed WITHOUT ANY WARRANTY, without even the implied warranty
// of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. For details, see
// the GNU AGPL at: http://www.gnu.org/licenses/agpl-3.0.html
//
// You can contact Ascensio System SIA at Lubanas st. 125a-25, Riga, Latvia, EU, LV-1021.
//
// The interactive user interfaces in modified source and object code versions of the Program must
// display Appropriate Legal Notices, as required under Section 5 of the GNU AGPL version 3.
//
// Pursuant to Section 7(b) of the License you must retain the original Product logo when
// distributing the program. Pursuant to Section 7(e) we decline to grant you any rights under
// trademark law for use of our trademarks.
//
// All the Product's GUI elements, including illustrations and icon sets, as well as technical writing
// content are licensed under the terms of the Creative Commons Attribution-ShareAlike 4.0
// International. See the License terms at http://creativecommons.org/licenses/by-sa/4.0/legalcode
2022-02-15 11:52:43 +00:00
namespace ASC.Core.Notify.Senders;
2022-04-01 16:17:21 +00:00
[Singletone]
2022-02-15 11:52:43 +00:00
public class SmtpSender : INotifySender
2020-10-19 15:53:15 +00:00
{
2022-02-15 11:52:43 +00:00
protected ILog Logger { get; set; }
protected readonly IConfiguration Configuration;
protected IServiceProvider ServiceProvider;
2022-02-15 11:52:43 +00:00
private string _host;
private int _port;
private bool _ssl;
private ICredentials _credentials;
protected bool UseCoreSettings;
2022-03-17 16:57:02 +00:00
const int _networkTimeout = 30000;
2022-02-15 11:52:43 +00:00
public SmtpSender(
2022-04-01 16:17:21 +00:00
IConfiguration configuration,
2022-02-15 11:52:43 +00:00
IServiceProvider serviceProvider,
IOptionsMonitor<ILog> options)
{
Logger = options.Get("ASC.Notify");
2022-04-01 16:17:21 +00:00
Configuration = configuration;
2022-02-15 11:52:43 +00:00
ServiceProvider = serviceProvider;
}
2019-05-15 14:56:09 +00:00
2022-02-15 11:52:43 +00:00
public virtual void Init(IDictionary<string, string> properties)
{
if (properties.ContainsKey("useCoreSettings") && bool.Parse(properties["useCoreSettings"]))
{
UseCoreSettings = true;
}
else
{
_host = properties["host"];
_port = properties.ContainsKey("port") ? int.Parse(properties["port"]) : 25;
_ssl = properties.ContainsKey("enableSsl") && bool.Parse(properties["enableSsl"]);
if (properties.TryGetValue("userName", out var property))
{
_credentials = new NetworkCredential(property, properties["password"]);
}
}
}
2022-02-15 11:52:43 +00:00
private void InitUseCoreSettings(CoreConfiguration configuration)
{
var s = configuration.SmtpSettings;
2019-05-15 14:56:09 +00:00
2022-02-15 11:52:43 +00:00
_host = s.Host;
_port = s.Port;
_ssl = s.EnableSSL;
_credentials = !string.IsNullOrEmpty(s.CredentialsUserName)
? new NetworkCredential(s.CredentialsUserName, s.CredentialsUserPassword)
: null;
}
2022-02-15 11:52:43 +00:00
public virtual NoticeSendResult Send(NotifyMessage m)
{
using var scope = ServiceProvider.CreateScope();
2022-04-01 16:17:21 +00:00
var tenantManager = scope.ServiceProvider.GetService<TenantManager>();
tenantManager.SetCurrentTenant(m.TenantId);
2022-04-01 16:17:21 +00:00
var configuration = scope.ServiceProvider.GetService<CoreConfiguration>();
2022-02-15 11:52:43 +00:00
var smtpClient = GetSmtpClient();
var result = NoticeSendResult.TryOnceAgain;
try
{
try
{
if (UseCoreSettings)
{
InitUseCoreSettings(configuration);
}
2022-02-15 11:52:43 +00:00
var mail = BuildMailMessage(m);
2022-02-15 11:52:43 +00:00
Logger.DebugFormat("SmtpSender - host={0}; port={1}; enableSsl={2} enableAuth={3}", _host, _port, _ssl, _credentials != null);
2022-02-15 11:52:43 +00:00
smtpClient.Connect(_host, _port,
_ssl ? SecureSocketOptions.Auto : SecureSocketOptions.None);
2022-02-15 11:52:43 +00:00
if (_credentials != null)
{
smtpClient.Authenticate(_credentials);
}
2022-02-15 11:52:43 +00:00
smtpClient.Send(mail);
result = NoticeSendResult.OK;
}
catch (Exception e)
{
Logger.ErrorFormat("Tenant: {0}, To: {1} - {2}", m.TenantId, m.Reciever, e);
2022-02-15 11:52:43 +00:00
throw;
}
}
catch (ObjectDisposedException)
{
result = NoticeSendResult.SendingImpossible;
}
catch (InvalidOperationException)
{
result = string.IsNullOrEmpty(_host) || _port == 0
? NoticeSendResult.SendingImpossible
: NoticeSendResult.TryOnceAgain;
}
catch (IOException)
{
result = NoticeSendResult.TryOnceAgain;
}
catch (MailKit.Net.Smtp.SmtpProtocolException)
{
result = NoticeSendResult.SendingImpossible;
}
catch (MailKit.Net.Smtp.SmtpCommandException e)
{
switch (e.StatusCode)
{
case MailKit.Net.Smtp.SmtpStatusCode.MailboxBusy:
case MailKit.Net.Smtp.SmtpStatusCode.MailboxUnavailable:
case MailKit.Net.Smtp.SmtpStatusCode.ExceededStorageAllocation:
result = NoticeSendResult.TryOnceAgain;
break;
case MailKit.Net.Smtp.SmtpStatusCode.MailboxNameNotAllowed:
case MailKit.Net.Smtp.SmtpStatusCode.UserNotLocalWillForward:
case MailKit.Net.Smtp.SmtpStatusCode.UserNotLocalTryAlternatePath:
result = NoticeSendResult.MessageIncorrect;
break;
default:
if (e.StatusCode != MailKit.Net.Smtp.SmtpStatusCode.Ok)
{
2022-02-15 11:52:43 +00:00
result = NoticeSendResult.TryOnceAgain;
}
2022-02-15 11:52:43 +00:00
break;
}
}
catch (Exception)
{
result = NoticeSendResult.SendingImpossible;
}
finally
{
if (smtpClient.IsConnected)
{
smtpClient.Disconnect(true);
}
2019-05-15 14:56:09 +00:00
2022-02-15 11:52:43 +00:00
smtpClient.Dispose();
}
return result;
}
2022-02-15 11:52:43 +00:00
private MimeMessage BuildMailMessage(NotifyMessage m)
{
var mimeMessage = new MimeMessage
{
Subject = m.Subject
};
var fromAddress = MailboxAddress.Parse(ParserOptions.Default, m.Sender);
2022-02-15 11:52:43 +00:00
mimeMessage.From.Add(fromAddress);
foreach (var to in m.Reciever.Split(new[] { '|' }, StringSplitOptions.RemoveEmptyEntries))
2022-02-15 11:52:43 +00:00
{
mimeMessage.To.Add(MailboxAddress.Parse(ParserOptions.Default, to));
}
2022-02-15 11:52:43 +00:00
if (m.ContentType == Pattern.HtmlContentType)
{
var textPart = new TextPart("plain")
{
Text = HtmlUtil.GetText(m.Content),
ContentTransferEncoding = ContentEncoding.QuotedPrintable
};
2022-02-15 11:52:43 +00:00
var multipartAlternative = new MultipartAlternative { textPart };
2022-02-15 11:52:43 +00:00
var htmlPart = new TextPart("html")
{
Text = GetHtmlView(m.Content),
ContentTransferEncoding = ContentEncoding.QuotedPrintable
};
if (m.Attachments != null && m.Attachments.Count > 0)
2022-02-15 11:52:43 +00:00
{
var multipartRelated = new MultipartRelated
{
Root = htmlPart
};
foreach (var attachment in m.Attachments)
{
2022-02-15 11:52:43 +00:00
var mimeEntity = ConvertAttachmentToMimePart(attachment);
if (mimeEntity != null)
{
multipartRelated.Add(mimeEntity);
}
}
2022-02-15 11:52:43 +00:00
multipartAlternative.Add(multipartRelated);
}
2022-02-15 11:52:43 +00:00
else
{
multipartAlternative.Add(htmlPart);
}
2022-02-15 11:52:43 +00:00
mimeMessage.Body = multipartAlternative;
}
else
{
mimeMessage.Body = new TextPart("plain")
{
Text = m.Content,
ContentTransferEncoding = ContentEncoding.QuotedPrintable
};
}
2022-02-15 11:52:43 +00:00
if (!string.IsNullOrEmpty(m.ReplyTo))
{
mimeMessage.ReplyTo.Add(MailboxAddress.Parse(ParserOptions.Default, m.ReplyTo));
}
mimeMessage.Headers.Add("Auto-Submitted", string.IsNullOrEmpty(m.AutoSubmitted) ? "auto-generated" : m.AutoSubmitted);
2022-02-15 11:52:43 +00:00
return mimeMessage;
}
2022-02-15 11:52:43 +00:00
protected string GetHtmlView(string body)
{
return $@"<!DOCTYPE html PUBLIC ""-//W3C//DTD HTML 4.01 Transitional//EN"">
<html>
<head>
<meta content=""text/html;charset=UTF-8"" http-equiv=""Content-Type"">
</head>
<body>{body}</body>
2022-02-15 11:52:43 +00:00
</html>";
}
2022-02-15 11:52:43 +00:00
private MailKit.Net.Smtp.SmtpClient GetSmtpClient()
2020-08-24 18:41:06 +00:00
{
2022-02-15 11:52:43 +00:00
var smtpClient = new MailKit.Net.Smtp.SmtpClient
{
2022-03-17 16:57:02 +00:00
Timeout = _networkTimeout
2022-02-15 11:52:43 +00:00
};
2022-02-15 11:52:43 +00:00
return smtpClient;
}
2022-02-15 11:52:43 +00:00
private static MimePart ConvertAttachmentToMimePart(NotifyMessageAttachment attachment)
{
try
2020-07-28 14:24:21 +00:00
{
2022-02-15 11:52:43 +00:00
if (attachment == null || string.IsNullOrEmpty(attachment.FileName) || string.IsNullOrEmpty(attachment.ContentId) || attachment.Content == null)
{
return null;
}
2022-02-15 11:52:43 +00:00
var extension = Path.GetExtension(attachment.FileName);
2022-02-15 11:52:43 +00:00
if (string.IsNullOrEmpty(extension))
{
return null;
}
2022-02-15 11:52:43 +00:00
return new MimePart("image", extension.TrimStart('.'))
{
ContentId = attachment.ContentId,
Content = new MimeContent(new MemoryStream(attachment.Content.ToByteArray())),
ContentDisposition = new ContentDisposition(ContentDisposition.Attachment),
ContentTransferEncoding = ContentEncoding.Base64,
FileName = attachment.FileName
};
}
catch (Exception)
2020-09-30 14:47:42 +00:00
{
2022-02-15 11:52:43 +00:00
return null;
2020-09-30 14:47:42 +00:00
}
2022-02-15 11:52:43 +00:00
}
}