26
26
import java .util .List ;
27
27
import java .util .Map ;
28
28
29
+ import org .jspecify .annotations .Nullable ;
29
30
import org .junit .jupiter .api .AfterEach ;
30
31
import org .junit .jupiter .api .Test ;
31
32
32
33
import org .springframework .beans .factory .annotation .Autowired ;
34
+ import org .springframework .context .ApplicationEvent ;
33
35
import org .springframework .context .ApplicationEventPublisher ;
34
36
import org .springframework .context .ConfigurableApplicationContext ;
35
37
import org .springframework .context .annotation .AnnotationConfigApplicationContext ;
43
45
import org .springframework .transaction .annotation .EnableTransactionManagement ;
44
46
import org .springframework .transaction .annotation .Propagation ;
45
47
import org .springframework .transaction .annotation .Transactional ;
48
+ import org .springframework .transaction .config .GlobalTransactionalEventErrorHandler ;
46
49
import org .springframework .transaction .support .TransactionSynchronization ;
47
50
import org .springframework .transaction .support .TransactionSynchronizationManager ;
48
51
import org .springframework .transaction .support .TransactionTemplate ;
@@ -99,12 +102,12 @@ void immediately() {
99
102
void immediatelyImpactsCurrentTransaction () {
100
103
load (ImmediateTestListener .class , BeforeCommitTestListener .class );
101
104
assertThatIllegalStateException ().isThrownBy (() ->
102
- this .transactionTemplate .execute (status -> {
103
- getContext ().publishEvent ("FAIL" );
104
- throw new AssertionError ("Should have thrown an exception at this point" );
105
- }))
106
- .withMessageContaining ("Test exception" )
107
- .withMessageContaining (EventCollector .IMMEDIATELY );
105
+ this .transactionTemplate .execute (status -> {
106
+ getContext ().publishEvent ("FAIL" );
107
+ throw new AssertionError ("Should have thrown an exception at this point" );
108
+ }))
109
+ .withMessageContaining ("Test exception" )
110
+ .withMessageContaining (EventCollector .IMMEDIATELY );
108
111
109
112
getEventCollector ().assertEvents (EventCollector .IMMEDIATELY , "FAIL" );
110
113
getEventCollector ().assertTotalEventsCount (1 );
@@ -369,6 +372,45 @@ void conditionFoundOnMetaAnnotation() {
369
372
getEventCollector ().assertNoEventReceived ();
370
373
}
371
374
375
+ @ Test
376
+ void afterCommitThrowException () {
377
+ doLoad (HandlerConfiguration .class , AfterCommitErrorHandlerTestListener .class );
378
+ this .transactionTemplate .execute (status -> {
379
+ getContext ().publishEvent ("test" );
380
+ getEventCollector ().assertNoEventReceived ();
381
+ return null ;
382
+ });
383
+ getEventCollector ().assertEvents (EventCollector .AFTER_COMMIT , "test" );
384
+ getEventCollector ().assertEvents (EventCollector .HANDLE_ERROR , "HANDLE_ERROR" );
385
+ getEventCollector ().assertTotalEventsCount (2 );
386
+ }
387
+
388
+ @ Test
389
+ void afterRollbackThrowException () {
390
+ doLoad (HandlerConfiguration .class , AfterRollbackErrorHandlerTestListener .class );
391
+ this .transactionTemplate .execute (status -> {
392
+ getContext ().publishEvent ("test" );
393
+ getEventCollector ().assertNoEventReceived ();
394
+ status .setRollbackOnly ();
395
+ return null ;
396
+ });
397
+ getEventCollector ().assertEvents (EventCollector .AFTER_ROLLBACK , "test" );
398
+ getEventCollector ().assertEvents (EventCollector .HANDLE_ERROR , "HANDLE_ERROR" );
399
+ getEventCollector ().assertTotalEventsCount (2 );
400
+ }
401
+
402
+ @ Test
403
+ void afterCompletionThrowException () {
404
+ doLoad (HandlerConfiguration .class , AfterCompletionErrorHandlerTestListener .class );
405
+ this .transactionTemplate .execute (status -> {
406
+ getContext ().publishEvent ("test" );
407
+ getEventCollector ().assertNoEventReceived ();
408
+ return null ;
409
+ });
410
+ getEventCollector ().assertEvents (EventCollector .AFTER_COMPLETION , "test" );
411
+ getEventCollector ().assertEvents (EventCollector .HANDLE_ERROR , "HANDLE_ERROR" );
412
+ getEventCollector ().assertTotalEventsCount (2 );
413
+ }
372
414
373
415
protected EventCollector getEventCollector () {
374
416
return this .eventCollector ;
@@ -442,6 +484,36 @@ public TransactionTemplate transactionTemplate() {
442
484
}
443
485
}
444
486
487
+ @ Configuration
488
+ @ EnableTransactionManagement
489
+ static class HandlerConfiguration {
490
+
491
+ @ Bean
492
+ public EventCollector eventCollector () {
493
+ return new EventCollector ();
494
+ }
495
+
496
+ @ Bean
497
+ public TestBean testBean (ApplicationEventPublisher eventPublisher ) {
498
+ return new TestBean (eventPublisher );
499
+ }
500
+
501
+ @ Bean
502
+ public CallCountingTransactionManager transactionManager () {
503
+ return new CallCountingTransactionManager ();
504
+ }
505
+
506
+ @ Bean
507
+ public TransactionTemplate transactionTemplate () {
508
+ return new TransactionTemplate (transactionManager ());
509
+ }
510
+
511
+ @ Bean
512
+ public AfterRollbackErrorHandler errorHandler (ApplicationEventPublisher eventPublisher ) {
513
+ return new AfterRollbackErrorHandler (eventPublisher );
514
+ }
515
+ }
516
+
445
517
446
518
@ Configuration
447
519
static class MulticasterWithCustomExecutor {
@@ -467,7 +539,9 @@ static class EventCollector {
467
539
468
540
public static final String AFTER_ROLLBACK = "AFTER_ROLLBACK" ;
469
541
470
- public static final String [] ALL_PHASES = {IMMEDIATELY , BEFORE_COMMIT , AFTER_COMMIT , AFTER_ROLLBACK };
542
+ public static final String HANDLE_ERROR = "HANDLE_ERROR" ;
543
+
544
+ public static final String [] ALL_PHASES = {IMMEDIATELY , BEFORE_COMMIT , AFTER_COMMIT , AFTER_ROLLBACK , HANDLE_ERROR };
471
545
472
546
private final MultiValueMap <String , Object > events = new LinkedMultiValueMap <>();
473
547
@@ -486,7 +560,7 @@ public void assertNoEventReceived(String... phases) {
486
560
for (String phase : phases ) {
487
561
List <Object > eventsForPhase = getEvents (phase );
488
562
assertThat (eventsForPhase .size ()).as ("Expected no events for phase '" + phase + "' " +
489
- "but got " + eventsForPhase + ":" ).isEqualTo (0 );
563
+ "but got " + eventsForPhase + ":" ).isEqualTo (0 );
490
564
}
491
565
}
492
566
@@ -504,7 +578,7 @@ public void assertTotalEventsCount(int number) {
504
578
size += entry .getValue ().size ();
505
579
}
506
580
assertThat (size ).as ("Wrong number of total events (" + this .events .size () + ") " +
507
- "registered phase(s)" ).isEqualTo (number );
581
+ "registered phase(s)" ).isEqualTo (number );
508
582
}
509
583
}
510
584
@@ -677,6 +751,51 @@ public void handleAfterCommit(String data) {
677
751
}
678
752
679
753
754
+ @ Component
755
+ static class AfterCommitErrorHandlerTestListener extends BaseTransactionalTestListener {
756
+
757
+ @ TransactionalEventListener (phase = AFTER_COMMIT , condition = "!'HANDLE_ERROR'.equals(#data)" )
758
+ public void handleBeforeCommit (String data ) {
759
+ handleEvent (EventCollector .AFTER_COMMIT , data );
760
+ throw new IllegalStateException ("test" );
761
+ }
762
+
763
+ @ EventListener (condition = "'HANDLE_ERROR'.equals(#data)" )
764
+ public void handleImmediately (String data ) {
765
+ handleEvent (EventCollector .HANDLE_ERROR , data );
766
+ }
767
+ }
768
+
769
+ @ Component
770
+ static class AfterRollbackErrorHandlerTestListener extends BaseTransactionalTestListener {
771
+
772
+ @ TransactionalEventListener (phase = AFTER_ROLLBACK , condition = "!'HANDLE_ERROR'.equals(#data)" )
773
+ public void handleBeforeCommit (String data ) {
774
+ handleEvent (EventCollector .AFTER_ROLLBACK , data );
775
+ throw new IllegalStateException ("test" );
776
+ }
777
+
778
+ @ EventListener (condition = "'HANDLE_ERROR'.equals(#data)" )
779
+ public void handleImmediately (String data ) {
780
+ handleEvent (EventCollector .HANDLE_ERROR , data );
781
+ }
782
+ }
783
+
784
+ @ Component
785
+ static class AfterCompletionErrorHandlerTestListener extends BaseTransactionalTestListener {
786
+
787
+ @ TransactionalEventListener (phase = AFTER_COMPLETION , condition = "!'HANDLE_ERROR'.equals(#data)" )
788
+ public void handleBeforeCommit (String data ) {
789
+ handleEvent (EventCollector .AFTER_COMPLETION , data );
790
+ throw new IllegalStateException ("test" );
791
+ }
792
+
793
+ @ EventListener (condition = "'HANDLE_ERROR'.equals(#data)" )
794
+ public void handleImmediately (String data ) {
795
+ handleEvent (EventCollector .HANDLE_ERROR , data );
796
+ }
797
+ }
798
+
680
799
static class EventTransactionSynchronization implements TransactionSynchronization {
681
800
682
801
private final int order ;
@@ -691,4 +810,18 @@ public int getOrder() {
691
810
}
692
811
}
693
812
813
+ static class AfterRollbackErrorHandler extends GlobalTransactionalEventErrorHandler {
814
+
815
+ private final ApplicationEventPublisher eventPublisher ;
816
+
817
+ AfterRollbackErrorHandler (ApplicationEventPublisher eventPublisher ) {
818
+ this .eventPublisher = eventPublisher ;
819
+ }
820
+
821
+ @ Override
822
+ public void handle (ApplicationEvent event , @ Nullable Throwable ex ) {
823
+ eventPublisher .publishEvent ("HANDLE_ERROR" );
824
+ }
825
+ }
826
+
694
827
}
0 commit comments