From 3bd69f766b9453fbe7205aab35b9d3b3cecc2770 Mon Sep 17 00:00:00 2001 From: Alexandr Gorshenin Date: Fri, 8 Nov 2024 15:50:44 +0000 Subject: [PATCH 1/2] Improve logging of driver and statuses --- jdbc/src/main/java/tech/ydb/jdbc/YdbDriver.java | 4 ++-- .../java/tech/ydb/jdbc/context/BaseYdbExecutor.java | 10 +--------- .../tech/ydb/jdbc/context/QueryServiceExecutor.java | 11 +---------- .../main/java/tech/ydb/jdbc/context/YdbContext.java | 2 +- .../main/java/tech/ydb/jdbc/context/YdbValidator.java | 7 +++++++ 5 files changed, 12 insertions(+), 22 deletions(-) diff --git a/jdbc/src/main/java/tech/ydb/jdbc/YdbDriver.java b/jdbc/src/main/java/tech/ydb/jdbc/YdbDriver.java index bce20e4..88e7455 100644 --- a/jdbc/src/main/java/tech/ydb/jdbc/YdbDriver.java +++ b/jdbc/src/main/java/tech/ydb/jdbc/YdbDriver.java @@ -47,7 +47,7 @@ public YdbConnection connect(String url, Properties info) throws SQLException { } YdbConfig config = YdbConfig.from(url, info); - LOGGER.log(Level.FINE, "About to connect to [{0}] using properties {1}", new Object[] { + LOGGER.log(Level.FINE, "Connect to [{0}] using properties {1}", new Object[] { config.getSafeUrl(), config.getSafeProps() }); @@ -80,7 +80,7 @@ public YdbContext getCachedContext(YdbConfig config) throws SQLException { // Was fixed in Java 9+ YdbContext context = cache.get(config); if (context != null) { - LOGGER.log(Level.FINE, "Reusing YDB connection to {0}", config.getSafeUrl()); + LOGGER.log(Level.FINEST, "Reusing YDB connection to {0}", config.getSafeUrl()); return context; } diff --git a/jdbc/src/main/java/tech/ydb/jdbc/context/BaseYdbExecutor.java b/jdbc/src/main/java/tech/ydb/jdbc/context/BaseYdbExecutor.java index 09c9701..eda2eab 100644 --- a/jdbc/src/main/java/tech/ydb/jdbc/context/BaseYdbExecutor.java +++ b/jdbc/src/main/java/tech/ydb/jdbc/context/BaseYdbExecutor.java @@ -7,12 +7,10 @@ import java.util.concurrent.atomic.AtomicReference; import tech.ydb.core.Result; -import tech.ydb.core.UnexpectedResultException; import tech.ydb.core.grpc.GrpcReadStream; import tech.ydb.jdbc.YdbConst; import tech.ydb.jdbc.YdbStatement; import tech.ydb.jdbc.YdbTracer; -import tech.ydb.jdbc.exception.ExceptionFactory; import tech.ydb.jdbc.impl.YdbQueryResult; import tech.ydb.jdbc.query.QueryType; import tech.ydb.jdbc.query.YdbQuery; @@ -48,13 +46,7 @@ public BaseYdbExecutor(YdbContext ctx) { } protected Session createNewTableSession(YdbValidator validator) throws SQLException { - try { - Result session = tableClient.createSession(sessionTimeout).join(); - validator.addStatusIssues(session.getStatus()); - return session.getValue(); - } catch (UnexpectedResultException ex) { - throw ExceptionFactory.createException("Cannot create session with " + ex.getStatus(), ex); - } + return validator.call("Get session", () -> tableClient.createSession(sessionTimeout)); } protected void closeCurrentResult() throws SQLException { diff --git a/jdbc/src/main/java/tech/ydb/jdbc/context/QueryServiceExecutor.java b/jdbc/src/main/java/tech/ydb/jdbc/context/QueryServiceExecutor.java index 1c10508..f8fd06a 100644 --- a/jdbc/src/main/java/tech/ydb/jdbc/context/QueryServiceExecutor.java +++ b/jdbc/src/main/java/tech/ydb/jdbc/context/QueryServiceExecutor.java @@ -15,12 +15,10 @@ import tech.ydb.common.transaction.TxMode; import tech.ydb.core.Issue; import tech.ydb.core.Result; -import tech.ydb.core.UnexpectedResultException; import tech.ydb.jdbc.YdbConst; import tech.ydb.jdbc.YdbResultSet; import tech.ydb.jdbc.YdbStatement; import tech.ydb.jdbc.YdbTracer; -import tech.ydb.jdbc.exception.ExceptionFactory; import tech.ydb.jdbc.impl.YdbQueryResult; import tech.ydb.jdbc.impl.YdbStaticResultSet; import tech.ydb.jdbc.query.QueryType; @@ -70,14 +68,7 @@ public QueryServiceExecutor(YdbContext ctx, int transactionLevel, boolean autoCo } protected QuerySession createNewQuerySession(YdbValidator validator) throws SQLException { - try { - Result result = queryClient.createSession(sessionTimeout).join(); - validator.addStatusIssues(result.getStatus()); - QuerySession session = result.getValue(); - return session; - } catch (UnexpectedResultException ex) { - throw ExceptionFactory.createException("Cannot create session with " + ex.getStatus(), ex); - } + return validator.call("Get query session", () -> queryClient.createSession(sessionTimeout)); } @Override diff --git a/jdbc/src/main/java/tech/ydb/jdbc/context/YdbContext.java b/jdbc/src/main/java/tech/ydb/jdbc/context/YdbContext.java index 200a54f..fad0c0d 100644 --- a/jdbc/src/main/java/tech/ydb/jdbc/context/YdbContext.java +++ b/jdbc/src/main/java/tech/ydb/jdbc/context/YdbContext.java @@ -271,7 +271,7 @@ public void deregister() { public static YdbContext createContext(YdbConfig config) throws SQLException { try { - LOGGER.log(Level.INFO, "Creating new YDB connection to {0}", config.getConnectionString()); + LOGGER.log(Level.FINE, "Creating new YDB context to {0}", config.getConnectionString()); YdbConnectionProperties connProps = new YdbConnectionProperties(config); YdbClientProperties clientProps = new YdbClientProperties(config); diff --git a/jdbc/src/main/java/tech/ydb/jdbc/context/YdbValidator.java b/jdbc/src/main/java/tech/ydb/jdbc/context/YdbValidator.java index 7b83a24..512854c 100644 --- a/jdbc/src/main/java/tech/ydb/jdbc/context/YdbValidator.java +++ b/jdbc/src/main/java/tech/ydb/jdbc/context/YdbValidator.java @@ -7,6 +7,8 @@ import java.util.List; import java.util.concurrent.CompletableFuture; import java.util.function.Supplier; +import java.util.logging.Level; +import java.util.logging.Logger; import tech.ydb.core.Issue; import tech.ydb.core.Result; @@ -20,6 +22,8 @@ * @author Aleksandr Gorshenin */ public class YdbValidator { + private static final Logger LOGGER = Logger.getLogger(YdbValidator.class.getName()); + private final List issues = new ArrayList<>(); public SQLWarning toSQLWarnings() { @@ -56,6 +60,7 @@ public void execute(String msg, YdbTracer tracer, Supplier R call(String msg, Supplier>> fn) throws addStatusIssues(result.getStatus()); return result.getValue(); } catch (UnexpectedResultException ex) { + LOGGER.log(Level.FINE, "call problem {0}", ex.getStatus()); throw ExceptionFactory.createException("Cannot call '" + msg + "' with " + ex.getStatus(), ex); } } @@ -80,6 +86,7 @@ public R call(String msg, YdbTracer tracer, Supplier Date: Mon, 11 Nov 2024 17:56:32 +0000 Subject: [PATCH 2/2] Added option cachedTransportsCount to shard JDBC connections --- .../tech/ydb/jdbc/settings/YdbConfig.java | 18 +++++++++++++-- .../ydb/jdbc/YdbDriverIntegrationTest.java | 23 +++++++++++++++++++ 2 files changed, 39 insertions(+), 2 deletions(-) diff --git a/jdbc/src/main/java/tech/ydb/jdbc/settings/YdbConfig.java b/jdbc/src/main/java/tech/ydb/jdbc/settings/YdbConfig.java index 4bdea69..5643233 100644 --- a/jdbc/src/main/java/tech/ydb/jdbc/settings/YdbConfig.java +++ b/jdbc/src/main/java/tech/ydb/jdbc/settings/YdbConfig.java @@ -10,6 +10,7 @@ import java.util.Map; import java.util.Objects; import java.util.Properties; +import java.util.concurrent.ThreadLocalRandom; import tech.ydb.core.utils.URITools; import tech.ydb.jdbc.YdbConst; @@ -50,6 +51,9 @@ public class YdbConfig { static final YdbProperty TRANSACTION_TRACER = YdbProperty.bool( "enableTxTracer", "Enable collecting of transaction execution traces", false ); + static final YdbProperty CACHED_TRANSPORT_COUNT = YdbProperty.integer( + "cachedTransportsCount", "Use specified count of YDB transports in context cache", 1 + ); private final String url; private final String username; @@ -67,6 +71,7 @@ public class YdbConfig { private final boolean fullScanDetectorEnabled; private final boolean txTracerEnabled; + private final int transportIndex; private YdbConfig( String url, String safeUrl, String connectionString, String username, String password, Properties props @@ -85,6 +90,13 @@ private YdbConfig( this.fullScanDetectorEnabled = FULLSCAN_DETECTOR_ENABLED.readValue(props).getValue(); this.txTracerEnabled = TRANSACTION_TRACER.readValue(props).getValue(); + + int transportsCount = CACHED_TRANSPORT_COUNT.readValue(props).getValue(); + if (transportsCount > 1) { + this.transportIndex = ThreadLocalRandom.current().nextInt(transportsCount); + } else { + this.transportIndex = 0; + } } public Properties getSafeProps() { @@ -144,12 +156,14 @@ public boolean equals(Object o) { return false; } YdbConfig that = (YdbConfig) o; - return Objects.equals(url, that.url) && Objects.equals(properties, that.properties); + return Objects.equals(url, that.url) + && Objects.equals(properties, that.properties) + && transportIndex == that.transportIndex; } @Override public int hashCode() { - return Objects.hash(url, properties); + return Objects.hash(url, properties, transportIndex); } public String getUrl() { diff --git a/jdbc/src/test/java/tech/ydb/jdbc/YdbDriverIntegrationTest.java b/jdbc/src/test/java/tech/ydb/jdbc/YdbDriverIntegrationTest.java index 58aebeb..dc54116 100644 --- a/jdbc/src/test/java/tech/ydb/jdbc/YdbDriverIntegrationTest.java +++ b/jdbc/src/test/java/tech/ydb/jdbc/YdbDriverIntegrationTest.java @@ -7,8 +7,10 @@ import java.util.ArrayDeque; import java.util.ArrayList; import java.util.Deque; +import java.util.HashSet; import java.util.List; import java.util.Properties; +import java.util.Set; import java.util.concurrent.CompletableFuture; import org.junit.jupiter.api.Assertions; @@ -195,6 +197,27 @@ public void testContextCacheConncurrent() throws SQLException { } } + @Test + public void testCachedTransportCount() throws SQLException { + Properties props = new Properties(); + props.put("cachedTransportsCount", "3"); + + List list = new ArrayList<>(); + + Set contexts = new HashSet<>(); + for (int idx = 0; idx < 20; idx++) { + Connection connection = DriverManager.getConnection(jdbcURL.build(), props); + contexts.add(connection.unwrap(YdbConnection.class).getCtx()); + list.add(connection); + } + + Assertions.assertEquals(3, contexts.size()); + + for (Connection connection: list) { + connection.close(); + } + } + @Test public void testResizeSessionPool() throws SQLException { String url = jdbcURL.build();