如何替换在被测函数中实例化的对象的函数调用

How to substitute a function call from object that is intantiated in function under test

我正在尝试编写一些单元测试。该应用程序有许多外部 API 调用,我想使用 NSubstitute 对其进行模拟。问题是这些调用使用需要在函数中实例化的服务对象,并且在替换后不能在构造函数中传递。

例如,在下面的代码中,我向 Quickbooks 添加了一个帐户:

public async Task AddQuickbooksAsync(int accountId)
{
    var qbChannel = await GetQuickbooksChannel();
    var account = await GetAsync(accountId);
    var quickbooksBO = new QuickbooksBO(qbChannel);
    quickbooksBO.AddAccount(account);
    quickbooksBO.UpdateRefreshedTokens(qbChannel);
    await context.SaveChangesAsync();
}

我想使用 NSubstitute 模拟以下函数调用,但不知道如何操作:

var quickbooksBO = new QuickbooksBO(qbChannel);
quickbooksBO.AddAccount(account);

我该怎么做?

您不能模拟局部变量。您可以创建一个虚拟方法 return QuickbooksBO 并替换它。

public async Task AddQuickbooksAsync(int accountId)
{
    var qbChannel = await GetQuickbooksChannel();
    var account = await GetAsync(accountId);
    var quickbooksBO = GetQbBO(qbChannel);
    quickbooksBO.AddAccount(account);
    quickbooksBO.UpdateRefreshedTokens(qbChannel);
    await context.SaveChangesAsync();
}

public virtual QuickbooksBO GetQbBO(Channel qbChannel)
{
    return new QuickBooks(qbChannel);
}

之后您可以在您的服务中替换它:

var s = Substitute.ForPartsOf<Service>();
s.GetQbBO(default).ReturnsForAnyArgs(substituteForQbBO);

您需要确定要模拟 QuickBooksBO 的哪些部分。通常这样的依赖应该是接口,没有接口单元测试是相当有限的。

如果 QuickbooksBO 方法是虚拟的,你可以做这样的事情来得到你的 substituteForQbBO:

var substituteForQbBO = Substitute.ForPartsOf<QuickbooksBO>();
substituteForQbBO.WhenForAnyArgs(x => x.AddAccount(default)).DoNotCallBase();
substituteForQbBO.WhenForAnyArgs(x => x.UpdateRefreshedTokens(default)).DoNotCallBase();