.NET 6開發(fā)TodoList應(yīng)用之實現(xiàn)PUT請求
需求
PUT請求本身其實可說的并不多,過程也和創(chuàng)建基本類似。在這篇文章中,重點是填上之前文章里留的一個坑,我們曾經(jīng)給TodoItem定義過一個標記完成的領(lǐng)域事件:TodoItemCompletedEvent,在SaveChangesAsync方法里做了一個DispatchEvents的操作。并且在DomainEventService實現(xiàn)IDomainEventService的Publish方法中暫時以下面的代碼代替了:
DomainEventService.cs
public async Task Publish(DomainEvent domainEvent) { // 在這里暫時什么都不做,到CQRS那一篇的時候再回來補充這里的邏輯 _logger.LogInformation("Publishing domain event. Event - {event}", domainEvent.GetType().Name); }
在前幾篇應(yīng)用MediatR實現(xiàn)CQRS的過程中,我們主要是和IRequest/IRequestHandler打的交道。那么本文將會涉及到另外一對常用的接口:INotification/INotificationHandler,來實現(xiàn)領(lǐng)域事件的處理。
目標
1.實現(xiàn)PUT請求;
2.實現(xiàn)領(lǐng)域事件的響應(yīng)處理;
原理與思路
實現(xiàn)PUT請求的原理和思路與實現(xiàn)POST請求類似,就不展開了。關(guān)于實現(xiàn)領(lǐng)域事件響應(yīng)的部分,我們需要實現(xiàn)INotification/INotificationHandler接口,并改寫Publish的實現(xiàn),讓它能發(fā)布領(lǐng)域事件通知。
實現(xiàn)
PUT請求
我們拿更新TodoItem的完成狀態(tài)來舉例,首先來自定義一個領(lǐng)域異常NotFoundException,位于Application/Common/Exceptions里:
NotFoundException.cs
namespace TodoList.Application.Common.Exceptions; public class NotFoundException : Exception { public NotFoundException() : base() { } public NotFoundException(string message) : base(message) { } public NotFoundException(string message, Exception innerException) : base(message, innerException) { } public NotFoundException(string name, object key) : base($"Entity \"{name}\" ({key}) was not found.") { } }
創(chuàng)建對應(yīng)的Command:
UpdateTodoItemCommand.cs
using MediatR; using TodoList.Application.Common.Exceptions; using TodoList.Application.Common.Interfaces; using TodoList.Domain.Entities; namespace TodoList.Application.TodoItems.Commands.UpdateTodoItem; public class UpdateTodoItemCommand : IRequest<TodoItem> { public Guid Id { get; set; } public string? Title { get; set; } public bool Done { get; set; } } public class UpdateTodoItemCommandHandler : IRequestHandler<UpdateTodoItemCommand, TodoItem> { private readonly IRepository<TodoItem> _repository; public UpdateTodoItemCommandHandler(IRepository<TodoItem> repository) { _repository = repository; } public async Task<TodoItem> Handle(UpdateTodoItemCommand request, CancellationToken cancellationToken) { var entity = await _repository.GetAsync(request.Id); if (entity == null) { throw new NotFoundException(nameof(TodoItem), request.Id); } entity.Title = request.Title ?? entity.Title; entity.Done = request.Done; await _repository.UpdateAsync(entity, cancellationToken); return entity; } }
實現(xiàn)Controller:
TodoItemController.cs
[HttpPut("{id:Guid}")] public async Task<ApiResponse<TodoItem>> Update(Guid id, [FromBody] UpdateTodoItemCommand command) { if (id != command.Id) { return ApiResponse<TodoItem>.Fail("Query id not match witch body"); } return ApiResponse<TodoItem>.Success(await _mediator.Send(command)); }
領(lǐng)域事件的發(fā)布和響應(yīng)
首先需要在Application/Common/Models定義一個泛型類,實現(xiàn)INotification接口,用于發(fā)布領(lǐng)域事件:
DomainEventNotification.cs
using MediatR; using TodoList.Domain.Base; namespace TodoList.Application.Common.Models; public class DomainEventNotification<TDomainEvent> : INotification where TDomainEvent : DomainEvent { public DomainEventNotification(TDomainEvent domainEvent) { DomainEvent = domainEvent; } public TDomainEvent DomainEvent { get; } }
接下來在Application/TodoItems/EventHandlers中創(chuàng)建對應(yīng)的Handler:
TodoItemCompletedEventHandler.cs
using MediatR; using Microsoft.Extensions.Logging; using TodoList.Application.Common.Models; using TodoList.Domain.Events; namespace TodoList.Application.TodoItems.EventHandlers; public class TodoItemCompletedEventHandler : INotificationHandler<DomainEventNotification<TodoItemCompletedEvent>> { private readonly ILogger<TodoItemCompletedEventHandler> _logger; public TodoItemCompletedEventHandler(ILogger<TodoItemCompletedEventHandler> logger) { _logger = logger; } public Task Handle(DomainEventNotification<TodoItemCompletedEvent> notification, CancellationToken cancellationToken) { var domainEvent = notification.DomainEvent; // 這里我們還是只做日志輸出,實際使用中根據(jù)需要進行業(yè)務(wù)邏輯處理,但是在Handler中不建議繼續(xù)Send其他Command或Notification _logger.LogInformation("TodoList Domain Event: {DomainEvent}", domainEvent.GetType().Name); return Task.CompletedTask; } }
最后去修改我們之前創(chuàng)建的DomainEventService,注入IMediator并發(fā)布領(lǐng)域事件,這樣就可以在Handler中進行響應(yīng)了。
DomainEventService.cs
using MediatR; using Microsoft.Extensions.Logging; using TodoList.Application.Common.Interfaces; using TodoList.Application.Common.Models; using TodoList.Domain.Base; namespace TodoList.Infrastructure.Services; public class DomainEventService : IDomainEventService { private readonly IMediator _mediator; private readonly ILogger<DomainEventService> _logger; public DomainEventService(IMediator mediator, ILogger<DomainEventService> logger) { _mediator = mediator; _logger = logger; } public async Task Publish(DomainEvent domainEvent) { _logger.LogInformation("Publishing domain event. Event - {event}", domainEvent.GetType().Name); await _mediator.Publish(GetNotificationCorrespondingToDomainEvent(domainEvent)); } private INotification GetNotificationCorrespondingToDomainEvent(DomainEvent domainEvent) { return (INotification)Activator.CreateInstance(typeof(DomainEventNotification<>).MakeGenericType(domainEvent.GetType()), domainEvent)!; } }
驗證
啟動Api項目,更新TodoItem的完成狀態(tài)。
請求
響應(yīng)
領(lǐng)域事件發(fā)布
總結(jié)
這篇文章主要在實現(xiàn)PUT請求的過程中介紹了如何通過MediatR去響應(yīng)領(lǐng)域事件,我們用的示例代碼中類似“創(chuàng)建TodoList”,包括后面會講到的“刪除TodoItem”之類的領(lǐng)域事件,都是相同的處理方式,我就不一一演示了。
可以看出來,在我們這個示例應(yīng)用程序的框架基本搭建完畢以后,進行領(lǐng)域業(yè)務(wù)的開發(fā)的思路是比較清晰的,模塊之間的耦合也處在一個理想的情況。
在我們來完成CRUD的最后一個請求之前,下一篇會簡單地介紹一下PATCH請求的相關(guān)內(nèi)容,這個請求實際應(yīng)用比較少,但是為了保持知識樹的完整性,還是會過一下。?
到此這篇關(guān)于.NET 6開發(fā)TodoList應(yīng)用之實現(xiàn)PUT請求的文章就介紹到這了,更多相關(guān).NET 6 實現(xiàn)PUT請求內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- .NET 6開發(fā)TodoList應(yīng)用之實現(xiàn)查詢分頁
- .NET 6開發(fā)TodoList應(yīng)用之實現(xiàn)ActionFilter
- .NET 6開發(fā)TodoList應(yīng)用之實現(xiàn)接口請求驗證
- .NET?6開發(fā)TodoList應(yīng)用之實現(xiàn)DELETE請求與HTTP請求冪等性
- .NET 6開發(fā)TodoList應(yīng)用之實現(xiàn)全局異常處理
- .NET 6開發(fā)TodoList應(yīng)用之使用AutoMapper實現(xiàn)GET請求
- .NET?6開發(fā)TodoList應(yīng)用之實現(xiàn)Repository模式
- .NET?6開發(fā)TodoList應(yīng)用之使用MediatR實現(xiàn)POST請求
- .NET 6開發(fā)TodoList應(yīng)用引入數(shù)據(jù)存儲
- .NET?6開發(fā)TodoList應(yīng)用引入第三方日志庫
- .NET 6開發(fā)TodoList應(yīng)用實現(xiàn)結(jié)構(gòu)搭建
- .NET?6開發(fā)TodoList應(yīng)用實現(xiàn)系列背景
- 使用.NET?6開發(fā)TodoList應(yīng)用之引入數(shù)據(jù)存儲的思路詳解
- 使用.NET?6開發(fā)TodoList應(yīng)用之領(lǐng)域?qū)嶓w創(chuàng)建原理和思路
- .NET?6開發(fā)TodoList應(yīng)用之請求日志組件HttpLogging介紹
相關(guān)文章
如何輕松搭建Windows8云平臺的開發(fā)環(huán)境
Windows Store應(yīng)用是基于Windows 8操作系統(tǒng)的新一代Windows應(yīng)用程序,其開發(fā)平臺以及運行模式和以往傳統(tǒng)平臺略有不同。為了幫助更多開發(fā)人員加入到Windows Store應(yīng)用開發(fā)行列,本篇將介紹如何在Windows Azure云平臺搭建Windows8應(yīng)用開發(fā)環(huán)境,本篇介紹的方法適合未安裝Windows8操作系統(tǒng),使用Mac或者Linux平臺的開發(fā)人員參考閱讀。2013-02-02