我想测试ConfirmDownloadInvoiceDate
方法。另外,我想用Order
属性的测试数据创建ConfirmationDownloadInvoiceDate
对象:
fixture.Create<Order>();
我的Order
课程:
public class Order
{
public DateTime? ConfirmationDownloadInvoiceDate { get; private set; }
public void ConfirmDownloadInvoiceDate(IDateTimeProvider timeProvider)
{
if (ConfirmationDownloadInvoiceDate == null)
{
ConfirmationDownloadInvoiceDate = timeProvider.Now();
}
}
}
是否可以用测试数据填充该属性?我尝试通过从ISpecimenBuilder
创建新类,但似乎它不起作用。
根据设计,AutoFixture仅在可公开写入时填写字段和属性,因为这是您自己作为客户端开发人员可以做的事情,如果您不是使用AutoFixture,而是手动编写测试数据安排阶段。在上面的Order
类中,ConfirmationDownloadInvoiceDate
属性没有公共setter,因此AutoFixture将忽略它。
显而易见,最简单的解决方法是将setter公之于众,但这并不总是有道理的。
在这种特殊情况下,您可以通过告诉AutoFixture在创建Order
对象时应该调用ConfirmDownloadInvoiceDate
方法来自定义Order
类的创建。
一种方法是首先创建IDateTimeProvider
的特定于测试的Stub实现,例如:
public class StubDateTimeProvider : IDateTimeProvider
{
public StubDateTimeProvider(DateTime value)
{
this.Value = value;
}
public DateTime Value { get; }
public DateTime Now()
{
return this.Value;
}
}
您还可以使用动态模拟库,如Moq,NSubstitute等。
使用存根调用ConfirmDownloadInvoiceDate
方法,例如:
[Fact]
public void AutoFillConfirmationDownloadInvoiceDate()
{
var fixture = new Fixture();
fixture.Customize<Order>(c => c
.Do(o => o.ConfirmDownloadInvoiceDate(fixture.Create<StubDateTimeProvider>())));
var actual = fixture.Create<Order>();
Assert.NotNull(actual.ConfirmationDownloadInvoiceDate);
Assert.NotEqual(default(DateTime), actual.ConfirmationDownloadInvoiceDate);
}
这个测试通过。您应该考虑在ICustomization
类中打包上述自定义。