我对Moq很困惑,我不确定这里有什么问题。
我想测试依赖于ILeadStorageService的LeadService,我想以这种方式配置Moq - 返回与Setup中传递的GUID相匹配的对象。
问题在于Moq设置/返回行,因为当我将依赖对象替换为其实例时 - 测试通过,但这是完全错误的。我不想只测试LeadService,而不是依赖存储。
public LeadService( IConfigurationDbContext configurationDbContext,
ILeadStorageService leadStorageService,
ILeadDeliveryService deliveryService)
{
this.configurationDbContext = configurationDbContext;
this.leadStorageService = leadStorageService;
this.deliveryService = deliveryService;
}
经过测试的方法
public TestLeadResponse ProcessTestLead(TestLeadRequest request)
{
var response = new TestLeadResponse()
{
Status = TestLeadStatus.Ok
};
try
{
var lead = leadStorageService.Get(request.LeadId);
if (lead == null)
{
throw new LeadNotFoundException(request.LeadId);
}
var buyerContract =
configurationDbContext.BuyerContracts.SingleOrDefault(bc => bc.Id == request.BuyerContractId);
if (buyerContract == null)
{
throw new BuyerContractNotFoundException(request.BuyerContractId);
}
response.DeliveryEntry = deliveryService.DeliverLead(lead, buyerContract);
}
catch (LeadNotFoundException e)
{
response.Status = TestLeadStatus.LeadNotFound;
response.StatusDescription = e.Message;
}
catch (BuyerContractNotFoundException e)
{
response.Status = TestLeadStatus.BuyerContractNotFound;
response.StatusDescription = e.Message;
}
return response;
}
然后进行测试准备:
[TestInitialize]
public void Initialize()
{
_leadIdStr = "2c3ac0c0-f0c2-4eb0-a55e-600ae3ada221";
_dbcontext = new ConfigurationDbContext();
_lead = PrepareLeadObject();
_buyerContract = PrepareBuyerContractObject(Id : 1, BuyerContractId : 1, BuyerTag: "GAME");
_leadDeliveryMock = new Mock<ILeadDeliveryService>();
_leadStorageMock = new Mock<ILeadStorageService>();
_leadStorageService = new LeadStorageService("LeadGeneration_Dev");
}
private Lead PrepareLeadObject()
{
var lead = new Lead() {CountryId = 1, Country = "NL", Id = Guid.Parse(_leadIdStr)};
return lead;
}
和测试本身:
[TestMethod]
public void LeadServiceTest_ProcessTestLeadWithWrongBuyerContractIDThrowsBuyerContractNotFoundException()
{
_leadDeliveryMock
.Setup(methodCall => methodCall.DeliverLead(_lead, _buyerContract))
.Returns<LeadDeliveryEntry>((r) => PrepareLeadDeliveryEntry());
// here is the problem!!!!
_leadStorageMock.Setup(mc => mc.Get(_leadId)).Returns((Lead l) => PrepareLeadObject());
//if i change to real object - test passes
//_service = new LeadService(_dbcontext, _leadStorageService, _leadDeliveryMock.Object);
_service = new LeadService(_dbcontext, _leadStorageMock.Object, _leadDeliveryMock.Object);
var response = _service.ProcessTestLead(new TestLeadRequest() { BuyerContractId = int.MaxValue, LeadId = _leadId });
Assert.IsNotNull(response);
Assert.AreEqual(response.Status, TestLeadStatus.BuyerContractNotFound);
}
我在_leadStorageMock.Setup()中缺少什么。返回()?
答案 0 :(得分:4)
Returns
扩展方法接受与您正在模拟的方法具有相同参数的委托。并且这些参数将在调用mocked方法期间传递给委托。因此,不是Lead
对象,而是获取传递给mc.Get
方法的参数 - 引导ID:
_leadStorageMock.Setup(mc => mc.Get(_leadId))
.Returns((Guid leadId) => PrepareLeadObject());
在返回值时,检查与访问调用参数相关的QuickStart部分。
请注意,有一堆Returns
扩展方法接受值函数作为参数:
Returns<T>(Func<T, TResult> valueFunction);
Returns<T1, T2>(Func<T1, T2, TResult> valueFunction);
Returns<T1, T2, T3>(Func<T1, T2, T3, TResult> valueFunction);
// etc
正如您所看到的那些值函数计算从mocked方法返回的值,但它们都接收不同数量的参数(最多16个)。这些参数将与您正在模拟的方法的参数完全匹配,并且在调用mocked方法期间它们将被传递给valueFunction
。因此,如果您使用两个参数模拟某个函数,则应使用相应的扩展名:
mock.Setup(m => m.Activate(It.IsAny<int>(), It.IsAny<bool>())
.Returns((int i, bool b) => b ? i : 0); // Func<T1, T2, TResult>