2019-04-09 23:13:32 +03:00
|
|
|
|
using Bit.Core.Abstractions;
|
|
|
|
|
using Bit.Core.Enums;
|
|
|
|
|
using Bit.Core.Models.Data;
|
|
|
|
|
using Bit.Core.Models.Domain;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.Linq;
|
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
|
|
|
|
|
|
namespace Bit.Core.Services
|
|
|
|
|
{
|
|
|
|
|
public class UserService : IUserService
|
|
|
|
|
{
|
|
|
|
|
private string _userId;
|
|
|
|
|
private string _email;
|
|
|
|
|
private string _stamp;
|
|
|
|
|
private KdfType? _kdf;
|
|
|
|
|
private int? _kdfIterations;
|
2021-04-15 07:54:58 +03:00
|
|
|
|
private bool? _emailVerified;
|
2021-09-24 21:14:26 +03:00
|
|
|
|
private bool? _forcePasswordReset;
|
2019-04-09 23:13:32 +03:00
|
|
|
|
|
|
|
|
|
private const string Keys_UserId = "userId";
|
|
|
|
|
private const string Keys_UserEmail = "userEmail";
|
|
|
|
|
private const string Keys_Stamp = "securityStamp";
|
|
|
|
|
private const string Keys_Kdf = "kdf";
|
|
|
|
|
private const string Keys_KdfIterations = "kdfIterations";
|
|
|
|
|
private const string Keys_OrganizationsFormat = "organizations_{0}";
|
2021-04-15 07:54:58 +03:00
|
|
|
|
private const string Keys_EmailVerified = "emailVerified";
|
2021-09-24 21:14:26 +03:00
|
|
|
|
private const string Keys_ForcePasswordReset = "forcePasswordReset";
|
2019-04-09 23:13:32 +03:00
|
|
|
|
|
|
|
|
|
private readonly IStorageService _storageService;
|
|
|
|
|
private readonly ITokenService _tokenService;
|
|
|
|
|
|
|
|
|
|
public UserService(IStorageService storageService, ITokenService tokenService)
|
|
|
|
|
{
|
|
|
|
|
_storageService = storageService;
|
|
|
|
|
_tokenService = tokenService;
|
|
|
|
|
}
|
|
|
|
|
|
2020-12-08 18:54:58 +03:00
|
|
|
|
public async Task SetInformationAsync(string userId, string email, KdfType kdf, int? kdfIterations)
|
2019-04-09 23:13:32 +03:00
|
|
|
|
{
|
|
|
|
|
_email = email;
|
|
|
|
|
_userId = userId;
|
|
|
|
|
_kdf = kdf;
|
|
|
|
|
_kdfIterations = kdfIterations;
|
|
|
|
|
await Task.WhenAll(
|
|
|
|
|
_storageService.SaveAsync(Keys_UserEmail, email),
|
|
|
|
|
_storageService.SaveAsync(Keys_UserId, userId),
|
|
|
|
|
_storageService.SaveAsync(Keys_Kdf, (int)kdf),
|
|
|
|
|
_storageService.SaveAsync(Keys_KdfIterations, kdfIterations));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public async Task SetSecurityStampAsync(string stamp)
|
|
|
|
|
{
|
|
|
|
|
_stamp = stamp;
|
|
|
|
|
await _storageService.SaveAsync(Keys_Stamp, stamp);
|
|
|
|
|
}
|
|
|
|
|
|
2021-04-15 07:54:58 +03:00
|
|
|
|
public async Task SetEmailVerifiedAsync(bool emailVerified)
|
|
|
|
|
{
|
|
|
|
|
_emailVerified = emailVerified;
|
|
|
|
|
await _storageService.SaveAsync(Keys_EmailVerified, emailVerified);
|
|
|
|
|
}
|
|
|
|
|
|
2021-09-24 21:14:26 +03:00
|
|
|
|
public async Task SetForcePasswordReset(bool forcePasswordReset)
|
|
|
|
|
{
|
|
|
|
|
_forcePasswordReset = forcePasswordReset;
|
|
|
|
|
await _storageService.SaveAsync(Keys_ForcePasswordReset, forcePasswordReset);
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-09 23:13:32 +03:00
|
|
|
|
public async Task<string> GetUserIdAsync()
|
|
|
|
|
{
|
2020-03-28 16:16:28 +03:00
|
|
|
|
if (_userId == null)
|
2019-04-09 23:13:32 +03:00
|
|
|
|
{
|
|
|
|
|
_userId = await _storageService.GetAsync<string>(Keys_UserId);
|
|
|
|
|
}
|
|
|
|
|
return _userId;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public async Task<string> GetEmailAsync()
|
|
|
|
|
{
|
2020-03-28 16:16:28 +03:00
|
|
|
|
if (_email == null)
|
2019-04-09 23:13:32 +03:00
|
|
|
|
{
|
|
|
|
|
_email = await _storageService.GetAsync<string>(Keys_UserEmail);
|
|
|
|
|
}
|
|
|
|
|
return _email;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public async Task<string> GetSecurityStampAsync()
|
|
|
|
|
{
|
2020-03-28 16:16:28 +03:00
|
|
|
|
if (_stamp == null)
|
2019-04-09 23:13:32 +03:00
|
|
|
|
{
|
|
|
|
|
_stamp = await _storageService.GetAsync<string>(Keys_Stamp);
|
|
|
|
|
}
|
|
|
|
|
return _stamp;
|
|
|
|
|
}
|
|
|
|
|
|
2021-04-15 07:54:58 +03:00
|
|
|
|
public async Task<bool> GetEmailVerifiedAsync()
|
|
|
|
|
{
|
|
|
|
|
if (_emailVerified == null)
|
|
|
|
|
{
|
|
|
|
|
_emailVerified = await _storageService.GetAsync<bool>(Keys_EmailVerified);
|
|
|
|
|
}
|
|
|
|
|
return _emailVerified.GetValueOrDefault();
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-09 23:13:32 +03:00
|
|
|
|
public async Task<KdfType?> GetKdfAsync()
|
|
|
|
|
{
|
2020-03-28 16:16:28 +03:00
|
|
|
|
if (_kdf == null)
|
2019-04-09 23:13:32 +03:00
|
|
|
|
{
|
|
|
|
|
_kdf = (KdfType?)(await _storageService.GetAsync<int?>(Keys_Kdf));
|
|
|
|
|
}
|
|
|
|
|
return _kdf;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public async Task<int?> GetKdfIterationsAsync()
|
|
|
|
|
{
|
2020-03-28 16:16:28 +03:00
|
|
|
|
if (_kdfIterations == null)
|
2019-04-09 23:13:32 +03:00
|
|
|
|
{
|
|
|
|
|
_kdfIterations = await _storageService.GetAsync<int?>(Keys_KdfIterations);
|
|
|
|
|
}
|
|
|
|
|
return _kdfIterations;
|
|
|
|
|
}
|
|
|
|
|
|
2021-09-24 21:14:26 +03:00
|
|
|
|
public async Task<bool> GetForcePasswordReset()
|
|
|
|
|
{
|
|
|
|
|
if (_forcePasswordReset == null)
|
|
|
|
|
{
|
|
|
|
|
_forcePasswordReset = await _storageService.GetAsync<bool>(Keys_ForcePasswordReset);
|
|
|
|
|
}
|
|
|
|
|
return _forcePasswordReset.GetValueOrDefault();
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-09 23:13:32 +03:00
|
|
|
|
public async Task ClearAsync()
|
|
|
|
|
{
|
|
|
|
|
var userId = await GetUserIdAsync();
|
|
|
|
|
await Task.WhenAll(
|
|
|
|
|
_storageService.RemoveAsync(Keys_UserId),
|
|
|
|
|
_storageService.RemoveAsync(Keys_UserEmail),
|
|
|
|
|
_storageService.RemoveAsync(Keys_Stamp),
|
|
|
|
|
_storageService.RemoveAsync(Keys_Kdf),
|
|
|
|
|
_storageService.RemoveAsync(Keys_KdfIterations),
|
2021-09-24 21:14:26 +03:00
|
|
|
|
_storageService.RemoveAsync(Keys_ForcePasswordReset),
|
2019-04-09 23:13:32 +03:00
|
|
|
|
ClearOrganizationsAsync(userId));
|
|
|
|
|
_userId = _email = _stamp = null;
|
|
|
|
|
_kdf = null;
|
|
|
|
|
_kdfIterations = null;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public async Task<bool> IsAuthenticatedAsync()
|
|
|
|
|
{
|
|
|
|
|
var token = await _tokenService.GetTokenAsync();
|
2020-03-28 16:16:28 +03:00
|
|
|
|
if (token == null)
|
2019-04-09 23:13:32 +03:00
|
|
|
|
{
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
var userId = await GetUserIdAsync();
|
|
|
|
|
return userId != null;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public async Task<bool> CanAccessPremiumAsync()
|
|
|
|
|
{
|
2020-09-18 22:07:32 +03:00
|
|
|
|
var authed = await IsAuthenticatedAsync();
|
|
|
|
|
if (!authed)
|
|
|
|
|
{
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-09 23:13:32 +03:00
|
|
|
|
var tokenPremium = _tokenService.GetPremium();
|
2020-03-28 16:16:28 +03:00
|
|
|
|
if (tokenPremium)
|
2019-04-09 23:13:32 +03:00
|
|
|
|
{
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
var orgs = await GetAllOrganizationAsync();
|
|
|
|
|
return orgs?.Any(o => o.UsersGetPremium && o.Enabled) ?? false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public async Task<Organization> GetOrganizationAsync(string id)
|
|
|
|
|
{
|
|
|
|
|
var userId = await GetUserIdAsync();
|
|
|
|
|
var organizations = await _storageService.GetAsync<Dictionary<string, OrganizationData>>(
|
|
|
|
|
string.Format(Keys_OrganizationsFormat, userId));
|
2020-03-28 16:16:28 +03:00
|
|
|
|
if (organizations == null || !organizations.ContainsKey(id))
|
2019-04-09 23:13:32 +03:00
|
|
|
|
{
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
return new Organization(organizations[id]);
|
|
|
|
|
}
|
2021-09-08 20:43:24 +03:00
|
|
|
|
|
|
|
|
|
public async Task<Organization> GetOrganizationByIdentifierAsync(string identifier)
|
|
|
|
|
{
|
|
|
|
|
var userId = await GetUserIdAsync();
|
|
|
|
|
var organizations = await GetAllOrganizationAsync();
|
|
|
|
|
|
|
|
|
|
if (organizations == null || organizations.Count == 0)
|
|
|
|
|
{
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return organizations.FirstOrDefault(o => o.Identifier == identifier);
|
|
|
|
|
}
|
2019-04-09 23:13:32 +03:00
|
|
|
|
|
|
|
|
|
public async Task<List<Organization>> GetAllOrganizationAsync()
|
|
|
|
|
{
|
|
|
|
|
var userId = await GetUserIdAsync();
|
|
|
|
|
var organizations = await _storageService.GetAsync<Dictionary<string, OrganizationData>>(
|
|
|
|
|
string.Format(Keys_OrganizationsFormat, userId));
|
|
|
|
|
return organizations?.Select(o => new Organization(o.Value)).ToList() ?? new List<Organization>();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public async Task ReplaceOrganizationsAsync(Dictionary<string, OrganizationData> organizations)
|
|
|
|
|
{
|
|
|
|
|
var userId = await GetUserIdAsync();
|
|
|
|
|
await _storageService.SaveAsync(string.Format(Keys_OrganizationsFormat, userId), organizations);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public async Task ClearOrganizationsAsync(string userId)
|
|
|
|
|
{
|
|
|
|
|
await _storageService.RemoveAsync(string.Format(Keys_OrganizationsFormat, userId));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|