2022-06-29 19:46:41 -04:00
|
|
|
|
using Bit.Core.Context;
|
2022-01-11 10:40:51 +01:00
|
|
|
|
using Bit.Core.Entities;
|
2015-12-08 22:57:38 -05:00
|
|
|
|
using Bit.Core.Repositories;
|
2018-08-28 17:40:08 -04:00
|
|
|
|
using Bit.Core.Services;
|
2016-05-19 19:10:24 -04:00
|
|
|
|
using Microsoft.AspNetCore.Identity;
|
2018-08-28 17:40:08 -04:00
|
|
|
|
using Microsoft.Extensions.DependencyInjection;
|
2015-12-08 22:57:38 -05:00
|
|
|
|
|
2023-04-14 13:25:56 -04:00
|
|
|
|
namespace Bit.Core.Auth.Identity;
|
2022-08-29 16:06:55 -04:00
|
|
|
|
|
2015-12-08 22:57:38 -05:00
|
|
|
|
public class UserStore :
|
|
|
|
|
|
IUserStore<User>,
|
|
|
|
|
|
IUserPasswordStore<User>,
|
|
|
|
|
|
IUserEmailStore<User>,
|
|
|
|
|
|
IUserTwoFactorStore<User>,
|
|
|
|
|
|
IUserSecurityStampStore<User>
|
|
|
|
|
|
{
|
2018-08-28 17:40:08 -04:00
|
|
|
|
private readonly IServiceProvider _serviceProvider;
|
2015-12-08 22:57:38 -05:00
|
|
|
|
private readonly IUserRepository _userRepository;
|
2021-02-04 12:54:21 -06:00
|
|
|
|
private readonly ICurrentContext _currentContext;
|
2022-08-29 16:06:55 -04:00
|
|
|
|
|
2015-12-27 00:14:56 -05:00
|
|
|
|
public UserStore(
|
2018-08-28 17:40:08 -04:00
|
|
|
|
IServiceProvider serviceProvider,
|
2015-12-27 00:14:56 -05:00
|
|
|
|
IUserRepository userRepository,
|
2021-02-04 12:54:21 -06:00
|
|
|
|
ICurrentContext currentContext)
|
2015-12-08 22:57:38 -05:00
|
|
|
|
{
|
2018-08-28 17:40:08 -04:00
|
|
|
|
_serviceProvider = serviceProvider;
|
2015-12-08 22:57:38 -05:00
|
|
|
|
_userRepository = userRepository;
|
2015-12-27 00:14:56 -05:00
|
|
|
|
_currentContext = currentContext;
|
2015-12-08 22:57:38 -05:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public void Dispose() { }
|
|
|
|
|
|
|
|
|
|
|
|
public async Task<IdentityResult> CreateAsync(User user, CancellationToken cancellationToken = default(CancellationToken))
|
|
|
|
|
|
{
|
|
|
|
|
|
await _userRepository.CreateAsync(user);
|
|
|
|
|
|
return IdentityResult.Success;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public async Task<IdentityResult> DeleteAsync(User user, CancellationToken cancellationToken = default(CancellationToken))
|
|
|
|
|
|
{
|
|
|
|
|
|
await _userRepository.DeleteAsync(user);
|
|
|
|
|
|
return IdentityResult.Success;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public async Task<User> FindByEmailAsync(string normalizedEmail, CancellationToken cancellationToken = default(CancellationToken))
|
2022-08-29 16:06:55 -04:00
|
|
|
|
{
|
2020-03-27 14:36:37 -04:00
|
|
|
|
if (_currentContext?.User != null && _currentContext.User.Email == normalizedEmail)
|
2015-12-08 22:57:38 -05:00
|
|
|
|
{
|
2017-01-24 22:15:21 -05:00
|
|
|
|
return _currentContext.User;
|
2015-12-08 22:57:38 -05:00
|
|
|
|
}
|
|
|
|
|
|
|
2017-01-24 22:15:21 -05:00
|
|
|
|
_currentContext.User = await _userRepository.GetByEmailAsync(normalizedEmail);
|
|
|
|
|
|
return _currentContext.User;
|
2022-08-29 16:06:55 -04:00
|
|
|
|
}
|
|
|
|
|
|
|
2015-12-08 22:57:38 -05:00
|
|
|
|
public async Task<User> FindByIdAsync(string userId, CancellationToken cancellationToken = default(CancellationToken))
|
2022-08-29 16:06:55 -04:00
|
|
|
|
{
|
2020-03-27 14:36:37 -04:00
|
|
|
|
if (_currentContext?.User != null &&
|
2017-01-24 22:15:21 -05:00
|
|
|
|
string.Equals(_currentContext.User.Id.ToString(), userId, StringComparison.InvariantCultureIgnoreCase))
|
2015-12-08 22:57:38 -05:00
|
|
|
|
{
|
2017-01-24 22:15:21 -05:00
|
|
|
|
return _currentContext.User;
|
2015-12-08 22:57:38 -05:00
|
|
|
|
}
|
|
|
|
|
|
|
2017-01-24 22:15:21 -05:00
|
|
|
|
Guid userIdGuid;
|
2015-12-08 22:57:38 -05:00
|
|
|
|
if (!Guid.TryParse(userId, out userIdGuid))
|
|
|
|
|
|
{
|
2017-01-24 22:15:21 -05:00
|
|
|
|
return null;
|
2015-12-08 22:57:38 -05:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
_currentContext.User = await _userRepository.GetByIdAsync(userIdGuid);
|
|
|
|
|
|
return _currentContext.User;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public async Task<User> FindByNameAsync(string normalizedUserName, CancellationToken cancellationToken = default(CancellationToken))
|
|
|
|
|
|
{
|
|
|
|
|
|
return await FindByEmailAsync(normalizedUserName, cancellationToken);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public Task<string> GetEmailAsync(User user, CancellationToken cancellationToken = default(CancellationToken))
|
|
|
|
|
|
{
|
|
|
|
|
|
return Task.FromResult(user.Email);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public Task<bool> GetEmailConfirmedAsync(User user, CancellationToken cancellationToken = default(CancellationToken))
|
|
|
|
|
|
{
|
2016-05-21 17:16:22 -04:00
|
|
|
|
return Task.FromResult(user.EmailVerified);
|
2015-12-08 22:57:38 -05:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public Task<string> GetNormalizedEmailAsync(User user, CancellationToken cancellationToken = default(CancellationToken))
|
|
|
|
|
|
{
|
|
|
|
|
|
return Task.FromResult(user.Email);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public Task<string> GetNormalizedUserNameAsync(User user, CancellationToken cancellationToken = default(CancellationToken))
|
|
|
|
|
|
{
|
|
|
|
|
|
return Task.FromResult(user.Email);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public Task<string> GetPasswordHashAsync(User user, CancellationToken cancellationToken = default(CancellationToken))
|
|
|
|
|
|
{
|
2015-12-29 21:45:21 -05:00
|
|
|
|
return Task.FromResult(user.MasterPassword);
|
2015-12-08 22:57:38 -05:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public Task<string> GetUserIdAsync(User user, CancellationToken cancellationToken = default(CancellationToken))
|
|
|
|
|
|
{
|
2015-12-29 21:45:21 -05:00
|
|
|
|
return Task.FromResult(user.Id.ToString());
|
2015-12-08 22:57:38 -05:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public Task<string> GetUserNameAsync(User user, CancellationToken cancellationToken = default(CancellationToken))
|
|
|
|
|
|
{
|
2015-12-29 21:45:21 -05:00
|
|
|
|
return Task.FromResult(user.Email);
|
2015-12-08 22:57:38 -05:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public Task<bool> HasPasswordAsync(User user, CancellationToken cancellationToken = default(CancellationToken))
|
|
|
|
|
|
{
|
2015-12-29 21:45:21 -05:00
|
|
|
|
return Task.FromResult(!string.IsNullOrWhiteSpace(user.MasterPassword));
|
2015-12-08 22:57:38 -05:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public Task SetEmailAsync(User user, string email, CancellationToken cancellationToken = default(CancellationToken))
|
|
|
|
|
|
{
|
|
|
|
|
|
user.Email = email;
|
2015-12-29 21:45:21 -05:00
|
|
|
|
return Task.FromResult(0);
|
2015-12-08 22:57:38 -05:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public Task SetEmailConfirmedAsync(User user, bool confirmed, CancellationToken cancellationToken = default(CancellationToken))
|
|
|
|
|
|
{
|
|
|
|
|
|
user.EmailVerified = confirmed;
|
2015-12-29 21:45:21 -05:00
|
|
|
|
return Task.FromResult(0);
|
2015-12-08 22:57:38 -05:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public Task SetNormalizedEmailAsync(User user, string normalizedEmail, CancellationToken cancellationToken = default(CancellationToken))
|
|
|
|
|
|
{
|
2017-01-14 10:02:37 -05:00
|
|
|
|
user.Email = normalizedEmail;
|
2015-12-08 22:57:38 -05:00
|
|
|
|
return Task.FromResult(0);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public Task SetNormalizedUserNameAsync(User user, string normalizedName, CancellationToken cancellationToken = default(CancellationToken))
|
|
|
|
|
|
{
|
2017-07-05 15:35:46 -04:00
|
|
|
|
user.Email = normalizedName;
|
2015-12-29 21:45:21 -05:00
|
|
|
|
return Task.FromResult(0);
|
2015-12-08 22:57:38 -05:00
|
|
|
|
}
|
|
|
|
|
|
|
2018-08-28 17:40:08 -04:00
|
|
|
|
public Task SetPasswordHashAsync(User user, string passwordHash, CancellationToken cancellationToken = default(CancellationToken))
|
2015-12-08 22:57:38 -05:00
|
|
|
|
{
|
2018-12-19 10:47:53 -05:00
|
|
|
|
user.MasterPassword = passwordHash;
|
|
|
|
|
|
return Task.FromResult(0);
|
2015-12-08 22:57:38 -05:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public Task SetUserNameAsync(User user, string userName, CancellationToken cancellationToken = default(CancellationToken))
|
|
|
|
|
|
{
|
|
|
|
|
|
user.Email = userName;
|
2015-12-29 21:45:21 -05:00
|
|
|
|
return Task.FromResult(0);
|
2015-12-08 22:57:38 -05:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public async Task<IdentityResult> UpdateAsync(User user, CancellationToken cancellationToken = default(CancellationToken))
|
2022-08-29 14:53:16 -04:00
|
|
|
|
{
|
2018-12-19 10:47:53 -05:00
|
|
|
|
user.RevisionDate = user.AccountRevisionDate = DateTime.UtcNow;
|
|
|
|
|
|
await _userRepository.ReplaceAsync(user);
|
|
|
|
|
|
return IdentityResult.Success;
|
2022-08-29 14:53:16 -04:00
|
|
|
|
}
|
|
|
|
|
|
|
2015-12-08 22:57:38 -05:00
|
|
|
|
public Task SetTwoFactorEnabledAsync(User user, bool enabled, CancellationToken cancellationToken)
|
2022-08-29 14:53:16 -04:00
|
|
|
|
{
|
2017-06-19 22:25:19 -04:00
|
|
|
|
// Do nothing...
|
2015-12-08 22:57:38 -05:00
|
|
|
|
return Task.FromResult(0);
|
2022-08-29 14:53:16 -04:00
|
|
|
|
}
|
|
|
|
|
|
|
2015-12-08 22:57:38 -05:00
|
|
|
|
public async Task<bool> GetTwoFactorEnabledAsync(User user, CancellationToken cancellationToken)
|
2022-08-29 16:06:55 -04:00
|
|
|
|
{
|
2018-12-19 10:47:53 -05:00
|
|
|
|
return await _serviceProvider.GetRequiredService<IUserService>().TwoFactorIsEnabledAsync(user);
|
2022-08-29 16:06:55 -04:00
|
|
|
|
}
|
|
|
|
|
|
|
2015-12-08 22:57:38 -05:00
|
|
|
|
public Task SetSecurityStampAsync(User user, string stamp, CancellationToken cancellationToken)
|
2022-08-29 16:06:55 -04:00
|
|
|
|
{
|
2015-12-08 22:57:38 -05:00
|
|
|
|
user.SecurityStamp = stamp;
|
|
|
|
|
|
return Task.FromResult(0);
|
2022-08-29 16:06:55 -04:00
|
|
|
|
}
|
|
|
|
|
|
|
2015-12-08 22:57:38 -05:00
|
|
|
|
public Task<string> GetSecurityStampAsync(User user, CancellationToken cancellationToken)
|
|
|
|
|
|
{
|
|
|
|
|
|
return Task.FromResult(user.SecurityStamp);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|