|
79 | 79 | import com.google.common.base.Strings;
|
80 | 80 | import com.google.common.base.Suppliers;
|
81 | 81 | import com.google.common.collect.ImmutableMap;
|
82 |
| -import com.google.common.collect.Sets; |
83 | 82 | import io.opentelemetry.api.OpenTelemetry;
|
84 | 83 | import java.io.IOException;
|
85 | 84 | import java.net.URL;
|
86 | 85 | import java.time.Duration;
|
87 | 86 | import java.util.ArrayList;
|
88 |
| -import java.util.Arrays; |
89 | 87 | import java.util.Collections;
|
90 | 88 | import java.util.HashMap;
|
91 |
| -import java.util.HashSet; |
92 | 89 | import java.util.List;
|
93 | 90 | import java.util.Locale;
|
94 | 91 | import java.util.Map;
|
95 | 92 | import java.util.Objects;
|
96 |
| -import java.util.Set; |
97 | 93 | import java.util.regex.Matcher;
|
98 | 94 | import java.util.regex.Pattern;
|
99 | 95 | import java.util.stream.Stream;
|
|
127 | 123 | */
|
128 | 124 | @InternalApi
|
129 | 125 | public class ConnectionOptions {
|
130 |
| - /** |
131 |
| - * Supported connection properties that can be included in the connection URI. |
132 |
| - * |
133 |
| - * @deprecated Replaced by {@link com.google.cloud.spanner.connection.ConnectionProperty}. |
134 |
| - */ |
135 |
| - @Deprecated |
136 |
| - public static class ConnectionProperty { |
137 |
| - private static final String[] BOOLEAN_VALUES = new String[] {"true", "false"}; |
138 |
| - private final String name; |
139 |
| - private final String description; |
140 |
| - private final String defaultValue; |
141 |
| - private final String[] validValues; |
142 |
| - private final int hashCode; |
143 |
| - |
144 |
| - private static ConnectionProperty createStringProperty(String name, String description) { |
145 |
| - return new ConnectionProperty(name, description, "", null); |
146 |
| - } |
147 |
| - |
148 |
| - private static ConnectionProperty createBooleanProperty( |
149 |
| - String name, String description, Boolean defaultValue) { |
150 |
| - return new ConnectionProperty( |
151 |
| - name, |
152 |
| - description, |
153 |
| - defaultValue == null ? "" : String.valueOf(defaultValue), |
154 |
| - BOOLEAN_VALUES); |
155 |
| - } |
156 |
| - |
157 |
| - private static ConnectionProperty createIntProperty( |
158 |
| - String name, String description, int defaultValue) { |
159 |
| - return new ConnectionProperty(name, description, String.valueOf(defaultValue), null); |
160 |
| - } |
161 |
| - |
162 |
| - private static ConnectionProperty createEmptyProperty(String name) { |
163 |
| - return new ConnectionProperty(name, "", "", null); |
164 |
| - } |
165 |
| - |
166 |
| - private ConnectionProperty( |
167 |
| - String name, String description, String defaultValue, String[] validValues) { |
168 |
| - Preconditions.checkNotNull(name); |
169 |
| - Preconditions.checkNotNull(description); |
170 |
| - Preconditions.checkNotNull(defaultValue); |
171 |
| - this.name = name; |
172 |
| - this.description = description; |
173 |
| - this.defaultValue = defaultValue; |
174 |
| - this.validValues = validValues; |
175 |
| - this.hashCode = name.toLowerCase().hashCode(); |
176 |
| - } |
177 |
| - |
178 |
| - @Override |
179 |
| - public int hashCode() { |
180 |
| - return hashCode; |
181 |
| - } |
182 |
| - |
183 |
| - @Override |
184 |
| - public boolean equals(Object o) { |
185 |
| - if (!(o instanceof ConnectionProperty)) { |
186 |
| - return false; |
187 |
| - } |
188 |
| - return ((ConnectionProperty) o).name.equalsIgnoreCase(this.name); |
189 |
| - } |
190 |
| - |
191 |
| - /** @return the name of this connection property. */ |
192 |
| - public String getName() { |
193 |
| - return name; |
194 |
| - } |
195 |
| - |
196 |
| - /** @return the description of this connection property. */ |
197 |
| - public String getDescription() { |
198 |
| - return description; |
199 |
| - } |
200 |
| - |
201 |
| - /** @return the default value of this connection property. */ |
202 |
| - public String getDefaultValue() { |
203 |
| - return defaultValue; |
204 |
| - } |
205 |
| - |
206 |
| - /** |
207 |
| - * @return the valid values for this connection property. <code>null</code> indicates no |
208 |
| - * restriction. |
209 |
| - */ |
210 |
| - public String[] getValidValues() { |
211 |
| - return validValues; |
212 |
| - } |
213 |
| - } |
214 | 126 |
|
215 | 127 | /**
|
216 | 128 | * Set this system property to true to enable transactional connection state by default for
|
@@ -380,210 +292,6 @@ static boolean isEnableTransactionalConnectionStateForPostgreSQL() {
|
380 | 292 | System.getProperty(ENABLE_TRANSACTIONAL_CONNECTION_STATE_FOR_POSTGRESQL_PROPERTY, "false"));
|
381 | 293 | }
|
382 | 294 |
|
383 |
| - /** |
384 |
| - * All valid connection properties. |
385 |
| - * |
386 |
| - * @deprecated Replaced by {@link ConnectionProperties#CONNECTION_PROPERTIES} |
387 |
| - */ |
388 |
| - @Deprecated |
389 |
| - public static final Set<ConnectionProperty> VALID_PROPERTIES = |
390 |
| - Collections.unmodifiableSet( |
391 |
| - new HashSet<>( |
392 |
| - Arrays.asList( |
393 |
| - ConnectionProperty.createBooleanProperty( |
394 |
| - AUTOCOMMIT_PROPERTY_NAME, |
395 |
| - "Should the connection start in autocommit (true/false)", |
396 |
| - DEFAULT_AUTOCOMMIT), |
397 |
| - ConnectionProperty.createBooleanProperty( |
398 |
| - READONLY_PROPERTY_NAME, |
399 |
| - "Should the connection start in read-only mode (true/false)", |
400 |
| - DEFAULT_READONLY), |
401 |
| - ConnectionProperty.createBooleanProperty( |
402 |
| - ROUTE_TO_LEADER_PROPERTY_NAME, |
403 |
| - "Should read/write transactions and partitioned DML be routed to leader region (true/false)", |
404 |
| - DEFAULT_ROUTE_TO_LEADER), |
405 |
| - ConnectionProperty.createBooleanProperty( |
406 |
| - RETRY_ABORTS_INTERNALLY_PROPERTY_NAME, |
407 |
| - "Should the connection automatically retry Aborted errors (true/false)", |
408 |
| - DEFAULT_RETRY_ABORTS_INTERNALLY), |
409 |
| - ConnectionProperty.createBooleanProperty( |
410 |
| - USE_VIRTUAL_THREADS_PROPERTY_NAME, |
411 |
| - "Use a virtual thread instead of a platform thread for each connection (true/false). " |
412 |
| - + "This option only has any effect if the application is running on Java 21 or higher. In all other cases, the option is ignored.", |
413 |
| - DEFAULT_USE_VIRTUAL_THREADS), |
414 |
| - ConnectionProperty.createBooleanProperty( |
415 |
| - USE_VIRTUAL_GRPC_TRANSPORT_THREADS_PROPERTY_NAME, |
416 |
| - "Use a virtual thread instead of a platform thread for the gRPC executor (true/false). " |
417 |
| - + "This option only has any effect if the application is running on Java 21 or higher. In all other cases, the option is ignored.", |
418 |
| - DEFAULT_USE_VIRTUAL_GRPC_TRANSPORT_THREADS), |
419 |
| - ConnectionProperty.createStringProperty( |
420 |
| - CREDENTIALS_PROPERTY_NAME, |
421 |
| - "The location of the credentials file to use for this connection. If neither this property or encoded credentials are set, the connection will use the default Google Cloud credentials for the runtime environment."), |
422 |
| - ConnectionProperty.createStringProperty( |
423 |
| - ENCODED_CREDENTIALS_PROPERTY_NAME, |
424 |
| - "Base64-encoded credentials to use for this connection. If neither this property or a credentials location are set, the connection will use the default Google Cloud credentials for the runtime environment."), |
425 |
| - ConnectionProperty.createStringProperty( |
426 |
| - CREDENTIALS_PROVIDER_PROPERTY_NAME, |
427 |
| - "The class name of the com.google.api.gax.core.CredentialsProvider implementation that should be used to obtain credentials for connections."), |
428 |
| - ConnectionProperty.createStringProperty( |
429 |
| - OAUTH_TOKEN_PROPERTY_NAME, |
430 |
| - "A valid pre-existing OAuth token to use for authentication for this connection. Setting this property will take precedence over any value set for a credentials file."), |
431 |
| - ConnectionProperty.createStringProperty( |
432 |
| - MIN_SESSIONS_PROPERTY_NAME, |
433 |
| - "The minimum number of sessions in the backing session pool. The default is 100."), |
434 |
| - ConnectionProperty.createStringProperty( |
435 |
| - MAX_SESSIONS_PROPERTY_NAME, |
436 |
| - "The maximum number of sessions in the backing session pool. The default is 400."), |
437 |
| - ConnectionProperty.createStringProperty( |
438 |
| - NUM_CHANNELS_PROPERTY_NAME, |
439 |
| - "The number of gRPC channels to use to communicate with Cloud Spanner. The default is 4."), |
440 |
| - ConnectionProperty.createStringProperty( |
441 |
| - ENDPOINT_PROPERTY_NAME, |
442 |
| - "The endpoint that the JDBC driver should connect to. " |
443 |
| - + "The default is the default Spanner production endpoint when autoConfigEmulator=false, " |
444 |
| - + "and the default Spanner emulator endpoint (localhost:9010) when autoConfigEmulator=true. " |
445 |
| - + "This property takes precedence over any host name at the start of the connection URL."), |
446 |
| - ConnectionProperty.createStringProperty( |
447 |
| - CHANNEL_PROVIDER_PROPERTY_NAME, |
448 |
| - "The name of the channel provider class. The name must reference an implementation of ExternalChannelProvider. If this property is not set, the connection will use the default grpc channel provider."), |
449 |
| - ConnectionProperty.createBooleanProperty( |
450 |
| - USE_PLAIN_TEXT_PROPERTY_NAME, |
451 |
| - "Use a plain text communication channel (i.e. non-TLS) for communicating with the server (true/false). Set this value to true for communication with the Cloud Spanner emulator.", |
452 |
| - DEFAULT_USE_PLAIN_TEXT), |
453 |
| - ConnectionProperty.createBooleanProperty( |
454 |
| - IS_EXPERIMENTAL_HOST_PROPERTY_NAME, |
455 |
| - "Set this value to true for communication with an Experimental Host.", |
456 |
| - DEFAULT_IS_EXPERIMENTAL_HOST), |
457 |
| - ConnectionProperty.createStringProperty( |
458 |
| - CLIENT_CERTIFICATE_PROPERTY_NAME, |
459 |
| - "Specifies the file path to the client certificate required for establishing an mTLS connection."), |
460 |
| - ConnectionProperty.createStringProperty( |
461 |
| - CLIENT_KEY_PROPERTY_NAME, |
462 |
| - "Specifies the file path to the client private key required for establishing an mTLS connection."), |
463 |
| - ConnectionProperty.createStringProperty( |
464 |
| - USER_AGENT_PROPERTY_NAME, |
465 |
| - "The custom user-agent property name to use when communicating with Cloud Spanner. This property is intended for internal library usage, and should not be set by applications."), |
466 |
| - ConnectionProperty.createStringProperty( |
467 |
| - OPTIMIZER_VERSION_PROPERTY_NAME, |
468 |
| - "Sets the default query optimizer version to use for this connection."), |
469 |
| - ConnectionProperty.createStringProperty( |
470 |
| - OPTIMIZER_STATISTICS_PACKAGE_PROPERTY_NAME, ""), |
471 |
| - ConnectionProperty.createBooleanProperty( |
472 |
| - "returnCommitStats", "", DEFAULT_RETURN_COMMIT_STATS), |
473 |
| - ConnectionProperty.createStringProperty( |
474 |
| - "maxCommitDelay", |
475 |
| - "The maximum commit delay in milliseconds that should be applied to commit requests from this connection."), |
476 |
| - ConnectionProperty.createBooleanProperty( |
477 |
| - "autoConfigEmulator", |
478 |
| - "Automatically configure the connection to try to connect to the Cloud Spanner emulator (true/false). " |
479 |
| - + "The instance and database in the connection string will automatically be created if these do not yet exist on the emulator. " |
480 |
| - + "Add dialect=postgresql to the connection string to make sure that the database that is created uses the PostgreSQL dialect.", |
481 |
| - false), |
482 |
| - ConnectionProperty.createBooleanProperty( |
483 |
| - "useAutoSavepointsForEmulator", |
484 |
| - "Automatically creates savepoints for each statement in a read/write transaction when using the Emulator. This is no longer needed when using Emulator version 1.5.23 or higher.", |
485 |
| - false), |
486 |
| - ConnectionProperty.createBooleanProperty( |
487 |
| - LENIENT_PROPERTY_NAME, |
488 |
| - "Silently ignore unknown properties in the connection string/properties (true/false)", |
489 |
| - DEFAULT_LENIENT), |
490 |
| - ConnectionProperty.createStringProperty( |
491 |
| - RPC_PRIORITY_NAME, |
492 |
| - "Sets the priority for all RPC invocations from this connection (HIGH/MEDIUM/LOW). The default is HIGH."), |
493 |
| - ConnectionProperty.createStringProperty( |
494 |
| - DDL_IN_TRANSACTION_MODE_PROPERTY_NAME, |
495 |
| - "Sets the behavior of a connection when a DDL statement is executed in a read/write transaction. The default is " |
496 |
| - + DEFAULT_DDL_IN_TRANSACTION_MODE |
497 |
| - + "."), |
498 |
| - ConnectionProperty.createStringProperty( |
499 |
| - DIALECT_PROPERTY_NAME, |
500 |
| - "Sets the dialect to use for new databases that are created by this connection."), |
501 |
| - ConnectionProperty.createStringProperty( |
502 |
| - DATABASE_ROLE_PROPERTY_NAME, |
503 |
| - "Sets the database role to use for this connection. The default is privileges assigned to IAM role"), |
504 |
| - ConnectionProperty.createBooleanProperty( |
505 |
| - DELAY_TRANSACTION_START_UNTIL_FIRST_WRITE_NAME, |
506 |
| - "Enabling this option will delay the actual start of a read/write transaction until the first write operation is seen in that transaction. " |
507 |
| - + "All reads that happen before the first write in a transaction will instead be executed as if the connection was in auto-commit mode. " |
508 |
| - + "Enabling this option will make read/write transactions lose their SERIALIZABLE isolation level. Read operations that are executed after " |
509 |
| - + "the first write operation in a read/write transaction will be executed using the read/write transaction. Enabling this mode can reduce locking " |
510 |
| - + "and improve performance for applications that can handle the lower transaction isolation semantics.", |
511 |
| - DEFAULT_DELAY_TRANSACTION_START_UNTIL_FIRST_WRITE), |
512 |
| - ConnectionProperty.createBooleanProperty( |
513 |
| - KEEP_TRANSACTION_ALIVE_PROPERTY_NAME, |
514 |
| - "Enabling this option will trigger the connection to keep read/write transactions alive by executing a SELECT 1 query once every 10 seconds " |
515 |
| - + "if no other statements are being executed. This option should be used with caution, as it can keep transactions alive and hold on to locks " |
516 |
| - + "longer than intended. This option should typically be used for CLI-type application that might wait for user input for a longer period of time.", |
517 |
| - DEFAULT_KEEP_TRANSACTION_ALIVE), |
518 |
| - ConnectionProperty.createBooleanProperty( |
519 |
| - TRACK_SESSION_LEAKS_PROPERTY_NAME, |
520 |
| - "Capture the call stack of the thread that checked out a session of the session pool. This will " |
521 |
| - + "pre-create a LeakedSessionException already when a session is checked out. This can be disabled, " |
522 |
| - + "for example if a monitoring system logs the pre-created exception. " |
523 |
| - + "If disabled, the LeakedSessionException will only be created when an " |
524 |
| - + "actual session leak is detected. The stack trace of the exception will " |
525 |
| - + "in that case not contain the call stack of when the session was checked out.", |
526 |
| - DEFAULT_TRACK_SESSION_LEAKS), |
527 |
| - ConnectionProperty.createBooleanProperty( |
528 |
| - TRACK_CONNECTION_LEAKS_PROPERTY_NAME, |
529 |
| - "Capture the call stack of the thread that created a connection. This will " |
530 |
| - + "pre-create a LeakedConnectionException already when a connection is created. " |
531 |
| - + "This can be disabled, for example if a monitoring system logs the pre-created exception. " |
532 |
| - + "If disabled, the LeakedConnectionException will only be created when an " |
533 |
| - + "actual connection leak is detected. The stack trace of the exception will " |
534 |
| - + "in that case not contain the call stack of when the connection was created.", |
535 |
| - DEFAULT_TRACK_CONNECTION_LEAKS), |
536 |
| - ConnectionProperty.createBooleanProperty( |
537 |
| - DATA_BOOST_ENABLED_PROPERTY_NAME, |
538 |
| - "Enable data boost for all partitioned queries that are executed by this connection. " |
539 |
| - + "This setting is only used for partitioned queries and is ignored by all other statements.", |
540 |
| - DEFAULT_DATA_BOOST_ENABLED), |
541 |
| - ConnectionProperty.createBooleanProperty( |
542 |
| - AUTO_PARTITION_MODE_PROPERTY_NAME, |
543 |
| - "Execute all queries on this connection as partitioned queries. " |
544 |
| - + "Executing a query that cannot be partitioned will fail. " |
545 |
| - + "Executing a query in a read/write transaction will also fail.", |
546 |
| - DEFAULT_AUTO_PARTITION_MODE), |
547 |
| - ConnectionProperty.createIntProperty( |
548 |
| - MAX_PARTITIONS_PROPERTY_NAME, |
549 |
| - "The max partitions hint value to use for partitioned queries. " |
550 |
| - + "Use 0 if you do not want to specify a hint.", |
551 |
| - DEFAULT_MAX_PARTITIONS), |
552 |
| - ConnectionProperty.createIntProperty( |
553 |
| - MAX_PARTITIONED_PARALLELISM_PROPERTY_NAME, |
554 |
| - "The maximum number of partitions that will be executed in parallel " |
555 |
| - + "for partitioned queries on this connection. Set this value to 0 to " |
556 |
| - + "dynamically use the number of processors available in the runtime.", |
557 |
| - DEFAULT_MAX_PARTITIONED_PARALLELISM), |
558 |
| - ConnectionProperty.createBooleanProperty( |
559 |
| - ENABLE_EXTENDED_TRACING_PROPERTY_NAME, |
560 |
| - "Include the SQL string in the OpenTelemetry traces that are generated " |
561 |
| - + "by this connection. The SQL string is added as the standard OpenTelemetry " |
562 |
| - + "attribute 'db.statement'.", |
563 |
| - DEFAULT_ENABLE_EXTENDED_TRACING), |
564 |
| - ConnectionProperty.createBooleanProperty( |
565 |
| - ENABLE_API_TRACING_PROPERTY_NAME, |
566 |
| - "Add OpenTelemetry traces for each individual RPC call. Enable this " |
567 |
| - + "to get a detailed view of each RPC that is being executed by your application, " |
568 |
| - + "or if you want to debug potential latency problems caused by RPCs that are " |
569 |
| - + "being retried.", |
570 |
| - DEFAULT_ENABLE_API_TRACING), |
571 |
| - ConnectionProperty.createBooleanProperty( |
572 |
| - ENABLE_END_TO_END_TRACING_PROPERTY_NAME, |
573 |
| - "Enable end-to-end tracing (true/false) to generate traces for both the time " |
574 |
| - + "that is spent in the client, as well as time that is spent in the Spanner server. " |
575 |
| - + "Server side traces can only go to Google Cloud Trace, so to see end to end traces, " |
576 |
| - + "the application should configure an exporter that exports the traces to Google Cloud Trace.", |
577 |
| - DEFAULT_ENABLE_END_TO_END_TRACING)))); |
578 |
| - |
579 |
| - private static final Set<ConnectionProperty> INTERNAL_PROPERTIES = |
580 |
| - Collections.unmodifiableSet( |
581 |
| - new HashSet<>( |
582 |
| - Collections.singletonList( |
583 |
| - ConnectionProperty.createStringProperty(USER_AGENT_PROPERTY_NAME, "")))); |
584 |
| - private static final Set<ConnectionProperty> INTERNAL_VALID_PROPERTIES = |
585 |
| - Sets.union(VALID_PROPERTIES, INTERNAL_PROPERTIES); |
586 |
| - |
587 | 295 | /**
|
588 | 296 | * Gets the default project-id for the current environment as defined by {@link
|
589 | 297 | * ServiceOptions#getDefaultProjectId()}, and if none could be found, the project-id of the given
|
|
0 commit comments