From e881f9b3efe92d3d697358e53c3ab912a43a59b7 Mon Sep 17 00:00:00 2001 From: Daniel Collingwood <82693586+danzuep@users.noreply.github.com> Date: Sun, 11 Aug 2024 00:49:50 +0800 Subject: [PATCH] #72 added SocketException reconnect retry --- .../EmailWorkerOptions.cs | 13 +- samples/WorkerServiceExample/Program.cs | 6 +- samples/WorkerServiceExample/Worker.cs | 10 +- .../Services/MailFolderMonitor.cs | 123 +++++++++++------- 4 files changed, 88 insertions(+), 64 deletions(-) diff --git a/samples/WorkerServiceExample/EmailWorkerOptions.cs b/samples/WorkerServiceExample/EmailWorkerOptions.cs index 1447a4c..a03a36e 100644 --- a/samples/WorkerServiceExample/EmailWorkerOptions.cs +++ b/samples/WorkerServiceExample/EmailWorkerOptions.cs @@ -1,10 +1,9 @@ -namespace MailKitSimplified.Sender.Models +namespace ExampleNamespace; + +public class EmailWorkerOptions { - public class EmailWorkerOptions - { - public const string SectionName = "EmailWorker"; + public const string SectionName = "EmailWorker"; - public string DefaultFromAddress { get; set; } = "noreply@localhost"; - public string DefaultToAddress { get; set; } = "noreply@localhost"; - } + public string DefaultFromAddress { get; set; } = "noreply@localhost"; + public string DefaultToAddress { get; set; } = "noreply@localhost"; } diff --git a/samples/WorkerServiceExample/Program.cs b/samples/WorkerServiceExample/Program.cs index e7bd9b7..01e55b9 100644 --- a/samples/WorkerServiceExample/Program.cs +++ b/samples/WorkerServiceExample/Program.cs @@ -1,11 +1,11 @@ using MailKitSimplified.Sender; using MailKitSimplified.Receiver; -using MailKitSimplified.Sender.Models; +using ExampleNamespace; -IHost host = Host.CreateDefaultBuilder(args) +using var host = Host.CreateDefaultBuilder(args) .ConfigureServices((context, services) => { - services.AddHostedService(); + services.AddHostedService(); //services.AddMailKitSimplifiedEmail(context.Configuration); services.AddScopedMailKitSimplifiedEmailSender(context.Configuration); services.AddScopedMailKitSimplifiedEmailReceiver(context.Configuration); diff --git a/samples/WorkerServiceExample/Worker.cs b/samples/WorkerServiceExample/Worker.cs index bd8cc78..294e524 100644 --- a/samples/WorkerServiceExample/Worker.cs +++ b/samples/WorkerServiceExample/Worker.cs @@ -8,7 +8,6 @@ using MailKitSimplified.Receiver.Extensions; using MailKitSimplified.Receiver.Services; using MailKitSimplified.Sender.Abstractions; -using MailKitSimplified.Sender.Models; using Microsoft.Extensions.Options; namespace ExampleNamespace; @@ -48,9 +47,9 @@ protected override async Task ExecuteAsync(CancellationToken cancellationToken = //await AddFlagsToNewestMessageSummaryAsync(cancellationToken); //await GetMailFolderCacheAsync(); //await CreateFolderAndMoveTopOneAsync(); - //await MonitorAsync(cancellationToken); + await MonitorAsync(cancellationToken); //await MoveToAsync(_processed, 1, cancellationToken); - await MonitorMoveAsync(cancellationToken); + //await MonitorMoveAsync(cancellationToken); } private static ImapReceiver CreateExchangeOAuth2ImapClientExample(SaslMechanismOAuth2 oauth2) @@ -494,17 +493,16 @@ async Task ProcessMessageAsync(IMessageSummary messageSummary) private async Task MonitorAsync(CancellationToken cancellationToken = default) { using var smtpSender = _serviceScope.ServiceProvider.GetRequiredService(); - var sendTask = DelayedSendAsync(TimeSpan.FromMilliseconds(500), smtpSender, cancellationToken); void ProcessMessage(IMessageSummary messageSummary) => _logger.LogInformation($"{_imapReceiver} message #{messageSummary.UniqueId} processed."); + var sendTask = DelayedSendAsync(TimeSpan.FromSeconds(3), smtpSender, cancellationToken); await _imapReceiver.MonitorFolder .SetMessageSummaryItems() - .SetIgnoreExistingMailOnConnect() + .SetIgnoreExistingMailOnConnect() .OnMessageArrival(ProcessMessage) .OnMessageDeparture(ProcessMessage) .IdleAsync(cancellationToken); await sendTask; - _logger.LogInformation($"{_imapReceiver} Monitoring complete."); } public override void Dispose() diff --git a/source/MailKitSimplified.Receiver/Services/MailFolderMonitor.cs b/source/MailKitSimplified.Receiver/Services/MailFolderMonitor.cs index 726143c..867dbcc 100644 --- a/source/MailKitSimplified.Receiver/Services/MailFolderMonitor.cs +++ b/source/MailKitSimplified.Receiver/Services/MailFolderMonitor.cs @@ -48,12 +48,12 @@ public MailFolderMonitor(IImapReceiver imapReceiver, IOptions { - _logger.LogInformation($"{_imapReceiver} message #{m.UniqueId} arrival processed."); + _logger.Log($"{_imapReceiver} message #{m.UniqueId} arrival processed."); return _completedTask; }; _messageDepartureMethod = (m) => { - _logger.LogInformation($"{_imapReceiver} message #{m.UniqueId} departure processed."); + _logger.Log($"{_imapReceiver} message #{m.UniqueId} departure processed."); return _completedTask; }; } @@ -181,30 +181,32 @@ public IMailFolderMonitor OnMessageDeparture(Action messageDepa public async Task IdleAsync(CancellationToken cancellationToken = default) { + _logger.Log($"{_imapReceiver} monitoring requested.", LogLevel.Trace); _cancel = CancellationTokenSource.CreateLinkedTokenSource(cancellationToken); try { var tasks = new Task[] { IdleStartAsync(_cancel.Token).ContinueWith(t => - _logger.LogError(t.Exception?.InnerException ?? t.Exception, + _logger.Log(t.Exception?.InnerException ?? t.Exception, "Idle client failed."), TaskContinuationOptions.OnlyOnFaulted), ProcessArrivalQueueAsync(_messageArrivalMethod, _cancel.Token).ContinueWith(t => - _logger.LogError(t.Exception?.InnerException ?? t.Exception, + _logger.Log(t.Exception?.InnerException ?? t.Exception, "Arrival queue processing failed."), TaskContinuationOptions.OnlyOnFaulted), ProcessDepartureQueueAsync(_messageDepartureMethod, _cancel.Token).ContinueWith(t => - _logger.LogError(t.Exception?.InnerException ?? t.Exception, + _logger.Log(t.Exception?.InnerException ?? t.Exception, "Departure queue processing failed."), TaskContinuationOptions.OnlyOnFaulted) }; await Task.WhenAll(tasks).ConfigureAwait(false); + _logger.Log($"{_imapReceiver} monitoring complete.", LogLevel.Information); } catch (OperationCanceledException) { - _logger.LogTrace($"{_imapReceiver} email monitoring cancelled."); + _logger.Log($"{_imapReceiver} email monitoring cancelled.", LogLevel.Trace); } catch (Exception ex) { - _logger.LogError(ex, $"{_imapReceiver} email monitoring failed."); + _logger.Log(ex, $"{_imapReceiver} email monitoring failed."); } } @@ -222,7 +224,7 @@ private async Task IdleStartAsync(CancellationToken cancellationToken = default) _mailFolder = await _imapReceiver.ConnectMailFolderAsync(cancellationToken).ConfigureAwait(false); _ = await _mailFolder.OpenAsync(FolderAccess.ReadOnly, cancellationToken).ConfigureAwait(false); var connectOption = _folderMonitorOptions.IgnoreExistingMailOnConnect ? "ignoring" : "fetching"; - _logger.LogInformation($"{_imapReceiver} ({_mailFolder.Count}) idle monitor started, {connectOption} existing emails."); + _logger.Log($"{_imapReceiver} ({_mailFolder.Count}) idle monitor started, {connectOption} existing emails."); _mailFolder.CountChanged += OnCountChanged; _mailFolder.MessageExpunged += OnMessageExpunged; @@ -233,15 +235,15 @@ private async Task IdleStartAsync(CancellationToken cancellationToken = default) } catch (OperationCanceledException) { - _logger.LogDebug("Initial fetch or idle task in mail folder monitor was cancelled."); + _logger.Log("Initial fetch or idle task in mail folder monitor was cancelled.", LogLevel.Debug); } catch (AuthenticationException ex) { - _logger.LogWarning(ex, "Stopping mail folder monitor service."); + _logger.Log(ex, "Stopping mail folder monitor service.", LogLevel.Warning); } catch (InvalidOperationException ex) { - _logger.LogError(ex, "IMAP client not available."); + _logger.Log(ex, "IMAP client not available.", LogLevel.Error); } finally { @@ -252,7 +254,7 @@ private async Task IdleStartAsync(CancellationToken cancellationToken = default) private void Disconnect(bool throwOnFirstException) { - _logger.LogTrace("Disconnecting IMAP idle client..."); + _logger.Log("Disconnecting IMAP idle client...", LogLevel.Trace); if (_mailFolder != null) { _mailFolder.MessageExpunged -= OnMessageExpunged; @@ -269,21 +271,45 @@ private void Disconnect(bool throwOnFirstException) _cancel?.Dispose(); } + /// Failed to authenticate private async ValueTask ReconnectAsync(CancellationToken cancellationToken = default) { - if (!cancellationToken.IsCancellationRequested) + int attemptCount = 0; + while (!cancellationToken.IsCancellationRequested && attemptCount < _folderMonitorOptions.MaxRetries) { - _ = await _imapReceiver.ConnectAuthenticatedImapClientAsync(cancellationToken).ConfigureAwait(false); - if (!_mailFolder.IsOpen) + try { - _ = await _mailFolder.OpenAsync(FolderAccess.ReadOnly, cancellationToken).ConfigureAwait(false); - _logger.LogTrace($"{_mailFolder.FullName} mail folder re-opened with ReadOnly access."); + _ = await _imapReceiver.ConnectAuthenticatedImapClientAsync(cancellationToken).ConfigureAwait(false); + if (!_mailFolder.IsOpen) + { + _ = await _mailFolder.OpenAsync(FolderAccess.ReadOnly, cancellationToken).ConfigureAwait(false); + _logger.Log($"{_mailFolder.FullName} mail folder re-opened with ReadOnly access.", LogLevel.Trace); + } + _ = await _fetchReceiver.ConnectAuthenticatedImapClientAsync(cancellationToken).ConfigureAwait(false); + if (!_fetchFolder.IsOpen) + { + _ = await _fetchFolder.OpenAsync(FolderAccess.ReadWrite, cancellationToken).ConfigureAwait(false); + _logger.Log($"{_fetchFolder.FullName} mail folder re-opened with ReadWrite access.", LogLevel.Trace); + } + break; } - _ = await _fetchReceiver.ConnectAuthenticatedImapClientAsync(cancellationToken).ConfigureAwait(false); - if (!_fetchFolder.IsOpen) + catch (ImapProtocolException ex) { - _ = await _fetchFolder.OpenAsync(FolderAccess.ReadWrite, cancellationToken).ConfigureAwait(false); - _logger.LogTrace($"{_fetchFolder.FullName} mail folder re-opened with ReadWrite access."); + await LogDelayAsync(ex, "IMAP protocol exception").ConfigureAwait(false); + } + catch (SocketException ex) + { + await LogDelayAsync(ex, "IMAP socket exception").ConfigureAwait(false); + } + + async Task LogDelayAsync(Exception exception, string exceptionType) + { + var message = $"{exceptionType} during connection attempt #{++attemptCount}, backing off for {_folderMonitorOptions.EmptyQueueMaxDelayMs}ms. {_imapReceiver}."; + if (attemptCount < _folderMonitorOptions.MaxRetries) + _logger.Log(exceptionType, LogLevel.Warning); + else + _logger.Log(exception, exceptionType, LogLevel.Error); + await Task.Delay(_folderMonitorOptions.EmptyQueueMaxDelayMs, cancellationToken).ConfigureAwait(false); } } } @@ -313,8 +339,9 @@ private async ValueTask WaitForNewMessagesAsync(CancellationToken cancellationTo } catch (OperationCanceledException) // includes TaskCanceledException { - _logger.LogTrace($"{_imapReceiver} mail folder idle wait task cancelled."); + _logger.Log($"Mail folder idle wait task cancelled. {_imapReceiver}.", LogLevel.Trace); _cancel.Cancel(false); + break; } catch (ImapProtocolException ex) { @@ -323,41 +350,41 @@ private async ValueTask WaitForNewMessagesAsync(CancellationToken cancellationTo _logger.Log(message, LogLevel.Debug); else _logger.Log(message, LogLevel.Information); - await ReconnectAsync(cancellationToken).ConfigureAwait(false); if (_folderMonitorOptions.IdleMinutes > FolderMonitorOptions.IdleMinutesGmail) _folderMonitorOptions.IdleMinutes = FolderMonitorOptions.IdleMinutesGmail; else if (_folderMonitorOptions.IdleMinutes == FolderMonitorOptions.IdleMinutesGmail) _folderMonitorOptions.IdleMinutes = 1; } - catch (ImapCommandException ex) + catch (ImapCommandException) { - _logger.LogInformation(ex, "IMAP command exception, rechecking server connection."); - await ReconnectAsync(cancellationToken).ConfigureAwait(false); + _logger.Log("IMAP command exception, rechecking server connection."); } catch (IOException) { - _logger.LogInformation("IMAP I/O exception, reconnecting."); - await ReconnectAsync(cancellationToken).ConfigureAwait(false); + _logger.Log("IMAP I/O exception, reconnecting."); } - catch (SocketException ex) + catch (SocketException) { - _logger.LogInformation(ex, "IMAP socket exception."); - await ReconnectAsync(cancellationToken).ConfigureAwait(false); + _logger.Log("IMAP socket exception, reconnecting."); } catch (ServiceNotConnectedException) { - _logger.LogInformation("IMAP service not connected, reconnecting."); - await ReconnectAsync(cancellationToken).ConfigureAwait(false); + _logger.Log("IMAP service not connected, reconnecting."); } catch (ServiceNotAuthenticatedException) { - _logger.LogInformation("IMAP service not authenticated, authenticating."); - await ReconnectAsync(cancellationToken).ConfigureAwait(false); + _logger.Log("IMAP service not authenticated, authenticating."); } catch (InvalidOperationException ex) { - _logger.LogError(ex, "IMAP client is being accessed by multiple threads."); + _logger.Log(ex, $"IMAP client is being accessed by multiple threads. {_imapReceiver}."); _cancel.Cancel(false); + break; + } + finally + { + if (!_cancel.IsCancellationRequested) + await ReconnectAsync(cancellationToken).ConfigureAwait(false); } } while (!cancellationToken.IsCancellationRequested && retryCount < _folderMonitorOptions.MaxRetries); } @@ -366,10 +393,10 @@ private async ValueTask ProcessMessagesArrivedAsync(bool firstConnection = { int startIndex = _messageCache.Count; await _fetchClient.NoOpAsync(cancellationToken).ConfigureAwait(false); - _logger.LogTrace($"{_fetchReceiver} ({_fetchFolder.Count}) fetching new message arrivals, starting from {startIndex}."); + _logger.Log($"{_fetchReceiver} ({_fetchFolder.Count}) fetching new message arrivals, starting from {startIndex}.", LogLevel.Trace); if (startIndex > _fetchFolder.Count) { - _logger.LogTrace($"{_fetchReceiver} start index {startIndex} is higher than fetched folder count of {_fetchFolder.Count}, monitored count is {_mailFolder.Count}."); + _logger.Log($"{_fetchReceiver} start index {startIndex} is higher than fetched folder count of {_fetchFolder.Count}, monitored count is {_mailFolder.Count}.", LogLevel.Trace); startIndex = _fetchFolder.Count; } var filter = _folderMonitorOptions.MessageSummaryItems | MessageSummaryItems.UniqueId; @@ -384,7 +411,7 @@ private async ValueTask ProcessMessagesArrivedAsync(bool firstConnection = { if (cancellationToken.IsCancellationRequested) break; - _logger.LogDebug($"{_imapReceiver} message #{mailItem.UniqueId} arrived."); + _logger.Log($"{_imapReceiver} message #{mailItem.UniqueId} arrived.", LogLevel.Debug); _arrivalQueue.Enqueue(mailItem); } } @@ -410,12 +437,12 @@ private async Task ProcessArrivalQueueAsync(Func messageA } catch (OperationCanceledException) { - _logger.LogTrace("Arrival queue cancelled."); + _logger.Log("Arrival queue cancelled.", LogLevel.Trace); break; } catch (Exception ex) { - _logger.LogWarning(ex, $"Error occurred processing arrival queue item, backing off for {_folderMonitorOptions.EmptyQueueMaxDelayMs}ms. {_imapReceiver} #{messageSummary.UniqueId}."); + _logger.Log(ex, $"Error occurred processing arrival queue item during attempt #{retryCount}, backing off for {_folderMonitorOptions.EmptyQueueMaxDelayMs}ms. {_imapReceiver} #{messageSummary.UniqueId}.", LogLevel.Warning); if (messageSummary != null) _arrivalQueue.Enqueue(messageSummary); await Task.Delay(_folderMonitorOptions.EmptyQueueMaxDelayMs, cancellationToken).ConfigureAwait(false); @@ -444,12 +471,12 @@ private async Task ProcessDepartureQueueAsync(Func messag } catch (OperationCanceledException) { - _logger.LogTrace("Departure queue cancelled."); + _logger.Log("Departure queue cancelled.", LogLevel.Trace); break; } catch (Exception ex) { - _logger.LogWarning(ex, $"Error occurred processing departure queue item, backing off for {_folderMonitorOptions.EmptyQueueMaxDelayMs}ms. {_imapReceiver} #{messageSummary.UniqueId}."); + _logger.Log(ex, $"Error occurred processing departure queue item during attempt #{retryCount}, backing off for {_folderMonitorOptions.EmptyQueueMaxDelayMs}ms. {_imapReceiver} #{messageSummary.UniqueId}.", LogLevel.Warning); if (messageSummary != null) _departureQueue.Enqueue(messageSummary); await Task.Delay(_folderMonitorOptions.EmptyQueueMaxDelayMs, cancellationToken).ConfigureAwait(false); @@ -474,16 +501,16 @@ private void OnCountChanged(object sender, EventArgs e) if (changeCount > 0) { _arrival?.Cancel(true); - _logger.LogTrace($"{_imapReceiver} message count increased by {changeCount} ({previousCount} to {presentCount})."); + _logger.Log($"{_imapReceiver} message count increased by {changeCount} ({previousCount} to {presentCount}).", LogLevel.Trace); } else if (changeCount < 0) { - _logger.LogTrace($"{_imapReceiver} message count decreased by {changeCount} ({previousCount} to {presentCount})."); + _logger.Log($"{_imapReceiver} message count decreased by {changeCount} ({previousCount} to {presentCount}).", LogLevel.Trace); } } else { - _logger.LogWarning($"{_imapReceiver} message count changed ({_messageCache.Count} to {_mailFolder.Count}), folder unknown."); + _logger.Log($"{_imapReceiver} message count changed ({_messageCache.Count} to {_mailFolder.Count}), folder unknown.", LogLevel.Warning); } } } @@ -507,10 +534,10 @@ private void OnMessageExpunged(object sender, MessageEventArgs e) _messageCache.RemoveAt(index); } _departureQueue.Enqueue(messageSummary); - _logger.LogTrace($"{_imapReceiver}[{index}] (count={cachedCount}) expunged, item #{messageSummary.UniqueId}."); + _logger.Log($"{_imapReceiver}[{index}] (count={cachedCount}) expunged, item #{messageSummary.UniqueId}.", LogLevel.Trace); } else - _logger.LogWarning($"{_imapReceiver}[{index}] (count={cachedCount}) was out of range."); + _logger.Log($"{_imapReceiver}[{index}] (count={cachedCount}) was out of range.", LogLevel.Warning); } }