首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >用AutoFac和AutoMock模拟AutoFac

用AutoFac和AutoMock模拟AutoFac
EN

Stack Overflow用户
提问于 2017-08-25 07:03:49
回答 3查看 6.8K关注 0票数 4

我正在尝试为我的AzureBlobRepository编写单元测试。存储库在构造函数中接收CloubBlobClient。我想嘲笑客户机,但这给出了一个例外:

代码语言:javascript
复制
using (var mock = AutoMock.GetLoose())
{
    var mockClient = mock.Mock<CloudBlobClient>();
}

无法在'Microsoft.WindowsAzure.Storage.Blob.CloudBlobClient'.类型上长度相同的多个构造函数之间进行选择在注册组件时,使用UsingConstructor()配置方法显式地选择构造函数。

当然,在我的单元测试中,我没有注册任何东西,所以这条消息没有多大帮助。

我还尝试了其他方法,例如提供NameParameters、TypedParameters或调用mock.Create而不是mock.Mock,但我尝试的每一种方法都返回相同的异常消息。

(同样的问题也发生在CloudBlobContainer上)

实现接口后的更新是我编写的一个单元测试的示例:

代码语言:javascript
复制
[TestMethod]
public void AzureBlobRepository_GetByIdAsync_ReturnsContent()
{    
    Guid blobId = Guid.NewGuid();
    Guid directoryName = Guid.NewGuid();
    string containerName = "unittest";

    using (var mock = AutoMock.GetLoose())
    {
        var mockClient = mock.Mock<ICloudBlobClient>();
        var mockContainer = mock.Mock<ICloudBlobContainer>();
        var mockDirectory = mock.Mock<ICloudBlobDirectory>();
        // notice that we're not using AutoMock here, it fails to create the mock
        var mockBlob = new Mock<CloudBlockBlob>(new Uri($"http://tempuri.org/{containerName}/{directoryName}/{blobId}"));
        mockBlob.Setup(m => m.DownloadTextAsync()).Returns(Task.FromResult("content"));

        mockClient.Setup(m => m.GetContainerReference(containerName))
            .Returns(mockContainer.Object);
        mockContainer.Setup(m => m.GetDirectoryReference(directoryName.ToString()))
            .Returns(mockDirectory.Object);
        mockDirectory.Setup(m => m.GetBlockBlobReference(blobId.ToString()))
            .Returns(mockBlob.Object);

        var repository = mock.Create<AzureBlobRepository>(
            new TypedParameter(typeof(ICloudBlobClient), mockClient.Object),
            new NamedParameter("container", containerName),
            new NamedParameter("directory", directoryName));

        var result = repository.GetByIdAsync(blobId, directoryName).Result;
        result.ShouldBe("content");
    }
}
EN

回答 3

Stack Overflow用户

回答已采纳

发布于 2017-08-25 12:07:16

这些类应被视为第三方的实现关注事项。这意味着你无法控制它们,我们不应该嘲笑我们无法控制的东西。它们应该封装在抽象中,您可以控制这些抽象,并在隔离测试时根据需要进行模拟。

代码语言:javascript
复制
public interface ICloudBlobClient {
    //...expose only the functionality I need
}

public class CloudBlobClientWrapper : ICloudBlobClient {
    private readonly CloudBlobClient client;

    public CloudBlobClientWrapper(CloudBlobClient client) {
        this.client = client;
    }

    //...implement interface wrapping
}

由于这个原因,类应该依赖抽象而不是具体化。模拟混凝土类往往会产生敲门效应。

包装器不需要完全包装客户端,但可以聚合功能,以避免公开实现方面的问题。

因此,现在在隔离测试时,您可以模拟您控制的抽象。

代码语言:javascript
复制
using (var mock = AutoMock.GetLoose()) {
    var mockClient = mock.Mock<ICloudBlobClient>();

    /// ...and the rest of the test.
}
票数 5
EN

Stack Overflow用户

发布于 2017-08-25 08:27:20

