Search code examples
c#.netnhibernateconsole-applicationsharp-architecture

How to setup NHibernate with Sharp Architecture in a multi-threaded console application?


I'm still pretty new to these technologies. The real problem here is how to manage the sessions per thread in the console app. Currently, if I run it as a single thread then all is well. As soon as I switch to a multi-threaded model, I'll start to see contention at the session level (because the Session object is not theadsafe by design) KeyNotFound exceptions (among others) start to get thrown.

In a web app, you'd do something like this:

    /// <summary>
    /// Due to issues on IIS7, the NHibernate initialization cannot reside in Init() but
    /// must only be called once.  Consequently, we invoke a thread-safe singleton class to
    /// ensure it's only initialized once.
    /// </summary>
    protected void Application_BeginRequest(object sender, EventArgs e)
    {
        NHibernateInitializer.Instance().InitializeNHibernateOnce(
            () => InitializeNHibernateSession());
    }

    /// <summary>
    /// If you need to communicate to multiple databases, you'd add a line to this method to
    /// initialize the other database as well.
    /// </summary>
    private void InitializeNHibernateSession()
    {            

        var path = ConfigurationManager.AppSettings["NHibernateConfig"];
        NHibernateSession.Init(
            webSessionStorage,
            new string[] { Server.MapPath("~/bin/foo.Data.dll") },
            new AutoPersistenceModelGenerator().Generate(),
            Server.MapPath("~/App_Configuration/" + path ));
    }

// sample of my console app... very simple
static void Main(string[] args)
{
  InitializeNHibernateSession();
  while(true)
  {
    Task.Factory.StartNew(() => SomeAwesomeLongRunningPieceOfWork());
  }
}

Which essentially executes the initialization once per thread (web request) in the global.asax.

Any ideas on how to set this (the session management) up in a console app?


Solution

  • This worked for me:

    // Three threads:
    for (int i = 0; i < 3; i++)
    {
       Thread curThread = new Thread(StartThread);
       curThread.Start();
    }
    
    private void StartThread()
    {
          NHibernateInitializer.Instance().InitializeNHibernateOnce(InitializeNHibernateSession);
          SomeAwesomeLongRunningPieceOfWork();            
    }
    
    private void InitializeNHibernateSession()
    {
       var path = ConfigurationManager.AppSettings["NHibernateConfig"];
    
       NHibernateSession.Init(
          new ThreadSessionStorage(),
          new string[] { "foo.Data.dll" },
          new AutoPersistenceModelGenerator().Generate(),
          "./App_Configuration/" + path);
    }
    

    The key was this class, which I got from:

    http://groups.google.com/group/sharp-architecture/browse_thread/thread/ce3d9c34bc2da629?fwc=1 http://groups.google.com/group/sharp-architecture/browse_thread/thread/51794671c91bc5e9/386efc30d4c0bf16#386efc30d4c0bf16

    public class ThreadSessionStorage : ISessionStorage
    {
        [ThreadStatic]
        private static ISession _session;
        public ISession Session
        {
            get
            {
                return _session;
            }
            set
            {
                _session = value;
            }
        }
    
        public ISession GetSessionForKey(string factoryKey)
        {
            return Session;
        }
    
        public void SetSessionForKey(string factoryKey, ISession session)
        {
            Session = session;
        }
    
        public IEnumerable<ISession> GetAllSessions()
        {
            return new List<ISession>() { Session };
        }
    }
    

    And it works just fine.