单元测试我的自定义会员提供商

时间:2013-08-22 18:00:46

标签: c# unit-testing ninject moq membership-provider

我有一个自定义成员资格提供程序,它连接到这样的用户存储库:

public class MyMembershipProvider : MembershipProvider {
   [Inject]
   public IUserRepository UserRepository { get; set; }
   ...
   //Required membership methods
}

我正在使用ninject作为我的DI。现在我想测试提供程序,并注入一个模拟用户存储库以允许我这样做。如下所示:

        ... 
        IList<User> users = new List<User> {
            new User { Email="matt@test.com", 
                       UserName="matt@test.com", 
                       Password="test"
            }
        };
        var mock = new Mock<IUserRepository>();

        mock.Setup(mr => mr.FindByUsername(
            It.IsAny<string>())).Returns((string s) => users.Where(
            x => x.UserName.Equals(s,StringComparison.OrdinalIgnoreCase)).Single());
        ...

这里是我不确定如何继续的地方,如何将我的模拟存储库注入我的提供程序,以便当调用提供程序的单元测试使用此模拟存储库时?

我在这里问正确的问题吗?

编辑 - 我的最终解决方案

为了我的价值,我从使用mock转移到使用InMemory存储库来维护状态,以便提供者正确地测试某些功能。现在我只用它来测试像我的提供者这样的东西。我最终得到了:

通用InMemoryRepository:

class InMemoryRepository<TEntity> : IRepository<TEntity> where TEntity : class {
    private int _incrementer = 0;
    public Dictionary<int, TEntity> List = new Dictionary<int, TEntity>();
    public string IDPropertyName {get; set;}

    public void Create(TEntity entity) {
        _incrementer++;
        entity.GetType().GetProperties().First(p => p.Name == IDPropertyName).SetValue(entity, _incrementer, null);
        List.Add(_incrementer,entity);
    }

    public TEntity GetById(int id) {
        return List[id];
    }

    public void Delete(TEntity entity) {
        var key = (int)entity.GetType().GetProperties().First(p => p.Name == IDPropertyName).GetValue(entity, null);
        List.Remove(key);
    }

    public void Update(TEntity entity) {
        var key = (int)entity.GetType().GetProperties().First(p => p.Name == IDPropertyName).GetValue(entity, null);
        List[key] = entity;
    }
}

然后我的实际用户存储库 - 我没有通用ID字段,这就是我使用IDPropertyName变量的原因:

class InMemoryUserRepository : InMemoryRepository<User>, IUserRepository {
    public InMemoryUserRepository() {
        this.IDPropertyName = "UserID";
    }

    public IQueryable<User> Users {
        get { return List.Select(x => x.Value).AsQueryable(); }
    }

    public User FindByUsername(string username) {
        int key = List.SingleOrDefault(x=>x.Value.UserName.Equals(username, StringComparison.OrdinalIgnoreCase)).Key;
        return List[key];
    }
}

我的会员资格测试基类:

[TestClass]
public class MyMembershipProviderTestsBaseClass : IntegrationTestsBase {
    protected MyMembershipProvider _provider;
    protected NameValueCollection _config;
    protected MembershipCreateStatus _status = new MembershipCreateStatus();

    [TestInitialize]
    public override void Initialize() {
        base.Initialize();

        // setup the membership provider
        _provider = new MyMembershipProvider();

        MembershipSection section = (MembershipSection) ConfigurationManager.GetSection("system.web/membership");
        NameValueCollection collection = section.Providers["MyMembershipProvider"].Parameters;
        _provider.Initialize(null, collection);
        _status = new MembershipCreateStatus();
    }

    [TestCleanup]
    public override void TestCleanup() {
        base.TestCleanup();
    }
}

然后我的测试:

    [TestMethod]
    public void Membership_CreateUser() {
        _provider.UserRepository = new InMemoryUserRepository();
        _provider.CreateUser(_email, out _status);
        Assert.AreEqual(MembershipCreateStatus.Success, _status);
    }

这个答案提供了灵感:https://stackoverflow.com/a/13073558/1803682

2 个答案:

答案 0 :(得分:3)

由于您已将存储库公开为属性,因此只需在测试类中创建提供程序的实例,并将该属性设置为模拟,如下所示:

public void Test()
{
  MyMembershipProvider provider = new MyMembershipProvder();
  provider.UserRepository = mock.Object;

  // Do test stuff here

  // Verify mock conditions
}

据推测,您的存储库实现正在使用UserRepository属性,因此在您测试它时,此代码将使用模拟的依赖项。

答案 1 :(得分:1)

您可以为Ninject设置测试模块,并使用单元测试项目中的测试模块创建Ninject内核。

这是控制容器的反转。您有一组绑定配置为作为网站运行,另一组绑定用于在测试中运行,另一组用于使用不同的后端(或其他)运行。

我这样做是为了让生产和测试代码以相同的方式初始化(​​通过Ninject),所有改变都是Ninject的配置。

或者做@chris house建议的事情。那也行。