Friday, March 4, 2011

Connection Management ASP.net

How do you manage your database connections in your ASP.Net application?

My understanding tells me the "best" way is to open a connection, make a query, close the connection - and do that multiple times because connection pooling makes the cost negligable.

The problem comes when I have a DAL where each method looks after its own connection. E.G.

User x = DAL.GetUserDetails();
Customer y = DAL.GetCustomer();

This is fine until we start talking about TransactionScope

using (TransactionScope t....
{
DAL.UpdateCustomer(y);
DAL.UpdateUser(x);
t.Complete();
}

ASP.Net wants to now use DTC because (I'm guessing) there are multiple connections involved.

Someone is going to say "Cache the connection somwhere" but I need to explicitly destroy the connection because of the way I am managing security (execute as / revert) and I don't want to have to make a call on every page to do that because someone will forget to make the call. I could also pass the connection into each method but thats not ideal because the page has to manage a connection.

Am I making sense or have I missed something fundamental here?

From stackoverflow
  • I was reading somewhere and I don't remember where that Microsoft was going to address that when you have two connections to the same DB that they would not escalate to DTC which would make this problem go away.

    Until then what we did was to develop our TransactionScope, our DAL's would then ask the TS for a new connection, and when we disposed the TS it would close the connection.

    The connections were stored in LogicalCallContext, although I would look at using HTTP Context instead. I left the company before the app went live but from what I;ve heard they haven't had any issues.

    so you'd have

    using (CustomTS.New()) { CustomerDal.Update() userDal.Update()
    }

    CustomTS had a static method that would get the current transaction and connection.

    Frustrating Developments : Nice idea, so if I understand correctly: in the dal methods I could look at TransactionScope.Current. If its not null use that transaction and the connection associated with that transaction. If it is null create a connection as I current do?
    Frustrating Developments : I'll have to give it a go before I give it the big tick
    JoshBerke : Yea basically that's it. You could make it easier on the Dal...have the TansactionScope.Current.GetConnection() method which always returns a connection. The class we had was very advanced wish I could share it but it's closed source...
    Frustrating Developments : TransactionScope is sealed and doesn't implement any interface so I'm stuck with writing my own instead of using what already exists - Why?
    JoshBerke : Yes you need to write your own from scratch....I'm sorry for not being clearer...I think we actually called it TransactionContext to differentiate from TS. Sorry
  • Implement it in a similar idea on how TransactionContext works.

    I started writing how to do this, but I think an example makes it clearer:

    public class MyConnectionManager : IDisposable
    {
        [ThreadStatic] // static per thread
        private static SqlConnection con;
    
        public static SqlConnection Connection
        {
            get
            {
                if (con == null)
                {
                    con = new SqlConnection();
                    con.Open();
                }
                return con;
            }
        }
    
        public void Dispose()
        {
            if (con != null)
            {
                con.Close();
            }
        }
    }
    
    public class Program
    {
        public void Run()
        {
            using(new MyConnectionManager())
            {
                MakeCall();
                MakeCall();
            } // Disposal happens here
        }
    
        public void MakeCall()
        {
            // The property can be accessed from anywhere
            SqlCommand cmd = new SqlCommand("SELECT 1", MyConnectionManager.Connection);
        }
    }
    

    This will cause any calls within the scope of the using statement to use the same connection, as long as they all run on the same thread. You can throw a transaction context in there as well.

    Caveat: In an ASP.NET WebPages scenario the thread can change between connection and page_load. See this article for more info. Also the thread is re-used. However if you are developing a DAL that is shared among applications, you may not be able to use the HttpContext, as mentioned, as there is none.

    JoshBerke : Don't use thread static, not in an asp.net app...google for the why (not enough chars in the comments to explain)
    Robert Wagner : @Josh - That is true to an extent, see my answer. However sometimes the DAL is shared among applications, also it's a bit different in WebService situations.
    lubos hasko : +1, Robert is right. `[ThreadStatic]` is not evil. Certainly you probably don't want to use it for private data just in case but I don't see how sqlconnection is private. Even if it leaks to another http request, there is no security risk if everybody is connecting to the same database anyway.

0 comments:

Post a Comment