有时在Command和DTO之间层次比较多,写了个验证Command的函数,能实现递归验证.
比如下面这些有层级关系的class定义,能通过一句代码来进行验证:
class A { [Required] public B B { get; set; } } class B { [Range(10, 20)] public int C { get; set; } [Range(typeof(Guid), "00000000-0000-0000-0000-000000000001", "00000000-0000-0000-0000-000000000010")] public Guid aa { get; set; } public CC ddd { get; set; } } class CC { [Range(10, 20)] public int DDDDD { get; set; } [Required(ErrorMessage = "必须填写用户名")] [Display(Name = "用户名")] public string UserName { get; set; } [Required(ErrorMessage = "必须填写密码")] [DataType(DataType.Password)] [Display(Name = "密码")] public string Password { get; set; } }
验证函数调用:
List<BrokenRule> errors = obj.IsValid();
代码如下:
public class BrokenRule { public string Key { get; set; } public string Message { get; set; } } public static class DataAnnotationHelper { public static List<BrokenRule> IsValid<T>(this T o, bool only1Level = false) { return IsValid(typeof(T), o, only1Level); } private static List<BrokenRule> IsValid(Type t, object o, bool only1Level) { List<BrokenRule> errors = new List<BrokenRule>(); var descriptor = GetTypeDescriptor(t); foreach (PropertyDescriptor propertyDescriptor in descriptor.GetProperties()) { foreach (var validationAttribute in propertyDescriptor.Attributes.OfType<ValidationAttribute>()) { if (!validationAttribute.IsValid(propertyDescriptor.GetValue(o))) { BrokenRule error = new BrokenRule(); error.Key = propertyDescriptor.Name; error.Message = validationAttribute.FormatErrorMessage(propertyDescriptor.Name); errors.Add(error); } } } if (!only1Level) { if (o.GetType().IsClass&&!o.GetType().Equals(typeof(string))) { foreach (var p in o.GetType().GetProperties()) { object pValue = p.GetValue(o, null); if (pValue != null) { List<BrokenRule> pErrors = IsValid(p.PropertyType, pValue, only1Level); errors.AddRange(pErrors); } } } } return errors; } private static ICustomTypeDescriptor GetTypeDescriptor(Type type) { return new AssociatedMetadataTypeTypeDescriptionProvider(type).GetTypeDescriptor(type); } }
效果图: