c# - 如何使用 Moq 对 Azure Key Vault 进行单元测试
问题描述
我需要模拟 Key Vault 的端点,以便知道我是否正在调用该函数来获取一次 Key Vault。
我正在使用 C# 和 Moq (Framework) 进行开发以进行测试。
界面如下:
public interface IKeyVaultConnection
{
string GetKeyVaultValue(string variableName);
}
public class KeyVaultConnection
{
public KeyVaultClient keyVaultClient;
private string endpointKeyVault;
public KeyVaultConnection(string keyVaultAddress = "DefaultEndpoint")
{
AzureServiceTokenProvider azureServiceTokenProvider = new AzureServiceTokenProvider();
keyVaultClient = new KeyVaultClient(new KeyVaultClient.AuthenticationCallback(azureServiceTokenProvider.KeyVaultTokenCallback));
endpointKeyVault = $"https://{ keyVaultAddress }.vault.azure.net";
}
private async Task<string> AsyncGetSecretValue(string keyName)
{
var secret = await keyVaultClient.GetSecretAsync($"{endpointKeyVault}/secrets/{ keyName }")
.ConfigureAwait(false);
return secret.Value;
}
public string GetKeyVaultValue(string variableName)
{
Task<string> task = Task.Run(async () => await AsyncGetSecretValue(variableName));
task.Wait();
return task.Result;
}
}
Mock<IKeyVaultConnection> mock = new Mock<IKeyVaultConnection>();
//----->>>>>> Need to setup the endpoint
mock.Setup(x => x.GetKeyVaultValue(It.IsAny<string>())).Returns(It.IsAny<string>());
// mock.Verify(x => x.GetKeyVaultValue(It.IsAny<string>()), Times.Once());
我需要的是伪造到端点的连接,以便我调用该函数一次并且我收到如下错误:
"KeyVaultErrorException: Operation returned an invalid status code 'NotFound'"
因为不提供端点。
如果我取消注释最后一行(mock.Verify(x => x.GetKeyVaultValue(It.IsAny<string>()), Times.Once());)
,我会得到:
"Expected invocation on the mock once, but was 0 times:
x => x.GetKeyVaultValue(It.IsAny<string>())"
解决方案
如果您将您的类视为一个不起眼的包装器,以便其他类可以在ISomethingThatReturnsValues
不知道的情况下依赖,KeyVaultClient
那么模拟KeyVaultClient
就没有必要了。我们只需要走这么远。我们不需要对我们依赖的框架类进行单元测试。
换句话说,我们是否需要验证
keyVaultClient.GetSecretAsync($"{endpointKeyVault}/secrets/{ keyName }")
...实际上调用端点来获取秘密?确实如此。就是KeyVaultClient.GetSecretAsync(string)
这样。如果测试失败,我们该怎么办?我们无法修复那个类。同样,如果我们沿着这条路得出其合乎逻辑的结论,我们将不得不测试各种东西。当我们创建一个List<string>
并添加一个字符串时,它真的被添加了吗?我们不测试这些东西,因为它们已经过测试,因此可以合理地假设它们按预期工作。
一个很好的测试将是一个集成测试,它验证你的类是否符合它的预期。或者,如果另一个类依赖于此,您可以为该类编写一个集成测试,如果您无法从密钥库中检索任何内容,该测试将失败。但这不需要任何嘲笑。
对不对类进行单元测试感到满意的一部分是最小化它,以便它除了调用内部类之外什么都不做。在这种情况下,您可以将您的public
andprivate
方法折叠成一种public
方法,也许还有一种方法可以提供一个async
选项:
public async Task<string> GetSecretValueAsync(string keyName)
{
return await keyVaultClient.GetSecretAsync($"{endpointKeyVault}/secrets/{ keyName }");
}
public string GetSecretValue(string keyName) => GetSecretValueAsync(keyName).Result;
现在更明显了:这个类本身并没有真正做任何需要单元测试的事情。它的有用目的是适应KeyVaultClient
你的接口——IKeyVaultConnection
这样其他类就不会直接依赖于KeyVaultClient
.
推荐阅读
- node.js - Google Pub-Sub 双向通信架构
- reactjs - 部署后反应项目不工作
- java - (初学者)期待什么“;” 是这里的意思吗?
- reactjs - 如何在 Mithril 中异步重绘特定的 DOM
- python - 猜数游戏给出错误答案的问题
- python - 发送 POST 时出现 Python 错误 - 需要类似字节的对象,而不是“str”
- python - 使用 re.findall() 和 re.escape() 匹配包含元字符的字符串?
- r - 在 edgarWebR 1.1.0 中使用地图功能时出错
- c++ - 这是我在 C++ 中的链表的代码。删除最后一个节点后,deleteAtEnd 方法出现问题,它显示一些垃圾值。我找不到错误
- amazon-web-services - 使用 k8s SDK 应用 yaml 文件