在问题的最后: Using Moq to set indexers in C#,有一个问题,有人突出了我遇到的问题。但他们没有找到解决方案。
具体来说,我正在尝试使用通用It.IsAny<string>
作为密钥的匹配器并通过It.IsAny<object>
设置值。通过索引访问并设置值时,它永远不会匹配,也不会访问我的回调方法。所以我的单元测试失败了。
var stateTable = new HashTable;
var httpSession = new Mock<HttpSessionStateBase>();
//works via httpSession.Add(key, value);
httpSession.Setup(x => x.Add(It.IsAny<string>(), It.IsAny<object>()))
.Callback((string index, object value) => {
var i = index;
var v = value;
stateData[i] = v;
});
//does not work via httpSession[key] = value;
httpSession.SetupSet(x => x[It.IsAny<string>()] = It.IsAny<object>())
.Callback( (string index, object value) => {
var i = index;
var v = value;
stateData[i] = v;
});
我正在使用Moq 4.0.10827
答案 0 :(得分:8)
根据我的经验,这永远不会有效,你不能将It.IsAny用作索引器表达式中的匹配器。但是,如果在索引器中放置具体值,它将匹配。 例如,以下内容确实有效:
httpSession.SetupSet(x => x["someValue"] = It.IsAny<object>())
.Callback( (string index, object value) => {
var i = index;
var v = value;
stateData[i] = v;
});
答案 1 :(得分:2)
这是我为验证阅读所做的工作。
_httpSessionStateBaseMock.VerifySet(x => x["keyname"] = It.IsAny<YourObject>(), Times.Once());
我刚读过的读物
_httpSessionStateBaseMock.Verify(x => x["keyname"],Times.Once());
答案 2 :(得分:0)
我不喜欢它,但我能够使用新的继承类来解决它。 然后我重写索引器并调用一个可以模拟的虚方法。
public class HttpSessionStateBaseProxy : HttpSessionStateBase
{
public virtual void SetItem(string key, object value)
{
base[key] = value;
}
public override object this[string name]
{
get
{
return base[name];
}
set
{
SetItem (name, value);
}
}
}
并且模仿是通过:
完成的Mock<HttpSessionStateBaseProxy> Session = new Mock<HttpSessionStateBaseProxy>(MockBehavior.Loose);
Session.CallBase = true;
Session.Setup(x => x.SetItem(It.IsAny<string>(),It.IsAny<object>()))
.Callback<string,object>((string index, object value) =>
{
if (!this.SessionItems.Contains(index)) this.SessionItems.Add(index, value);
else this.SessionItems[index] = value;
});
Session.Setup(s => s[It.IsAny<string>()]).Returns<string>(key =>
{
return this.SessionItems[key];
});