net.snowflake.client.jdbc.telemetry.TelemetryClient Maven / Gradle / Ivy
/*
* Copyright (c) 2012-2019 Snowflake Computing Inc. All rights reserved.
*/
package net.snowflake.client.jdbc.telemetry;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.node.ArrayNode;
import com.fasterxml.jackson.databind.node.ObjectNode;
import java.io.IOException;
import java.rmi.UnexpectedException;
import java.sql.Connection;
import java.sql.SQLException;
import java.util.LinkedList;
import java.util.Objects;
import java.util.concurrent.Future;
import net.snowflake.client.core.HttpUtil;
import net.snowflake.client.core.ObjectMapperFactory;
import net.snowflake.client.core.SFBaseSession;
import net.snowflake.client.core.SFSession;
import net.snowflake.client.jdbc.SnowflakeConnectionV1;
import net.snowflake.client.jdbc.SnowflakeSQLException;
import net.snowflake.client.jdbc.telemetryOOB.TelemetryThreadPool;
import net.snowflake.client.log.SFLogger;
import net.snowflake.client.log.SFLoggerFactory;
import net.snowflake.client.util.Stopwatch;
import org.apache.http.HttpHeaders;
import org.apache.http.client.methods.HttpPost;
import org.apache.http.entity.StringEntity;
import org.apache.http.impl.client.CloseableHttpClient;
/**
* Copyright (c) 2018-2019 Snowflake Computing Inc. All rights reserved.
*
* Telemetry Service Interface
*/
public class TelemetryClient implements Telemetry {
private static final SFLogger logger = SFLoggerFactory.getLogger(SFBaseSession.class);
private static final String SF_PATH_TELEMETRY = "/telemetry/send";
private static final String SF_PATH_TELEMETRY_SESSIONLESS = "/telemetry/send/sessionless";
// if the number of cached logs is larger than this threshold,
// the telemetry connector will flush the buffer automatically.
private final int forceFlushSize;
private static final int DEFAULT_FORCE_FLUSH_SIZE = 100;
private final String serverUrl;
private final String telemetryUrl;
private final SFSession session;
private LinkedList logBatch;
private static final ObjectMapper mapper = ObjectMapperFactory.getObjectMapper();
private boolean isClosed;
// HTTP client object used to communicate with other machine
private final CloseableHttpClient httpClient;
// the authorization type speficied in sessionless header
private String authType;
// JWT/OAuth token
private String token;
private Object locker = new Object();
// false if meet any error when sending metrics
private boolean isTelemetryServiceAvailable = true;
// Retry timeout for the HTTP request
private static final int TELEMETRY_HTTP_RETRY_TIMEOUT_IN_SEC = 1000;
private TelemetryClient(SFSession session, int flushSize) {
this.session = session;
this.serverUrl = session.getUrl();
this.httpClient = null;
if (this.serverUrl.endsWith("/")) {
this.telemetryUrl =
this.serverUrl.substring(0, this.serverUrl.length() - 1) + SF_PATH_TELEMETRY;
} else {
this.telemetryUrl = this.serverUrl + SF_PATH_TELEMETRY;
}
this.logBatch = new LinkedList<>();
this.isClosed = false;
this.forceFlushSize = flushSize;
}
/**
* Constructor for creating a sessionless telemetry client
*
* @param httpClient client object used to communicate with other machine
* @param serverUrl server url
* @param authType authorization type, should be either KEYPAIR_JWY or OAUTH
* @param flushSize maximum size of telemetry batch before flush
*/
private TelemetryClient(
CloseableHttpClient httpClient, String serverUrl, String authType, int flushSize) {
this.session = null;
this.serverUrl = serverUrl;
this.httpClient = httpClient;
if (!Objects.equals(authType, "KEYPAIR_JWT") && !Objects.equals(authType, "OAUTH")) {
throw new IllegalArgumentException(
"Invalid authType, should be \"KEYPAIR_JWT\" or \"OAUTH\"");
}
this.authType = authType;
if (this.serverUrl.endsWith("/")) {
this.telemetryUrl =
this.serverUrl.substring(0, this.serverUrl.length() - 1) + SF_PATH_TELEMETRY_SESSIONLESS;
} else {
this.telemetryUrl = this.serverUrl + SF_PATH_TELEMETRY_SESSIONLESS;
}
this.logBatch = new LinkedList<>();
this.isClosed = false;
this.forceFlushSize = flushSize;
logger.debug(
"Initializing telemetry client with telemetry url: {}, flush size: {}, auth type: {}",
telemetryUrl,
forceFlushSize,
authType);
}
/**
* Return whether the client can be used to add/send metrics
*
* @return whether client is enabled
*/
public boolean isTelemetryEnabled() {
return (this.session == null || this.session.isClientTelemetryEnabled())
&& this.isTelemetryServiceAvailable;
}
/** Disable any use of the client to add/send metrics */
public void disableTelemetry() {
logger.debug("Disabling telemetry");
this.isTelemetryServiceAvailable = false;
}
/**
* Initialize the telemetry connector
*
* @param conn connection with the session to use for the connector
* @param flushSize maximum size of telemetry batch before flush
* @return a telemetry connector
*/
public static Telemetry createTelemetry(Connection conn, int flushSize) {
try {
return createTelemetry(
(SFSession) conn.unwrap(SnowflakeConnectionV1.class).getSFBaseSession(), flushSize);
} catch (SQLException ex) {
logger.debug("Input connection is not a SnowflakeConnection", false);
return null;
}
}
/**
* Initialize the telemetry connector
*
* @param conn connection with the session to use for the connector
* @return a telemetry connector
*/
public static Telemetry createTelemetry(Connection conn) {
return createTelemetry(conn, DEFAULT_FORCE_FLUSH_SIZE);
}
/**
* Initialize the telemetry connector
*
* @param session session to use for telemetry dumps
* @return a telemetry connector
*/
public static Telemetry createTelemetry(SFSession session) {
return createTelemetry(session, DEFAULT_FORCE_FLUSH_SIZE);
}
/**
* Initialize the telemetry connector
*
* @param session session to use for telemetry dumps
* @param flushSize maximum size of telemetry batch before flush
* @return a telemetry connector
*/
public static Telemetry createTelemetry(SFSession session, int flushSize) {
return new TelemetryClient(session, flushSize);
}
/**
* Initialize the sessionless telemetry connector using KEYPAIR_JWT as the default auth type
*
* @param httpClient client object used to communicate with other machine
* @param serverUrl server url
* @return a telemetry connector
*/
public static Telemetry createSessionlessTelemetry(
CloseableHttpClient httpClient, String serverUrl) {
// By default, use KEYPAIR_JWT as the auth type
return createSessionlessTelemetry(
httpClient, serverUrl, "KEYPAIR_JWT", DEFAULT_FORCE_FLUSH_SIZE);
}
/**
* Initialize the sessionless telemetry connector
*
* @param httpClient client object used to communicate with other machine
* @param serverUrl server url
* @param authType authorization type for sessionless telemetry
* @return a telemetry connector
*/
public static Telemetry createSessionlessTelemetry(
CloseableHttpClient httpClient, String serverUrl, String authType) {
return createSessionlessTelemetry(httpClient, serverUrl, authType, DEFAULT_FORCE_FLUSH_SIZE);
}
/**
* Initialize the sessionless telemetry connector
*
* @param httpClient client object used to communicate with other machine
* @param serverUrl server url
* @param authType authorization type for sessionless telemetry
* @param flushSize maximum size of telemetry batch before flush
* @return a telemetry connector
*/
public static Telemetry createSessionlessTelemetry(
CloseableHttpClient httpClient, String serverUrl, String authType, int flushSize) {
return new TelemetryClient(httpClient, serverUrl, authType, flushSize);
}
/**
* Add log to batch to be submitted to telemetry. Send batch if forceFlushSize reached
*
* @param log entry to add
*/
@Override
public void addLogToBatch(TelemetryData log) {
if (isClosed) {
logger.debug("Telemetry already closed", false);
return;
}
if (!isTelemetryEnabled()) {
return; // if disable, do nothing
}
synchronized (locker) {
this.logBatch.add(log);
}
int logBatchSize = this.logBatch.size();
if (logBatchSize >= this.forceFlushSize) {
logger.debug("Force flushing telemetry batch of size: {}", logBatchSize);
this.sendBatchAsync();
}
}
/**
* Add log to batch to be submitted to telemetry. Send batch if forceFlushSize reached
*
* @param message json node of log
* @param timeStamp timestamp to use for log
*/
public void addLogToBatch(ObjectNode message, long timeStamp) {
this.addLogToBatch(new TelemetryData(message, timeStamp));
}
/** Close telemetry connector and send any unsubmitted logs */
@Override
public void close() {
if (isClosed) {
logger.debug("Telemetry client already closed", false);
return;
}
try {
// sendBatch when close is synchronous, otherwise client might be closed
// before data was sent.
sendBatchAsync().get();
} catch (Throwable e) {
logger.debug("Error when sending batch data, {}", e);
} finally {
this.isClosed = true;
}
}
/**
* Return whether the client has been closed
*
* @return whether client is closed
*/
public boolean isClosed() {
return this.isClosed;
}
@Override
public Future sendBatchAsync() {
return TelemetryThreadPool.getInstance()
.submit(
() -> {
try {
return this.sendBatch();
} catch (Throwable e) {
logger.debug("Failed to send telemetry data, {}", e);
return false;
}
});
}
@Override
public void postProcess(String queryId, String sqlState, int vendorCode, Throwable ex) {
// This is a no-op.
}
/**
* Send all cached logs to server
*
* @return whether the logs were sent successfully
* @throws IOException if closed or uploading batch fails
*/
private boolean sendBatch() throws IOException {
if (isClosed) {
throw new IOException("Telemetry connector is closed");
}
if (!isTelemetryEnabled()) {
return false;
}
LinkedList tmpList;
synchronized (locker) {
tmpList = this.logBatch;
this.logBatch = new LinkedList<>();
}
if (this.session != null && this.session.isClosed()) {
throw new UnexpectedException("Session is closed when sending log");
}
if (!tmpList.isEmpty()) {
Stopwatch stopwatch = new Stopwatch();
stopwatch.start();
// session shared with JDBC
String payload = logsToString(tmpList);
logger.debugNoMask("Payload of telemetry is : " + payload);
HttpPost post = new HttpPost(this.telemetryUrl);
post.setEntity(new StringEntity(payload));
post.setHeader("Content-type", "application/json");
if (this.session == null) {
post.setHeader(HttpHeaders.AUTHORIZATION, "Bearer " + this.token);
post.setHeader("X-Snowflake-Authorization-Token-Type", this.authType);
post.setHeader(HttpHeaders.ACCEPT, "application/json");
} else {
post.setHeader(
HttpHeaders.AUTHORIZATION,
"Snowflake Token=\"" + this.session.getSessionToken() + "\"");
}
String response = null;
try {
response =
this.session == null
? HttpUtil.executeGeneralRequest(
post,
TELEMETRY_HTTP_RETRY_TIMEOUT_IN_SEC,
0,
(int) HttpUtil.getSocketTimeout().toMillis(),
0,
this.httpClient)
: HttpUtil.executeGeneralRequest(
post,
TELEMETRY_HTTP_RETRY_TIMEOUT_IN_SEC,
this.session.getAuthTimeout(),
this.session.getHttpClientSocketTimeout(),
0,
this.session.getHttpClientKey());
stopwatch.stop();
logger.debug(
"Sending telemetry took {} ms. Batch size: {}",
stopwatch.elapsedMillis(),
tmpList.size());
} catch (SnowflakeSQLException e) {
disableTelemetry(); // when got error like 404 or bad request, disable telemetry in this
// telemetry instance
logger.error(
"Telemetry request failed, response: {}, exception: {}", response, e.getMessage());
return false;
}
}
return true;
}
/**
* Send a log to the server, along with any existing logs waiting to be sent
*
* @param log entry to send
* @return whether the logs were sent successfully
* @throws IOException if closed or uploading batch fails
*/
public boolean sendLog(TelemetryData log) throws IOException {
addLogToBatch(log);
return sendBatch();
}
/**
* Send a log to the server, along with any existing logs waiting to be sent
*
* @param message json node of log
* @param timeStamp timestamp to use for log
* @return whether the logs were sent successfully
* @throws IOException if closed or uploading batch fails
*/
public boolean sendLog(ObjectNode message, long timeStamp) throws IOException {
return this.sendLog(new TelemetryData(message, timeStamp));
}
/**
* convert a list of log to a JSON object
*
* @param telemetryData a list of log
* @return the result json string
*/
static ObjectNode logsToJson(LinkedList telemetryData) {
ObjectNode node = mapper.createObjectNode();
ArrayNode logs = mapper.createArrayNode();
for (TelemetryData data : telemetryData) {
logs.add(data.toJson());
}
node.set("logs", logs);
return node;
}
/**
* convert a list of log to a JSON String
*
* @param telemetryData a list of log
* @return the result json string
*/
static String logsToString(LinkedList telemetryData) {
return logsToJson(telemetryData).toString();
}
/**
* For test use only
*
* @return the number of cached logs
*/
public int bufferSize() {
return this.logBatch.size();
}
/**
* For test use only
*
* @return a copy of the logs currently in the buffer
*/
public LinkedList logBuffer() {
return new LinkedList<>(this.logBatch);
}
/**
* Refresh the JWT/OAuth token
*
* @param token latest JWT/OAuth token
*/
public void refreshToken(String token) {
this.token = token;
}
}