From edc5bbf0d9d4faf48fd9a8d479d5bc5de938c82d Mon Sep 17 00:00:00 2001 From: Arpan Mishra Date: Mon, 6 Feb 2023 13:29:38 +0530 Subject: [PATCH 01/16] fix: prevent illegal negative timeout values into thread sleep() method while retrying exceptions in unit tests. * For details on issue see - /~https://github.com/googleapis/java-spanner/issues/2206 --- .../com/google/cloud/spanner/it/ITClosedSessionTest.java | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/google-cloud-spanner/src/test/java/com/google/cloud/spanner/it/ITClosedSessionTest.java b/google-cloud-spanner/src/test/java/com/google/cloud/spanner/it/ITClosedSessionTest.java index aeb0256285b..227611a10de 100644 --- a/google-cloud-spanner/src/test/java/com/google/cloud/spanner/it/ITClosedSessionTest.java +++ b/google-cloud-spanner/src/test/java/com/google/cloud/spanner/it/ITClosedSessionTest.java @@ -251,7 +251,10 @@ public void testTransactionManager() throws InterruptedException { break; } } catch (AbortedException e) { - Thread.sleep(e.getRetryDelayInMillis()); + long retryDelayInMillis = e.getRetryDelayInMillis(); + if(retryDelayInMillis > 0) { + Thread.sleep(retryDelayInMillis); + } txn = manager.resetForRetry(); } } From 4cd497b05eab3e3b6b89b582bfafde80d42c1518 Mon Sep 17 00:00:00 2001 From: Arpan Mishra Date: Wed, 8 Feb 2023 15:27:18 +0530 Subject: [PATCH 02/16] Fixing lint issues. --- .../java/com/google/cloud/spanner/it/ITClosedSessionTest.java | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/google-cloud-spanner/src/test/java/com/google/cloud/spanner/it/ITClosedSessionTest.java b/google-cloud-spanner/src/test/java/com/google/cloud/spanner/it/ITClosedSessionTest.java index 227611a10de..efbffcfa899 100644 --- a/google-cloud-spanner/src/test/java/com/google/cloud/spanner/it/ITClosedSessionTest.java +++ b/google-cloud-spanner/src/test/java/com/google/cloud/spanner/it/ITClosedSessionTest.java @@ -252,7 +252,7 @@ public void testTransactionManager() throws InterruptedException { } } catch (AbortedException e) { long retryDelayInMillis = e.getRetryDelayInMillis(); - if(retryDelayInMillis > 0) { + if (retryDelayInMillis > 0) { Thread.sleep(retryDelayInMillis); } txn = manager.resetForRetry(); From 3269c6b0dac3f7f774b25486485ddec0a24d7dbe Mon Sep 17 00:00:00 2001 From: Arpan Mishra Date: Wed, 27 Dec 2023 19:36:52 +0530 Subject: [PATCH 03/16] feat: add support for Directed Read options. --- .../cloud/spanner/AbstractReadContext.java | 6 +++ .../com/google/cloud/spanner/Options.java | 49 ++++++++++++++++++- .../cloud/spanner/spi/v1/GapicSpannerRpc.java | 1 + .../google/cloud/spanner/it/ITReadTest.java | 34 +++++++++++++ 4 files changed, 88 insertions(+), 2 deletions(-) diff --git a/google-cloud-spanner/src/main/java/com/google/cloud/spanner/AbstractReadContext.java b/google-cloud-spanner/src/main/java/com/google/cloud/spanner/AbstractReadContext.java index a0b25cb64c0..089a82d960a 100644 --- a/google-cloud-spanner/src/main/java/com/google/cloud/spanner/AbstractReadContext.java +++ b/google-cloud-spanner/src/main/java/com/google/cloud/spanner/AbstractReadContext.java @@ -623,6 +623,9 @@ ExecuteSqlRequest.Builder getExecuteSqlRequestBuilder( if (options.hasDataBoostEnabled()) { builder.setDataBoostEnabled(options.dataBoostEnabled()); } + if (options.hasDirectedReadOptions()) { + builder.setDirectedReadOptions(options.directedReadOptions()); + } builder.setSeqno(getSeqNo()); builder.setQueryOptions(buildQueryOptions(statement.getQueryOptions())); builder.setRequestOptions(buildRequestOptions(options)); @@ -811,6 +814,9 @@ ResultSet readInternalWithOptions( if (readOptions.hasDataBoostEnabled()) { builder.setDataBoostEnabled(readOptions.dataBoostEnabled()); } + if (readOptions.hasDirectedReadOptions()) { + builder.setDirectedReadOptions(readOptions.directedReadOptions()); + } final int prefetchChunks = readOptions.hasPrefetchChunks() ? readOptions.prefetchChunks() : defaultPrefetchChunks; ResumableStreamIterator stream = diff --git a/google-cloud-spanner/src/main/java/com/google/cloud/spanner/Options.java b/google-cloud-spanner/src/main/java/com/google/cloud/spanner/Options.java index 2bd35ec7853..3f61367a9e1 100644 --- a/google-cloud-spanner/src/main/java/com/google/cloud/spanner/Options.java +++ b/google-cloud-spanner/src/main/java/com/google/cloud/spanner/Options.java @@ -17,6 +17,7 @@ package com.google.cloud.spanner; import com.google.common.base.Preconditions; +import com.google.spanner.v1.DirectedReadOptions; import com.google.spanner.v1.RequestOptions.Priority; import java.io.Serializable; import java.util.Objects; @@ -224,6 +225,19 @@ public static CreateUpdateDeleteAdminApiOption validateOnly(Boolean validateOnly return new ValidateOnlyOption(validateOnly); } + /** + * Option to request DirectedRead for ReadOnlyTransaction and SingleUseTransaction. + * + *

