我正在使用一个在函数中使用 out 参数的库,我需要使用该函数测试我的代码。
因此,尝试通过我在项目的其余部分中一直使用的最小起订量来让模拟来拯救我。
问题
我知道下面有一堵文字墙,所以问题(提前)是:
- 根据下面的线索:Moq 能否模拟一个带有构造函数的项目,该构造函数需要通常不会单独调用的参数?
- 这是我的测试代码的问题吗?与图书馆?使用验证库?
- 我是否在没有参数的情况下使用 Moq?
- 我从哪里开始调试它?
更新:到目前为止的线索
我认为这是模拟 IXLRow 接口(interface)的模拟方面的问题。通常情况下,XLRow 似乎只是从工作簿中实例化,而不是通过 new XLRow()
- 这是一个因素吗?
以下测试通过时(注意:mocks):
[Fact]
public void TryGetValueCanReturnTrueForVieldWithAnInteger_WhenAccessingFromRow()
{
var workbook = new XLWorkbook();
workbook.Worksheets.Add("TestWS");
var wb = workbook.Worksheet("TestWS");
wb.Cell("A1").Value = "12345";
// NOTE: Here we're referring to the row as part of an instantiated
// workbook instead of Mocking it by itself
int output;
Assert.True(wb.Row(1).Cell("A").TryGetValue(out output));
}
代码
获取有效对象模拟的方法片段():
// ...other code that sets up other parts of the row correctly
int isAnyInt = 0; //I don't care about this value, only the true/false
// set this to false to true to mimic a row being a legitimate integer
mock.Setup(m => m.Cell("B").TryGetValue(out isAnyInt)).Returns(true);
测试快乐路径的 xUnit 测试 -- 获取有效行的模拟,然后确保它通过验证。 注意:此测试通过。
[Fact]
public void Validate_GivenValidRow_ReturnsValid()
{
var mockRow = TestHelper.GetMockValidInvoiceDetailsWorksheetRow();
var validationResult = new InvoiceDetailsWorksheetRowValidator().Validate(mockRow.Object);
Assert.True(validationResult.IsValid);
}
一个 xUnit 测试(基本上,“验证器是否因单元格不是整数而失败?”) 注意:此测试通过。
[Fact]
public void Validate_GivenNonNumericClaimantID_ReturnsInvalid()
{
int outint = 0;
// Get a mock of a valid row
var mockRow = TestHelper.GetMockValidInvoiceDetailsWorksheetRow();
// change the TryGetValue result to false
mockRow.Setup(m => m.Cell("B").TryGetValue(out outint)).Returns(false);
var validationResult = new InvoiceDetailsWorksheetRowValidator().Validate(mockRow.Object);
Assert.False(validationResult.IsValid);
Assert.Equal("ClaimantID column value is not a number.", validationResult.Errors.First().ErrorMessage);
}
验证器(使用 FluentValidation):
public class InvoiceDetailsWorksheetRowValidator : AbstractValidator<IXLRow>
{
public InvoiceDetailsWorksheetRowValidator()
{
RuleFor(x => x.Cell("B"))
.Must(BeAnInt).WithMessage("ClaimantID column value is not a number.")
.OverridePropertyName("ClaimantIDColumn");
}
private bool BeAnInt(IXLCell cellToCheck)
{
int result;
var successful = cellToCheck.TryGetValue(out result);
return successful;
}
}
供引用,库中的方法:
public Boolean TryGetValue<T>(out T value)
{
var currValue = Value;
if (currValue == null)
{
value = default(T);
return true;
}
bool b;
if (TryGetTimeSpanValue(out value, currValue, out b)) return b;
if (TryGetRichStringValue(out value)) return true;
if (TryGetStringValue(out value, currValue)) return true;
var strValue = currValue.ToString();
if (typeof(T) == typeof(bool)) return TryGetBasicValue<T, bool>(out value, strValue, bool.TryParse);
if (typeof(T) == typeof(sbyte)) return TryGetBasicValue<T, sbyte>(out value, strValue, sbyte.TryParse);
if (typeof(T) == typeof(byte)) return TryGetBasicValue<T, byte>(out value, strValue, byte.TryParse);
if (typeof(T) == typeof(short)) return TryGetBasicValue<T, short>(out value, strValue, short.TryParse);
if (typeof(T) == typeof(ushort)) return TryGetBasicValue<T, ushort>(out value, strValue, ushort.TryParse);
if (typeof(T) == typeof(int)) return TryGetBasicValue<T, int>(out value, strValue, int.TryParse);
if (typeof(T) == typeof(uint)) return TryGetBasicValue<T, uint>(out value, strValue, uint.TryParse);
if (typeof(T) == typeof(long)) return TryGetBasicValue<T, long>(out value, strValue, long.TryParse);
if (typeof(T) == typeof(ulong)) return TryGetBasicValue<T, ulong>(out value, strValue, ulong.TryParse);
if (typeof(T) == typeof(float)) return TryGetBasicValue<T, float>(out value, strValue, float.TryParse);
if (typeof(T) == typeof(double)) return TryGetBasicValue<T, double>(out value, strValue, double.TryParse);
if (typeof(T) == typeof(decimal)) return TryGetBasicValue<T, decimal>(out value, strValue, decimal.TryParse);
if (typeof(T) == typeof(XLHyperlink))
{
XLHyperlink tmp = GetHyperlink();
if (tmp != null)
{
value = (T)Convert.ChangeType(tmp, typeof(T));
return true;
}
value = default(T);
return false;
}
try
{
value = (T)Convert.ChangeType(currValue, typeof(T));
return true;
}
catch
{
value = default(T);
return false;
}
}
问题
第一个测试通过。但是当我运行这个测试时,它失败了:
[Fact]
public void Validate_GivenNonNumericInvoiceNumber_ReturnsInvalid()
{
int outint = 0; // I don't care about this value
// Get a mock of a valid worksheet row
var mockRow = TestHelper.GetMockValidInvoiceDetailsWorksheetRow();
mockRow.Setup(m => m.Cell("E").TryGetValue(out outint)).Returns(false);
// Validates & asserts
var validationResult = new InvoiceDetailsWorksheetRowValidator().Validate(mockRow.Object);
Assert.False(validationResult.IsValid);
// Placed here to ensure it's the only error message. This is where it fails.
Assert.Equal("InvoiceNumber column value is not a number.",validationResult.Errors.First().ErrorMessage);
}
但它不会失败,因为尚未实现验证 - 它会失败,因为另一个项目首先无效,即使我从获得有效模拟返回它 - 通过测试的相同有效模拟否则。
确切地说,消息是:
Assert.Equal() Failure
Position: First difference is at position 0
Expected: InvoiceNumber column value is not a number.
Actual: ClaimantID column value is not a number.
我希望:
- 它以与其他测试相同的方式工作,或者
- 为了快乐的道路也会失败。
但是当快乐路径(例如有效模拟)通过时,但测试失败,因为该方法无效(通过与“有效”模拟的一部分相同的验证的相同方法)......这让我完全困惑.
供引用
- 我使用的库是 ClosedXML
- 我使用的验证库是 FluentValidation
- 我正在使用 xUnit.NET进行单元测试。
最佳答案
我认为您不需要测试 TryGetValue 在图书馆。
把 BeAnInt 在一个单独的类中说 XLCellHelpers,使用 IXLCell 的模拟测试它
为 XLCellHelpers 创建一个接口(interface),比如 IXLCellHelpers,将其注入(inject)您的验证器:InvoiceDetailsWorksheetRowValidator
模拟 IXLCellHelpers 以测试验证器。
像这样:
using System;
public class InvoiceDetailsWorksheetRowValidator : AbstractValidator<IXLRow>
{
private readonly IXlCellHelpers xlCellHelpers;
InvoiceDetailsWorksheetRowValidator(IXlCellHelpers xlCellHelpers)
{
this.xlCellHelpers = xlCellHelpers;
}
public InvoiceDetailsWorksheetRowValidator()
{
RuleFor(x => x.Cell("B"))
.Must(this.xlCellHelpers.BeAnInt).WithMessage("ClaimantID column value is not a number.")
.OverridePropertyName("ClaimantIDColumn");
}
}
public interface IXlCellHelpers
{
bool BeAnInt(IXLCell cellToCheck);
}
public class XlCellHelpers : IXlCellHelpers
{
publi bool BeAnInt(IXLCell cellToCheck)
{
int result;
var successful = cellToCheck.TryGetValue(out result);
return successful;
}
}
关于c# - 如何模拟带有 out 参数的方法?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/19140182/