Using the Repository Pattern with ASP.NET MVC and Entity Framework

原文:http://www.codeguru.com/csharp/.net/net_asp/mvc/using-the-repository-pattern-with-asp.net-mvc-and-entity-framework.htm

Introduction

Data driven web applications need to have a neat strategy for data access. One of the important aspects of this strategy is the separation between the physical database, queries and other data access logic from the rest of the application. Repository pattern is a popular way to achieve such an isolation. This article discusses the basics of Repository pattern in the context of Entity Framework and ASP.NET MVC. It also illustrates how a repository can be built around your entities.

Overview of the Repository Pattern?

Most of the business applications need to access data residing in one or the other data store. The simplest approach is to write all the data access code in the main application itself. Consider, for example, that you have an ASP.NET MVC controller named CustomerController. The Customer controller class has several action methods that ultimately perform typical CRUD (Create, Read, Update and Delete) operations on the underlying database. Let's further assume that you are using Entity Framework for database access. In this case your application would do something like this:

   
   
       
            IBM Worklight Compared to "Do-It-Yourself" Mobile Platforms       
Download Now       
 
 

 

As you can see, various actions of the Customer controller (not all are shown in the figure) directly instantiate the EF data context and fire queries to retrieve the data. They also INSERT, UPDATE and DELETE data using the data context and DbSet. The EF in turn talks with the underlying SQL Server database. Although the above application works as expected it suffers from the drawback that the database access code (creating data context, writing queries, manipulating data, persisting changes etc.) is embedded directly inside the action methods. This design can cause code duplication and the controller is susceptible to change even for a minute change in the data access logic. For example, if an application is modifying a customer from two controllers, each controller will repeat the same code. And any future modifications also need to be done at two places.

To tackle this shortcoming Repository pattern can be introduced. The following line summarizes the purpose of the Repository pattern:

Mediates between the domain and data mapping layers using a collection-like interface for accessing domain objects.

(Read more about repository pattern here.)

Thus a repository acts like a middleman between the rest of the application and the data access logic. A repository isolates all the data access code from rest of the application. In doing so you are benefited by having a simple point of change in case modifications are necessary. Additionally, testing your controllers becomes easy because the testing framework need not run against the actual database access code. An in-memory or local pseudo-database would do the trick. With a repository introduced, the above figure can be changed to:

With the changed design, the Customer controller won't talk with EF data context directly. Additionally, there won't be queries or any other database operations in the action methods. All these operations are wrapped by the Customer repository. The Customer repository in turn uses EF data context to get the job done. Notice that the Customer repository has methods such as SelectAll(), SelectByID(), Insert(), Update() and Delete(). The Customer controller uses these methods to get its job done. If you see the Customer repository - it is offering an in-memory collection like interface to its consumer (for example, many collection classes expose Add() and Remove() methods and allow you to query them).

Creating Model using Entity Framework

Now that you understand the basics of Repository pattern, let's create a sample application that illustrates what we've discussed so far. Create a new ASP.NET MVC web application based on the empty template. Right click on the Models folder and add an ADO.NET Entity Data Model for the Customers table of the Northwind database. The following figure shows how the Customer entity looks:

Various views of the Customer controller need the Customer entity for their display so it needs to be passed between repository, controller and the views. 

Creating Customer Repository

A repository typically does at least five operations - Selecting all records from a table, selecting a single record based on its primary key, Insert, Update and Delete. This list is, however, not rigid. You may have more or fewer methods in the repository. For the sake of our example let's decide that these five operations are needed from the Customer repository. To enforce that all the repository classes have these five methods we will define an interface - ICustomerRepository - that has these methods and then we will implement this interface in a class. Creating an interface will also help us during testing where we may be required to define an alternate in-memory repository for the sake of simplicity. The ICustomerRepository interface looks like this:

  1. public interface ICustomerRepository
  2. {
  3. IEnumerable<Customer> SelectAll();
  4. Customer SelectByID(string id);
  5. void Insert(Customer obj);
  6. void Update(Customer obj);
  7. void Delete(string id);
  8. void Save();
  9. }

