На просторах интернета появились библиотеки, позволяющие упростить и ускорить построение бизнес‑логики разрабатываемого приложения. Одна из таких библиотек — MediatR. В данной статье я хочу описать небольшой пример из реального проекта. Проект, web приложение, предназначен для автоматизации некоторого бизнес‑процесса. В рамках данного проекта была реализована задача по согласованию, где использовался инструментарий библиотеки MediatR. Я не буду уделять особого внимания моментам, связанным с установкой и настройкой данной библиотеки в проекте, выделю только то, что необходимо для решения нашей задачи.
Описание библиотеки MediatR
Несколько строк хочу посвятить описанию самой библиотеки. MediatR — это библиотека с открытым кодом, которая помогает реализовать поведенческий шаблон проектирования «Посредник» (Mediator). Шаблон «Посредник» способствует организации взаимодействия множества объектов без создания ссылок друг на друга, то есть отпадает необходимость в передаче объектам информации друг о друге. Это порождает слабую связанность, которая оказывает положительный эффект при изменении или доработках объектов. MediatR получил широкое применение при проектировании и разработке микросервисов. Ссылка на исходный код проекта библиотек располагается здесь. Бинарный код MediatR можно найти на сайте Nuget. В состав библиотеки входит обширный инструментарий для реализации логики разрабатываемых приложений, я остановлюсь на обработчиках и опишу способ их применения.
Разработка задачи
Вернемся к задаче. Внесу некоторое уточнение по технологиям, используемым в проекте. Проект использует ASP.NET MVC и Unity — библиотеку внедрения зависимостей.
Еще раз напомню, что в рамках данного проекта реализуется подзадача по согласованию неких событий, не будем углубляться в подробности, каких. В согласовании принимают участие несколько участников. После успешного согласования все ответственные должны получить оповещение о том, что согласование завершено, например, в виде электронного письма. Для большей наглядности изобразим алгоритм задачи в виде схемы, которая изображена на рисунке 1. Здесь возникает вопрос, как используемая библиотека поможет нам в реализации данного процесса? При изучении содержимого библиотеки MediatR, в ее составе можно обнаружить интерфейсы, которые используются при создании пользовательских классов. Рассмотрим только те, которые мы будем использовать при написании функционала для нашей задачи:
IRequestHandler<IRequest,TResponse> интерфейс для реализации класса обработчика.
IRequest интерфейс, который применяется при описании объекта запроса, используемого классом обработчика.
IRequestPreProcessor — интерфейс для описания класса, который будет выполнятся перед вызовом класса обработчика.
IRequestPostProcessor<TRequest, TResult> интерфейс для описания класса, который будет выполнятся после вызова класса обработчика.
Тут прослеживается некоторая логика поведения. При запросе основного обработчика (IRequestHandler) логика библиотеки MediatR запускает еще несколько вспомогательных классов обработки, которые наследуются от интерфейсов IRequestPreProcessor и IRequestPostProcessor. Первый вызывается перед запуском, а второй после завершения выполнения кода основного обработчика. Это правило поведения будет очень полезно при реализации задачи.
После уточнения требований приступим к воплощению функционала. Напишем несколько классов для обработки, которые будут описывать основной функционал по согласованию. Перечислим их:
StartHandler – запускает согласование.
ApproveHadler – обрабатывает подтверждение со стороны участника согласования.
RejectHandler – обрабатывает отклонение со стороны участника согласования.
Как я уже упоминал, все классы будут наследоваться от интерфейса:
IRequestHandler<IRequest,TResponse>
Реализуем их:
..
public class StartHandler : IRequestHandler<StartRequest, string>
{
public Task<string> Handle(
StartRequest request,
CancellationToken cancellationToken)
{
..
}
}
..
..
public class ApproveHandler : IRequestHandler<ApproveRequest, string>
{
public Task<string> Handle(
ApproveRequest request,
CancellationToken cancellationToken)
{
..
}
}
..
..
public class RejectHandler : IRequestHandler<RejectRequest, string>
{
public Task<string> Handle(
RejectLRequest request,
CancellationToken cancellationToken)
{
..
}
}
..
Объекты запроса StartRequest, ApproveRequest и RejectRequest опишем ниже.
..
public class StartRequest : BaseRequest
{
}
..
public class ApproveRequest : BaseRequest
{
public string Comments { get; set; }
public IConfig Iconfig { get; set; }
}
..
public class RejectRequest : BaseRequest
{
public string Comments { get; set; }
}
..
public class BaseRequest : IRequest<string>
{
public IUnitOfWork IUnitOfWork { get; set; }
public IUserContext IUserContext { get; set; }
public int? EventID { get; set; }
}
..
Дополнительно нам потребуются доработка вспомогательных обработчиков, которые будут вызываться после выполнения основных обработчиков, описанных выше.
Приведу пример таких пре‑обработчиков и пост‑обработчиков.
..
public class GenericRequestPreProcessorBehavior<TRequest> :
IRequestPreProcessor<TRequest>
{
public Task Process(TRequest request, CancellationToken token)
{
//код предобработки, вызывается перед запуском основного обработчика
..
}
}
..
public class GenericRequestPostProcessorBehavior<TRequest, TResult> :
IRequestPostProcessor<TRequest, TResult>
{
public Task Process(TRequest request, TResult result)
{
//постобработка после того, как согласование было запущено
if (request is StartRequest)
{
//код для обработки после запуска согласования
..
}
..
//постобработка после отклонения мероприятия
if (request is RejectRequest) {
//код для обработки после отклонения согласования
//одним из участников
..
}
..
//постобработка после завершения согласования
if (request is ApproveRequest) {
//код для обработки после подтверждения одним
//из участников согласования
..
}
..
return Task.FromResult<TResult>(result);
}
}
Библиотека MediatR очень хорошо работает с различными типами IoC‑контейнеров. Здесь можно найти примеры настройки MediatR и различных библиотек, реализующих внедрение зависимостей. Как я упоминал выше, в нашем проекте используется библиотека Unity. Основной момент, на который нужно обратить внимание — это регистрация в контейнере внедрения зависимостей для наших основных обработчиков и вспомогательных обработчиков. Если этого не сделать, то функционал работать не будет. Приведу часть кода, где описана регистрация в IoC контейнере классов обработчиков.
..
public static class UnityConfig
{
private static UnityContainer container;
public static void RegisterComponents()
{
container = new UnityContainer();
…
StringBuilder sb = new StringBuilder();
container.RegisterInstance<TextWriter>(new StringWriter(sb))
.RegisterMediator(new HierarchicalLifetimeManager())
.RegisterMediatorHandlers(
Assembly.GetAssembly(typeof(StartRequest)));
container.RegisterInstance<TextWriter>(new StringWriter(sb))
.RegisterMediator(new HierarchicalLifetimeManager())
.RegisterMediatorHandlers(
Assembly.GetAssembly(typeof(RejectRequest)));
container.RegisterInstance<TextWriter>(new StringWriter(sb))
.RegisterMediator(new HierarchicalLifetimeManager())
.RegisterMediatorHandlers(
Assembly.GetAssembly(typeof(ApproveRequest)));
...
container.RegisterType(
typeof(IRequestPreProcessor<>),
typeof(GenericRequestPreProcessorBehavior<>),
"GenericRequestPreProcessorBehavior");
container.RegisterType(
typeof(IRequestPostProcessor<,>),
typeof(GenericRequestPostProcessorBehavior<,>),
"GenericRequestPostProcessorBehavior");
...
DependencyResolver.SetResolver(
new UnityDependencyResolver(container));
}
}
Нужно отметить, что используемые методы:
public static IUnityContainer RegisterMediator(
this IUnityContainer container,
LifetimeManager lifetimeManager)
..
и
public static IUnityContainer RegisterMediatorHandlers(
this IUnityContainer container,
Assembly assembly)
..
являются расширениями для контейнера внедрения зависимостей Unity и не являются частью библиотеки. Пример их реализации можно найти на просторах интернета, причем вариант реализации может зависеть от версии библиотеки MediatR и версии Framework .NET, поэтому я не буду приводить примеры реализации этих расширений в данной статье.
После объявления и описания основных функций перейдем к окончательной реализации методов контроллера, отвечающего за работу с согласованием со стороны сервера. Вот как будет выглядеть этот функционал:
..
/// <summary>
/// Контроллер для запуска, подтверждения и отмены согласования
/// </summary>
public class AgreementController : BaseController
{
public AgreementController(IUnitOfWork IUnitOfWork)
: base(IUnitOfWork)
{
}
/// <summary>
/// Запуск процесса согласования для события
/// </summary>
/// <param name=”id">идентификатор события</param>
/// <returns>Task<ActionResult></returns>
public Task<ActionResult> Start(int id)
{
string error = string.Empty;
var eventId = id;
var userContext = CurrentUserContext;
return Task.Run<ActionResult>(async() => {
try
{
var response = await mediator.Send(
new StartRequest() {
IUnitOfWork = this.IUnitOfWork,
IUserContext = userContext,
EventID = eventId });
if (!response.StartsWith("OK")) {
return Json(new JsonMessageView() {
Status = "ERROR",
Message = response},
JsonRequestBehavior.AllowGet);
}
return Json(new JsonMessageView() {
Status= "OK",
Message= $"{eventId}",
JsonRequestBehavior.AllowGet);
}
catch (Exception ex)
{
error = ex.Message;
}
return Json(new JsonMessageView(){
Status = "ERROR",
Message = $"ОШИБКА. {error}"},
JsonRequestBehavior.AllowGet);
});
}
/// <summary>
/// Согласовать
/// </summary>
/// <param name=”id">идентификатор события</param>
/// <param name="comments">комментарий</param>
/// <returns>Task<ActionResult></returns>
public async Task<ActionResult> Approve(int id, string comments)
{
string error = string.Empty;
int eventId = id;
var userContext = CurrentUserContext;
return await Task.Run<ActionResult>(async () => {
try
{
var response = await mediator.Send(new ApproveRequest()
{
IUnitOfWork = this.IUnitOfWork,
IUserContext = userContext,
EventID = eventId,
Comments = comments ?? string.Empty,
IConfig = new AppConfiguration()
});
if (!response.StartsWith("OK"))
{
return Json(new JsonMessageView() {
Status = "ERROR",
Message = response},
JsonRequestBehavior.AllowGet);
}
return Json(new JsonMessageView() {
Status = "OK",
Message = $"{eventId}"},
JsonRequestBehavior.AllowGet);
}
catch (Exception ex)
{
error = ex.Message;
}
return Json(new JsonMessageView() {
Status = "ERROR",
Message = $"ОШИБКА. {error}"},
JsonRequestBehavior.AllowGet);
});
}
/// <summary>
/// Отклонить
/// </summary>
/// <param name="id">идентификатор события</param>
/// <param name="comments">комментарий</param>
/// <returns>Task<ActionResult></returns>
public async Task<ActionResult> Reject(int id,string сomments)
{
string error = string.Empty;
int eventId = id;
var userContext = CurrentUserContext;
return await Task.Run<ActionResult>(async() => {
try
{
var response = await mediator.Send(new RejectRequest()
{
IUnitOfWork = this.IUnitOfWork,
IUserContext = userContext,
EventID = eventId,
Comments = comments ?? string.Empty
});
if (!response.StartsWith("OK"))
{
return Json(new JsonMessageView() {
Status = "ERROR",
Message = response},
JsonRequestBehavior.AllowGet);
}
return Json(new JsonMessageView() {
Status = "OK",
Message = $"{eventId}" },
JsonRequestBehavior.AllowGet);
}
catch (Exception ex)
{
error = ex.Message;
}
return Json(new JsonMessageView() {
Status = "ERROR",
Message = $"ОШИБКА. {error}"},
JsonRequestBehavior.AllowGet);
});
}
..
}
Реализация методов контроллера, отвечающего за обработку согласования, завершена. Теперь несколько слов по описанию, как это работает. После вызова метода контроллера, код, указанный ниже,
var response = await mediator.Send(new ApproveRequest()
{
IUnitOfWork = this.IUnitOfWork,
IUserContext = userContext,
EventID = eventId,
Comments = comments ?? string.Empty,
IConfig = new AppConfiguration()
});
запускает обработчики, которые выполняются в последовательности, указанной на рисунке 2. Сначала будет вызван метод Process(TRequest request, CancellationToken token) класса GenericRequestPreProcessorBehavior. Затем запустится метод Handle(ApproveRequest request, CancellationToken cancellationToken) основного обработчика ApproveHandler, где находится основная логика обработки согласования. После выполнения этого метода выполнится метод Process(TRequest request, TResult result) класса GenericRequestPostProcessorBehavior. В данном методе выполняется проверка, что все участники вынесли свое решение. Если условие выполнено, то нужно запустить процедуру оповещения всех ответственных сотрудников о том, что согласование завершено.
Положительные и отрицательные стороны использования библиотеки
Рассмотрим, какие бонусы мы получим. К положительным моментам использования библиотеки можно отнести:
Уменьшения связанности между объектами.
При сложной логике взаимодействия между множеством объектов позволяет настроить передачу сообщений между объектами.
Позволяет с меньшими затратами и, не ломая текущую архитектуру проекта, создать расширение и доработку функционала.
К недостаткам использования данной библиотеки можно отнести:
Сложность настраивания и подключения библиотек к проекту.
Сложность реализации функционала. Описание дополнительных вспомогательных классов, которые нужны при написании обработчиков.
Если контекст базы данных передается в обработчик, то это нужно учитывать при создании логики, чтобы не возникло ошибок, связанных с одновременным доступом к контексту из разных потоков.
Заключение
В статье я описал одну из моделей поведения, реализованную с помощью обработчиков. Другая модель поведения, входящая в состав библиотеки MediatR, которая не была рассмотрена в статье — это событийная модель. Таким образом, бизнес-логика может быть построена на обработчиках, на событиях или на их комбинации, что предоставляет большую гибкость для разработчика при построении архитектуры приложений. Это дает возможность описывать сложную логику поведения при реализации сложных бизнес-процессов. Хотя применение вспомогательной библиотеки требует дополнительных усилий, связанных с изучением ее инфраструктуры, применение MediatR также помогает организовать слабые связи между разрабатываемыми модулями, что является для современных проектов основным требованием. Это упрощает доработку и развитие программного продукта. Поэтому любой разработчик, который будет поддерживать или развивать приложение, бизнес-логика которого описана при помощи библиотеки MediatR, в большинстве случаев сможет вносить изменения в программу, не опасаясь, что сломает логику ее работы.
Дополнительные ссылки:
Адрес проекта библиотеки https://github.com/jbogard/MediatR
Адрес библиотеки сайта nuget https://www.nuget.org/packages/MediatR/
Примеры использования различных библиотек по внедрению зависимостей — https://github.com/jbogard/MediatR/wiki
Комментарии (12)
dopusteam
20.04.2023 05:45+1string error = string.Empty; try { var response = ///; if (!response.StartsWith("OK")) { // return ERROR(error) } // return OK } catch (Exception ex) { error = ex.Message; } // return ERROR(error)
Зачем вы пишете такой код? Почему бы из catch не вернуть результат и не создавать string error непонятно зачем?
response.StartsWith серьёзно? Почему бы типами не разрулить?
rtv128 Автор
20.04.2023 05:45-1А какие преимущества мы получим, если будем использовать типы? И почему данная реализация недопустима?
Oceanshiver
20.04.2023 05:45+1Для меня вообще большая загадка как MediatR стал хоть сколько-то используемым в .net сообществе. Его бы в антипаттерны запихнуть.
shai_hulud
20.04.2023 05:45+1По тому что все программисты проходят через стадию "абстракция на абстракцию как в книжке про паттерны", эта стадия является гиперкомпенсацией предыдущей, где все фигачили в один статичный класс с глобальными переменными.
zerg903
20.04.2023 05:45-1В заключение статьи хорошо отмечено, что «MediatR не является реализацией шаблона Посредник, а является внутренней шиной». Как мне кажется, это основной плюс данной библиотеки. Это позволяет отделить в тестах специфику web от бизнес-логики.
Так, если замокать ISender и создать клиента с помощью WebApplicationFactory, то мы можем легко протестировать, что http-запрос корректно преобразовался в Request, который отправился во внутреннюю шину и полученный из нее Response преобразован в ожидаемый http-ответ.Бизнес-логику, реализованную в хэндлерах (как слой юзкейсов над сервисами), можно тестировать классическими unit-тестами.
Но, автор MediatR попытался превратить библиотеку в «комбайн», использование которого и вызывают такие холивары. Я предпочитаю свою light-реализацию данного функционала.
Реализация на интерфейсах – хорошая практика, если у вас небольшое приложение. Но, если у вашего Web API 100+ методов, то подход MediatR-а позволяет получить слабосвязанный код без нагромождения десятков интерфейсов.
zerg903
20.04.2023 05:45Промахнулся, это был ответ на сообщение @Oceanshiver ".. вообще большая загадка как MediatR стал хоть сколько-то используемым .."
dopusteam
20.04.2023 05:45Так, если замокать ISender и создать клиента с помощью WebApplicationFactory, то мы можем легко протестировать, что http-запрос корректно преобразовался в Request, который отправился во внутреннюю шину и полученный из нее Response преобразован в ожидаемый http-ответ.
А в чем сложность сделать то же самое с интерфейсами и сервисами обычными?
dopusteam
Между какими конкретно объектами? И чем это лучше, чем просто вызывать сервис через интерфейс?
Что это вообще значит? Ну и если у меня сложная логика взаимодействия между объектами, я б предпочёл упростить её, а не делать всё через
одно местомедиатр, т.к. это даже навигацию по коду усложнит и даст возможность создать циклические зависимости, что невозможно при работе через интерфейсы и сервисы.Как именно? Какая то общая фраза
Всё ещё не раскрыта тема, зачем оно вообще.
Это в любом случае нужно, что с медиатром, что без.
Это в любом случае стоит учитывать, что с медиатром, что без
А зачем вы в запрос к медиатру пихаете зависимости? Оо
Это дикая связанность какая то, которую вы якобы избежали
Почему бы хэндлеру не получить всё из DI?
s207883
Слабая связанность она только кажется таковой. На самом деле код сильно связан, но, благодаря посреднику, ты не знаешь как. Лично для меня это скорее минус.
Плюсы в виде сложной навигации по коду и прочее вы уже описали.
Если делать все тоже самое, но использовать явный вызов, то ничего особо не изменится.
Реально плюсом можно считать простоту распиливания такого кода на микросервисы, где посредник заменяется брокером сообщений. Но такой себе плюс.
dopusteam
Я сомневаюсь, что медиатр реально поможет распилить. Выдернуть условный handler в отдельный микросервис нельзя, т.к. он использует какие то общие типы и зависимости чаще всего. Так же где то конфигурится весь этот медиатр, там всякие цепочки и прочее. Но при этом смотря на handler я понятия не имею, от кого он реально зависит и могу ли я его просто так вытащить в отдельный микросервис
rtv128 Автор
Здравствуйте. В данной статье я хотел показать возможности данной библиотеки в разрезе реализации бизнес-логики, а не как реализатора стандартного шаблона “посредник”. С моей точки зрения, наличие у основного обработчика пре-обработчиков и пост-обработчиков дает некоторое преимущество в реализации некоторых стандартных процессов. Как бонус, пре-обработчик и пост-обработчик позволяют уменьшить связанность логики, так как физически они изолированы от основного обработчика и время их выполнения разнесены во времени. Это дает возможность вносить изменения в пред-обработчик и пост-обработчик, при этом не затрагивая логики работы основного обработчика и наоборот. Конечно, все зависит от масштаба доработок.