نگاشت اشیاء در AutoMapper توسط Attribute ها #2 - تبدیل ویژگیها به نگاشت
پنج شنبه 28 آبان 1394 8:04 AM
پس از معرفی ویژگیهای لازم، در ادامه با نحوهی تبدیل این ویژگیها به معادل نگاشت آنها در automapper خواهم پرداخت.
متد زیر هستهی اصلی عملیات است و کلیهی نگاشتهای لازم را انجام میدهد. این متد وظیفهی تبدیل نگاشتها را دارد. نگاشتهایی که با Attributes مشخص شدهاند:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
|
public static void Initialize(Assembly assembly) { //register global convertors. AutoMapper.Mapper.CreateMap<DateTime, string >().ConvertUsing<DateTimeToPersianDateTimeConverter>(); var typesToMap = from t in assembly.GetTypes() let attr = t.GetCustomAttribute<MapFromAttribute>() where attr != null select new {SourceType = attr.SourceType, Destination = t, Attribute = attr}; foreach (var map in typesToMap) { AutoMapper.Mapper.CreateMap(map.SourceType, map.Destination) .DoMapForMemberAttribute() // for different property names in source and destination .DoIgnoreMapAttribute() // ignore specified properties .DoUseValueResolverAttribute() // set value resolvers .DoIgnoreAllNonExisting() // its have to be the latest. ; } //endeach AutoMapper.Mapper.AssertConfigurationIsValid(); } |
ورودی این متد اسملبی مربوط به ویوومدل میباشد (برای زمانیکه ویوومدلها در اسمبلی دیگری باشند).
در سطر اول، اقدام به رجیستر کردن کلیهی مبدلهای سراسری میکنیم. در این سطر مبدل تاریخ به کوچی خورشیدی مورد استفاده قرار گرفته است. سپس در اسمبلی داده شده، کلیه نوعهایی که ویژگی MapFromAttribute را دارند، یافته و جدا میکنیم. در حلقهی foreach ابتدا نگاشت نوع مبدأ و مقصد را انجام میدهیم. خروجی این متد از نوع IMappingExpression است. گر چه این اینترفیس برای تغییر بسته است، ولی قابل توسعه میباشد و عملیات را توسط متدهای الحاقی انجام میدهیم(اصل OCP).
اگر به نحوهی نامگذاری متدهای الحاقی تعریف شده دقت کرده باشید، تنها کلمهی Do به ابتدای نام ویژگیها اضافه شده است.
متد الحاقی DoMapFormMemberAttribute
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
public static IMappingExpression DoMapForMemberAttribute( this IMappingExpression expression) { var ok = from p in expression.TypeMap.DestinationType.GetProperties() let attr = p.GetCustomAttribute<MapForMemberAttribute>() where attr != null select new {AttributeValue = attr, PropertyName = p.Name}; foreach (var property in ok) { expression.ForMember(property.PropertyName, opt => opt.MapFrom(property.AttributeValue.MemberToMap)); } return expression; } |
هر IMappingExpression دارای امکاناتی برای نگهداری و انجام فعالیت بر روی یک نگاشت میباشد. در کوئری ابتدای متد، کلیهی پروپرتیهایی را که دارای ویژگی MapForMemeberAttribute میباشند، یافته و جدا میکنیم. این پروپرتیها از نظر معادل اسمی در نوع مبدأ و مقصد متفاوت هستند. سپس در حلقه، کار اتصال پروپرتی مبدأ و مقصد صورت میگیرد.
متد الحاقی DoIgnoreMapAttribute
1
2
3
4
5
6
7
8
9
10
|
public static IMappingExpression DoIgnoreAttribute( this IMappingExpression expression) { foreach (var property in expression.TypeMap.DestinationType.GetProperties() .Where(x => x.GetCustomAttribute<IgnoreMapAttribute>() != null )) { expression.ForMember(property.Name, opt => opt.Ignore()); } return expression; } |
این متد کلیهی پروپرتیهایی را که دارای ویژگی IgnoreMapAttribute باشند، از گردونهی نگاشت automapper خارج میکند. به عنوان مثال پروپرتی Password در ویوومدل مربوط به تغییر گذرواژه را نظر بگیرید. این پروپرتی نباید مقدار معادلی در شیء EF داشته باشد. از طرفی هم باید در ویوو وجودداشته باشد. با استفاده از این ویژگی هیچ نگاشتی انجام نمیشود و میتوان تضمین کرد که گذرواژه به ویوومدل و ویوو راه پیدا نمیکند.
متد الحاقی DoUseValueResolverAttribute
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
public static IMappingExpression DoUseValueResolverAttribute( this IMappingExpression expression) { var ok = from p in expression.TypeMap.DestinationType.GetProperties() let attr = p.GetCustomAttribute<UseValueResolverAttribute>() where attr != null select new {AttributeValue = attr, PropertyName = p.Name}; foreach (var property in ok) { expression.ForMember(property.PropertyName, opt => opt.ResolveUsing(property.AttributeValue.ValueResolver)); } return expression; } |
به شیوهی قبل، ابتدا نوع هایی را که دارای ویژگی UseValueResolverAttribute باشند، یافته و جدا میکنیم. سپس در حلقه، کار نگاشت متناظر در automapper انجام میگیرد. لازم به ذکر است که متد opt.ResolveUsing یک شیء با کارآیی (can do) اینترفیس IValueResolver را به عنوان آرگومان میگیرد.
متد الحاقی DoIgnoreAllNonExisting
1
2
3
4
5
6
7
8
9
10
11
12
13
|
public static IMappingExpression DoIgnoreAllNonExisting( this IMappingExpression expression) { var attr = expression.TypeMap.DestinationType.GetCustomAttribute<MapFromAttribute>(); if (attr?.IgnoreAllNonExistingProperty == false ) //instead of if(attr == null || attr.IgnoreAllNonExistingProperty == false) return expression; foreach (var property in expression.TypeMap.GetUnmappedPropertyNames()) { expression.ForMember(property, opt => opt.Ignore()); } return expression; } |
این متد برحسب پرچم تعیین شده در هنگام بکارگیری ویژگی MapFromAttribute رفتار میکند. به این صورت که اگر موقع تعریف، مقدار IgnoreAllNonExistingProperty را صحیح اعلام کنیم، تمام پروپرتیهای مقصد را که معادل اسمی در مبدأ نداشته باشند و همچنین هیچگونه تنظیمی جهت مشخص سازی تکلیف نگاشت آنها صورت نگرفته باشد، از گردونهی نگاشت Automapper خارج میکند.
توضیح تکمیلی: پس از تنظیم کلیهی نگاشتها در automapper جهت اطمینان از صحت تنظیمات، فراخوانی متد AutoMapper.Mapper.AssertConfigurationIsValid الزامی است. یکی از عواملی که باعث شکست این متد میشود، وجود پروپرتیهایی در نوع مقصد است، بطوریکه معادل اسمی در نوع مبدأ نداشته باشند و یا تنظیمی جهت مشخص سازی نگاشت آن انجام نشده باشد (پروپرتی که قابل نگاشت نباشد). در حقیقت این شکست بسیار مفید است. به این صورت که اگر این شکست صورت نگیرد در حین نگاشت مقادیر، باید از null یا مقدار default بدون اطلاع برنامه نویس برای مقداردهی پروپرتی استفاده کند و این یک حالت نامعلوم شیء است. اگر میخواهید این پروپرتیها مقدار پیشفرضی بگیرند و همچنین باعث شکست عملیات هم نشوند، باید بطور صریح این موضوع را اعلام کنید. این اعلام یا باید به همین روش صورت بگیرد یا باید از ویژگی IgnorMapAttribute استفاده شود. تنها تفاوت این دو، نحوهی اعمال تنظیم میباشد. IgnorMapAttribute باید روی تک تک پروپرتیهای مدنظر قرار گیرد، ولی در روش اول تنها کافیست که مقدار true تنظیم گردد. بهنظر استفاده از IgnoreMapAttribute باعث طولانی شدن کدها میشود؛ اما توصیه میشود که از همین شیوه استفاده کنید.
تا اینجا کدهای مورد نیاز نوشته شدند. در ادامه به ارائهی یک مثال برای نگاشت اشیاء در Automapper توسط Attributeها میپردازم.
مدل سادهی زیر را در نظر بگیرید:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
|
public class Student { public virtual int Id { set ; get ; } public virtual string Name { set ; get ; } public virtual string Family { set ; get ; } public virtual string Email { set ; get ; } public virtual DateTime RegisterDateTime { set ; get ; } public virtual ICollection<Book> Books { set ; get ; } } public class Book { public virtual int Id { set ; get ; } public virtual string Name { set ; get ; } public virtual DateTime BorrowDateTime { set ; get ; } public virtual DateTime ExpiredDateTime { set ; get ; } public virtual decimal Price { set ; get ; } [ForeignKey( "StudentIdFk" )] public virtual Student Student { set ; get ; } public virtual int StudentIdFk { set ; get ; } } |
با ویوومدل متناظر ذیل:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
|
[MapFrom( typeof (Student), ignoreAllNonExistingProperty: true , alsoCopyMetadata: true )] public class AdminStudentViewModel { // [IgnoreMap] public int Id { set ; get ; } [MapForMember( "Name" )] public string FirstName { set ; get ; } [MapForMember( "Family" )] public string LastName { set ; get ; } [IgnoreMap] public string Email { set ; get ; } [MapForMember( "RegisterDateTime" )] public string RegisterDateTimePersian { set ; get ; } [UseValueResolver( typeof (BookCountValueResolver))] public int BookCounts { set ; get ; } [UseValueResolver( typeof (BookPriceValueResolver))] public decimal TotalBookPrice { set ; get ; } }; |
در تنظیم ویژگی MapFromAttribute ابتدا نوع مبدأ (Student) را مشخص کردیم و بعد صراحتاً گفتیم که از نگاشت پروپرتیهای بلاتکلیف صرف نظر کند و همچنین پرچم انتقال Data Annotationهای EF به ویوومدل را هم برافراشتیم. توسط MapForMember پروپرتی FirstName را به پروپرتی Name در مبدأ تنظیم کردیم و LastName را به Family. همچنین Email را بصورت صریح از نگاشت شدن منع کردیم. پروپرتی BookCounts تعداد کتابها را محاسبه میکند و TotalBookPrice قیمت کلیهی کتابها را. برای این موارد از تأمین کنندهی داده (Value Resolver) استفاده کردیم. این تأمین کنندهها میتوانند اینچنین پیاده سازی شوند:
1
2
3
4
5
6
7
8
|
public class BookCountValueResolver : ValueResolver<Student, int > { protected override int ResolveCore(Student source) => source.Books.Count; }; public class BookPriceValueResolver : ValueResolver<Student, decimal > { protected override decimal ResolveCore(Student source) => source.Books.Sum(b => b.Price); }; |
نحوهی پیکربندی و مشاهدهی نتایج را در یک برنامهی تحت کنسول پیاده سازی کردم. متد Main آن میتواند اینچنین باشد:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
|
static void Main( string [] args) { var assemblyToLoad = Assembly.GetAssembly( typeof (AdminStudentViewModel)); //get assembly global::AttributesForAutomapper.Configuration.Initialize(assemblyToLoad); //init automaper IList<Student> lst; using (var context = new MySampleContext()) { lst = context.Students.Include(x => x.Books).ToList(); } foreach (var student in lst) { WriteLine( $ "[{student.Id}]*\n{student.Name} {student.Family}.\nmailto:{student.Email}.\nRegistered at'{student.RegisterDateTime}'" ); foreach (var book in student.Books) WriteLine($ "\tBook name:{book.Name}, Book price:{book.Price}" ); } var lstViewModel = AutoMapper.Mapper.Map<IList<Student>, IList<AdminStudentViewModel>>(lst); foreach (var adminStudentViewModel in lstViewModel) { WriteLine( $ "[{adminStudentViewModel.Id}]*\n\t{adminStudentViewModel.FirstName} {adminStudentViewModel.LastName}.\n\t" + $ "mailto:{adminStudentViewModel.Email}.\n\tRegistered at'{adminStudentViewModel.RegisterDateTimePersian}'\n\t" + $ "Book Counts: {adminStudentViewModel.BookCounts} with total price of {adminStudentViewModel.TotalBookPrice}" ); } WriteLine( "Press any key to exit..." ); ReadKey(); } |
ابتدا اسمبلی مربوط به ویوومدلها را مشخص میکنیم. سپس این اسمبلی را جهت تبدیل ویژگیها به نگاشتهای معتبر automapper به متد Initialize ارسال میکنیم. تنها بکار بردن همین دوسطر برای اعمال تنظیمها مورد نیاز میباشد. بعد از اجرای موفق متد Initialize، نگاشتهای اشیاء آماده هستند.
نمونهی خروجی:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
[1]* Morteza Raeisi. mailto:MrRaeisi@outlook.com. Registered at '23/08/1392 19:11:43' // I'm using Windows 10 with Persian calendar as default, On other OS or calendar settings, this value is different. Book name:AutoMapper Attr, Book price:1000.00 Book name:Second Book, Book price:2500.00 Book name:Hungry Book, Book price:2500.00 ... [1]* Morteza Raeisi. //MapForMemebers mailto:. // IgnoreMap Registered at '1392/08/23 19:11' // Convert using Book Counts: 3 with total price of 6000.00 // Value resolvers ... |