com.google.cloud.spanner.jdbc.JdbcDriver Maven / Gradle / Ivy
/*
* Copyright 2019 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.jdbc;
import com.google.api.core.InternalApi;
import com.google.auth.oauth2.GoogleCredentials;
import com.google.cloud.spanner.SessionPoolOptions;
import com.google.cloud.spanner.SessionPoolOptionsHelper;
import com.google.cloud.spanner.SpannerException;
import com.google.cloud.spanner.connection.ConnectionOptions;
import com.google.cloud.spanner.connection.ConnectionOptions.ConnectionProperty;
import com.google.rpc.Code;
import io.opentelemetry.api.OpenTelemetry;
import java.sql.Connection;
import java.sql.Driver;
import java.sql.DriverManager;
import java.sql.DriverPropertyInfo;
import java.sql.SQLException;
import java.sql.SQLFeatureNotSupportedException;
import java.sql.SQLWarning;
import java.util.Map.Entry;
import java.util.Properties;
import java.util.logging.Logger;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
/**
* JDBC {@link Driver} for Google Cloud Spanner.
*
* Usage:
*
*
*
* {@code
* String url = "jdbc:cloudspanner:/projects/my_project_id/"
* + "instances/my_instance_id/databases/my_database_name?"
* + "credentials=/home/cloudspanner-keys/my-key.json;autocommit=false";
* try (Connection connection = DriverManager.getConnection(url)) {
* try(ResultSet rs = connection.createStatement().executeQuery("SELECT SingerId, AlbumId, MarketingBudget FROM Albums")) {
* while(rs.next()) {
* // do something
* }
* }
* }
* }
*
*
*
* The connection that is returned will implement the interface {@link CloudSpannerJdbcConnection}.
* The JDBC connection URL must be specified in the following format:
*
*
* jdbc:cloudspanner:[//host[:port]]/projects/project-id[/instances/instance-id[/databases/database-name]][\?property-name=property-value[;property-name=property-value]*]?
*
*
* The property-value strings should be url-encoded.
*
* The project-id part of the URI may be filled with the placeholder DEFAULT_PROJECT_ID. This
* placeholder is replaced by the default project id of the environment that is requesting a
* connection.
*
*
The supported properties are:
*
*
* - credentials (String): URL for the credentials file to use for the connection. If you do not
* specify any credentials at all, the default credentials of the environment as returned by
* {@link GoogleCredentials#getApplicationDefault()} is used.
*
- autocommit (boolean): Sets the initial autocommit mode for the connection. Default is true.
*
- readonly (boolean): Sets the initial readonly mode for the connection. Default is false.
*
- autoConfigEmulator (boolean): Automatically configure the connection to try to connect to
* the Cloud Spanner emulator. You do not need to specify any host or port in the connection
* string as long as the emulator is running on the default host/port (localhost:9010). The
* instance and database in the connection string will automatically be created if these do
* not yet exist on the emulator. This means that you do not need to execute any `gcloud`
* commands on the emulator to create the instance and database before you can connect to it.
* Setting this property to true also enables running concurrent transactions on the emulator.
* The emulator aborts any concurrent transaction on the emulator, and the JDBC driver works
* around this by automatically setting a savepoint after each statement that is executed.
* When the transaction has been aborted by the emulator and the JDBC connection wants to
* continue with that transaction, the transaction is replayed up until the savepoint that had
* automatically been set after the last statement that was executed before the transaction
* was aborted by the emulator.
*
- endpoint (string): Set this property to specify a custom endpoint that the JDBC driver
* should connect to. You can use this property in combination with the autoConfigEmulator
* property to instruct the JDBC driver to connect to an emulator instance that uses a
* randomly assigned port numer. See ConcurrentTransactionOnEmulatorTest
* for a concrete example of how to use this property.
*
- usePlainText (boolean): Sets whether the JDBC connection should establish an unencrypted
* connection to the server. This option can only be used when connecting to a local emulator
* that does not require an encrypted connection, and that does not require authentication.
*
- optimizerVersion (string): The query optimizer version to use for the connection. The value
* must be either a valid version number or
LATEST
. If no value is specified, the
* query optimizer version specified in the environment variable
* SPANNER_OPTIMIZER_VERSION
is used. If no query optimizer version is specified in the
* connection URL or in the environment variable, the default query optimizer version of Cloud
* Spanner is used.
* - oauthtoken (String): A valid OAuth2 token to use for the JDBC connection. The token must
* have been obtained with one or both of the scopes
* 'https://www.googleapis.com/auth/spanner.admin' and/or
* 'https://www.googleapis.com/auth/spanner.data'. If you specify both a credentials file and
* an OAuth token, the JDBC driver will throw an exception when you try to obtain a
* connection.
*
- retryAbortsInternally (boolean): Sets the initial retryAbortsInternally mode for the
* connection. Default is true. @see {@link
* CloudSpannerJdbcConnection#setRetryAbortsInternally(boolean)} for more information.
*
- minSessions (int): Sets the minimum number of sessions in the backing session pool.
* Defaults to 100.
*
- maxSessions (int): Sets the maximum number of sessions in the backing session pool.
* Defaults to 400.
*
- numChannels (int): Sets the number of gRPC channels to use. Defaults to 4.
*
- rpcPriority (String): Sets the priority for all RPC invocations from this connection.
* Defaults to HIGH.
*
*/
public class JdbcDriver implements Driver {
/**
* The info {@link Properties} object that is passed to the JDBC driver may contain an entry with
* this key and an {@link io.opentelemetry.api.OpenTelemetry} instance as its value. This {@link
* io.opentelemetry.api.OpenTelemetry} instance will be used for tracing and metrics in the JDBC
* connection.
*/
public static final String OPEN_TELEMETRY_PROPERTY_KEY = "openTelemetry";
private static final String JDBC_API_CLIENT_LIB_TOKEN = "sp-jdbc";
// Updated to version 2 when upgraded to Java 8 (JDBC 4.2)
static final int MAJOR_VERSION = 2;
static final int MINOR_VERSION = 0;
private static final String JDBC_URL_FORMAT =
"jdbc:" + ConnectionOptions.Builder.SPANNER_URI_FORMAT;
private static final Pattern URL_PATTERN = Pattern.compile(JDBC_URL_FORMAT);
@InternalApi
public static String getClientLibToken() {
return JDBC_API_CLIENT_LIB_TOKEN;
}
static {
try {
register();
} catch (SQLException e) {
java.sql.DriverManager.println("Registering driver failed: " + e.getMessage());
}
}
private static JdbcDriver registeredDriver;
static void register() throws SQLException {
if (isRegistered()) {
throw new IllegalStateException(
"Driver is already registered. It can only be registered once.");
}
JdbcDriver registeredDriver = new JdbcDriver();
DriverManager.registerDriver(registeredDriver);
JdbcDriver.registeredDriver = registeredDriver;
}
/**
* According to JDBC specification, this driver is registered against {@link DriverManager} when
* the class is loaded. To avoid leaks, this method allow unregistering the driver so that the
* class can be gc'ed if necessary.
*
* @throws IllegalStateException if the driver is not registered
* @throws SQLException if deregistering the driver fails
*/
static void deregister() throws SQLException {
if (!isRegistered()) {
throw new IllegalStateException(
"Driver is not registered (or it has not been registered using Driver.register() method)");
}
ConnectionOptions.closeSpanner();
DriverManager.deregisterDriver(registeredDriver);
registeredDriver = null;
}
/** @return {@code true} if the driver is registered against {@link DriverManager} */
static boolean isRegistered() {
return registeredDriver != null;
}
/**
* @return the registered JDBC driver for Cloud Spanner.
* @throws SQLException if the driver has not been registered.
*/
static JdbcDriver getRegisteredDriver() throws SQLException {
if (isRegistered()) {
return registeredDriver;
}
throw JdbcSqlExceptionFactory.of(
"The driver has not been registered", Code.FAILED_PRECONDITION);
}
public JdbcDriver() {}
@Override
public Connection connect(String url, Properties info) throws SQLException {
if (url != null && url.startsWith("jdbc:cloudspanner")) {
try {
Matcher matcher = URL_PATTERN.matcher(url);
if (matcher.matches()) {
// strip 'jdbc:' from the URL, add any extra properties and pass on to the generic
// Connection API
String connectionUri = appendPropertiesToUrl(url.substring(5), info);
ConnectionOptions options = buildConnectionOptions(connectionUri, info);
JdbcConnection connection = new JdbcConnection(url, options);
if (options.getWarnings() != null) {
connection.pushWarning(new SQLWarning(options.getWarnings()));
}
return connection;
}
} catch (SpannerException e) {
throw JdbcSqlExceptionFactory.of(e);
} catch (IllegalArgumentException e) {
throw JdbcSqlExceptionFactory.of(e.getMessage(), Code.INVALID_ARGUMENT, e);
} catch (Exception e) {
throw JdbcSqlExceptionFactory.of(e.getMessage(), Code.UNKNOWN, e);
}
throw JdbcSqlExceptionFactory.of("invalid url: " + url, Code.INVALID_ARGUMENT);
}
return null;
}
private ConnectionOptions buildConnectionOptions(String connectionUrl, Properties info) {
ConnectionOptions.Builder builder =
ConnectionOptions.newBuilder().setTracingPrefix("JDBC").setUri(connectionUrl);
if (info.containsKey(OPEN_TELEMETRY_PROPERTY_KEY)
&& info.get(OPEN_TELEMETRY_PROPERTY_KEY) instanceof OpenTelemetry) {
builder.setOpenTelemetry((OpenTelemetry) info.get(OPEN_TELEMETRY_PROPERTY_KEY));
}
// Enable multiplexed sessions by default for the JDBC driver.
builder.setSessionPoolOptions(
SessionPoolOptionsHelper.useMultiplexedSessions(SessionPoolOptions.newBuilder()).build());
return builder.build();
}
private String appendPropertiesToUrl(String url, Properties info) {
StringBuilder res = new StringBuilder(url);
for (Entry
© 2015 - 2025 Weber Informatics LLC | Privacy Policy