如何使用 Moq Azure Key Vault 进行单元测试

Vic*_*hez 1 c# unit-testing moq azure azure-keyvault

我需要模拟 Key Vault 的端点,以便知道我是否调用该函数来获取 Key Vault 一次。

我正在使用 C# 和 Moq(框架)开发它来进行测试。

界面如下:

public interface IKeyVaultConnection
{
    string GetKeyVaultValue(string variableName);
}
Run Code Online (Sandbox Code Playgroud)
 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;
        }
}

Run Code Online (Sandbox Code Playgroud)
 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());
Run Code Online (Sandbox Code Playgroud)

我需要的是伪造与端点的连接,以便我调用该函数一次并且收到如下错误:

"KeyVaultErrorException: Operation returned an invalid status code 'NotFound'"
Run Code Online (Sandbox Code Playgroud)

因为没有提供端点。

如果我取消注释最后一行,(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>())"
Run Code Online (Sandbox Code Playgroud)

Sco*_*nen 6

如果您将您的类视为一个不起眼的包装器,以便其他类可以在ISomethingThatReturnsValues不知情的情况下依赖它KeyVaultClient,那么模拟KeyVaultClient就没有必要了。我们只需要走到这一步。我们不需要对我们依赖的框架类进行单元测试。

换句话说,我们是否需要验证

keyVaultClient.GetSecretAsync($"{endpointKeyVault}/secrets/{ keyName }")
Run Code Online (Sandbox Code Playgroud)

...实际上调用端点来获取秘密?确实如此。就是这样KeyVaultClient.GetSecretAsync(string)。如果测试失败,我们该怎么办?我们无法修复该类。同样,如果我们沿着这条路得出合乎逻辑的结论,我们就必须测试各种各样的东西。当我们创建List<string>并添加一个字符串时,它真的被添加了吗?我们不测试这些东西,因为它们已经过测试,所以可以合理地假设它们按预期工作。

对此的一个很好的测试是集成测试,它验证您的类是否达到了预期的效果。或者,如果另一个类依赖于此,您可以为该类编写集成测试,如果您无法从密钥保管库检索任何内容,该测试将会失败。但这不需要任何嘲笑。

不对类进行单元测试的部分原因是最小化它,这样除了调用内部类之外,它实际上什么也不做。在这种情况下,您可以将您的publicprivate方法折叠成一个public方法,也许还有一个方法来提供一个async选项:

    public async Task<string> GetSecretValueAsync(string keyName)
    {
        return await keyVaultClient.GetSecretAsync($"{endpointKeyVault}/secrets/{ keyName }");
    }

    public string GetSecretValue(string keyName) => GetSecretValueAsync(keyName).Result;
Run Code Online (Sandbox Code Playgroud)

现在更明显了:这个类本身并没有真正做任何需要单元测试的事情。它的有用目的是适应KeyVaultClient您的接口 -IKeyVaultConnection以便其他类不直接依赖于KeyVaultClient.