net.snowflake.client.jdbc.SnowflakeSQLLoggedException Maven / Gradle / Ivy
/*
* Copyright (c) 2012-2020 Snowflake Computing Inc. All rights reserved.
*/
package net.snowflake.client.jdbc;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.node.ObjectNode;
import com.google.api.client.util.Strings;
import java.io.PrintWriter;
import java.io.StringWriter;
import java.sql.SQLException;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Future;
import java.util.concurrent.TimeUnit;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import net.minidev.json.JSONObject;
import net.snowflake.client.core.ObjectMapperFactory;
import net.snowflake.client.core.SFBaseSession;
import net.snowflake.client.core.SFException;
import net.snowflake.client.jdbc.telemetry.Telemetry;
import net.snowflake.client.jdbc.telemetry.TelemetryField;
import net.snowflake.client.jdbc.telemetry.TelemetryUtil;
import net.snowflake.client.jdbc.telemetryOOB.TelemetryEvent;
import net.snowflake.client.jdbc.telemetryOOB.TelemetryService;
import net.snowflake.client.log.SFLogger;
import net.snowflake.client.log.SFLoggerFactory;
import net.snowflake.common.core.LoginInfoDTO;
import net.snowflake.common.core.SqlState;
/**
* @author mknister
* This SnowflakeSQLLoggedException class extends the SnowflakeSQLException class to add OOB
* telemetry data for sql exceptions. Not all sql exceptions require OOB telemetry logging so
* the exceptions in this class should only be thrown if there is a need for logging the
* exception with OOB telemetry.
*/
public class SnowflakeSQLLoggedException extends SnowflakeSQLException {
private static final SFLogger logger =
SFLoggerFactory.getLogger(SnowflakeSQLLoggedException.class);
private static final ObjectMapper mapper = ObjectMapperFactory.getObjectMapper();
/**
* Function to create a TelemetryEvent log from the JSONObject and exception and send it via OOB
* telemetry
*
* @param value JSONnode containing relevant information specific to the exception constructor
* that should be included in the telemetry data, such as sqlState or vendorCode
* @param ex The exception being thrown
* @param oobInstance Out of band telemetry instance through which log will be passed
*/
private static void sendOutOfBandTelemetryMessage(
JSONObject value, SQLException ex, TelemetryService oobInstance) {
TelemetryEvent.LogBuilder logBuilder = new TelemetryEvent.LogBuilder();
StringWriter sw = new StringWriter();
PrintWriter pw = new PrintWriter(sw);
ex.printStackTrace(pw);
String stackTrace = maskStacktrace(sw.toString());
value.put("Stacktrace", stackTrace);
value.put("Exception", ex.getClass().getSimpleName());
TelemetryEvent log =
logBuilder.withName("Exception: " + ex.getClass().getSimpleName()).withValue(value).build();
oobInstance.report(log);
}
/**
* Function to create a TelemetryClient log and send it via in-band telemetry
*
* @param value ObjectNode containing information specific to the exception constructor that
* should be included in the telemetry log, such as SQLState or reason for the error
* @param ex The exception being thrown
* @param ibInstance Telemetry instance through which telemetry log will be sent
* @return true if in-band telemetry log sent successfully or false if it did not
*/
private static Future sendInBandTelemetryMessage(
ObjectNode value, SQLException ex, Telemetry ibInstance) {
StringWriter sw = new StringWriter();
PrintWriter pw = new PrintWriter(sw);
ex.printStackTrace(pw);
String stackTrace = maskStacktrace(sw.toString());
value.put("Stacktrace", stackTrace);
value.put("Exception", ex.getClass().getSimpleName());
// For SQLFeatureNotSupportedExceptions, add in reason for failure as " not
// supported"
if (value.get("SQLState").toString().contains(SqlState.FEATURE_NOT_SUPPORTED)) {
String reason = "";
StackTraceElement[] stackTraceArray = ex.getStackTrace();
if (stackTraceArray.length >= 1) {
reason = ex.getStackTrace()[0].getMethodName() + " not supported";
}
value.put("reason", reason);
}
ibInstance.addLogToBatch(TelemetryUtil.buildJobData(value));
return ibInstance.sendBatchAsync();
}
/**
* Helper function to remove sensitive data (error message, reason) from the stacktrace.
*
* @param stackTrace original stacktrace
* @return stack trace with sensitive data removed
*/
static String maskStacktrace(String stackTrace) {
Pattern STACKTRACE_BEGINNING =
Pattern.compile(
"(com|net)(\\.snowflake\\.client\\.jdbc\\.Snowflake)(SQLLogged|LoggedFeatureNotSupported|SQL)(Exception)([\\s\\S]*?)(\\n\\t?at\\snet|com\\.)",
Pattern.MULTILINE | Pattern.CASE_INSENSITIVE);
Matcher matcher = STACKTRACE_BEGINNING.matcher(stackTrace);
// Remove the reason from after the stack trace (in group #5 of regex pattern)
if (matcher.find()) {
return matcher.replaceAll("$1$2$3$4$6");
}
return stackTrace;
}
/**
* Helper function to create JSONObject node for OOB telemetry log
*
* @param queryId query ID
* @param SQLState the SQL state
* @param vendorCode the vendor code
* @return JSONObject with data about SQLException
*/
static JSONObject createOOBValue(String queryId, String SQLState, int vendorCode) {
JSONObject oobValue = new JSONObject();
oobValue.put("type", TelemetryField.SQL_EXCEPTION.toString());
oobValue.put("DriverType", LoginInfoDTO.SF_JDBC_APP_ID);
oobValue.put("DriverVersion", SnowflakeDriver.implementVersion);
if (!Strings.isNullOrEmpty(queryId)) {
oobValue.put("QueryID", queryId);
}
if (!Strings.isNullOrEmpty(SQLState)) {
oobValue.put("SQLState", SQLState);
}
if (vendorCode != -1) {
oobValue.put("ErrorNumber", vendorCode);
}
return oobValue;
}
/**
* Helper function to create ObjectNode for IB telemetry log
*
* @param queryId query ID
* @param SQLState the SQL state
* @param vendorCode the vendor code
* @return ObjectNode for IB telemetry log
*/
static ObjectNode createIBValue(String queryId, String SQLState, int vendorCode) {
ObjectNode ibValue = mapper.createObjectNode();
ibValue.put("type", TelemetryField.SQL_EXCEPTION.toString());
ibValue.put("DriverType", LoginInfoDTO.SF_JDBC_APP_ID);
ibValue.put("DriverVersion", SnowflakeDriver.implementVersion);
if (!Strings.isNullOrEmpty(queryId)) {
ibValue.put("QueryID", queryId);
}
if (!Strings.isNullOrEmpty(SQLState)) {
ibValue.put("SQLState", SQLState);
}
if (vendorCode != -1) {
ibValue.put("ErrorNumber", vendorCode);
}
return ibValue;
}
/**
* Function to construct log data based on possible exception inputs and send data through in-band
* telemetry, or oob if in-band does not work
*
* @param queryId query ID if exists
* @param SQLState SQLState
* @param vendorCode vendor code
* @param session session object, which is needed to send in-band telemetry but not oob. Might be
* null, in which case oob is used.
* @param ex Exception object
*/
public static void sendTelemetryData(
String queryId, String SQLState, int vendorCode, SFBaseSession session, SQLException ex) {
Telemetry ibInstance = null;
// if session is not null, try sending data using in-band telemetry
if (session != null) {
ibInstance = session.getTelemetryClient();
}
// if in-band instance is successfully created, compile sql exception data into an in-band
// telemetry log
if (ibInstance != null) {
ObjectNode ibValue = createIBValue(queryId, SQLState, vendorCode);
// try to send in-band data asynchronously
ExecutorService threadExecutor = Executors.newSingleThreadExecutor();
Telemetry finalIbInstance = ibInstance;
try {
threadExecutor.submit(
() -> {
boolean inBandSuccess;
Future sendInBand = sendInBandTelemetryMessage(ibValue, ex, finalIbInstance);
// record whether in band telemetry message sent with boolean value inBandSuccess
try {
inBandSuccess = sendInBand.get(10, TimeUnit.SECONDS);
} catch (Exception e) {
inBandSuccess = false;
}
// In-band failed so send OOB telemetry instead
if (!inBandSuccess) {
logger.debug(
"In-band telemetry message failed to send. Sending out-of-band message instead");
JSONObject oobValue = createOOBValue(queryId, SQLState, vendorCode);
sendOutOfBandTelemetryMessage(oobValue, ex, TelemetryService.getInstance());
}
});
} finally {
// Send the shutdown signal to the executor service
threadExecutor.shutdown();
// Add an extra hook in the telemetry client, if extra error handling is needed
ibInstance.postProcess(queryId, SQLState, vendorCode, ex);
}
}
// In-band is not possible so send OOB telemetry instead
else {
JSONObject oobValue = createOOBValue(queryId, SQLState, vendorCode);
sendOutOfBandTelemetryMessage(oobValue, ex, TelemetryService.getInstance());
}
}
/**
* @param session SFBaseSession
* @param reason exception reason
* @param SQLState the SQL state
* @param vendorCode the vendor code
* @param queryId the query ID
*/
public SnowflakeSQLLoggedException(
SFBaseSession session, String reason, String SQLState, int vendorCode, String queryId) {
super(queryId, reason, SQLState, vendorCode);
sendTelemetryData(queryId, SQLState, vendorCode, session, this);
}
/**
* @param session SFBaseSession
* @param vendorCode the vendor code
* @param SQLState the SQL state
*/
public SnowflakeSQLLoggedException(SFBaseSession session, int vendorCode, String SQLState) {
super(SQLState, vendorCode);
sendTelemetryData(null, SQLState, vendorCode, session, this);
}
/**
* @param queryId the query ID
* @param session SFBaseSession
* @param vendorCode the vendor code
* @param SQLState the SQL state
*/
public SnowflakeSQLLoggedException(
String queryId, SFBaseSession session, int vendorCode, String SQLState) {
super(queryId, SQLState, vendorCode);
sendTelemetryData(queryId, SQLState, vendorCode, session, this);
}
/**
* use {@link SnowflakeSQLLoggedException#SnowflakeSQLLoggedException(String, SFBaseSession,
* String, String)}
*
* @param session SFBaseSession
* @param SQLState the SQL state
* @param reason exception reason
*/
@Deprecated
public SnowflakeSQLLoggedException(SFBaseSession session, String SQLState, String reason) {
this(null, session, SQLState, reason);
}
/**
* @param queryId the query ID
* @param session SFBaseSession
* @param SQLState the SQL state
* @param reason the exception reason
*/
public SnowflakeSQLLoggedException(
String queryId, SFBaseSession session, String SQLState, String reason) {
super(reason, SQLState);
sendTelemetryData(queryId, SQLState, -1, session, this);
}
/**
* @param session SFBaseSession
* @param vendorCode the vendor code
* @param SQLState the SQL state
* @param params additional parameters
*/
public SnowflakeSQLLoggedException(
SFBaseSession session, int vendorCode, String SQLState, Object... params) {
this(null, session, vendorCode, SQLState, params);
}
/**
* @param queryId the query ID
* @param session SFBaseSession
* @param vendorCode the vendor code
* @param SQLState the SQL state
* @param params additional parameters
*/
public SnowflakeSQLLoggedException(
String queryId, SFBaseSession session, int vendorCode, String SQLState, Object... params) {
super(queryId, SQLState, vendorCode, params);
sendTelemetryData(queryId, SQLState, vendorCode, session, this);
}
/**
* @param session SFBaseSession
* @param errorCode the error code
* @param ex Throwable exception
* @param params additional parameters
*/
public SnowflakeSQLLoggedException(
SFBaseSession session, ErrorCode errorCode, Throwable ex, Object... params) {
super(ex, errorCode, params);
sendTelemetryData(null, errorCode.getSqlState(), errorCode.getMessageCode(), session, this);
}
/**
* @param session SFBaseSession
* @param SQLState the SQL state
* @param vendorCode the vendor code
* @param ex Throwable exception
* @param params additional parameters
*/
public SnowflakeSQLLoggedException(
SFBaseSession session, String SQLState, int vendorCode, Throwable ex, Object... params) {
super(ex, SQLState, vendorCode, params);
sendTelemetryData(null, SQLState, vendorCode, session, this);
}
/**
* @param queryId the query ID
* @param session SFBaseSession
* @param SQLState the SQL state
* @param vendorCode the vendor code
* @param ex Throwable exception
* @param params additional parameters
*/
public SnowflakeSQLLoggedException(
String queryId,
SFBaseSession session,
String SQLState,
int vendorCode,
Throwable ex,
Object... params) {
super(queryId, ex, SQLState, vendorCode, params);
sendTelemetryData(queryId, SQLState, vendorCode, session, this);
}
/**
* use {@link SnowflakeSQLLoggedException#SnowflakeSQLLoggedException(String, SFBaseSession,
* ErrorCode, Object...)}
*
* @param session SFBaseSession
* @param errorCode the error code
* @param params additional parameters
*/
@Deprecated
public SnowflakeSQLLoggedException(SFBaseSession session, ErrorCode errorCode, Object... params) {
this(null, session, errorCode, params);
}
/**
* @param queryId the query ID
* @param session SFBaseSession
* @param errorCode the error code
* @param params additional parameters
*/
public SnowflakeSQLLoggedException(
String queryId, SFBaseSession session, ErrorCode errorCode, Object... params) {
super(queryId, errorCode, params);
sendTelemetryData(queryId, null, -1, session, this);
}
/**
* @param session SFBaseSession
* @param e throwable exception
*/
public SnowflakeSQLLoggedException(SFBaseSession session, SFException e) {
super(e);
sendTelemetryData(null, null, -1, session, this);
}
/**
* use {@link SnowflakeSQLLoggedException#SnowflakeSQLLoggedException(String, SFBaseSession,
* String)}
*
* @param session SFBaseSession
* @param reason exception reason
*/
@Deprecated
public SnowflakeSQLLoggedException(SFBaseSession session, String reason) {
this(null, session, reason);
}
/**
* @param queryId the query ID
* @param session SFBaseSession
* @param reason exception reason
*/
public SnowflakeSQLLoggedException(String queryId, SFBaseSession session, String reason) {
super(queryId, reason, null);
sendTelemetryData(queryId, null, -1, session, this);
}
}