From b95fae518a917c02c92598dc2000ae8426e73c31 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jakub=20Szczerbi=C5=84ski?= Date: Fri, 23 Aug 2024 12:25:59 +0200 Subject: [PATCH 1/7] Test if rest request is retried after connection is closed --- .../client/jdbc/BaseWiremockTest.java | 242 ++++++++++++ .../snowflake/client/jdbc/ProxyLatestIT.java | 360 ++++-------------- .../client/jdbc/RestRequestTest.java | 24 ++ .../client/jdbc/RestRequestWiremockTest.java | 74 ++++ 4 files changed, 419 insertions(+), 281 deletions(-) create mode 100644 src/test/java/net/snowflake/client/jdbc/BaseWiremockTest.java create mode 100644 src/test/java/net/snowflake/client/jdbc/RestRequestWiremockTest.java diff --git a/src/test/java/net/snowflake/client/jdbc/BaseWiremockTest.java b/src/test/java/net/snowflake/client/jdbc/BaseWiremockTest.java new file mode 100644 index 000000000..e0376ec8d --- /dev/null +++ b/src/test/java/net/snowflake/client/jdbc/BaseWiremockTest.java @@ -0,0 +1,242 @@ +package net.snowflake.client.jdbc; + +import static junit.framework.TestCase.assertEquals; +import static net.snowflake.client.AbstractDriverIT.getConnectionParameters; +import static net.snowflake.client.jdbc.SnowflakeUtil.systemGetProperty; +import static org.awaitility.Awaitility.await; +import static org.junit.Assume.*; + +import java.io.File; +import java.io.IOException; +import java.io.UnsupportedEncodingException; +import java.net.ServerSocket; +import java.nio.file.Paths; +import java.sql.*; +import java.time.Duration; +import java.util.Map; +import java.util.Properties; +import net.snowflake.client.RunningNotOnGithubActionsMac; +import net.snowflake.client.RunningNotOnJava21; +import net.snowflake.client.RunningNotOnJava8; +import net.snowflake.client.core.HttpUtil; +import net.snowflake.client.log.SFLogger; +import net.snowflake.client.log.SFLoggerFactory; +import org.apache.http.client.methods.CloseableHttpResponse; +import org.apache.http.client.methods.HttpGet; +import org.apache.http.client.methods.HttpPost; +import org.apache.http.entity.StringEntity; +import org.apache.http.impl.client.CloseableHttpClient; +import org.apache.http.impl.client.HttpClients; +import org.junit.After; +import org.junit.AfterClass; +import org.junit.BeforeClass; + +public abstract class BaseWiremockTest { + + protected static final SFLogger logger = SFLoggerFactory.getLogger(BaseWiremockTest.class); + protected static final String WIREMOCK_HOME_DIR = ".wiremock"; + protected static final String WIREMOCK_M2_PATH = + "/.m2/repository/org/wiremock/wiremock-standalone/3.8.0/wiremock-standalone-3.8.0.jar"; + protected static final String WIREMOCK_HOST = "localhost"; + protected static final String TRUST_STORE_PROPERTY = "javax.net.ssl.trustStore"; + protected static int httpProxyPort; + protected static int httpsProxyPort; + private static String originalTrustStorePath; + protected static Process wiremockStandalone; + + @BeforeClass + public static void setUpClass() { + assumeFalse(RunningNotOnJava8.isRunningOnJava8()); + assumeFalse(RunningNotOnJava21.isRunningOnJava21()); + assumeFalse( + RunningNotOnGithubActionsMac + .isRunningOnGithubActionsMac()); // disabled until issue with access to localhost + // (https://github.com/snowflakedb/snowflake-jdbc/pull/1807#discussion_r1686229430) is fixed on + // github actions mac image. Ticket to enable when fixed: SNOW-1555950 + originalTrustStorePath = systemGetProperty(TRUST_STORE_PROPERTY); + startWiremockStandAlone(); + } + + @After + public void tearDown() { + restoreTrustStorePathProperty(); + resetWiremock(); + HttpUtil.httpClient.clear(); + } + + @AfterClass + public static void tearDownClass() { + stopWiremockStandAlone(); + } + + protected static void startWiremockStandAlone() { + // retrying in case of fail in port bindings + await() + .alias("wait for wiremock responding") + .atMost(Duration.ofSeconds(10)) + .until( + () -> { + try { + httpProxyPort = findFreePort(); + httpsProxyPort = findFreePort(); + wiremockStandalone = + new ProcessBuilder( + "java", + "-jar", + getWiremockStandAlonePath(), + "--root-dir", + System.getProperty("user.dir") + + File.separator + + WIREMOCK_HOME_DIR + + File.separator, + "--enable-browser-proxying", // work as forward proxy + "--proxy-pass-through", + "false", // pass through only matched requests + "--port", + String.valueOf(httpProxyPort), + "--https-port", + String.valueOf(httpsProxyPort), + "--https-keystore", + getResourceURL("wiremock" + File.separator + "ca-cert.jks"), + "--ca-keystore", + getResourceURL("wiremock" + File.separator + "ca-cert.jks")) + .inheritIO() + .start(); + waitForWiremock(); + return true; + } catch (Exception e) { + logger.warn("Failed to start wiremock, retrying: ", e); + return false; + } + }); + } + + protected void resetWiremock() { + HttpPost postRequest; + postRequest = new HttpPost("http://" + WIREMOCK_HOST + ":" + getAdminPort() + "/__admin/reset"); + try (CloseableHttpClient client = HttpClients.createDefault(); + CloseableHttpResponse response = client.execute(postRequest)) { + assertEquals(200, response.getStatusLine().getStatusCode()); + } catch (IOException e) { + throw new RuntimeException(e); + } + } + + private static String getWiremockStandAlonePath() { + return System.getProperty("user.home") + WIREMOCK_M2_PATH; + } + + private static void waitForWiremock() { + await() + .pollDelay(Duration.ofSeconds(1)) + .atMost(Duration.ofSeconds(3)) + .until(BaseWiremockTest::isWiremockResponding); + } + + private static boolean isWiremockResponding() { + try (CloseableHttpClient httpClient = HttpClients.createDefault()) { + HttpGet request = + new HttpGet(String.format("http://%s:%d/__admin/mappings", WIREMOCK_HOST, httpProxyPort)); + CloseableHttpResponse response = httpClient.execute(request); + return response.getStatusLine().getStatusCode() == 200; + } catch (Exception e) { + logger.warn("Waiting for wiremock to respond: ", e); + } + return false; + } + + protected static void stopWiremockStandAlone() { + if (wiremockStandalone != null) { + wiremockStandalone.destroyForcibly(); + await() + .alias("stop wiremock") + .atMost(Duration.ofSeconds(10)) + .until(() -> !wiremockStandalone.isAlive()); + } + } + + private static int findFreePort() { + try { + ServerSocket socket = new ServerSocket(0); + int port = socket.getLocalPort(); + socket.close(); + return port; + } catch (Exception e) { + throw new RuntimeException(e); + } + } + + protected Properties getProperties() { + Map params = getConnectionParameters(); + Properties props = new Properties(); + props.put("host", params.get("host")); + props.put("port", params.get("port")); + props.put("account", params.get("account")); + props.put("user", params.get("user")); + props.put("role", params.get("role")); + props.put("password", params.get("password")); + props.put("warehouse", params.get("warehouse")); + props.put("db", params.get("database")); + props.put("ssl", params.get("ssl")); + props.put("insecureMode", true); // OCSP disabled for wiremock proxy tests + return props; + } + + protected HttpPost createWiremockPostRequest(String body, String path) { + HttpPost postRequest = new HttpPost("http://" + WIREMOCK_HOST + ":" + getAdminPort() + path); + final StringEntity entity; + try { + entity = new StringEntity(body); + } catch (UnsupportedEncodingException e) { + throw new RuntimeException(e); + } + postRequest.setEntity(entity); + postRequest.setHeader("Accept", "application/json"); + postRequest.setHeader("Content-type", "application/json"); + return postRequest; + } + + protected static void restoreTrustStorePathProperty() { + if (originalTrustStorePath != null) { + System.setProperty(TRUST_STORE_PROPERTY, originalTrustStorePath); + } else { + System.clearProperty(TRUST_STORE_PROPERTY); + } + } + + private int getAdminPort() { + return httpProxyPort; + } + + private static String getResourceURL(String relativePath) { + return Paths.get(systemGetProperty("user.dir"), "src", "test", "resources", relativePath) + .toAbsolutePath() + .toString(); + } + + protected void setCustomTrustStorePropertyPath() { + System.setProperty( + TRUST_STORE_PROPERTY, getResourceURL("wiremock" + File.separator + "ca-cert.jks")); + } + + protected void importMapping(String mappingImport) { + HttpPost request = createWiremockPostRequest(mappingImport, "/__admin/mappings/import"); + try (CloseableHttpClient httpClient = HttpClients.createDefault(); + CloseableHttpResponse response = httpClient.execute(request)) { + assumeTrue(response.getStatusLine().getStatusCode() == 200); + } catch (Exception e) { + logger.error("Importing mapping failed", e); + assumeNoException(e); + } + } + + protected void addMapping(String mapping) { + HttpPost postRequest = createWiremockPostRequest(mapping, "/__admin/mappings"); + try (CloseableHttpClient client = HttpClients.createDefault(); + CloseableHttpResponse response = client.execute(postRequest)) { + assertEquals(201, response.getStatusLine().getStatusCode()); + } catch (IOException e) { + throw new RuntimeException(e); + } + } +} diff --git a/src/test/java/net/snowflake/client/jdbc/ProxyLatestIT.java b/src/test/java/net/snowflake/client/jdbc/ProxyLatestIT.java index d3ee887dd..22781d3d9 100644 --- a/src/test/java/net/snowflake/client/jdbc/ProxyLatestIT.java +++ b/src/test/java/net/snowflake/client/jdbc/ProxyLatestIT.java @@ -1,269 +1,49 @@ package net.snowflake.client.jdbc; import static junit.framework.TestCase.assertEquals; -import static net.snowflake.client.AbstractDriverIT.getConnectionParameters; -import static net.snowflake.client.jdbc.SnowflakeUtil.systemGetProperty; -import static org.awaitility.Awaitility.await; import static org.junit.Assert.assertTrue; -import static org.junit.Assume.assumeFalse; import com.fasterxml.jackson.databind.JsonNode; import com.fasterxml.jackson.databind.ObjectMapper; -import java.io.File; import java.io.IOException; -import java.io.UnsupportedEncodingException; -import java.net.ServerSocket; -import java.nio.file.Paths; import java.sql.Connection; import java.sql.DriverManager; import java.sql.ResultSet; import java.sql.SQLException; import java.sql.Statement; -import java.time.Duration; -import java.util.Map; import java.util.Objects; import java.util.Properties; -import net.snowflake.client.RunningNotOnGithubActionsMac; -import net.snowflake.client.RunningNotOnJava21; -import net.snowflake.client.RunningNotOnJava8; import net.snowflake.client.category.TestCategoryOthers; -import net.snowflake.client.core.HttpUtil; -import net.snowflake.client.log.SFLogger; -import net.snowflake.client.log.SFLoggerFactory; import org.apache.http.client.methods.CloseableHttpResponse; -import org.apache.http.client.methods.HttpGet; import org.apache.http.client.methods.HttpPost; -import org.apache.http.entity.StringEntity; import org.apache.http.impl.client.CloseableHttpClient; import org.apache.http.impl.client.HttpClients; import org.apache.http.util.EntityUtils; import org.junit.After; -import org.junit.AfterClass; -import org.junit.BeforeClass; import org.junit.Test; import org.junit.experimental.categories.Category; @Category(TestCategoryOthers.class) -public class ProxyLatestIT { - - private static final SFLogger logger = SFLoggerFactory.getLogger(ProxyLatestIT.class); - private static final String WIREMOCK_HOME_DIR = ".wiremock"; - private static final String WIREMOCK_M2_PATH = - "/.m2/repository/org/wiremock/wiremock-standalone/3.8.0/wiremock-standalone-3.8.0.jar"; - private static final String WIREMOCK_HOST = "localhost"; - private static final String TRUST_STORE_PROPERTY = "javax.net.ssl.trustStore"; - private static int httpProxyPort; - private static int httpsProxyPort; - private static String originalTrustStorePath; - private static Process wiremockStandalone; - - @BeforeClass - public static void setUpClass() { - assumeFalse(RunningNotOnJava8.isRunningOnJava8()); - assumeFalse(RunningNotOnJava21.isRunningOnJava21()); - assumeFalse( - RunningNotOnGithubActionsMac - .isRunningOnGithubActionsMac()); // disabled until issue with access to localhost - // (https://github.com/snowflakedb/snowflake-jdbc/pull/1807#discussion_r1686229430) is fixed on - // github actions mac image. Ticket to enable when fixed: SNOW-1555950 - originalTrustStorePath = systemGetProperty(TRUST_STORE_PROPERTY); - startWiremockStandAlone(); - } +public class ProxyLatestIT extends BaseWiremockTest { @After public void tearDown() { - restoreTrustStorePathProperty(); + super.tearDown(); unsetJvmProperties(); - resetWiremock(); - HttpUtil.httpClient.clear(); - } - - @AfterClass - public static void tearDownClass() { - stopWiremockStandAlone(); - } - - @Test - public void testProxyIsUsedWhenSetInProperties() throws SQLException { - setCustomTrustStorePropertyPath(); - Properties props = getProperties(); - addProxyProperties(props); - proxyAll(props); - - connectAndVerifySimpleQuery(props); - verifyProxyWasUsed(); - } - - @Test - public void testProxyIsUsedWhenSetInJVMParams() throws SQLException { - setCustomTrustStorePropertyPath(); - Properties props = getProperties(); - setJvmProperties(props); - proxyAll(props); - - connectAndVerifySimpleQuery(props); - verifyProxyWasUsed(); - } - - @Test - public void testProxyNotUsedWhenNonProxyHostsMatchingInProperties() throws SQLException { - Properties props = getProperties(); - addProxyProperties(props); - props.put("nonProxyHosts", "*"); - proxyAll(props); - - connectAndVerifySimpleQuery(props); - verifyProxyNotUsed(); - } - - @Test - public void testProxyIsUsedWhenNonProxyHostsNotMatchingInProperties() throws SQLException { - setCustomTrustStorePropertyPath(); - Properties props = getProperties(); - addProxyProperties(props); - props.put("nonProxyHosts", "notMatchingHost"); - proxyAll(props); - - connectAndVerifySimpleQuery(props); - verifyProxyWasUsed(); - } - - @Test - public void testProxyNotUsedWhenNonProxyHostsMatchingInJVMParams() throws SQLException { - Properties props = getProperties(); - setJvmProperties(props); - System.setProperty("http.nonProxyHosts", "*"); - proxyAll(props); - - connectAndVerifySimpleQuery(props); - verifyProxyNotUsed(); - } - - @Test - public void testProxyUsedWhenNonProxyHostsNotMatchingInJVMParams() throws SQLException { - setCustomTrustStorePropertyPath(); - Properties props = getProperties(); - setJvmProperties(props); - System.setProperty("http.nonProxyHosts", "notMatchingHost"); - proxyAll(props); - - connectAndVerifySimpleQuery(props); - verifyProxyWasUsed(); - } - - private static void startWiremockStandAlone() { - // retrying in case of fail in port bindings - await() - .alias("wait for wiremock responding") - .atMost(Duration.ofSeconds(10)) - .until( - () -> { - try { - httpProxyPort = findFreePort(); - httpsProxyPort = findFreePort(); - wiremockStandalone = - new ProcessBuilder( - "java", - "-jar", - getWiremockStandAlonePath(), - "--root-dir", - System.getProperty("user.dir") - + File.separator - + WIREMOCK_HOME_DIR - + File.separator, - "--enable-browser-proxying", // work as forward proxy - "--proxy-pass-through", - "false", // pass through only matched requests - "--port", - String.valueOf(httpProxyPort), - "--https-port", - String.valueOf(httpsProxyPort), - "--https-keystore", - getResourceURL("wiremock" + File.separator + "ca-cert.jks"), - "--ca-keystore", - getResourceURL("wiremock" + File.separator + "ca-cert.jks")) - .inheritIO() - .start(); - waitForWiremock(); - return true; - } catch (Exception e) { - logger.warn("Failed to start wiremock, retrying: ", e); - return false; - } - }); } - private void resetWiremock() { - HttpPost postRequest; - postRequest = new HttpPost("http://" + WIREMOCK_HOST + ":" + getAdminPort() + "/__admin/reset"); - try (CloseableHttpClient client = HttpClients.createDefault(); - CloseableHttpResponse response = client.execute(postRequest)) { - assertEquals(200, response.getStatusLine().getStatusCode()); - } catch (IOException e) { - throw new RuntimeException(e); - } - } - - private static String getWiremockStandAlonePath() { - return System.getProperty("user.home") + WIREMOCK_M2_PATH; - } - - private static void waitForWiremock() { - await() - .pollDelay(Duration.ofSeconds(1)) - .atMost(Duration.ofSeconds(3)) - .until(ProxyLatestIT::isWiremockResponding); - } - - private static boolean isWiremockResponding() { - try (CloseableHttpClient httpClient = HttpClients.createDefault()) { - HttpGet request = - new HttpGet(String.format("http://%s:%d/__admin/mappings", WIREMOCK_HOST, httpProxyPort)); - CloseableHttpResponse response = httpClient.execute(request); - return response.getStatusLine().getStatusCode() == 200; - } catch (Exception e) { - logger.warn("Waiting for wiremock to respond: ", e); - } - return false; - } - - private static void stopWiremockStandAlone() { - if (wiremockStandalone != null) { - wiremockStandalone.destroyForcibly(); - await() - .alias("stop wiremock") - .atMost(Duration.ofSeconds(10)) - .until(() -> !wiremockStandalone.isAlive()); - } + private String getProxyProtocol(Properties props) { + return props.get("ssl").toString().equals("on") ? "https" : "http"; } - private static int findFreePort() { - try { - ServerSocket socket = new ServerSocket(0); - int port = socket.getLocalPort(); - socket.close(); - return port; - } catch (Exception e) { - throw new RuntimeException(e); + private int getProxyPort(String proxyProtocol) { + if (Objects.equals(proxyProtocol, "http")) { + return httpProxyPort; + } else { + return httpsProxyPort; } } - private Properties getProperties() { - Map params = getConnectionParameters(); - Properties props = new Properties(); - props.put("host", params.get("host")); - props.put("port", params.get("port")); - props.put("account", params.get("account")); - props.put("user", params.get("user")); - props.put("role", params.get("role")); - props.put("password", params.get("password")); - props.put("warehouse", params.get("warehouse")); - props.put("db", params.get("database")); - props.put("ssl", params.get("ssl")); - props.put("insecureMode", true); // OCSP disabled for wiremock proxy tests - return props; - } - private void addProxyProperties(Properties props) { String proxyProtocol = getProxyProtocol(props); props.put("useProxy", "true"); @@ -294,16 +74,22 @@ private void unsetJvmProperties() { System.clearProperty("https.proxyPort"); } - private String getProxyProtocol(Properties props) { - return props.get("ssl").toString().equals("on") ? "https" : "http"; + private String getSnowflakeUrl(Properties props) { + String protocol = getProxyProtocol(props); + return String.format("%s://%s:%s", protocol, props.get("host"), props.get("port")); } - private int getProxyPort(String proxyProtocol) { - if (Objects.equals(proxyProtocol, "http")) { - return httpProxyPort; - } else { - return httpsProxyPort; - } + private void proxyAll(Properties props) { + String template = + "{\n" + + " \"request\": {\n" + + " \"method\": \"ANY\",\n" + + " \"urlPattern\": \".*\"\n" + + " }\n," + + " \"response\": { \"proxyBaseUrl\": \"%s\" }\n" + + "}"; + String body = String.format(template, getSnowflakeUrl(props)); + addMapping(body); } private void verifyProxyWasUsed() { @@ -315,20 +101,6 @@ private void verifyProxyNotUsed() { verifyRequestToProxy(".*", 0); } - private void proxyAll(Properties props) { - String body = - String.format( - "{\"request\": {\"method\": \"ANY\", \"urlPattern\": \".*\"},\"response\": {\"proxyBaseUrl\": \"%s\"}}", - getSnowflakeUrl(props)); - HttpPost postRequest = createWiremockPostRequest(body, "/__admin/mappings"); - try (CloseableHttpClient client = HttpClients.createDefault(); - CloseableHttpResponse response = client.execute(postRequest)) { - assertEquals(201, response.getStatusLine().getStatusCode()); - } catch (IOException e) { - throw new RuntimeException(e); - } - } - private void connectAndVerifySimpleQuery(Properties props) throws SQLException { try (Connection con = DriverManager.getConnection( @@ -358,45 +130,71 @@ private void verifyRequestToProxy(String pathPattern, int expectedCount) { } } - private HttpPost createWiremockPostRequest(String body, String path) { - HttpPost postRequest = new HttpPost("http://" + WIREMOCK_HOST + ":" + getAdminPort() + path); - final StringEntity entity; - try { - entity = new StringEntity(body); - } catch (UnsupportedEncodingException e) { - throw new RuntimeException(e); - } - postRequest.setEntity(entity); - postRequest.setHeader("Accept", "application/json"); - postRequest.setHeader("Content-type", "application/json"); - return postRequest; + @Test + public void testProxyIsUsedWhenSetInProperties() throws SQLException { + setCustomTrustStorePropertyPath(); + Properties props = getProperties(); + addProxyProperties(props); + proxyAll(props); + + connectAndVerifySimpleQuery(props); + verifyProxyWasUsed(); } - private static void restoreTrustStorePathProperty() { - if (originalTrustStorePath != null) { - System.setProperty(TRUST_STORE_PROPERTY, originalTrustStorePath); - } else { - System.clearProperty(TRUST_STORE_PROPERTY); - } + @Test + public void testProxyIsUsedWhenSetInJVMParams() throws SQLException { + setCustomTrustStorePropertyPath(); + Properties props = getProperties(); + setJvmProperties(props); + proxyAll(props); + + connectAndVerifySimpleQuery(props); + verifyProxyWasUsed(); } - private int getAdminPort() { - return httpProxyPort; + @Test + public void testProxyNotUsedWhenNonProxyHostsMatchingInProperties() throws SQLException { + Properties props = getProperties(); + addProxyProperties(props); + props.put("nonProxyHosts", "*"); + proxyAll(props); + + connectAndVerifySimpleQuery(props); + verifyProxyNotUsed(); } - private String getSnowflakeUrl(Properties props) { - String protocol = getProxyProtocol(props); - return String.format("%s://%s:%s", protocol, props.get("host"), props.get("port")); + @Test + public void testProxyIsUsedWhenNonProxyHostsNotMatchingInProperties() throws SQLException { + setCustomTrustStorePropertyPath(); + Properties props = getProperties(); + addProxyProperties(props); + props.put("nonProxyHosts", "notMatchingHost"); + proxyAll(props); + + connectAndVerifySimpleQuery(props); + verifyProxyWasUsed(); } - private static String getResourceURL(String relativePath) { - return Paths.get(systemGetProperty("user.dir"), "src", "test", "resources", relativePath) - .toAbsolutePath() - .toString(); + @Test + public void testProxyNotUsedWhenNonProxyHostsMatchingInJVMParams() throws SQLException { + Properties props = getProperties(); + setJvmProperties(props); + System.setProperty("http.nonProxyHosts", "*"); + proxyAll(props); + + connectAndVerifySimpleQuery(props); + verifyProxyNotUsed(); } - private void setCustomTrustStorePropertyPath() { - System.setProperty( - TRUST_STORE_PROPERTY, getResourceURL("wiremock" + File.separator + "ca-cert.jks")); + @Test + public void testProxyUsedWhenNonProxyHostsNotMatchingInJVMParams() throws SQLException { + setCustomTrustStorePropertyPath(); + Properties props = getProperties(); + setJvmProperties(props); + System.setProperty("http.nonProxyHosts", "notMatchingHost"); + proxyAll(props); + + connectAndVerifySimpleQuery(props); + verifyProxyWasUsed(); } } diff --git a/src/test/java/net/snowflake/client/jdbc/RestRequestTest.java b/src/test/java/net/snowflake/client/jdbc/RestRequestTest.java index ae56a49f7..6e1a26428 100644 --- a/src/test/java/net/snowflake/client/jdbc/RestRequestTest.java +++ b/src/test/java/net/snowflake/client/jdbc/RestRequestTest.java @@ -16,6 +16,7 @@ import static org.mockito.Mockito.when; import java.io.IOException; +import java.net.SocketException; import java.net.SocketTimeoutException; import java.util.ArrayList; import java.util.List; @@ -492,6 +493,29 @@ public CloseableHttpResponse answer(InvocationOnMock invocation) throws Throwabl } } + @Test + public void testConnectionClosedRetriesSuccessful() throws IOException, SnowflakeSQLException { + CloseableHttpClient client = mock(CloseableHttpClient.class); + when(client.execute(any(HttpUriRequest.class))) + .then( + new Answer() { + int callCount = 0; + + @Override + public CloseableHttpResponse answer(InvocationOnMock invocationOnMock) + throws Throwable { + callCount += 1; + if (callCount >= 1) { + return successResponse(); + } else { + throw new SocketException("Connection reset"); + } + } + }); + + execute(client, "fakeurl.com/?requestId=abcd-1234", 0, 0, 0, true, false, 1); + } + @Test(expected = SnowflakeSQLException.class) public void testLoginMaxRetries() throws IOException, SnowflakeSQLException { boolean telemetryEnabled = TelemetryService.getInstance().isEnabled(); diff --git a/src/test/java/net/snowflake/client/jdbc/RestRequestWiremockTest.java b/src/test/java/net/snowflake/client/jdbc/RestRequestWiremockTest.java new file mode 100644 index 000000000..ef32c5f9d --- /dev/null +++ b/src/test/java/net/snowflake/client/jdbc/RestRequestWiremockTest.java @@ -0,0 +1,74 @@ +package net.snowflake.client.jdbc; + +import static org.junit.Assume.*; + +import java.util.concurrent.atomic.AtomicBoolean; +import net.snowflake.client.core.ExecTimeTelemetryData; +import org.apache.http.client.methods.CloseableHttpResponse; +import org.apache.http.client.methods.HttpGet; +import org.apache.http.impl.client.CloseableHttpClient; +import org.apache.http.impl.client.HttpClientBuilder; +import org.junit.Test; + +public class RestRequestWiremockTest extends BaseWiremockTest { + + String connectionResetByPeerScenario = + "{\n" + + " \"mappings\": [\n" + + " {\n" + + " \"scenarioName\": \"Connection reset by peer\",\n" + + " \"requiredScenarioState\": \"Started\",\n" + + " \"newScenarioState\": \"Connection is stable\",\n" + + " \"request\": {\n" + + " \"method\": \"GET\",\n" + + " \"url\": \"/endpoint\"\n" + + " },\n" + + " \"response\": {\n" + + " \"fault\": \"CONNECTION_RESET_BY_PEER\"\n" + + " }\n" + + " },\n" + + " {\n" + + " \"scenarioName\": \"Connection reset by peer\",\n" + + " \"requiredScenarioState\": \"Connection is stable\",\n" + + " \"request\": {\n" + + " \"method\": \"GET\",\n" + + " \"url\": \"/endpoint\"\n" + + " },\n" + + " \"response\": {\n" + + " \"status\": 200\n" + + " }\n" + + " }\n" + + " ],\n" + + " \"importOptions\": {\n" + + " \"duplicatePolicy\": \"IGNORE\",\n" + + " \"deleteAllNotInImport\": true\n" + + " }" + + "}"; + + @Test + public void testProxyIsUsedWhenSetInProperties() throws Exception { + importMapping(connectionResetByPeerScenario); + HttpClientBuilder httpClientBuilder = HttpClientBuilder.create().disableAutomaticRetries(); + try (CloseableHttpClient httpClient = httpClientBuilder.build()) { + HttpGet request = + new HttpGet(String.format("http://%s:%d/endpoint", WIREMOCK_HOST, httpProxyPort)); + RestRequest.execute( + httpClient, + request, + 0, + 0, + 0, + 0, + 0, + new AtomicBoolean(false), + false, + false, + false, + false, + new ExecTimeTelemetryData()); + + CloseableHttpResponse response = httpClient.execute(request); + assert (response.getStatusLine().getStatusCode() == 200); + } + } +} From 5c16b1d54e6751d000017063436e9d5fd6efb2b0 Mon Sep 17 00:00:00 2001 From: Jakub Szczerbinski Date: Fri, 23 Aug 2024 12:47:00 +0200 Subject: [PATCH 2/7] Fix imports --- .../java/net/snowflake/client/jdbc/BaseWiremockTest.java | 5 +++-- .../net/snowflake/client/jdbc/RestRequestWiremockTest.java | 2 -- 2 files changed, 3 insertions(+), 4 deletions(-) diff --git a/src/test/java/net/snowflake/client/jdbc/BaseWiremockTest.java b/src/test/java/net/snowflake/client/jdbc/BaseWiremockTest.java index e0376ec8d..3624d8a4e 100644 --- a/src/test/java/net/snowflake/client/jdbc/BaseWiremockTest.java +++ b/src/test/java/net/snowflake/client/jdbc/BaseWiremockTest.java @@ -4,14 +4,15 @@ import static net.snowflake.client.AbstractDriverIT.getConnectionParameters; import static net.snowflake.client.jdbc.SnowflakeUtil.systemGetProperty; import static org.awaitility.Awaitility.await; -import static org.junit.Assume.*; +import static org.junit.Assume.assumeFalse; +import static org.junit.Assume.assumeNoException; +import static org.junit.Assume.assumeTrue; import java.io.File; import java.io.IOException; import java.io.UnsupportedEncodingException; import java.net.ServerSocket; import java.nio.file.Paths; -import java.sql.*; import java.time.Duration; import java.util.Map; import java.util.Properties; diff --git a/src/test/java/net/snowflake/client/jdbc/RestRequestWiremockTest.java b/src/test/java/net/snowflake/client/jdbc/RestRequestWiremockTest.java index ef32c5f9d..2054343a5 100644 --- a/src/test/java/net/snowflake/client/jdbc/RestRequestWiremockTest.java +++ b/src/test/java/net/snowflake/client/jdbc/RestRequestWiremockTest.java @@ -1,7 +1,5 @@ package net.snowflake.client.jdbc; -import static org.junit.Assume.*; - import java.util.concurrent.atomic.AtomicBoolean; import net.snowflake.client.core.ExecTimeTelemetryData; import org.apache.http.client.methods.CloseableHttpResponse; From d691fdbe9e56ee70b5fb7a1f10beac624a224e12 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jakub=20Szczerbi=C5=84ski?= Date: Fri, 23 Aug 2024 14:06:09 +0200 Subject: [PATCH 3/7] Add test category --- .../net/snowflake/client/jdbc/RestRequestWiremockTest.java | 3 +++ 1 file changed, 3 insertions(+) diff --git a/src/test/java/net/snowflake/client/jdbc/RestRequestWiremockTest.java b/src/test/java/net/snowflake/client/jdbc/RestRequestWiremockTest.java index 2054343a5..76a484a78 100644 --- a/src/test/java/net/snowflake/client/jdbc/RestRequestWiremockTest.java +++ b/src/test/java/net/snowflake/client/jdbc/RestRequestWiremockTest.java @@ -1,13 +1,16 @@ package net.snowflake.client.jdbc; import java.util.concurrent.atomic.AtomicBoolean; +import net.snowflake.client.category.TestCategoryOthers; import net.snowflake.client.core.ExecTimeTelemetryData; import org.apache.http.client.methods.CloseableHttpResponse; import org.apache.http.client.methods.HttpGet; import org.apache.http.impl.client.CloseableHttpClient; import org.apache.http.impl.client.HttpClientBuilder; import org.junit.Test; +import org.junit.experimental.categories.Category; +@Category(TestCategoryOthers.class) public class RestRequestWiremockTest extends BaseWiremockTest { String connectionResetByPeerScenario = From edafaab7466e3a0df003af92ade0fd991d41006b Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jakub=20Szczerbi=C5=84ski?= Date: Fri, 23 Aug 2024 14:34:34 +0200 Subject: [PATCH 4/7] Rename port variables in base wiremock test --- .../snowflake/client/jdbc/BaseWiremockTest.java | 16 ++++++++-------- .../net/snowflake/client/jdbc/ProxyLatestIT.java | 4 ++-- .../client/jdbc/RestRequestWiremockTest.java | 2 +- 3 files changed, 11 insertions(+), 11 deletions(-) diff --git a/src/test/java/net/snowflake/client/jdbc/BaseWiremockTest.java b/src/test/java/net/snowflake/client/jdbc/BaseWiremockTest.java index 3624d8a4e..c0390910b 100644 --- a/src/test/java/net/snowflake/client/jdbc/BaseWiremockTest.java +++ b/src/test/java/net/snowflake/client/jdbc/BaseWiremockTest.java @@ -40,8 +40,8 @@ public abstract class BaseWiremockTest { "/.m2/repository/org/wiremock/wiremock-standalone/3.8.0/wiremock-standalone-3.8.0.jar"; protected static final String WIREMOCK_HOST = "localhost"; protected static final String TRUST_STORE_PROPERTY = "javax.net.ssl.trustStore"; - protected static int httpProxyPort; - protected static int httpsProxyPort; + protected static int wiremockHttpPort; + protected static int wiremockHttpsPort; private static String originalTrustStorePath; protected static Process wiremockStandalone; @@ -78,8 +78,8 @@ protected static void startWiremockStandAlone() { .until( () -> { try { - httpProxyPort = findFreePort(); - httpsProxyPort = findFreePort(); + wiremockHttpPort = findFreePort(); + wiremockHttpsPort = findFreePort(); wiremockStandalone = new ProcessBuilder( "java", @@ -94,9 +94,9 @@ protected static void startWiremockStandAlone() { "--proxy-pass-through", "false", // pass through only matched requests "--port", - String.valueOf(httpProxyPort), + String.valueOf(wiremockHttpPort), "--https-port", - String.valueOf(httpsProxyPort), + String.valueOf(wiremockHttpsPort), "--https-keystore", getResourceURL("wiremock" + File.separator + "ca-cert.jks"), "--ca-keystore", @@ -137,7 +137,7 @@ private static void waitForWiremock() { private static boolean isWiremockResponding() { try (CloseableHttpClient httpClient = HttpClients.createDefault()) { HttpGet request = - new HttpGet(String.format("http://%s:%d/__admin/mappings", WIREMOCK_HOST, httpProxyPort)); + new HttpGet(String.format("http://%s:%d/__admin/mappings", WIREMOCK_HOST, wiremockHttpPort)); CloseableHttpResponse response = httpClient.execute(request); return response.getStatusLine().getStatusCode() == 200; } catch (Exception e) { @@ -206,7 +206,7 @@ protected static void restoreTrustStorePathProperty() { } private int getAdminPort() { - return httpProxyPort; + return wiremockHttpPort; } private static String getResourceURL(String relativePath) { diff --git a/src/test/java/net/snowflake/client/jdbc/ProxyLatestIT.java b/src/test/java/net/snowflake/client/jdbc/ProxyLatestIT.java index 22781d3d9..4fca52d1c 100644 --- a/src/test/java/net/snowflake/client/jdbc/ProxyLatestIT.java +++ b/src/test/java/net/snowflake/client/jdbc/ProxyLatestIT.java @@ -38,9 +38,9 @@ private String getProxyProtocol(Properties props) { private int getProxyPort(String proxyProtocol) { if (Objects.equals(proxyProtocol, "http")) { - return httpProxyPort; + return wiremockHttpPort; } else { - return httpsProxyPort; + return wiremockHttpsPort; } } diff --git a/src/test/java/net/snowflake/client/jdbc/RestRequestWiremockTest.java b/src/test/java/net/snowflake/client/jdbc/RestRequestWiremockTest.java index 76a484a78..cfdcd788f 100644 --- a/src/test/java/net/snowflake/client/jdbc/RestRequestWiremockTest.java +++ b/src/test/java/net/snowflake/client/jdbc/RestRequestWiremockTest.java @@ -52,7 +52,7 @@ public void testProxyIsUsedWhenSetInProperties() throws Exception { HttpClientBuilder httpClientBuilder = HttpClientBuilder.create().disableAutomaticRetries(); try (CloseableHttpClient httpClient = httpClientBuilder.build()) { HttpGet request = - new HttpGet(String.format("http://%s:%d/endpoint", WIREMOCK_HOST, httpProxyPort)); + new HttpGet(String.format("http://%s:%d/endpoint", WIREMOCK_HOST, wiremockHttpPort)); RestRequest.execute( httpClient, request, From 93bdbeeeaedf7bbea750d45e85401a9ddc49311f Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jakub=20Szczerbi=C5=84ski?= Date: Fri, 23 Aug 2024 17:11:26 +0200 Subject: [PATCH 5/7] Rename test --- ...{RestRequestWiremockTest.java => RestRequestWiremockIT.java} | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) rename src/test/java/net/snowflake/client/jdbc/{RestRequestWiremockTest.java => RestRequestWiremockIT.java} (97%) diff --git a/src/test/java/net/snowflake/client/jdbc/RestRequestWiremockTest.java b/src/test/java/net/snowflake/client/jdbc/RestRequestWiremockIT.java similarity index 97% rename from src/test/java/net/snowflake/client/jdbc/RestRequestWiremockTest.java rename to src/test/java/net/snowflake/client/jdbc/RestRequestWiremockIT.java index cfdcd788f..eeb7b91d8 100644 --- a/src/test/java/net/snowflake/client/jdbc/RestRequestWiremockTest.java +++ b/src/test/java/net/snowflake/client/jdbc/RestRequestWiremockIT.java @@ -11,7 +11,7 @@ import org.junit.experimental.categories.Category; @Category(TestCategoryOthers.class) -public class RestRequestWiremockTest extends BaseWiremockTest { +public class RestRequestWiremockIT extends BaseWiremockTest { String connectionResetByPeerScenario = "{\n" From 2054afa42b9321cfdcd6692b25372e15b5fcaea2 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jakub=20Szczerbi=C5=84ski?= Date: Fri, 23 Aug 2024 17:54:29 +0200 Subject: [PATCH 6/7] Fix formatting --- src/test/java/net/snowflake/client/jdbc/BaseWiremockTest.java | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/src/test/java/net/snowflake/client/jdbc/BaseWiremockTest.java b/src/test/java/net/snowflake/client/jdbc/BaseWiremockTest.java index c0390910b..5a2fe8e96 100644 --- a/src/test/java/net/snowflake/client/jdbc/BaseWiremockTest.java +++ b/src/test/java/net/snowflake/client/jdbc/BaseWiremockTest.java @@ -137,7 +137,8 @@ private static void waitForWiremock() { private static boolean isWiremockResponding() { try (CloseableHttpClient httpClient = HttpClients.createDefault()) { HttpGet request = - new HttpGet(String.format("http://%s:%d/__admin/mappings", WIREMOCK_HOST, wiremockHttpPort)); + new HttpGet( + String.format("http://%s:%d/__admin/mappings", WIREMOCK_HOST, wiremockHttpPort)); CloseableHttpResponse response = httpClient.execute(request); return response.getStatusLine().getStatusCode() == 200; } catch (Exception e) { From b1a3190f323364baf00ed9c0e207688a5ecfc921 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jakub=20Szczerbi=C5=84ski?= Date: Thu, 29 Aug 2024 12:27:31 +0200 Subject: [PATCH 7/7] Make RestRequestWiremockIT 'latest' --- ...tRequestWiremockIT.java => RestRequestWiremockLatestIT.java} | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) rename src/test/java/net/snowflake/client/jdbc/{RestRequestWiremockIT.java => RestRequestWiremockLatestIT.java} (97%) diff --git a/src/test/java/net/snowflake/client/jdbc/RestRequestWiremockIT.java b/src/test/java/net/snowflake/client/jdbc/RestRequestWiremockLatestIT.java similarity index 97% rename from src/test/java/net/snowflake/client/jdbc/RestRequestWiremockIT.java rename to src/test/java/net/snowflake/client/jdbc/RestRequestWiremockLatestIT.java index eeb7b91d8..76856b985 100644 --- a/src/test/java/net/snowflake/client/jdbc/RestRequestWiremockIT.java +++ b/src/test/java/net/snowflake/client/jdbc/RestRequestWiremockLatestIT.java @@ -11,7 +11,7 @@ import org.junit.experimental.categories.Category; @Category(TestCategoryOthers.class) -public class RestRequestWiremockIT extends BaseWiremockTest { +public class RestRequestWiremockLatestIT extends BaseWiremockTest { String connectionResetByPeerScenario = "{\n"