@@ -25,6 +25,7 @@ namespace NHibernate.Test.SystemTransactions
25
25
public class DistributedSystemTransactionFixtureAsync : SystemTransactionFixtureBase
26
26
{
27
27
private static readonly ILog _log = LogManager . GetLogger ( typeof ( DistributedSystemTransactionFixtureAsync ) ) ;
28
+ protected override bool UseConnectionOnSystemTransactionPrepare => true ;
28
29
29
30
protected override bool AppliesTo ( Dialect . Dialect dialect )
30
31
=> dialect . SupportsDistributedTransactions && base . AppliesTo ( dialect ) ;
@@ -79,6 +80,7 @@ public async Task SupportsPromotingToDistributedAsync()
79
80
[ Test ]
80
81
public async Task WillNotCrashOnPrepareFailureAsync ( )
81
82
{
83
+ IgnoreIfUnsupported ( false ) ;
82
84
var tx = new TransactionScope ( TransactionScopeAsyncFlowOption . Enabled ) ;
83
85
var disposeCalled = false ;
84
86
try
@@ -111,9 +113,10 @@ public async Task WillNotCrashOnPrepareFailureAsync()
111
113
}
112
114
}
113
115
114
- [ Test ]
115
- public async Task CanRollbackTransactionAsync ( [ Values ( false , true ) ] bool explicitFlush )
116
+ [ Theory ]
117
+ public async Task CanRollbackTransactionAsync ( bool explicitFlush )
116
118
{
119
+ IgnoreIfUnsupported ( explicitFlush ) ;
117
120
var tx = new TransactionScope ( TransactionScopeAsyncFlowOption . Enabled ) ;
118
121
var disposeCalled = false ;
119
122
try
@@ -150,9 +153,10 @@ public async Task CanRollbackTransactionAsync([Values(false, true)] bool explici
150
153
AssertNoPersons ( ) ;
151
154
}
152
155
153
- [ Test ]
154
- public async Task CanRollbackTransactionFromScopeAsync ( [ Values ( false , true ) ] bool explicitFlush )
156
+ [ Theory ]
157
+ public async Task CanRollbackTransactionFromScopeAsync ( bool explicitFlush )
155
158
{
159
+ IgnoreIfUnsupported ( explicitFlush ) ;
156
160
using ( new TransactionScope ( TransactionScopeAsyncFlowOption . Enabled ) )
157
161
using ( var s = OpenSession ( ) )
158
162
{
@@ -167,10 +171,11 @@ public async Task CanRollbackTransactionFromScopeAsync([Values(false, true)] boo
167
171
AssertNoPersons ( ) ;
168
172
}
169
173
170
- [ Test ]
174
+ [ Theory ]
171
175
[ 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 )
173
177
{
178
+ IgnoreIfUnsupported ( explicitFlush ) ;
174
179
try
175
180
{
176
181
using ( var txscope = new TransactionScope ( TransactionScopeAsyncFlowOption . Enabled ) )
@@ -198,10 +203,11 @@ public async Task RollbackOutsideNhAsync([Values(false, true)] bool explicitFlus
198
203
AssertNoPersons ( ) ;
199
204
}
200
205
201
- [ Test ]
206
+ [ Theory ]
202
207
[ 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 )
204
209
{
210
+ IgnoreIfUnsupported ( explicitFlush ) ;
205
211
using ( new TransactionScope ( TransactionScopeAsyncFlowOption . Enabled ) )
206
212
{
207
213
using ( var s = OpenSession ( ) )
@@ -218,10 +224,11 @@ public async Task TransactionInsertWithRollBackFromScopeAsync([Values(false, tru
218
224
AssertNoPersons ( ) ;
219
225
}
220
226
221
- [ Test ]
227
+ [ Theory ]
222
228
[ 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 )
224
230
{
231
+ IgnoreIfUnsupported ( explicitFlush ) ;
225
232
try
226
233
{
227
234
using ( var txscope = new TransactionScope ( TransactionScopeAsyncFlowOption . Enabled ) )
@@ -248,12 +255,13 @@ public async Task TransactionInsertWithRollBackTaskAsync([Values(false, true)] b
248
255
AssertNoPersons ( ) ;
249
256
}
250
257
251
- [ Test ]
258
+ [ Theory ]
252
259
[ Description ( @"Two session in two txscope
253
260
(without an explicit NH transaction)
254
261
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 )
256
263
{
264
+ IgnoreIfUnsupported ( explicitFlush ) ;
257
265
object savedId ;
258
266
var createdAt = DateTime . Today ;
259
267
using ( var txscope = new TransactionScope ( TransactionScopeAsyncFlowOption . Enabled ) )
@@ -291,12 +299,13 @@ public async Task TransactionInsertLoadWithRollBackFromScopeAsync([Values(false,
291
299
}
292
300
}
293
301
294
- [ Test ]
302
+ [ Theory ]
295
303
[ Description ( @"Two session in two txscope
296
304
(without an explicit NH transaction)
297
305
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 )
299
307
{
308
+ IgnoreIfUnsupported ( explicitFlush ) ;
300
309
object savedId ;
301
310
var createdAt = DateTime . Today ;
302
311
using ( var txscope = new TransactionScope ( TransactionScopeAsyncFlowOption . Enabled ) )
@@ -344,9 +353,10 @@ public async Task TransactionInsertLoadWithRollBackTaskAsync([Values(false, true
344
353
}
345
354
}
346
355
347
- [ Test ]
348
- public async Task CanDeleteItemInDtcAsync ( [ Values ( false , true ) ] bool explicitFlush )
356
+ [ Theory ]
357
+ public async Task CanDeleteItemInDtcAsync ( bool explicitFlush )
349
358
{
359
+ IgnoreIfUnsupported ( explicitFlush ) ;
350
360
object id ;
351
361
using ( var tx = new TransactionScope ( TransactionScopeAsyncFlowOption . Enabled ) )
352
362
{
@@ -411,9 +421,10 @@ public async Task NH1744Async()
411
421
}
412
422
}
413
423
414
- [ Test ]
415
- public async Task CanUseSessionWithManyScopesAsync ( [ Values ( false , true ) ] bool explicitFlush )
424
+ [ Theory ]
425
+ public async Task CanUseSessionWithManyScopesAsync ( bool explicitFlush )
416
426
{
427
+ IgnoreIfUnsupported ( explicitFlush ) ;
417
428
// Note that this fails with ConnectionReleaseMode.OnClose and SqlServer:
418
429
// System.Data.SqlClient.SqlException : Microsoft Distributed Transaction Coordinator (MS DTC) has stopped this transaction.
419
430
// 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
480
491
}
481
492
}
482
493
483
- [ Test ]
484
- public async Task CanUseSessionOutsideOfScopeAfterScopeAsync ( [ Values ( false , true ) ] bool explicitFlush )
494
+ [ Theory ]
495
+ public async Task CanUseSessionOutsideOfScopeAfterScopeAsync ( bool explicitFlush )
485
496
{
497
+ IgnoreIfUnsupported ( explicitFlush ) ;
486
498
// Note that this fails with ConnectionReleaseMode.OnClose and Npgsql (< 3.2.5?):
487
499
// NpgsqlOperationInProgressException: The connection is already in state 'Executing'
488
500
// Not much an issue since it is advised to not use ConnectionReleaseMode.OnClose.
@@ -509,9 +521,11 @@ public async Task CanUseSessionOutsideOfScopeAfterScopeAsync([Values(false, true
509
521
}
510
522
}
511
523
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 )
514
527
{
528
+ IgnoreIfUnsupported ( explicitFlush ) ;
515
529
for ( var i = 1 ; i <= 10 ; i ++ )
516
530
{
517
531
// 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
560
574
}
561
575
}
562
576
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
+
563
628
[ Test ]
564
629
public async Task DoNotDeadlockOnAfterTransactionWaitAsync ( )
565
630
{
@@ -633,4 +698,10 @@ public void InDoubt(Enlistment enlistment)
633
698
}
634
699
}
635
700
}
636
- }
701
+
702
+ [ TestFixture ]
703
+ public class DistributedSystemTransactionWithoutConnectionFromPrepareFixtureAsync : DistributedSystemTransactionFixtureAsync
704
+ {
705
+ protected override bool UseConnectionOnSystemTransactionPrepare => false ;
706
+ }
707
+ }
0 commit comments