com.github.cassandra.jdbc.provider.datastax.DataStaxSessionFactory Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of cassandra-jdbc-driver Show documentation
Show all versions of cassandra-jdbc-driver Show documentation
Type 4 JDBC driver for Apache Cassandra built on top of existing great libs like java driver from
DataStax. It supports Cassandra 2.x and above with improved SQL compatibility.
/**
* Copyright (C) 2015-2017, Zhichun Wu
*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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.github.cassandra.jdbc.provider.datastax;
import com.datastax.driver.core.*;
import com.datastax.driver.core.policies.DCAwareRoundRobinPolicy;
import com.datastax.driver.core.policies.RoundRobinPolicy;
import com.datastax.driver.extras.codecs.joda.InstantCodec;
import com.datastax.driver.extras.codecs.joda.LocalDateCodec;
import com.datastax.driver.extras.codecs.joda.LocalTimeCodec;
import com.github.cassandra.jdbc.CassandraConfiguration;
import com.github.cassandra.jdbc.provider.datastax.codecs.JavaSqlTimeCodec;
import com.google.common.base.Splitter;
import com.google.common.base.Strings;
import com.google.common.cache.Cache;
import com.google.common.cache.CacheBuilder;
import com.google.common.cache.RemovalListener;
import com.google.common.cache.RemovalNotification;
import com.google.common.reflect.ClassPath;
import org.pmw.tinylog.Logger;
import java.util.concurrent.Callable;
/**
* Session factory.
*/
final class DataStaxSessionFactory {
private final static Cache _sessionCache;
static {
_sessionCache = CacheBuilder.newBuilder().weakValues().removalListener(
new RemovalListener() {
public void onRemoval(RemovalNotification notification) {
DataStaxSessionWrapper session = notification.getValue();
Logger.debug("Closing [{}] (cause: {})...", session, notification.getCause());
if (session != null) {
try {
session.close();
} catch (Throwable t) {
Logger.debug(t, "Error occurred when closing session");
}
}
Logger.debug("Closed [{0}].", session);
}
}).build();
}
private static DataStaxSessionWrapper newSession(CassandraConfiguration config) {
return newSession(config, null);
}
private static DataStaxSessionWrapper newSession(CassandraConfiguration config, String keyspace) {
keyspace = Strings.isNullOrEmpty(keyspace)
|| config.getKeyspace().equals(keyspace) ? config.getKeyspace() : keyspace;
Logger.debug("Connecting to [{}]...", config.getConnectionUrl());
Cluster.Builder builder = Cluster.builder();
// add contact points
for (String host : Splitter.on(',').trimResults().omitEmptyStrings().split(config.getHosts())) {
builder.addContactPoint(host);
}
// set port if specified
if (config.getPort() > 0) {
builder.withPort(config.getPort());
}
// set socket options
SocketOptions socketOptions = new SocketOptions();
socketOptions.setConnectTimeoutMillis(config.getConnectionTimeout());
socketOptions.setReadTimeoutMillis(config.getReadTimeout());
socketOptions.setKeepAlive(config.isKeepAlive());
builder.withSocketOptions(socketOptions);
// set query options
QueryOptions queryOptions = new QueryOptions();
queryOptions.setConsistencyLevel(ConsistencyLevel.valueOf(config.getConsistencyLevel().name()));
if (config.getFetchSize() > 0) {
queryOptions.setFetchSize(config.getFetchSize());
}
builder.withQueryOptions(queryOptions);
// set pool options - use same defaults as in V3
PoolingOptions poolOptions = new PoolingOptions();
poolOptions.setConnectionsPerHost(HostDistance.LOCAL,
config.getAdditionalProperty("corePoolLocal", 1),
config.getAdditionalProperty("maxPoolLocal", 1));
poolOptions.setConnectionsPerHost(HostDistance.REMOTE,
config.getAdditionalProperty("corePoolRemote", 1),
config.getAdditionalProperty("maxPoolRemote", 1));
poolOptions.setIdleTimeoutSeconds(
config.getAdditionalProperty("idleTimeoutSeconds", poolOptions.getIdleTimeoutSeconds()));
poolOptions.setPoolTimeoutMillis(
config.getAdditionalProperty("poolTimeoutMillis", poolOptions.getPoolTimeoutMillis()));
poolOptions.setHeartbeatIntervalSeconds(
config.getAdditionalProperty("heartbeatIntervalSeconds", poolOptions.getHeartbeatIntervalSeconds()));
poolOptions.setMaxRequestsPerConnection(HostDistance.LOCAL,
config.getAdditionalProperty("maxRequestsPerConnectionLocal", 800));
poolOptions.setMaxRequestsPerConnection(HostDistance.REMOTE,
config.getAdditionalProperty("maxRequestsPerConnectionRemote", 200));
poolOptions.setNewConnectionThreshold(HostDistance.LOCAL,
config.getAdditionalProperty("newConnectionThresholdLocal", 1024));
poolOptions.setNewConnectionThreshold(HostDistance.REMOTE,
config.getAdditionalProperty("newConnectionThresholdRemote", 256));
builder.withPoolingOptions(poolOptions);
// set compression
builder.withCompression(ProtocolOptions.Compression.valueOf(config.getCompression().name()));
// add custom codecs
CodecRegistry registry = new CodecRegistry();
registry.register(LocalDateCodec.instance, LocalTimeCodec.instance, InstantCodec.instance);
String packageName = JavaSqlTimeCodec.class.getPackage().getName();
try {
// FIXME one exception will stop loading the rest codecs
for (ClassPath.ClassInfo info : ClassPath.from(
DataStaxSessionFactory.class.getClassLoader()).getTopLevelClasses()) {
if (packageName.equals(info.getPackageName())) {
Logger.debug("Registering codec: {}", info.getName());
registry.register((TypeCodec) info.load().getField("instance").get(null));
}
}
} catch (Exception e) {
Logger.warn(e, "Failed to register codec");
}
builder.withCodecRegistry(registry);
// FIXME set policies based on configuration
if (!Strings.isNullOrEmpty(config.getLocalDc())) {
builder.withLoadBalancingPolicy(DCAwareRoundRobinPolicy.builder().withLocalDc(config.getLocalDc()).build());
} else {
builder.withLoadBalancingPolicy(new RoundRobinPolicy());
}
// build the cluster
Cluster cluster = builder.withCredentials(config.getUserName(),
config.getPassword()).build();
Logger.debug("Connected to [{}({})] successfully", config.getConnectionUrl(), cluster.hashCode());
Metadata metadata = cluster.getMetadata();
Logger.info("Connected to cluster@{}: {}", cluster.hashCode(), metadata.getClusterName());
for (Host host : metadata.getAllHosts()) {
Logger.info("-> Datacenter: {}, Host: {}, Rack: {}",
host.getDatacenter(),
host.getAddress(),
host.getRack());
}
return new DataStaxSessionWrapper(cluster.connect(keyspace));
}
static DataStaxSessionWrapper getSession(final CassandraConfiguration config) {
return getSession(config, null);
}
static DataStaxSessionWrapper getSession(final CassandraConfiguration config, final String keyspace) {
final String targetKeyspace = Strings.isNullOrEmpty(keyspace)
|| config.getKeyspace().equals(keyspace) ? config.getKeyspace() : keyspace;
DataStaxSessionWrapper session = null;
try {
session = _sessionCache.get(config.getConnectionUrl(), new Callable() {
public DataStaxSessionWrapper call() throws Exception {
return newSession(config, targetKeyspace);
}
});
if (session.isClosed() || !session.getLoggedKeyspace().equals(targetKeyspace)) {
// FIXME this will cause connection issues in other threads
_sessionCache.invalidate(config.getConnectionUrl());
session = getSession(config, targetKeyspace);
}
// active the session and increase the reference counter
session.open();
} catch (Exception e) {
Logger.error(e, "Failed to obtain session object");
throw new RuntimeException(e);
}
return session;
}
}