* Separation of messages. Introducing Wino.Messages library. * Wino.Server and Wino.Packaging projects. Enabling full trust for UWP and app service connection manager basics. * Remove debug code. * Enable generating assembly info to deal with unsupported os platform warnings. * Fix server-client connection. * UIMessage communication. Single instancing for server and re-connection mechanism on suspension. * Removed IWinoSynchronizerFactory from UWP project. * Removal of background task service from core. * Delegating changes to UI and triggering new background synchronization. * Fix build error. * Moved core lib messages to Messaging project. * Better client-server communication. Handling of requests in the server. New synchronizer factory in the server. * WAM broker and MSAL token caching for OutlookAuthenticator. Handling account creation for Outlook. * WinoServerResponse basics. * Delegating protocol activation for Gmail authenticator. * Adding margin to searchbox to match action bar width. * Move libraries into lib folder. * Storing base64 encoded mime on draft creation instead of MimeMessage object. Fixes serialization/deserialization issue with S.T.Json * Scrollbar adjustments * WınoExpander for thread expander layout ıssue. * Handling synchronizer state changes. * Double init on background activation. * FIxing packaging issues and new Wino Mail launcher protocol for activation from full thrust process. * Remove debug deserialization. * Remove debug code. * Making sure the server connection is established when the app is launched. * Thrust -> Trust string replacement... * Rename package to Wino Mail * Enable translated values in the server. * Fixed an issue where toast activation can't find the clicked mail after the folder is initialized. * Revert debug code. * Change server background sync to every 3 minute and Inbox only synchronization. * Revert google auth changes. * App preferences page. * Changing tray icon visibility on preference change. * Start the server with invisible tray icon if set to invisible. * Reconnect button on the title bar. * Handling of toast actions. * Enable x86 build for server during packaging. * Get rid of old background tasks and v180 migration. * Terminate client when Exit clicked in server. * Introducing SynchronizationSource to prevent notifying UI after server tick synchronization. * Remove confirmAppClose restricted capability and unused debug code in manifest. * Closing the reconnect info popup when reconnect is clicked. * Custom RetryHandler for OutlookSynchronizer and separating client/server logs. * Running server on Windows startup. * Fix startup exe. * Fix for expander list view item paddings. * Force full sync on app launch instead of Inbox. * Fix draft creation. * Fix an issue with custom folder sync logic. * Reporting back account sync progress from server. * Fix sending drafts and missing notifications for imap. * Changing imap folder sync requirements. * Retain file count is set to 3. * Disabled swipe gestures temporarily due to native crash with SwipeControl * Save all attachments implementation. * Localization for save all attachments button. * Fix logging dates for logs. * Fixing ARM64 build. * Add ARM64 build config to packaging project. * Comment out OutOfProcPDB for ARM64. * Hnadling GONE response for Outlook folder synchronization.
272 lines
13 KiB
C#
272 lines
13 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using System.Threading.Tasks;
|
|
using Wino.Core.Domain;
|
|
using Wino.Core.Domain.Entities;
|
|
using Wino.Core.Domain.Enums;
|
|
using Wino.Core.Domain.Exceptions;
|
|
using Wino.Core.Domain.Interfaces;
|
|
using Wino.Core.Domain.Models.Folders;
|
|
using Wino.Core.Domain.Models.MailItem;
|
|
using Wino.Core.Requests;
|
|
|
|
namespace Wino.Core.Services
|
|
{
|
|
/// <summary>
|
|
/// Intermediary processor for converting a user action to executable Wino requests.
|
|
/// Primarily responsible for batching requests by AccountId and FolderId.
|
|
/// </summary>
|
|
public class WinoRequestProcessor : BaseDatabaseService, IWinoRequestProcessor
|
|
{
|
|
private readonly IFolderService _folderService;
|
|
private readonly IKeyPressService _keyPressService;
|
|
private readonly IPreferencesService _preferencesService;
|
|
private readonly IAccountService _accountService;
|
|
private readonly IDialogService _dialogService;
|
|
private readonly IMailService _mailService;
|
|
|
|
/// <summary>
|
|
/// Set of rules that defines which action should be executed if user wants to toggle an action.
|
|
/// </summary>
|
|
private readonly List<ToggleRequestRule> _toggleRequestRules =
|
|
[
|
|
new ToggleRequestRule(MailOperation.MarkAsRead, MailOperation.MarkAsUnread, new System.Func<IMailItem, bool>((item) => item.IsRead)),
|
|
new ToggleRequestRule(MailOperation.MarkAsUnread, MailOperation.MarkAsRead, new System.Func<IMailItem, bool>((item) => !item.IsRead)),
|
|
new ToggleRequestRule(MailOperation.SetFlag, MailOperation.ClearFlag, new System.Func<IMailItem, bool>((item) => item.IsFlagged)),
|
|
new ToggleRequestRule(MailOperation.ClearFlag, MailOperation.SetFlag, new System.Func<IMailItem, bool>((item) => !item.IsFlagged)),
|
|
];
|
|
|
|
public WinoRequestProcessor(IDatabaseService databaseService,
|
|
IFolderService folderService,
|
|
IKeyPressService keyPressService,
|
|
IPreferencesService preferencesService,
|
|
IAccountService accountService,
|
|
IDialogService dialogService,
|
|
IMailService mailService) : base(databaseService)
|
|
{
|
|
_folderService = folderService;
|
|
_keyPressService = keyPressService;
|
|
_preferencesService = preferencesService;
|
|
_accountService = accountService;
|
|
_dialogService = dialogService;
|
|
_mailService = mailService;
|
|
}
|
|
|
|
public async Task<List<IRequest>> PrepareRequestsAsync(MailOperationPreperationRequest preperationRequest)
|
|
{
|
|
var action = preperationRequest.Action;
|
|
var moveTargetStructure = preperationRequest.MoveTargetFolder;
|
|
|
|
// Ask confirmation for permanent delete operation.
|
|
// Drafts are always hard deleted without any protection.
|
|
|
|
if (!preperationRequest.IgnoreHardDeleteProtection && ((action == MailOperation.SoftDelete && _keyPressService.IsShiftKeyPressed()) || action == MailOperation.HardDelete))
|
|
{
|
|
if (_preferencesService.IsHardDeleteProtectionEnabled)
|
|
{
|
|
var shouldDelete = await _dialogService.ShowHardDeleteConfirmationAsync();
|
|
|
|
if (!shouldDelete) return default;
|
|
}
|
|
|
|
action = MailOperation.HardDelete;
|
|
}
|
|
|
|
// Make sure there is a move target folder if action is move.
|
|
// Let user pick a folder to move from the dialog.
|
|
|
|
if (action == MailOperation.Move && moveTargetStructure == null)
|
|
{
|
|
// TODO: Handle multiple accounts for move operation.
|
|
// What happens if we move 2 different mails from 2 different accounts?
|
|
|
|
var accountId = preperationRequest.MailItems.FirstOrDefault().AssignedAccount.Id;
|
|
|
|
moveTargetStructure = await _dialogService.PickFolderAsync(accountId, PickFolderReason.Move, _folderService);
|
|
|
|
if (moveTargetStructure == null)
|
|
return default;
|
|
}
|
|
|
|
var requests = new List<IRequest>();
|
|
|
|
// TODO: Fix: Collection was modified; enumeration operation may not execute
|
|
foreach (var item in preperationRequest.MailItems)
|
|
{
|
|
var singleRequest = await GetSingleRequestAsync(item, action, moveTargetStructure, preperationRequest.ToggleExecution);
|
|
|
|
if (singleRequest == null) continue;
|
|
|
|
requests.Add(singleRequest);
|
|
}
|
|
|
|
return requests;
|
|
}
|
|
|
|
private async Task<IRequest> GetSingleRequestAsync(MailCopy mailItem, MailOperation action, IMailItemFolder moveTargetStructure, bool shouldToggleActions)
|
|
{
|
|
if (mailItem.AssignedAccount == null) throw new ArgumentException(Translator.Exception_NullAssignedAccount);
|
|
if (mailItem.AssignedFolder == null) throw new ArgumentException(Translator.Exception_NullAssignedFolder);
|
|
|
|
// Rule: Soft deletes from Trash folder must perform Hard Delete.
|
|
if (action == MailOperation.SoftDelete && mailItem.AssignedFolder.SpecialFolderType == SpecialFolderType.Deleted)
|
|
action = MailOperation.HardDelete;
|
|
|
|
// Rule: SoftDelete draft items must be performed as hard delete.
|
|
if (action == MailOperation.SoftDelete && mailItem.IsDraft)
|
|
action = MailOperation.HardDelete;
|
|
|
|
// Rule: Soft/Hard deletes on local drafts are always discard local draft.
|
|
if ((action == MailOperation.SoftDelete || action == MailOperation.HardDelete) && mailItem.IsLocalDraft)
|
|
action = MailOperation.DiscardLocalDraft;
|
|
|
|
// Rule: Toggle actions must be reverted if ToggleExecution is passed true.
|
|
if (shouldToggleActions)
|
|
{
|
|
var toggleRule = _toggleRequestRules.Find(a => a.SourceAction == action);
|
|
|
|
if (toggleRule != null && toggleRule.Condition(mailItem))
|
|
{
|
|
action = toggleRule.TargetAction;
|
|
}
|
|
}
|
|
|
|
if (action == MailOperation.MarkAsRead)
|
|
return new MarkReadRequest(mailItem, true);
|
|
else if (action == MailOperation.MarkAsUnread)
|
|
return new MarkReadRequest(mailItem, false);
|
|
else if (action == MailOperation.SetFlag)
|
|
return new ChangeFlagRequest(mailItem, true);
|
|
else if (action == MailOperation.ClearFlag)
|
|
return new ChangeFlagRequest(mailItem, false);
|
|
else if (action == MailOperation.HardDelete)
|
|
return new DeleteRequest(mailItem);
|
|
else if (action == MailOperation.Move)
|
|
{
|
|
if (moveTargetStructure == null)
|
|
throw new InvalidMoveTargetException();
|
|
|
|
// TODO
|
|
// Rule: You can't move items to non-move target folders;
|
|
// Rule: You can't move items from a folder to itself.
|
|
|
|
//if (!moveTargetStructure.IsMoveTarget || moveTargetStructure.FolderId == mailItem.AssignedFolder.Id)
|
|
// throw new InvalidMoveTargetException();
|
|
|
|
var pickedFolderItem = await _folderService.GetFolderAsync(moveTargetStructure.Id);
|
|
|
|
return new MoveRequest(mailItem, mailItem.AssignedFolder, pickedFolderItem);
|
|
}
|
|
else if (action == MailOperation.Archive)
|
|
{
|
|
// For IMAP and Outlook: Validate archive folder exists.
|
|
// Gmail doesn't need archive folder existence.
|
|
|
|
MailItemFolder archiveFolder = null;
|
|
|
|
bool shouldRequireArchiveFolder = mailItem.AssignedAccount.ProviderType == MailProviderType.Outlook
|
|
|| mailItem.AssignedAccount.ProviderType == MailProviderType.IMAP4
|
|
|| mailItem.AssignedAccount.ProviderType == MailProviderType.Office365;
|
|
|
|
if (shouldRequireArchiveFolder)
|
|
{
|
|
archiveFolder = await _folderService.GetSpecialFolderByAccountIdAsync(mailItem.AssignedAccount.Id, SpecialFolderType.Archive)
|
|
?? throw new UnavailableSpecialFolderException(SpecialFolderType.Archive, mailItem.AssignedAccount.Id);
|
|
}
|
|
|
|
return new ArchiveRequest(true, mailItem, mailItem.AssignedFolder, archiveFolder);
|
|
}
|
|
else if (action == MailOperation.MarkAsNotJunk)
|
|
{
|
|
var inboxFolder = await _folderService.GetSpecialFolderByAccountIdAsync(mailItem.AssignedAccount.Id, SpecialFolderType.Inbox)
|
|
?? throw new UnavailableSpecialFolderException(SpecialFolderType.Inbox, mailItem.AssignedAccount.Id);
|
|
|
|
return new MoveRequest(mailItem, mailItem.AssignedFolder, inboxFolder);
|
|
}
|
|
else if (action == MailOperation.UnArchive)
|
|
{
|
|
var inboxFolder = await _folderService.GetSpecialFolderByAccountIdAsync(mailItem.AssignedAccount.Id, SpecialFolderType.Inbox)
|
|
?? throw new UnavailableSpecialFolderException(SpecialFolderType.Inbox, mailItem.AssignedAccount.Id);
|
|
|
|
return new ArchiveRequest(false, mailItem, mailItem.AssignedFolder, inboxFolder);
|
|
}
|
|
else if (action == MailOperation.SoftDelete)
|
|
{
|
|
var trashFolder = await _folderService.GetSpecialFolderByAccountIdAsync(mailItem.AssignedAccount.Id, SpecialFolderType.Deleted)
|
|
?? throw new UnavailableSpecialFolderException(SpecialFolderType.Deleted, mailItem.AssignedAccount.Id);
|
|
|
|
return new MoveRequest(mailItem, mailItem.AssignedFolder, trashFolder);
|
|
}
|
|
else if (action == MailOperation.MoveToJunk)
|
|
{
|
|
var junkFolder = await _folderService.GetSpecialFolderByAccountIdAsync(mailItem.AssignedAccount.Id, SpecialFolderType.Junk)
|
|
?? throw new UnavailableSpecialFolderException(SpecialFolderType.Junk, mailItem.AssignedAccount.Id);
|
|
|
|
return new MoveRequest(mailItem, mailItem.AssignedFolder, junkFolder);
|
|
}
|
|
else if (action == MailOperation.AlwaysMoveToFocused || action == MailOperation.AlwaysMoveToOther)
|
|
return new AlwaysMoveToRequest(mailItem, action == MailOperation.AlwaysMoveToFocused);
|
|
else if (action == MailOperation.DiscardLocalDraft)
|
|
await _mailService.DeleteMailAsync(mailItem.AssignedAccount.Id, mailItem.Id);
|
|
else
|
|
throw new NotSupportedException(string.Format(Translator.Exception_UnsupportedAction, action));
|
|
|
|
return null;
|
|
}
|
|
|
|
public async Task<IRequestBase> PrepareFolderRequestAsync(FolderOperationPreperationRequest request)
|
|
{
|
|
if (request == null || request.Folder == null) return default;
|
|
|
|
IRequestBase change = null;
|
|
|
|
var folder = request.Folder;
|
|
var operation = request.Action;
|
|
|
|
switch (request.Action)
|
|
{
|
|
case FolderOperation.Pin:
|
|
case FolderOperation.Unpin:
|
|
await _folderService.ChangeStickyStatusAsync(folder.Id, operation == FolderOperation.Pin);
|
|
break;
|
|
|
|
case FolderOperation.Rename:
|
|
var newFolderName = await _dialogService.ShowTextInputDialogAsync(folder.FolderName, Translator.DialogMessage_RenameFolderTitle, Translator.DialogMessage_RenameFolderMessage, Translator.FolderOperation_Rename);
|
|
|
|
if (!string.IsNullOrEmpty(newFolderName))
|
|
{
|
|
change = new RenameFolderRequest(folder, folder.FolderName, newFolderName);
|
|
}
|
|
|
|
break;
|
|
case FolderOperation.Empty:
|
|
var mailsToDelete = await _mailService.GetMailsByFolderIdAsync(folder.Id).ConfigureAwait(false);
|
|
|
|
change = new EmptyFolderRequest(folder, mailsToDelete);
|
|
|
|
break;
|
|
case FolderOperation.MarkAllAsRead:
|
|
|
|
var unreadItems = await _mailService.GetUnreadMailsByFolderIdAsync(folder.Id).ConfigureAwait(false);
|
|
|
|
if (unreadItems.Any())
|
|
change = new MarkFolderAsReadRequest(folder, unreadItems);
|
|
|
|
break;
|
|
//case FolderOperation.Delete:
|
|
// var isConfirmed = await _dialogService.ShowConfirmationDialogAsync($"'{folderStructure.FolderName}' is going to be deleted. Do you want to continue?", "Are you sure?", "Yes delete.");
|
|
|
|
// if (isConfirmed)
|
|
// change = new DeleteFolderRequest(accountId, folderStructure.RemoteFolderId, folderStructure.FolderId);
|
|
|
|
// break;
|
|
//default:
|
|
// throw new NotImplementedException();
|
|
}
|
|
|
|
return change;
|
|
}
|
|
}
|
|
}
|