4

使用 asp.net 标识RTW 版本

我需要在事务中执行几个操作,包括UserMananger函数调用和我的其他操作DbContext(例如:创建新用户,将其添加到组并执行一些业务逻辑操作)。

我该怎么做?

我的想法随之而来。

交易范围

using (var scope = new TransactionScope(TransactionScopeOption.Required))
{
    // Do what I need
    if (everythingIsOk) scope.Complete();
}

问题是:UserManager函数都是异步的,TransactionScope并不是为使用 async/await 而设计的。它似乎在 .Net Framework 4.5.1 中得到解决。但是我使用 Azure 网站来托管我的项目构建,所以我还不能以 4.5.1 为目标。

数据库事务

public class SomeController : Controller
{
    private MyDbContext DbContext { get; set; }
    private UserManager<User> UserManager { get; set; }

    public AccountController()
    {
        DbContext = new MyDbContext()
        var userStore = new UserStore<IdentityUser>(DbContext);
        UserManager = new UserManager<IdentityUser>(userStore);
    }

    public async ActionResult SomeAction()
    {
        // UserManager uses the same db context, so they can share db transaction
        using (var tran = DbContext.Database.BeginTransaction())
        {
            try
            {
                // Do what I need
                if (everythingIsOk)
                    tran.Commit();
                else
                {
                    tran.Rollback();
                }
            }
            catch (Exception)
            {
                tran.Rollback();
            }
        }
    }
}

这似乎可行,但我该如何对其进行单元测试?

UserManager<>构造函数接受IUserStore<>,所以我可以轻松地存根它。

UserStore<>构造函数接受DbContext,不知道如何存根。

4

1 回答 1

0

您可以实现自己的测试用户存储,可以为您的单元测试存根。

如果您想在测试中使用实际的 EF UserStore,这也可以,但默认情况下它将使用 DefaultConnection 字符串创建数据库。如果您想确保每个测试都有一个干净的数据库,您可以指定一个 DatabaseInitializer 以始终在测试中删除/重新创建表。

于 2013-10-15T17:36:31.310 回答