diff --git a/flint-core/src/main/java/org/opensearch/flint/core/logging/ExceptionMessages.java b/flint-core/src/main/java/org/opensearch/flint/core/logging/ExceptionMessages.java new file mode 100644 index 000000000..34f5dd257 --- /dev/null +++ b/flint-core/src/main/java/org/opensearch/flint/core/logging/ExceptionMessages.java @@ -0,0 +1,74 @@ +/* + * Copyright OpenSearch Contributors + * SPDX-License-Identifier: Apache-2.0 + */ + +package org.opensearch.flint.core.logging; + +import com.amazonaws.services.s3.model.AmazonS3Exception; + +/** + * Define constants for common error messages and utility methods to handle them. + */ +public class ExceptionMessages { + public static final String SyntaxErrorPrefix = "Syntax error"; + public static final String S3ErrorPrefix = "Fail to read data from S3. Cause"; + public static final String GlueErrorPrefix = "Fail to read data from Glue. Cause"; + public static final String QueryAnalysisErrorPrefix = "Fail to analyze query. Cause"; + public static final String SparkExceptionErrorPrefix = "Spark exception. Cause"; + public static final String QueryRunErrorPrefix = "Fail to run query. Cause"; + public static final String GlueAccessDeniedMessage = "Access denied in AWS Glue service. Please check permissions."; + + /** + * Extracts the root cause from a throwable and returns a sanitized error message. + * + * @param e The throwable from which to extract the root cause. + * @param maxLength The maximum length of the returned error message. Default is 1000 characters. + * @return A sanitized and possibly truncated error message string. + */ + public static String extractRootCause(Throwable e, int maxLength) { + return truncateMessage(redactMessage(rootCause(e)), maxLength); + } + + /** + * Overloads extractRootCause to provide a non-truncated error message. + * + * @param e The throwable from which to extract the root cause. + * @return A sanitized error message string. + */ + public static String extractRootCause(Throwable e) { + return redactMessage(rootCause(e)); + } + + private static Throwable rootCause(Throwable e) { + Throwable cause = e; + while (cause.getCause() != null && cause.getCause() != cause) { + cause = cause.getCause(); + } + return cause; + } + + private static String redactMessage(Throwable cause) { + if (cause instanceof AmazonS3Exception) { + AmazonS3Exception e = (AmazonS3Exception) cause; + return String.format("%s: serviceName=[%s], statusCode=[%d]", + S3ErrorPrefix, e.getServiceName(), e.getStatusCode()); + } else { + if (cause.getLocalizedMessage() != null) { + return cause.getLocalizedMessage(); + } else if (cause.getMessage() != null) { + return cause.getMessage(); + } else { + return cause.toString(); + } + } + } + + private static String truncateMessage(String message, int maxLength) { + if (message.length() > maxLength) { + return message.substring(0, maxLength) + "..."; + } else { + return message; + } + } +} diff --git a/flint-core/src/main/java/org/opensearch/flint/core/logging/ExceptionMessages.scala b/flint-core/src/main/java/org/opensearch/flint/core/logging/ExceptionMessages.scala deleted file mode 100644 index 71bb729f5..000000000 --- a/flint-core/src/main/java/org/opensearch/flint/core/logging/ExceptionMessages.scala +++ /dev/null @@ -1,62 +0,0 @@ -/* - * Copyright OpenSearch Contributors - * SPDX-License-Identifier: Apache-2.0 - */ - -package org.opensearch.flint.core.logging - -import com.amazonaws.services.s3.model.AmazonS3Exception - -// Define constants for common error messages -object ExceptionMessages { - val SyntaxErrorPrefix = "Syntax error" - val S3ErrorPrefix = "Fail to read data from S3. Cause" - val GlueErrorPrefix = "Fail to read data from Glue. Cause" - val QueryAnalysisErrorPrefix = "Fail to analyze query. Cause" - val SparkExceptionErrorPrefix = "Spark exception. Cause" - val QueryRunErrorPrefix = "Fail to run query. Cause" - val GlueAccessDeniedMessage = "Access denied in AWS Glue service. Please check permissions." - - /** - * Extracts the root cause from a throwable and returns a sanitized error message. - * - * @param e - * The throwable from which to extract the root cause. - * @param maxLength - * The maximum length of the returned error message. Default is 100 characters. - * @return - * A sanitized and possibly truncated error message string. - */ - def extractRootCause(e: Throwable, maxLength: Int = 1000): String = { - truncateMessage(redactMessage(rootCause(e)), maxLength) - } - - private def rootCause(e: Throwable): Throwable = { - var cause = e - while (cause.getCause != null && cause.getCause != cause) { - cause = cause.getCause - } - cause - } - - private def redactMessage(cause: Throwable): String = cause match { - case e: AmazonS3Exception => - s"$S3ErrorPrefix: serviceName=[${e.getServiceName}], statusCode=[${e.getStatusCode}]" - case _ => - if (cause.getLocalizedMessage != null) { - cause.getLocalizedMessage - } else if (cause.getMessage != null) { - cause.getMessage - } else { - cause.toString - } - } - - private def truncateMessage(message: String, maxLength: Int): String = { - if (message.length > maxLength) { - message.take(maxLength) + "..." - } else { - message - } - } -} diff --git a/flint-core/src/test/java/org/opensearch/flint/core/logging/ExceptionMessagesTest.java b/flint-core/src/test/java/org/opensearch/flint/core/logging/ExceptionMessagesTest.java new file mode 100644 index 000000000..b074be066 --- /dev/null +++ b/flint-core/src/test/java/org/opensearch/flint/core/logging/ExceptionMessagesTest.java @@ -0,0 +1,49 @@ +/* + * Copyright OpenSearch Contributors + * SPDX-License-Identifier: Apache-2.0 + */ + +package org.opensearch.flint.core.logging; + +import static org.junit.Assert.assertEquals; +import static org.opensearch.flint.core.logging.ExceptionMessages.extractRootCause; + +import com.amazonaws.services.s3.model.AmazonS3Exception; +import org.apache.spark.SparkException; +import org.junit.Test; +import org.opensearch.OpenSearchException; + +public class ExceptionMessagesTest { + + @Test + public void testExtractRootCauseFromS3Exception() { + AmazonS3Exception exception = new AmazonS3Exception("test"); + exception.setServiceName("S3"); + exception.setStatusCode(400); + + assertEquals( + "Fail to read data from S3. Cause: serviceName=[S3], statusCode=[400]", + extractRootCause( + new SparkException("test", exception))); + } + + @Test + public void testExtractRootCauseFromOtherException() { + OpenSearchException exception = new OpenSearchException("Write is blocked"); + + assertEquals( + "Write is blocked", + extractRootCause( + new SparkException("test", exception))); + } + + @Test + public void testExtractRootCauseFromOtherExceptionWithLongMessage() { + OpenSearchException exception = new OpenSearchException("Write is blocked due to cluster is readonly"); + + assertEquals( + "Write is blocked due...", + extractRootCause( + new SparkException("test", exception), 20)); + } +} \ No newline at end of file diff --git a/flint-spark-integration/src/main/scala/org/opensearch/flint/spark/FlintSparkIndexMonitor.scala b/flint-spark-integration/src/main/scala/org/opensearch/flint/spark/FlintSparkIndexMonitor.scala index 99997d931..29acaea6b 100644 --- a/flint-spark-integration/src/main/scala/org/opensearch/flint/spark/FlintSparkIndexMonitor.scala +++ b/flint-spark-integration/src/main/scala/org/opensearch/flint/spark/FlintSparkIndexMonitor.scala @@ -210,7 +210,7 @@ class FlintSparkIndexMonitor( .finalLog(latest => exception match { case Some(ex) => - latest.copy(state = FAILED, error = extractRootCause(ex)) + latest.copy(state = FAILED, error = extractRootCause(ex, 1000)) case None => latest.copy(state = FAILED) })