Примечание: Теперь, когда я это напечатал, я должен извиниться за очень длинный вопрос, однако, я думаю, что весь код и информация, представленная здесь, каким-то образом относится к делу.
Хорошо, я получаю странные ошибки "Session Is Closed" в случайных точках в моем приложении веб-форм ASP.NET. Однако сегодня это, наконец, происходит в одном и том же месте снова и снова. Я почти уверен, что в моем коде нет ничего, что могло бы уничтожить или закрыть сеанс, так как используемые биты кода хорошо отделены от всего остального кода, как вы увидите ниже.
Я также использую ninject в качестве МОК, что может / не может быть важным.
Хорошо, итак, сначала мои SessionFactoryProvider
и SessionProvider
классы:
SessionFactoryProvider
public class SessionFactoryProvider : IDisposable
{
ISessionFactory sessionFactory;
public ISessionFactory GetSessionFactory()
{
if (sessionFactory == null)
sessionFactory =
Fluently.Configure()
.Database(
MsSqlConfiguration.MsSql2005.ConnectionString(p =>
p.FromConnectionStringWithKey("QoiSqlConnection")))
.Mappings(m =>
m.FluentMappings.AddFromAssemblyOf<JobMapping>())
.BuildSessionFactory();
return sessionFactory;
}
public void Dispose()
{
if (sessionFactory != null)
sessionFactory.Dispose();
}
}
SessionProvider
public class SessionProvider : IDisposable
{
ISessionFactory sessionFactory;
ISession session;
public SessionProvider(SessionFactoryProvider sessionFactoryProvider)
{
this.sessionFactory = sessionFactoryProvider.GetSessionFactory();
}
public ISession GetCurrentSession()
{
if (session == null)
session = sessionFactory.OpenSession();
return session;
}
public void Dispose()
{
if (session != null)
{
session.Dispose();
}
}
}
Эти два класса связаны с Ninject следующим образом:
NHibernateModule
public class NHibernateModule : StandardModule
{
public override void Load()
{
Bind<SessionFactoryProvider>().ToSelf().Using<SingletonBehavior>();
Bind<SessionProvider>().ToSelf().Using<OnePerRequestBehavior>();
}
}
и, насколько я могу судить, работает, как ожидалось.
Теперь мой BaseDao<T>
класс:
BaseDao
public class BaseDao<T> : IDao<T> where T : EntityBase
{
private SessionProvider sessionManager;
protected ISession session { get { return sessionManager.GetCurrentSession(); } }
public BaseDao(SessionProvider sessionManager)
{
this.sessionManager = sessionManager;
}
public T GetBy(int id)
{
return session.Get<T>(id);
}
public void Save(T item)
{
using (var transaction = session.BeginTransaction())
{
session.SaveOrUpdate(item);
transaction.Commit();
}
}
public void Delete(T item)
{
using (var transaction = session.BeginTransaction())
{
session.Delete(item);
transaction.Commit();
}
}
public IList<T> GetAll()
{
return session.CreateCriteria<T>().List<T>();
}
public IQueryable<T> Query()
{
return session.Linq<T>();
}
}
Который связан в Ninject так:
DaoModule
public class DaoModule : StandardModule
{
public override void Load()
{
Bind(typeof(IDao<>)).To(typeof(BaseDao<>))
.Using<OnePerRequestBehavior>();
}
}
Теперь веб-запрос, который вызывает это, - это когда я сохраняю объект, он не выполнялся до тех пор, пока я сегодня не сделал некоторые изменения в модели, однако изменения в моей модели в любом случае не изменили код доступа к данным. Хотя он изменил несколько отображений NHibernate (я могу опубликовать их, если кому-то интересно)
Насколько я могу судить, вызывается BaseDao<SomeClass>.Get
, затем вызывается BaseDao<SomeOtherClass>.Get
, затем вызывается BaseDao<TypeImTryingToSave>.Save
.
это третий вызов на линии в Save()
using (var transaction = session.BeginTransaction())
, который завершается с ошибкой "Сессия закрыта!" точнее исключение:
Session is closed!
Object name: 'ISession'.
Description: An unhandled exception occurred during the execution of the current web request. Please review the stack trace for more information about the error and where it originated in the code.
Exception Details: System.ObjectDisposedException: Session is closed!
Object name: 'ISession'.
И действительно, отладчик показывает третий раз, когда сеанс запрашивается у SessionProvider
, он действительно закрыт и не подключен.
Я проверил, что Dispose
на моем SessionFactoryProvider
и на SessionProvider
вызываются в конце запроса, а не раньше, чем на моем Дао будет сделан вызов Save
.
Так что теперь я немного застрял. Несколько вещей приходят на ум.
- Я что-то делаю явно неправильно?
- NHibernate когда-либо закрывает сессии без моего ведома?
- Какие-нибудь обходные пути или идеи относительно того, что я мог бы сделать?
Заранее спасибо