تا اینجا با اصول توسعهی برنامههای مبتنی بر Blazor Server آشنا شدیم. در ادامهی این سری، روش توسعه برنامههای مبتنی بر Blazor WASM را بررسی خواهیم کرد و پیش از شروع آن، باید بتوان امکانات سمت سرور مورد نیاز این نوع برنامههای سمت کلاینت را از طریق یک Web API تامین کرد که شامل دریافت و ارائهی اطلاعات و همچنین اعتبارسنجی و احراز هویت مبتنی بر JWT یکپارچهی با ASP.NET Core Identity است.
ایجاد پروژهی ASP.NET Core Web API
برای تامین اطلاعات برنامهی سمت کلاینت Blazor WASM و همچنین فراهم آوردن زیرساخت اعتبارسنجی کاربران آن، نیاز به یک پروژهی ASP.NET Core Web API داریم که آنرا با اجرای دستور dotnet new webapi در یک پوشهی خالی، برای مثال به نام BlazorWasm.WebApi ایجاد میکنیم.
البته این پروژه، از زیرساختی که در برنامهی Blazor Server بررسی شدهی تا این قسمت، ایجاد کردیم نیز استفاده خواهد کرد. همانطور که پیشتر نیز عنوان شد، هدف از قسمت Blazor Server مثال این سری، آشنایی با مدل برنامه نویسی خاص آن بود؛ وگرنه میتوان کل این پروژه را با Blazor Server و یا کل آنرا با Web API + Blazor WASM نیز پیاده سازی کرد. در این مثال، قسمتهای مدیریتی برنامهی مدیریت هتل را توسط Blazor Server (مانند قسمتهای تعریف اتاقها و امکانات رفاهی هتل) و قسمت مخصوص کاربران آنرا مانند رزرو کردن اتاقها، توسط Blazor WASM پیاده سازی میکنیم. به همین جهت قسمتهایی از این دو پروژه، مانند سرویسهای استفاده شدهی در پروژهی Blazor server، در پروژهی Web API مکمل Blazor WASM، قابلیت استفادهی مجدد را دارند.
افزودن سرویسهای آغازین مورد نیاز، به پروژهی Web API
در فایل آغازین BlazorWasm\BlazorWasm.WebApi\Startup.cs، برای شروع به تکمیل Web API، نیاز به این سرویسها را داریم:
namespace BlazorWasm.WebApi
{
public class Startup
{
//...
public void ConfigureServices(IServiceCollection services)
{
services.AddAutoMapper(typeof(MappingProfile).Assembly);
services.AddScoped<IHotelRoomService, HotelRoomService>();
services.AddScoped<IAmenityService, AmenityService>();
services.AddScoped<IHotelRoomImageService, HotelRoomImageService>();
var connectionString = Configuration.GetConnectionString("DefaultConnection");
services.AddDbContext<ApplicationDbContext>(options => options.UseSqlServer(connectionString));
services.AddIdentity<IdentityUser, IdentityRole>()
.AddEntityFrameworkStores<ApplicationDbContext>()
.AddDefaultTokenProviders();
//...
در اینجا سرویسهای AutoMapper، تنظیمات ابتدایی DbContext برنامه، به همراه سرویسهای Identity (بدون UI آن) و افزودن سرویسهای اتاقها و امکانات رفاهی هتل را نیاز داریم. به همین جهت ارجاعات و وابستگیهای زیر را به فایل csproj جاری اضافه میکنیم تا پروژههای DataAccess ،Services و Mappings قابل دسترسی و استفاده شوند:
<Project Sdk="Microsoft.NET.Sdk.Web">
<PropertyGroup>
<TargetFramework>net5.0</TargetFramework>
</PropertyGroup>
<ItemGroup>
<PackageReference Include="AutoMapper.Extensions.Microsoft.DependencyInjection" Version="8.1.1" />
<PackageReference Include="Swashbuckle.AspNetCore" Version="5.6.3" />
</ItemGroup>
<ItemGroup>
<ProjectReference Include="..\..\BlazorServer\BlazorServer.DataAccess\BlazorServer.DataAccess.csproj" />
<ProjectReference Include="..\..\BlazorServer\BlazorServer.Services\BlazorServer.Services.csproj" />
<ProjectReference Include="..\..\BlazorServer\BlazorServer.Models.Mappings\BlazorServer.Models.Mappings.csproj" />
</ItemGroup>
</Project>
همچنین در این پروژه نیز از همان بانک اطلاعاتی پروژهی Blazor Server که تاکنون تکمیل کردیم، استفاده میکنیم. بنابراین محتوای فایل BlazorWasm\BlazorWasm.WebApi\appsettings.json آن نیز مشابهاست:
{
"ConnectionStrings": {
"DefaultConnection": "Server=(localdb)\\mssqllocaldb;Database=HotelManagement;Trusted_Connection=True;MultipleActiveResultSets=true"
}
}
تعریف کنترلر HotelRoom
در ادامه کدهای اولین کنترلر Web API را مشاهده میکنید که مرتبط است با بازگشت اطلاعات تمام اتاقهای ثبت شده و یا بازگشت اطلاعات یک اتاق ثبت شده:
using BlazorServer.Models;
using BlazorServer.Services;
using Microsoft.AspNetCore.Http;
using Microsoft.AspNetCore.Mvc;
using System.Collections.Generic;
using System.Threading.Tasks;
namespace BlazorWasm.WebApi.Controllers
{
[Route("api/[controller]")]
public class HotelRoomController : ControllerBase
{
private readonly IHotelRoomService _hotelRoomService;
public HotelRoomController(IHotelRoomService hotelRoomService)
{
_hotelRoomService = hotelRoomService;
}
[HttpGet]
public IAsyncEnumerable<HotelRoomDTO> GetHotelRooms()
{
return _hotelRoomService.GetAllHotelRoomsAsync();
}
[HttpGet("{roomId}")]
public async Task<IActionResult> GetHotelRoom(int? roomId)
{
if (roomId == null)
{
return BadRequest(new ErrorModel
{
Title = "",
ErrorMessage = "Invalid Room Id",
StatusCode = StatusCodes.Status400BadRequest
});
}
var roomDetails = await _hotelRoomService.GetHotelRoomAsync(roomId.Value);
if (roomDetails == null)
{
return BadRequest(new ErrorModel
{
Title = "",
ErrorMessage = "Invalid Room Id",
StatusCode = StatusCodes.Status404NotFound
});
}
return Ok(roomDetails);
}
}
}
- این کنترلر، از سرویس IHotelRoomService که در قسمتهای قبل تکمیل کردیم، استفاده میکند.
- ErrorModel آنرا در همان پروژهی قبلی مدلها، در فایل BlazorServer\BlazorServer.Models\ErrorModel.cs به صورت زیر ایجاد کردهایم:
namespace BlazorServer.Models
{
public class ErrorModel
{
public string Title { get; set; }
public int StatusCode { get; set; }
public string ErrorMessage { get; set; }
}
}
در این حالت اگر برنامهی Web API را اجرا کنیم، به خروجی Swagger زیر میرسیم که جزئیات این فناوری را در سری «
مستند سازی ASP.NET Core 2x API توسط OpenAPI Swagger» پیشتر بررسی کردیم:
یکی از مزایای آن، امکان آزمایش API تهیه شده، بدون نیاز به تهیهی هیچ نوع کلاینت خاصی است. برای مثال اگر بر روی api/hotelroom آن کلیک کنیم، گزینهی «try it out» آن ظاهر شده و با کلیک بر روی آن، اینبار دکمهی execute ظاهر میشود. در ادامه با کلیک بر روی دکمهی اجرای آن، اکشن متد GetHotelRooms اجرا شده و خروجی زیر ظاهر میشود:
و یا اگر بخواهیم متد GetHotelRoom را توسط آن آزمایش کنیم، بر اساس پارامترهای آن، رابط کاربری زیر را تشکیل میدهد که امکان دریافت شمارهی اتاق را دارد:
انجام تنظیمات ابتدایی CORS و خروجی JSON برنامه
قرار است این API را از طریق پروژهی Blazor سمت کلاینت خود استفاده کنیم که آدرس آن، با آدرس API یکی نیست. به همین جهت نیاز است
تنظیمات CORS را به صورت زیر اضافه کنیم:
namespace BlazorWasm.WebApi
{
public class Startup
{
public void ConfigureServices(IServiceCollection services)
{
// ...
services.AddCors(o => o.AddPolicy("HotelManagement", builder =>
{
builder.AllowAnyOrigin().AllowAnyMethod().AllowAnyHeader();
}));
services.AddControllers()
.AddJsonOptions(options =>
{
options.JsonSerializerOptions.PropertyNamingPolicy = null;
// To avoid `JsonSerializationException: Self referencing loop detected error`
options.JsonSerializerOptions.ReferenceHandler = ReferenceHandler.Preserve;
});
// ...
}
public void Configure(IApplicationBuilder app, IWebHostEnvironment env)
{
// ...
app.UseCors("HotelManagement");
app.UseRouting();
app.UseAuthentication();
// ...
}
}
}
در اینجا علاوه بر تنظیمات CORS، تنظیمات JsonSerializer را هم تغییر دادهایم تا خطاهای Self referencing loop را در حین ارائهی خروجیهای Web API، مشاهده نکنیم (همان نکتهی «
تهیه خروجی JSON از مدلهای مرتبط، بدون Stack overflow»).
کدهای کامل این مطلب را از اینجا میتوانید دریافت کنید: Blazor-5x-Part-24.zip