How to mock a method call that takes a dynamic obj

2020-03-11 03:02发布

Say I have the following:

public interface ISession 
{
   T Get<T>(dynamic filter); }
}

And I have the following code that I want to test:

var user1 = session.Get<User>(new {Name = "test 1"});
var user2 = session.Get<User>(new {Name = "test 2"});

How would I mock this call?

Using Moq, I tired doing this:

var sessionMock = new Mock<ISession>();
sessionMock.Setup(x => x.Get<User>(new {Name = "test 1")).Returns(new User{Id = 1});
sessionMock.Setup(x => x.Get<User>(new {Name = "test 1")).Returns(new User{Id = 2});

And that didn't work. The returned results is null

I also tried to do the following with Rhino Mocks:

var session = MockRepository.GenerateStub<ISession>();
session.Stub(x => x.Get<User>(new {Name = "test 1"})).Return(new User{Id=1});

No luck either. Null again.

So how would I do this?

Thanks,

3条回答
Fickle 薄情
2楼-- · 2020-03-11 03:36

Moq provided It.IsAny<T> for that case

sessionMock.Setup(x => x.Get<User>(It.IsAny<object>()).Returns(new User());

*dynamic is any object

查看更多
放我归山
3楼-- · 2020-03-11 03:41

You can use the It.Is<object> matcher together with reflection. You cannot use dynamic in expression trees so It.Is<dynamic> won't work that's why you need reflection to get the your property value by name:

sessionMock
    .Setup(x => x.Get<User>(
        It.Is<object>(d => d.GetPropertyValue<string>("Name") == "test 1")))
    .Returns(new User{Id = 1});
sessionMock
    .Setup(x => x.Get<User>(
        It.Is<object>(d => d.GetPropertyValue<string>("Name") == "test 2")))
    .Returns(new User { Id = 2 });

Where GetPropertyValue is a little helper:

public static class ReflectionExtensions
{
    public static T GetPropertyValue<T>(this object obj, string propertyName)
    {
        return (T) obj.GetType().GetProperty(propertyName).GetValue(obj, null);
    }
}
查看更多
时光不老,我们不散
4楼-- · 2020-03-11 03:44

First of all, anonymous objects are not really dynamic.

If you used dynamic objects like

dynamic user1Filter = new ExpandoObject();
user1Filter.Name = "test 1";
var user1 = session.Get<User>(user1Filter);

you could mock it like

sessionMock.Setup(x => x.Get<User>(DynamicFilter.HasName("test 1")));

by implementing custom argument matcher:

static class DynamicFilter
{
    [Matcher] public static object HasName(string name) { return null; }
    public static bool HasName(dynamic filter, string name)
    {
        string passedName = filter.Name; //dynamic expression
        return name.Equals(passedName);
    }
}
查看更多
登录 后发表回答