com.alibaba.hologres.client.HoloClient Maven / Gradle / Ivy
/*
* Copyright (c) 2020. Alibaba Group Holding Limited
*/
package com.alibaba.hologres.client;
import com.alibaba.hologres.client.exception.ExceptionCode;
import com.alibaba.hologres.client.exception.HoloClientException;
import com.alibaba.hologres.client.exception.HoloClientWithDetailsException;
import com.alibaba.hologres.client.function.FunctionWithSQLException;
import com.alibaba.hologres.client.impl.ExecutionPool;
import com.alibaba.hologres.client.impl.action.CopyAction;
import com.alibaba.hologres.client.impl.action.PutAction;
import com.alibaba.hologres.client.impl.action.ScanAction;
import com.alibaba.hologres.client.impl.action.SqlAction;
import com.alibaba.hologres.client.impl.binlog.BinlogOffset;
import com.alibaba.hologres.client.impl.binlog.Committer;
import com.alibaba.hologres.client.impl.binlog.TableSchemaSupplier;
import com.alibaba.hologres.client.impl.binlog.action.BinlogAction;
import com.alibaba.hologres.client.impl.collector.ActionCollector;
import com.alibaba.hologres.client.impl.collector.BatchState;
import com.alibaba.hologres.client.impl.copy.CopyContext;
import com.alibaba.hologres.client.impl.copy.InternalPipedOutputStream;
import com.alibaba.hologres.client.model.ExportContext;
import com.alibaba.hologres.client.model.HoloVersion;
import com.alibaba.hologres.client.model.ImportContext;
import com.alibaba.hologres.client.model.Partition;
import com.alibaba.hologres.client.model.Record;
import com.alibaba.hologres.client.model.RecordScanner;
import com.alibaba.hologres.client.model.TableName;
import com.alibaba.hologres.client.model.TableSchema;
import com.alibaba.hologres.client.model.WriteMode;
import com.alibaba.hologres.client.model.checkandput.CheckAndPutCondition;
import com.alibaba.hologres.client.model.checkandput.CheckAndPutRecord;
import com.alibaba.hologres.client.model.checkandput.CheckCompareOp;
import com.alibaba.hologres.client.utils.IdentifierUtil;
import com.alibaba.hologres.client.utils.Tuple;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.io.Closeable;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.io.PipedInputStream;
import java.io.PipedOutputStream;
import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.Types;
import java.util.ArrayList;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.NavigableMap;
import java.util.TreeMap;
import java.util.concurrent.ArrayBlockingQueue;
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.atomic.AtomicBoolean;
/**
* 线程不安全,每个client,最多会创建2个JDBC connection,读写分离.
*/
public class HoloClient implements Closeable {
public static final Logger LOGGER = LoggerFactory.getLogger(HoloClient.class);
static {
LOGGER.info("=========holo-client version==========");
LOGGER.info("version:{}", com.alibaba.hologres.client.Version.version);
LOGGER.info("revision:{}", com.alibaba.hologres.client.Version.revision);
LOGGER.info("date:{}", com.alibaba.hologres.client.Version.date);
LOGGER.info("======================================");
}
static {
// Load DriverManager first to avoid deadlock between DriverManager's
// static initialization block and specific driver class's static
// initialization block when two different driver classes are loading
// concurrently using Class.forName while DriverManager is uninitialized
// before.
//
// This could happen in JDK 8 but not above as driver loading has been
// moved out of DriverManager's static initialization block since JDK 9.
DriverManager.getDrivers();
}
private ActionCollector collector;
/**
* 是否使用fixed fe.
* 开启的话会创建fixed pool,用于执行点查、写入以及prefix scan.
*/
private final boolean useFixedFe;
private ExecutionPool fixedPool = null;
private ExecutionPool pool = null;
private final HoloConfig config;
/**
* 在AsyncCommit为true,调用put方法时,当记录数>=writeBatchSize 或 总记录字节数数>=writeBatchByteSize 调用flush进行提交.
* 否则每次调用put都会调用flush.
* 默认为true
*
* @HasGetter
* @HasSetter
*/
boolean asyncCommit = true;
/**
* 是否shading打包后的环境.
* 打包后的运行环境,isShadingEnv=true,将把jdbc:postgresql:改为jdbc:hologres:
* 本地跑测试时,isShadingEnv=false。此时pgjdbc没有shading,仍然要使用jdbc:postgresql:
*/
boolean isShadingEnv = false;
boolean isEmbeddedPool = false;
boolean isEmbeddedFixedPool = false;
public HoloClient(HoloConfig config) throws HoloClientException {
try {
DriverManager.getDrivers();
Class.forName("com.alibaba.hologres.org.postgresql.Driver");
isShadingEnv = true;
} catch (Exception e) {
try {
DriverManager.getDrivers();
Class.forName("org.postgresql.Driver");
} catch (Exception e2) {
throw new HoloClientException(ExceptionCode.INTERNAL_ERROR, "load driver fail", e);
}
}
checkConfig(config);
this.config = config;
this.useFixedFe = config.isUseFixedFe();
}
private void checkConfig(HoloConfig config) throws HoloClientException {
if (config.getJdbcUrl() == null || config.getJdbcUrl().isEmpty()) {
throw new HoloClientException(ExceptionCode.INVALID_Config, "jdbcUrl cannot be null");
}
if (config.getPassword() == null || config.getPassword().isEmpty()) {
throw new HoloClientException(ExceptionCode.INVALID_Config, "password cannot be null");
}
if (config.getUsername() == null || config.getUsername().isEmpty()) {
throw new HoloClientException(ExceptionCode.INVALID_Config, "username cannot be null");
}
if (config.getWriteBatchSize() < 1) {
throw new HoloClientException(ExceptionCode.INVALID_Config, "batchSize must > 0");
}
if (config.getWriteBatchByteSize() < 1) {
throw new HoloClientException(ExceptionCode.INVALID_Config, "batchByteSize must > 0");
}
}
public TableSchema getTableSchema(String tableName) throws HoloClientException {
return getTableSchema(TableName.valueOf(tableName), false);
}
public TableSchema getTableSchema(String tableName, boolean noCache) throws HoloClientException {
return getTableSchema(TableName.valueOf(tableName), noCache);
}
public TableSchema getTableSchema(TableName tableName) throws HoloClientException {
return getTableSchema(tableName, false);
}
public TableSchema getTableSchema(TableName tableName, boolean noCache) throws HoloClientException {
ensurePoolOpen();
return pool.getOrSubmitTableSchema(tableName, noCache);
}
private void checkGet(Get get) throws HoloClientException {
if (get == null) {
throw new HoloClientException(ExceptionCode.CONSTRAINT_VIOLATION, "Get cannot be null");
}
if (get.getRecord().getSchema().getPrimaryKeys().length == 0) {
throw new HoloClientException(ExceptionCode.CONSTRAINT_VIOLATION, "Get table must have primary key:" + get.getRecord().getSchema().getTableNameObj().getFullName());
}
for (int index : get.getRecord().getKeyIndex()) {
if (!get.getRecord().isSet(index) || null == get.getRecord().getObject(index)) {
throw new HoloClientException(ExceptionCode.CONSTRAINT_VIOLATION, "Get primary key cannot be null:" + get.getRecord().getSchema().getColumnSchema()[index].getName());
}
}
}
private void checkPut(Put put) throws HoloClientException {
if (put == null) {
throw new HoloClientException(ExceptionCode.CONSTRAINT_VIOLATION, "Put cannot be null");
}
for (int index : put.getRecord().getKeyIndex()) {
if ((!put.getRecord().isSet(index) || null == put.getRecord().getObject(index)) && put.getRecord().getSchema().getColumn(index).getDefaultValue() == null) {
throw new HoloClientWithDetailsException(ExceptionCode.CONSTRAINT_VIOLATION, "Put primary key cannot be null:" + put.getRecord().getSchema().getColumnSchema()[index].getName(), put.getRecord());
}
}
if (put.getRecord().getSchema().isPartitionParentTable() && (!put.getRecord().isSet(put.getRecord().getSchema().getPartitionIndex()) || null == put.getRecord().getObject(put.getRecord().getSchema().getPartitionIndex()))) {
throw new HoloClientWithDetailsException(ExceptionCode.CONSTRAINT_VIOLATION, "Put partition key cannot be null:" + put.getRecord().getSchema().getColumnSchema()[put.getRecord().getSchema().getPartitionIndex()].getName(), put.getRecord());
}
if (put.getRecord().getType() == Put.MutationType.DELETE && put.getRecord().getSchema().getPrimaryKeys().length == 0) {
throw new HoloClientWithDetailsException(ExceptionCode.CONSTRAINT_VIOLATION, "Delete Put table must have primary key:" + put.getRecord().getSchema().getTableNameObj().getFullName(), put.getRecord());
}
}
/** 检查CheckAndPut内容并进行重写. */
private void checkCheckAndPut(CheckAndPut put) throws HoloClientException {
if (put == null) {
throw new HoloClientException(ExceptionCode.CONSTRAINT_VIOLATION, "CheckAndPut cannot be null");
}
if (put.getRecord().getSchema().getPrimaryKeys().length == 0) {
throw new HoloClientWithDetailsException(ExceptionCode.CONSTRAINT_VIOLATION, "CheckAndPut need table have primary key:" + put.getRecord().getSchema().getTableNameObj().getFullName(), put.getRecord());
}
for (int index : put.getRecord().getKeyIndex()) {
if ((!put.getRecord().isSet(index) || null == put.getRecord().getObject(index)) && put.getRecord().getSchema().getColumn(index).getDefaultValue() == null) {
throw new HoloClientWithDetailsException(ExceptionCode.CONSTRAINT_VIOLATION, "CheckAndPut primary key cannot be null:" + put.getRecord().getSchema().getColumnSchema()[index].getName(), put.getRecord());
}
}
if (put.getRecord().getSchema().isPartitionParentTable() && (!put.getRecord().isSet(put.getRecord().getSchema().getPartitionIndex()) || null == put.getRecord().getObject(put.getRecord().getSchema().getPartitionIndex()))) {
throw new HoloClientWithDetailsException(ExceptionCode.CONSTRAINT_VIOLATION, "CheckAndPut partition key cannot be null:" + put.getRecord().getSchema().getColumnSchema()[put.getRecord().getSchema().getPartitionIndex()].getName(), put.getRecord());
}
if (config.getWriteMode() == WriteMode.INSERT_OR_IGNORE) {
throw new HoloClientException(ExceptionCode.NOT_SUPPORTED, "CheckAndPut not supports writeMode insertOrIgnore.");
}
CheckAndPutRecord record = put.getRecord();
String checkColumnName = record.getCheckAndPutCondition().getCheckColumnName();
CheckCompareOp checkOp = record.getCheckAndPutCondition().getCheckOp();
Object checkValue = record.getCheckAndPutCondition().getCheckValue();
Object nullValue = record.getCheckAndPutCondition().getNullValue();
Integer checkColumnIndex = record.getSchema().getColumnIndex(checkColumnName);
if (checkColumnIndex == null || checkColumnIndex < 0) {
throw new HoloClientWithDetailsException(ExceptionCode.INVALID_REQUEST, "checkColumn " + checkColumnName + " is not exists in table " + put.getRecord().getSchema().getTableNameObj().getFullName(), put.getRecord());
} else {
// CheckAndPutCondition 可能是通过columnName初始化的
put.getRecord().getCheckAndPutCondition().setCheckColumn(record.getSchema().getColumn(checkColumnIndex));
}
if (checkOp == CheckCompareOp.IS_NULL || checkOp == CheckCompareOp.IS_NOT_NULL) {
// is null,is not null不需要做coalesce,不关心nullValue,checkValue的值
nullValue = null;
checkValue = null;
put.getRecord().getCheckAndPutCondition().setNullValue(nullValue);
put.getRecord().getCheckAndPutCondition().setCheckValue(checkValue);
} else {
// >,>=,=,<>,<,<=操作符不能和null进行比较(结果恒为false),需要提供nullValue进行重写,相当于sql中的coalesce函数
if (nullValue == null && record.getCheckAndPutCondition().getCheckColumn().getAllowNull()) {
LOGGER.warn("When a field allows null, it is recommended to set nullValue to prevent null fields from being updated.");
}
// checkValue没有设置,表示要使用当前put中的checkColumn值和已有的值进行比较,所以Record的checkColumn必须被set
if (checkValue == null && !put.getRecord().isSet(checkColumnIndex)) {
throw new HoloClientWithDetailsException(ExceptionCode.INVALID_REQUEST, "checkColumn " + checkColumnName + " should be set when not set checkValue.", put.getRecord());
}
// checkValue没有设置,表示要使用当前put中的checkColumn值和已有的值进行比较,所以Record的checkColumn必须被set,且对于delete来说不能设置为null(抛出异常: internal error: binaryrow should not be empty)
if (checkValue == null && put.getRecord().getObject(checkColumnIndex) == null && put.getRecord().getType() == Put.MutationType.DELETE) {
// delete from table where (pk = $1 and $2 > checkColumn). 这里的$2不能是null
throw new HoloClientWithDetailsException(ExceptionCode.INVALID_REQUEST, "checkColumn " + checkColumnName + " should be set not null when not set checkValue and mutationType is delete.", put.getRecord());
}
}
// delete不支持攒批,直接将delete的checkValue设置为put中的checkColumn字段的值, 这样不同的Condition会分批提交
if (checkValue == null && put.getRecord().getType() == Put.MutationType.DELETE && checkOp != CheckCompareOp.IS_NULL && checkOp != CheckCompareOp.IS_NOT_NULL) {
checkValue = put.getRecord().getObject(checkColumnIndex);
put.getRecord().getCheckAndPutCondition().setCheckValue(checkValue);
}
}
public CompletableFuture get(Get get) throws HoloClientException {
ensurePoolOpen();
checkGet(get);
get.setStartTime(System.nanoTime());
get.setFuture(new CompletableFuture<>());
if (get.isFullColumn()) {
for (int i = 0; i < get.getRecord().getSchema().getColumnSchema().length; ++i) {
if (!get.getRecord().isSet(i)) {
get.getRecord().setObject(i, null);
}
}
}
try {
if (rewriteForPartitionTable(get.getRecord(), false, false)) {
get.getFuture().complete(null);
}
} catch (HoloClientException e) {
get.getFuture().completeExceptionally(e);
}
collector.appendGet(get);
return get.getFuture();
}
public List> get(List gets) throws HoloClientException {
ensurePoolOpen();
for (Get get : gets) {
checkGet(get);
get.setStartTime(System.nanoTime());
get.setFuture(new CompletableFuture<>());
if (get.isFullColumn()) {
for (int i = 0; i < get.getRecord().getSchema().getColumnSchema().length; ++i) {
if (!get.getRecord().isSet(i)) {
get.getRecord().setObject(i, null);
}
}
}
try {
if (rewriteForPartitionTable(get.getRecord(), false, false)) {
get.getFuture().complete(null);
}
} catch (HoloClientException e) {
get.getFuture().completeExceptionally(e);
}
}
List> ret = new ArrayList<>();
collector.appendGet(gets);
for (Get get : gets) {
ret.add(get.getFuture());
}
return ret;
}
public CompletableFuture sql(FunctionWithSQLException func) throws HoloClientException {
ensurePoolOpen();
SqlAction action = new SqlAction<>(func);
while (!pool.submit(action)) {
}
return action.getFuture();
}
public RecordScanner scan(Scan scan) throws HoloClientException {
return doScan(scan).getResult();
}
public CompletableFuture asyncScan(Scan scan) throws HoloClientException {
return doScan(scan).getFuture();
}
private ScanAction doScan(Scan scan) throws HoloClientException {
ensurePoolOpen();
ScanAction action = new ScanAction(scan);
ExecutionPool execPool = useFixedFe ? fixedPool : pool;
while (!execPool.submit(action)) {
}
return action;
}
private void ensurePoolOpen() throws HoloClientException {
if (pool == null) {
synchronized (this) {
if (pool == null) {
ExecutionPool temp = new ExecutionPool("embedded-" + config.getAppName(), config, isShadingEnv, false);
// 当useFixedFe为true 则不赋值新的collector,调用这个函数仅仅为了把pool标记为started.
ActionCollector tempCollector = temp.register(this, config);
if (!this.useFixedFe) {
collector = tempCollector;
}
pool = temp;
isEmbeddedPool = true;
}
}
}
if (!pool.isRunning()) {
throw new HoloClientException(ExceptionCode.ALREADY_CLOSE,
"already close at " + pool.getCloseReasonStack().l, pool.getCloseReasonStack().r);
}
if (useFixedFe && fixedPool == null) {
synchronized (this) {
if (fixedPool == null) {
ExecutionPool temp = new ExecutionPool("embedded-fixed-" + config.getAppName(), config, isShadingEnv, true);
// 当useFixedFe为true时, 只会使用这个fixedPool注册的collector.
collector = temp.register(this, config);
fixedPool = temp;
isEmbeddedFixedPool = true;
}
}
}
if (useFixedFe && !fixedPool.isRunning()) {
throw new HoloClientException(ExceptionCode.ALREADY_CLOSE, "already close");
}
}
public synchronized void setPool(ExecutionPool pool) throws HoloClientException {
if (pool.isFixedPool()) {
throw new HoloClientException(ExceptionCode.INTERNAL_ERROR, "fixed pool recived, require is not fixed");
}
ExecutionPool temp = pool;
ActionCollector tempCollector = temp.register(this, config);
if (!this.useFixedFe) {
collector = tempCollector;
}
this.pool = temp;
isEmbeddedPool = false;
}
public synchronized void setFixedPool(ExecutionPool fixedPool) throws HoloClientException {
if (!useFixedFe || fixedPool == null || !fixedPool.isFixedPool()) {
throw new HoloClientException(ExceptionCode.INTERNAL_ERROR, "fixedPool required not null, and enable HoloConfig:useFixedFe");
}
ExecutionPool tempFixedPool = fixedPool;
this.fixedPool = tempFixedPool;
collector = tempFixedPool.register(this, config);
isEmbeddedFixedPool = false;
}
private void tryThrowException() throws HoloClientException {
if (pool != null) {
pool.tryThrowException();
}
if (fixedPool != null) {
fixedPool.tryThrowException();
}
}
/**
* 如果读写分区表,修改对应操作的schema为分区子表.
*
* @param record 操作的Record
* @param createIfNotExists dynamicPartition为true,且是非delete的put操作时,自动创建分区
* @param exceptionIfNotExists 分区表不存在时是否抛出异常,get和delete操作发现子表不存在不会抛出异常
* @return 是否可以忽略本次操作,比如delete(PUT)但是分区子表不存在的时候;GET但分区子表不存在的时候
* @throws HoloClientException 获取分区或者根据分区信息获取TableSchema异常 那么complete exception
*/
private boolean rewriteForPartitionTable(Record record, boolean createIfNotExists, boolean exceptionIfNotExists) throws HoloClientException {
TableSchema schema = record.getSchema();
if (schema.isPartitionParentTable()) {
boolean isStr = Types.VARCHAR == schema.getColumn(schema.getPartitionIndex()).getType() || Types.DATE == schema.getColumn(schema.getPartitionIndex()).getType();
String value = String.valueOf(record.getObject(schema.getPartitionIndex()));
Partition partition = pool.getOrSubmitPartition(schema.getTableNameObj(), value, isStr, createIfNotExists);
if (partition != null) {
TableSchema newSchema = pool.getOrSubmitTableSchema(TableName.valueOf(IdentifierUtil.quoteIdentifier(partition.getSchemaName(), true), IdentifierUtil.quoteIdentifier(partition.getTableName(), true)), false);
record.changeToChildSchema(newSchema);
} else if (exceptionIfNotExists) {
throw new HoloClientWithDetailsException(ExceptionCode.TABLE_NOT_FOUND, "child table is not found", record);
} else {
return true;
}
}
return false;
}
public void put(Put put) throws HoloClientException {
ensurePoolOpen();
tryThrowException();
checkPut(put);
ExecutionPool execPool = useFixedFe ? fixedPool : pool;
if (!rewriteForPartitionTable(put.getRecord(), config.isDynamicPartition() && !Put.MutationType.DELETE.equals(put.getRecord().getType()), !Put.MutationType.DELETE.equals(put.getRecord().getType()))) {
if (!asyncCommit) {
Record r = put.getRecord();
PutAction action = new PutAction(Collections.singletonList(r), r.getByteSize(), config.getWriteMode(), BatchState.SizeEnough);
while (!execPool.submit(action)) {
}
action.getResult();
} else {
collector.append(put.getRecord());
}
}
}
public CompletableFuture putAsync(Put put) throws HoloClientException {
ensurePoolOpen();
tryThrowException();
checkPut(put);
CompletableFuture ret = new CompletableFuture<>();
if (!rewriteForPartitionTable(put.getRecord(), config.isDynamicPartition() && !Put.MutationType.DELETE.equals(put.getRecord().getType()), !Put.MutationType.DELETE.equals(put.getRecord().getType()))) {
put.getRecord().setPutFuture(ret);
collector.append(put.getRecord());
} else {
put.getRecord().setPutFuture(ret);
ret.complete(null);
}
return ret;
}
public void put(List puts) throws HoloClientException {
ensurePoolOpen();
tryThrowException();
HoloClientWithDetailsException detailException = null;
List putList = new ArrayList<>();
for (Put put : puts) {
try {
checkPut(put);
if (!rewriteForPartitionTable(put.getRecord(), config.isDynamicPartition() && !Put.MutationType.DELETE.equals(put.getRecord().getType()), !Put.MutationType.DELETE.equals(put.getRecord().getType()))) {
putList.add(put);
}
} catch (HoloClientWithDetailsException e) {
if (detailException == null) {
detailException = e;
} else {
detailException.merge(e);
}
}
}
for (Put put : putList) {
collector.append(put.getRecord());
}
if (!asyncCommit) {
collector.flush(false);
}
if (detailException != null) {
throw detailException;
}
}
/**
* 假设checkOp为GREATER
* 当checkValue不为null时, 用checkValue与表中checkColumn的当前值比较,满足checkOp时,则执行put,相当于sql `checkValue > old.column1`.
* 当checkValue为null时, 用Put中的新值与表中checkColumn当前值和进行比较,满足checkOp时,则执行put,相当于sql `excluded.column1 > old.column1`.
*
* @param checkColumn column
* @param checkOp op
* @param checkValue value
* @param nullValue 当前值为null时,视做nullValue。相当于sql `coalesce(old.column1, nullValue)`.
* @param put put
* @throws HoloClientException e
*/
public void checkAndPut(String checkColumn, CheckCompareOp checkOp, Object checkValue, Object nullValue, Put put) throws HoloClientException {
TableSchema schema = put.getRecord().getSchema();
CheckAndPutRecord checkAndPutRecord = new CheckAndPutRecord(put.getRecord(),
new CheckAndPutCondition(schema.getColumn(schema.getColumnIndex(checkColumn)), checkOp, checkValue, nullValue));
checkAndPut(new CheckAndPut(checkAndPutRecord));
}
public void checkAndPut(String checkColumn, CheckCompareOp checkOp, Object checkValue, Put put) throws HoloClientException {
checkAndPut(checkColumn, checkOp, checkValue, null, put);
}
/**
* 兼容beta版本接口. 不建议使用, checkValue会从put的record中获取, 只能每次单条写入无法攒批.
*/
public void checkAndPut(String checkColumn, CheckCompareOp checkOp, Put put) throws HoloClientException {
checkAndPut(checkColumn, checkOp, put.getRecord().getObject(checkColumn), null, put);
}
public void checkAndPut(String checkColumn, Object nullValue, CheckCompareOp checkOp, Put put) throws HoloClientException {
checkAndPut(checkColumn, checkOp, null, nullValue, put);
}
/**
* 直接传入CheckAndPut.
*
* @param put CheckAndPut
* @throws HoloClientException e
*/
public void checkAndPut(CheckAndPut put) throws HoloClientException {
ensurePoolOpen();
tryThrowException();
checkCheckAndPut(put);
final CheckAndPutRecord record = put.getRecord();
ExecutionPool execPool = useFixedFe ? fixedPool : pool;
if (!rewriteForPartitionTable(put.getRecord(), config.isDynamicPartition() && !Put.MutationType.DELETE.equals(put.getRecord().getType()), !Put.MutationType.DELETE.equals(put.getRecord().getType()))) {
if (!asyncCommit) {
PutAction action = new PutAction(Collections.singletonList(record), record.getByteSize(), config.getWriteMode(), BatchState.SizeEnough);
while (!execPool.submit(action)) {
}
action.getResult();
} else {
collector.append(put.getRecord());
}
}
}
public ExportContext exportData(Exporter exporter) throws HoloClientException {
ensurePoolOpen();
tryThrowException();
int threadSize = exporter.getThreadSize();
int maxThread = this.config.readThreadSize > 1 ? this.config.readThreadSize - 1 : 1;
if (threadSize > maxThread) {
LOGGER.warn("Thread size is larger than max read thread size of holo client, will be using {}", maxThread);
threadSize = maxThread;
}
int shardCount = Command.getShardCount(this, exporter.getSchema());
int startShard = exporter.getStartShardId() == -1 ? 0 : exporter.getStartShardId();
int endShard = exporter.getEndShardId() == -1 ? shardCount : exporter.getEndShardId();
if (threadSize > (endShard - startShard)) {
threadSize = endShard - startShard;
LOGGER.warn("Thread size is larger than shard count, will be using thread size {}", threadSize);
}
OutputStream os = exporter.getOutputStream();
InputStream[] istreams = new InputStream[threadSize];
OutputStream[] ostreams = new OutputStream[threadSize];
if (os == null) {
for (int t = 0; t < threadSize; t++) {
PipedOutputStream ostream = new InternalPipedOutputStream();
PipedInputStream istream = new PipedInputStream();
ostreams[t] = ostream;
istreams[t] = istream;
try {
(ostream).connect(istream);
} catch (IOException e) {
throw new HoloClientException(ExceptionCode.INTERNAL_ERROR, "should not happen", e);
}
}
} else {
ostreams[0] = os;
}
int shardSize = (endShard - startShard) / threadSize;
int remain = (endShard - startShard) % threadSize;
CopyAction[] actions = new CopyAction[threadSize];
for (int t = 0; t < threadSize; t++) {
int end;
if (remain > 0) {
end = startShard + shardSize + 1;
remain--;
} else {
end = startShard + shardSize;
}
CopyAction action = new CopyAction(exporter.getSchema(), ostreams[t], null, startShard, end, CopyAction.Mode.OUT);
startShard = end;
actions[t] = action;
while (!pool.submit(action)) {
}
}
try {
CopyContext[] copyContexts = new CopyContext[threadSize];
CompletableFuture[] futures = new CompletableFuture[threadSize];
for (int t = 0; t < threadSize; t++) {
copyContexts[t] = actions[t].getReadyToStart().get();
futures[t] = actions[t].getFuture();
}
return new ExportContext(futures, copyContexts, istreams);
} catch (InterruptedException e) {
throw new HoloClientException(ExceptionCode.INTERNAL_ERROR, "interrupt", e);
} catch (ExecutionException e) {
Throwable cause = e.getCause();
if (cause instanceof HoloClientException) {
throw (HoloClientException) cause;
} else {
throw new HoloClientException(ExceptionCode.INTERNAL_ERROR, "", cause);
}
}
}
public ImportContext importData(Importer importer) throws HoloClientException {
ensurePoolOpen();
tryThrowException();
int threadSize = importer.getThreadSize();
int maxThread = this.config.writeThreadSize > 1 ? this.config.writeThreadSize - 1 : 1;
if (threadSize > maxThread) {
LOGGER.warn("Thread size is larger than max write thread size of holo client, will be using {}", maxThread);
threadSize = maxThread;
}
int shardCount = Command.getShardCount(this, importer.getSchema());
int startShard = importer.getStartShardId() == -1 ? 0 : importer.getStartShardId();
int endShard = importer.getEndShardId() == -1 ? shardCount : importer.getEndShardId();
if (threadSize > (endShard - startShard)) {
threadSize = endShard - startShard;
LOGGER.warn("Thread size is larger than shard count, will be using thread size {}", threadSize);
}
InputStream is = importer.getInputStream();
InputStream[] istreams = new InputStream[threadSize];
OutputStream[] ostreams = new OutputStream[threadSize];
if (is == null) {
for (int t = 0; t < threadSize; t++) {
PipedInputStream istream = new PipedInputStream(importer.getBufferSize() > 0 ? importer.getBufferSize() : 1024);
PipedOutputStream ostream = new PipedOutputStream();
istreams[t] = istream;
ostreams[t] = ostream;
try {
ostream.connect(istream);
} catch (IOException e) {
throw new HoloClientException(ExceptionCode.INTERNAL_ERROR, "should not happen", e);
}
}
} else {
istreams[0] = is;
}
int shardSize = (endShard - startShard) / threadSize;
int remain = (endShard - startShard) % threadSize;
CopyAction[] actions = new CopyAction[threadSize];
NavigableMap shardMap = new TreeMap<>();
for (int t = 0; t < threadSize; t++) {
int end;
if (remain > 0) {
end = startShard + shardSize + 1;
remain--;
} else {
end = startShard + shardSize;
}
CopyAction action = new CopyAction(importer.getSchema(), null, istreams[t], startShard, end, CopyAction.Mode.IN);
action.setBufferSize(importer.getBufferSize());
shardMap.put(startShard, t);
startShard = end;
actions[t] = action;
while (!pool.submit(action)) {
}
}
try {
CopyContext[] copyContexts = new CopyContext[threadSize];
CompletableFuture[] futures = new CompletableFuture[threadSize];
for (int t = 0; t < threadSize; t++) {
copyContexts[t] = actions[t].getReadyToStart().get();
futures[t] = actions[t].getFuture();
}
return new ImportContext(shardMap, futures, copyContexts, ostreams, shardCount);
} catch (InterruptedException e) {
throw new HoloClientException(ExceptionCode.INTERNAL_ERROR, "interrupt", e);
} catch (ExecutionException e) {
Throwable cause = e.getCause();
if (cause instanceof HoloClientException) {
throw (HoloClientException) cause;
} else {
throw new HoloClientException(ExceptionCode.INTERNAL_ERROR, "", cause);
}
}
}
public BinlogShardGroupReader binlogSubscribe(Subscribe subscribe) throws HoloClientException {
ensurePoolOpen();
TableSchemaSupplier supplier = new TableSchemaSupplier() {
@Override
public TableSchema apply() throws HoloClientException {
return HoloClient.this.getTableSchema(subscribe.getTableName(), true);
}
};
TableSchema schema = supplier.apply();
int shardCount = Command.getShardCount(this, schema);
if (subscribe.getSlotName() != null && !Command.getSlotNames(this, schema).contains(subscribe.getSlotName())) {
throw new HoloClientException(ExceptionCode.INVALID_REQUEST, String.format("The table %s has no slot named %s", schema.getTableNameObj().getFullName(), subscribe.getSlotName()));
}
if (subscribe.getSlotName() == null) {
HoloVersion holoVersion = Command.getHoloVersion(this);
if (holoVersion.compareTo(new HoloVersion("2.1.0")) < 0) {
throw new HoloClientException(ExceptionCode.INVALID_REQUEST, String.format("For hologres instance version lower than r2.1.0, need to provide slotName to subscribe binlog. your version is %s", holoVersion));
}
}
Map offsetMap = subscribe.getOffsetMap();
if (null != offsetMap) {
for (Integer shardId : offsetMap.keySet()) {
if (shardId < 0 || shardId >= shardCount) {
throw new HoloClientException(ExceptionCode.INVALID_REQUEST, String.format("invalid shard id [%s] for table %s", shardId, subscribe.getTableName()));
}
}
} else {
offsetMap = new HashMap<>();
for (int i = 0; i < shardCount; i++) {
offsetMap.put(i, new BinlogOffset().setTimestamp(subscribe.getBinlogReadStartTime()));
}
}
BinlogShardGroupReader reader = null;
try {
AtomicBoolean started = new AtomicBoolean(true);
Map committerMap = new HashMap<>();
reader = new BinlogShardGroupReader(config, subscribe, offsetMap.size(), committerMap, started);
for (Map.Entry entry : offsetMap.entrySet()) {
BlockingQueue, Long>> queue = new ArrayBlockingQueue<>(1);
Committer committer = new Committer(queue);
committerMap.put(entry.getKey(), committer);
BinlogAction action = new BinlogAction(subscribe.getTableName(), subscribe.getSlotName(), entry.getKey(), entry.getValue().getSequence(), entry.getValue().getStartTimeText(), reader.getCollector(), supplier, queue);
if (this.useFixedFe) {
reader.addThread(fixedPool.submitOneShotAction(started, entry.getKey(), action));
} else {
reader.addThread(pool.submitOneShotAction(started, entry.getKey(), action));
}
}
} catch (HoloClientException e) {
if (null != reader) {
reader.close();
}
throw e;
}
return reader;
}
public void flush() throws HoloClientException {
ensurePoolOpen();
collector.flush(false);
}
public boolean isAsyncCommit() {
return asyncCommit;
}
public void setAsyncCommit(boolean asyncCommit) {
this.asyncCommit = asyncCommit;
}
private void closeInternal() throws HoloClientException{
HoloClientException exception = null;
if (pool != null && pool.isRegister(this)) {
try {
tryThrowException();
flush();
} catch (HoloClientException e) {
LOGGER.error("fail when close", e);
exception = e;
}
pool.unregister(this);
if (isEmbeddedPool) {
pool.close();
}
}
if (fixedPool != null && fixedPool.isRegister(this)) {
fixedPool.unregister(this);
if (isEmbeddedFixedPool) {
fixedPool.close();
}
}
if (null != exception) {
throw exception;
}
}
@Override
public void close() {
try {
closeInternal();
} catch (HoloClientException e) {
throw new RuntimeException(e);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy