117 lines
4.9 KiB
C#
117 lines
4.9 KiB
C#
// Copyright (c) .NET Foundation. All rights reserved.
|
|
// Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information.
|
|
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using System.Threading.Tasks;
|
|
|
|
namespace Microsoft.AspNet.Identity
|
|
{
|
|
/// <summary>
|
|
/// Provides the default password policy for Identity.
|
|
/// </summary>
|
|
/// <typeparam name="TUser">The type that represents a user.</typeparam>
|
|
public class PasswordValidator<TUser> : IPasswordValidator<TUser> where TUser : class
|
|
{
|
|
/// <summary>
|
|
/// Constructions a new instance of <see cref="PasswordValidator{TUser}"/>.
|
|
/// </summary>
|
|
/// <param name="errors">The <see cref="IdentityErrorDescriber"/> to retrieve error text from.</param>
|
|
public PasswordValidator(IdentityErrorDescriber errors = null)
|
|
{
|
|
Describer = errors ?? new IdentityErrorDescriber();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the <see cref="IdentityErrorDescriber"/> used to supply error text.
|
|
/// </summary>
|
|
/// <value>The <see cref="IdentityErrorDescriber"/> used to supply error text.</value>
|
|
public IdentityErrorDescriber Describer { get; private set; }
|
|
|
|
/// <summary>
|
|
/// Validates a password as an asynchronous operation.
|
|
/// </summary>
|
|
/// <param name="manager">The <see cref="UserManager{TUser}"/> to retrieve the <paramref name="user"/> properties from.</param>
|
|
/// <param name="user">The user whose password should be validated.</param>
|
|
/// <param name="password">The password supplied for validation</param>
|
|
/// <returns>The task object representing the asynchronous operation.</returns>
|
|
public virtual Task<IdentityResult> ValidateAsync(UserManager<TUser> manager, TUser user, string password)
|
|
{
|
|
if (password == null)
|
|
{
|
|
throw new ArgumentNullException(nameof(password));
|
|
}
|
|
if (manager == null)
|
|
{
|
|
throw new ArgumentNullException(nameof(manager));
|
|
}
|
|
var errors = new List<IdentityError>();
|
|
var options = manager.Options.Password;
|
|
if (string.IsNullOrWhiteSpace(password) || password.Length < options.RequiredLength)
|
|
{
|
|
errors.Add(Describer.PasswordTooShort(options.RequiredLength));
|
|
}
|
|
if (options.RequireNonLetterOrDigit && password.All(IsLetterOrDigit))
|
|
{
|
|
errors.Add(Describer.PasswordRequiresNonLetterAndDigit());
|
|
}
|
|
if (options.RequireDigit && !password.Any(IsDigit))
|
|
{
|
|
errors.Add(Describer.PasswordRequiresDigit());
|
|
}
|
|
if (options.RequireLowercase && !password.Any(IsLower))
|
|
{
|
|
errors.Add(Describer.PasswordRequiresLower());
|
|
}
|
|
if (options.RequireUppercase && !password.Any(IsUpper))
|
|
{
|
|
errors.Add(Describer.PasswordRequiresUpper());
|
|
}
|
|
return
|
|
Task.FromResult(errors.Count == 0
|
|
? IdentityResult.Success
|
|
: IdentityResult.Failed(errors.ToArray()));
|
|
}
|
|
|
|
/// <summary>
|
|
/// Returns a flag indicting whether the supplied character is a digit.
|
|
/// </summary>
|
|
/// <param name="c">The character to check if it is a digit.</param>
|
|
/// <returns>True if the character is a digit, otherwise false.</returns>
|
|
public virtual bool IsDigit(char c)
|
|
{
|
|
return c >= '0' && c <= '9';
|
|
}
|
|
|
|
/// <summary>
|
|
/// Returns a flag indicting whether the supplied character is a lower case ASCII letter.
|
|
/// </summary>
|
|
/// <param name="c">The character to check if it is a lower case ASCII letter.</param>
|
|
/// <returns>True if the character is a lower case ASCII letter, otherwise false.</returns>
|
|
public virtual bool IsLower(char c)
|
|
{
|
|
return c >= 'a' && c <= 'z';
|
|
}
|
|
|
|
/// <summary>
|
|
/// Returns a flag indicting whether the supplied character is an upper case ASCII letter.
|
|
/// </summary>
|
|
/// <param name="c">The character to check if it is an upper case ASCII letter.</param>
|
|
/// <returns>True if the character is an upper case ASCII letter, otherwise false.</returns>
|
|
public virtual bool IsUpper(char c)
|
|
{
|
|
return c >= 'A' && c <= 'Z';
|
|
}
|
|
|
|
/// <summary>
|
|
/// Returns a flag indicting whether the supplied character is an ASCII letter or digit.
|
|
/// </summary>
|
|
/// <param name="c">The character to check if it is an ASCII letter or digit.</param>
|
|
/// <returns>True if the character is an ASCII letter or digit, otherwise false.</returns>
|
|
public virtual bool IsLetterOrDigit(char c)
|
|
{
|
|
return IsUpper(c) || IsLower(c) || IsDigit(c);
|
|
}
|
|
}
|
|
} |