2020-12-12 16:20:48 +00:00
#pragma warning disable CA1307
2020-05-13 02:10:35 +00:00
using System ;
2020-10-27 00:31:10 +00:00
using System.Collections.Concurrent ;
2020-05-13 02:10:35 +00:00
using System.Collections.Generic ;
using System.Globalization ;
using System.Linq ;
using System.Text ;
2020-05-15 21:24:01 +00:00
using System.Text.RegularExpressions ;
2020-05-13 02:10:35 +00:00
using System.Threading.Tasks ;
2020-05-15 21:24:01 +00:00
using Jellyfin.Data.Entities ;
2020-05-13 02:10:35 +00:00
using Jellyfin.Data.Enums ;
2020-08-14 00:48:28 +00:00
using Jellyfin.Data.Events ;
2020-08-15 19:55:15 +00:00
using Jellyfin.Data.Events.Users ;
2020-05-19 19:58:25 +00:00
using MediaBrowser.Common ;
2020-05-13 02:10:35 +00:00
using MediaBrowser.Common.Cryptography ;
2020-06-25 00:36:58 +00:00
using MediaBrowser.Common.Extensions ;
2020-05-13 02:10:35 +00:00
using MediaBrowser.Common.Net ;
using MediaBrowser.Controller.Authentication ;
2020-05-19 19:58:25 +00:00
using MediaBrowser.Controller.Drawing ;
2020-08-15 19:55:15 +00:00
using MediaBrowser.Controller.Events ;
2020-05-13 02:10:35 +00:00
using MediaBrowser.Controller.Library ;
using MediaBrowser.Controller.Net ;
using MediaBrowser.Model.Configuration ;
using MediaBrowser.Model.Cryptography ;
using MediaBrowser.Model.Dto ;
using MediaBrowser.Model.Users ;
2020-07-12 18:45:52 +00:00
using Microsoft.EntityFrameworkCore ;
2020-05-13 02:10:35 +00:00
using Microsoft.Extensions.Logging ;
2020-05-15 21:24:01 +00:00
namespace Jellyfin.Server.Implementations.Users
2020-05-13 02:10:35 +00:00
{
2020-05-19 23:44:55 +00:00
/// <summary>
/// Manages the creation and retrieval of <see cref="User"/> instances.
/// </summary>
2020-05-13 02:10:35 +00:00
public class UserManager : IUserManager
{
private readonly JellyfinDbProvider _dbProvider ;
2020-08-15 19:55:15 +00:00
private readonly IEventManager _eventManager ;
2020-05-13 02:10:35 +00:00
private readonly ICryptoProvider _cryptoProvider ;
private readonly INetworkManager _networkManager ;
2020-05-19 19:58:25 +00:00
private readonly IApplicationHost _appHost ;
private readonly IImageProcessor _imageProcessor ;
2020-05-20 23:47:41 +00:00
private readonly ILogger < UserManager > _logger ;
2020-06-17 14:24:25 +00:00
private readonly IReadOnlyCollection < IPasswordResetProvider > _passwordResetProviders ;
private readonly IReadOnlyCollection < IAuthenticationProvider > _authenticationProviders ;
private readonly InvalidAuthProvider _invalidAuthProvider ;
private readonly DefaultAuthenticationProvider _defaultAuthenticationProvider ;
private readonly DefaultPasswordResetProvider _defaultPasswordResetProvider ;
2020-05-13 02:10:35 +00:00
2020-10-27 16:12:08 +00:00
private readonly IDictionary < Guid , User > _users ;
2020-10-27 00:31:10 +00:00
2020-05-19 23:44:55 +00:00
/// <summary>
/// Initializes a new instance of the <see cref="UserManager"/> class.
/// </summary>
/// <param name="dbProvider">The database provider.</param>
2020-08-15 19:55:15 +00:00
/// <param name="eventManager">The event manager.</param>
2020-05-19 23:44:55 +00:00
/// <param name="cryptoProvider">The cryptography provider.</param>
/// <param name="networkManager">The network manager.</param>
/// <param name="appHost">The application host.</param>
/// <param name="imageProcessor">The image processor.</param>
/// <param name="logger">The logger.</param>
2020-05-13 02:10:35 +00:00
public UserManager (
JellyfinDbProvider dbProvider ,
2020-08-15 19:55:15 +00:00
IEventManager eventManager ,
2020-05-13 02:10:35 +00:00
ICryptoProvider cryptoProvider ,
INetworkManager networkManager ,
2020-05-19 19:58:25 +00:00
IApplicationHost appHost ,
IImageProcessor imageProcessor ,
2020-05-20 23:47:41 +00:00
ILogger < UserManager > logger )
2020-05-13 02:10:35 +00:00
{
_dbProvider = dbProvider ;
2020-08-15 19:55:15 +00:00
_eventManager = eventManager ;
2020-05-13 02:10:35 +00:00
_cryptoProvider = cryptoProvider ;
_networkManager = networkManager ;
2020-05-19 19:58:25 +00:00
_appHost = appHost ;
_imageProcessor = imageProcessor ;
2020-05-13 02:10:35 +00:00
_logger = logger ;
2020-06-17 14:24:25 +00:00
2020-06-20 21:58:09 +00:00
_passwordResetProviders = appHost . GetExports < IPasswordResetProvider > ( ) ;
_authenticationProviders = appHost . GetExports < IAuthenticationProvider > ( ) ;
2020-06-17 14:24:25 +00:00
_invalidAuthProvider = _authenticationProviders . OfType < InvalidAuthProvider > ( ) . First ( ) ;
_defaultAuthenticationProvider = _authenticationProviders . OfType < DefaultAuthenticationProvider > ( ) . First ( ) ;
_defaultPasswordResetProvider = _passwordResetProviders . OfType < DefaultPasswordResetProvider > ( ) . First ( ) ;
2020-10-27 00:31:10 +00:00
2020-10-27 16:12:08 +00:00
_users = new ConcurrentDictionary < Guid , User > ( ) ;
2020-10-27 00:31:10 +00:00
using var dbContext = _dbProvider . CreateContext ( ) ;
foreach ( var user in dbContext . Users
. Include ( user = > user . Permissions )
. Include ( user = > user . Preferences )
. Include ( user = > user . AccessSchedules )
. Include ( user = > user . ProfileImage )
. AsEnumerable ( ) )
{
_users . Add ( user . Id , user ) ;
}
2020-05-13 02:10:35 +00:00
}
/// <inheritdoc/>
2020-06-09 16:21:21 +00:00
public event EventHandler < GenericEventArgs < User > > ? OnUserUpdated ;
2020-05-13 02:10:35 +00:00
2020-05-19 23:44:55 +00:00
/// <inheritdoc/>
2020-10-27 16:12:08 +00:00
public IEnumerable < User > Users = > _users . Values ;
2020-05-13 02:10:35 +00:00
2020-05-19 23:44:55 +00:00
/// <inheritdoc/>
2020-10-27 16:12:08 +00:00
public IEnumerable < Guid > UsersIds = > _users . Keys ;
2020-05-13 02:10:35 +00:00
2020-05-19 23:44:55 +00:00
/// <inheritdoc/>
2020-06-09 16:21:21 +00:00
public User ? GetUserById ( Guid id )
2020-05-13 02:10:35 +00:00
{
if ( id = = Guid . Empty )
{
throw new ArgumentException ( "Guid can't be empty" , nameof ( id ) ) ;
}
2020-10-27 00:31:10 +00:00
_users . TryGetValue ( id , out var user ) ;
return user ;
2020-05-13 02:10:35 +00:00
}
2020-05-19 23:44:55 +00:00
/// <inheritdoc/>
2020-06-09 16:21:21 +00:00
public User ? GetUserByName ( string name )
2020-05-13 02:10:35 +00:00
{
if ( string . IsNullOrWhiteSpace ( name ) )
{
throw new ArgumentException ( "Invalid username" , nameof ( name ) ) ;
}
2020-10-27 00:31:10 +00:00
return _users . Values . FirstOrDefault ( u = > string . Equals ( u . Username , name , StringComparison . OrdinalIgnoreCase ) ) ;
2020-05-13 02:10:35 +00:00
}
2020-05-19 23:44:55 +00:00
/// <inheritdoc/>
2020-05-15 21:24:01 +00:00
public async Task RenameUser ( User user , string newName )
2020-05-13 02:10:35 +00:00
{
if ( user = = null )
{
throw new ArgumentNullException ( nameof ( user ) ) ;
}
2021-02-17 10:30:14 +00:00
ThrowIfInvalidUsername ( newName ) ;
2020-05-13 02:10:35 +00:00
2020-07-20 02:21:30 +00:00
if ( user . Username . Equals ( newName , StringComparison . Ordinal ) )
2020-05-13 02:10:35 +00:00
{
throw new ArgumentException ( "The new and old names must be different." ) ;
}
2021-03-25 23:48:30 +00:00
await using var dbContext = _dbProvider . CreateContext ( ) ;
if ( await dbContext . Users
. AsQueryable ( )
. Where ( u = > u . Username = = newName & & u . Id ! = user . Id )
. AnyAsync ( )
. ConfigureAwait ( false ) )
2020-05-13 02:10:35 +00:00
{
throw new ArgumentException ( string . Format (
CultureInfo . InvariantCulture ,
"A user with the name '{0}' already exists." ,
newName ) ) ;
}
user . Username = newName ;
await UpdateUserAsync ( user ) . ConfigureAwait ( false ) ;
2020-05-15 21:24:01 +00:00
OnUserUpdated ? . Invoke ( this , new GenericEventArgs < User > ( user ) ) ;
2020-05-13 02:10:35 +00:00
}
2020-05-19 23:44:55 +00:00
/// <inheritdoc/>
2020-05-15 21:24:01 +00:00
public async Task UpdateUserAsync ( User user )
2020-05-13 02:10:35 +00:00
{
2020-07-12 18:45:52 +00:00
await using var dbContext = _dbProvider . CreateContext ( ) ;
2020-05-13 02:10:35 +00:00
dbContext . Users . Update ( user ) ;
2020-11-15 16:30:04 +00:00
_users [ user . Id ] = user ;
2020-05-13 02:10:35 +00:00
await dbContext . SaveChangesAsync ( ) . ConfigureAwait ( false ) ;
}
2020-07-22 18:57:29 +00:00
internal async Task < User > CreateUserInternalAsync ( string name , JellyfinDb dbContext )
{
// TODO: Remove after user item data is migrated.
2020-10-06 02:51:52 +00:00
var max = await dbContext . Users . AsQueryable ( ) . AnyAsync ( ) . ConfigureAwait ( false )
? await dbContext . Users . AsQueryable ( ) . Select ( u = > u . InternalId ) . MaxAsync ( ) . ConfigureAwait ( false )
2020-07-22 18:57:29 +00:00
: 0 ;
2020-10-27 00:31:10 +00:00
var user = new User (
2020-07-22 18:57:29 +00:00
name ,
2021-03-06 22:43:01 +00:00
_defaultAuthenticationProvider . GetType ( ) . FullName ! ,
_defaultPasswordResetProvider . GetType ( ) . FullName ! )
2020-07-22 18:57:29 +00:00
{
InternalId = max + 1
} ;
2020-10-27 00:31:10 +00:00
2021-03-17 21:42:45 +00:00
user . AddDefaultPermissions ( ) ;
user . AddDefaultPreferences ( ) ;
2020-10-27 00:31:10 +00:00
_users . Add ( user . Id , user ) ;
return user ;
2020-07-22 18:57:29 +00:00
}
2020-05-19 23:44:55 +00:00
/// <inheritdoc/>
2020-07-22 18:57:29 +00:00
public async Task < User > CreateUserAsync ( string name )
2020-05-13 02:10:35 +00:00
{
2021-02-17 10:30:14 +00:00
ThrowIfInvalidUsername ( name ) ;
2020-05-15 21:24:01 +00:00
2021-02-17 01:48:41 +00:00
if ( Users . Any ( u = > u . Username . Equals ( name , StringComparison . OrdinalIgnoreCase ) ) )
{
throw new ArgumentException ( string . Format (
CultureInfo . InvariantCulture ,
"A user with the name '{0}' already exists." ,
name ) ) ;
}
2020-10-06 02:51:52 +00:00
await using var dbContext = _dbProvider . CreateContext ( ) ;
2020-05-13 02:10:35 +00:00
2020-07-22 18:57:29 +00:00
var newUser = await CreateUserInternalAsync ( name , dbContext ) . ConfigureAwait ( false ) ;
2020-05-20 17:49:44 +00:00
2020-05-13 02:10:35 +00:00
dbContext . Users . Add ( newUser ) ;
2020-07-22 18:57:29 +00:00
await dbContext . SaveChangesAsync ( ) . ConfigureAwait ( false ) ;
2020-05-13 02:10:35 +00:00
2020-08-15 19:55:15 +00:00
await _eventManager . PublishAsync ( new UserCreatedEventArgs ( newUser ) ) . ConfigureAwait ( false ) ;
2020-05-15 21:24:01 +00:00
2020-05-13 02:10:35 +00:00
return newUser ;
}
2020-05-19 23:44:55 +00:00
/// <inheritdoc/>
2020-12-11 15:15:43 +00:00
public async Task DeleteUserAsync ( Guid userId )
2020-05-13 02:10:35 +00:00
{
2020-10-27 00:31:10 +00:00
if ( ! _users . TryGetValue ( userId , out var user ) )
2020-05-13 02:10:35 +00:00
{
2020-06-25 00:36:58 +00:00
throw new ResourceNotFoundException ( nameof ( userId ) ) ;
2020-05-13 02:10:35 +00:00
}
2020-10-27 00:31:10 +00:00
if ( _users . Count = = 1 )
2020-05-13 02:10:35 +00:00
{
throw new InvalidOperationException ( string . Format (
CultureInfo . InvariantCulture ,
"The user '{0}' cannot be deleted because there must be at least one user in the system." ,
user . Username ) ) ;
}
if ( user . HasPermission ( PermissionKind . IsAdministrator )
& & Users . Count ( i = > i . HasPermission ( PermissionKind . IsAdministrator ) ) = = 1 )
{
throw new ArgumentException (
string . Format (
CultureInfo . InvariantCulture ,
"The user '{0}' cannot be deleted because there must be at least one admin user in the system." ,
user . Username ) ,
2020-06-25 00:19:47 +00:00
nameof ( userId ) ) ;
}
2020-12-11 15:15:43 +00:00
await using var dbContext = _dbProvider . CreateContext ( ) ;
2020-05-13 02:10:35 +00:00
dbContext . Users . Remove ( user ) ;
2020-12-11 15:15:43 +00:00
await dbContext . SaveChangesAsync ( ) . ConfigureAwait ( false ) ;
2020-10-27 00:31:10 +00:00
_users . Remove ( userId ) ;
2020-08-15 19:55:15 +00:00
2020-12-11 15:15:43 +00:00
await _eventManager . PublishAsync ( new UserDeletedEventArgs ( user ) ) . ConfigureAwait ( false ) ;
2020-05-13 02:10:35 +00:00
}
2020-05-19 23:44:55 +00:00
/// <inheritdoc/>
2020-05-15 21:24:01 +00:00
public Task ResetPassword ( User user )
2020-05-13 02:10:35 +00:00
{
return ChangePassword ( user , string . Empty ) ;
}
2020-05-19 23:44:55 +00:00
/// <inheritdoc/>
2021-04-10 20:59:41 +00:00
public Task ResetEasyPassword ( User user )
2020-05-13 02:10:35 +00:00
{
2021-04-10 20:59:41 +00:00
return ChangeEasyPassword ( user , string . Empty , null ) ;
2020-05-13 02:10:35 +00:00
}
2020-05-19 23:44:55 +00:00
/// <inheritdoc/>
2020-05-15 21:24:01 +00:00
public async Task ChangePassword ( User user , string newPassword )
2020-05-13 02:10:35 +00:00
{
if ( user = = null )
{
throw new ArgumentNullException ( nameof ( user ) ) ;
}
await GetAuthenticationProvider ( user ) . ChangePassword ( user , newPassword ) . ConfigureAwait ( false ) ;
await UpdateUserAsync ( user ) . ConfigureAwait ( false ) ;
2020-08-15 19:55:15 +00:00
await _eventManager . PublishAsync ( new UserPasswordChangedEventArgs ( user ) ) . ConfigureAwait ( false ) ;
2020-05-13 02:10:35 +00:00
}
2020-05-19 23:44:55 +00:00
/// <inheritdoc/>
2021-04-10 20:59:41 +00:00
public async Task ChangeEasyPassword ( User user , string newPassword , string? newPasswordSha1 )
2020-05-13 02:10:35 +00:00
{
2020-06-24 15:45:11 +00:00
if ( newPassword ! = null )
{
newPasswordSha1 = _cryptoProvider . CreatePasswordHash ( newPassword ) . ToString ( ) ;
}
if ( string . IsNullOrWhiteSpace ( newPasswordSha1 ) )
{
throw new ArgumentNullException ( nameof ( newPasswordSha1 ) ) ;
}
user . EasyPassword = newPasswordSha1 ;
2021-04-10 21:11:59 +00:00
await UpdateUserAsync ( user ) . ConfigureAwait ( false ) ;
2020-05-13 02:10:35 +00:00
2020-08-15 19:55:15 +00:00
_eventManager . Publish ( new UserPasswordChangedEventArgs ( user ) ) ;
2020-05-13 02:10:35 +00:00
}
2020-05-19 23:44:55 +00:00
/// <inheritdoc/>
2020-06-09 16:21:21 +00:00
public UserDto GetUserDto ( User user , string? remoteEndPoint = null )
2020-05-13 02:10:35 +00:00
{
2020-06-08 03:11:51 +00:00
var hasPassword = GetAuthenticationProvider ( user ) . HasPassword ( user ) ;
2020-05-13 02:10:35 +00:00
return new UserDto
{
2020-05-19 19:58:25 +00:00
Name = user . Username ,
2020-05-13 02:10:35 +00:00
Id = user . Id ,
2020-05-19 19:58:25 +00:00
ServerId = _appHost . SystemId ,
2020-06-08 03:11:51 +00:00
HasPassword = hasPassword ,
HasConfiguredPassword = hasPassword ,
2020-06-08 00:16:51 +00:00
HasConfiguredEasyPassword = ! string . IsNullOrEmpty ( user . EasyPassword ) ,
2020-05-13 02:10:35 +00:00
EnableAutoLogin = user . EnableAutoLogin ,
LastLoginDate = user . LastLoginDate ,
LastActivityDate = user . LastActivityDate ,
2020-05-19 19:58:25 +00:00
PrimaryImageTag = user . ProfileImage ! = null ? _imageProcessor . GetImageCacheTag ( user ) : null ,
2020-05-13 02:10:35 +00:00
Configuration = new UserConfiguration
{
SubtitleMode = user . SubtitleMode ,
HidePlayedInLatest = user . HidePlayedInLatest ,
EnableLocalPassword = user . EnableLocalPassword ,
PlayDefaultAudioTrack = user . PlayDefaultAudioTrack ,
DisplayCollectionsView = user . DisplayCollectionsView ,
DisplayMissingEpisodes = user . DisplayMissingEpisodes ,
AudioLanguagePreference = user . AudioLanguagePreference ,
RememberAudioSelections = user . RememberAudioSelections ,
EnableNextEpisodeAutoPlay = user . EnableNextEpisodeAutoPlay ,
RememberSubtitleSelections = user . RememberSubtitleSelections ,
2020-05-19 19:58:25 +00:00
SubtitleLanguagePreference = user . SubtitleLanguagePreference ? ? string . Empty ,
2020-05-13 02:10:35 +00:00
OrderedViews = user . GetPreference ( PreferenceKind . OrderedViews ) ,
GroupedFolders = user . GetPreference ( PreferenceKind . GroupedFolders ) ,
MyMediaExcludes = user . GetPreference ( PreferenceKind . MyMediaExcludes ) ,
LatestItemsExcludes = user . GetPreference ( PreferenceKind . LatestItemExcludes )
} ,
Policy = new UserPolicy
{
MaxParentalRating = user . MaxParentalAgeRating ,
EnableUserPreferenceAccess = user . EnableUserPreferenceAccess ,
2020-06-07 23:37:47 +00:00
RemoteClientBitrateLimit = user . RemoteClientBitrateLimit ? ? 0 ,
2020-05-15 21:24:01 +00:00
AuthenticationProviderId = user . AuthenticationProviderId ,
2020-05-13 02:10:35 +00:00
PasswordResetProviderId = user . PasswordResetProviderId ,
InvalidLoginAttemptCount = user . InvalidLoginAttemptCount ,
2020-05-28 05:08:37 +00:00
LoginAttemptsBeforeLockout = user . LoginAttemptsBeforeLockout ? ? - 1 ,
2020-10-04 17:34:53 +00:00
MaxActiveSessions = user . MaxActiveSessions ,
2020-05-13 02:10:35 +00:00
IsAdministrator = user . HasPermission ( PermissionKind . IsAdministrator ) ,
IsHidden = user . HasPermission ( PermissionKind . IsHidden ) ,
IsDisabled = user . HasPermission ( PermissionKind . IsDisabled ) ,
EnableSharedDeviceControl = user . HasPermission ( PermissionKind . EnableSharedDeviceControl ) ,
EnableRemoteAccess = user . HasPermission ( PermissionKind . EnableRemoteAccess ) ,
EnableLiveTvManagement = user . HasPermission ( PermissionKind . EnableLiveTvManagement ) ,
EnableLiveTvAccess = user . HasPermission ( PermissionKind . EnableLiveTvAccess ) ,
EnableMediaPlayback = user . HasPermission ( PermissionKind . EnableMediaPlayback ) ,
EnableAudioPlaybackTranscoding = user . HasPermission ( PermissionKind . EnableAudioPlaybackTranscoding ) ,
EnableVideoPlaybackTranscoding = user . HasPermission ( PermissionKind . EnableVideoPlaybackTranscoding ) ,
EnableContentDeletion = user . HasPermission ( PermissionKind . EnableContentDeletion ) ,
EnableContentDownloading = user . HasPermission ( PermissionKind . EnableContentDownloading ) ,
EnableSyncTranscoding = user . HasPermission ( PermissionKind . EnableSyncTranscoding ) ,
EnableMediaConversion = user . HasPermission ( PermissionKind . EnableMediaConversion ) ,
EnableAllChannels = user . HasPermission ( PermissionKind . EnableAllChannels ) ,
EnableAllDevices = user . HasPermission ( PermissionKind . EnableAllDevices ) ,
EnableAllFolders = user . HasPermission ( PermissionKind . EnableAllFolders ) ,
EnableRemoteControlOfOtherUsers = user . HasPermission ( PermissionKind . EnableRemoteControlOfOtherUsers ) ,
EnablePlaybackRemuxing = user . HasPermission ( PermissionKind . EnablePlaybackRemuxing ) ,
ForceRemoteSourceTranscoding = user . HasPermission ( PermissionKind . ForceRemoteSourceTranscoding ) ,
EnablePublicSharing = user . HasPermission ( PermissionKind . EnablePublicSharing ) ,
AccessSchedules = user . AccessSchedules . ToArray ( ) ,
BlockedTags = user . GetPreference ( PreferenceKind . BlockedTags ) ,
2020-12-13 15:15:26 +00:00
EnabledChannels = user . GetPreferenceValues < Guid > ( PreferenceKind . EnabledChannels ) ,
2020-05-13 02:10:35 +00:00
EnabledDevices = user . GetPreference ( PreferenceKind . EnabledDevices ) ,
2020-12-13 15:15:26 +00:00
EnabledFolders = user . GetPreferenceValues < Guid > ( PreferenceKind . EnabledFolders ) ,
2020-05-27 00:52:05 +00:00
EnableContentDeletionFromFolders = user . GetPreference ( PreferenceKind . EnableContentDeletionFromFolders ) ,
2020-06-13 22:26:46 +00:00
SyncPlayAccess = user . SyncPlayAccess ,
2020-12-13 15:15:26 +00:00
BlockedChannels = user . GetPreferenceValues < Guid > ( PreferenceKind . BlockedChannels ) ,
BlockedMediaFolders = user . GetPreferenceValues < Guid > ( PreferenceKind . BlockedMediaFolders ) ,
BlockUnratedItems = user . GetPreferenceValues < UnratedItem > ( PreferenceKind . BlockUnratedItems )
2020-05-13 02:10:35 +00:00
}
} ;
}
2020-05-19 23:44:55 +00:00
/// <inheritdoc/>
2020-06-09 16:21:21 +00:00
public async Task < User ? > AuthenticateUser (
2020-05-13 02:10:35 +00:00
string username ,
string password ,
string passwordSha1 ,
string remoteEndPoint ,
bool isUserSession )
{
if ( string . IsNullOrWhiteSpace ( username ) )
{
_logger . LogInformation ( "Authentication request without username has been denied (IP: {IP})." , remoteEndPoint ) ;
throw new ArgumentNullException ( nameof ( username ) ) ;
}
2020-07-12 18:45:52 +00:00
var user = Users . FirstOrDefault ( i = > string . Equals ( username , i . Username , StringComparison . OrdinalIgnoreCase ) ) ;
2021-02-06 20:59:27 +00:00
var authResult = await AuthenticateLocalUser ( username , password , user , remoteEndPoint )
. ConfigureAwait ( false ) ;
var authenticationProvider = authResult . authenticationProvider ;
var success = authResult . success ;
2020-05-13 02:10:35 +00:00
2021-02-26 00:02:27 +00:00
if ( user = = null )
2020-05-13 02:10:35 +00:00
{
string updatedUsername = authResult . username ;
if ( success
& & authenticationProvider ! = null
2021-02-06 20:59:27 +00:00
& & authenticationProvider is not DefaultAuthenticationProvider )
2020-05-13 02:10:35 +00:00
{
// Trust the username returned by the authentication provider
username = updatedUsername ;
// Search the database for the user again
// the authentication provider might have created it
2020-07-12 18:45:52 +00:00
user = Users . FirstOrDefault ( i = > string . Equals ( username , i . Username , StringComparison . OrdinalIgnoreCase ) ) ;
2020-05-13 02:10:35 +00:00
2020-10-29 23:16:39 +00:00
if ( authenticationProvider is IHasNewUserPolicy hasNewUserPolicy & & user ! = null )
2020-05-13 02:10:35 +00:00
{
2020-10-30 00:30:33 +00:00
await UpdatePolicyAsync ( user . Id , hasNewUserPolicy . GetNewUserPolicy ( ) ) . ConfigureAwait ( false ) ;
2020-05-13 02:10:35 +00:00
}
}
}
if ( success & & user ! = null & & authenticationProvider ! = null )
{
var providerId = authenticationProvider . GetType ( ) . FullName ;
2021-03-06 22:43:01 +00:00
if ( providerId ! = null & & ! string . Equals ( providerId , user . AuthenticationProviderId , StringComparison . OrdinalIgnoreCase ) )
2020-05-13 02:10:35 +00:00
{
user . AuthenticationProviderId = providerId ;
await UpdateUserAsync ( user ) . ConfigureAwait ( false ) ;
}
}
if ( user = = null )
{
_logger . LogInformation (
"Authentication request for {UserName} has been denied (IP: {IP})." ,
username ,
remoteEndPoint ) ;
throw new AuthenticationException ( "Invalid username or password entered." ) ;
}
if ( user . HasPermission ( PermissionKind . IsDisabled ) )
{
_logger . LogInformation (
"Authentication request for {UserName} has been denied because this account is currently disabled (IP: {IP})." ,
username ,
remoteEndPoint ) ;
throw new SecurityException (
$"The {user.Username} account is currently disabled. Please consult with your administrator." ) ;
}
if ( ! user . HasPermission ( PermissionKind . EnableRemoteAccess ) & &
! _networkManager . IsInLocalNetwork ( remoteEndPoint ) )
{
_logger . LogInformation (
"Authentication request for {UserName} forbidden: remote access disabled and user not in local network (IP: {IP})." ,
username ,
remoteEndPoint ) ;
throw new SecurityException ( "Forbidden." ) ;
}
if ( ! user . IsParentalScheduleAllowed ( ) )
{
_logger . LogInformation (
"Authentication request for {UserName} is not allowed at this time due parental restrictions (IP: {IP})." ,
username ,
remoteEndPoint ) ;
throw new SecurityException ( "User is not allowed access at this time." ) ;
}
// Update LastActivityDate and LastLoginDate, then save
if ( success )
{
if ( isUserSession )
{
user . LastActivityDate = user . LastLoginDate = DateTime . UtcNow ;
}
2020-05-15 21:24:01 +00:00
user . InvalidLoginAttemptCount = 0 ;
2020-05-30 04:20:59 +00:00
await UpdateUserAsync ( user ) . ConfigureAwait ( false ) ;
2020-05-13 02:10:35 +00:00
_logger . LogInformation ( "Authentication request for {UserName} has succeeded." , user . Username ) ;
}
else
{
2020-07-21 18:25:52 +00:00
await IncrementInvalidLoginAttemptCount ( user ) . ConfigureAwait ( false ) ;
2020-05-13 02:10:35 +00:00
_logger . LogInformation (
"Authentication request for {UserName} has been denied (IP: {IP})." ,
user . Username ,
remoteEndPoint ) ;
}
return success ? user : null ;
}
2020-05-19 23:44:55 +00:00
/// <inheritdoc/>
2020-05-13 02:10:35 +00:00
public async Task < ForgotPasswordResult > StartForgotPasswordProcess ( string enteredUsername , bool isInNetwork )
{
var user = string . IsNullOrWhiteSpace ( enteredUsername ) ? null : GetUserByName ( enteredUsername ) ;
if ( user ! = null & & isInNetwork )
{
var passwordResetProvider = GetPasswordResetProvider ( user ) ;
2020-06-20 21:58:09 +00:00
var result = await passwordResetProvider
. StartForgotPasswordProcess ( user , isInNetwork )
. ConfigureAwait ( false ) ;
await UpdateUserAsync ( user ) . ConfigureAwait ( false ) ;
return result ;
2020-05-13 02:10:35 +00:00
}
return new ForgotPasswordResult
{
2020-05-28 05:08:37 +00:00
Action = ForgotPasswordAction . InNetworkRequired ,
2020-05-13 02:10:35 +00:00
PinFile = string . Empty
} ;
}
2020-05-19 23:44:55 +00:00
/// <inheritdoc/>
2020-05-13 02:10:35 +00:00
public async Task < PinRedeemResult > RedeemPasswordResetPin ( string pin )
{
foreach ( var provider in _passwordResetProviders )
{
var result = await provider . RedeemPasswordResetPin ( pin ) . ConfigureAwait ( false ) ;
if ( result . Success )
{
return result ;
}
}
2021-10-26 11:56:30 +00:00
return new PinRedeemResult ( ) ;
2020-05-13 02:10:35 +00:00
}
2020-06-09 18:01:21 +00:00
/// <inheritdoc />
2020-07-22 18:57:29 +00:00
public async Task InitializeAsync ( )
2020-06-09 18:01:21 +00:00
{
// TODO: Refactor the startup wizard so that it doesn't require a user to already exist.
2020-10-27 00:31:10 +00:00
if ( _users . Any ( ) )
2020-06-09 18:01:21 +00:00
{
return ;
}
var defaultName = Environment . UserName ;
2020-07-28 09:03:08 +00:00
if ( string . IsNullOrWhiteSpace ( defaultName ) | | ! IsValidUsername ( defaultName ) )
2020-06-09 18:01:21 +00:00
{
defaultName = "MyJellyfinUser" ;
}
_logger . LogWarning ( "No users, creating one with username {UserName}" , defaultName ) ;
2020-10-27 00:31:10 +00:00
await using var dbContext = _dbProvider . CreateContext ( ) ;
2020-07-22 18:57:29 +00:00
var newUser = await CreateUserInternalAsync ( defaultName , dbContext ) . ConfigureAwait ( false ) ;
2020-06-09 18:01:21 +00:00
newUser . SetPermission ( PermissionKind . IsAdministrator , true ) ;
newUser . SetPermission ( PermissionKind . EnableContentDeletion , true ) ;
newUser . SetPermission ( PermissionKind . EnableRemoteControlOfOtherUsers , true ) ;
2020-07-22 19:07:13 +00:00
dbContext . Users . Add ( newUser ) ;
2020-07-22 18:57:29 +00:00
await dbContext . SaveChangesAsync ( ) . ConfigureAwait ( false ) ;
2020-06-09 18:01:21 +00:00
}
2020-05-19 23:44:55 +00:00
/// <inheritdoc/>
2020-05-13 02:10:35 +00:00
public NameIdPair [ ] GetAuthenticationProviders ( )
{
return _authenticationProviders
. Where ( provider = > provider . IsEnabled )
. OrderBy ( i = > i is DefaultAuthenticationProvider ? 0 : 1 )
. ThenBy ( i = > i . Name )
. Select ( i = > new NameIdPair
{
Name = i . Name ,
Id = i . GetType ( ) . FullName
} )
. ToArray ( ) ;
}
2020-05-19 23:44:55 +00:00
/// <inheritdoc/>
2020-05-13 02:10:35 +00:00
public NameIdPair [ ] GetPasswordResetProviders ( )
{
return _passwordResetProviders
. Where ( provider = > provider . IsEnabled )
. OrderBy ( i = > i is DefaultPasswordResetProvider ? 0 : 1 )
. ThenBy ( i = > i . Name )
. Select ( i = > new NameIdPair
{
Name = i . Name ,
Id = i . GetType ( ) . FullName
} )
. ToArray ( ) ;
}
2020-05-19 23:44:55 +00:00
/// <inheritdoc/>
2020-10-30 00:30:33 +00:00
public async Task UpdateConfigurationAsync ( Guid userId , UserConfiguration config )
2020-05-13 02:10:35 +00:00
{
2020-10-30 00:30:33 +00:00
await using var dbContext = _dbProvider . CreateContext ( ) ;
2020-07-14 12:47:46 +00:00
var user = dbContext . Users
. Include ( u = > u . Permissions )
. Include ( u = > u . Preferences )
. Include ( u = > u . AccessSchedules )
. Include ( u = > u . ProfileImage )
. FirstOrDefault ( u = > u . Id = = userId )
? ? throw new ArgumentException ( "No user exists with given Id!" ) ;
2020-05-13 02:10:35 +00:00
user . SubtitleMode = config . SubtitleMode ;
user . HidePlayedInLatest = config . HidePlayedInLatest ;
user . EnableLocalPassword = config . EnableLocalPassword ;
user . PlayDefaultAudioTrack = config . PlayDefaultAudioTrack ;
user . DisplayCollectionsView = config . DisplayCollectionsView ;
user . DisplayMissingEpisodes = config . DisplayMissingEpisodes ;
user . AudioLanguagePreference = config . AudioLanguagePreference ;
user . RememberAudioSelections = config . RememberAudioSelections ;
user . EnableNextEpisodeAutoPlay = config . EnableNextEpisodeAutoPlay ;
user . RememberSubtitleSelections = config . RememberSubtitleSelections ;
user . SubtitleLanguagePreference = config . SubtitleLanguagePreference ;
user . SetPreference ( PreferenceKind . OrderedViews , config . OrderedViews ) ;
user . SetPreference ( PreferenceKind . GroupedFolders , config . GroupedFolders ) ;
user . SetPreference ( PreferenceKind . MyMediaExcludes , config . MyMediaExcludes ) ;
user . SetPreference ( PreferenceKind . LatestItemExcludes , config . LatestItemsExcludes ) ;
2020-05-28 05:08:37 +00:00
dbContext . Update ( user ) ;
2020-11-15 16:30:04 +00:00
_users [ user . Id ] = user ;
2020-10-30 00:30:33 +00:00
await dbContext . SaveChangesAsync ( ) . ConfigureAwait ( false ) ;
2020-05-13 02:10:35 +00:00
}
2020-05-19 23:44:55 +00:00
/// <inheritdoc/>
2020-10-30 00:30:33 +00:00
public async Task UpdatePolicyAsync ( Guid userId , UserPolicy policy )
2020-05-13 02:10:35 +00:00
{
2020-10-30 00:30:33 +00:00
await using var dbContext = _dbProvider . CreateContext ( ) ;
2020-07-19 02:29:27 +00:00
var user = dbContext . Users
. Include ( u = > u . Permissions )
. Include ( u = > u . Preferences )
. Include ( u = > u . AccessSchedules )
. Include ( u = > u . ProfileImage )
. FirstOrDefault ( u = > u . Id = = userId )
? ? throw new ArgumentException ( "No user exists with given Id!" ) ;
2020-06-13 23:30:45 +00:00
// The default number of login attempts is 3, but for some god forsaken reason it's sent to the server as "0"
2020-05-23 20:07:42 +00:00
int? maxLoginAttempts = policy . LoginAttemptsBeforeLockout switch
2020-05-23 18:53:24 +00:00
{
- 1 = > null ,
0 = > 3 ,
_ = > policy . LoginAttemptsBeforeLockout
} ;
2020-05-13 02:10:35 +00:00
user . MaxParentalAgeRating = policy . MaxParentalRating ;
user . EnableUserPreferenceAccess = policy . EnableUserPreferenceAccess ;
user . RemoteClientBitrateLimit = policy . RemoteClientBitrateLimit ;
2020-05-15 21:24:01 +00:00
user . AuthenticationProviderId = policy . AuthenticationProviderId ;
2020-05-13 02:10:35 +00:00
user . PasswordResetProviderId = policy . PasswordResetProviderId ;
user . InvalidLoginAttemptCount = policy . InvalidLoginAttemptCount ;
2020-05-23 20:07:42 +00:00
user . LoginAttemptsBeforeLockout = maxLoginAttempts ;
2020-10-04 15:50:00 +00:00
user . MaxActiveSessions = policy . MaxActiveSessions ;
2020-05-27 00:52:05 +00:00
user . SyncPlayAccess = policy . SyncPlayAccess ;
2020-05-13 02:10:35 +00:00
user . SetPermission ( PermissionKind . IsAdministrator , policy . IsAdministrator ) ;
user . SetPermission ( PermissionKind . IsHidden , policy . IsHidden ) ;
user . SetPermission ( PermissionKind . IsDisabled , policy . IsDisabled ) ;
user . SetPermission ( PermissionKind . EnableSharedDeviceControl , policy . EnableSharedDeviceControl ) ;
user . SetPermission ( PermissionKind . EnableRemoteAccess , policy . EnableRemoteAccess ) ;
user . SetPermission ( PermissionKind . EnableLiveTvManagement , policy . EnableLiveTvManagement ) ;
user . SetPermission ( PermissionKind . EnableLiveTvAccess , policy . EnableLiveTvAccess ) ;
user . SetPermission ( PermissionKind . EnableMediaPlayback , policy . EnableMediaPlayback ) ;
user . SetPermission ( PermissionKind . EnableAudioPlaybackTranscoding , policy . EnableAudioPlaybackTranscoding ) ;
user . SetPermission ( PermissionKind . EnableVideoPlaybackTranscoding , policy . EnableVideoPlaybackTranscoding ) ;
user . SetPermission ( PermissionKind . EnableContentDeletion , policy . EnableContentDeletion ) ;
user . SetPermission ( PermissionKind . EnableContentDownloading , policy . EnableContentDownloading ) ;
user . SetPermission ( PermissionKind . EnableSyncTranscoding , policy . EnableSyncTranscoding ) ;
user . SetPermission ( PermissionKind . EnableMediaConversion , policy . EnableMediaConversion ) ;
user . SetPermission ( PermissionKind . EnableAllChannels , policy . EnableAllChannels ) ;
user . SetPermission ( PermissionKind . EnableAllDevices , policy . EnableAllDevices ) ;
user . SetPermission ( PermissionKind . EnableAllFolders , policy . EnableAllFolders ) ;
user . SetPermission ( PermissionKind . EnableRemoteControlOfOtherUsers , policy . EnableRemoteControlOfOtherUsers ) ;
user . SetPermission ( PermissionKind . EnablePlaybackRemuxing , policy . EnablePlaybackRemuxing ) ;
user . SetPermission ( PermissionKind . ForceRemoteSourceTranscoding , policy . ForceRemoteSourceTranscoding ) ;
user . SetPermission ( PermissionKind . EnablePublicSharing , policy . EnablePublicSharing ) ;
user . AccessSchedules . Clear ( ) ;
foreach ( var policyAccessSchedule in policy . AccessSchedules )
{
user . AccessSchedules . Add ( policyAccessSchedule ) ;
}
2020-06-13 22:23:13 +00:00
// TODO: fix this at some point
2020-12-11 22:00:43 +00:00
user . SetPreference ( PreferenceKind . BlockUnratedItems , policy . BlockUnratedItems ? ? Array . Empty < UnratedItem > ( ) ) ;
2020-05-13 02:10:35 +00:00
user . SetPreference ( PreferenceKind . BlockedTags , policy . BlockedTags ) ;
2020-12-11 22:00:43 +00:00
user . SetPreference ( PreferenceKind . EnabledChannels , policy . EnabledChannels ) ;
2020-05-13 02:10:35 +00:00
user . SetPreference ( PreferenceKind . EnabledDevices , policy . EnabledDevices ) ;
2020-12-11 22:00:43 +00:00
user . SetPreference ( PreferenceKind . EnabledFolders , policy . EnabledFolders ) ;
2020-05-13 02:10:35 +00:00
user . SetPreference ( PreferenceKind . EnableContentDeletionFromFolders , policy . EnableContentDeletionFromFolders ) ;
2020-05-28 05:08:37 +00:00
dbContext . Update ( user ) ;
2020-11-15 16:30:04 +00:00
_users [ user . Id ] = user ;
2020-10-30 00:30:33 +00:00
await dbContext . SaveChangesAsync ( ) . ConfigureAwait ( false ) ;
2020-05-13 02:10:35 +00:00
}
2020-06-11 22:28:49 +00:00
/// <inheritdoc/>
2020-10-30 00:30:33 +00:00
public async Task ClearProfileImageAsync ( User user )
2020-06-11 21:51:02 +00:00
{
2021-10-04 13:43:40 +00:00
if ( user . ProfileImage = = null )
{
return ;
}
2020-10-30 00:30:33 +00:00
await using var dbContext = _dbProvider . CreateContext ( ) ;
2020-06-13 20:38:17 +00:00
dbContext . Remove ( user . ProfileImage ) ;
2020-10-30 00:30:33 +00:00
await dbContext . SaveChangesAsync ( ) . ConfigureAwait ( false ) ;
2020-10-27 16:28:37 +00:00
user . ProfileImage = null ;
2020-11-15 16:30:04 +00:00
_users [ user . Id ] = user ;
2020-06-11 21:51:02 +00:00
}
2021-02-17 10:30:14 +00:00
internal static void ThrowIfInvalidUsername ( string name )
{
if ( ! string . IsNullOrWhiteSpace ( name ) & & IsValidUsername ( name ) )
{
return ;
}
throw new ArgumentException ( "Usernames can contain unicode symbols, numbers (0-9), dashes (-), underscores (_), apostrophes ('), and periods (.)" , nameof ( name ) ) ;
}
2020-05-29 19:32:31 +00:00
private static bool IsValidUsername ( string name )
2020-05-13 02:10:35 +00:00
{
2020-05-15 21:24:01 +00:00
// This is some regex that matches only on unicode "word" characters, as well as -, _ and @
// In theory this will cut out most if not all 'control' characters which should help minimize any weirdness
2020-07-26 11:57:22 +00:00
// Usernames can contain letters (a-z + whatever else unicode is cool with), numbers (0-9), at-signs (@), dashes (-), underscores (_), apostrophes ('), periods (.) and spaces ( )
2021-02-17 10:30:14 +00:00
return Regex . IsMatch ( name , @"^[\w\ \-'._@]+$" ) ;
2020-05-13 02:10:35 +00:00
}
2020-05-15 21:24:01 +00:00
private IAuthenticationProvider GetAuthenticationProvider ( User user )
2020-05-13 02:10:35 +00:00
{
return GetAuthenticationProviders ( user ) [ 0 ] ;
}
2020-05-15 21:24:01 +00:00
private IPasswordResetProvider GetPasswordResetProvider ( User user )
2020-05-13 02:10:35 +00:00
{
return GetPasswordResetProviders ( user ) [ 0 ] ;
}
2020-06-09 16:21:21 +00:00
private IList < IAuthenticationProvider > GetAuthenticationProviders ( User ? user )
2020-05-13 02:10:35 +00:00
{
var authenticationProviderId = user ? . AuthenticationProviderId ;
var providers = _authenticationProviders . Where ( i = > i . IsEnabled ) . ToList ( ) ;
if ( ! string . IsNullOrEmpty ( authenticationProviderId ) )
{
providers = providers . Where ( i = > string . Equals ( authenticationProviderId , i . GetType ( ) . FullName , StringComparison . OrdinalIgnoreCase ) ) . ToList ( ) ;
}
if ( providers . Count = = 0 )
{
// Assign the user to the InvalidAuthProvider since no configured auth provider was valid/found
_logger . LogWarning (
2020-05-15 21:24:01 +00:00
"User {Username} was found with invalid/missing Authentication Provider {AuthenticationProviderId}. Assigning user to InvalidAuthProvider until this is corrected" ,
2020-05-13 02:10:35 +00:00
user ? . Username ,
user ? . AuthenticationProviderId ) ;
providers = new List < IAuthenticationProvider >
{
_invalidAuthProvider
} ;
}
return providers ;
}
2020-05-15 21:24:01 +00:00
private IList < IPasswordResetProvider > GetPasswordResetProviders ( User user )
2020-05-13 02:10:35 +00:00
{
2020-10-03 15:03:23 +00:00
var passwordResetProviderId = user . PasswordResetProviderId ;
2020-05-13 02:10:35 +00:00
var providers = _passwordResetProviders . Where ( i = > i . IsEnabled ) . ToArray ( ) ;
if ( ! string . IsNullOrEmpty ( passwordResetProviderId ) )
{
providers = providers . Where ( i = >
string . Equals ( passwordResetProviderId , i . GetType ( ) . FullName , StringComparison . OrdinalIgnoreCase ) )
. ToArray ( ) ;
}
if ( providers . Length = = 0 )
{
providers = new IPasswordResetProvider [ ]
{
_defaultPasswordResetProvider
} ;
}
return providers ;
}
2020-06-09 16:21:21 +00:00
private async Task < ( IAuthenticationProvider ? authenticationProvider , string username , bool success ) > AuthenticateLocalUser (
2020-05-13 02:10:35 +00:00
string username ,
string password ,
2020-06-09 16:21:21 +00:00
User ? user ,
2020-05-13 02:10:35 +00:00
string remoteEndPoint )
{
bool success = false ;
2020-06-09 16:21:21 +00:00
IAuthenticationProvider ? authenticationProvider = null ;
2020-05-13 02:10:35 +00:00
foreach ( var provider in GetAuthenticationProviders ( user ) )
{
var providerAuthResult =
await AuthenticateWithProvider ( provider , username , password , user ) . ConfigureAwait ( false ) ;
var updatedUsername = providerAuthResult . username ;
success = providerAuthResult . success ;
if ( success )
{
authenticationProvider = provider ;
username = updatedUsername ;
break ;
}
}
if ( ! success
& & _networkManager . IsInLocalNetwork ( remoteEndPoint )
& & user ? . EnableLocalPassword = = true
& & ! string . IsNullOrEmpty ( user . EasyPassword ) )
{
// Check easy password
var passwordHash = PasswordHash . Parse ( user . EasyPassword ) ;
var hash = _cryptoProvider . ComputeHash (
passwordHash . Id ,
Encoding . UTF8 . GetBytes ( password ) ,
passwordHash . Salt . ToArray ( ) ) ;
success = passwordHash . Hash . SequenceEqual ( hash ) ;
}
return ( authenticationProvider , username , success ) ;
}
private async Task < ( string username , bool success ) > AuthenticateWithProvider (
IAuthenticationProvider provider ,
string username ,
string password ,
2020-06-09 16:21:21 +00:00
User ? resolvedUser )
2020-05-13 02:10:35 +00:00
{
try
{
var authenticationResult = provider is IRequiresResolvedUser requiresResolvedUser
? await requiresResolvedUser . Authenticate ( username , password , resolvedUser ) . ConfigureAwait ( false )
: await provider . Authenticate ( username , password ) . ConfigureAwait ( false ) ;
if ( authenticationResult . Username ! = username )
{
_logger . LogDebug ( "Authentication provider provided updated username {1}" , authenticationResult . Username ) ;
username = authenticationResult . Username ;
}
return ( username , true ) ;
}
catch ( AuthenticationException ex )
{
_logger . LogError ( ex , "Error authenticating with provider {Provider}" , provider . Name ) ;
return ( username , false ) ;
}
}
2020-07-21 18:25:52 +00:00
private async Task IncrementInvalidLoginAttemptCount ( User user )
2020-05-13 02:10:35 +00:00
{
2020-05-23 01:45:31 +00:00
user . InvalidLoginAttemptCount + + ;
2020-05-13 02:10:35 +00:00
int? maxInvalidLogins = user . LoginAttemptsBeforeLockout ;
2020-05-23 01:45:31 +00:00
if ( maxInvalidLogins . HasValue & & user . InvalidLoginAttemptCount > = maxInvalidLogins )
2020-05-13 02:10:35 +00:00
{
user . SetPermission ( PermissionKind . IsDisabled , true ) ;
2020-08-15 19:55:15 +00:00
await _eventManager . PublishAsync ( new UserLockedOutEventArgs ( user ) ) . ConfigureAwait ( false ) ;
2020-05-13 02:10:35 +00:00
_logger . LogWarning (
2020-05-15 21:24:01 +00:00
"Disabling user {Username} due to {Attempts} unsuccessful login attempts." ,
2020-05-13 02:10:35 +00:00
user . Username ,
2020-05-23 01:45:31 +00:00
user . InvalidLoginAttemptCount ) ;
2020-05-13 02:10:35 +00:00
}
2020-07-21 18:25:52 +00:00
await UpdateUserAsync ( user ) . ConfigureAwait ( false ) ;
2020-05-13 02:10:35 +00:00
}
}
}