در
قسمتهای قبل سعی شد یک دید کلی از نحوه استفاده از این زیرساخت ارائه شود؛ در این قسمت علاوه بر بررسی مکانیزم Eventing، با جزئیات بیشتری به استفاده از سرویسهای پیادهسازی شده پرداخته خواهد شد.
مکانیزم Eventing
استفاده از رخدادها، یکی از راهحلهای رسیدن به طراحی با Loose Coupling (اتصال سست و ضعیف، وابستگی ضعیف) میباشد؛ همچنین برای حذف چرخه در فرآیند وابستگی مولفههای سیستم نیز مورد استفاده قرار میگیرد. در این زیرساخت برای Application Layer مبتنیبر CRUD، مکانیزم
BusinessEvent با هدف در معرض دید قراردادن
یکسری نقاط قابل گسترش توسط سایر بخشهای سیستم، تعبیه شده است. برای استفاده از این مکانیزم لازم است بسته نیوگت زیر را نصب کنید:
PM> Install-Package DNTFrameworkCore
سپس امکان این را خواهید داشت که مشترک رخدادهای مرتبط با عملیات CUD متناظر با موجودیتهای سیستم، شوید. به عنوان مثال، برای اینکه بتوان مشترک رخداد ویرایش مرتبط با موجودیت Task شد، باید به شکل زیر عمل کرد:
public class TaskEditingBusinessEventHandler : BusinessEventHandler<EditingBusinessEvent<TaskModel, int>>
{
private readonly ILogger<TaskEditingBusinessEventHandler> _logger;
public TaskEditingBusinessEventHandler(ILogger<TaskEditingBusinessEventHandler> logger)
{
_logger = logger ?? throw new ArgumentNullException(nameof(logger));
}
public override Task<Result> Handle(EditingBusinessEvent<TaskModel, int> @event)
{
foreach (var model in @event.Models)
{
_logger.LogInformation($"Title changed from: {model.OriginalValue.Title} to: {model.NewValue.Title}");
}
return Task.FromResult(Ok());
}
}
کار با پیادهسازی واسط جنریک IBusinessEventHandler یا ارثبری از کلاس جنریک BusinessEventHandler آغاز میشود؛ سپس نیاز است Type Parameter متناظر را نیز مشخص کنیم. برای این منظور در تکه کد بالا از رخداد جنریک EditingBusinessEvent استفاده شده است. همچنین همانطور که ملاحظه میکنید، نیاز است نوع Model مورد نظر نیز مشخص شده باشد؛ در اینجا از TaskModel به عنوان Model/DTO عملیات CUD موجودیت Task استفاده شده است.
رخدادهای Creating/Created/Deleting/Deleted دارای خصوصیتی بنام Models هستند که نوع آن IEnumerable<TModel> میباشد. ولی این خصوصیت در رخدادهای Editing/Edited از نوع IEnumerable<ModifiedModel<TModel>> میباشد؛ در این صورت به مقادیر موجود در بانک اطلاعاتی و همچنین مقادیری که توسط استفاده کننده از سرویس جاری به عنوان آرگومان به متد ویرایش ارسال شده است، دسترسی خواهیم داشت.
public class ModifiedModel<TValue>
{
public TValue NewValue { get; set; }
public TValue OriginalValue { get; set; }
}
نکته: همانطور که در قسمتهای قبل اشاره شد، Application Layer مدنظر ما با یک Model/DTO برای عملیات CUD کار میکند؛ از این جهت، منطق تجاری و همچنین قواعد تجاری برفراز همان Model/DTO اجرا خواهند شد و بهتبع آن، اگر سایر بخشهای سیستم نیز قصد گسترش منطق تجاری مرتبط با یک موجودیت را دارند، باید با همان Model/DTO کار کنند.
چه زمانی استفاده از مکانیزم BusinessEvent مطرح شده توصیه میشود؟
به طور کلی محدودیتی در استفاده از آن وجود ندارد؛ در مواردی مشابه اگر قصد اعمال یکسری قواعد تجاری توسط سایر مولفههای سیستم را دارید و قصد ندارید ارجاعی به آن مولفه در مولفه جاری وجود داشته باشد یا بدلیل ایجاد چرخه، این امکان وجود ندارد، میتوان از این مکانیزم بهره برد. برای مثال زمانی که یکسری قواعد تجاری جدید قرار است از سمت مولفه فروش بر روی مولفه مرتبط با مدیریت محصولات اعمال شود.
نکته: اگر قصد ارائه یک رخداد سفارشی را دارید، میتوانید واسط
IEventBus را تزریق کرده و از متد TriggerAsync آن استفاده کنید.
استفاده از سرویسهای موجودیتها
OOP : Everything is an object
CRUD-based thinking : Everything is CRUD
استفاده از سرویسهای موجودیتها به تولید CrudController مرتبط ختم نمیشود و در تفکر مبتنیبر CRUD، تمام عملیات مرتبط با یک موجودیت از یک تونل واحد عبور خواهند کرد. مسئول این تونل در ابتدا متد Create میباشد و در ادامه توسط متد Edit مدیریت میشود. به عنوان مثال، اگر امروز در یک سیستم رستورانی با نحوههای فروش مختلف، قرار باشد در زمان ثبت گروه کالایی جدید و براساس تنظیمات سیستم، آن گروه کالایی به صورت خودکار به لیست گروههای کالایی مرتبط با تمام نحوههای فروش اضافه شود، این امر باید از طریق منطق تجاری توسعه داده شده برای نحوه فروش، انجام پذیرد. قرار نیست شما منطق تجاری مرتبط با نحوه فروش را دور بزنید و به صورت دستی شروع به ثبت این اطلاعات در بانک اطلاعاتی کنید. در این شرایط میبایست با استفاده از مکانیزم BusinessEvent به شکل زیر عمل کرد:
public class ItemCategoryCreatedBusinessEventHandler : IBusinessEventHandler<CreatedBusinessEvent<ItemCategoryModel, int>>
{
private readonly ISaleMethodService _saleMethodService;
public TaskEditingBusinessEventHandler(ISaleMethodService saleMethodService)
{
_saleMethodService = saleMethodService ?? throw new ArgumentNullException(nameof(saleMethodService));
}
public override Task<Result> Handle(CreatedBusinessEvent<ItemCategoryModel, int> @event)
{
var methods = _saleMethodService.FindAsnc();
foreach (var method in methods)
{
foreach (var model in @event.Models)
{
method.ItemCategories.Add(new SaleMethodItemCategoryModel
{
ItemCategoryId = model.Id,
TrackingState = TrackingState.Added;
});
}
}
return _saleMethodService.EditAsync(methods);
}
این آبونه شدن به رخداد Created مرتبط با گروه کالایی، از سمت مولفه فروش انجام گرفته است. در بدنه متد Handle، ابتدا لیست نحوههای فروش موجود در سیستم توسط متد FindAsync بدون پارامتر واکشی شده و سپس با پیمایش خصوصیت Models مرتبط با رخداد مدنظر، بهازای تکتک گروههای کالایی ثبت شده، یک وهله از SaleMethodItemCategoryModel به عنوان Detail موجودیت SaleMethod اضافه میشود. سپس با استفاده از متد EditAsync لیست این نحوههای فروش را ویرایش خواهیم کرد.
نکته: در حد امکان این هندلرها را به صورت تک مسئولیتی طراحی کرده و توسعه دهید؛ این قضیه برای نوشتن آزمونهای واحد مرتبط با هندلرها، حیاتی میباشد.