tech.ydb.jdbc.context.YdbContext Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of ydb-jdbc-driver Show documentation
Show all versions of ydb-jdbc-driver Show documentation
JDBC Driver over YDB Java SDK
package tech.ydb.jdbc.context;
import java.sql.SQLException;
import java.util.Map;
import java.util.Objects;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.logging.Level;
import java.util.logging.Logger;
import tech.ydb.core.grpc.GrpcTransport;
import tech.ydb.core.grpc.GrpcTransportBuilder;
import tech.ydb.jdbc.exception.YdbConfigurationException;
import tech.ydb.jdbc.query.YdbQueryOptions;
import tech.ydb.jdbc.settings.ParsedProperty;
import tech.ydb.jdbc.settings.YdbClientProperties;
import tech.ydb.jdbc.settings.YdbClientProperty;
import tech.ydb.jdbc.settings.YdbConnectionProperties;
import tech.ydb.jdbc.settings.YdbConnectionProperty;
import tech.ydb.jdbc.settings.YdbOperationProperties;
import tech.ydb.scheme.SchemeClient;
import tech.ydb.table.TableClient;
import tech.ydb.table.impl.PooledTableClient;
import tech.ydb.table.rpc.grpc.GrpcTableRpc;
/**
*
* @author Aleksandr Gorshenin
*/
public class YdbContext implements AutoCloseable {
private static final Logger LOGGER = Logger.getLogger(YdbContext.class.getName());
private static final int SESSION_POOL_DEFAULT_MIN_SIZE = 0;
private static final int SESSION_POOL_DEFAULT_MAX_SIZE = 50;
private static final int SESSION_POOL_RESIZE_STEP = 50;
private static final int SESSION_POOL_RESIZE_THRESHOLD = 10;
private final YdbConfig config;
private final GrpcTransport grpcTransport;
private final PooledTableClient tableClient;
private final SchemeClient schemeClient;
private final YdbQueryOptions queryOptions;
private final boolean autoResizeSessionPool;
private final AtomicInteger connectionsCount = new AtomicInteger();
private YdbContext(YdbConfig config, GrpcTransport transport, PooledTableClient tableClient, boolean autoResize) {
this.config = config;
this.grpcTransport = Objects.requireNonNull(transport);
this.tableClient = Objects.requireNonNull(tableClient);
this.schemeClient = SchemeClient.newClient(transport).build();
this.queryOptions = YdbQueryOptions.createFrom(config.getOperationProperties());
this.autoResizeSessionPool = autoResize;
}
public String getDatabase() {
return grpcTransport.getDatabase();
}
public SchemeClient getSchemeClient() {
return schemeClient;
}
public TableClient getTableClient() {
return tableClient;
}
public String getUrl() {
return config.getUrl();
}
public YdbQueryOptions getQueryOptions() {
return queryOptions;
}
public int getConnectionsCount() {
return connectionsCount.get();
}
public YdbOperationProperties getOperationProperties() {
return config.getOperationProperties();
}
@Override
public void close() {
try {
schemeClient.close();
tableClient.close();
grpcTransport.close();
} catch (Exception e) {
LOGGER.log(Level.SEVERE, "Unable to close client: " + e.getMessage(), e);
}
}
public void register() {
int actual = connectionsCount.incrementAndGet();
int maxSize = tableClient.sessionPoolStats().getMaxSize();
if (autoResizeSessionPool && actual > maxSize - SESSION_POOL_RESIZE_THRESHOLD) {
int newSize = maxSize + SESSION_POOL_RESIZE_STEP;
if (maxSize == tableClient.sessionPoolStats().getMaxSize()) {
tableClient.updatePoolMaxSize(newSize);
}
}
}
public void deregister() {
int actual = connectionsCount.decrementAndGet();
int maxSize = tableClient.sessionPoolStats().getMaxSize();
if (autoResizeSessionPool && maxSize > SESSION_POOL_RESIZE_STEP) {
if (actual < maxSize - SESSION_POOL_RESIZE_STEP - 2 * SESSION_POOL_RESIZE_THRESHOLD) {
int newSize = maxSize - SESSION_POOL_RESIZE_STEP;
if (maxSize == tableClient.sessionPoolStats().getMaxSize()) {
tableClient.updatePoolMaxSize(newSize);
}
}
}
}
public static YdbContext createContext(YdbConfig config) throws SQLException {
try {
YdbConnectionProperties connProps = config.getConnectionProperties();
YdbClientProperties clientProps = config.getClientProperties();
LOGGER.log(Level.INFO, "Creating new YDB connection to {0}", connProps.getConnectionString());
GrpcTransport grpcTransport = buildGrpcTransport(connProps);
PooledTableClient.Builder tableClient = PooledTableClient.newClient(
GrpcTableRpc.useTransport(grpcTransport)
);
boolean autoResize = buildTableClient(tableClient, clientProps);
return new YdbContext(config, grpcTransport, tableClient.build(), autoResize);
} catch (Exception ex) {
throw new YdbConfigurationException("Cannot connect to YDB", ex);
}
}
public static GrpcTransport buildGrpcTransport(YdbConnectionProperties props) {
GrpcTransportBuilder builder = GrpcTransport.forConnectionString(props.getConnectionString());
for (Map.Entry, ParsedProperty> entry : props.getParams().entrySet()) {
if (entry.getValue() != null) {
entry.getKey().getSetter().accept(builder, entry.getValue().getParsedValue());
}
}
if (props.hasStaticCredentials()) {
builder = builder.withAuthProvider(props.getStaticCredentials());
}
return builder.build();
}
private static boolean buildTableClient(TableClient.Builder builder, YdbClientProperties props) {
for (Map.Entry, ParsedProperty> entry : props.getParams().entrySet()) {
if (entry.getValue() != null) {
entry.getKey().getSetter().accept(builder, entry.getValue().getParsedValue());
}
}
ParsedProperty minSizeConfig = props.getProperty(YdbClientProperty.SESSION_POOL_SIZE_MIN);
ParsedProperty maxSizeConfig = props.getProperty(YdbClientProperty.SESSION_POOL_SIZE_MAX);
if (minSizeConfig == null && maxSizeConfig == null) {
return true;
}
int minSize = SESSION_POOL_DEFAULT_MIN_SIZE;
int maxSize = SESSION_POOL_DEFAULT_MAX_SIZE;
if (minSizeConfig != null) {
minSize = Math.max(0, minSizeConfig.getParsedValue());
maxSize = Math.max(maxSize, minSize);
}
if (maxSizeConfig != null) {
maxSize = Math.max(minSize + 1, maxSizeConfig.getParsedValue());
}
builder.sessionPoolSize(minSize, maxSize);
return false;
}
}