RESTful API中的NHibernate会话管理与任务/线程

时间:2015-10-19 14:58:27

标签: c# asp.net multithreading api nhibernate

我正在构建一组API。其中一个是身份验证API,它返回JWT令牌。我正在尝试使用ActionFiltersAttribute实现每个操作的会话方法。我的控制器使用以下属性进行修饰:

public class NHibernateSessionAttribute : ActionFilterAttribute
    {
        public override void OnActionExecuting(HttpActionContext actionContext)
        {
            var session = NHibernateSessionManager.SessionFactory.OpenSession();
            session.BeginTransaction();            
            CurrentSessionContext.Bind(session);
        }

        public override void OnActionExecuted(HttpActionExecutedContext actionExecutedContext)
        {
            var session = CurrentSessionContext.Unbind(NHibernateSessionManager.SessionFactory)            
            if (session != null)
            {
                if (session.Transaction.IsActive)
                {
                    try
                    {
                        session.Transaction.Commit();
                    }
                    catch
                    {
                        session.Transaction.Rollback();
                    }
                }

                session.Close();
            }
        }
    }

问题出在哪里?使用NHibernate来管理用户而不是实体框架我已经实现了所有需要的ASP.NET身份接口,并且它们都返回Task<T>。例如,执行以下操作:

AccountController.cs

public async Task<IHttpActionResult> ChangePassword(ChangePasswordBindingModel model)
        {
            if (!ModelState.IsValid)
            {
                return BadRequest(ModelState);
            }

            IdentityResult result = await UserManager.ChangePasswordAsync(User.Identity.GetUserId(), model.OldPassword,
                model.NewPassword);

            if (!result.Succeeded)
            {
                return GetErrorResult(result);
            }

            return Ok();
        }

内部有ChangePasswordAsync个调用新任务中包含代码的几种方法,其中SessionFactory.GetCurrentSession()会导致NullException。据我所知,因为那是另一个线程和上下文。 在代码中,第一次保存执行没有失败,第二次没有。重复的代码只是为了说明情况。

UserStore.cs

public System.Threading.Tasks.Task UpdateAsync(UserModel user)
    {
        if (user == null)
        {
            throw new ArgumentNullException("user");
        }
        //Here the Session is found
        DataProviderI<UserModel, int> prov = new DataProviderImplGeneric<UserModel, int>();
        prov.Save(user);


        return Task.Factory.StartNew(() =>
        {
            //Here the Session is NOT found
            DataProviderI<UserModel, int> prov2 = new DataProviderImplGeneric<UserModel, int>();
            prov.Save(user);
        });
    }

在所有ISession期间,处理此问题并获得相同Action的最佳方法是什么?

据我所知,NHibernate不支持异步调用,我可以在void case上返回类型为Task.FromResult(0)的方法重构方法,或者Task.FromResult<T>(T),其中T是一个对象,但我想知道是否有另一种解决方案可以利用并行性

1 个答案:

答案 0 :(得分:1)

似乎你正在处理的问题源于HttpContext在任务内部时为null,因此无法访问存储在上下文变量中的NHibernate会话。

您可以通过在调用任务之前获取ISession来解决此问题。

将构造函数添加到DataProviderImplGeneric中,以便手动传入一个。

DataProviderImplGeneric(ISession session) {
      this.session = session;
}

只需在从任务内部调用会话之前检索会话。

var session = GetCurrentNHibernateSession();

return Task.Factory.StartNew(() => {
    var dataProvider = DataProviderImplGeneric<UserModel, int>(session);
    return dataProvider.Save(user);
}