sqlOptions => sqlOptions.EnableRetryOn
Failure());
return new ApplicationDbContext(optionsBuilder.Options);
}
Как вероятно вы помните, выделенный полужирным вызов EnableRetryOnFailure()
выбирает стратегию повтора SQL Server, которая будет автоматически повторять операции, потерпевших неудачу из-за кратковременных ошибок.
Добавьте еще один статический метод, который будет создавать новый экземпляр ApplicationDbContext
с применением того же самого подключения и транзакции, что и в переданном исходном контексте. Этот метод демонстрирует способ создания экземпляра ApplicationDbContext
из существующего экземпляра с целью совместного использования подключения и транзакции:
public static ApplicationDbContext GetSecondContext(
ApplicationDbContext oldContext,
IDbContextTransaction trans)
{
var optionsBuilder = new DbContextOptionsBuilder
optionsBuilder.UseSqlServer(
oldContext.Database.GetDbConnection(),
sqlServerOptions => sqlServerOptions.EnableRetryOnFailure());
var context = new ApplicationDbContext(optionsBuilder.Options);
context.Database.UseTransaction(trans.GetDbTransaction());
return context;
}
Добавление класса BaseTest
Создайте в проекте новый каталог по имени Base
и добавьте туда новый файл класса BaseTest.cs
. Модифицируйте операторы using
следующим образом:
using System;
using System.Data;
using AutoLot.Dal.EfStructures;
using Microsoft.EntityFrameworkCore;
using Microsoft.EntityFrameworkCore.Storage;
using Microsoft.Extensions.Configuration;
Сделайте класс абстрактным и реализующим IDisposable
. Добавьте два защищенных свойства readonly
для хранения экземпляров реализации IConfiguration
икласса ApplicationDbContext
и освободите экземпляр ApplicationDbContext
в виртуальном методе Dispose()
:
namespace AutoLot.Dal.Tests.Base
{
public abstract class BaseTest : IDisposable
{
protected readonly IConfiguration Configuration;
protected readonly ApplicationDbContext Context;
public virtual void Dispose()
{
Context.Dispose();
}
}
}
Инфраструктура тестирования xUnit предоставляет механизм для запуска кода до и после прогона IDisposable
, перед прогоном каждого теста будут выполнять код в конструкторе класса (в конструкторе базового класса и конструкторе производного класса в этом случае), называемый настройкой Dispose()
(в производном и в базовом классах), называемый освобождением
Добавьте защищенный конструктор, который создает экземпляр реализации IConfiguration
и присваивает его защищенной переменной класса. С применением конфигурации создайте экземпляр ApplicationDbContext
, используя класс TestHelpers
, и присвойте его защищенной переменной класса:
protected BaseTest()
{
Configuration = TestHelpers.GetConfiguration();
Context = TestHelpers.GetContext(Configuration);
}
Добавление вспомогательных методов для выполнения тестов в транзакциях
Последние два метода в классе BaseTest
позволяют выполнять тестовые методы в транзакциях. Методы будут принимать в единственном параметре делегат Action
, создавать явную транзакцию (или вовлекать существующую транзакцию), выполнять делегат Action
и затем проводить откат транзакции. Так делается для того, чтобы любые тесты создания/обновления/удаления оставляли базу данных в состоянии, в котором она пребывала до прогона теста. Поскольку класс ApplicationDbContext
сконфигурирован с целью включения повторений при возникновении кратковременных ошибок, весь процесс обязан выполняться в соответствии со стратегией выполнения ApplicationDbContext
.