Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Repository pattern and mapping between domain models and Entity Framework

My repositories deal with and provide persistence for a rich domain model. I do not want to expose the anemic, Entity Framework data entity to my business layers, so I need some way of mapping between them.

In most cases, constructing a domain model instance from a data entity requires the use of parameterised constructors and methods (since it is rich). It is not as simple as a property/field match. AutoMapper could be used for the opposite situation (mapping to data entities) but not when creating domain models.

Below is the core of my repository pattern.

The EntityFrameworkRepository class works with two generic types:

  • TDomainModel: The rich domain model
  • TEntityModel: The Entity Framework data entity

Two abstract methods are defined:

  • ToDataEntity(TDomainModel): To convert to data entities (for Add() and Update() methods)
  • ToDomainModel(TEntityModel): To construct domain models (for the Find() method).

Concrete implementations of these methods would define the mapping required for the repository in question.

public interface IRepository<T> where T : DomainModel {     T Find(int id);     void Add(T item);     void Update(T item); }  public abstract class EntityFrameworkRepository<TDomainModel, TEntityModel> : IRepository<TDomainModel>     where TDomainModel : DomainModel     where TEntityModel : EntityModel {     public EntityFrameworkRepository(IUnitOfWork unitOfWork)     {         // ...     }      public virtual TDomainModel Find(int id)     {         var entity = context.Set<TEntityModel>().Find(id);          return ToDomainModel(entity);     }      public virtual void Add(TDomainModel item)     {         context.Set<TEntityModel>().Add(ToDataEntity(item));     }      public virtual void Update(TDomainModel item)     {         var entity = ToDataEntity(item);          DbEntityEntry dbEntityEntry = context.Entry<TEntityModel>(entity);          if (dbEntityEntry.State == EntityState.Detached)         {             context.Set<TEntityModel>().Attach(entity);              dbEntityEntry.State = EntityState.Modified;         }     }      protected abstract TEntityModel ToDataEntity(TDomainModel domainModel);     protected abstract TDomainModel ToDomainModel(TEntityModel dataEntity); } 

Here is a basic example of a repository implementation:

public interface ICompanyRepository : IRepository<Company> {     // Any specific methods could be included here }  public class CompanyRepository : EntityFrameworkRepository<Company, CompanyTableEntity>, ICompanyRepository {     protected CompanyTableEntity ToDataEntity(Company domainModel)     {         return new CompanyTable()         {             Name = domainModel.Name,             City = domainModel.City             IsActive = domainModel.IsActive         };     }      protected Company ToDomainModel(CompanyTableEntity dataEntity)      {         return new Company(dataEntity.Name, dataEntity.IsActive)         {             City = dataEntity.City         }     } } 

Problem:

A Company might be composed of many Departments. If I want to eagerly load these from the CompanyRepository when fetching a Company then where would I define the mapping between a Department and a DepartmentDataEntity?

I could provide more mapping methods in the CompanyRepository, but this will soon get messy. There would soon be duplicated mapping methods across the system.

What is a better approach to the above problem?

like image 890
Dave New Avatar asked Jan 06 '14 07:01

Dave New


People also ask

Does Entity Framework use repository pattern?

No, the repository/unit-of-work pattern (shortened to Rep/UoW) isn't useful with EF Core. EF Core already implements a Rep/UoW pattern, so layering another Rep/UoW pattern on top of EF Core isn't helpful.

What is the benefit of repository pattern in Entity Framework?

Benefits of Repository PatternIt centralizes data logic or business logic and service logic. It gives a substitution point for the unit tests. Provides a flexible architecture. If you want to modify the data access logic or business access logic, you don't need to change the repository logic.

What is generic repository pattern in Entity Framework?

The generic repository pattern implements in a separate class library project. It uses the "Code First" development approach and creates a database from a model, using migration. This article demonstrates a sample Application, which has one too many relationships in ASP.NET Core with Entity Framework Core.

What are approaches to domain modeling in Entity Framework?

In this article I will be describing the three approaches to using Entity Framework (EF)- Model First, Database First and Code First. The Entity Framework provides three separate approaches to work with data models and each one has its own pros and cons.


1 Answers

My repositories deal with and provide persistence for a rich domain model. I do not want to expose the anemic, Entity Framework data entity to my business layers, so I need some way of mapping between them.

If you you use Entity Framework, it can map Rich Domain Model itself.

I've answered the similar question "Advice on mapping of entities to domain objects" recently.

I've been using NHibernate and know that in Entity Framework you can also specify mapping rules from DB tables to your POCO objects. It is an extra effort to develop another abstraction layer over Entity Framework entities. Let the ORM be responsible for all of the mappings, state tracking, unit of work and identity map implementation, etc. Modern ORMs know how to handle all these issues.

AutoMapper could be used for the opposite situation (mapping to data entities) but not when creating domain models.

You are completely right.

Automapper is useful when one entity can be mapped into another without additional dependencies (e.g. Repositories, Services, ...).

... where would I define the mapping between a Department and a DepartmentDataEntity?

I would put it into DepartmentRepository and add method IList<Department> FindByCompany(int companyId) in order to retreive company's departments.

I could provide more mapping methods in the CompanyRepository, but this will soon get messy. There would soon be duplicated mapping methods across the system.

What is a better approach to the above problem?

If it is needed to get list of Departments for another entity, a new method should be added to DepartmentRepository and simply used where it is needed.

like image 115
Ilya Palkin Avatar answered Sep 17 '22 21:09

Ilya Palkin