Cannot seem to moq EF CodeFirst 4.1.Help anyone?

后端 未结 2 991
礼貌的吻别
礼貌的吻别 2021-01-01 07:25

I have been given the task to evaluate codeFirst and possible to use for all our future projects. The evaluation is based on using codeFirst with an existing database.

相关标签:
2条回答
  • 2021-01-01 07:50

    When I started with repository and unit of work patterns I used the implementation similar to this (it is for ObjectContext API but converting it to DbContext API is simple). We used that implementation with MOQ and Unity without any problems. By the time implementations of repository and unit of work have evolve as well as the approach of injecting. Later on we found that whole this approach has serious pitfalls but that was alredy discussed in other questions I referenced here (I highly recommend you to go through these links).

    It is very surprising that you are evaluating the EFv4.1 with high emphasis on mocking and unit testing and in the same time you defined service method which is not unit-testable (with mocking) at all. The main problem of you service method is that you are not passing repository/context as dependency and because of that you can't mock it. The only way to test your service and don't use the real repository is using some very advanced approach = replacing mocking and MOQ with detouring (for example Moles framework).

    First what you must do is replacing your service code with:

    public class StudentService : IStudentService
    {
        private readonly IStudentRepository _studentRepository;
    
        public StudentService(IStudentRepository studentRepository)
        {
            _studentRepository = studentRepository;
        }
    
        public IEnumerable<Student> GetAll()
        {
             return _studentRepository.GetAll().ToList();
        }
    }
    

    Btw. this is absolutely useless code and example of silly layering which doesn't offer any useful functionality. Just wrapping the call to repository only shows that service is not needed at all as well as unit testing this method is not needed. The main point here is integration test for GetAll method.

    Anyway if you want to unit thest such method with MOQ you will do:

    [TestClass]
    public class StudentsServiveTest
    {
        private Mock<IRespository<Student>> _repo;
    
        [TestInitialize]
        public void Init()
        {
            _repo = new Mock<IRepository<Student>>();
            _repo.Setup(r => r.GetAll()).Returns(() => new Student[] 
                { 
                    new Student { StudentId = 1, Name = "A", Surname = "B" },
                    new Student { StudentId = 2, Name = "B", Surname = "C" }
                });
        }
    
        [TestMethod]
        public void ShouldReturnAllStudents()
        {
            var service = new StudentsService(_repo.Object);
            var data = service.GetAll();
            _repo.Verify(r => r.GetAll(), Times.Once());
    
            Assert.IsNotNull(data);
            Assert.AreEqual(2, data.Count);
        }
    }
    
    0 讨论(0)
  • 2021-01-01 07:53

    The issue from what I can see is that you are throwing away the mock object and newing up a new instance

    _studentRepository = new StudentRepository(ctx);
    

    Perhaps add a method on the interface to add the context object and reuse the same instance that was injected in the constructor.

    using (var ctx = new SchoolContext("SchoolDB"))
        {
            _studentRepository.Context = ctx;
            var students = _studentRepository.GetAll().ToList();
            return students;
        } 
    }
    
    0 讨论(0)
提交回复
热议问题