چکیده: با استفاده از ویژگی Action method selector در ASP.NET MVC می توانید اکشن متد خاصی را با توجه به نیازهای بیزینستان اجرا و یا رد نمایید.
ما می توانیم به راحتی فیلترهای سفارشی برای authentication ، ثبت رویداد ها و خطاها و غیره پیاده سازی نماییم. در نتیجه می توانیم کنترلرها و Viewهای سفارشی برای توسعه آبجکت مدلها در application مان پیاده سازی نماییم.
پردازش request در MVC از Controller شروع می شود. این آبجکت مسئول نگه داری اکشن متدهای متفاوتی است که اکشنی که باید اجرا شود درون request توسط http مشخص می شود. فرایند request در MVC با استفاده از IActionInvoker که درون خود دارد تعیین می کند که کدام اکشن متد باید فراخوانده شده و اجرا شود. نمودار زیر نشان می دهد که چطور کنترلر، اکشن متد مورد نظر را بر اساس attribute فرامی خواند.
کنترلر در MVC میتواند چند متد با یک نام داشته باشد. این اکشن متدها در MVC با Attribute های HttpGet و HttpPost در بالای خود تعریف می شوند. این Attribute ها تعیین می کنند که request ارسال شده بر اساس GET و یا POST بودنش کدام یک از اکشن متدها باید اجرا گردد. نمودار زیر رفتار Action Invoker را برا یانتخاب اکشن متد در MVC Controller نشان می دهد.
نمودار بالا نحوه انتخاب و اجرا شدن اکشن متد را نشان می دهد. از آنجایی که ما میدانیم Route در MVC شامل نام کنترلر و متد است، سپس اکشن مورد نظر بر اساس نام اکشن با استفاده از ویژگی Action Name Selector که درون خود Mvc وجود دارد فراخوانده می شود.
اما اگر کنترلر دارای اکشن متدها با یک نام یکسان است (در مثال ما متد create) که در اکثر موارد اتفاق می افتد سپس ویژگی توکار ActionMethod selector وارد فرایند می شود و اکشن مورد نظر را بر اساس نوع http request که در دو حالت Get و یا Post است فرا می خواند. این اطلاعات که request از چه نوعی است درون Http Request header قرا می گیرد. که در این حالت MVC از کلاس ActionMethodSelectorAttribute برای فراخوانی متد مورد نظر جهت اجرا استفاده می نماید.
حال ما میتوانیم یک Action Method Selector سفارشی ایجاد نماییم و کنترلر در MVC را مجبور نماییم که یک اکشن متد ویژه را بر اساس مقداری که در Http header درون request پاس داده می شود را فراخوانی نماید. به این منظور ما در این مقاله قصد داریم که طی یک مثال به شما نشان دهیم که چطور با استفاده از این ویژگی میخواهیم محصولاتی را در یک دسته بندی خاص که نام دسته درون http Header در request قرار گرفته می شود را برگردانیم. و این ویژگی را شما برای هرنوع موردی که در پروژه خود دارید می توانید به کار گیرید.
پیاده سازی Action Method Selector
مرحله 1: ویژوال استودیو را باز نمایید و یک پروژه جدید با نام MVC_CustomActionMethodSelector از نوع ASP.NET Web Application را مانند شکل زیر ایجاد نمایید:
بر روی ok کلیک نمایید سپس در پنجره بعدی template را از نوع empty انتخاب نمایید:
مرحله 2: در فولدر مدلها یک فایل کلاس به نام ModelClasses.cs ایجاد نمایید و کدهای زیر را درون آن قرار دهید:
using System.Collections.Generic; namespace MVC_CustomActionMethodSelector.Models { public class Product { public int ProductId { get ; set ; } public string ProductName { get ; set ; } public int Price { get ; set ; } public string Category { get ; set ; } } public class ElectronicsProductsDatabase : List<Product> { public ElectronicsProductsDatabase() { Add( new Product() { ProductId = 1, ProductName = "Desktop" , Price = 34000, Category = "Electronics" }); Add( new Product() { ProductId = 2, ProductName = "Laptop" , Price = 34000, Category = "Electronics" }); Add( new Product() { ProductId = 3, ProductName = "Router" , Price = 34000, Category = "Electronics" }); Add( new Product() { ProductId = 4, ProductName = "Mouse" , Price = 34000, Category = "Electronics" }); Add( new Product() { ProductId = 5, ProductName = "USB HDD" , Price = 34000, Category = "Electronics" }); Add( new Product() { ProductId = 6, ProductName = "LCD" , Price = 34000, Category = "Electronics" }); } } public class DataAccessElectronics { public List<Product> GetDataElectronics() { return new ElectronicsProductsDatabase(); } } public class MechanicalProductsDatabase : List<Product> { public MechanicalProductsDatabase() { Add( new Product() { ProductId = 1, ProductName = "JCB" , Price = 314000, Category = "Mechanical" }); Add( new Product() { ProductId = 2, ProductName = "Tractor" , Price = 134000, Category = "Mechanical" }); Add( new Product() { ProductId = 3, ProductName = "Leath Machine" , Price = 74000, Category = "Mechanical" }); Add( new Product() { ProductId = 4, ProductName = "Crusher" , Price = 104000, Category = "Mechanical" }); Add( new Product() { ProductId = 5, ProductName = "Mixer" , Price = 143000, Category = "Mechanical" }); } } public class DataAccessMechanical { public List<Product> GetDataMechanical() { return new MechanicalProductsDatabase(); } } } |
کد بالا شامل کلاسهای زیر می باشد:
- Product – این یک کلاس موجودیت شامل اطلاعات محصولات می باشد.
- ElectronicsProductsDatabase – این کلاس از کلاس List<Product> مشتق شده می باشد و شامل داده های پیش فرض برا ی محصولات الکترونیکی می باشد.
- DataAccessElectronics – این کلاس شامل متد برای برگرداندن تمام محصولات الکترونیکی می باشد.
- MechanicalProductsDatabase – این کلاس شامل داده های پیش فرض برا ی محصولات مکانیکی می باشد.
- DataAccessMechanical – این کلاس شامل متد برای برگرداندن تمام محصولات مکانیکی می باشد.
توجه: در یک پروژه واقعی، شما می توانید از Entity framework برای map کردن داده ها به دیتابیس استفاده نمایید.
مرحله 3: در پروژه یک فولدر به نام CustomActionMethodSelector ایجاد نمایید. در این فولدر یک فایل کلاس جدید به نام MyActionMethodSelector ایجاد نمایید و کدهای زیر را درون آن قرار دهید:
using System.Reflection; using System.Web.Mvc; namespace MVC_CustomActionMethodSelector.CustomActioMethodSelector { public class ElectronicsActionMethodSelectorAttribute : ActionMethodSelectorAttribute { public override bool IsValidForRequest(ControllerContext controllerContext, MethodInfo methodInfo) { return controllerContext.HttpContext.Request.Headers[ "P-Electronics" ] != null ; } } public class MechanicalActionMethodSelectorAttribute : ActionMethodSelectorAttribute { public override bool IsValidForRequest(ControllerContext controllerContext, MethodInfo methodInfo) { return controllerContext.HttpContext.Request.Headers[ "P-Mechanical" ] != null ; } } } |
کد بالا شامل کلاسهای زیر می باشد:
ElectronicsActionMethodSelectorAttribute
- این کلاس از کلاس ActionMethodSelectorAttribute مشتق شده است.
- این کلاس متد IsValidateForRequest را override می کند. این متد، آبجکت ControllerContext و MethodInfo را به عنوان ورودی می گیرد. ControllerContext حاوی context کنترلر بارگزاری شده در request فرستاده شده برای اجرای اکشن مورد نظر می باشد.
- این متد بررسی می کند که آیا request header ارسالی شامل عبارت P-Electronics: true می باشد یا خیر. اگر جواب مثبت باشد سپس اکشن متدی که حاوی ElectronicsActionMethodSelectorAttribute می باشد اجرا خواهد شد در غیر اینصورت خطای 404 برگردانده می شود.
MechanicalActionMethodSelectorAttribute
این کلاس نیز دقیقا مانند کلاس بالا می باشد با این تفاوت که request header ارسالی را برای عبارت P-Mechanical: true بررسی می نماید.
مرحله 4: در پوشه controllers، یک کنترلر با نام ProductController ایجاد نمایید و متدهای زیر را درون آن کپی نمایید:
using System.Web.Mvc; using MVC_CustomActionMethodSelector.Models; using MVC_CustomActionMethodSelector.CustomActioMethodSelector; namespace MVC_CustomActionMethodSelector.Controllers { public class ProductController : Controller { // GET: Product public ActionResult Index() { return View(); } [ElectronicsActionMethodSelector] public ActionResult GetElectronics() { var ds = new DataAccessElectronics(); var products = ds.GetDataElectronics(); return View( "IndexElectronics" , products); } [MechanicalActionMethodSelector] public ActionResult GetMechanical() {ttronutes var ds = new DataAccessMechanical(); var products = ds.GetDataMechanical(); return View( "IndexMechanical" , products); } } } |
کد بالا اکشن GetElectronics() و GetMechanical() با attribute های ElectronicsActionMethodSelector و MechanicalActionMethodSelector نشان می دهد که بر روی آنها اعمال شده اند. این به این معنیست زمانی که ما request را به هریک از این متدها با مقادیر P-Electronics and P-Mechanical ارسال می کنیم، این متدها فراخوانده و اجرا می شوند.
view های تولید شده برای اکشن متدهای بالا به فرض مثال IndexElectronics.cshtml و IndexMechanical.cshtml می باشند.
مرحله 5: حال یک breakpoint داخل اکشن های GetElectronics و GetMechanical و متد IsValidForRequest در کلاس Action method selector سفارشی که در مرحله 3 ایجاد نمودیم قرار دهید. ما پروژه را با ابزار fiddler تست می نماییم.
application را اجرا نمایید و در تب composer در ابزار fiddler آدرس زیر را وارد نمایید:
http://localhost:44083/Product/GetElectronics
روی دکمه Execute کلیک نمایید که در نتیجه با خطاس 404 مواجه خواهید شد:
حال request header را با مقدار P-Electronics: true به روزنمایید:
روی دکمه Execute کلیک نمایید. در نتیجه کد وارد debug می شود. مقدار header را مانند شکل زیر بررسی نمایید:
مقدار header به عنوان P-Electronics دریافت شده است. زمانی که عملیات debug را ادامه دهید مشاهده خواهید نمود که اکشن GetElectronics مانند شکل زیر فراخوانی خواهد شد:
F5 را بزنید. در نتیجه اجرا با نتیجه زیر در Web View ابزار Fiddler کامل خواهد شد:
در نتیجه لیست تمام محصولات الکترونیکی به عنوان نتیجه برگشت داده میشود.
نتیجه گیری
ASP.NET MVC یک framework کامل برای توسعه application می باشد. ویژگی Action method selector باعث می شود ما عملیت invoke شدن اکشن متد مان را بر اساس نیاز بیزنسی مان به آسانی می سازد.