Skip to content

Commit 7f478eb

Browse files
NH-2928 - Option for not requiring connection availability from commit prepare phase.
1 parent 25e5c48 commit 7f478eb

25 files changed

+677
-271
lines changed

src/NHibernate.Test/Async/SystemTransactions/DistributedSystemTransactionFixture.cs

+94-23
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,7 @@ namespace NHibernate.Test.SystemTransactions
2525
public class DistributedSystemTransactionFixtureAsync : SystemTransactionFixtureBase
2626
{
2727
private static readonly ILog _log = LogManager.GetLogger(typeof(DistributedSystemTransactionFixtureAsync));
28+
protected override bool UseConnectionOnSystemTransactionPrepare => true;
2829

2930
protected override bool AppliesTo(Dialect.Dialect dialect)
3031
=> dialect.SupportsDistributedTransactions && base.AppliesTo(dialect);
@@ -79,6 +80,7 @@ public async Task SupportsPromotingToDistributedAsync()
7980
[Test]
8081
public async Task WillNotCrashOnPrepareFailureAsync()
8182
{
83+
IgnoreIfUnsupported(false);
8284
var tx = new TransactionScope(TransactionScopeAsyncFlowOption.Enabled);
8385
var disposeCalled = false;
8486
try
@@ -111,9 +113,10 @@ public async Task WillNotCrashOnPrepareFailureAsync()
111113
}
112114
}
113115

