Проблема с настройкой StructureMap / nhibernate: - PullRequest
1 голос
/ 28 февраля 2010

Я получаю следующую ошибку:

Невозможно получить доступ к удаленному объекту. Название объекта: «AdoTransaction».

Установка следует примеру, приведенному на http://trason.net/journal/2009/10/7/bootstrapping-nhibernate-with-structuremap.html

вот класс IUnitOfWork (точно такой же, как в ссылке):

public class UnitOfWork : IUnitOfWork
{
    private readonly ISessionFactory _sessionFactory;
    private readonly ITransaction _transaction;

    public UnitOfWork(ISessionFactory sessionFactory)
    {
        _sessionFactory = sessionFactory;
        CurrentSession = _sessionFactory.OpenSession();
        _transaction = CurrentSession.BeginTransaction();
    }

    public ISession CurrentSession { get; private set; }

    public void Dispose()
    {
        CurrentSession.Close();
        CurrentSession = null;
    }

    public void Commit()
    {
        _transaction.Commit();
    }

}

вот NHibernateModule (опять же!):

public class NHibernateModule : IHttpModule, IDisposable
{
    private IUnitOfWork _unitOfWork;

    public void Init(HttpApplication context)
    {
        context.BeginRequest += ContextBeginRequest;
        context.EndRequest += ContextEndRequest;
    }

    private void ContextBeginRequest(object sender, EventArgs e)
    {
        _unitOfWork = ObjectFactory.GetInstance<IUnitOfWork>();
    }

    private void ContextEndRequest(object sender, EventArgs e)
    {
        Dispose();
    }

    public void Dispose()
    {
        if (_unitOfWork == null) return;
        _unitOfWork.Commit();
        _unitOfWork.Dispose();
    }
}

вот мой репо:

 public class Repository<T> : IRepository<T>
{
    public readonly IUnitOfWork _uow;

    public Repository(IUnitOfWork uow)
    {
        _uow = uow;
    }

    public Repository()
    {

    }

    #region IRepository<T> Members

    public IList<T> GetAll()
    {
        using (var session = _uow.CurrentSession)
        {
            return session.CreateCriteria(typeof (T)).List<T>();
        }
    }

    public IList<T> FindAll<T>(IList<Expression<Func<T, bool>>> criteria)
    {
        var session = _uow.CurrentSession;

        var query = from item in session.SessionFactory.OpenSession().Query<T>()
                              select item;
        foreach (var criterion in criteria)
        {
            query = query.Where(criterion);
        }
        return query.ToList();
    }

    public T FindFirst<T>(IList<Expression<Func<T, bool>>> criteria)
    {

        var col = FindAll(criteria);

        if (col.Count > 0)
        {
            return col.First();
        }
        else
        {
            return default(T);
        }
    }

    public T Get(int id)
    {
        using (var session = _uow.CurrentSession)
        {
            return session.Get<T>(id);
        }
    }

    public void Save(T entity)
    {
        using (var session = _uow.CurrentSession)
        {
            session.Save(entity);
        }
    }

    public void Update(T entity)
    {
        using (var session = _uow.CurrentSession)
        {
            session.Update(entity);
            session.Flush();
        }
    }

    #endregion
}
}

вот мой загрузчик:

 public class BootStrapper : IBootstrapper
{
    private static bool _hasStarted;

    public virtual void BootstrapStructureMap()
    {
        ObjectFactory.Initialize(x =>
        {
            x.Scan(s =>
            {
                s.TheCallingAssembly();
                s.AssemblyContainingType<User>();
                s.AssemblyContainingType<UserRepository>();
                s.AssemblyContainingType<NHibernateRegistry>();
                s.LookForRegistries();
            });

            // Repositories
            x.For<WmcStar.Data.IUserRepository>()
                .CacheBy(InstanceScope.HttpContext)
                .TheDefault.Is.OfConcreteType<UserRepository>();

            x.For<IDatabaseBuilder>().TheDefaultIsConcreteType<DatabaseBuilder>(); 

        });
    }

    public static void Restart()
    {
        if (_hasStarted)
        {
            ObjectFactory.ResetDefaults();
        }
        else
        {
            Bootstrap();
            _hasStarted = true;
        }
    }

    public static void Bootstrap()
    {
        new BootStrapper().BootstrapStructureMap();
    }

}

вот мое NHibernateRegistry:

