Как высмеять вызов метода, который принимает динамический объект

Скажем, у меня есть следующее:

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

И у меня есть следующий код, который я хочу проверить:

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

Как мне высмеять этот вызов?

Используя Moq, я устал делать это:

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});

И это не сработало. Возвращенные результаты: null

Я также попытался сделать следующее с Rhino Mocks:

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

Не повезло. Нуль снова.

Итак, как бы я это сделал?

Спасибо,

Ответ 1

В решении должно использоваться совпадение It.Is<object> вместе с отражением. Вы не можете использовать динамическое выражение в деревьях выражений, поэтому It.Is<dynamic> не будет работать, поэтому вам нужно отразить, чтобы получить значение свойства по имени:

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 });

Где GetPropertyValue - маленький помощник:

public static class ReflectionExtensions
{
    public static T GetPropertyValue<T>(this object obj, string propertyName)
    {
        return (T) obj.GetType().GetProperty(propertyName).GetValue(obj, null);
    }
}

Ответ 2

Прежде всего, анонимные объекты на самом деле не являются dynamic.

Если вы использовали dynamic объекты типа

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

вы могли бы издеваться над ним, как

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

путем создания настраиваемого аргумента:

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);
    }
}

Ответ 3

Moq при условии It.IsAny<T> для этого случая

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

* dynamic - любой объект