com.taosdata.jdbc.ws.Transport Maven / Gradle / Ivy
package com.taosdata.jdbc.ws;
import com.taosdata.jdbc.TSDBError;
import com.taosdata.jdbc.TSDBErrorNumbers;
import com.taosdata.jdbc.common.SerializeBlock;
import com.taosdata.jdbc.enums.WSFunction;
import com.taosdata.jdbc.rs.ConnectionParam;
import com.taosdata.jdbc.utils.CompletableFutureTimeout;
import com.taosdata.jdbc.ws.entity.Request;
import com.taosdata.jdbc.ws.entity.Response;
import org.java_websocket.exceptions.WebsocketNotConnectedException;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.nio.ByteBuffer;
import java.sql.SQLException;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.TimeoutException;
import java.util.function.Consumer;
import static com.taosdata.jdbc.TSDBErrorNumbers.ERROR_CONNECTION_TIMEOUT;
/**
* send message
*/
public class Transport implements AutoCloseable {
public static final int DEFAULT_MESSAGE_WAIT_TIMEOUT = 60_000;
private final WSClient client;
private final InFlightRequest inFlightRequest;
private long timeout;
private boolean closed = false;
public Transport(WSFunction function, ConnectionParam param, InFlightRequest inFlightRequest) throws SQLException {
this.client = WSClient.getInstance(param, function, this);
this.inFlightRequest = inFlightRequest;
this.timeout = param.getRequestTimeout();
}
public void setTextMessageHandler(Consumer textMessageHandler) {
client.setTextMessageHandler(textMessageHandler);
}
public void setBinaryMessageHandler(Consumer binaryMessageHandler) {
client.setBinaryMessageHandler(binaryMessageHandler);
}
public void setTimeout(long timeout) {
this.timeout = timeout;
}
@SuppressWarnings("all")
public Response send(Request request) throws SQLException {
Response response = null;
CompletableFuture completableFuture = new CompletableFuture<>();
String reqString = request.toString();
try {
inFlightRequest.put(new FutureResponse(request.getAction(), request.id(), completableFuture));
} catch (InterruptedException | TimeoutException e) {
throw new SQLException(e);
}
try {
client.send(reqString);
} catch (WebsocketNotConnectedException e) {
if (!client.reconnectBlocking()) {
inFlightRequest.remove(request.getAction(), request.id());
throw TSDBError.createSQLException(TSDBErrorNumbers.ERROR_RESTFul_Client_IOException, "Websocket Not Connected Exception");
}
try {
client.send(reqString);
}catch (Exception ex){
inFlightRequest.remove(request.getAction(), request.id());
throw TSDBError.createSQLException(TSDBErrorNumbers.ERROR_RESTFul_Client_IOException, e.getMessage());
}
}
CompletableFuture responseFuture = CompletableFutureTimeout.orTimeout(
completableFuture, timeout, TimeUnit.MILLISECONDS, reqString);
try {
response = responseFuture.get();
} catch (InterruptedException | ExecutionException e) {
inFlightRequest.remove(request.getAction(), request.id());
throw TSDBError.createSQLException(TSDBErrorNumbers.ERROR_RESTFul_Client_IOException, e.getMessage());
}
return response;
}
public Response send(String action, long reqId, long stmtId, long type, byte[] rawData) throws SQLException {
ByteArrayOutputStream buffer = new ByteArrayOutputStream();
try {
buffer.write(SerializeBlock.longToBytes(reqId));
buffer.write(SerializeBlock.longToBytes(stmtId));
buffer.write(SerializeBlock.longToBytes(type));
buffer.write(rawData);
} catch (IOException e) {
throw new SQLException("data serialize error!", e);
}
Response response = null;
CompletableFuture completableFuture = new CompletableFuture<>();
try {
inFlightRequest.put(new FutureResponse(action, reqId, completableFuture));
} catch (InterruptedException | TimeoutException e) {
throw new SQLException(e);
}
try {
client.send(buffer.toByteArray());
} catch (WebsocketNotConnectedException e) {
if (!client.reconnectBlocking()) {
inFlightRequest.remove(action, reqId);
throw TSDBError.createSQLException(TSDBErrorNumbers.ERROR_RESTFul_Client_IOException, "Websocket Not Connected Exception");
}
try {
client.send(buffer.toByteArray());
}catch (Exception ex){
inFlightRequest.remove(action, reqId);
throw TSDBError.createSQLException(TSDBErrorNumbers.ERROR_RESTFul_Client_IOException, e.getMessage());
}
}
String reqString = "action:" + action + ", reqId:" + reqId + ", stmtId:" + stmtId + ", bindType" + type;
CompletableFuture responseFuture = CompletableFutureTimeout.orTimeout(completableFuture, timeout, TimeUnit.MILLISECONDS, reqString);
try {
response = responseFuture.get();
} catch (InterruptedException | ExecutionException e) {
inFlightRequest.remove(action, reqId);
throw TSDBError.createSQLException(TSDBErrorNumbers.ERROR_RESTFul_Client_IOException, e.getMessage());
}
return response;
}
public void sendWithoutRep(Request request) throws SQLException {
try {
client.send(request.toString());
} catch (WebsocketNotConnectedException e) {
if (!client.reconnectBlocking()) {
throw TSDBError.createSQLException(TSDBErrorNumbers.ERROR_RESTFul_Client_IOException, "Websocket Not Connected Exception");
}
try {
client.send(request.toString());
}catch (Exception ex){
throw TSDBError.createSQLException(TSDBErrorNumbers.ERROR_RESTFul_Client_IOException, e.getMessage());
}
}
}
public boolean isClosed() {
return closed;
}
public void disconnectAndReconnect() throws SQLException {
try {
client.closeBlocking();
if (!client.reconnectBlockingWithoutRetry()){
throw TSDBError.createSQLException(TSDBErrorNumbers.ERROR_RESTFul_Client_IOException, "websocket reconnect failed!");
}
} catch (Exception e) {
throw TSDBError.createSQLException(TSDBErrorNumbers.ERROR_RESTFul_Client_IOException, e.getMessage());
}
}
@Override
public synchronized void close() {
closed = true;
inFlightRequest.close();
client.shutdown();
}
public static void checkConnection(Transport transport, int connectTimeout) throws SQLException {
try {
if (!transport.client.connectBlocking(connectTimeout, TimeUnit.MILLISECONDS)) {
transport.close();
throw TSDBError.createSQLException(ERROR_CONNECTION_TIMEOUT,
"can't create connection with server within: " + connectTimeout + " milliseconds");
}
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
transport.close();
throw new SQLException("create websocket connection has been Interrupted ", e);
}
}
public void shutdown() {
closed = true;
if (inFlightRequest.hasInFlightRequest()) {
CompletableFuture future = CompletableFuture.runAsync(() -> {
try {
TimeUnit.MILLISECONDS.sleep(timeout);
} catch (InterruptedException e) {
// ignore
}
});
future.thenRun(this::close);
} else {
close();
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy