Microsoft.AspNet.Identity.PasswordValidator可以通过一些参数进行配置,这些参数定义了最小密码长度和复杂度规则。如果用户输入的密码不符合标准,PasswordValidator将抛出一个描述违规的异常,但它没有提供其他方法来向用户描述规则(例如:"密码必须是8-12个字符,并且必须包含…"等)。
我搜索了StackOverflow和网络,一般情况下,我需要的东西一定有数百个例子,但它开始看起来像我是唯一一个需要这个的人,我知道情况并非如此。在我编写自己的实现之前,我必须问一下……
我是否完全忽略了这个问题公认的"标准"答案?
这听起来可能有点粗俗,但是向PasswordValidator的ValidateAsync()方法发送一个空字符串将完成这项工作。UserManager的默认配置示例:
var rules = await UserManager.PasswordValidator.ValidateAsync("");
rules.Errors.First() :
Passwords must be at least 6 characters.
Passwords must have at least one non letter or digit character.
Passwords must have at least one digit ('0'-'9').
Passwords must have at least one lowercase ('a'-'z').
Passwords must have at least one uppercase ('A'-'Z').
你甚至可以列出它们
var list = rules.Errors.First().Split('.');
这是我想到的,因为我需要得到一些工作,并没有找到我正在寻找的固定标准答案。为了方便起见,我将其实现为PasswordValidator类的扩展,尽管这并不是严格必要的。
public static string ToDescription(this PasswordValidator validator)
{
var options = new List<string>();
if (validator.RequireUppercase) options.Add("upper-case letters");
if (validator.RequireLowercase) options.Add("lower-case letters");
if (validator.RequireDigit) options.Add("numbers");
if (validator.RequireNonLetterOrDigit) options.Add("special characters");
var result = "Passwords must be at least "
+ validator.RequiredLength + " characters long"
+ (options.Any() ? ", and include " + options.OxfordJoin() : ".");
return result;
}
我还添加了OxfordJoin扩展,将需求列表转换为格式良好的句子:
public static string OxfordJoin(this IEnumerable<string> source)
{
var array = source.ToArray();
var joined = string.Join(", ", array);
var result = (array.Length > 2)
? joined.ReplaceLast(",", ", and")
: joined.ReplaceLast(",", " and");
return result;
}
…最后是"so standard it should just in the framework already" ReplaceLast implementation:
public static string ReplaceLast(this string value, string oldValue, string newValue)
{
var position = value.LastIndexOf(oldValue, StringComparison.InvariantCultureIgnoreCase);
var result = (position > -1)
? value.Substring(0, position) + newValue + value.Substring(position + 1)
: value;
return result;
}