//------------------------------------------------------------------------------ // // Copyright (c) Microsoft Corporation. All rights reserved. // //------------------------------------------------------------------------------ #region Using directives using System; using System.Diagnostics; using System.Transactions; using System.Data; using System.Data.Common; using System.Threading; #endregion namespace System.Workflow.Runtime.Hosting { /// /// This class wraps a local transaction (DbTransaction) by implementing IPromotableSinglePhaseNotification: /// It instantiate a DbTransaction and never allows promotion to a DTC transaction. /// internal sealed class LocalTransaction : IPromotableSinglePhaseNotification { readonly DbTransaction transaction; System.Data.Common.DbConnection connection; ManualResetEvent handle; object syncRoot = new Object(); #region Constructor /// /// Wraps a local transaction inside /// /// internal LocalTransaction(DbResourceAllocator dbHelper, ManualResetEvent handle) { if (null == handle) throw new ArgumentNullException("handle"); // Open a connection that specifically does not auto-enlist ("Enlist=false" in connection string) // to prevent auto-promotion of the transaction this.connection = dbHelper.OpenNewConnectionNoEnlist(); this.transaction = this.connection.BeginTransaction(); this.handle = handle; } #endregion Constructor #region Accessors public System.Data.Common.DbConnection Connection { get { return this.connection; } } public DbTransaction Transaction { get { return this.transaction; } } #endregion Accessors #region IPromotableSinglePhaseNotification Members public void Initialize() { } public void SinglePhaseCommit(SinglePhaseEnlistment en) { if (en == null) throw new ArgumentNullException("en"); // // Wait until IPendingWork members have completed (WinOE bugs 17580 and 13395) try { handle.WaitOne(); } catch (ObjectDisposedException) { // If an ObjectDisposedException is thrown because // the WaitHandle has already closed, nothing to worry // about. Move on. } lock (syncRoot) { try { this.transaction.Commit(); en.Committed(); } catch (Exception e) { en.Aborted(e); } finally { if ((null != connection) && (ConnectionState.Closed != connection.State)) { connection.Close(); connection = null; } } } } public void Rollback(SinglePhaseEnlistment en) { if (en == null) throw new ArgumentNullException("en"); // // Wait until IPendingWork members have completed (WinOE bugs 17580 and 13395) try { handle.WaitOne(); } catch (ObjectDisposedException) { // If an ObjectDisposedException is thrown because // the WaitHandle has already closed, nothing to worry // about. Move on. } // DbTransaction.Dispose will call Rollback if the DbTransaction is not Zombied. // Not safe to call DbTransaction.Rollback here as the the underlining // DbTransaction may have already been Rolled back lock (syncRoot) { if (null != transaction) transaction.Dispose(); if ((null != connection) && (ConnectionState.Closed != connection.State)) { connection.Close(); connection = null; } en.Aborted(); } } public byte[] Promote() { throw new TransactionPromotionException( ExecutionStringManager.PromotionNotSupported); } #endregion IPromotableSinglePhaseNotification Members } }