2022-01-24 19:25:46 +03:00
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
|
using Bit.App.Abstractions;
|
|
|
|
|
using Bit.App.Pages;
|
|
|
|
|
using Bit.App.Pages.Accounts;
|
|
|
|
|
using Bit.Core.Abstractions;
|
|
|
|
|
using Bit.Core.Enums;
|
|
|
|
|
using Bit.Core.Utilities;
|
|
|
|
|
using Xamarin.Forms;
|
|
|
|
|
|
|
|
|
|
namespace Bit.App.Utilities
|
|
|
|
|
{
|
|
|
|
|
public interface IVerificationActionsFlowHelper
|
|
|
|
|
{
|
|
|
|
|
IVerificationActionsFlowHelper Configure(VerificationFlowAction action,
|
|
|
|
|
IActionFlowParmeters parameters = null,
|
|
|
|
|
string verificatioCodeMainActionText = null,
|
|
|
|
|
bool isVerificationCodeMainActionStyleDanger = false);
|
|
|
|
|
IActionFlowParmeters GetParameters();
|
|
|
|
|
Task ValidateAndExecuteAsync();
|
|
|
|
|
Task ExecuteAsync(IActionFlowParmeters parameters);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public interface IActionFlowParmeters
|
|
|
|
|
{
|
2022-02-03 23:46:45 +03:00
|
|
|
|
VerificationType VerificationType { get; set; }
|
|
|
|
|
|
2022-01-24 19:25:46 +03:00
|
|
|
|
string Secret { get; set; }
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public class DefaultActionFlowParameters : IActionFlowParmeters
|
|
|
|
|
{
|
2022-02-03 23:46:45 +03:00
|
|
|
|
public VerificationType VerificationType { get; set; }
|
|
|
|
|
|
2022-01-24 19:25:46 +03:00
|
|
|
|
public string Secret { get; set; }
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public interface IActionFlowExecutioner
|
|
|
|
|
{
|
|
|
|
|
Task Execute(IActionFlowParmeters parameters);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public enum VerificationFlowAction
|
|
|
|
|
{
|
|
|
|
|
ExportVault,
|
|
|
|
|
DeleteAccount
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Verifies and execute actions on the corresponding order.
|
|
|
|
|
///
|
|
|
|
|
/// Use: From the caller
|
|
|
|
|
/// - Configure it on <see cref="Configure(VerificationFlowAction, IActionFlowParmeters)"/>
|
|
|
|
|
/// - Call <see cref="ValidateAndExecuteAsync"/>
|
|
|
|
|
///
|
|
|
|
|
/// For every <see cref="VerificationFlowAction"/> we need an implementation of <see cref="IActionFlowExecutioner"/>
|
|
|
|
|
/// and it to be configured in the inner dictionary.
|
|
|
|
|
/// Also, inherit from <see cref="DefaultActionFlowParameters"/> if more custom parameters are needed for the executioner.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public class VerificationActionsFlowHelper : IVerificationActionsFlowHelper
|
|
|
|
|
{
|
|
|
|
|
private readonly IKeyConnectorService _keyConnectorService;
|
|
|
|
|
private readonly IPasswordRepromptService _passwordRepromptService;
|
2022-02-03 23:46:45 +03:00
|
|
|
|
private readonly ICryptoService _cryptoService;
|
2022-01-24 19:25:46 +03:00
|
|
|
|
|
|
|
|
|
private VerificationFlowAction? _action;
|
|
|
|
|
private IActionFlowParmeters _parameters;
|
|
|
|
|
private string _verificationCodeMainActionText;
|
|
|
|
|
private bool _isVerificationCodeMainActionStyleDanger;
|
|
|
|
|
|
|
|
|
|
private readonly Dictionary<VerificationFlowAction, IActionFlowExecutioner> _actionExecutionerDictionary = new Dictionary<VerificationFlowAction, IActionFlowExecutioner>();
|
|
|
|
|
|
|
|
|
|
public VerificationActionsFlowHelper(IKeyConnectorService keyConnectorService,
|
2022-02-03 23:46:45 +03:00
|
|
|
|
IPasswordRepromptService passwordRepromptService,
|
|
|
|
|
ICryptoService cryptoService)
|
2022-01-24 19:25:46 +03:00
|
|
|
|
{
|
|
|
|
|
_keyConnectorService = keyConnectorService;
|
|
|
|
|
_passwordRepromptService = passwordRepromptService;
|
2022-02-03 23:46:45 +03:00
|
|
|
|
_cryptoService = cryptoService;
|
2022-01-24 19:25:46 +03:00
|
|
|
|
|
|
|
|
|
_actionExecutionerDictionary.Add(VerificationFlowAction.DeleteAccount, ServiceContainer.Resolve<IDeleteAccountActionFlowExecutioner>("deleteAccountActionFlowExecutioner"));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public IVerificationActionsFlowHelper Configure(VerificationFlowAction action,
|
|
|
|
|
IActionFlowParmeters parameters = null,
|
|
|
|
|
string verificationCodeMainActionText = null,
|
|
|
|
|
bool isVerificationCodeMainActionStyleDanger = false)
|
|
|
|
|
{
|
|
|
|
|
_action = action;
|
|
|
|
|
_parameters = parameters;
|
|
|
|
|
_verificationCodeMainActionText = verificationCodeMainActionText;
|
|
|
|
|
_isVerificationCodeMainActionStyleDanger = isVerificationCodeMainActionStyleDanger;
|
|
|
|
|
|
|
|
|
|
return this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public IActionFlowParmeters GetParameters()
|
|
|
|
|
{
|
|
|
|
|
if (_parameters is null)
|
|
|
|
|
{
|
|
|
|
|
_parameters = new DefaultActionFlowParameters();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return _parameters;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public async Task ValidateAndExecuteAsync()
|
|
|
|
|
{
|
|
|
|
|
var verificationType = await _keyConnectorService.GetUsesKeyConnector()
|
|
|
|
|
? VerificationType.OTP
|
|
|
|
|
: VerificationType.MasterPassword;
|
|
|
|
|
|
|
|
|
|
switch (verificationType)
|
|
|
|
|
{
|
|
|
|
|
case VerificationType.MasterPassword:
|
|
|
|
|
var (password, valid) = await _passwordRepromptService.ShowPasswordPromptAndGetItAsync();
|
|
|
|
|
if (!valid)
|
|
|
|
|
{
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2022-02-03 23:46:45 +03:00
|
|
|
|
var parameters = GetParameters();
|
|
|
|
|
parameters.Secret = await _cryptoService.HashPasswordAsync(password, null);
|
|
|
|
|
parameters.VerificationType = VerificationType.MasterPassword;
|
|
|
|
|
await ExecuteAsync(parameters);
|
2022-01-24 19:25:46 +03:00
|
|
|
|
break;
|
|
|
|
|
case VerificationType.OTP:
|
|
|
|
|
await Application.Current.MainPage.Navigation.PushModalAsync(new NavigationPage(
|
|
|
|
|
new VerificationCodePage(_verificationCodeMainActionText, _isVerificationCodeMainActionStyleDanger)));
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
throw new NotImplementedException($"There is no implementation for {verificationType}");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Executes the action with the given parameters after we have gotten the verification secret
|
|
|
|
|
/// </summary>
|
|
|
|
|
public async Task ExecuteAsync(IActionFlowParmeters parameters)
|
|
|
|
|
{
|
|
|
|
|
if (!_action.HasValue)
|
|
|
|
|
{
|
|
|
|
|
// this should never happen
|
|
|
|
|
throw new InvalidOperationException("A problem occurred while getting the action value after validation");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!_actionExecutionerDictionary.TryGetValue(_action.Value, out var executioner))
|
|
|
|
|
{
|
|
|
|
|
throw new InvalidOperationException($"There is no executioner for {_action}");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
await executioner.Execute(GetParameters());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|