0

我有一个实现 IValidatableObject 的 POCO 对象。

public class Documentation : IValidatableObject
{
    [Key]
    public int DocumentationId { get; set; }

    [ForeignKey("Project")]
    public int ProjectId { get; set; }

    public virtual Project Project { get; set; }

    public string FileGuid { get; set; }

    public IEnumerable<ValidationResult> Validate(ValidationContext validationContext)
    {
        return new[] { new ValidationResult("File has not been uploaded", new[] { "FileGuid" }) };
    }

}

为什么 DbContext 会运行验证而 DbDomainService 不会?

此测试通过了 DbContext:

    [TestMethod, ExpectedException(typeof(DbEntityValidationException))]
    public void TestDbContext()
    {
        SampleDbContext ctx = new SampleDbContext();
        var p = new Project()
        {
            ProjectName = "UnitTest",
        };
        var d = new Documentation()
        {
            FileGuid = "UnitTestDoc",
        };
        p.Documentations = new List<Documentation>();
        p.Documentations.Add(d);
        ctx.Projects.Add(p);
        ctx.SaveChanges();
    }

虽然这没有(没有抛出异常):

    [TestMethod, ExpectedException(typeof(ValidationException))]
    public void TestDbDomain()
    {
        SampleDomainService svc = new SampleDomainService();
        svc.Initialize(ServiceProvider.CreateDomainServiceContext());
        var p = new Project()
        {
            ProjectName = "UnitTest",
        };
        var d = new Documentation()
        {
            FileGuid = "UnitTestDoc",
            Project = p,
        };
        ChangeSet changeSet = new ChangeSet(
            new [] {
                new ChangeSetEntry(1, p, null, DomainOperation.Insert),
                new ChangeSetEntry(2, d, null, DomainOperation.Insert),
            }
        );
        svc.Submit(changeSet);
    }

示例代码在这里

4

1 回答 1

0

如上所示,原始代码存在两个问题。

首先,DomainService 不会为 IValidatableObject 验证错误抛出 ValidationException。只有 DataAnnotation 验证会抛出 ValidationException。所以首先要修复的是我的测试用例:

[TestMethod] //, ExpectedException(typeof(ValidationException))]
public void TestDbDomain()
{
    //... setup

    bool success = svc.Submit(changeSet);

    bool foundError = (from item in changeSet.ChangeSetEntries
                       where item.HasError
                       from validationError in item.ValidationErrors
                       select validationError).Any();

    Assert.IsTrue(foundError);
}

其次,在 Telerik 的 JustDecompiler 的一点帮助下(非常感谢!),事实证明 DomainServices.ValidateChanges 只有在类中的某个成员已使用验证指令注释时才有效。

public class Documentation : IValidatableObject
{
    ...

    [DataType("File")]
    public string FileGuid { get; set; }

    ...
}

DataType("File") 指令不会在验证机制中触发任何验证,但它对于设置标志很有用。

如果您对细节感兴趣,请查看 System.DomainServices.Server.ValidationUtilities.ValidateObjectRecursive:

private static bool ValidateObjectRecursive(object instance, string memberPath, ValidationContext validationContext, List<ValidationResult> validationResults)
{
    MetaType metaType = MetaType.GetMetaType(instance.GetType());
    if (!metaType.RequiresValidation)
    {
        return true;
    }

    // ... checks for IValidatableObject later
}

这显然是一个错误,因为 IValidatableObject 的存在应该设置了 metaType.RequiresValidation 标志。

于 2013-06-23T14:30:15.440 回答