HBASE-26157 Expose some IA.LimitedPrivate interface in TestingHBaseCluster (#3643)

Signed-off-by: Geoffrey Jacoby <gjacoby@apache.org>
This commit is contained in:
Duo Zhang 2021-09-02 16:39:09 +08:00
parent ccd90269d2
commit 49a3e15aa9
4 changed files with 196 additions and 0 deletions

View File

@ -17,9 +17,15 @@
*/ */
package org.apache.hadoop.hbase.testing; package org.apache.hadoop.hbase.testing;
import java.util.List;
import java.util.Optional;
import java.util.concurrent.CompletableFuture; import java.util.concurrent.CompletableFuture;
import org.apache.hadoop.conf.Configuration; import org.apache.hadoop.conf.Configuration;
import org.apache.hadoop.hbase.HBaseInterfaceAudience;
import org.apache.hadoop.hbase.ServerName; import org.apache.hadoop.hbase.ServerName;
import org.apache.hadoop.hbase.client.RegionInfo;
import org.apache.hadoop.hbase.regionserver.OnlineRegions;
import org.apache.hadoop.hbase.regionserver.Region;
import org.apache.yetus.audience.InterfaceAudience; import org.apache.yetus.audience.InterfaceAudience;
/** /**
@ -114,6 +120,41 @@ public interface TestingHBaseCluster {
*/ */
void start() throws Exception; void start() throws Exception;
/**
* Get the address of active master if there is one.
*/
Optional<ServerName> getActiveMasterAddress();
/**
* Get all the backup master addresses.
*/
List<ServerName> getBackupMasterAddresses();
/**
* Get all the region server addresses.
*/
List<ServerName> getRegionServerAddresses();
/**
* Get the server side {@link Region} interface for the specific region.
* <p/>
* This is used for CPs to test something which can only be accessed at server side, such as tags.
*/
@InterfaceAudience.LimitedPrivate(HBaseInterfaceAudience.COPROC)
Optional<Region> getRegion(RegionInfo regionInfo);
/**
* Get the server side {@link OnlineRegions} interface for the specific region server.
* <p/>
* You could list the addresses of all the region server through the
* {@link #getRegionServerAddresses()} method.
* <p/>
* This is used for CPs to test something which can only be accessed at server side, such as tags.
* And also you could use the returned interface to get all regions on this region server, etc.
*/
@InterfaceAudience.LimitedPrivate(HBaseInterfaceAudience.COPROC)
Optional<OnlineRegions> getOnlineRegionsInterface(ServerName serverName);
/** /**
* Return whether the cluster is running. * Return whether the cluster is running.
* <p/> * <p/>

View File

@ -18,13 +18,20 @@
package org.apache.hadoop.hbase.testing; package org.apache.hadoop.hbase.testing;
import java.util.List; import java.util.List;
import java.util.Optional;
import java.util.concurrent.CompletableFuture; import java.util.concurrent.CompletableFuture;
import java.util.concurrent.ExecutorService; import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors; import java.util.concurrent.Executors;
import java.util.stream.Collectors;
import org.apache.hadoop.conf.Configuration; import org.apache.hadoop.conf.Configuration;
import org.apache.hadoop.hbase.HBaseTestingUtility; import org.apache.hadoop.hbase.HBaseTestingUtility;
import org.apache.hadoop.hbase.ServerName; import org.apache.hadoop.hbase.ServerName;
import org.apache.hadoop.hbase.StartMiniClusterOption; import org.apache.hadoop.hbase.StartMiniClusterOption;
import org.apache.hadoop.hbase.client.RegionInfo;
import org.apache.hadoop.hbase.master.HMaster;
import org.apache.hadoop.hbase.regionserver.HRegion;
import org.apache.hadoop.hbase.regionserver.OnlineRegions;
import org.apache.hadoop.hbase.regionserver.Region;
import org.apache.hadoop.hbase.util.JVMClusterUtil.MasterThread; import org.apache.hadoop.hbase.util.JVMClusterUtil.MasterThread;
import org.apache.hadoop.hbase.util.JVMClusterUtil.RegionServerThread; import org.apache.hadoop.hbase.util.JVMClusterUtil.RegionServerThread;
import org.apache.yetus.audience.InterfaceAudience; import org.apache.yetus.audience.InterfaceAudience;
@ -172,4 +179,38 @@ class TestingHBaseClusterImpl implements TestingHBaseCluster {
public void startRegionServer(String hostname, int port) throws Exception { public void startRegionServer(String hostname, int port) throws Exception {
util.getMiniHBaseCluster().startRegionServer(hostname, port); util.getMiniHBaseCluster().startRegionServer(hostname, port);
} }
@Override
public Optional<ServerName> getActiveMasterAddress() {
return Optional.ofNullable(util.getMiniHBaseCluster().getMaster()).map(HMaster::getServerName);
}
@Override
public List<ServerName> getBackupMasterAddresses() {
return util.getMiniHBaseCluster().getMasterThreads().stream().map(MasterThread::getMaster)
.filter(m -> !m.isActiveMaster()).map(HMaster::getServerName).collect(Collectors.toList());
}
@Override
public List<ServerName> getRegionServerAddresses() {
return util.getMiniHBaseCluster().getRegionServerThreads().stream()
.map(t -> t.getRegionServer().getServerName()).collect(Collectors.toList());
}
@Override
public Optional<Region> getRegion(RegionInfo regionInfo) {
for (RegionServerThread t : util.getMiniHBaseCluster().getRegionServerThreads()) {
for (HRegion region : t.getRegionServer().getRegions()) {
if (region.getRegionInfo().equals(regionInfo)) {
return Optional.of(region);
}
}
}
return Optional.empty();
}
@Override
public Optional<OnlineRegions> getOnlineRegionsInterface(ServerName serverName) {
return Optional.ofNullable(util.getMiniHBaseCluster().getRegionServer(serverName));
}
} }

View File

@ -144,4 +144,11 @@ public class TestTestingHBaseCluster {
CLUSTER.startRegionServer(DNS.getHostname(CLUSTER.getConf(), ServerType.REGIONSERVER), 0); CLUSTER.startRegionServer(DNS.getHostname(CLUSTER.getConf(), ServerType.REGIONSERVER), 0);
Waiter.waitFor(CLUSTER.getConf(), 30000, () -> admin.getRegionServers().size() == 4); Waiter.waitFor(CLUSTER.getConf(), 30000, () -> admin.getRegionServers().size() == 4);
} }
@Test
public void testGetAddresses() throws Exception {
assertTrue(CLUSTER.getActiveMasterAddress().isPresent());
assertEquals(1, CLUSTER.getBackupMasterAddresses().size());
assertEquals(3, CLUSTER.getRegionServerAddresses().size());
}
} }

View File

@ -0,0 +1,107 @@
/**
* 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.testing;
import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertFalse;
import static org.junit.Assert.assertSame;
import java.io.IOException;
import java.util.List;
import org.apache.hadoop.hbase.HBaseClassTestRule;
import org.apache.hadoop.hbase.ServerName;
import org.apache.hadoop.hbase.TableName;
import org.apache.hadoop.hbase.client.Admin;
import org.apache.hadoop.hbase.client.ColumnFamilyDescriptorBuilder;
import org.apache.hadoop.hbase.client.Connection;
import org.apache.hadoop.hbase.client.ConnectionFactory;
import org.apache.hadoop.hbase.client.RegionInfo;
import org.apache.hadoop.hbase.client.RegionInfoBuilder;
import org.apache.hadoop.hbase.client.RegionLocator;
import org.apache.hadoop.hbase.client.TableDescriptorBuilder;
import org.apache.hadoop.hbase.regionserver.OnlineRegions;
import org.apache.hadoop.hbase.regionserver.Region;
import org.apache.hadoop.hbase.testclassification.LargeTests;
import org.apache.hadoop.hbase.testclassification.MiscTests;
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;
import org.apache.hbase.thirdparty.com.google.common.io.Closeables;
@Category({ MiscTests.class, LargeTests.class })
public class TestTestingHBaseClusterImplForCPs {
@ClassRule
public static final HBaseClassTestRule CLASS_RULE =
HBaseClassTestRule.forClass(TestTestingHBaseClusterImplForCPs.class);
private static TestingHBaseCluster CLUSTER;
private static TableName NAME = TableName.valueOf("test");
private static byte[] CF = Bytes.toBytes("cf");
private static Connection CONN;
private static Admin ADMIN;
@BeforeClass
public static void setUpBeforeClass() throws Exception {
CLUSTER = TestingHBaseCluster.create(TestingHBaseClusterOption.builder().numMasters(2)
.numRegionServers(3).numDataNodes(3).build());
CLUSTER.start();
CONN = ConnectionFactory.createConnection(CLUSTER.getConf());
ADMIN = CONN.getAdmin();
ADMIN.createTable(TableDescriptorBuilder.newBuilder(NAME)
.setColumnFamily(ColumnFamilyDescriptorBuilder.of(CF)).build());
ADMIN.balancerSwitch(false, true);
}
@AfterClass
public static void tearDownAfterClass() throws Exception {
Closeables.close(ADMIN, true);
Closeables.close(CONN, true);
if (CLUSTER.isClusterRunning()) {
CLUSTER.stop();
}
}
@Test
public void testGetRegion() throws IOException {
List<RegionInfo> infos = ADMIN.getRegions(NAME);
assertEquals(1, infos.size());
RegionInfo info = infos.get(0);
Region region = CLUSTER.getRegion(info).get();
ServerName loc;
try (RegionLocator locator = CONN.getRegionLocator(NAME)) {
loc = locator.getRegionLocation(info.getStartKey()).getServerName();
}
OnlineRegions onlineRegionsInterface = CLUSTER.getOnlineRegionsInterface(loc).get();
List<? extends Region> regions = onlineRegionsInterface.getRegions(NAME);
assertEquals(1, regions.size());
assertSame(region, regions.get(0));
assertFalse(CLUSTER
.getRegion(RegionInfoBuilder.newBuilder(TableName.valueOf("whatever")).build()).isPresent());
assertFalse(CLUSTER.getOnlineRegionsInterface(ServerName.valueOf("whatever,1,1")).isPresent());
}
}