我们需要确保基于 EF Core 的代码已执行测试的特定类型的数据库级操作(例如,任何命令执行或任何事务提交(。
假设应该触发一个真实的数据库,我们不能通过DbContext
嘲笑来隔离它。它看起来如何:
[Fact]
public async Task Test()
{
using (var context = new FooContext())
{
//context-related code to produce queries and commands
}
Assert.True(/* any context-related transaction has been committed */);
}
可能吗?
EF Core 不提供自己的跟踪机制。但是,它会记录大量数据库交互事件。我们可以收集这些日志消息并检查其EventId
以确定是否发生了特定操作。下面是 EF Core 使用的关系事件列表:
EF Core 1.1.2:RelationalEventId enum。
EF Core 2.0.0 预览版 1:关系事件 ID 类(重大更改!
我们需要做的就是创建一个假记录器并将其传递给上下文:
[Fact]
public async Task TransactionCommit_Logger_ContainsEvent()
{
var logger = new FakeLogger();
var factoryMock = new Mock<ILoggerFactory>();
factoryMock.Setup(f => f.CreateLogger(It.IsAny<string>()))
.Returns(logger);
using (var context = new FooContext(factoryMock.Object))
{
using (var transaction = await context.Database.BeginTransactionAsync())
{
transaction.Commit();
}
}
Assert.True(logger.Events.Contains((int)RelationalEventId.CommittingTransaction));
}
FakeLogger
将记录的事件 ID 添加到Events
列表中。
public class FakeLogger : ILogger
{
public void Log<TState>(LogLevel logLevel, EventId eventId, TState state, Exception exception,
Func<TState, Exception, string> formatter)
{
Events.Add(eventId.Id);
}
public List<int> Events { get; set; } = new List<int>();
public bool IsEnabled(LogLevel logLevel) => true;
public IDisposable BeginScope<TState>(TState state) => null;
}
调用 UseLoggerFactory
将工厂实例附加到上下文:
public class FooContext : FooParentContext
{
private readonly ILoggerFactory _loggerFactory;
public FooContext() { }
public FooContext(ILoggerFactory loggerFactory)
{
_loggerFactory = loggerFactory;
}
protected override void OnConfiguring(DbContextOptionsBuilder optionsBuilder)
{
base.OnConfiguring(optionsBuilder);
optionsBuilder.UseLoggerFactory(_loggerFactory);
}
}
附言您可以更深入地分析日志消息,甚至是 EF 生成的原始 SQL。