The ICustomerRepository interface has five methods as listed below:

  • SelectAll() : This method is intended to return all the Customer entities as an enumerable collection (such as a generic List).
  • SelectByID() : This method accepts a string representing a customer ID (CustomerID is a character column in the database) and returns a single Customer entity matching that ID.
  • Insert(): This method accepts a Customer object and adds it to the Customers DbSet.
  • Update() : This method accepts a Customer object and marks it as a modified Customer in the DbSet.
  • Delete() : This method accepts a CustomerID and removes that Customer entity from the Customers DbSet.
  • Save() : This method saves the changes to Northwind database.

Next, add CustomerRepository class to the project and implement ICustomerRepository in it. The following code shows the completed CustomerRepository class.

  1. public class CustomerRepository:ICustomerRepository
  2. {
  3. private NorthwindEntities db = null;
  4.  
  5. public CustomerRepository()
  6. {
  7. this.db = new NorthwindEntities();
  8. }
  9.  
  10. public CustomerRepository(NorthwindEntities db)
  11. {
  12. this.db = db;
  13. }
  14.  
  15. public IEnumerable<Customer> SelectAll()
  16. {
  17. return db.Customers.ToList();
  18. }
  19.  
  20. public Customer SelectByID(string id)
  21. {
  22. return db.Customers.Find(id);
  23. }
  24.  
  25. public void Insert(Customer obj)
  26. {
  27. db.Customers.Add(obj);
  28. }
  29.  
  30. public void Update(Customer obj)
  31. {
  32. db.Entry(obj).State = EntityState.Modified;
  33. }
  34.  
  35. public void Delete(string id)
  36. {
  37. Customer existing = db.Customers.Find(id);
  38. db.Customers.Remove(existing);
  39. }
  40.  
  41. public void Save()
  42. {
  43. db.SaveChanges();
  44. }
  45. }

The CustomerRepository class implements all the five methods discussed above. Notice that it has two constructor definitions - one that takes no parameters and the one that accepts the data context instance. This second version will be useful when you wish to pass the context from outside (such as during testing or while using the Unit of Work pattern). All the method implementations of CustomerRepository are quite straightforward and hence we won't go into detailed discussion of these methods.

Using Customer Repository in a Controller

Now that you have built the Customer repository, let's use it in a controller. So, add a controller class inside the Controllers folder and name it CustomerController. The following code shows how CustomerController looks:

  1. public class CustomerController : Controller
  2. {
  3. private ICustomerRepository repository = null;
  4. public CustomerController()
  5. {
  6. this.repository = new CustomerRepository();
  7. }
  8. public CustomerController(ICustomerRepository repository)
  9. {
  10. this.repository = repository;
  11. }
  12.  
  13.  
  14. public ActionResult Index()
  15. {
  16. List<Customer> model = (List<Customer>)repository.SelectAll();
  17. return View(model);
  18. }
  19.  
  20. public ActionResult New()
  21. {
  22. return View();
  23. }
  24.  
  25. public ActionResult Insert(Customer obj)
  26. {
  27. repository.Insert(obj);
  28. repository.Save();
  29. return View();
  30. }
  31.  
  32. public ActionResult Edit(string id)
  33. {
  34. Customer existing = repository.SelectByID(id);
  35. return View(existing);
  36. }
  37.  
  38. public ActionResult Update(Customer obj)
  39. {
  40. repository.Update(obj);
  41. repository.Save();
  42. return View();
  43. }
  44.  
  45. public ActionResult ConfirmDelete(string id)
  46. {
  47. Customer existing = repository.SelectByID(id);
  48. return View(existing);
  49. }
  50.  
  51. public ActionResult Delete(string id)
  52. {
  53. repository.Delete(id);
  54. repository.Save();
  55. return View();
  56. }
  57.  
  58. }

The Customer controller has two versions of the constructor and seven action methods. Notice that there is a private variable of type ICustomerRepository at the class level. The parameter less constructor sets this variable to an instance of CustomerRepository. The other version of the constructor accepts an implementation of ICustomerRepository from the external world and sets it to the private variable. This second version is useful during testing where you will supply a mock implementation of Customer repository from the test project.

The seven methods defined by the Customer controller are as follows:

  • Index() : Displays Index view and passes a List of Customer entities as its model.
  • New() : Displays New view.
  • Insert() : New view submits data to this method. It receives the data as a Customer instance and then inserts a customer using the repository.
  • Edit() : Displays Edit view. It accepts a CustomerID as an id route parameter and populates the Edit view with the data of the existing Customer.
  • Update() : Edit view submits data to this method. It receives the data as a Customer instance and then updates a customer using the repository.
  • ConfirmDelete() : Displays ConfirmDelete view.
  • Delete() : ConfirmDelete view submits to this action method. The action then deletes the Customer using the repository.

