C постепенным переходом проектов на .NET Core фреймворки все большую популярность набирает стандартная реализация внедрения зависимостей от Microsoft. На мой взгляд эта реализация все еще уступает Autofac, но в этой статье речь пойдет не об этом. При использовании любого фреймворка для внедрения зависимостей рано или поздно разработчики сталкиваются с проблемой забытой или неправильной регистрации зависимостей, что влечет за собой ошибку в рантайме приложения.

Хорошо, если разработчик, привнеся ошибку в код, самостоятельно ее обнаружил во время отладки, а если нет? Если ошибка ускользнула от разработчика, QA-инженера, прошла через все стадии доставки изменений кода до пользователей и оказалась в production? В этом случае есть вопросы к контроля качества в команде, однако, нельзя отрицать тот факт, что работа с зависимостями в .NET приложении несет в себе большой регрессионный риск.
В основе юнит-тестирования внедрения зависимостей лежит простая идея - отлавливать проблемы с DI в приложении еще на этапе автоматического запуска юнит-тестов, настроенного в вашем ci/cd пайплайне. Дальнейшее повествование будет вестись в контексте ASP.NET Core приложения.
Пишем юнит тесты
Перед написанием юнит тестов определю некоторые требования к ним. Тесты должны:
Поддерживать автоматический assembly-сканнинг всех точек входа в приложение, в случае ASP.NET Core это будут MVC контроллеры.
Иметь базовый класс или набор extension-методов для переиспользования в множестве проектов.
Поддерживать проверку внедрения зависимостей для BackgroundServices, слушателей ServiceBus или других кастомных классов в Console Applications.
Последнее требование специфично конкретно для моего проекта, но уверен, что практически в любом приложении с микросервисной архитектурой можно найти работу с брокерами сообщений, фоновыми сервисами, джобами и триггерами. Тесты можно адаптировать под любой вид зависимостей.
Целевым фреймворком для указанного ниже кода является .NET 5, однако, код легко адаптировать и для .NET 6+. Начнем с основы и напишем базовый класс для тестирования зависимостей API контроллеров:
public abstract class ApiDependencyInjectionTestBase<TStartup> where TStartup : class { protected IEnumerable<object> ResolveEntryPoints() { var host = WebHost.CreateDefaultBuilder() .UseStartup<TStartup>() .Build(); var serviceProvider = host.Services; using var serviceScope = serviceProvider.CreateScope(); var baseClassType = typeof(ControllerBase); var typesToResolve = typeof(TStartup).Assembly .GetTypes() .Where(x => x.IsClass && !x.IsInterface && !x.IsAbstract && baseClassType.IsAssignableFrom(x)) .ToList(); foreach (var typeToResolve in typesToResolve) { yield return ActivatorUtilities.CreateInstance(serviceProvider, typeToResolve); } } }
Базовый generic-класс содержит один единственный метод ResolveEntryPoints , который внутри себя создает WebHost на основе заданного Startup класса, с помощью рефлексии ищет всех наследников базового класса ControllerBase в сборке и пытается их внедрить вместе со всеми зависимостями. В случае, если какая-нибудь из зависимостей не будет должным образом зарегистрирована, при попытке вызова ActivatorUtilities.CreateInstance() будет выброшено соответствующее исключение. В реализации юнит теста нам остается только использовать Startup класс в качестве generic-типа и проверить, что список внедренных контроллеров не пустой и действительно содержит объекты:
public class ApiDependencyInjectionTests : ApiDependencyInjectionTestBase<Startup> { [Fact] public void DependenciesRegistrationTest() { var instances = ResolveEntryPoints(); Assert.NotEmpty(instances); foreach (var instance in instances) { Assert.NotNull(instance); } } }
В своем проекте я поставляю базовый ApiDependencyInjectionTestBase класс в NuGet пакете и пишу простой тест на внедрение зависимостей в каждом MVC-приложении.
Однако, кроме ASP.NET приложений конкретно в моем проекте используются еще и Console Applications, которые хостят разного рода фоновые задачи или обработчики сообщений от брокеров. Для этих целей напишем чуть более сложный базовый класс WebJobsDependencyInjectionTestBase:
public abstract class WebJobsDependencyInjectionTestBase { private IServiceProvider serviceProvider { get; set; } protected void SetUpServiceProvider( string appSettingsPath, Action<IConfiguration> setConfiguration, Action<IServiceCollection> configureServices) { var configuration = new ConfigurationBuilder() .AddJsonFile(appSettingsPath) .Build(); setConfiguration(configuration); var host = new HostBuilder() .ConfigureHostConfiguration(configBuilder => { configBuilder.AddConfiguration(configuration); }) .ConfigureServices(configureServices) .Build(); serviceProvider = host.Services; } protected IEnumerable<object> ResolveEntryPoints(Assembly assembly, params Type[] additionalTypesToResolve) { using var serviceScope = serviceProvider.CreateScope(); var messageHandlerInterface = typeof(IMessageHandler); var backgroundServiceBaseClass = typeof(BackgroundService); var typesToResolve = assembly .GetTypes() .Where(x => x.IsClass && !x.IsInterface && !x.IsAbstract && (messageHandlerInterface.IsAssignableFrom(x) || backgroundServiceBaseClass.IsAssignableFrom(x))) .ToList(); typesToResolve.AddRange(additionalTypesToResolve); foreach (var typeToResolve in typesToResolve) { yield return ActivatorUtilities.CreateInstance(serviceProvider, typeToResolve); } } }
Поскольку конфигурирование ServiceProvider в случае отсутствия Startup класса является чуть более сложной задачей и требует дополнительных параметров, выделим отдельный метод SetUpServiceProvider. В этот раз нам пришлось отдельно создавать конфигурацию приложения с помощью пути к файлу настроек appsettings.json и использовать Action параметры для конфигурирования ServiceProvider. Метод ResolveEntryPoints также усложнился и использует для Assembly-сканнинга интерфейс IMessageHandler и базовый класс BackgroundService . Кроме этого добавлена возможность принимать кастомный список типов, которые могут использоваться в конкретном приложении как точки входа.
Вот так может выглядеть реализация теста:
public class WebJobsDependencyInjectionTests : WebJobsDependencyInjectionTestBase { [Fact] public void DependenciesRegistrationTest() { SetUpServiceProvider(appSettingsPath: "appsettings.json", configuration => Program.Configuration = configuration, services => Program.ConfigureServices(services)); var instances = ResolveEntryPoints(typeof(MesageHandlerWithDependency).Assembly, typeof(CustomClassWithDependency)); Assert.NotEmpty(instances); foreach (var instance in instances) { Assert.NotNull(instance); } } }
При этом отмечу, что в Program класс необходимо сделать публичным, добавить метод public static ServiceProvider ConfigureServices(IServiceCollection services) и свойство public static IConfiguration Configuration { get; set; }.В метод ResolveEntryPoints необходимо передать Assembly для сканирования, а также при необходимости список кастомных классов для резолва.
Заключение
Юнит-тесты на внедрение зависимостей могут покрыть большую часть возможных проблем с зависимостями. Приведенная реализация тестов является рабочим решением, но стоит помнить, что в реальных проектах часто необходимо проверять дополнительные зависимости, например, если вы используете FromServices атрибуты, разного рода Middlewares, профили Automapper, обработчики MediatR и т.д.
C демонстрационным проектом и тестами можно ознакомиться на GitHub
