簡體   English   中英

c#mocking IFormFile CopyToAsync()方法

[英]c# mocking IFormFile CopyToAsync() method

我正在進行異步函數的單元測試,該函數將IFormFile列表轉換為我自己的任意數據庫文件類列表。

將文件數據轉換為字節數組的方法是:

internal async Task<List<File>> ConvertFormFilesToFiles(ICollection<IFormFile> formFiles)
{
    var file = new File
    {
        InsertDateTime = DateTime.Now,
        LastChangeDateTime = DateTime.Now
    };
    if (formFile.Length > 0)
    {
        using (var memoryStream = new MemoryStream())
        {
            await formFile.CopyToAsync(memoryStream, CancellationToken.None);
            file.FileData = memoryStream.ToArray();
        }
    }
    // ...
}

該函數接收IFormFiles的ICollection,因此它是可模擬的。

現在我有這樣的測試代碼:

//Arrange
var fileConverter = new FilesConverter();
using (var fileStream = new FileStream("Files/uploadme.txt", FileMode.Open, FileAccess.Read))
{
    using (var memoryStream = new MemoryStream())
    {
        var newMemoryStream = new MemoryStream();
        fileStream.CopyTo(memoryStream);
        FormFileMock.Setup(f => f.CopyToAsync(newMemoryStream, CancellationToken.None)).Returns(Task.CompletedTask);
        // some other setups

        //Act
        var result = await fileConverter.ConvertFormFilesToFiles(new List<IFormFile> { FormFileMock.Object });
        //Assert
        Assert.IsTrue(result.Any());
    }
}

創建newMemoryStream變量是因為該函數使用新的空內存流調用CopyToAsync方法(我不確定是否有必要)。

問題是await formFile.CopyToAsync(memoryStream,CancellationToken.None)不會將任何數據復制到memoryStream。

我知道這可能不受歡迎,因為現在使用模擬框架完全“在”,但為什么不簡單地讓框架成為框架並采用簡單,簡單的方法? 您可以在不進行任何FormFile情況下創建FormFile 真正的交易:

var fileConverter = new FilesConverter(FilesConverterLoggerMock.Object, FileDataMock.Object);

// access to a real file should really not be in a "unit" test, but anyway: 
using (var stream = new MemoryStream(File.ReadAllBytes("Files/uploadme.txt"))
{
  // create a REAL form file
  var formFile = new FormFile(stream , 0, stream.Length, "name", "filename");

  //Act
  var result = await fileConverter.ConvertFormFilesToFiles(new List<IFormFile> { formFile });

  //Assert
  Assert.IsTrue(result.Any());
}

問題是await formFile.CopyToAsync(memoryStream, CancellationToken.None)不會將任何數據復制到memoryStream

根據你的設置。 什么都不會被復制。

您只需將呼叫設置為已完成返回。 沒有實現任何實際功能。

在返回任務之前,您需要添加一個Callback來執行一些所需的功能。

FormFileMock
    .Setup(_ => _.CopyToAsync(It.IsAny<Stream>(), CancellationToken.None))
    .Callback<Stream, CancellationToken>((stream, token) => {
        //memory stream in this scope is the one that was populated
        //when you called **fileStream.CopyTo(memoryStream);** in the test
        memoryStream.CopyTo(stream);
    }) 
    .Returns(Task.CompletedTask);

我結合了@Nkosi和@nvoigt的答案。 正如@nvoigt指出的那樣:訪問真實文件應該不是在“單元”測試中。 所以我用一個像這樣的字節數組替換了文件:

using (var memoryStream = new MemoryStream(new byte[]{1,2,3,4}))

而不是一個完整的文件。

我按照@Nkosi的建議在模擬對象上實現了.callback

FormFileMock.Setup(f => f.CopyToAsync(It.IsAny<Stream>(), CancellationToken.None))
    .Callback<Stream, CancellationToken>((stream, token) =>
    {
        // with memoryStream being the stream from inside the using statement
        memoryStream.CopyTo(stream);
    }).Returns(Task.CompletedTask);

現在它有效。

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM