Jednostka pracy i ogólne repozytorium z ramami Entity Framework 5

Korzystam z ASP.NET MVC 4 z Entity Framework 5. Mam klasy modeli i mapy jednostek do mapowania istniejących tabel do tych klas modeli. Wszystko to jest dobrze skonfigurowane i działa świetnie.

Teraz chcę to wyśmiać. Stworzyłem Unit Of Work, który pobiera DataContext i używa Repozytorium rodzajowego. Po tym zbudowałem usługi, aby móc pobierać dane z wielu repozytoriów jednocześnie i potrzebowałem tylko jednej instancji DataContext. To także działa świetnie.

Teraz problem: Chcę przetestować usługi za pomocą fałszywych danych. Gdy tworzę instancję Unit Of Work, chcę móc wstawić DataContext, który jest wyszydzany zamiast rzeczywistego DataContext.

Próbowałem utworzyć interfejs IContext i pozwolić, aby prawdziwy i wyszydzony DataContext zaimplementował go, ale napotkał problemy z DbSet. Próbowałem użyć IDbSet i utworzyć FakeDbSet, ale bez powodzenia. Czytałem też w internecie, że wyśmiewanie kontekstu za pomocą IDbSet i używanie FakeDbSet to złe podejście.

Czy masz pojęcie, jaki byłby najlepszy sposób osiągnięcia tego celu? To, co mam teraz, to zachowanie, które chciałbym zachować, ale naprawdę chciałbym móc wyszyfrować dane z klas Model w DataContext.

Zdaję sobie sprawę, że Entity Framework zawiera już zachowanie Unit Of Work i nie trzeba dodawać dodatkowego zachowania. Ale chciałem zawinąć to do innej klasy, która śledzi wszystkie repozytoria (zwana klasą UnitOfWork).

Edytować: Napisałem dwa artykuły wyjaśniające moje rozwiązanie za pomocą LINQ i Entity Framework.

http://gaui.is/how-to-mock-the-datacontext-linq/

http://gaui.is/how-to-mock-the-datacontext-entity-framework/

Oto mój kod:

IRepository.cs

public interface IRepository<T> where T : class
{
    void Add(T entity);
    void Delete(T entity);
    void Update(T entity);
    T GetById(long Id);
    IEnumerable<T> All();
    IEnumerable<T> Find(Expression<Func<T, bool>> predicate);
}

IUnitOfWork.cs

public interface IUnitOfWork : IDisposable
{
    IRepository<TEntity> GetRepository<TEntity>() where TEntity : class;
    void Save();
}

Repository.cs

public class Repository<T> : IRepository<T> where T : class
{
    private readonly IDbContext _context;
    private readonly IDbSet<T> _dbset;

    public Repository(IDbContext context)
    {
        _context = context;
        _dbset = context.Set<T>();
    }

    public virtual void Add(T entity)
    {
        _dbset.Add(entity);
    }

    public virtual void Delete(T entity)
    {
        var entry = _context.Entry(entity);
        entry.State = System.Data.EntityState.Deleted;
    }

    public virtual void Update(T entity)
    {
        var entry = _context.Entry(entity);
        _dbset.Attach(entity);
        entry.State = System.Data.EntityState.Modified;
    }

    public virtual T GetById(long id)
    {
        return _dbset.Find(id);
    }

    public virtual IEnumerable<T> All()
    {
        return _dbset;
    }

    public IEnumerable<T> Find(Expression<Func<T, bool>> predicate)
    {
        return _dbset.Where(predicate);
    }
}

UnitOfWork.cs

public class UnitOfWork<TContext> : IUnitOfWork where TContext : IDbContext, new()
{
    private readonly IDbContext _ctx;
    private Dictionary<Type, object> _repositories;
    private bool _disposed;

    public UnitOfWork()
    {
        _ctx = new TContext();
        _repositories = new Dictionary<Type, object>();
        _disposed = false;
    }

    public IRepository<TEntity> GetRepository<TEntity>() where TEntity : class
    {
        if (_repositories.Keys.Contains(typeof(TEntity)))
            return _repositories[typeof(TEntity)] as IRepository<TEntity>;

        var repository = new Repository<TEntity>(_ctx);
        _repositories.Add(typeof(TEntity), repository);
        return repository;
    }

    public void Save()
    {
        _ctx.SaveChanges();
    }

    public void Dispose()
    {
        Dispose(true);
        GC.SuppressFinalize(this);
    }

    protected virtual void Dispose(bool disposing)
    {
        if (!this._disposed)
        {
            if (disposing)
            {
                _ctx.Dispose();
            }

            this._disposed = true;
        }
    }
}

ExampleService.cs

public class ExampleService
{
    private IRepository<Example> m_repo;

    public ExampleService(IUnitOfWork uow)
    {
        m_repo = uow.GetRepository<Example>();
    }

    public void Add(Example Example)
    {
        m_repo.Add(Example);
    }

    public IEnumerable<Example> getAll()
    {
        return m_repo.All();
    }
}

ExampleController.cs

public IEnumerable<Example> GetAll()
{
    // Create Unit Of Work object
    IUnitOfWork uow = new UnitOfWork<AppDataContext>();

    // Create Service with Unit Of Work attached to the DataContext
    ExampleService service = new ExampleService(uow);

    return service.getAll();
}

questionAnswers(2)

yourAnswerToTheQuestion