We won't go into the details of the four views mentioned above (Index, New, Edit and ConfirmDelete). They are quite simple and you can add them on your own for the sake of testing.

You just created and successfully used Repository pattern! As you can see from the controller code, nowhere did you use data context or EF operations. You always called some or the other method of the CustomerRepository to get the job done. Thus all your data access code is now separated into the repository.

Testing a Controller

Earlier we mentioned that repository pattern also helps during the testing phase. Let's see how by creating a simple test. Add a new Test project to the same solution and refer the MVC project into it. Inside the test project we will define another repository that works on some in-memory collection instead of the actual database. To create the repository add a class to the test project and write the following code to it:

  1. class TestCustomerRepository:ICustomerRepository
  2. {
  3. private List<Customer> data = new List<Customer>();
  4.  
  5. public IEnumerable<Customer> SelectAll()
  6. {
  7. return data;
  8. }
  9.  
  10. public Customer SelectByID(string id)
  11. {
  12. return data.Find(m => m.CustomerID == id);
  13. }
  14.  
  15. public void Insert(Customer obj)
  16. {
  17. data.Add(obj);
  18. }
  19.  
  20. public void Update(Customer obj)
  21. {
  22. Customer existing = data.Find(m => m.CustomerID == obj.CustomerID);
  23. existing = obj;
  24. }
  25.  
  26. public void Delete(string id)
  27. {
  28. Customer existing = data.Find(m => m.CustomerID == id);
  29. data.Remove(existing);
  30. }
  31.  
  32. public void Save()
  33. {
  34. //nothing here
  35. }
  36. }

As you can see TestCustomerRepository class implements the same ICustomerRepository interface defined in the MVC project. It then implements all five methods against an in-memory List of Customer entities. Although not shown in the above code, you could have pre-populated the List with some mock data.

Once TestCustomerRepository is created you can instantiate it in a test method and pass it to the CustomerController like this:

  1. [TestMethod]
  2. public void TestMethod1()
  3. {
  4. TestCustomerRepository repository = new TestCustomerRepository();
  5. CustomerController controller = new CustomerController(repository);
  6. var result = (ViewResult)controller.Index();
  7. List<Customer> data = (List<Customer>)result.ViewData.Model;
  8. Assert.IsFalse(data.Count <= 0);
  9. }

Recollect that CustomerController has an overloaded version of the constructor that takes any implementation of ICustomerRepository from the external world. The above code creates an instance of TestCustomerRepository and passes it to the CustomerController. It then invokes the Index() action method of the controller. The model of the Index view is obtained using the ViewData.Model property. The Assert checks whether there are any items in the model collection using IsFalse() method. In this case since no data is added to the generic List of Customer, the Count will be 0 and hence the condition will evaluate to true causing the assertion to fail.

Making the Repository Generic

Although the above example works great, it has a drawback. It expects you to have a separate repository for each entity in the application. For example, CustomerRepository for Customer entity, EmployeeRepository for Employee entity and so on. This can be too much work, especially if all the repositories are doing the same kind of operations (typical CRUD as in our example). Wouldn't it be nice to create a generic repository that can be used with any entity? Let's attempt to do just that.

Add the following interface to the ASP.NET MVC project:

  1. public interface IGenericRepository<T> where T:class
  2. {
  3. IEnumerable<T> SelectAll();
  4. T SelectByID(object id);
  5. void Insert(T obj);
  6. void Update(T obj);
  7. void Delete(object id);
  8. void Save();
  9. }

The IGenericRepository interface is a generic interface that defines the same set of methods as before. Notice that this time instead of Customer entity it uses T everywhere. Also notice that SelectByID() and Delete() methods now accept object parameter instead of string. This is necessary because different tables may have different types of primary keys (Customers table has a string primary key whereas Employees table has an integer primary key).