看这个CloudBlobContainer。此类型包含三个构造函数。创建类型的实例需要构造函数。尝试键入您的代码new CloudBlobContainer,您将需要选择三个构造函数之一。AutoMock无法知道必须选择什么构造函数。

您可以向AutoMock说明如何创建CloudBlobContainer

示例:

代码语言:javascript
复制
using (var mock = AutoMock.GetLoose())
{
    mock.Provide<CloudBlobContainer, CloudBlobContainer>(new NamedParameter("uri", new Uri("your uri")));
    var mockClient = mock.Mock<CloudBlobClient>();
}
票数 0
EN

Stack Overflow用户

发布于 2019-04-17 06:20:23

我成功地用NSubstitute来模拟它,我只模拟了我使用的函数。

代码语言:javascript
复制
/// <summary>
/// Create a mock for CloudBlobClient
/// </summary>
/// <param name="containerExists"></param>
/// <returns></returns>
private CloudBlobClient GetMock(bool containerExists = true)
{
    var items = new List<IListBlobItem>();
    var client = Substitute.For<CloudBlobClient>(new Uri("http://foo.bar/"), null);
    var container = Substitute.For<CloudBlobContainer>(new Uri("http://foo.bar/"));
    client.GetContainerReference(Arg.Any<string>()).Returns(container);
    container.ExistsAsync(Arg.Any<CancellationToken>()).Returns(Task.FromResult(containerExists));
    container.ListBlobsSegmentedAsync(Arg.Any<string>(), Arg.Any<bool>(), 
                                        Arg.Any<BlobListingDetails>(), 
                                        Arg.Any<int?>(), 
                                        Arg.Any<BlobContinuationToken>(), 
                                        Arg.Any<BlobRequestOptions>(), 
                                        Arg.Any<OperationContext>(), 
                                        Arg.Any<CancellationToken>())
                                        .Returns(ci => new BlobResultSegment(items.ToArray(), null));

    container.GetBlockBlobReference(Arg.Any<string>()).Returns(ci => GetBlockBlobMock(ci.ArgAt<string>(0), items));
    return client;
}

/// <summary>
/// Create a mock for CloudBlockBlob
/// </summary>
/// <param name="name"></param>
/// <param name="listBlobItems"></param>
/// <returns></returns>
private CloudBlockBlob GetBlockBlobMock(string name, List<IListBlobItem> listBlobItems)
{
    var created = DateTimeOffset.Now;
    var bufferStream = new MemoryStream();
    var blob = Substitute.For<CloudBlockBlob>(new Uri("https://foo.blob.core.windows.net/bar/" + name + ".txt"));
    //We can't mock the value the normal way, use reflection to change its value!
    blob.Properties.GetType().GetProperty(nameof(blob.Properties.Created)).SetValue(blob.Properties, created);
    //we cant mock properties! (Dam this wont work)
    blob.UploadFromStreamAsync(Arg.Any<Stream>(),
                                Arg.Any<AccessCondition>(),
                                Arg.Any<BlobRequestOptions>(),
                                Arg.Any<OperationContext>(),
                                Arg.Any<CancellationToken>()).Returns(ci => {
                                    var stream = ci.Arg<Stream>();
                                    stream.CopyTo(bufferStream);
                                    listBlobItems.Add(blob);
                                    return Task.CompletedTask;
                                });

    blob.DownloadToStreamAsync(Arg.Any<Stream>(),
                                Arg.Any<AccessCondition>(),
                                Arg.Any<BlobRequestOptions>(),
                                Arg.Any<OperationContext>(),
                                Arg.Any<CancellationToken>()).Returns(ci =>
                                {
                                    var stream = ci.Arg<Stream>();
                                    bufferStream.Position = 0;
                                    bufferStream.CopyTo(stream);
                                    stream.Position = 0;
                                    return Task.CompletedTask;
                                });
    return blob;
}

我不是100%确定它100%的准确性,但它允许我运行一些单元测试!

票数 0
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/45876042

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档