测试EF保存更改修饰符.传入DbPropertyValues

本文关键字:传入 DbPropertyValues EF 保存更改 测试 | 更新日期: 2023-09-27 18:17:32

试图通过覆盖EF SaveChanges方法在c#中做一些业务逻辑。
这个想法是有一些高级的计算,比如如果这个字段改变了,更新这个字段。这个字段是子类减去其他一些字段的和,你知道高级商业垃圾。

因为它真的很复杂,我们想测试它的填充物。添加测试工作得很好,但我们似乎无法测试更新的测试,因为我们已经编写了一个接口,其中有问题的方法通过了签名是这样的

void Update(object entity, DbPropertyValues currentValues, DbPropertyValues originalValues);

当用完整的EF调用它时,效果很好

    public override int SaveChanges()
    {
        var added = ChangeTracker.Entries().Where(p => p.State == EntityState.Added).Select(p => p.Entity);
        var updated = ChangeTracker.Entries().Where(p => p.State == EntityState.Modified).Select(p => p);
        var context = new ChangeAndValidationContext();
        foreach (var item in added)
        {
            var strategy = context.SelectStrategy(item);
            strategy.Add(item);
        }
        foreach (var item in updated)
        {
            var strategy = context.SelectStrategy(item);
            strategy.Update(item.Entity, item.CurrentValues, item.OriginalValues);
        }
        return base.SaveChanges();
   }

我们只是不能弄清楚如何在DbPropertyValues原始或更新我们的测试。请帮助我们找出如何测试该方法。

测试EF保存更改修饰符.传入DbPropertyValues

如果您有Visual Studio 2012 Update 2+,您可以通过右键单击测试项目中的项目引用来为EntityFramework"添加Fakes Assembly"。

添加后,您可以创建完全在您控制之下的System.Data.Entity.Infrastructure.Fakes.ShimDbPropertyValues实例。例如

var shim = new System.Data.Entity.Infrastructure.Fakes.ShimDbPropertyValues();
shim.ItemGetString = s => "Hello, World!";

GetString被调用到那个fake/shim DbPropertyValues时,它将返回"Hello, World!"

更多细节在这里:https://msdn.microsoft.com/en-us/library/hh549175.aspx

我决定更好的方法是改变策略的预期。而不是

void Update(object entity, DbPropertyValues currentValues, DbPropertyValues originalValues);

我让它接受

void Update(object entity, Dictionary<string, object> currentValues, Dictionary<string, object> originalValues);
这意味着我改变了传递给更新方法的值
foreach (var item in updated)
{
      var strategy = context.SelectStrategy(item);
      strategy.Update(item.Entity, item.CurrentValues.ValuesToValuesDictionary(), item.OriginalValues.ValuesToValuesDictionary());
}

然后我创建了扩展方法

public static class DbPropertyValueExtensions
{
    public static Dictionary<string, object> ValuesToValuesDictionary(this DbPropertyValues vals)
    {
        var retVal = new Dictionary<string, object>();
        foreach (var propertyName in vals.PropertyNames)
        {
            if (!retVal.ContainsKey(propertyName))
            {
                retVal.Add(propertyName, vals[propertyName]);
            }
        }
        return retVal;
    }
}

这意味着我的测试需要在那些字典中通过。

    [Test]
    public void DateLastModifiedUpdatesOnUpdate()
    {
        //Arrange
        var toTest = LossFactoryHelper.Create();
        var lossCheckAndValidationAddStrategy = new LossChangeAndValidationStrategy();
        var now = DateTime.UtcNow;
        var originalValues = toTest.GetValuesNow();
        //Act

        toTest.mny_deductible = -1;
        var currentValues = toTest.GetValuesNow();
        lossCheckAndValidationAddStrategy.Update(toTest, originalValues, currentValues);
        //Assert
        Assert.GreaterOrEqual(toTest.clc_DateLastModified, now);
    }

和Extension方法,以帮助获得值的快照,而不必一遍又一遍地创建字典

public static class ReflectionToGetCurrentValuesExtension
{
    public static Dictionary<string, object> GetValuesNow(this object obj)
    {
        var retVal = new Dictionary<string, object>();
        var type = obj.GetType();
        PropertyInfo[] properties = type.GetProperties();
        foreach (PropertyInfo property in properties)
        {
            if (property.CanRead && property.CanWrite)
            {
                if (!retVal.ContainsKey(property.Name))
                {
                    retVal.Add(property.Name, property.GetValue(obj));
                }
            }
        }
        return retVal;
    }
}