ru.yandex.clickhouse.except.ClickHouseExceptionSpecifier Maven / Gradle / Ivy
The newest version!
package ru.yandex.clickhouse.except;
import org.apache.http.conn.ConnectTimeoutException;
import ru.yandex.clickhouse.util.Utils;
import java.net.ConnectException;
import java.net.SocketTimeoutException;
import com.clickhouse.client.logging.Logger;
import com.clickhouse.client.logging.LoggerFactory;
/**
* Specify ClickHouse exception to ClickHouseException and fill it with a vendor
* code.
*/
public final class ClickHouseExceptionSpecifier {
private static final Logger log = LoggerFactory.getLogger(ClickHouseExceptionSpecifier.class);
private ClickHouseExceptionSpecifier() {
}
public static ClickHouseException specify(Throwable cause, String host, int port) {
return specify(cause != null ? cause.getMessage() : null, cause, host, port);
}
public static ClickHouseException specify(String clickHouseMessage, String host, int port) {
return specify(clickHouseMessage, null, host, port);
}
public static ClickHouseException specify(String clickHouseMessage) {
return specify(clickHouseMessage, "unknown", -1);
}
/**
* Here we expect the ClickHouse error message to be of the following format:
* "Code: 10, e.displayText() = DB::Exception: ...".
*/
private static ClickHouseException specify(String clickHouseMessage, Throwable cause, String host, int port) {
if (Utils.isNullOrEmptyString(clickHouseMessage) && cause != null) {
return getException(cause, host, port);
}
try {
int code;
if (clickHouseMessage.startsWith("Poco::Exception. Code: 1000, ")) {
code = 1000;
} else {
// Code: 175, e.displayText() = DB::Exception:
code = getErrorCode(clickHouseMessage);
}
// ошибку в изначальном виде все-таки укажем
Throwable messageHolder = cause != null ? cause : new Throwable(clickHouseMessage);
if (code == -1) {
return getException(messageHolder, host, port);
}
return new ClickHouseException(code, messageHolder, host, port);
} catch (Exception e) {
log.error(
"Unsupported ClickHouse error format, please fix ClickHouseExceptionSpecifier, message: %s, error: %s",
clickHouseMessage, e.getMessage());
return new ClickHouseUnknownException(clickHouseMessage, cause, host, port);
}
}
private static int getErrorCode(String errorMessage) {
int startIndex = errorMessage.indexOf(' ');
if (startIndex >= 0) {
for (int i = ++startIndex, len = errorMessage.length(); i < len; i++) {
char ch = errorMessage.charAt(i);
if (ch == '.' || ch == ',' || Character.isWhitespace(ch)) {
try {
return Integer.parseInt(errorMessage.substring(startIndex, i));
} catch (NumberFormatException e) {
// ignore
}
break;
}
}
}
return -1;
}
private static ClickHouseException getException(Throwable cause, String host, int port) {
if (cause instanceof SocketTimeoutException)
// if we've got SocketTimeoutException, we'll say that the query is not good.
// This is not the same as SOCKET_TIMEOUT of clickhouse
// but it actually could be a failing ClickHouse
{
return new ClickHouseException(ClickHouseErrorCode.TIMEOUT_EXCEEDED.code, cause, host, port);
} else if (cause instanceof ConnectTimeoutException || cause instanceof ConnectException)
// couldn't connect to ClickHouse during connectTimeout
{
return new ClickHouseException(ClickHouseErrorCode.NETWORK_ERROR.code, cause, host, port);
} else {
return new ClickHouseUnknownException(cause, host, port);
}
}
}