AnswerDebugging a resource leak
As it turns out, there are a LOT of issues with this code:
public class QueueActions : IDisposable { UnmanagedDatabaseConnection database; public string Name { get; private set; } public class QueueActions( UnmanagedDatabaseConnectionFactory factory) { database = factory.Create(); database.Open(()=> Name = database.ReadName()); } // assume proper GC finalizer impl public void Dispose() { database.Dispose(); } }
And the code using this:
using(var factory = CreateFactory()) { ThreadPool.QueueUserWorkItem(()=> { using(var actions = new QueueActions(factory)) { actions.Send("abc"); } }); }
To begin with, what happens if we close the factory between the first and second lines in QueueActions constructors?
We already have an unmanaged resource, but when we try to open it, we are going to get an exception. Since the exception is thrown from the constructor, it will NOT invoke the usual using logic, and the code will not be disposed.
Furthermore, and the reason for the blog post about it. Dispose itself can also fail.
Here is the actual stack trace that caused this blog post:
Microsoft.Isam.Esent.Interop.EsentErrorException: Error TermInProgress (JET_errTermInProgress, Termination in progress) at Microsoft.Isam.Esent.Interop.Api.Check(Int32 err) in Api.cs: line 1492 at Microsoft.Isam.Esent.Interop.Api.JetCloseTable(JET_SESID sesid, JET_TABLEID tableid) in Api.cs: line 372 at Microsoft.Isam.Esent.Interop.Table.ReleaseResource() in D:\Work\esent\EsentInterop\Table.cs: line 97 at Microsoft.Isam.Esent.Interop.EsentResource.Dispose() in EsentResource.cs: line 63 at Rhino.Queues.Storage.AbstractActions.Dispose() in AbstractActions.cs: line 146
More posts in "Answer" series:
- (22 Jan 2025) What does this code do?
- (05 Jan 2023) what does this code print?
- (15 Dec 2022) What does this code print?
- (07 Apr 2022) Why is this code broken?
- (20 Jan 2017) What does this code do?
- (16 Aug 2011) Modifying execution approaches
- (30 Apr 2011) Stopping the leaks
- (24 Dec 2010) This code should never hit production
- (21 Dec 2010) Your own ThreadLocal
- (11 Feb 2010) Debugging a resource leak
- (03 Sep 2009) The lazy loaded inheritance many to one association OR/M conundrum
- (04 Sep 2008) Don't stop with the first DSL abstraction
- (12 Jun 2008) How many tests?
Comments
Comment preview