The DirectedReadOptions can be used to indicate which replicas or regions should be used for + * non-transactional reads or queries. Not all requests can be sent to non-leader replicas. In + * particular, some requests such as reads within read-write transactions must be sent to a + * designated leader replica. These requests ignore DirectedReadOptions. + */ + public static ReadAndQueryOption directedRead(DirectedReadOptions directedReadOptions) { + return new DirectedReadOption(directedReadOptions); + } + + /** Option to request {@link CommitStats} for read/write transactions. */ static final class CommitStatsOption extends InternalOption implements TransactionOption { @Override @@ -231,7 +245,6 @@ void appendToOptions(Options options) { options.withCommitStats = true; } } - static final CommitStatsOption COMMIT_STATS_OPTION = new CommitStatsOption(); /** Option to request Optimistic Concurrency Control for read/write transactions. */ @@ -325,6 +338,22 @@ void appendToOptions(Options options) { } } + static final class DirectedReadOption extends InternalOption implements ReadAndQueryOption { + private final DirectedReadOptions directedReadOptions; + + DirectedReadOption(DirectedReadOptions directedReadOptions) { + this.directedReadOptions = + Preconditions.checkNotNull( + directedReadOptions, "DirectedReadOptions cannot be null"); + ; + } + + @Override + void appendToOptions(Options options) { + options.directedReadOptions = directedReadOptions; + } + } + private boolean withCommitStats; private Long limit; private Integer prefetchChunks; @@ -338,6 +367,7 @@ void appendToOptions(Options options) { private Boolean validateOnly; private Boolean withOptimisticLock; private Boolean dataBoostEnabled; + private DirectedReadOptions directedReadOptions; // Construction is via factory methods below. private Options() {} @@ -438,6 +468,14 @@ Boolean dataBoostEnabled() { return dataBoostEnabled; } + boolean hasDirectedReadOptions() { + return directedReadOptions != null; + } + + DirectedReadOptions directedReadOptions() { + return directedReadOptions; + } + @Override public String toString() { StringBuilder b = new StringBuilder(); @@ -477,6 +515,9 @@ public String toString() { if (dataBoostEnabled != null) { b.append("dataBoostEnabled: ").append(dataBoostEnabled).append(' '); } + if (directedReadOptions != null) { + b.append("directedReadOptions: ").append(directedReadOptions).append(' '); + } return b.toString(); } @@ -512,7 +553,8 @@ public boolean equals(Object o) { && Objects.equals(etag(), that.etag()) && Objects.equals(validateOnly(), that.validateOnly()) && Objects.equals(withOptimisticLock(), that.withOptimisticLock()) - && Objects.equals(dataBoostEnabled(), that.dataBoostEnabled()); + && Objects.equals(dataBoostEnabled(), that.dataBoostEnabled()) + && Objects.equals(directedReadOptions(), that.directedReadOptions()); } @Override @@ -557,6 +599,9 @@ public int hashCode() { if (dataBoostEnabled != null) { result = 31 * result + dataBoostEnabled.hashCode(); } + if (directedReadOptions != null) { + result = 31 * result + directedReadOptions.hashCode(); + } return result; } diff --git a/google-cloud-spanner/src/main/java/com/google/cloud/spanner/spi/v1/GapicSpannerRpc.java b/google-cloud-spanner/src/main/java/com/google/cloud/spanner/spi/v1/GapicSpannerRpc.java index d75b6636a56..f5003948cdf 100644 --- a/google-cloud-spanner/src/main/java/com/google/cloud/spanner/spi/v1/GapicSpannerRpc.java +++ b/google-cloud-spanner/src/main/java/com/google/cloud/spanner/spi/v1/GapicSpannerRpc.java @@ -165,6 +165,7 @@ import com.google.spanner.v1.CommitResponse; import com.google.spanner.v1.CreateSessionRequest; import com.google.spanner.v1.DeleteSessionRequest; +import com.google.spanner.v1.DirectedReadOptions; import com.google.spanner.v1.ExecuteBatchDmlRequest; import com.google.spanner.v1.ExecuteBatchDmlResponse; import com.google.spanner.v1.ExecuteSqlRequest; diff --git a/google-cloud-spanner/src/test/java/com/google/cloud/spanner/it/ITReadTest.java b/google-cloud-spanner/src/test/java/com/google/cloud/spanner/it/ITReadTest.java index c28b48c529a..2d888996465 100644 --- a/google-cloud-spanner/src/test/java/com/google/cloud/spanner/it/ITReadTest.java +++ b/google-cloud-spanner/src/test/java/com/google/cloud/spanner/it/ITReadTest.java @@ -21,6 +21,9 @@ import static com.google.cloud.spanner.testing.EmulatorSpannerHelper.isUsingEmulator; import static com.google.common.truth.Truth.assertThat; import static com.google.common.truth.Truth.assertWithMessage; +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertFalse; +import static org.junit.Assert.assertTrue; import static org.junit.Assert.fail; import com.google.cloud.spanner.Database; @@ -42,6 +45,9 @@ import com.google.cloud.spanner.Type; import com.google.cloud.spanner.connection.ConnectionOptions; import com.google.cloud.spanner.testing.RemoteSpannerHelper; +import com.google.spanner.v1.DirectedReadOptions; +import com.google.spanner.v1.DirectedReadOptions.IncludeReplicas; +import com.google.spanner.v1.DirectedReadOptions.ReplicaSelection; import io.grpc.Context; import java.util.ArrayList; import java.util.Arrays; @@ -77,6 +83,17 @@ public class ITReadTest { private static final Type TABLE_TYPE = Type.struct( StructField.of("key", Type.string()), StructField.of("stringvalue", Type.string())); + private static DirectedReadOptions DIRECTED_READ_OPTIONS = + DirectedReadOptions.newBuilder() + .setIncludeReplicas( + IncludeReplicas.newBuilder() + .addReplicaSelections( + ReplicaSelection.newBuilder() + .setLocation("us-west1") + .setType(ReplicaSelection.Type.READ_ONLY) + .build()) + .setAutoFailoverDisabled(true)) + .build(); private static DatabaseClient googleStandardSQLClient; private static DatabaseClient postgreSQLClient; @@ -336,6 +353,23 @@ public void rowsAreSnapshots() { assertThat(rows.get(2).getString(1)).isEqualTo("v4"); } + @Test + public void pointReadWithDirectedReadOptions() { + try (ResultSet rs = + getClient(dialect.dialect) + .singleUse() + .read( + TABLE_NAME, + KeySet.singleKey(Key.of("k1")), + ALL_COLUMNS, + Options.directedRead(DIRECTED_READ_OPTIONS))) { + assertTrue(rs.next()); + assertEquals("k1", rs.getString(0)); + assertEquals("v1", rs.getString(1)); + assertFalse(rs.next()); + } + } + @Test public void invalidDatabase() { RemoteSpannerHelper helper = env.getTestHelper(); From 8725334476e5178181b5b9e68287a50d0b72fd88 Mon Sep 17 00:00:00 2001 From: Arpan Mishra Date: Wed, 27 Dec 2023 19:37:57 +0530 Subject: [PATCH 04/16] chore: fix lint issues. --- .../src/main/java/com/google/cloud/spanner/Options.java | 5 ++--- .../com/google/cloud/spanner/spi/v1/GapicSpannerRpc.java | 1 - 2 files changed, 2 insertions(+), 4 deletions(-) diff --git a/google-cloud-spanner/src/main/java/com/google/cloud/spanner/Options.java b/google-cloud-spanner/src/main/java/com/google/cloud/spanner/Options.java index 3f61367a9e1..dda12b60d64 100644 --- a/google-cloud-spanner/src/main/java/com/google/cloud/spanner/Options.java +++ b/google-cloud-spanner/src/main/java/com/google/cloud/spanner/Options.java @@ -237,7 +237,6 @@ public static ReadAndQueryOption directedRead(DirectedReadOptions directedReadOp return new DirectedReadOption(directedReadOptions); } - /** Option to request {@link CommitStats} for read/write transactions. */ static final class CommitStatsOption extends InternalOption implements TransactionOption { @Override @@ -245,6 +244,7 @@ void appendToOptions(Options options) { options.withCommitStats = true; } } + static final CommitStatsOption COMMIT_STATS_OPTION = new CommitStatsOption(); /** Option to request Optimistic Concurrency Control for read/write transactions. */ @@ -343,8 +343,7 @@ static final class DirectedReadOption extends InternalOption implements ReadAndQ DirectedReadOption(DirectedReadOptions directedReadOptions) { this.directedReadOptions = - Preconditions.checkNotNull( - directedReadOptions, "DirectedReadOptions cannot be null"); + Preconditions.checkNotNull(directedReadOptions, "DirectedReadOptions cannot be null"); ; } diff --git a/google-cloud-spanner/src/main/java/com/google/cloud/spanner/spi/v1/GapicSpannerRpc.java b/google-cloud-spanner/src/main/java/com/google/cloud/spanner/spi/v1/GapicSpannerRpc.java index f5003948cdf..d75b6636a56 100644 --- a/google-cloud-spanner/src/main/java/com/google/cloud/spanner/spi/v1/GapicSpannerRpc.java +++ b/google-cloud-spanner/src/main/java/com/google/cloud/spanner/spi/v1/GapicSpannerRpc.java @@ -165,7 +165,6 @@ import com.google.spanner.v1.CommitResponse; import com.google.spanner.v1.CreateSessionRequest; import com.google.spanner.v1.DeleteSessionRequest; -import com.google.spanner.v1.DirectedReadOptions; import com.google.spanner.v1.ExecuteBatchDmlRequest; import com.google.spanner.v1.ExecuteBatchDmlResponse; import com.google.spanner.v1.ExecuteSqlRequest; From 23113842eaa11ab971577d8374aed9d6c7b79951 Mon Sep 17 00:00:00 2001 From: Arpan Mishra Date: Wed, 27 Dec 2023 19:59:08 +0530 Subject: [PATCH 05/16] test: add unit tests for options class. --- .../com/google/cloud/spanner/OptionsTest.java | 75 +++++++++++++++++-- 1 file changed, 69 insertions(+), 6 deletions(-) diff --git a/google-cloud-spanner/src/test/java/com/google/cloud/spanner/OptionsTest.java b/google-cloud-spanner/src/test/java/com/google/cloud/spanner/OptionsTest.java index d40f9b39ea1..b75f8a21327 100644 --- a/google-cloud-spanner/src/test/java/com/google/cloud/spanner/OptionsTest.java +++ b/google-cloud-spanner/src/test/java/com/google/cloud/spanner/OptionsTest.java @@ -24,6 +24,9 @@ import static org.junit.Assert.assertTrue; import com.google.cloud.spanner.Options.RpcPriority; +import com.google.spanner.v1.DirectedReadOptions; +import com.google.spanner.v1.DirectedReadOptions.IncludeReplicas; +import com.google.spanner.v1.DirectedReadOptions.ReplicaSelection; import com.google.spanner.v1.RequestOptions.Priority; import org.junit.Test; import org.junit.runner.RunWith; @@ -32,6 +35,13 @@ /** Unit tests for {@link Options}. */ @RunWith(JUnit4.class) public class OptionsTest { + private static final DirectedReadOptions DIRECTED_READ_OPTIONS = + DirectedReadOptions.newBuilder() + .setIncludeReplicas( + IncludeReplicas.newBuilder() + .addReplicaSelections( + ReplicaSelection.newBuilder().setLocation("us-west1").build())) + .build(); @Test public void negativeLimitsNotAllowed() { @@ -65,13 +75,18 @@ public void zeroPrefetchChunksNotAllowed() { public void allOptionsPresent() { Options options = Options.fromReadOptions( - Options.limit(10), Options.prefetchChunks(1), Options.dataBoostEnabled(true)); + Options.limit(10), + Options.prefetchChunks(1), + Options.dataBoostEnabled(true), + Options.directedRead(DIRECTED_READ_OPTIONS)); assertThat(options.hasLimit()).isTrue(); assertThat(options.limit()).isEqualTo(10); assertThat(options.hasPrefetchChunks()).isTrue(); assertThat(options.prefetchChunks()).isEqualTo(1); assertThat(options.hasDataBoostEnabled()).isTrue(); assertTrue(options.dataBoostEnabled()); + assertTrue(options.hasDirectedReadOptions()); + assertEquals(DIRECTED_READ_OPTIONS, options.directedReadOptions()); } @Test @@ -84,6 +99,7 @@ public void allOptionsAbsent() { assertThat(options.hasPriority()).isFalse(); assertThat(options.hasTag()).isFalse(); assertThat(options.hasDataBoostEnabled()).isFalse(); + assertThat(options.hasDirectedReadOptions()).isFalse(); assertThat(options.toString()).isEqualTo(""); assertThat(options.equals(options)).isTrue(); assertThat(options.equals(null)).isFalse(); @@ -161,14 +177,29 @@ public void readOptionsTest() { boolean dataBoost = true; Options options = Options.fromReadOptions( - Options.limit(limit), Options.tag(tag), Options.dataBoostEnabled(true)); + Options.limit(limit), + Options.tag(tag), + Options.dataBoostEnabled(true), + Options.directedRead(DIRECTED_READ_OPTIONS)); assertThat(options.toString()) .isEqualTo( - "limit: " + limit + " " + "tag: " + tag + " " + "dataBoostEnabled: " + dataBoost + " "); + "limit: " + + limit + + " " + + "tag: " + + tag + + " " + + "dataBoostEnabled: " + + dataBoost + + " " + + "directedReadOptions: " + + DIRECTED_READ_OPTIONS + + " "); assertThat(options.tag()).isEqualTo(tag); assertEquals(dataBoost, options.dataBoostEnabled()); - assertThat(options.hashCode()).isEqualTo(-96091607); + assertEquals(DIRECTED_READ_OPTIONS, options.directedReadOptions()); + assertThat(options.hashCode()).isEqualTo(-1667681729); } @Test @@ -199,7 +230,10 @@ public void queryOptionsTest() { boolean dataBoost = true; Options options = Options.fromQueryOptions( - Options.prefetchChunks(chunks), Options.tag(tag), Options.dataBoostEnabled(true)); + Options.prefetchChunks(chunks), + Options.tag(tag), + Options.dataBoostEnabled(true), + Options.directedRead(DIRECTED_READ_OPTIONS)); assertThat(options.toString()) .isEqualTo( "prefetchChunks: " @@ -210,11 +244,15 @@ public void queryOptionsTest() { + " " + "dataBoostEnabled: " + dataBoost + + " " + + "directedReadOptions: " + + DIRECTED_READ_OPTIONS + " "); assertThat(options.prefetchChunks()).isEqualTo(chunks); assertThat(options.tag()).isEqualTo(tag); assertEquals(dataBoost, options.dataBoostEnabled()); - assertThat(options.hashCode()).isEqualTo(1274581983); + assertEquals(DIRECTED_READ_OPTIONS, options.directedReadOptions()); + assertThat(options.hashCode()).isEqualTo(-187561950); } @Test @@ -630,4 +668,29 @@ public void optimisticLockHashCode() { assertEquals(option1.hashCode(), option2.hashCode()); assertNotEquals(option1.hashCode(), option3.hashCode()); } + + @Test + public void directedReadEquality() { + Options option1 = Options.fromReadOptions(Options.directedRead(DIRECTED_READ_OPTIONS)); + Options option2 = Options.fromReadOptions(Options.directedRead(DIRECTED_READ_OPTIONS)); + Options option3 = Options.fromTransactionOptions(); + + assertEquals(option1, option2); + assertNotEquals(option1, option3); + } + + @Test + public void directedReadHashCode() { + Options option1 = Options.fromReadOptions(Options.directedRead(DIRECTED_READ_OPTIONS)); + Options option2 = Options.fromReadOptions(Options.directedRead(DIRECTED_READ_OPTIONS)); + Options option3 = Options.fromTransactionOptions(); + + assertEquals(option1.hashCode(), option2.hashCode()); + assertNotEquals(option1.hashCode(), option3.hashCode()); + } + + @Test + public void directedReadsNullNotAllowed() { + assertThrows(NullPointerException.class, () -> Options.directedRead(null)); + } } From bf5e6e0f98336f952bab464f485f02e6f81018ad Mon Sep 17 00:00:00 2001 From: Arpan Mishra Date: Wed, 27 Dec 2023 20:09:27 +0530 Subject: [PATCH 06/16] test: add tests using mock spanner. --- .../cloud/spanner/DatabaseClientImplTest.java | 55 +++++++++++++++++++ 1 file changed, 55 insertions(+) diff --git a/google-cloud-spanner/src/test/java/com/google/cloud/spanner/DatabaseClientImplTest.java b/google-cloud-spanner/src/test/java/com/google/cloud/spanner/DatabaseClientImplTest.java index aea8a4dcb64..944209f682a 100644 --- a/google-cloud-spanner/src/test/java/com/google/cloud/spanner/DatabaseClientImplTest.java +++ b/google-cloud-spanner/src/test/java/com/google/cloud/spanner/DatabaseClientImplTest.java @@ -75,6 +75,9 @@ import com.google.spanner.v1.BatchWriteResponse; import com.google.spanner.v1.CommitRequest; import com.google.spanner.v1.DeleteSessionRequest; +import com.google.spanner.v1.DirectedReadOptions; +import com.google.spanner.v1.DirectedReadOptions.IncludeReplicas; +import com.google.spanner.v1.DirectedReadOptions.ReplicaSelection; import com.google.spanner.v1.ExecuteBatchDmlRequest; import com.google.spanner.v1.ExecuteSqlRequest; import com.google.spanner.v1.ExecuteSqlRequest.QueryMode; @@ -169,6 +172,20 @@ public class DatabaseClientImplTest { .setStatus(STATUS_OK) .addAllIndexes(ImmutableList.of(2, 3)) .build()); + private static final DirectedReadOptions DIRECTED_READ_OPTIONS1 = + DirectedReadOptions.newBuilder() + .setIncludeReplicas( + IncludeReplicas.newBuilder() + .addReplicaSelections( + ReplicaSelection.newBuilder().setLocation("us-west1").build())) + .build(); + private static final DirectedReadOptions DIRECTED_READ_OPTIONS2 = + DirectedReadOptions.newBuilder() + .setIncludeReplicas( + IncludeReplicas.newBuilder() + .addReplicaSelections( + ReplicaSelection.newBuilder().setLocation("us-east1").build())) + .build(); private Spanner spanner; private Spanner spannerWithEmptySessionPool; private static ExecutorService executor; @@ -1518,6 +1535,22 @@ public void testExecuteQueryWithTag() { assertThat(request.getRequestOptions().getTransactionTag()).isEmpty(); } + @Test + public void testExecuteQueryWithDirectedReadOptions() { + DatabaseClient client = + spanner.getDatabaseClient(DatabaseId.of(TEST_PROJECT, TEST_INSTANCE, TEST_DATABASE)); + try (ResultSet resultSet = + client.singleUse().executeQuery(SELECT1, Options.directedRead(DIRECTED_READ_OPTIONS1))) { + while (resultSet.next()) {} + } + + List requests = mockSpanner.getRequestsOfType(ExecuteSqlRequest.class); + assertEquals(1, requests.size()); + ExecuteSqlRequest request = requests.get(0); + assertTrue(request.hasDirectedReadOptions()); + assertEquals(DIRECTED_READ_OPTIONS1, request.getDirectedReadOptions()); + } + @Test public void testExecuteReadWithTag() { DatabaseClient client = @@ -1542,6 +1575,28 @@ public void testExecuteReadWithTag() { assertThat(request.getRequestOptions().getTransactionTag()).isEmpty(); } + @Test + public void testExecuteReadWithDirectedReadOptions() { + DatabaseClient client = + spanner.getDatabaseClient(DatabaseId.of(TEST_PROJECT, TEST_INSTANCE, TEST_DATABASE)); + try (ResultSet resultSet = + client + .singleUse() + .read( + READ_TABLE_NAME, + KeySet.singleKey(Key.of(1L)), + READ_COLUMN_NAMES, + Options.directedRead(DIRECTED_READ_OPTIONS1))) { + while (resultSet.next()) {} + } + + List requests = mockSpanner.getRequestsOfType(ReadRequest.class); + assertEquals(1, requests.size()); + ReadRequest request = requests.get(0); + assertTrue(request.hasDirectedReadOptions()); + assertEquals(DIRECTED_READ_OPTIONS1, request.getDirectedReadOptions()); + } + @Test public void testReadWriteExecuteQueryWithTag() { DatabaseClient client = From 5a3427e29ff588af18a945d309d624537f8ed52b Mon Sep 17 00:00:00 2001 From: Arpan Mishra Date: Thu, 28 Dec 2023 00:15:18 +0530 Subject: [PATCH 07/16] test: add unit test for partitioned read. --- .../java/com/google/cloud/spanner/DatabaseClientImplTest.java | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/google-cloud-spanner/src/test/java/com/google/cloud/spanner/DatabaseClientImplTest.java b/google-cloud-spanner/src/test/java/com/google/cloud/spanner/DatabaseClientImplTest.java index 944209f682a..2cd32afc4f3 100644 --- a/google-cloud-spanner/src/test/java/com/google/cloud/spanner/DatabaseClientImplTest.java +++ b/google-cloud-spanner/src/test/java/com/google/cloud/spanner/DatabaseClientImplTest.java @@ -2890,7 +2890,8 @@ public void testBackendPartitionQueryOptions() { .setOptimizerVersion("1") .setOptimizerStatisticsPackage("custom-package") .build()) - .build()); + .build(), + Options.directedRead(DIRECTED_READ_OPTIONS1)); try (ResultSet rs = transaction.execute(partitions.get(0))) { // Just iterate over the results to execute the query. while (rs.next()) {} @@ -2909,6 +2910,7 @@ public void testBackendPartitionQueryOptions() { assertThat(executeSqlRequest.getQueryOptions().getOptimizerVersion()).isEqualTo("1"); assertThat(executeSqlRequest.getQueryOptions().getOptimizerStatisticsPackage()) .isEqualTo("custom-package"); + assertThat(executeSqlRequest.getDirectedReadOptions()).isEqualTo(DIRECTED_READ_OPTIONS1); } } From 5c8028ea2c6dc99ee319f674a72283278adc0212 Mon Sep 17 00:00:00 2001 From: Arpan Mishra Date: Thu, 28 Dec 2023 11:56:08 +0530 Subject: [PATCH 08/16] test: add unit test for partitioned read. --- .../cloud/spanner/DatabaseClientImplTest.java | 51 ++++++++++++++++--- .../cloud/spanner/MockSpannerTestUtil.java | 1 + 2 files changed, 45 insertions(+), 7 deletions(-) diff --git a/google-cloud-spanner/src/test/java/com/google/cloud/spanner/DatabaseClientImplTest.java b/google-cloud-spanner/src/test/java/com/google/cloud/spanner/DatabaseClientImplTest.java index 2cd32afc4f3..09bad8b717f 100644 --- a/google-cloud-spanner/src/test/java/com/google/cloud/spanner/DatabaseClientImplTest.java +++ b/google-cloud-spanner/src/test/java/com/google/cloud/spanner/DatabaseClientImplTest.java @@ -21,6 +21,7 @@ import static com.google.cloud.spanner.MockSpannerTestUtil.READ_ONE_KEY_VALUE_STATEMENT; import static com.google.cloud.spanner.MockSpannerTestUtil.READ_TABLE_NAME; import static com.google.cloud.spanner.MockSpannerTestUtil.SELECT1; +import static com.google.cloud.spanner.MockSpannerTestUtil.SELECT1_FROM_TABLE; import static com.google.cloud.spanner.MockSpannerTestUtil.SELECT1_RESULTSET; import static com.google.cloud.spanner.SpannerApiFutures.get; import static com.google.common.truth.Truth.assertThat; @@ -179,13 +180,6 @@ public class DatabaseClientImplTest { .addReplicaSelections( ReplicaSelection.newBuilder().setLocation("us-west1").build())) .build(); - private static final DirectedReadOptions DIRECTED_READ_OPTIONS2 = - DirectedReadOptions.newBuilder() - .setIncludeReplicas( - IncludeReplicas.newBuilder() - .addReplicaSelections( - ReplicaSelection.newBuilder().setLocation("us-east1").build())) - .build(); private Spanner spanner; private Spanner spannerWithEmptySessionPool; private static ExecutorService executor; @@ -203,6 +197,8 @@ public static void startStaticServer() throws IOException { StatementResult.exception( INVALID_UPDATE_STATEMENT, Status.INVALID_ARGUMENT.withDescription("invalid statement").asRuntimeException())); + mockSpanner.putStatementResult( + StatementResult.query(SELECT1_FROM_TABLE, MockSpannerTestUtil.SELECT1_RESULTSET)); mockSpanner.setBatchWriteResult(BATCH_WRITE_RESPONSES); executor = Executors.newSingleThreadExecutor(); @@ -2914,6 +2910,47 @@ public void testBackendPartitionQueryOptions() { } } + @Test + public void testBackendPartitionReadOptions() { + // Use a Spanner instance with MinSession=0 and WriteFraction=0.0 to prevent background requests + // from the session pool interfering with the test case. + try (Spanner spanner = + SpannerOptions.newBuilder() + .setProjectId("[PROJECT]") + .setChannelProvider(channelProvider) + .setCredentials(NoCredentials.getInstance()) + .setSessionPoolOption(SessionPoolOptions.newBuilder().setMinSessions(0).build()) + .build() + .getService()) { + BatchClient client = + spanner.getBatchClient(DatabaseId.of("[PROJECT]", "[INSTANCE]", "[DATABASE")); + BatchReadOnlyTransaction transaction = + client.batchReadOnlyTransaction(TimestampBound.strong()); + List partitions = + transaction.partitionRead( + PartitionOptions.newBuilder().setMaxPartitions(10L).build(), + "FOO", + KeySet.all(), + Lists.newArrayList("1"), + Options.directedRead(DIRECTED_READ_OPTIONS1)); + try (ResultSet rs = transaction.execute(partitions.get(0))) { + // Just iterate over the results to execute the query. + while (rs.next()) {} + } finally { + transaction.cleanup(); + } + // Check if the last query executed is a DeleteSessionRequest and the second last query + // executed is a ExecuteSqlRequest and was executed using a custom optimizer version and + // statistics package. + List requests = mockSpanner.getRequests(); + assert requests.size() >= 2 : "required to have at least 2 requests"; + assertThat(requests.get(requests.size() - 1)).isInstanceOf(DeleteSessionRequest.class); + assertThat(requests.get(requests.size() - 2)).isInstanceOf(ReadRequest.class); + ReadRequest readRequest = (ReadRequest) requests.get(requests.size() - 2); + assertThat(readRequest.getDirectedReadOptions()).isEqualTo(DIRECTED_READ_OPTIONS1); + } + } + @Test public void testAsyncQuery() throws Exception { final int EXPECTED_ROW_COUNT = 10; diff --git a/google-cloud-spanner/src/test/java/com/google/cloud/spanner/MockSpannerTestUtil.java b/google-cloud-spanner/src/test/java/com/google/cloud/spanner/MockSpannerTestUtil.java index af336d3f582..83bb1728ac0 100644 --- a/google-cloud-spanner/src/test/java/com/google/cloud/spanner/MockSpannerTestUtil.java +++ b/google-cloud-spanner/src/test/java/com/google/cloud/spanner/MockSpannerTestUtil.java @@ -49,6 +49,7 @@ public class MockSpannerTestUtil { .build()) .setMetadata(SELECT1_METADATA) .build(); + public static final Statement SELECT1_FROM_TABLE = Statement.of("SELECT 1 FROM FOO WHERE 1=1"); static final String TEST_PROJECT = "my-project"; static final String TEST_INSTANCE = "my-instance"; From 12e93a709b492135bab0f1b61fbd41205ebfc268 Mon Sep 17 00:00:00 2001 From: Arpan Mishra Date: Fri, 29 Dec 2023 17:35:09 +0530 Subject: [PATCH 09/16] chore: adding option in spanner options. --- .../cloud/spanner/AbstractReadContext.java | 13 +++ .../com/google/cloud/spanner/SessionImpl.java | 3 + .../google/cloud/spanner/SpannerOptions.java | 23 ++++ .../spanner/AbstractReadContextTest.java | 22 ++++ .../cloud/spanner/DatabaseClientImplTest.java | 81 ++++++++++++++ .../com/google/cloud/spanner/OptionsTest.java | 2 - .../cloud/spanner/SpannerOptionsTest.java | 24 +++++ .../cloud/spanner/it/ITDirectedReadsTest.java | 100 ++++++++++++++++++ 8 files changed, 266 insertions(+), 2 deletions(-) create mode 100644 google-cloud-spanner/src/test/java/com/google/cloud/spanner/it/ITDirectedReadsTest.java diff --git a/google-cloud-spanner/src/main/java/com/google/cloud/spanner/AbstractReadContext.java b/google-cloud-spanner/src/main/java/com/google/cloud/spanner/AbstractReadContext.java index 089a82d960a..200f0fe4168 100644 --- a/google-cloud-spanner/src/main/java/com/google/cloud/spanner/AbstractReadContext.java +++ b/google-cloud-spanner/src/main/java/com/google/cloud/spanner/AbstractReadContext.java @@ -42,6 +42,7 @@ import com.google.common.util.concurrent.MoreExecutors; import com.google.protobuf.ByteString; import com.google.spanner.v1.BeginTransactionRequest; +import com.google.spanner.v1.DirectedReadOptions; import com.google.spanner.v1.ExecuteBatchDmlRequest; import com.google.spanner.v1.ExecuteSqlRequest; import com.google.spanner.v1.ExecuteSqlRequest.QueryMode; @@ -72,6 +73,7 @@ abstract static class Builder, T extends AbstractReadCon private Span span = Tracing.getTracer().getCurrentSpan(); private int defaultPrefetchChunks = SpannerOptions.Builder.DEFAULT_PREFETCH_CHUNKS; private QueryOptions defaultQueryOptions = SpannerOptions.Builder.DEFAULT_QUERY_OPTIONS; + private DirectedReadOptions defaultDirectedReadOption; private ExecutorProvider executorProvider; private Clock clock = new Clock(); @@ -117,6 +119,11 @@ B setClock(Clock clock) { return self(); } + B setDefaultDirectedReadOption(DirectedReadOptions directedReadOptions) { + this.defaultDirectedReadOption = Preconditions.checkNotNull(directedReadOptions); + return self(); + } + abstract T build(); } @@ -399,6 +406,7 @@ void initTransaction() { private final int defaultPrefetchChunks; private final QueryOptions defaultQueryOptions; + private final DirectedReadOptions defaultDirectedReadOptions; private final Clock clock; @GuardedBy("lock") @@ -423,6 +431,7 @@ void initTransaction() { this.rpc = builder.rpc; this.defaultPrefetchChunks = builder.defaultPrefetchChunks; this.defaultQueryOptions = builder.defaultQueryOptions; + this.defaultDirectedReadOptions = builder.defaultDirectedReadOption; this.span = builder.span; this.executorProvider = builder.executorProvider; this.clock = builder.clock; @@ -625,6 +634,8 @@ ExecuteSqlRequest.Builder getExecuteSqlRequestBuilder( } if (options.hasDirectedReadOptions()) { builder.setDirectedReadOptions(options.directedReadOptions()); + } else if (defaultDirectedReadOptions != null) { + builder.setDirectedReadOptions(defaultDirectedReadOptions); } builder.setSeqno(getSeqNo()); builder.setQueryOptions(buildQueryOptions(statement.getQueryOptions())); @@ -816,6 +827,8 @@ ResultSet readInternalWithOptions( } if (readOptions.hasDirectedReadOptions()) { builder.setDirectedReadOptions(readOptions.directedReadOptions()); + } else if (defaultDirectedReadOptions != null) { + builder.setDirectedReadOptions(defaultDirectedReadOptions); } final int prefetchChunks = readOptions.hasPrefetchChunks() ? readOptions.prefetchChunks() : defaultPrefetchChunks; diff --git a/google-cloud-spanner/src/main/java/com/google/cloud/spanner/SessionImpl.java b/google-cloud-spanner/src/main/java/com/google/cloud/spanner/SessionImpl.java index 0e763dbc93d..aed74e7789f 100644 --- a/google-cloud-spanner/src/main/java/com/google/cloud/spanner/SessionImpl.java +++ b/google-cloud-spanner/src/main/java/com/google/cloud/spanner/SessionImpl.java @@ -255,6 +255,7 @@ public ReadContext singleUse(TimestampBound bound) { .setRpc(spanner.getRpc()) .setDefaultQueryOptions(spanner.getDefaultQueryOptions(databaseId)) .setDefaultPrefetchChunks(spanner.getDefaultPrefetchChunks()) + .setDefaultDirectedReadOption(spanner.getOptions().getDirectedReadOptions()) .setSpan(currentSpan) .setExecutorProvider(spanner.getAsyncExecutorProvider()) .build()); @@ -274,6 +275,7 @@ public ReadOnlyTransaction singleUseReadOnlyTransaction(TimestampBound bound) { .setRpc(spanner.getRpc()) .setDefaultQueryOptions(spanner.getDefaultQueryOptions(databaseId)) .setDefaultPrefetchChunks(spanner.getDefaultPrefetchChunks()) + .setDefaultDirectedReadOption(spanner.getOptions().getDirectedReadOptions()) .setSpan(currentSpan) .setExecutorProvider(spanner.getAsyncExecutorProvider()) .buildSingleUseReadOnlyTransaction()); @@ -293,6 +295,7 @@ public ReadOnlyTransaction readOnlyTransaction(TimestampBound bound) { .setRpc(spanner.getRpc()) .setDefaultQueryOptions(spanner.getDefaultQueryOptions(databaseId)) .setDefaultPrefetchChunks(spanner.getDefaultPrefetchChunks()) + .setDefaultDirectedReadOption(spanner.getOptions().getDirectedReadOptions()) .setSpan(currentSpan) .setExecutorProvider(spanner.getAsyncExecutorProvider()) .build()); diff --git a/google-cloud-spanner/src/main/java/com/google/cloud/spanner/SpannerOptions.java b/google-cloud-spanner/src/main/java/com/google/cloud/spanner/SpannerOptions.java index ba22ec54487..919c2ba6e68 100644 --- a/google-cloud-spanner/src/main/java/com/google/cloud/spanner/SpannerOptions.java +++ b/google-cloud-spanner/src/main/java/com/google/cloud/spanner/SpannerOptions.java @@ -33,6 +33,7 @@ import com.google.cloud.TransportOptions; import com.google.cloud.grpc.GcpManagedChannelOptions; import com.google.cloud.grpc.GrpcTransportOptions; +import com.google.cloud.spanner.Options.DirectedReadOption; import com.google.cloud.spanner.Options.QueryOption; import com.google.cloud.spanner.Options.UpdateOption; import com.google.cloud.spanner.admin.database.v1.DatabaseAdminSettings; @@ -50,6 +51,7 @@ import com.google.common.collect.ImmutableMap; import com.google.common.collect.ImmutableSet; import com.google.common.util.concurrent.ThreadFactoryBuilder; +import com.google.spanner.v1.DirectedReadOptions; import com.google.spanner.v1.ExecuteSqlRequest; import com.google.spanner.v1.ExecuteSqlRequest.QueryOptions; import com.google.spanner.v1.SpannerGrpc; @@ -137,6 +139,7 @@ public class SpannerOptions extends ServiceOptions { private final String compressorName; private final boolean leaderAwareRoutingEnabled; private final boolean attemptDirectPath; + private final DirectedReadOptions directedReadOption; /** Interface that can be used to provide {@link CallCredentials} to {@link SpannerOptions}. */ public interface CallCredentialsProvider { @@ -627,6 +630,7 @@ private SpannerOptions(Builder builder) { compressorName = builder.compressorName; leaderAwareRoutingEnabled = builder.leaderAwareRoutingEnabled; attemptDirectPath = builder.attemptDirectPath; + directedReadOption = builder.directedReadOption; } /** @@ -729,6 +733,7 @@ public static class Builder private String emulatorHost = System.getenv("SPANNER_EMULATOR_HOST"); private boolean leaderAwareRoutingEnabled = true; private boolean attemptDirectPath = true; + private DirectedReadOptions directedReadOption; private static String createCustomClientLibToken(String token) { return token + " " + ServiceOptions.getGoogApiClientLibName(); @@ -789,6 +794,7 @@ private Builder() { this.channelConfigurator = options.channelConfigurator; this.interceptorProvider = options.interceptorProvider; this.attemptDirectPath = options.attemptDirectPath; + this.directedReadOption = options.directedReadOption; } @Override @@ -1153,6 +1159,19 @@ public Builder setAsyncExecutorProvider(CloseableExecutorProvider provider) { return this; } + /** + * Sets the {@link DirectedReadOption} that specify which replicas or regions should be used for + * non-transactional reads or queries. + * + *

DirectedReadOptions set at the request level will take precedence over the options set + * using this method. + */ + public Builder setDirectedReadOption(DirectedReadOptions directedReadOptions) { + this.directedReadOption = + Preconditions.checkNotNull(directedReadOptions, "DirectedReadOptions cannot be null"); + return this; + } + /** * Specifying this will allow the client to prefetch up to {@code prefetchChunks} {@code * PartialResultSet} chunks for each read and query. The data size of each chunk depends on the @@ -1371,6 +1390,10 @@ public boolean isLeaderAwareRoutingEnabled() { return leaderAwareRoutingEnabled; } + public DirectedReadOptions getDirectedReadOptions() { + return directedReadOption; + } + @BetaApi public boolean isAttemptDirectPath() { return attemptDirectPath; diff --git a/google-cloud-spanner/src/test/java/com/google/cloud/spanner/AbstractReadContextTest.java b/google-cloud-spanner/src/test/java/com/google/cloud/spanner/AbstractReadContextTest.java index 31b73581f6b..16e4aa9600d 100644 --- a/google-cloud-spanner/src/test/java/com/google/cloud/spanner/AbstractReadContextTest.java +++ b/google-cloud-spanner/src/test/java/com/google/cloud/spanner/AbstractReadContextTest.java @@ -25,6 +25,9 @@ import com.google.api.gax.core.ExecutorProvider; import com.google.cloud.spanner.Options.RpcPriority; import com.google.cloud.spanner.spi.v1.SpannerRpc; +import com.google.spanner.v1.DirectedReadOptions; +import com.google.spanner.v1.DirectedReadOptions.IncludeReplicas; +import com.google.spanner.v1.DirectedReadOptions.ReplicaSelection; import com.google.spanner.v1.ExecuteBatchDmlRequest; import com.google.spanner.v1.ExecuteSqlRequest; import com.google.spanner.v1.ExecuteSqlRequest.QueryMode; @@ -45,6 +48,14 @@ @RunWith(Parameterized.class) public class AbstractReadContextTest { + private static final DirectedReadOptions DIRECTED_READ_OPTIONS = + DirectedReadOptions.newBuilder() + .setIncludeReplicas( + IncludeReplicas.newBuilder() + .addReplicaSelections( + ReplicaSelection.newBuilder().setLocation("us-west1").build())) + .build(); + @Parameter(0) public QueryOptions defaultQueryOptions; @@ -250,4 +261,15 @@ public void executeSqlRequestBuilderWithRequestOptionsWithTxnTag() { .isEqualTo("app=spanner,env=test,action=query"); assertThat(request.getRequestOptions().getTransactionTag()).isEqualTo("app=spanner,env=test"); } + + @Test + public void testGetExecuteSqlRequestBuilderWithDirectedReadOptions() { + ExecuteSqlRequest.Builder request = + context.getExecuteSqlRequestBuilder( + Statement.of("SELECT * FROM FOO"), + QueryMode.NORMAL, + Options.fromQueryOptions(Options.directedRead(DIRECTED_READ_OPTIONS)), + false); + assertEquals(DIRECTED_READ_OPTIONS, request.getDirectedReadOptions()); + } } diff --git a/google-cloud-spanner/src/test/java/com/google/cloud/spanner/DatabaseClientImplTest.java b/google-cloud-spanner/src/test/java/com/google/cloud/spanner/DatabaseClientImplTest.java index 09bad8b717f..ce5eddab40f 100644 --- a/google-cloud-spanner/src/test/java/com/google/cloud/spanner/DatabaseClientImplTest.java +++ b/google-cloud-spanner/src/test/java/com/google/cloud/spanner/DatabaseClientImplTest.java @@ -180,6 +180,13 @@ public class DatabaseClientImplTest { .addReplicaSelections( ReplicaSelection.newBuilder().setLocation("us-west1").build())) .build(); + private static final DirectedReadOptions DIRECTED_READ_OPTIONS2 = + DirectedReadOptions.newBuilder() + .setIncludeReplicas( + IncludeReplicas.newBuilder() + .addReplicaSelections( + ReplicaSelection.newBuilder().setLocation("us-east1").build())) + .build(); private Spanner spanner; private Spanner spannerWithEmptySessionPool; private static ExecutorService executor; @@ -1547,6 +1554,29 @@ public void testExecuteQueryWithDirectedReadOptions() { assertEquals(DIRECTED_READ_OPTIONS1, request.getDirectedReadOptions()); } + @Test + public void testExecuteQueryWithDirectedReadOptionsViaSpannerOptions() { + Spanner spannerWithDirectedReadOptions = + spanner + .getOptions() + .toBuilder() + .setDirectedReadOption(DIRECTED_READ_OPTIONS2) + .build() + .getService(); + DatabaseClient client = + spannerWithDirectedReadOptions.getDatabaseClient( + DatabaseId.of(TEST_PROJECT, TEST_INSTANCE, TEST_DATABASE)); + try (ResultSet resultSet = client.singleUse().executeQuery(SELECT1)) { + while (resultSet.next()) {} + } + + List requests = mockSpanner.getRequestsOfType(ExecuteSqlRequest.class); + assertEquals(requests.size(), 1); + ExecuteSqlRequest request = requests.get(0); + assertTrue(request.hasDirectedReadOptions()); + assertEquals(DIRECTED_READ_OPTIONS2, request.getDirectedReadOptions()); + } + @Test public void testExecuteReadWithTag() { DatabaseClient client = @@ -1593,6 +1623,57 @@ public void testExecuteReadWithDirectedReadOptions() { assertEquals(DIRECTED_READ_OPTIONS1, request.getDirectedReadOptions()); } + @Test + public void testExecuteReadWithDirectedReadOptionsViaSpannerOptions() { + Spanner spannerWithDirectedReadOptions = + spanner + .getOptions() + .toBuilder() + .setDirectedReadOption(DIRECTED_READ_OPTIONS2) + .build() + .getService(); + DatabaseClient client = + spannerWithDirectedReadOptions.getDatabaseClient( + DatabaseId.of(TEST_PROJECT, TEST_INSTANCE, TEST_DATABASE)); + try (ResultSet resultSet = + client.singleUse().read(READ_TABLE_NAME, KeySet.singleKey(Key.of(1L)), READ_COLUMN_NAMES)) { + while (resultSet.next()) {} + } + + List requests = mockSpanner.getRequestsOfType(ReadRequest.class); + assertEquals(requests.size(), 1); + ReadRequest request = requests.get(0); + assertTrue(request.hasDirectedReadOptions()); + assertEquals(DIRECTED_READ_OPTIONS2, request.getDirectedReadOptions()); + } + + @Test + public void testReadWriteExecuteQueryWithDirectedReadOptionsViaSpannerOptions() { + Spanner spannerWithDirectedReadOptions = + spanner + .getOptions() + .toBuilder() + .setDirectedReadOption(DIRECTED_READ_OPTIONS2) + .build() + .getService(); + DatabaseClient client = + spannerWithDirectedReadOptions.getDatabaseClient( + DatabaseId.of(TEST_PROJECT, TEST_INSTANCE, TEST_DATABASE)); + TransactionRunner runner = client.readWriteTransaction(); + runner.run( + transaction -> { + try (ResultSet resultSet = transaction.executeQuery(SELECT1)) { + while (resultSet.next()) {} + } + return null; + }); + + List requests = mockSpanner.getRequestsOfType(ExecuteSqlRequest.class); + assertEquals(requests.size(), 1); + ExecuteSqlRequest request = requests.get(0); + assertFalse(request.hasDirectedReadOptions()); + } + @Test public void testReadWriteExecuteQueryWithTag() { DatabaseClient client = diff --git a/google-cloud-spanner/src/test/java/com/google/cloud/spanner/OptionsTest.java b/google-cloud-spanner/src/test/java/com/google/cloud/spanner/OptionsTest.java index b75f8a21327..e0bbf81f297 100644 --- a/google-cloud-spanner/src/test/java/com/google/cloud/spanner/OptionsTest.java +++ b/google-cloud-spanner/src/test/java/com/google/cloud/spanner/OptionsTest.java @@ -199,7 +199,6 @@ public void readOptionsTest() { assertThat(options.tag()).isEqualTo(tag); assertEquals(dataBoost, options.dataBoostEnabled()); assertEquals(DIRECTED_READ_OPTIONS, options.directedReadOptions()); - assertThat(options.hashCode()).isEqualTo(-1667681729); } @Test @@ -252,7 +251,6 @@ public void queryOptionsTest() { assertThat(options.tag()).isEqualTo(tag); assertEquals(dataBoost, options.dataBoostEnabled()); assertEquals(DIRECTED_READ_OPTIONS, options.directedReadOptions()); - assertThat(options.hashCode()).isEqualTo(-187561950); } @Test diff --git a/google-cloud-spanner/src/test/java/com/google/cloud/spanner/SpannerOptionsTest.java b/google-cloud-spanner/src/test/java/com/google/cloud/spanner/SpannerOptionsTest.java index 635838512c7..21143dc77e0 100644 --- a/google-cloud-spanner/src/test/java/com/google/cloud/spanner/SpannerOptionsTest.java +++ b/google-cloud-spanner/src/test/java/com/google/cloud/spanner/SpannerOptionsTest.java @@ -48,6 +48,9 @@ import com.google.spanner.v1.CommitRequest; import com.google.spanner.v1.CreateSessionRequest; import com.google.spanner.v1.DeleteSessionRequest; +import com.google.spanner.v1.DirectedReadOptions; +import com.google.spanner.v1.DirectedReadOptions.IncludeReplicas; +import com.google.spanner.v1.DirectedReadOptions.ReplicaSelection; import com.google.spanner.v1.ExecuteBatchDmlRequest; import com.google.spanner.v1.ExecuteSqlRequest; import com.google.spanner.v1.ExecuteSqlRequest.QueryOptions; @@ -698,6 +701,27 @@ public void testLeaderAwareRoutingEnablement() { .isLeaderAwareRoutingEnabled()); } + @Test + public void testSetDirectedReadOptions() { + final DirectedReadOptions directedReadOptions = + DirectedReadOptions.newBuilder() + .setIncludeReplicas( + IncludeReplicas.newBuilder() + .addReplicaSelections( + ReplicaSelection.newBuilder().setLocation("us-west1").build()) + .build()) + .build(); + SpannerOptions options = + SpannerOptions.newBuilder() + .setProjectId("[PROJECT]") + .setDirectedReadOption(directedReadOptions) + .build(); + assertEquals(options.getDirectedReadOptions(), directedReadOptions); + assertThrows( + NullPointerException.class, + () -> SpannerOptions.newBuilder().setDirectedReadOption(null).build()); + } + @Test public void testSpannerCallContextTimeoutConfigurator_NullValues() { SpannerCallContextTimeoutConfigurator configurator = diff --git a/google-cloud-spanner/src/test/java/com/google/cloud/spanner/it/ITDirectedReadsTest.java b/google-cloud-spanner/src/test/java/com/google/cloud/spanner/it/ITDirectedReadsTest.java new file mode 100644 index 00000000000..e3f592ca9d5 --- /dev/null +++ b/google-cloud-spanner/src/test/java/com/google/cloud/spanner/it/ITDirectedReadsTest.java @@ -0,0 +1,100 @@ +/* + * Copyright 2023 Google LLC + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package com.google.cloud.spanner.it; + +import static com.google.cloud.spanner.MockSpannerTestUtil.SELECT1; +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertThrows; +import static org.junit.Assert.assertTrue; + +import com.google.cloud.spanner.Database; +import com.google.cloud.spanner.DatabaseClient; +import com.google.cloud.spanner.ErrorCode; +import com.google.cloud.spanner.IntegrationTestEnv; +import com.google.cloud.spanner.Options; +import com.google.cloud.spanner.ParallelIntegrationTest; +import com.google.cloud.spanner.ResultSet; +import com.google.cloud.spanner.Spanner; +import com.google.cloud.spanner.SpannerException; +import com.google.cloud.spanner.SpannerOptions; +import com.google.cloud.spanner.TransactionRunner; +import com.google.spanner.v1.DirectedReadOptions; +import com.google.spanner.v1.DirectedReadOptions.IncludeReplicas; +import com.google.spanner.v1.DirectedReadOptions.ReplicaSelection; +import org.junit.AfterClass; +import org.junit.BeforeClass; +import org.junit.ClassRule; +import org.junit.Test; +import org.junit.experimental.categories.Category; +import org.junit.runner.RunWith; +import org.junit.runners.JUnit4; + +@Category(ParallelIntegrationTest.class) +@RunWith(JUnit4.class) +public class ITDirectedReadsTest { + + private static final DirectedReadOptions DIRECTED_READ_OPTIONS = + DirectedReadOptions.newBuilder() + .setIncludeReplicas( + IncludeReplicas.newBuilder() + .addReplicaSelections( + ReplicaSelection.newBuilder().setLocation("us-west1").build())) + .build(); + + @ClassRule public static IntegrationTestEnv env = new IntegrationTestEnv(); + private static Database db; + + @BeforeClass + public static void setUp() { + db = + env.getTestHelper() + .createTestDatabase("CREATE TABLE TEST (ID INT64, NAME STRING(100)) PRIMARY KEY (ID)"); + } + + @AfterClass + public static void tearDown() { + db.drop(); + } + + @Test + public void testReadWriteTransactionRunner_queryWithDirectedReadOptions_throwsError() { + // Directed Read Options set at an RPC level is not acceptable for RW transaction + SpannerOptions options = env.getTestHelper().getOptions().toBuilder().build(); + try (Spanner spanner = options.getService()) { + DatabaseClient client = spanner.getDatabaseClient(db.getId()); + TransactionRunner runner = client.readWriteTransaction(); + SpannerException e = + assertThrows( + SpannerException.class, + () -> + runner.run( + transaction -> { + try (ResultSet resultSet = + transaction.executeQuery( + SELECT1, Options.directedRead(DIRECTED_READ_OPTIONS))) { + while (resultSet.next()) {} + } + return null; + })); + + assertEquals(ErrorCode.INVALID_ARGUMENT, e.getErrorCode()); + assertTrue( + e.getMessage() + .contains("Directed reads can only be performed in a read-only transaction.")); + } + } +} From 9122ece1b8e2819953e32029948953586860efcb Mon Sep 17 00:00:00 2001 From: Arpan Mishra Date: Sat, 30 Dec 2023 02:26:01 +0530 Subject: [PATCH 10/16] chore: fix NPE. --- .../main/java/com/google/cloud/spanner/AbstractReadContext.java | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/google-cloud-spanner/src/main/java/com/google/cloud/spanner/AbstractReadContext.java b/google-cloud-spanner/src/main/java/com/google/cloud/spanner/AbstractReadContext.java index 200f0fe4168..4d63da59879 100644 --- a/google-cloud-spanner/src/main/java/com/google/cloud/spanner/AbstractReadContext.java +++ b/google-cloud-spanner/src/main/java/com/google/cloud/spanner/AbstractReadContext.java @@ -120,7 +120,7 @@ B setClock(Clock clock) { } B setDefaultDirectedReadOption(DirectedReadOptions directedReadOptions) { - this.defaultDirectedReadOption = Preconditions.checkNotNull(directedReadOptions); + this.defaultDirectedReadOption = directedReadOptions; return self(); } From c8df52ca83d64f72dfc193c191e1cd317a1d39a5 Mon Sep 17 00:00:00 2001 From: Arpan Mishra Date: Sat, 30 Dec 2023 18:19:13 +0530 Subject: [PATCH 11/16] chore: disabling test on emulator. --- .../java/com/google/cloud/spanner/it/ITDirectedReadsTest.java | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/google-cloud-spanner/src/test/java/com/google/cloud/spanner/it/ITDirectedReadsTest.java b/google-cloud-spanner/src/test/java/com/google/cloud/spanner/it/ITDirectedReadsTest.java index e3f592ca9d5..c3ae3c25f4d 100644 --- a/google-cloud-spanner/src/test/java/com/google/cloud/spanner/it/ITDirectedReadsTest.java +++ b/google-cloud-spanner/src/test/java/com/google/cloud/spanner/it/ITDirectedReadsTest.java @@ -17,9 +17,11 @@ package com.google.cloud.spanner.it; import static com.google.cloud.spanner.MockSpannerTestUtil.SELECT1; +import static com.google.cloud.spanner.testing.EmulatorSpannerHelper.isUsingEmulator; import static org.junit.Assert.assertEquals; import static org.junit.Assert.assertThrows; import static org.junit.Assert.assertTrue; +import static org.junit.Assume.assumeFalse; import com.google.cloud.spanner.Database; import com.google.cloud.spanner.DatabaseClient; @@ -73,6 +75,8 @@ public static void tearDown() { @Test public void testReadWriteTransactionRunner_queryWithDirectedReadOptions_throwsError() { // Directed Read Options set at an RPC level is not acceptable for RW transaction + + assumeFalse("Emulator does not support directed reads", isUsingEmulator()); SpannerOptions options = env.getTestHelper().getOptions().toBuilder().build(); try (Spanner spanner = options.getService()) { DatabaseClient client = spanner.getDatabaseClient(db.getId()); From ead5ab63361559a71a81e448f347433a7ab7ed01 Mon Sep 17 00:00:00 2001 From: Arpan Mishra Date: Tue, 2 Jan 2024 22:18:34 +0530 Subject: [PATCH 12/16] chore: adding test for query in RW transaction. --- .../cloud/spanner/it/ITDirectedReadsTest.java | 36 +++++++++++++++++++ 1 file changed, 36 insertions(+) diff --git a/google-cloud-spanner/src/test/java/com/google/cloud/spanner/it/ITDirectedReadsTest.java b/google-cloud-spanner/src/test/java/com/google/cloud/spanner/it/ITDirectedReadsTest.java index c3ae3c25f4d..f0a37412bf1 100644 --- a/google-cloud-spanner/src/test/java/com/google/cloud/spanner/it/ITDirectedReadsTest.java +++ b/google-cloud-spanner/src/test/java/com/google/cloud/spanner/it/ITDirectedReadsTest.java @@ -27,6 +27,8 @@ import com.google.cloud.spanner.DatabaseClient; import com.google.cloud.spanner.ErrorCode; import com.google.cloud.spanner.IntegrationTestEnv; +import com.google.cloud.spanner.Key; +import com.google.cloud.spanner.KeySet; import com.google.cloud.spanner.Options; import com.google.cloud.spanner.ParallelIntegrationTest; import com.google.cloud.spanner.ResultSet; @@ -34,6 +36,7 @@ import com.google.cloud.spanner.SpannerException; import com.google.cloud.spanner.SpannerOptions; import com.google.cloud.spanner.TransactionRunner; +import com.google.common.collect.Lists; import com.google.spanner.v1.DirectedReadOptions; import com.google.spanner.v1.DirectedReadOptions.IncludeReplicas; import com.google.spanner.v1.DirectedReadOptions.ReplicaSelection; @@ -101,4 +104,37 @@ public void testReadWriteTransactionRunner_queryWithDirectedReadOptions_throwsEr .contains("Directed reads can only be performed in a read-only transaction.")); } } + + @Test + public void testReadWriteTransactionRunner_readWithDirectedReadOptions_throwsError() { + // Directed Read Options set at an RPC level is not acceptable for RW transaction + + assumeFalse("Emulator does not support directed reads", isUsingEmulator()); + SpannerOptions options = env.getTestHelper().getOptions().toBuilder().build(); + try (Spanner spanner = options.getService()) { + DatabaseClient client = spanner.getDatabaseClient(db.getId()); + TransactionRunner runner = client.readWriteTransaction(); + SpannerException e = + assertThrows( + SpannerException.class, + () -> + runner.run( + transaction -> { + try (ResultSet resultSet = + transaction.read( + "TEST", + KeySet.singleKey(Key.of(1L)), + Lists.newArrayList("NAME"), + Options.directedRead(DIRECTED_READ_OPTIONS))) { + while (resultSet.next()) {} + } + return null; + })); + + assertEquals(ErrorCode.INVALID_ARGUMENT, e.getErrorCode()); + assertTrue( + e.getMessage() + .contains("Directed reads can only be performed in a read-only transaction.")); + } + } } From 641cb0be5f62e8fc44bf019300e5d5a6442b2167 Mon Sep 17 00:00:00 2001 From: Arpan Mishra Date: Tue, 2 Jan 2024 22:22:45 +0530 Subject: [PATCH 13/16] chore: adding IT for transaction manager interface. --- .../cloud/spanner/it/ITDirectedReadsTest.java | 43 +++++++++++++++++++ 1 file changed, 43 insertions(+) diff --git a/google-cloud-spanner/src/test/java/com/google/cloud/spanner/it/ITDirectedReadsTest.java b/google-cloud-spanner/src/test/java/com/google/cloud/spanner/it/ITDirectedReadsTest.java index f0a37412bf1..a61d1ed41a8 100644 --- a/google-cloud-spanner/src/test/java/com/google/cloud/spanner/it/ITDirectedReadsTest.java +++ b/google-cloud-spanner/src/test/java/com/google/cloud/spanner/it/ITDirectedReadsTest.java @@ -19,10 +19,12 @@ import static com.google.cloud.spanner.MockSpannerTestUtil.SELECT1; import static com.google.cloud.spanner.testing.EmulatorSpannerHelper.isUsingEmulator; import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertNotNull; import static org.junit.Assert.assertThrows; import static org.junit.Assert.assertTrue; import static org.junit.Assume.assumeFalse; +import com.google.cloud.spanner.AbortedException; import com.google.cloud.spanner.Database; import com.google.cloud.spanner.DatabaseClient; import com.google.cloud.spanner.ErrorCode; @@ -35,6 +37,8 @@ import com.google.cloud.spanner.Spanner; import com.google.cloud.spanner.SpannerException; import com.google.cloud.spanner.SpannerOptions; +import com.google.cloud.spanner.TransactionContext; +import com.google.cloud.spanner.TransactionManager; import com.google.cloud.spanner.TransactionRunner; import com.google.common.collect.Lists; import com.google.spanner.v1.DirectedReadOptions; @@ -137,4 +141,43 @@ public void testReadWriteTransactionRunner_readWithDirectedReadOptions_throwsErr .contains("Directed reads can only be performed in a read-only transaction.")); } } + + @Test + public void testReadWriteTransactionManager_readWithDirectedReadOptionsViaRequest_throwsError() { + // Directed Read Options set at an RPC level is not acceptable for RW transaction + SpannerOptions options = env.getTestHelper().getOptions().toBuilder().build(); + try (Spanner spanner = options.getService()) { + DatabaseClient client = spanner.getDatabaseClient(db.getId()); + try (TransactionManager manager = client.transactionManager()) { + SpannerException e = + assertThrows( + SpannerException.class, + () -> { + TransactionContext transaction = manager.begin(); + try { + while (true) { + + ResultSet resultSet = + transaction.read( + "TEST", + KeySet.singleKey(Key.of(1L)), + Lists.newArrayList("NAME"), + Options.directedRead(DIRECTED_READ_OPTIONS)); + while (resultSet.next()) {} + + manager.commit(); + assertNotNull(manager.getCommitTimestamp()); + break; + } + } catch (AbortedException ex) { + transaction = manager.resetForRetry(); + } + }); + assertEquals(ErrorCode.INVALID_ARGUMENT, e.getErrorCode()); + assertTrue( + e.getMessage() + .contains("Directed reads can only be performed in a read-only transaction.")); + } + } + } } From 54b8ee0ff0bbdf360a09d4d14e4196a30b323742 Mon Sep 17 00:00:00 2001 From: Arpan Mishra Date: Wed, 3 Jan 2024 12:30:27 +0530 Subject: [PATCH 14/16] chore: disable IT for emulator. --- .../java/com/google/cloud/spanner/it/ITDirectedReadsTest.java | 2 ++ 1 file changed, 2 insertions(+) diff --git a/google-cloud-spanner/src/test/java/com/google/cloud/spanner/it/ITDirectedReadsTest.java b/google-cloud-spanner/src/test/java/com/google/cloud/spanner/it/ITDirectedReadsTest.java index a61d1ed41a8..c068bde0a40 100644 --- a/google-cloud-spanner/src/test/java/com/google/cloud/spanner/it/ITDirectedReadsTest.java +++ b/google-cloud-spanner/src/test/java/com/google/cloud/spanner/it/ITDirectedReadsTest.java @@ -145,6 +145,8 @@ public void testReadWriteTransactionRunner_readWithDirectedReadOptions_throwsErr @Test public void testReadWriteTransactionManager_readWithDirectedReadOptionsViaRequest_throwsError() { // Directed Read Options set at an RPC level is not acceptable for RW transaction + + assumeFalse("Emulator does not support directed reads", isUsingEmulator()); SpannerOptions options = env.getTestHelper().getOptions().toBuilder().build(); try (Spanner spanner = options.getService()) { DatabaseClient client = spanner.getDatabaseClient(db.getId()); From 87aa78dd2b845daaca72f40dbe97fd01a005e40c Mon Sep 17 00:00:00 2001 From: Arpan Mishra Date: Wed, 3 Jan 2024 16:00:54 +0530 Subject: [PATCH 15/16] chore: PR comments. --- .../cloud/spanner/AbstractReadContext.java | 2 +- .../google/cloud/spanner/BatchClientImpl.java | 8 ++- .../com/google/cloud/spanner/SessionImpl.java | 6 +-- .../google/cloud/spanner/SpannerOptions.java | 14 +++--- .../cloud/spanner/DatabaseClientImplTest.java | 50 ++++++++++++++----- .../cloud/spanner/SpannerOptionsTest.java | 4 +- .../cloud/spanner/it/ITDirectedReadsTest.java | 4 +- 7 files changed, 58 insertions(+), 30 deletions(-) diff --git a/google-cloud-spanner/src/main/java/com/google/cloud/spanner/AbstractReadContext.java b/google-cloud-spanner/src/main/java/com/google/cloud/spanner/AbstractReadContext.java index 4d63da59879..0f4310f9b4d 100644 --- a/google-cloud-spanner/src/main/java/com/google/cloud/spanner/AbstractReadContext.java +++ b/google-cloud-spanner/src/main/java/com/google/cloud/spanner/AbstractReadContext.java @@ -119,7 +119,7 @@ B setClock(Clock clock) { return self(); } - B setDefaultDirectedReadOption(DirectedReadOptions directedReadOptions) { + B setDefaultDirectedReadOptions(DirectedReadOptions directedReadOptions) { this.defaultDirectedReadOption = directedReadOptions; return self(); } diff --git a/google-cloud-spanner/src/main/java/com/google/cloud/spanner/BatchClientImpl.java b/google-cloud-spanner/src/main/java/com/google/cloud/spanner/BatchClientImpl.java index 0191a11be1c..eab90a266c9 100644 --- a/google-cloud-spanner/src/main/java/com/google/cloud/spanner/BatchClientImpl.java +++ b/google-cloud-spanner/src/main/java/com/google/cloud/spanner/BatchClientImpl.java @@ -60,7 +60,9 @@ public BatchReadOnlyTransaction batchReadOnlyTransaction(TimestampBound bound) { .setDefaultQueryOptions( sessionClient.getSpanner().getDefaultQueryOptions(sessionClient.getDatabaseId())) .setExecutorProvider(sessionClient.getSpanner().getAsyncExecutorProvider()) - .setDefaultPrefetchChunks(sessionClient.getSpanner().getDefaultPrefetchChunks()), + .setDefaultPrefetchChunks(sessionClient.getSpanner().getDefaultPrefetchChunks()) + .setDefaultDirectedReadOptions( + sessionClient.getSpanner().getOptions().getDirectedReadOptions()), checkNotNull(bound)); } @@ -77,7 +79,9 @@ public BatchReadOnlyTransaction batchReadOnlyTransaction(BatchTransactionId batc .setDefaultQueryOptions( sessionClient.getSpanner().getDefaultQueryOptions(sessionClient.getDatabaseId())) .setExecutorProvider(sessionClient.getSpanner().getAsyncExecutorProvider()) - .setDefaultPrefetchChunks(sessionClient.getSpanner().getDefaultPrefetchChunks()), + .setDefaultPrefetchChunks(sessionClient.getSpanner().getDefaultPrefetchChunks()) + .setDefaultDirectedReadOptions( + sessionClient.getSpanner().getOptions().getDirectedReadOptions()), batchTransactionId); } diff --git a/google-cloud-spanner/src/main/java/com/google/cloud/spanner/SessionImpl.java b/google-cloud-spanner/src/main/java/com/google/cloud/spanner/SessionImpl.java index aed74e7789f..53bf37feb05 100644 --- a/google-cloud-spanner/src/main/java/com/google/cloud/spanner/SessionImpl.java +++ b/google-cloud-spanner/src/main/java/com/google/cloud/spanner/SessionImpl.java @@ -255,7 +255,7 @@ public ReadContext singleUse(TimestampBound bound) { .setRpc(spanner.getRpc()) .setDefaultQueryOptions(spanner.getDefaultQueryOptions(databaseId)) .setDefaultPrefetchChunks(spanner.getDefaultPrefetchChunks()) - .setDefaultDirectedReadOption(spanner.getOptions().getDirectedReadOptions()) + .setDefaultDirectedReadOptions(spanner.getOptions().getDirectedReadOptions()) .setSpan(currentSpan) .setExecutorProvider(spanner.getAsyncExecutorProvider()) .build()); @@ -275,7 +275,7 @@ public ReadOnlyTransaction singleUseReadOnlyTransaction(TimestampBound bound) { .setRpc(spanner.getRpc()) .setDefaultQueryOptions(spanner.getDefaultQueryOptions(databaseId)) .setDefaultPrefetchChunks(spanner.getDefaultPrefetchChunks()) - .setDefaultDirectedReadOption(spanner.getOptions().getDirectedReadOptions()) + .setDefaultDirectedReadOptions(spanner.getOptions().getDirectedReadOptions()) .setSpan(currentSpan) .setExecutorProvider(spanner.getAsyncExecutorProvider()) .buildSingleUseReadOnlyTransaction()); @@ -295,7 +295,7 @@ public ReadOnlyTransaction readOnlyTransaction(TimestampBound bound) { .setRpc(spanner.getRpc()) .setDefaultQueryOptions(spanner.getDefaultQueryOptions(databaseId)) .setDefaultPrefetchChunks(spanner.getDefaultPrefetchChunks()) - .setDefaultDirectedReadOption(spanner.getOptions().getDirectedReadOptions()) + .setDefaultDirectedReadOptions(spanner.getOptions().getDirectedReadOptions()) .setSpan(currentSpan) .setExecutorProvider(spanner.getAsyncExecutorProvider()) .build()); diff --git a/google-cloud-spanner/src/main/java/com/google/cloud/spanner/SpannerOptions.java b/google-cloud-spanner/src/main/java/com/google/cloud/spanner/SpannerOptions.java index 919c2ba6e68..18f22dff66f 100644 --- a/google-cloud-spanner/src/main/java/com/google/cloud/spanner/SpannerOptions.java +++ b/google-cloud-spanner/src/main/java/com/google/cloud/spanner/SpannerOptions.java @@ -139,7 +139,7 @@ public class SpannerOptions extends ServiceOptions { private final String compressorName; private final boolean leaderAwareRoutingEnabled; private final boolean attemptDirectPath; - private final DirectedReadOptions directedReadOption; + private final DirectedReadOptions directedReadOptions; /** Interface that can be used to provide {@link CallCredentials} to {@link SpannerOptions}. */ public interface CallCredentialsProvider { @@ -630,7 +630,7 @@ private SpannerOptions(Builder builder) { compressorName = builder.compressorName; leaderAwareRoutingEnabled = builder.leaderAwareRoutingEnabled; attemptDirectPath = builder.attemptDirectPath; - directedReadOption = builder.directedReadOption; + directedReadOptions = builder.directedReadOptions; } /** @@ -733,7 +733,7 @@ public static class Builder private String emulatorHost = System.getenv("SPANNER_EMULATOR_HOST"); private boolean leaderAwareRoutingEnabled = true; private boolean attemptDirectPath = true; - private DirectedReadOptions directedReadOption; + private DirectedReadOptions directedReadOptions; private static String createCustomClientLibToken(String token) { return token + " " + ServiceOptions.getGoogApiClientLibName(); @@ -794,7 +794,7 @@ private Builder() { this.channelConfigurator = options.channelConfigurator; this.interceptorProvider = options.interceptorProvider; this.attemptDirectPath = options.attemptDirectPath; - this.directedReadOption = options.directedReadOption; + this.directedReadOptions = options.directedReadOptions; } @Override @@ -1166,8 +1166,8 @@ public Builder setAsyncExecutorProvider(CloseableExecutorProvider provider) { *

DirectedReadOptions set at the request level will take precedence over the options set * using this method. */ - public Builder setDirectedReadOption(DirectedReadOptions directedReadOptions) { - this.directedReadOption = + public Builder setDirectedReadOptions(DirectedReadOptions directedReadOptions) { + this.directedReadOptions = Preconditions.checkNotNull(directedReadOptions, "DirectedReadOptions cannot be null"); return this; } @@ -1391,7 +1391,7 @@ public boolean isLeaderAwareRoutingEnabled() { } public DirectedReadOptions getDirectedReadOptions() { - return directedReadOption; + return directedReadOptions; } @BetaApi diff --git a/google-cloud-spanner/src/test/java/com/google/cloud/spanner/DatabaseClientImplTest.java b/google-cloud-spanner/src/test/java/com/google/cloud/spanner/DatabaseClientImplTest.java index ce5eddab40f..92471e6b341 100644 --- a/google-cloud-spanner/src/test/java/com/google/cloud/spanner/DatabaseClientImplTest.java +++ b/google-cloud-spanner/src/test/java/com/google/cloud/spanner/DatabaseClientImplTest.java @@ -1539,7 +1539,7 @@ public void testExecuteQueryWithTag() { } @Test - public void testExecuteQueryWithDirectedReadOptions() { + public void testExecuteQuery_withDirectedReadOptionsViaRequest() { DatabaseClient client = spanner.getDatabaseClient(DatabaseId.of(TEST_PROJECT, TEST_INSTANCE, TEST_DATABASE)); try (ResultSet resultSet = @@ -1555,12 +1555,12 @@ public void testExecuteQueryWithDirectedReadOptions() { } @Test - public void testExecuteQueryWithDirectedReadOptionsViaSpannerOptions() { + public void testExecuteQuery_withDirectedReadOptionsViaSpannerOptions() { Spanner spannerWithDirectedReadOptions = spanner .getOptions() .toBuilder() - .setDirectedReadOption(DIRECTED_READ_OPTIONS2) + .setDirectedReadOptions(DIRECTED_READ_OPTIONS2) .build() .getService(); DatabaseClient client = @@ -1577,6 +1577,30 @@ public void testExecuteQueryWithDirectedReadOptionsViaSpannerOptions() { assertEquals(DIRECTED_READ_OPTIONS2, request.getDirectedReadOptions()); } + @Test + public void testExecuteQuery_whenMultipleDirectedReadsOptions_preferRequestOption() { + Spanner spannerWithDirectedReadOptions = + spanner + .getOptions() + .toBuilder() + .setDirectedReadOptions(DIRECTED_READ_OPTIONS2) + .build() + .getService(); + DatabaseClient client = + spannerWithDirectedReadOptions.getDatabaseClient( + DatabaseId.of(TEST_PROJECT, TEST_INSTANCE, TEST_DATABASE)); + try (ResultSet resultSet = + client.singleUse().executeQuery(SELECT1, Options.directedRead(DIRECTED_READ_OPTIONS1))) { + while (resultSet.next()) {} + } + + List requests = mockSpanner.getRequestsOfType(ExecuteSqlRequest.class); + assertEquals(requests.size(), 1); + ExecuteSqlRequest request = requests.get(0); + assertTrue(request.hasDirectedReadOptions()); + assertEquals(DIRECTED_READ_OPTIONS1, request.getDirectedReadOptions()); + } + @Test public void testExecuteReadWithTag() { DatabaseClient client = @@ -1629,7 +1653,7 @@ public void testExecuteReadWithDirectedReadOptionsViaSpannerOptions() { spanner .getOptions() .toBuilder() - .setDirectedReadOption(DIRECTED_READ_OPTIONS2) + .setDirectedReadOptions(DIRECTED_READ_OPTIONS2) .build() .getService(); DatabaseClient client = @@ -1653,7 +1677,7 @@ public void testReadWriteExecuteQueryWithDirectedReadOptionsViaSpannerOptions() spanner .getOptions() .toBuilder() - .setDirectedReadOption(DIRECTED_READ_OPTIONS2) + .setDirectedReadOptions(DIRECTED_READ_OPTIONS2) .build() .getService(); DatabaseClient client = @@ -2860,7 +2884,7 @@ public void testNestedTransactionsUsingTwoDatabases() throws InterruptedExceptio @Test public void testBackendQueryOptions() { - // Use a Spanner instance with MinSession=0 and WriteFraction=0.0 to prevent background requests + // Use a Spanner instance with MinSession=0 to prevent background requests // from the session pool interfering with the test case. try (Spanner spanner = SpannerOptions.newBuilder() @@ -2901,7 +2925,7 @@ public void testBackendQueryOptions() { @Test public void testBackendQueryOptionsWithAnalyzeQuery() { - // Use a Spanner instance with MinSession=0 and WriteFraction=0.0 to prevent background requests + // Use a Spanner instance with MinSession=0 to prevent background requests // from the session pool interfering with the test case. try (Spanner spanner = SpannerOptions.newBuilder() @@ -2944,7 +2968,7 @@ public void testBackendQueryOptionsWithAnalyzeQuery() { @Test public void testBackendPartitionQueryOptions() { - // Use a Spanner instance with MinSession=0 and WriteFraction=0.0 to prevent background requests + // Use a Spanner instance with MinSession=0 to prevent background requests // from the session pool interfering with the test case. try (Spanner spanner = SpannerOptions.newBuilder() @@ -2976,8 +3000,8 @@ public void testBackendPartitionQueryOptions() { transaction.cleanup(); } // Check if the last query executed is a DeleteSessionRequest and the second last query - // executed is a ExecuteSqlRequest and was executed using a custom optimizer version and - // statistics package. + // executed is a ExecuteSqlRequest and was executed using a custom optimizer version, + // statistics package and directed read options. List requests = mockSpanner.getRequests(); assert requests.size() >= 2 : "required to have at least 2 requests"; assertThat(requests.get(requests.size() - 1)).isInstanceOf(DeleteSessionRequest.class); @@ -2993,7 +3017,7 @@ public void testBackendPartitionQueryOptions() { @Test public void testBackendPartitionReadOptions() { - // Use a Spanner instance with MinSession=0 and WriteFraction=0.0 to prevent background requests + // Use a Spanner instance with MinSession=0 to prevent background requests // from the session pool interfering with the test case. try (Spanner spanner = SpannerOptions.newBuilder() @@ -3021,8 +3045,8 @@ public void testBackendPartitionReadOptions() { transaction.cleanup(); } // Check if the last query executed is a DeleteSessionRequest and the second last query - // executed is a ExecuteSqlRequest and was executed using a custom optimizer version and - // statistics package. + // executed is a ExecuteSqlRequest and was executed using a custom optimizer version, + // statistics package and directed read options. List requests = mockSpanner.getRequests(); assert requests.size() >= 2 : "required to have at least 2 requests"; assertThat(requests.get(requests.size() - 1)).isInstanceOf(DeleteSessionRequest.class); diff --git a/google-cloud-spanner/src/test/java/com/google/cloud/spanner/SpannerOptionsTest.java b/google-cloud-spanner/src/test/java/com/google/cloud/spanner/SpannerOptionsTest.java index 21143dc77e0..42e53a6b8e7 100644 --- a/google-cloud-spanner/src/test/java/com/google/cloud/spanner/SpannerOptionsTest.java +++ b/google-cloud-spanner/src/test/java/com/google/cloud/spanner/SpannerOptionsTest.java @@ -714,12 +714,12 @@ public void testSetDirectedReadOptions() { SpannerOptions options = SpannerOptions.newBuilder() .setProjectId("[PROJECT]") - .setDirectedReadOption(directedReadOptions) + .setDirectedReadOptions(directedReadOptions) .build(); assertEquals(options.getDirectedReadOptions(), directedReadOptions); assertThrows( NullPointerException.class, - () -> SpannerOptions.newBuilder().setDirectedReadOption(null).build()); + () -> SpannerOptions.newBuilder().setDirectedReadOptions(null).build()); } @Test diff --git a/google-cloud-spanner/src/test/java/com/google/cloud/spanner/it/ITDirectedReadsTest.java b/google-cloud-spanner/src/test/java/com/google/cloud/spanner/it/ITDirectedReadsTest.java index c068bde0a40..217da5f4bc5 100644 --- a/google-cloud-spanner/src/test/java/com/google/cloud/spanner/it/ITDirectedReadsTest.java +++ b/google-cloud-spanner/src/test/java/com/google/cloud/spanner/it/ITDirectedReadsTest.java @@ -80,7 +80,7 @@ public static void tearDown() { } @Test - public void testReadWriteTransactionRunner_queryWithDirectedReadOptions_throwsError() { + public void testReadWriteTransactionRunner_queryWithDirectedReadOptionsViaRequest_throwsError() { // Directed Read Options set at an RPC level is not acceptable for RW transaction assumeFalse("Emulator does not support directed reads", isUsingEmulator()); @@ -110,7 +110,7 @@ public void testReadWriteTransactionRunner_queryWithDirectedReadOptions_throwsEr } @Test - public void testReadWriteTransactionRunner_readWithDirectedReadOptions_throwsError() { + public void testReadWriteTransactionRunner_readWithDirectedReadOptionsViaRequest_throwsError() { // Directed Read Options set at an RPC level is not acceptable for RW transaction assumeFalse("Emulator does not support directed reads", isUsingEmulator()); From db8b52ccd6690e14d051a0f8e65a1b3badffe600 Mon Sep 17 00:00:00 2001 From: Arpan Mishra Date: Wed, 3 Jan 2024 16:14:44 +0530 Subject: [PATCH 16/16] chore: address PR comments. --- .../google/cloud/spanner/SpannerOptions.java | 13 +++ .../cloud/spanner/DatabaseClientImplTest.java | 94 +++++++++++++++++++ 2 files changed, 107 insertions(+) diff --git a/google-cloud-spanner/src/main/java/com/google/cloud/spanner/SpannerOptions.java b/google-cloud-spanner/src/main/java/com/google/cloud/spanner/SpannerOptions.java index 18f22dff66f..877ea72e467 100644 --- a/google-cloud-spanner/src/main/java/com/google/cloud/spanner/SpannerOptions.java +++ b/google-cloud-spanner/src/main/java/com/google/cloud/spanner/SpannerOptions.java @@ -1165,6 +1165,19 @@ public Builder setAsyncExecutorProvider(CloseableExecutorProvider provider) { * *

DirectedReadOptions set at the request level will take precedence over the options set * using this method. + * + *

An example below of how {@link DirectedReadOptions} can be constructed by including a + * replica. + * + *


+     * DirectedReadOptions.newBuilder()
+     *           .setIncludeReplicas(
+     *               IncludeReplicas.newBuilder()
+     *                   .addReplicaSelections(
+     *                       ReplicaSelection.newBuilder().setLocation("us-east1").build()))
+     *           .build();
+     *           }
+     * 
*/ public Builder setDirectedReadOptions(DirectedReadOptions directedReadOptions) { this.directedReadOptions = diff --git a/google-cloud-spanner/src/test/java/com/google/cloud/spanner/DatabaseClientImplTest.java b/google-cloud-spanner/src/test/java/com/google/cloud/spanner/DatabaseClientImplTest.java index 92471e6b341..e6263568fe5 100644 --- a/google-cloud-spanner/src/test/java/com/google/cloud/spanner/DatabaseClientImplTest.java +++ b/google-cloud-spanner/src/test/java/com/google/cloud/spanner/DatabaseClientImplTest.java @@ -2976,6 +2976,7 @@ public void testBackendPartitionQueryOptions() { .setChannelProvider(channelProvider) .setCredentials(NoCredentials.getInstance()) .setSessionPoolOption(SessionPoolOptions.newBuilder().setMinSessions(0).build()) + .setDirectedReadOptions(DIRECTED_READ_OPTIONS2) .build() .getService()) { BatchClient client = @@ -3015,6 +3016,56 @@ public void testBackendPartitionQueryOptions() { } } + @Test + public void + testBackendPartitionQueryOptions_whenDirectedReadOptionsViaSpannerOptions_assertOptions() { + // Use a Spanner instance with MinSession=0 to prevent background requests + // from the session pool interfering with the test case. + try (Spanner spanner = + SpannerOptions.newBuilder() + .setProjectId("[PROJECT]") + .setChannelProvider(channelProvider) + .setCredentials(NoCredentials.getInstance()) + .setSessionPoolOption(SessionPoolOptions.newBuilder().setMinSessions(0).build()) + .setDirectedReadOptions(DIRECTED_READ_OPTIONS2) + .build() + .getService()) { + BatchClient client = + spanner.getBatchClient(DatabaseId.of("[PROJECT]", "[INSTANCE]", "[DATABASE")); + BatchReadOnlyTransaction transaction = + client.batchReadOnlyTransaction(TimestampBound.strong()); + List partitions = + transaction.partitionQuery( + PartitionOptions.newBuilder().setMaxPartitions(10L).build(), + Statement.newBuilder(SELECT1.getSql()) + .withQueryOptions( + QueryOptions.newBuilder() + .setOptimizerVersion("1") + .setOptimizerStatisticsPackage("custom-package") + .build()) + .build()); + try (ResultSet rs = transaction.execute(partitions.get(0))) { + // Just iterate over the results to execute the query. + while (rs.next()) {} + } finally { + transaction.cleanup(); + } + // Check if the last query executed is a DeleteSessionRequest and the second last query + // executed is a ExecuteSqlRequest and was executed using a custom optimizer version, + // statistics package and directed read options. + List requests = mockSpanner.getRequests(); + assert requests.size() >= 2 : "required to have at least 2 requests"; + assertThat(requests.get(requests.size() - 1)).isInstanceOf(DeleteSessionRequest.class); + assertThat(requests.get(requests.size() - 2)).isInstanceOf(ExecuteSqlRequest.class); + ExecuteSqlRequest executeSqlRequest = (ExecuteSqlRequest) requests.get(requests.size() - 2); + assertThat(executeSqlRequest.getQueryOptions()).isNotNull(); + assertThat(executeSqlRequest.getQueryOptions().getOptimizerVersion()).isEqualTo("1"); + assertThat(executeSqlRequest.getQueryOptions().getOptimizerStatisticsPackage()) + .isEqualTo("custom-package"); + assertThat(executeSqlRequest.getDirectedReadOptions()).isEqualTo(DIRECTED_READ_OPTIONS2); + } + } + @Test public void testBackendPartitionReadOptions() { // Use a Spanner instance with MinSession=0 to prevent background requests @@ -3025,6 +3076,7 @@ public void testBackendPartitionReadOptions() { .setChannelProvider(channelProvider) .setCredentials(NoCredentials.getInstance()) .setSessionPoolOption(SessionPoolOptions.newBuilder().setMinSessions(0).build()) + .setDirectedReadOptions(DIRECTED_READ_OPTIONS2) .build() .getService()) { BatchClient client = @@ -3056,6 +3108,48 @@ public void testBackendPartitionReadOptions() { } } + @Test + public void + testBackendPartitionReadOptions_whenDirectedReadOptionsViaSpannerOptions_assertOptions() { + // Use a Spanner instance with MinSession=0 to prevent background requests + // from the session pool interfering with the test case. + try (Spanner spanner = + SpannerOptions.newBuilder() + .setProjectId("[PROJECT]") + .setChannelProvider(channelProvider) + .setCredentials(NoCredentials.getInstance()) + .setSessionPoolOption(SessionPoolOptions.newBuilder().setMinSessions(0).build()) + .setDirectedReadOptions(DIRECTED_READ_OPTIONS2) + .build() + .getService()) { + BatchClient client = + spanner.getBatchClient(DatabaseId.of("[PROJECT]", "[INSTANCE]", "[DATABASE")); + BatchReadOnlyTransaction transaction = + client.batchReadOnlyTransaction(TimestampBound.strong()); + List partitions = + transaction.partitionRead( + PartitionOptions.newBuilder().setMaxPartitions(10L).build(), + "FOO", + KeySet.all(), + Lists.newArrayList("1")); + try (ResultSet rs = transaction.execute(partitions.get(0))) { + // Just iterate over the results to execute the query. + while (rs.next()) {} + } finally { + transaction.cleanup(); + } + // Check if the last query executed is a DeleteSessionRequest and the second last query + // executed is a ExecuteSqlRequest and was executed using a custom optimizer version, + // statistics package and directed read options. + List requests = mockSpanner.getRequests(); + assert requests.size() >= 2 : "required to have at least 2 requests"; + assertThat(requests.get(requests.size() - 1)).isInstanceOf(DeleteSessionRequest.class); + assertThat(requests.get(requests.size() - 2)).isInstanceOf(ReadRequest.class); + ReadRequest readRequest = (ReadRequest) requests.get(requests.size() - 2); + assertThat(readRequest.getDirectedReadOptions()).isEqualTo(DIRECTED_READ_OPTIONS2); + } + } + @Test public void testAsyncQuery() throws Exception { final int EXPECTED_ROW_COUNT = 10;