Now, add a class to the ASP.NET MVC project that implements IGenericRepository interface. This class is shown below:

  1. public class GenericRepository<T>:IGenericRepository<T> where T : class
  2. {
  3. private NorthwindEntities db = null;
  4. private DbSet<T> table = null;
  5.  
  6. public GenericRepository()
  7. {
  8. this.db = new NorthwindEntities();
  9. table = db.Set<T>();
  10. }
  11.  
  12. public GenericRepository(NorthwindEntities db)
  13. {
  14. this.db = db;
  15. table = db.Set<T>();
  16. }
  17.  
  18. public IEnumerable<T> SelectAll()
  19. {
  20. return table.ToList();
  21. }
  22.  
  23. public T SelectByID(object id)
  24. {
  25. return table.Find(id);
  26. }
  27.  
  28. public void Insert(T obj)
  29. {
  30. table.Add(obj);
  31. }
  32.  
  33. public void Update(T obj)
  34. {
  35. table.Attach(obj);
  36. db.Entry(obj).State = EntityState.Modified;
  37. }
  38.  
  39. public void Delete(object id)
  40. {
  41. T existing = table.Find(id);
  42. table.Remove(existing);
  43. }
  44.  
  45. public void Save()
  46. {
  47. db.SaveChanges();
  48. }
  49. }

The GenericRepository is a generic class and implements IGenericRepository. Notice that since this class uses generic type T you can't access a DbSet as a property of data context. That's why a generic DbSet variable is declared at the top that points to an appropriate DbSet based on the type of T.

Once GenericRepository is ready you can use it in the Customer controller like this:

  1. public class CustomerController : Controller
  2. {
  3. private IGenericRepository<Customer> repository = null;
  4.  
  5. public CustomerController()
  6. {
  7. this.repository = new GenericRepository<Customer>();
  8. }
  9.  
  10. public CustomerController(IGenericRepository<Customer> repository)
  11. {
  12. this.repository = repository;
  13. }
  14. ...
  15. ...
  16. }

As shown above, the IGenericRepository variable is declared with Customer as its type. The constrictor then assigns an instanced of GenericRepository() or some other implementation of IGenericRepository to this variable. 

Testing a Controller using Generic Repository

Just like you tested the CustomerController by creating a mock implementation of ICustomerRepository, you can also test it by creating a mock implementation of IGenericRepository. The following code shows one such implementation:

  1. class TestGenericRepository<T>:IGenericRepository<T> where T:class
  2. {
  3. private List<T> data = new List<T>();
  4.  
  5. public IEnumerable<T> SelectAll()
  6. {
  7. return data;
  8. }
  9.  
  10. public T SelectByID(object id)
  11. {
  12. return data.FirstOrDefault();
  13. }
  14.  
  15. public void Insert(T obj)
  16. {
  17. data.Add(obj);
  18. }
  19.  
  20. public void Update(T obj)
  21. {
  22. T existing = data.FirstOrDefault();
  23. existing = obj;
  24. }
  25.  
  26. public void Delete(object id)
  27. {
  28. data.RemoveAt(0);
  29. }
  30.  
  31. public void Save()
  32. {
  33. //nothing here
  34. }
  35. }

The TestGenericRepository class creates an implementation of IGenericRepository that works against an in-memory List of entities. Notice that just for the sake of testing, SelectByID(), Update() and Delete() methods return the first item from the List.

Now you can write another test method that uses TestGenericRepository repository.

  1. [TestMethod]
  2. public void TestMethod2()
  3. {
  4. TestGenericRepository<Customer> repository = new TestGenericRepository<Customer>();
  5. CustomerController controller = new CustomerController(repository);
  6. var result = (ViewResult)controller.Index();
  7. List<Customer> data = (List<Customer>)result.ViewData.Model;
  8. Assert.IsFalse(data.Count <= 0);
  9. }

As you can see this time TestGenericRepository is instantiated for the Customer entity and passed to the CustomerController. The remaining part of the test is identical to the earlier test method you wrote.

Summary

The Repository pattern allows you to separate data access code from the rest of the system. This isolation promotes code reuse, minimizes future modifications to code and also simplifies testing of the controller classes. This article gave you an understanding of the Repository pattern in the context of ASP.NET MVC and Entity Framework. You created entity specific repositories as well as generic repositories. You also used them in a test project.

    

Related Articles

  •             Preventing Cross Site Scripting Attacks in ASP.NET MVC 4            
  •             Test Driven Development in Asp.Net MVC Architecture            
  •             Introduction to ASP.NET vNext            
  •             Overview of OWIN and Katana       

你可能感兴趣的:(Using the Repository Pattern with ASP.NET MVC and Entity Framework)