public class NHibernateRegistry : Registry
{
    public NHibernateRegistry()
    {
        var cfg = new Configuration()
             .SetProperty(NHibernate.Cfg.Environment.ReleaseConnections, "on_close")
             .SetProperty(NHibernate.Cfg.Environment.Dialect, typeof(NHibernate.Dialect.MsSql2005Dialect).AssemblyQualifiedName)
             .SetProperty(NHibernate.Cfg.Environment.ConnectionDriver, typeof(NHibernate.Driver.SqlClientDriver).AssemblyQualifiedName)
             .SetProperty(NHibernate.Cfg.Environment.ConnectionString, @"my connstring")
             .SetProperty(NHibernate.Cfg.Environment.ProxyFactoryFactoryClass, typeof(ProxyFactoryFactory).AssemblyQualifiedName)
             .AddAssembly(typeof(User).Assembly);

        var sessionFactory = cfg.BuildSessionFactory();

        For<Configuration>().Singleton().Use(cfg);

        For<ISessionFactory>().Singleton().Use(sessionFactory);


        For<ISession>().HybridHttpOrThreadLocalScoped()
            .Use(ctx => ctx.GetInstance<ISessionFactory>().OpenSession());

        For<IUnitOfWork>().HybridHttpOrThreadLocalScoped()
            .Use<UnitOfWork>();

        For<IDatabaseBuilder>().Use<DatabaseBuilder>();
        SetAllProperties(x => x.OfType<IUnitOfWork>());
    }
}

и, наконец, вот мой global.asax:

    public class Global : System.Web.HttpApplication
{

    protected void Application_Start(object sender, EventArgs e)
    {
        BootStrapper.Bootstrap();
        new SchemaExport(ObjectFactory.GetInstance<Configuration>()).Execute(false, true, false);
        ObjectFactory.GetInstance<IDatabaseBuilder>().RebuildDatabase();

        AutoMapper.Mapper.CreateMap<WmcStar.Model.User, WmcStar.Data.Dto.User>();
    }
}

Кто-нибудь получил какие-либо подсказки относительно того, что могло бы вызвать это?

ш: //

Ответы [ 4 ]

4 голосов
/ 22 марта 2010

Я столкнулся с этой проблемой при использовании StructureMap и NHibernate.В основном, у меня есть карта структуры карты именованного объекта (например, CurrentUser).В этом отображении он использует сеанс NHibernate для извлечения текущего пользователя из БД.

Затем в моем коде где-то я делал что-то вроде:

using (var transaction = _session.BeginTransaction())
{
  var user = ObjectFactory.GetNamedInstance<User>("CurrentUser");
  var myObjects = _session.QueryOver<myObject>().Where(x => x.User == CurrentUser).Future();
  transaction.Commit();
}

Ошибка выдаетсятранзакция располагается как часть текущей пользовательской операции.Поэтому, когда выполнение программы достигает строки транзакции.Commit (), транзакция для фиксации отсутствует.Мое решение состояло в том, чтобы просто переместить пользовательскую строку за пределы оператора транзакции:

var user = ObjectFactory.GetNamedInstance<User>("CurrentUser");
using (var transaction = _session.BeginTransaction())
{
  var myObjects = _session.QueryOver<MyObject>().Where(x => x.User == CurrentUser).Future();
  transaction.Commit();
}

Это позволит транзакции вокруг текущего поиска пользователя быть независимой от транзакции для MyObject.

2 голосов
/ 28 февраля 2010

Я не знаю, какая именно проблема вызывает исключение, но я обнаружил некоторые потенциальные проблемы:

  1. Зафиксируйте единицу работы в конце запроса и откатите ее в случае неудачной фиксации.
  2. Откатить единицу работы при возникновении исключения во время запроса.
  3. Не вызывайте dispose в конце запроса, он злоупотребляет IDisposable класса.
  4. Утилизируйте единицу работы в конце запроса, если она еще не удалена.
  5. Не размещайте сеанс в каждом методе вашего хранилища.
  6. Утилизируйте сессию в методе утилизации единицы работы.
  7. Вам не нужно явно закрывать сеанс. достаточно утилизации.
  8. Удалите транзакцию перед сеансом в методе dispose.
  9. Используйте проверки в методе утилизации единицы работы, чтобы предотвратить утечки памяти и сделать метод удаления мусора стойким.

После ввода этого списка, я думаю, исключение вызвано использованием блоков в хранилище.

Если вы хотите объяснить одну из этих вещей, пожалуйста, спросите меня в комментариях.

2 голосов
/ 28 февраля 2010

Возможно, это не имеет ничего общего с комбинацией StructureMap и NHibernate, а также с вашим кодом? Я предполагаю, что не вижу кода, но, скорее всего, вы удаляете транзакцию и затем пытаетесь получить к ней доступ.

0 голосов
/ 16 марта 2018

Когда я удаляю объект, я сталкиваюсь с тем же сообщением об ошибке. В моих обстоятельствах первопричиной является одно из значений свойства сущности NULL. После очистки значений NULL команда выполнена успешно.

...