From fe0ed36f6c6170c82da614bf943a09fc86c1b0aa Mon Sep 17 00:00:00 2001 From: Piotr Fus Date: Wed, 29 Nov 2023 09:18:43 +0100 Subject: [PATCH] SNOW-980046 Make QueryStatus thread safe --- .../snowflake/client/core/QueryStatus.java | 77 ++++++++++++++----- .../client/core/QueryStatusTest.java | 38 +++++++++ 2 files changed, 95 insertions(+), 20 deletions(-) create mode 100644 src/test/java/net/snowflake/client/core/QueryStatusTest.java diff --git a/src/main/java/net/snowflake/client/core/QueryStatus.java b/src/main/java/net/snowflake/client/core/QueryStatus.java index 73268cf43..265f32c93 100644 --- a/src/main/java/net/snowflake/client/core/QueryStatus.java +++ b/src/main/java/net/snowflake/client/core/QueryStatus.java @@ -1,23 +1,44 @@ package net.snowflake.client.core; +import java.util.Arrays; +import java.util.List; + /** Copied from QueryDTO.java in Global Services. */ -public enum QueryStatus { - RUNNING(0, "RUNNING"), - ABORTING(1, "ABORTING"), - SUCCESS(2, "SUCCESS"), - FAILED_WITH_ERROR(3, "FAILED_WITH_ERROR"), - ABORTED(4, "ABORTED"), - QUEUED(5, "QUEUED"), - FAILED_WITH_INCIDENT(6, "FAILED_WITH_INCIDENT"), - DISCONNECTED(7, "DISCONNECTED"), - RESUMING_WAREHOUSE(8, "RESUMING_WAREHOUSE"), +public class QueryStatus { + public static final QueryStatus RUNNING = new QueryStatus(0, "RUNNING"); + public static final QueryStatus ABORTING = new QueryStatus(1, "ABORTING"); + public static final QueryStatus SUCCESS = new QueryStatus(2, "SUCCESS"); + public static final QueryStatus FAILED_WITH_ERROR = new QueryStatus(3, "FAILED_WITH_ERROR"); + public static final QueryStatus ABORTED = new QueryStatus(4, "ABORTED"); + public static final QueryStatus QUEUED = new QueryStatus(5, "QUEUED"); + public static final QueryStatus FAILED_WITH_INCIDENT = new QueryStatus(6, "FAILED_WITH_INCIDENT"); + public static final QueryStatus DISCONNECTED = new QueryStatus(7, "DISCONNECTED"); + public static final QueryStatus RESUMING_WAREHOUSE = new QueryStatus(8, "RESUMING_WAREHOUSE"); // purposeful typo. Is present in QueryDTO.java. - QUEUED_REPAIRING_WAREHOUSE(9, "QUEUED_REPARING_WAREHOUSE"), - RESTARTED(10, "RESTARTED"), + public static final QueryStatus QUEUED_REPAIRING_WAREHOUSE = + new QueryStatus(9, "QUEUED_REPARING_WAREHOUSE"); + public static final QueryStatus RESTARTED = new QueryStatus(10, "RESTARTED"); /** The state when a statement is waiting on a lock on resource held by another statement. */ - BLOCKED(11, "BLOCKED"), - NO_DATA(12, "NO_DATA"); + public static final QueryStatus BLOCKED = new QueryStatus(11, "BLOCKED"); + + public static final QueryStatus NO_DATA = new QueryStatus(12, "NO_DATA"); + + private static final List allStatuses = + Arrays.asList( + RUNNING, + ABORTING, + SUCCESS, + FAILED_WITH_ERROR, + ABORTED, + QUEUED, + FAILED_WITH_INCIDENT, + DISCONNECTED, + RESUMING_WAREHOUSE, + QUEUED_REPAIRING_WAREHOUSE, + RESTARTED, + BLOCKED, + NO_DATA); private final int value; private final String description; @@ -83,13 +104,29 @@ public static boolean isAnError(QueryStatus status) { public static QueryStatus getStatusFromString(String description) { if (description != null) { - for (QueryStatus st : QueryStatus.values()) { - if (description.equalsIgnoreCase(st.getDescription())) { - return st; - } - } - return QueryStatus.NO_DATA; + return allStatuses.stream() + .filter(queryStatus -> queryStatus.description.equals(description)) + .findFirst() + .map(queryStatus -> new QueryStatus(queryStatus.value, queryStatus.description)) + .orElse(new QueryStatus(NO_DATA.value, NO_DATA.description)); } return null; } + + @Override + public boolean equals(Object obj) { + if (obj == null) { + return false; + } + if (!(obj instanceof QueryStatus)) { + return false; + } + QueryStatus another = (QueryStatus) obj; + return value == another.value && description.equals(another.description); + } + + @Override + public int hashCode() { + return value; + } } diff --git a/src/test/java/net/snowflake/client/core/QueryStatusTest.java b/src/test/java/net/snowflake/client/core/QueryStatusTest.java new file mode 100644 index 000000000..685a4264a --- /dev/null +++ b/src/test/java/net/snowflake/client/core/QueryStatusTest.java @@ -0,0 +1,38 @@ +package net.snowflake.client.core; + +import static org.junit.Assert.assertEquals; + +import org.junit.Test; + +public class QueryStatusTest { + @Test + public void testThreadSafetyByConstructor() { + QueryStatus status1 = + new QueryStatus(QueryStatus.RUNNING.getValue(), QueryStatus.RUNNING.getDescription()); + QueryStatus status2 = + new QueryStatus(QueryStatus.RUNNING.getValue(), QueryStatus.RUNNING.getDescription()); + status1.setErrorCode(1); + status2.setErrorCode(2); + + assertEquals(status1.getErrorCode(), 1); + assertEquals(status2.getErrorCode(), 2); + } + + @Test + public void testThreadSafetyByDescription() { + QueryStatus status1 = QueryStatus.getStatusFromString(QueryStatus.RUNNING.getDescription()); + QueryStatus status2 = QueryStatus.getStatusFromString(QueryStatus.RUNNING.getDescription()); + status1.setErrorCode(1); + status2.setErrorCode(2); + + assertEquals(status1.getErrorCode(), 1); + assertEquals(status2.getErrorCode(), 2); + } + + @Test + public void testEquality() { + assertEquals( + QueryStatus.RUNNING, + new QueryStatus(QueryStatus.RUNNING.getValue(), QueryStatus.RUNNING.getDescription())); + } +}