首页 > 解决方案 > 如何从被测函数中实例化的对象替换函数调用

问题描述

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

例如,在下面的代码中,我将一个帐户添加到 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);

我怎样才能做到这一点?

标签: c#.netmoqxunitnsubstitute

解决方案


您不能模拟局部变量。您可以创建一个将返回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();

推荐阅读