org.apache.omid.transaction.HBaseTransactionManager Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of phoenix-server-hbase-2.6
Show all versions of phoenix-server-hbase-2.6
Phoenix HBase Server Side JAR
The newest version!
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.apache.omid.transaction;
import java.io.IOException;
import java.util.HashSet;
import java.util.Map;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.Executors;
import org.apache.hadoop.hbase.client.Connection;
import org.apache.hadoop.hbase.client.ConnectionFactory;
import org.apache.hadoop.hbase.client.Get;
import org.apache.hadoop.hbase.client.Result;
import org.apache.hadoop.hbase.util.Bytes;
import org.apache.omid.committable.CommitTable;
import org.apache.omid.committable.hbase.HBaseCommitTable;
import org.apache.omid.committable.hbase.HBaseCommitTableConfig;
import org.apache.omid.tools.hbase.HBaseLogin;
import org.apache.omid.tso.client.CellId;
import org.apache.omid.tso.client.OmidClientConfiguration.ConflictDetectionLevel;
import org.apache.omid.tso.client.TSOClient;
import org.apache.omid.tso.client.TSOProtocol;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.apache.phoenix.thirdparty.com.google.common.base.Optional;
import org.apache.phoenix.thirdparty.com.google.common.util.concurrent.ListeningExecutorService;
import org.apache.phoenix.thirdparty.com.google.common.util.concurrent.MoreExecutors;
import org.apache.phoenix.thirdparty.com.google.common.util.concurrent.ThreadFactoryBuilder;
public class HBaseTransactionManager extends AbstractTransactionManager implements HBaseTransactionClient {
private static final Logger LOG = LoggerFactory.getLogger(HBaseTransactionManager.class);
private final Connection connection;
private static class HBaseTransactionFactory implements TransactionFactory {
@Override
public HBaseTransaction createTransaction(long transactionId, long epoch, AbstractTransactionManager tm) {
return new HBaseTransaction(transactionId, epoch, new HashSet(), new HashSet(),
tm, tm.isLowLatency());
}
}
// ----------------------------------------------------------------------------------------------------------------
// Construction
// ----------------------------------------------------------------------------------------------------------------
public static TransactionManager newInstance() throws IOException, InterruptedException {
return newInstance(new HBaseOmidClientConfiguration());
}
public static TransactionManager newInstance(HBaseOmidClientConfiguration configuration)
throws IOException, InterruptedException {
//Logging in to Secure HBase if required
HBaseLogin.loginIfNeeded(configuration);
return builder(configuration).build();
}
public static class Builder {
// Required parameters
private final HBaseOmidClientConfiguration hbaseOmidClientConf;
// Optional parameters - initialized to default values
private Optional tsoClient = Optional.absent();
private Optional commitTableClient = Optional.absent();
private Optional commitTableWriter = Optional.absent();
private Optional postCommitter = Optional.absent();
public Builder(HBaseOmidClientConfiguration hbaseOmidClientConf) {
this.hbaseOmidClientConf = hbaseOmidClientConf;
}
public Builder tsoClient(TSOProtocol tsoClient) {
this.tsoClient = Optional.of(tsoClient);
return this;
}
public Builder commitTableClient(CommitTable.Client client) {
this.commitTableClient = Optional.of(client);
return this;
}
public Builder commitTableWriter(CommitTable.Writer writer) {
this.commitTableWriter = Optional.of(writer);
return this;
}
Builder postCommitter(PostCommitActions postCommitter) {
this.postCommitter = Optional.of(postCommitter);
return this;
}
public HBaseTransactionManager build() throws IOException, InterruptedException {
Connection connection = ConnectionFactory.createConnection(hbaseOmidClientConf.getHBaseConfiguration());
CommitTable.Client commitTableClient = this.commitTableClient.or(buildCommitTableClient(connection)).get();
CommitTable.Writer commitTableWriter = this.commitTableWriter.or(buildCommitTableWriter(connection)).get();
PostCommitActions postCommitter = this.postCommitter.or(buildPostCommitter(commitTableClient, connection)).get();
TSOProtocol tsoClient = this.tsoClient.or(buildTSOClient()).get();
return new HBaseTransactionManager(hbaseOmidClientConf,
postCommitter,
tsoClient,
commitTableClient,
commitTableWriter,
new HBaseTransactionFactory(),
connection);
}
private Optional buildTSOClient() throws IOException, InterruptedException {
return Optional.of((TSOProtocol) TSOClient.newInstance(hbaseOmidClientConf.getOmidClientConfiguration()));
}
private Optional buildCommitTableClient(Connection connection) throws IOException {
HBaseCommitTableConfig commitTableConf = new HBaseCommitTableConfig();
commitTableConf.setTableName(hbaseOmidClientConf.getCommitTableName());
CommitTable commitTable = new HBaseCommitTable(connection, commitTableConf);
return Optional.of(commitTable.getClient());
}
private Optional buildCommitTableWriter(Connection connection) throws IOException {
HBaseCommitTableConfig commitTableConf = new HBaseCommitTableConfig();
commitTableConf.setTableName(hbaseOmidClientConf.getCommitTableName());
CommitTable commitTable = new HBaseCommitTable(connection, commitTableConf);
return Optional.of(commitTable.getWriter());
}
private Optional buildPostCommitter(CommitTable.Client commitTableClient, Connection connection) {
PostCommitActions postCommitter;
PostCommitActions syncPostCommitter = new HBaseSyncPostCommitter(hbaseOmidClientConf.getMetrics(),
commitTableClient, connection);
switch(hbaseOmidClientConf.getPostCommitMode()) {
case ASYNC:
ListeningExecutorService postCommitExecutor =
MoreExecutors.listeningDecorator(Executors.newSingleThreadExecutor(
new ThreadFactoryBuilder().setNameFormat("postCommit-%d").build()));
postCommitter = new HBaseAsyncPostCommitter(syncPostCommitter, postCommitExecutor);
break;
case SYNC:
default:
postCommitter = syncPostCommitter;
break;
}
return Optional.of(postCommitter);
}
}
public static Builder builder(HBaseOmidClientConfiguration hbaseOmidClientConf) {
return new Builder(hbaseOmidClientConf);
}
private HBaseTransactionManager(HBaseOmidClientConfiguration hBaseOmidClientConfiguration,
PostCommitActions postCommitter,
TSOProtocol tsoClient,
CommitTable.Client commitTableClient,
CommitTable.Writer commitTableWriter,
HBaseTransactionFactory hBaseTransactionFactory, Connection connection) {
super(hBaseOmidClientConfiguration.getMetrics(),
postCommitter,
tsoClient,
commitTableClient,
commitTableWriter,
hBaseTransactionFactory);
this.connection = connection;
}
// ----------------------------------------------------------------------------------------------------------------
// AbstractTransactionManager overwritten methods
// ----------------------------------------------------------------------------------------------------------------
@Override
public void closeResources() throws IOException {
connection.close();
}
@Override
public void preCommit(AbstractTransaction extends CellId> transaction) throws TransactionManagerException {
try {
// Flush all pending writes
HBaseTransaction hBaseTx = enforceHBaseTransactionAsParam(transaction);
hBaseTx.flushTables();
} catch (IOException e) {
throw new TransactionManagerException("Exception while flushing writes", e);
}
}
@Override
public void preRollback(AbstractTransaction extends CellId> transaction) throws TransactionManagerException {
try {
// Flush all pending writes
HBaseTransaction hBaseTx = enforceHBaseTransactionAsParam(transaction);
hBaseTx.flushTables();
} catch (IOException e) {
throw new TransactionManagerException("Exception while flushing writes", e);
}
}
@Override
public long getHashForTable(byte[] tableName) {
return HBaseCellId.getHasher().putBytes(tableName).hash().asLong();
}
@Override
public long getLowWatermark() throws TransactionException {
try {
return commitTableClient.readLowWatermark().get();
} catch (ExecutionException ee) {
throw new TransactionException("Error reading low watermark", ee.getCause());
} catch (InterruptedException ie) {
Thread.currentThread().interrupt();
throw new TransactionException("Interrupted reading low watermark", ie);
}
}
// ----------------------------------------------------------------------------------------------------------------
// Helper methods
// ----------------------------------------------------------------------------------------------------------------
static HBaseTransaction enforceHBaseTransactionAsParam(AbstractTransaction extends CellId> tx) {
if (tx instanceof HBaseTransaction) {
return (HBaseTransaction) tx;
} else {
throw new IllegalArgumentException(
"The transaction object passed is not an instance of HBaseTransaction");
}
}
public void setConflictDetectionLevel(ConflictDetectionLevel conflictDetectionLevel) {
tsoClient.setConflictDetectionLevel(conflictDetectionLevel);
}
public ConflictDetectionLevel getConflictDetectionLevel() {
return tsoClient.getConflictDetectionLevel();
}
static class CommitTimestampLocatorImpl implements CommitTimestampLocator {
private HBaseCellId hBaseCellId;
private final Map commitCache;
private TableAccessWrapper tableAccessWrapper;
CommitTimestampLocatorImpl(HBaseCellId hBaseCellId, Map commitCache, TableAccessWrapper tableAccessWrapper) {
this.hBaseCellId = hBaseCellId;
this.commitCache = commitCache;
this.tableAccessWrapper = tableAccessWrapper;
}
CommitTimestampLocatorImpl(HBaseCellId hBaseCellId, Map commitCache) {
this.hBaseCellId = hBaseCellId;
this.commitCache = commitCache;
this.tableAccessWrapper = null;
this.tableAccessWrapper = new HTableAccessWrapper(hBaseCellId.getTable().getHTable(), hBaseCellId.getTable().getHTable());
}
@Override
public Optional readCommitTimestampFromCache(long startTimestamp) {
if (commitCache.containsKey(startTimestamp)) {
return Optional.of(commitCache.get(startTimestamp));
}
return Optional.absent();
}
@Override
public Optional readCommitTimestampFromShadowCell(long startTimestamp) throws IOException {
Get get = new Get(hBaseCellId.getRow());
byte[] family = hBaseCellId.getFamily();
byte[] shadowCellQualifier = CellUtils.addShadowCellSuffixPrefix(hBaseCellId.getQualifier());
get.addColumn(family, shadowCellQualifier);
get.readVersions(1);
get.setTimestamp(startTimestamp);
Result result = tableAccessWrapper.get(get);
if (result.containsColumn(family, shadowCellQualifier)) {
return Optional.of(Bytes.toLong(result.getValue(family, shadowCellQualifier)));
}
return Optional.absent();
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy