2024-08-05 00:36:26 +02:00
|
|
|
|
using System;
|
|
|
|
|
|
using System.Linq;
|
|
|
|
|
|
using System.Threading;
|
|
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
|
|
using CommunityToolkit.Mvvm.Messaging;
|
|
|
|
|
|
using Wino.Core.Domain.Enums;
|
|
|
|
|
|
using Wino.Core.Domain.Interfaces;
|
|
|
|
|
|
using Wino.Core.Domain.Models.Server;
|
|
|
|
|
|
using Wino.Core.Domain.Models.Synchronization;
|
|
|
|
|
|
using Wino.Messaging.Server;
|
|
|
|
|
|
using Wino.Messaging.UI;
|
|
|
|
|
|
using Wino.Server.Core;
|
|
|
|
|
|
|
|
|
|
|
|
namespace Wino.Server.MessageHandlers
|
|
|
|
|
|
{
|
|
|
|
|
|
/// <summary>
|
2024-12-27 00:18:46 +01:00
|
|
|
|
/// Handler for NewMailSynchronizationRequested from the client.
|
2024-08-05 00:36:26 +02:00
|
|
|
|
/// </summary>
|
2024-12-27 00:18:46 +01:00
|
|
|
|
public class MailSynchronizationRequestHandler : ServerMessageHandler<NewMailSynchronizationRequested, MailSynchronizationResult>
|
2024-08-05 00:36:26 +02:00
|
|
|
|
{
|
2024-12-24 18:30:25 +01:00
|
|
|
|
public override WinoServerResponse<MailSynchronizationResult> FailureDefaultResponse(Exception ex)
|
|
|
|
|
|
=> WinoServerResponse<MailSynchronizationResult>.CreateErrorResponse(ex.Message);
|
2024-08-05 00:36:26 +02:00
|
|
|
|
|
|
|
|
|
|
private readonly ISynchronizerFactory _synchronizerFactory;
|
|
|
|
|
|
private readonly INotificationBuilder _notificationBuilder;
|
|
|
|
|
|
private readonly IFolderService _folderService;
|
|
|
|
|
|
|
2024-12-27 00:18:46 +01:00
|
|
|
|
public MailSynchronizationRequestHandler(ISynchronizerFactory synchronizerFactory,
|
2024-08-05 00:36:26 +02:00
|
|
|
|
INotificationBuilder notificationBuilder,
|
|
|
|
|
|
IFolderService folderService)
|
|
|
|
|
|
{
|
|
|
|
|
|
_synchronizerFactory = synchronizerFactory;
|
|
|
|
|
|
_notificationBuilder = notificationBuilder;
|
|
|
|
|
|
_folderService = folderService;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-12-27 00:18:46 +01:00
|
|
|
|
protected override async Task<WinoServerResponse<MailSynchronizationResult>> HandleAsync(NewMailSynchronizationRequested message, CancellationToken cancellationToken = default)
|
2024-08-05 00:36:26 +02:00
|
|
|
|
{
|
|
|
|
|
|
var synchronizer = await _synchronizerFactory.GetAccountSynchronizerAsync(message.Options.AccountId);
|
|
|
|
|
|
|
|
|
|
|
|
// 1. Don't send message for sync completion when we execute requests.
|
|
|
|
|
|
// People are usually interested in seeing the notification after they trigger the synchronization.
|
|
|
|
|
|
|
|
|
|
|
|
// 2. Don't send message for sync completion when we are synchronizing from the server.
|
|
|
|
|
|
// It happens very common and there is no need to send a message for each synchronization.
|
|
|
|
|
|
|
|
|
|
|
|
bool shouldReportSynchronizationResult =
|
2024-12-24 18:30:25 +01:00
|
|
|
|
message.Options.Type != MailSynchronizationType.ExecuteRequests &&
|
2025-02-15 12:53:32 +01:00
|
|
|
|
message.Options.Type != MailSynchronizationType.IMAPIdle &&
|
2024-08-05 00:36:26 +02:00
|
|
|
|
message.Source == SynchronizationSource.Client;
|
|
|
|
|
|
|
|
|
|
|
|
try
|
|
|
|
|
|
{
|
2024-12-24 18:30:25 +01:00
|
|
|
|
var synchronizationResult = await synchronizer.SynchronizeMailsAsync(message.Options, cancellationToken).ConfigureAwait(false);
|
2024-08-05 00:36:26 +02:00
|
|
|
|
|
2024-08-17 03:43:37 +02:00
|
|
|
|
if (synchronizationResult.DownloadedMessages?.Any() ?? false || !synchronizer.Account.Preferences.IsNotificationsEnabled)
|
2024-08-05 00:36:26 +02:00
|
|
|
|
{
|
|
|
|
|
|
var accountInboxFolder = await _folderService.GetSpecialFolderByAccountIdAsync(message.Options.AccountId, SpecialFolderType.Inbox);
|
|
|
|
|
|
|
|
|
|
|
|
if (accountInboxFolder != null)
|
|
|
|
|
|
{
|
|
|
|
|
|
await _notificationBuilder.CreateNotificationsAsync(accountInboxFolder.Id, synchronizationResult.DownloadedMessages);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
var isSynchronizationSucceeded = synchronizationResult.CompletedState == SynchronizationCompletedState.Success;
|
|
|
|
|
|
|
2025-02-15 12:53:32 +01:00
|
|
|
|
// IDLE requests might be canceled successfully.
|
|
|
|
|
|
if (message.Options.Type == MailSynchronizationType.IMAPIdle && synchronizationResult.CompletedState == SynchronizationCompletedState.Canceled)
|
|
|
|
|
|
{
|
|
|
|
|
|
isSynchronizationSucceeded = true;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-09-12 00:02:14 +02:00
|
|
|
|
// Update badge count of the notification task.
|
|
|
|
|
|
if (isSynchronizationSucceeded)
|
|
|
|
|
|
{
|
|
|
|
|
|
await _notificationBuilder.UpdateTaskbarIconBadgeAsync();
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-08-05 00:36:26 +02:00
|
|
|
|
if (shouldReportSynchronizationResult)
|
|
|
|
|
|
{
|
|
|
|
|
|
var completedMessage = new AccountSynchronizationCompleted(message.Options.AccountId,
|
|
|
|
|
|
isSynchronizationSucceeded ? SynchronizationCompletedState.Success : SynchronizationCompletedState.Failed,
|
|
|
|
|
|
message.Options.GroupedSynchronizationTrackingId);
|
|
|
|
|
|
|
|
|
|
|
|
WeakReferenceMessenger.Default.Send(completedMessage);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-12-24 18:30:25 +01:00
|
|
|
|
return WinoServerResponse<MailSynchronizationResult>.CreateSuccessResponse(synchronizationResult);
|
2024-08-05 00:36:26 +02:00
|
|
|
|
}
|
|
|
|
|
|
// TODO: Following cases might always be thrown from server. Handle them properly.
|
|
|
|
|
|
|
|
|
|
|
|
//catch (AuthenticationAttentionException)
|
|
|
|
|
|
//{
|
|
|
|
|
|
// // TODO
|
|
|
|
|
|
// // await SetAccountAttentionAsync(accountId, AccountAttentionReason.InvalidCredentials);
|
|
|
|
|
|
//}
|
|
|
|
|
|
//catch (SystemFolderConfigurationMissingException)
|
|
|
|
|
|
//{
|
|
|
|
|
|
// // TODO
|
|
|
|
|
|
// // await SetAccountAttentionAsync(accountId, AccountAttentionReason.MissingSystemFolderConfiguration);
|
|
|
|
|
|
//}
|
|
|
|
|
|
catch (Exception)
|
|
|
|
|
|
{
|
|
|
|
|
|
throw;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|