Skip to content

Commit 308a465

Browse files
authored
fix: PDML retry settings were not applied for aborted tx (#232)
* fix: PDML retry settings were not applied for aborted tx The PartitionedDML retry settings were only applied for the RPC, and not for the generic retryer that would retry the PDML transaction if it was aborted by Spanner. This could cause long-running PDML transactions to fail with an Aborted exception. Fixes #199 * fix: add ignored diff to clirr
1 parent be7d713 commit 308a465

File tree

5 files changed

+38
-11
lines changed

5 files changed

+38
-11
lines changed

google-cloud-spanner/clirr-ignored-differences.xml

+5
Original file line numberDiff line numberDiff line change
@@ -170,5 +170,10 @@
170170
<className>com/google/cloud/spanner/spi/v1/GapicSpannerRpc</className>
171171
<method>com.google.spanner.v1.ResultSet executePartitionedDml(com.google.spanner.v1.ExecuteSqlRequest, java.util.Map, org.threeten.bp.Duration)</method>
172172
</difference>
173+
<difference>
174+
<differenceType>7012</differenceType>
175+
<className>com/google/cloud/spanner/spi/v1/SpannerRpc</className>
176+
<method>com.google.api.gax.retrying.RetrySettings getPartitionedDmlRetrySettings()</method>
177+
</difference>
173178

174179
</differences>

google-cloud-spanner/src/main/java/com/google/cloud/spanner/PartitionedDMLTransaction.java

+2-1
Original file line numberDiff line numberDiff line change
@@ -87,7 +87,8 @@ public com.google.spanner.v1.ResultSet call() throws Exception {
8787
}
8888
};
8989
com.google.spanner.v1.ResultSet resultSet =
90-
SpannerRetryHelper.runTxWithRetriesOnAborted(callable);
90+
SpannerRetryHelper.runTxWithRetriesOnAborted(
91+
callable, rpc.getPartitionedDmlRetrySettings());
9192
if (!resultSet.hasStats()) {
9293
throw new IllegalArgumentException(
9394
"Partitioned DML response missing stats possibly due to non-DML statement as input");

google-cloud-spanner/src/main/java/com/google/cloud/spanner/SpannerRetryHelper.java

+8
Original file line numberDiff line numberDiff line change
@@ -53,6 +53,14 @@ class SpannerRetryHelper {
5353

5454
/** Executes the {@link Callable} and retries if it fails with an {@link AbortedException}. */
5555
static <T> T runTxWithRetriesOnAborted(Callable<T> callable) {
56+
return runTxWithRetriesOnAborted(callable, txRetrySettings);
57+
}
58+
59+
/**
60+
* Executes the {@link Callable} and retries if it fails with an {@link AbortedException} using
61+
* the specific {@link RetrySettings}.
62+
*/
63+
static <T> T runTxWithRetriesOnAborted(Callable<T> callable, RetrySettings retrySettings) {
5664
try {
5765
return RetryHelper.runWithRetries(
5866
callable, txRetrySettings, new TxRetryAlgorithm<>(), NanoClock.getDefaultClock());

google-cloud-spanner/src/main/java/com/google/cloud/spanner/spi/v1/GapicSpannerRpc.java

+20-10
Original file line numberDiff line numberDiff line change
@@ -29,6 +29,7 @@
2929
import com.google.api.gax.grpc.InstantiatingGrpcChannelProvider;
3030
import com.google.api.gax.longrunning.OperationFuture;
3131
import com.google.api.gax.retrying.ResultRetryAlgorithm;
32+
import com.google.api.gax.retrying.RetrySettings;
3233
import com.google.api.gax.retrying.TimedAttemptSettings;
3334
import com.google.api.gax.rpc.AlreadyExistsException;
3435
import com.google.api.gax.rpc.ApiClientHeaderProvider;
@@ -217,6 +218,7 @@ private void awaitTermination() throws InterruptedException {
217218
private boolean rpcIsClosed;
218219
private final SpannerStub spannerStub;
219220
private final SpannerStub partitionedDmlStub;
221+
private final RetrySettings partitionedDmlRetrySettings;
220222
private final InstanceAdminStub instanceAdminStub;
221223
private final DatabaseAdminStubSettings databaseAdminStubSettings;
222224
private final DatabaseAdminStub databaseAdminStub;
@@ -300,7 +302,7 @@ public GapicSpannerRpc(final SpannerOptions options) {
300302

301303
// Set a keepalive time of 120 seconds to help long running
302304
// commit GRPC calls succeed
303-
.setKeepAliveTime(Duration.ofSeconds(GRPC_KEEPALIVE_SECONDS))
305+
.setKeepAliveTime(Duration.ofSeconds(GRPC_KEEPALIVE_SECONDS * 1000))
304306

305307
// Then check if SpannerOptions provides an InterceptorProvider. Create a default
306308
// SpannerInterceptorProvider if none is provided
@@ -336,21 +338,24 @@ public GapicSpannerRpc(final SpannerOptions options) {
336338
.setCredentialsProvider(credentialsProvider)
337339
.setStreamWatchdogProvider(watchdogProvider)
338340
.build());
341+
partitionedDmlRetrySettings =
342+
options
343+
.getSpannerStubSettings()
344+
.executeSqlSettings()
345+
.getRetrySettings()
346+
.toBuilder()
347+
.setInitialRpcTimeout(options.getPartitionedDmlTimeout())
348+
.setMaxRpcTimeout(options.getPartitionedDmlTimeout())
349+
.setTotalTimeout(options.getPartitionedDmlTimeout())
350+
.setRpcTimeoutMultiplier(1.0)
351+
.build();
339352
SpannerStubSettings.Builder pdmlSettings = options.getSpannerStubSettings().toBuilder();
340353
pdmlSettings
341354
.setTransportChannelProvider(channelProvider)
342355
.setCredentialsProvider(credentialsProvider)
343356
.setStreamWatchdogProvider(watchdogProvider)
344357
.executeSqlSettings()
345-
.setRetrySettings(
346-
options
347-
.getSpannerStubSettings()
348-
.executeSqlSettings()
349-
.getRetrySettings()
350-
.toBuilder()
351-
.setInitialRpcTimeout(options.getPartitionedDmlTimeout())
352-
.setMaxRpcTimeout(options.getPartitionedDmlTimeout())
353-
.build());
358+
.setRetrySettings(partitionedDmlRetrySettings);
354359
this.partitionedDmlStub = GrpcSpannerStub.create(pdmlSettings.build());
355360

356361
this.instanceAdminStub =
@@ -1060,6 +1065,11 @@ public ResultSet executePartitionedDml(
10601065
return get(partitionedDmlStub.executeSqlCallable().futureCall(request, context));
10611066
}
10621067

1068+
@Override
1069+
public RetrySettings getPartitionedDmlRetrySettings() {
1070+
return partitionedDmlRetrySettings;
1071+
}
1072+
10631073
@Override
10641074
public StreamingCall executeQuery(
10651075
ExecuteSqlRequest request, ResultStreamConsumer consumer, @Nullable Map<Option, ?> options) {

google-cloud-spanner/src/main/java/com/google/cloud/spanner/spi/v1/SpannerRpc.java

+3
Original file line numberDiff line numberDiff line change
@@ -19,6 +19,7 @@
1919
import com.google.api.core.ApiFuture;
2020
import com.google.api.core.InternalApi;
2121
import com.google.api.gax.longrunning.OperationFuture;
22+
import com.google.api.gax.retrying.RetrySettings;
2223
import com.google.cloud.ServiceRpc;
2324
import com.google.cloud.spanner.SpannerException;
2425
import com.google.cloud.spanner.admin.database.v1.stub.DatabaseAdminStub;
@@ -283,6 +284,8 @@ StreamingCall read(
283284

284285
ResultSet executePartitionedDml(ExecuteSqlRequest request, @Nullable Map<Option, ?> options);
285286

287+
RetrySettings getPartitionedDmlRetrySettings();
288+
286289
StreamingCall executeQuery(
287290
ExecuteSqlRequest request, ResultStreamConsumer consumer, @Nullable Map<Option, ?> options);
288291

0 commit comments

Comments
 (0)