ADO.NET Connection Pool Performance Counters

Aug 13, 2017 windows ado-dotnet

I'm currently focusing on performance of my clients most used APIs. A few core services get hit a lot in their architecture, so they must be optimised for read and write, yet we are seeing some strange results from our apps when deployed. One theory could be that there are not enough connections open through to the database. We've noticed lots of task switching too in the app, could this be due to waiting for a free connection? This is a demo app to see how connections and connection pools behave on the local machine. I'll use .NET ADO SQL Performance Counters to help us.

I've written a sample app to demonstrate this. There's a JMeter runner which hits a simple controller. The controller uses the database correctly and should give us the results I want to demo.

screenshot

Here's the core code from the app, note that I don't have to open the connection, it's done internally inside Dapper.

public class ValuesController: ApiController {
  public async Task < IHttpActionResult > Get(string id) {
    using(var connection = new SqlConnection(ConfigurationManager.ConnectionStrings["database"].ConnectionString)) {
      // this is inefficient, just an example here
      var result = await connection.QueryAsync("SELECT [Data] FROM Example WHERE [Id]=@id", new {
        id
      });
      if (result.Count() == 0) {
        return NotFound();
      }
      return Ok(result);
    }
  }
}

Connections take resources to connect between the client and server. Keeping these connections opened and pooled lets them be shared between executions within the same process. .NET has connection pooling built into it which can be controlled via code or the connection strings.

This app performed well on my machine, with JMeter reporting sub 5 milliseconds response times with a connect time of below 1 milliseconds.

Performance Counters

If we want to see all the values, we must raise the level of performance counter detail:

<system.diagnostics>
  <switches>
    <add name="ConnectionPoolPerformanceCounterDetail" value="4" />
  </switches>
</system.diagnostics>

HardConnectsPerSecond

screenshot

How many connections are being opened and closed explicitly each second. You would expect one connection to be opened per database, or when there aren't enough connections in the connection pool. If you see this when the app is under load after making initial connections, then your app is closing connections and you should find a way to pool them - basically don't close them.

HardDisconnectsPerSecond

screenshot

Same as HardConnectsPerSecond except this is how many times a connection is being closed. I wouldn't expect to see any values whilst the app is running under load in the majority of applications since the connections should be pooled. If you see this your app is being disconnected from the server or you're closing the connection.

SoftConnectsPerSecond

screenshot

This counter shows how many connections are being opened by the application but have been intercepted by the connection pool. If there was already an unused connection pool connection open, then then open command would not open another connection, instead the connection pool would give it an open connection. If there wasn't an open connection and the connection pool is allowed to open one, you'll see the HardConnectsPerSecond counter increase. If you're calling connection.Open() anywhere (e.g., inside Dapper code, or your own ADO.NET SqlConnection using statements) then you should see soft connects. Microsoft say you should always Open() the connection but don't worry about closing it.

SoftDisconnectsPerSecond

screenshot

The same as above apart from this is soft disconnects. The value should be the similar or the same to SoftConnectsPerSecond. If your app is not disconnecting, then you are keeping connections open and it might lead to locking. .NET normally takes care of this for you.

NumberOfNonPooledConnections

screenshot

An early warning count of connections that don't take advantage of connection pooling. I'd expect this to be zero.

NumberOfPooledConnections

screenshot

This is an important graph. It states how many connections are being used within the pools. This is a count across all the pools so may be misleading. If you're putting a lot of traffic through to your app and this value is low, then you may be being bottlenecked by the connections as the app is having to share connections in its connection pool a lot (you may see task switching in your profiling). Increase the minimum pool connection count or check your connection opening code. A value of 1 here is bad. Here under load my app with one connection pool (for my one database connection) has 5 pooled connections - they may be in an active or waiting stage, but there are 5 connected to the SQL server on my machine.

NumberOfActiveConnectionPoolGroups

screenshot

Shows the number of active connection pool groups, which is different to the number of active connection pools, there should be one per database generally.  If this is zero or less than your number of different connection strings, then perhaps you're not utilising connection pooling. See NumberOfPooledConnections above.

NumberOfInactiveConnectionPoolGroups

screenshot

I'd expect this value to be zero unless your connections explicitly don't use connection pooling. ADO.NET uses connection pooling by default. See also NumberOfActiveConnectionPools.

NumberOfActiveConnectionPools

screenshot

Each connection belongs to one connection pool. If you're using connection pooling then you will likely have one connection pool per database, with multiple active connections within it.

NumberOfInactiveConnectionPools

screenshot

This is the number of connection pools that were created but their connections aren't currently being used.

NumberOfActiveConnections

screenshot

This is a good one to watch. In my example application, I saw a range of values here. The lowest I saw was 0 and the highest I saw was 4. If this is stuck at 1, you may somehow be locking your database code so only one task or thread can use it at once. When NumberOfActiveConnections increases generally the NumberOfFreeConnections decreases since they are being used up.

NumberOfFreeConnections

screenshot

These are the number of open connections that are not being used. When the NumberOfFreeConnections increases, it means the connection is still alive in the pool but not currently being used by any of your code.

NumberOfStatisConnections

screenshot

I've no idea what this means, and it had a value of 0 throughout. Google didn't help me either except for some Oracle documentation where it's not used - so this is probably a legacy performance counter.

NumberOfReclaimedConnections

screenshot

This value was always zero under load. A value here indicates .NET garbage collection released a connection, and you should have disposed or closed it in your code.

Further reading