我正在构建一组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是一个对象,但我想知道是否有另一种解决方案可以利用并行性
答案 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);
}