70
70
class TransactionRunnerImpl implements SessionTransaction , TransactionRunner {
71
71
private static final Tracer tracer = Tracing .getTracer ();
72
72
private static final Logger txnLogger = Logger .getLogger (TransactionRunner .class .getName ());
73
+ /**
74
+ * (Part of) the error message that is returned by Cloud Spanner if a transaction is cancelled
75
+ * because it was invalidated by a later transaction in the same session.
76
+ */
77
+ private static final String TRANSACTION_CANCELLED_MESSAGE = "invalidated by a later transaction" ;
73
78
74
79
@ VisibleForTesting
75
80
static class TransactionContextImpl extends AbstractReadContext implements TransactionContext {
@@ -372,8 +377,7 @@ public void run() {
372
377
}
373
378
span .addAnnotation ("Commit Failed" , TraceUtil .getExceptionAnnotations (e ));
374
379
TraceUtil .endSpanWithFailure (opSpan , e );
375
- onError ((SpannerException ) e , false );
376
- res .setException (e );
380
+ res .setException (onError ((SpannerException ) e , false ));
377
381
}
378
382
}
379
383
}),
@@ -519,7 +523,7 @@ public void onTransactionMetadata(Transaction transaction, boolean shouldInclude
519
523
}
520
524
521
525
@ Override
522
- public void onError (SpannerException e , boolean withBeginTransaction ) {
526
+ public SpannerException onError (SpannerException e , boolean withBeginTransaction ) {
523
527
// If the statement that caused an error was the statement that included a BeginTransaction
524
528
// option, we simulate an aborted transaction to force a retry of the entire transaction. This
525
529
// will cause the retry to execute an explicit BeginTransaction RPC and then the actual
@@ -536,21 +540,41 @@ public void onError(SpannerException e, boolean withBeginTransaction) {
536
540
SpannerExceptionFactory .createAbortedExceptionWithRetryDelay (
537
541
"Aborted due to failed initial statement" , e , 0 , 1 )));
538
542
}
543
+ SpannerException exceptionToThrow ;
544
+ if (withBeginTransaction
545
+ && e .getErrorCode () == ErrorCode .CANCELLED
546
+ && e .getMessage ().contains (TRANSACTION_CANCELLED_MESSAGE )) {
547
+ // If the first statement of a transaction fails because it was invalidated by a later
548
+ // transaction, then the transaction should be retried with an explicit BeginTransaction
549
+ // RPC. It could be that this occurred because of a previous transaction that timed out or
550
+ // was cancelled by the client, but that was sent to Cloud Spanner and that was still active
551
+ // on the backend.
552
+ exceptionToThrow =
553
+ SpannerExceptionFactory .newSpannerException (
554
+ ErrorCode .ABORTED ,
555
+ e .getMessage (),
556
+ SpannerExceptionFactory .createAbortedExceptionWithRetryDelay (
557
+ "Aborted due to failed initial statement" , e , 0 , 1 ));
558
+ } else {
559
+ exceptionToThrow = e ;
560
+ }
539
561
540
- if (e .getErrorCode () == ErrorCode .ABORTED ) {
562
+ if (exceptionToThrow .getErrorCode () == ErrorCode .ABORTED ) {
541
563
long delay = -1L ;
542
- if (e instanceof AbortedException ) {
543
- delay = ((AbortedException ) e ).getRetryDelayInMillis ();
564
+ if (exceptionToThrow instanceof AbortedException ) {
565
+ delay = ((AbortedException ) exceptionToThrow ).getRetryDelayInMillis ();
544
566
}
545
567
if (delay == -1L ) {
546
- txnLogger .log (Level .FINE , "Retry duration is missing from the exception." , e );
568
+ txnLogger .log (
569
+ Level .FINE , "Retry duration is missing from the exception." , exceptionToThrow );
547
570
}
548
571
549
572
synchronized (lock ) {
550
573
retryDelayInMillis = delay ;
551
574
aborted = true ;
552
575
}
553
576
}
577
+ return exceptionToThrow ;
554
578
}
555
579
556
580
@ Override
@@ -607,8 +631,8 @@ public long executeUpdate(Statement statement, UpdateOption... options) {
607
631
// For standard DML, using the exact row count.
608
632
return resultSet .getStats ().getRowCountExact ();
609
633
} catch (Throwable t ) {
610
- onError (SpannerExceptionFactory . asSpannerException ( t ), builder . getTransaction (). hasBegin ());
611
- throw t ;
634
+ throw onError (
635
+ SpannerExceptionFactory . asSpannerException ( t ), builder . getTransaction (). hasBegin ()) ;
612
636
}
613
637
}
614
638
@@ -661,8 +685,7 @@ public Long apply(ResultSet input) {
661
685
@ Override
662
686
public Long apply (Throwable input ) {
663
687
SpannerException e = SpannerExceptionFactory .asSpannerException (input );
664
- onError (e , builder .getTransaction ().hasBegin ());
665
- throw e ;
688
+ throw onError (e , builder .getTransaction ().hasBegin ());
666
689
}
667
690
},
668
691
MoreExecutors .directExecutor ());
@@ -730,8 +753,8 @@ public long[] batchUpdate(Iterable<Statement> statements, UpdateOption... option
730
753
}
731
754
return results ;
732
755
} catch (Throwable e ) {
733
- onError (SpannerExceptionFactory . asSpannerException ( e ), builder . getTransaction (). hasBegin ());
734
- throw e ;
756
+ throw onError (
757
+ SpannerExceptionFactory . asSpannerException ( e ), builder . getTransaction (). hasBegin ()) ;
735
758
}
736
759
}
737
760
@@ -788,8 +811,7 @@ public long[] apply(ExecuteBatchDmlResponse batchDmlResponse) {
788
811
@ Override
789
812
public long [] apply (Throwable input ) {
790
813
SpannerException e = SpannerExceptionFactory .asSpannerException (input );
791
- onError (e , builder .getTransaction ().hasBegin ());
792
- throw e ;
814
+ throw onError (e , builder .getTransaction ().hasBegin ());
793
815
}
794
816
},
795
817
MoreExecutors .directExecutor ());
0 commit comments