114-
[Test]
115-
public async Task CanRollbackTransactionAsync([Values(false, true)] bool explicitFlush)
116+
[Theory]
117+
public async Task CanRollbackTransactionAsync(bool explicitFlush)
116118
{
119+
IgnoreIfUnsupported(explicitFlush);
117120
var tx = new TransactionScope(TransactionScopeAsyncFlowOption.Enabled);
118121
var disposeCalled = false;
119122
try
@@ -150,9 +153,10 @@ public async Task CanRollbackTransactionAsync([Values(false, true)] bool explici
150153
AssertNoPersons();
151154
}
152155

153-
[Test]
154-
public async Task CanRollbackTransactionFromScopeAsync([Values(false, true)] bool explicitFlush)
156+
[Theory]
157+
public async Task CanRollbackTransactionFromScopeAsync(bool explicitFlush)
155158
{
159+
IgnoreIfUnsupported(explicitFlush);
156160
using (new TransactionScope(TransactionScopeAsyncFlowOption.Enabled))
157161
using (var s = OpenSession())
158162
{
@@ -167,10 +171,11 @@ public async Task CanRollbackTransactionFromScopeAsync([Values(false, true)] boo
167171
AssertNoPersons();
168172
}
169173

170-
[Test]
174+
[Theory]
171175
[Description("Another action inside the transaction do the rollBack outside nh-session-scope.")]
172-
public async Task RollbackOutsideNhAsync([Values(false, true)] bool explicitFlush)
176+
public async Task RollbackOutsideNhAsync(bool explicitFlush)
173177
{
178+
IgnoreIfUnsupported(explicitFlush);
174179
try
175180
{
176181
using (var txscope = new TransactionScope(TransactionScopeAsyncFlowOption.Enabled))
@@ -198,10 +203,11 @@ public async Task RollbackOutsideNhAsync([Values(false, true)] bool explicitFlus
198203
AssertNoPersons();
199204
}
200205

201-
[Test]
206+
[Theory]
202207
[Description("rollback inside nh-session-scope should not commit save and the transaction should be aborted.")]
203-
public async Task TransactionInsertWithRollBackFromScopeAsync([Values(false, true)] bool explicitFlush)
208+
public async Task TransactionInsertWithRollBackFromScopeAsync(bool explicitFlush)
204209
{
210+
IgnoreIfUnsupported(explicitFlush);
205211
using (new TransactionScope(TransactionScopeAsyncFlowOption.Enabled))
206212
{
207213
using (var s = OpenSession())
@@ -218,10 +224,11 @@ public async Task TransactionInsertWithRollBackFromScopeAsync([Values(false, tru
218224
AssertNoPersons();
219225
}
220226

221-
[Test]
227+
[Theory]
222228
[Description("rollback inside nh-session-scope should not commit save and the transaction should be aborted.")]
223-
public async Task TransactionInsertWithRollBackTaskAsync([Values(false, true)] bool explicitFlush)
229+
public async Task TransactionInsertWithRollBackTaskAsync(bool explicitFlush)
224230
{
231+
IgnoreIfUnsupported(explicitFlush);
225232
try
226233
{
227234
using (var txscope = new TransactionScope(TransactionScopeAsyncFlowOption.Enabled))
@@ -248,12 +255,13 @@ public async Task TransactionInsertWithRollBackTaskAsync([Values(false, true)] b
248255
AssertNoPersons();
249256
}
250257

251-
[Test]
258+
[Theory]
252259
[Description(@"Two session in two txscope
253260
(without an explicit NH transaction)
254261
and with a rollback in the second dtc and a rollback outside nh-session-scope.")]
255-
public async Task TransactionInsertLoadWithRollBackFromScopeAsync([Values(false, true)] bool explicitFlush)
262+
public async Task TransactionInsertLoadWithRollBackFromScopeAsync(bool explicitFlush)
256263
{
264+
IgnoreIfUnsupported(explicitFlush);
257265
object savedId;
258266
var createdAt = DateTime.Today;
259267
using (var txscope = new TransactionScope(TransactionScopeAsyncFlowOption.Enabled))
@@ -291,12 +299,13 @@ public async Task TransactionInsertLoadWithRollBackFromScopeAsync([Values(false,
291299
}
292300
}
293301

294-
[Test]
302+
[Theory]
295303
[Description(@"Two session in two txscope
296304
(without an explicit NH transaction)
297305
and with a rollback in the second dtc and a ForceRollback outside nh-session-scope.")]
298-
public async Task TransactionInsertLoadWithRollBackTaskAsync([Values(false, true)] bool explicitFlush)
306+
public async Task TransactionInsertLoadWithRollBackTaskAsync(bool explicitFlush)
299307
{
308+
IgnoreIfUnsupported(explicitFlush);
300309
object savedId;
301310
var createdAt = DateTime.Today;
302311
using (var txscope = new TransactionScope(TransactionScopeAsyncFlowOption.Enabled))
@@ -344,9 +353,10 @@ public async Task TransactionInsertLoadWithRollBackTaskAsync([Values(false, true
344353
}
345354
}
346355

347-
[Test]
348-
public async Task CanDeleteItemInDtcAsync([Values(false, true)] bool explicitFlush)
356+
[Theory]
357+
public async Task CanDeleteItemInDtcAsync(bool explicitFlush)
349358
{
359+
IgnoreIfUnsupported(explicitFlush);
350360
object id;
351361
using (var tx = new TransactionScope(TransactionScopeAsyncFlowOption.Enabled))
352362
{
@@ -411,9 +421,10 @@ public async Task NH1744Async()
411421
}
412422
}
413423

414-
[Test]
415-
public async Task CanUseSessionWithManyScopesAsync([Values(false, true)] bool explicitFlush)
424+
[Theory]
425+
public async Task CanUseSessionWithManyScopesAsync(bool explicitFlush)
416426
{
427+
IgnoreIfUnsupported(explicitFlush);
417428
// Note that this fails with ConnectionReleaseMode.OnClose and SqlServer:
418429
// System.Data.SqlClient.SqlException : Microsoft Distributed Transaction Coordinator (MS DTC) has stopped this transaction.
419430
// Not much an issue since it is advised to not use ConnectionReleaseMode.OnClose.
@@ -480,9 +491,10 @@ public async Task CanUseSessionWithManyScopesAsync([Values(false, true)] bool ex
480491
}
481492
}
482493

483-
[Test]
484-
public async Task CanUseSessionOutsideOfScopeAfterScopeAsync([Values(false, true)] bool explicitFlush)
494+
[Theory]
495+
public async Task CanUseSessionOutsideOfScopeAfterScopeAsync(bool explicitFlush)
485496
{
497+
IgnoreIfUnsupported(explicitFlush);
486498
// Note that this fails with ConnectionReleaseMode.OnClose and Npgsql (< 3.2.5?):
487499
// NpgsqlOperationInProgressException: The connection is already in state 'Executing'
488500
// Not much an issue since it is advised to not use ConnectionReleaseMode.OnClose.
@@ -509,9 +521,11 @@ public async Task CanUseSessionOutsideOfScopeAfterScopeAsync([Values(false, true
509521
}
510522
}
511523

512-
[Test(Description = "Do not fail, but warn in case a delayed after scope disposal commit is made.")]
513-
public async Task DelayedTransactionCompletionAsync([Values(false, true)] bool explicitFlush)
524+
[Theory]
525+
[Description("Do not fail, but warn in case a delayed after scope disposal commit is made.")]
526+
public async Task DelayedTransactionCompletionAsync(bool explicitFlush)
514527
{
528+
IgnoreIfUnsupported(explicitFlush);
515529
for (var i = 1; i <= 10; i++)
516530
{
517531
// Isolation level must be read committed on the control session: reading twice while expecting some data insert
@@ -560,6 +574,57 @@ public async Task DelayedTransactionCompletionAsync([Values(false, true)] bool e
560574
}
561575
}
562576

577+
// Taken and adjusted from NH1632 When_commiting_items_in_DTC_transaction_will_add_items_to_2nd_level_cache
578+
[Test]
579+
public async Task WhenCommittingItemsAfterSessionDisposalWillAddThemTo2ndLevelCacheAsync()
580+
{
581+
int id;
582+
const string notNullData = "test";
583+
using (var tx = new TransactionScope(TransactionScopeAsyncFlowOption.Enabled))
584+
{
585+
using (var s = Sfi.OpenSession())
586+
{
587+
var person = new CacheablePerson { NotNullData = notNullData };
588+
await (s.SaveAsync(person));
589+
id = person.Id;
590+
591+
ForceEscalationToDistributedTx.Escalate();
592+
593+
await (s.FlushAsync());
594+
}
595+
tx.Complete();
596+
}
597+
598+
DodgeTransactionCompletionDelayIfRequired();
599+
600+
using (var tx = new TransactionScope(TransactionScopeAsyncFlowOption.Enabled))
601+
{
602+
using (var s = OpenSession())
603+
{
604+
ForceEscalationToDistributedTx.Escalate();
605+
606+
var person = await (s.LoadAsync<CacheablePerson>(id));
607+
Assert.That(person.NotNullData, Is.EqualTo(notNullData));
608+
}
609+
tx.Complete();
610+
}
611+
612+
// Closing the connection to ensure we can't actually use it.
613+
var connection = await (Sfi.ConnectionProvider.GetConnectionAsync(CancellationToken.None));
614+
Sfi.ConnectionProvider.CloseConnection(connection);
615+
616+
// The session is supposed to succeed because the second level cache should have the
617+
// entity to load, allowing the session to not use the connection at all.
618+
// Will fail if a transaction manager tries to enlist user supplied connection. Do
619+
// not add a transaction scope below.
620+
using (var s = Sfi.WithOptions().Connection(connection).OpenSession())
621+
{
622+
CacheablePerson person = null;
623+
Assert.DoesNotThrowAsync(async () => person = await (s.LoadAsync<CacheablePerson>(id)), "Failed loading entity from second level cache.");
624+
Assert.That(person.NotNullData, Is.EqualTo(notNullData));
625+
}
626+
}
627+
563628
[Test]
564629
public async Task DoNotDeadlockOnAfterTransactionWaitAsync()
565630
{
@@ -633,4 +698,10 @@ public void InDoubt(Enlistment enlistment)
633698
}
634699
}
635700
}
636-
}
701+
702+
[TestFixture]
703+
public class DistributedSystemTransactionWithoutConnectionFromPrepareFixtureAsync : DistributedSystemTransactionFixtureAsync
704+
{
705+
protected override bool UseConnectionOnSystemTransactionPrepare => false;
706+
}
707+
}

0 commit comments

Comments
 (0)