HBASE-27060 Allow sharing connections between AggregationClient instances (#4566)
Signed-off-by: Duo Zhang <zhangduo@apache.org>
This commit is contained in:
parent
e8f04a8a7f
commit
5c273dacd0
|
@ -78,6 +78,7 @@ public class AggregationClient implements Closeable {
|
||||||
// TODO: This class is not used. Move to examples?
|
// TODO: This class is not used. Move to examples?
|
||||||
private static final Logger log = LoggerFactory.getLogger(AggregationClient.class);
|
private static final Logger log = LoggerFactory.getLogger(AggregationClient.class);
|
||||||
private final Connection connection;
|
private final Connection connection;
|
||||||
|
private final boolean manageConnection;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* An RpcController implementation for use here in this endpoint.
|
* An RpcController implementation for use here in this endpoint.
|
||||||
|
@ -127,25 +128,73 @@ public class AggregationClient implements Closeable {
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Constructor with Conf object
|
* Creates AggregationClient with no underlying Connection. Users of this constructor should limit
|
||||||
* @param cfg Configuration to use
|
* themselves to methods here which take a {@link Table} argument, such as
|
||||||
|
* {@link #rowCount(Table, ColumnInterpreter, Scan)}. Use of methods which instead take a
|
||||||
|
* TableName, such as {@link #rowCount(TableName, ColumnInterpreter, Scan)}, will throw an
|
||||||
|
* IOException.
|
||||||
|
*/
|
||||||
|
public AggregationClient() {
|
||||||
|
this(null, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates AggregationClient using the passed in Connection, which will be used by methods taking
|
||||||
|
* a {@link TableName} to create the necessary {@link Table} for the call. The Connection is
|
||||||
|
* externally managed by the caller and will not be closed if {@link #close()} is called. There is
|
||||||
|
* no need to call {@link #close()} for AggregationClients created this way.
|
||||||
|
* @param connection the connection to use
|
||||||
|
*/
|
||||||
|
public AggregationClient(Connection connection) {
|
||||||
|
this(connection, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates AggregationClient with internally managed Connection, which will be used by methods
|
||||||
|
* taking a {@link TableName} to create the necessary {@link Table} for the call. The Connection
|
||||||
|
* will immediately be created will be closed when {@link #close()} is called. It's important to
|
||||||
|
* call {@link #close()} when done with this AggregationClient and to otherwise treat it as a
|
||||||
|
* shared Singleton.
|
||||||
|
* @param cfg Configuration to use to create connection
|
||||||
*/
|
*/
|
||||||
public AggregationClient(Configuration cfg) {
|
public AggregationClient(Configuration cfg) {
|
||||||
|
// Create a connection on construction. Will use it making each of the calls below.
|
||||||
|
this(createConnection(cfg), true);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static Connection createConnection(Configuration cfg) {
|
||||||
try {
|
try {
|
||||||
// Create a connection on construction. Will use it making each of the calls below.
|
return ConnectionFactory.createConnection(cfg);
|
||||||
this.connection = ConnectionFactory.createConnection(cfg);
|
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
throw new RuntimeException(e);
|
throw new RuntimeException(e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private AggregationClient(Connection connection, boolean manageConnection) {
|
||||||
|
this.connection = connection;
|
||||||
|
this.manageConnection = manageConnection;
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void close() throws IOException {
|
public void close() throws IOException {
|
||||||
if (this.connection != null && !this.connection.isClosed()) {
|
if (manageConnection && this.connection != null && !this.connection.isClosed()) {
|
||||||
this.connection.close();
|
this.connection.close();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// visible for tests
|
||||||
|
boolean isClosed() {
|
||||||
|
return manageConnection && this.connection != null && this.connection.isClosed();
|
||||||
|
}
|
||||||
|
|
||||||
|
private Connection getConnection() throws IOException {
|
||||||
|
if (connection == null) {
|
||||||
|
throw new IOException(
|
||||||
|
"Connection not initialized. Use the correct constructor, or use the methods taking a Table");
|
||||||
|
}
|
||||||
|
return connection;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* It gives the maximum value of a column for a given column family for the given range. In case
|
* It gives the maximum value of a column for a given column family for the given range. In case
|
||||||
* qualifier is null, a max of all values for the given family is returned.
|
* qualifier is null, a max of all values for the given family is returned.
|
||||||
|
@ -159,7 +208,7 @@ public class AggregationClient implements Closeable {
|
||||||
public <R, S, P extends Message, Q extends Message, T extends Message> R
|
public <R, S, P extends Message, Q extends Message, T extends Message> R
|
||||||
max(final TableName tableName, final ColumnInterpreter<R, S, P, Q, T> ci, final Scan scan)
|
max(final TableName tableName, final ColumnInterpreter<R, S, P, Q, T> ci, final Scan scan)
|
||||||
throws Throwable {
|
throws Throwable {
|
||||||
try (Table table = connection.getTable(tableName)) {
|
try (Table table = getConnection().getTable(tableName)) {
|
||||||
return max(table, ci, scan);
|
return max(table, ci, scan);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -226,7 +275,7 @@ public class AggregationClient implements Closeable {
|
||||||
public <R, S, P extends Message, Q extends Message, T extends Message> R
|
public <R, S, P extends Message, Q extends Message, T extends Message> R
|
||||||
min(final TableName tableName, final ColumnInterpreter<R, S, P, Q, T> ci, final Scan scan)
|
min(final TableName tableName, final ColumnInterpreter<R, S, P, Q, T> ci, final Scan scan)
|
||||||
throws Throwable {
|
throws Throwable {
|
||||||
try (Table table = connection.getTable(tableName)) {
|
try (Table table = getConnection().getTable(tableName)) {
|
||||||
return min(table, ci, scan);
|
return min(table, ci, scan);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -298,7 +347,7 @@ public class AggregationClient implements Closeable {
|
||||||
public <R, S, P extends Message, Q extends Message, T extends Message> long
|
public <R, S, P extends Message, Q extends Message, T extends Message> long
|
||||||
rowCount(final TableName tableName, final ColumnInterpreter<R, S, P, Q, T> ci, final Scan scan)
|
rowCount(final TableName tableName, final ColumnInterpreter<R, S, P, Q, T> ci, final Scan scan)
|
||||||
throws Throwable {
|
throws Throwable {
|
||||||
try (Table table = connection.getTable(tableName)) {
|
try (Table table = getConnection().getTable(tableName)) {
|
||||||
return rowCount(table, ci, scan);
|
return rowCount(table, ci, scan);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -368,7 +417,7 @@ public class AggregationClient implements Closeable {
|
||||||
public <R, S, P extends Message, Q extends Message, T extends Message> S
|
public <R, S, P extends Message, Q extends Message, T extends Message> S
|
||||||
sum(final TableName tableName, final ColumnInterpreter<R, S, P, Q, T> ci, final Scan scan)
|
sum(final TableName tableName, final ColumnInterpreter<R, S, P, Q, T> ci, final Scan scan)
|
||||||
throws Throwable {
|
throws Throwable {
|
||||||
try (Table table = connection.getTable(tableName)) {
|
try (Table table = getConnection().getTable(tableName)) {
|
||||||
return sum(table, ci, scan);
|
return sum(table, ci, scan);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -437,7 +486,7 @@ public class AggregationClient implements Closeable {
|
||||||
private <R, S, P extends Message, Q extends Message, T extends Message> Pair<S, Long> getAvgArgs(
|
private <R, S, P extends Message, Q extends Message, T extends Message> Pair<S, Long> getAvgArgs(
|
||||||
final TableName tableName, final ColumnInterpreter<R, S, P, Q, T> ci, final Scan scan)
|
final TableName tableName, final ColumnInterpreter<R, S, P, Q, T> ci, final Scan scan)
|
||||||
throws Throwable {
|
throws Throwable {
|
||||||
try (Table table = connection.getTable(tableName)) {
|
try (Table table = getConnection().getTable(tableName)) {
|
||||||
return getAvgArgs(table, ci, scan);
|
return getAvgArgs(table, ci, scan);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -623,7 +672,7 @@ public class AggregationClient implements Closeable {
|
||||||
*/
|
*/
|
||||||
public <R, S, P extends Message, Q extends Message, T extends Message> double std(
|
public <R, S, P extends Message, Q extends Message, T extends Message> double std(
|
||||||
final TableName tableName, ColumnInterpreter<R, S, P, Q, T> ci, Scan scan) throws Throwable {
|
final TableName tableName, ColumnInterpreter<R, S, P, Q, T> ci, Scan scan) throws Throwable {
|
||||||
try (Table table = connection.getTable(tableName)) {
|
try (Table table = getConnection().getTable(tableName)) {
|
||||||
return std(table, ci, scan);
|
return std(table, ci, scan);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -727,7 +776,7 @@ public class AggregationClient implements Closeable {
|
||||||
*/
|
*/
|
||||||
public <R, S, P extends Message, Q extends Message, T extends Message> R median(
|
public <R, S, P extends Message, Q extends Message, T extends Message> R median(
|
||||||
final TableName tableName, ColumnInterpreter<R, S, P, Q, T> ci, Scan scan) throws Throwable {
|
final TableName tableName, ColumnInterpreter<R, S, P, Q, T> ci, Scan scan) throws Throwable {
|
||||||
try (Table table = connection.getTable(tableName)) {
|
try (Table table = getConnection().getTable(tableName)) {
|
||||||
return median(table, ci, scan);
|
return median(table, ci, scan);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,124 @@
|
||||||
|
/*
|
||||||
|
* 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.hadoop.hbase.client.coprocessor;
|
||||||
|
|
||||||
|
import static org.junit.Assert.assertFalse;
|
||||||
|
import static org.junit.Assert.assertTrue;
|
||||||
|
import static org.junit.Assert.fail;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import org.apache.hadoop.conf.Configuration;
|
||||||
|
import org.apache.hadoop.hbase.HBaseClassTestRule;
|
||||||
|
import org.apache.hadoop.hbase.HBaseTestingUtil;
|
||||||
|
import org.apache.hadoop.hbase.TableName;
|
||||||
|
import org.apache.hadoop.hbase.client.Connection;
|
||||||
|
import org.apache.hadoop.hbase.client.ConnectionFactory;
|
||||||
|
import org.apache.hadoop.hbase.client.Scan;
|
||||||
|
import org.apache.hadoop.hbase.client.Table;
|
||||||
|
import org.apache.hadoop.hbase.coprocessor.AggregateImplementation;
|
||||||
|
import org.apache.hadoop.hbase.coprocessor.CoprocessorHost;
|
||||||
|
import org.apache.hadoop.hbase.testclassification.CoprocessorTests;
|
||||||
|
import org.apache.hadoop.hbase.testclassification.MediumTests;
|
||||||
|
import org.apache.hadoop.hbase.util.Bytes;
|
||||||
|
import org.junit.AfterClass;
|
||||||
|
import org.junit.BeforeClass;
|
||||||
|
import org.junit.ClassRule;
|
||||||
|
import org.junit.Test;
|
||||||
|
import org.junit.experimental.categories.Category;
|
||||||
|
|
||||||
|
@Category({ MediumTests.class, CoprocessorTests.class })
|
||||||
|
public class TestAggregationClient {
|
||||||
|
|
||||||
|
@ClassRule
|
||||||
|
public static final HBaseClassTestRule CLASS_RULE =
|
||||||
|
HBaseClassTestRule.forClass(TestAggregationClient.class);
|
||||||
|
|
||||||
|
private static final HBaseTestingUtil UTIL = new HBaseTestingUtil();
|
||||||
|
|
||||||
|
private static final TableName TABLE_NAME = TableName.valueOf("TestAggregationClient");
|
||||||
|
|
||||||
|
private static final byte[] CF = Bytes.toBytes("CF");
|
||||||
|
|
||||||
|
private static Connection CONN;
|
||||||
|
|
||||||
|
private static Table TABLE;
|
||||||
|
|
||||||
|
@BeforeClass
|
||||||
|
public static void setUp() throws Exception {
|
||||||
|
Configuration conf = UTIL.getConfiguration();
|
||||||
|
conf.setStrings(CoprocessorHost.REGION_COPROCESSOR_CONF_KEY,
|
||||||
|
AggregateImplementation.class.getName());
|
||||||
|
UTIL.startMiniCluster(1);
|
||||||
|
UTIL.createTable(TABLE_NAME, CF);
|
||||||
|
CONN = ConnectionFactory.createConnection(conf);
|
||||||
|
TABLE = CONN.getTable(TABLE_NAME);
|
||||||
|
}
|
||||||
|
|
||||||
|
@AfterClass
|
||||||
|
public static void tearDown() throws Exception {
|
||||||
|
CONN.close();
|
||||||
|
UTIL.shutdownMiniCluster();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void itCreatesConnectionless() throws Throwable {
|
||||||
|
AggregationClient client = new AggregationClient();
|
||||||
|
assertFalse(client.isClosed());
|
||||||
|
|
||||||
|
try {
|
||||||
|
client.rowCount(TABLE_NAME, new LongColumnInterpreter(), new Scan());
|
||||||
|
fail("Expected IOException");
|
||||||
|
} catch (Throwable e) {
|
||||||
|
assertTrue(e instanceof IOException);
|
||||||
|
assertTrue(e.getMessage().contains("Connection not initialized"));
|
||||||
|
}
|
||||||
|
|
||||||
|
client.rowCount(TABLE, new LongColumnInterpreter(), new Scan());
|
||||||
|
|
||||||
|
client.close();
|
||||||
|
assertFalse(CONN.isClosed());
|
||||||
|
assertFalse(client.isClosed());
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void itCreatesExternalConnection() throws Throwable {
|
||||||
|
AggregationClient client = new AggregationClient(CONN);
|
||||||
|
assertFalse(client.isClosed());
|
||||||
|
|
||||||
|
client.rowCount(TABLE_NAME, new LongColumnInterpreter(), new Scan());
|
||||||
|
client.rowCount(TABLE, new LongColumnInterpreter(), new Scan());
|
||||||
|
|
||||||
|
client.close();
|
||||||
|
assertFalse(CONN.isClosed());
|
||||||
|
assertFalse(client.isClosed());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void itCreatesManagedConnection() throws Throwable {
|
||||||
|
AggregationClient client = new AggregationClient(CONN.getConfiguration());
|
||||||
|
assertFalse(client.isClosed());
|
||||||
|
|
||||||
|
client.rowCount(TABLE_NAME, new LongColumnInterpreter(), new Scan());
|
||||||
|
client.rowCount(TABLE, new LongColumnInterpreter(), new Scan());
|
||||||
|
|
||||||
|
client.close();
|
||||||
|
assertFalse(CONN.isClosed());
|
||||||
|
assertTrue(client.isClosed());
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue