Skip to content

Commit

Permalink
HBASE-25682 Add a new command to update the configuration of all RSs …
Browse files Browse the repository at this point in the history
…in a RSGroup (#3080)

* HBASE-25682 Add a new command to update the configuration of all RSs in a RSGroup

Signed-off-by: Pankaj Kumar<[email protected]>
  • Loading branch information
ZhaoBQ authored May 21, 2021
1 parent 15e8611 commit a1177b3
Show file tree
Hide file tree
Showing 13 changed files with 240 additions and 3 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -1845,6 +1845,14 @@ void deleteTableSnapshots(Pattern tableNamePattern, Pattern snapshotNamePattern)
*/
void updateConfiguration() throws IOException;

/**
* Update the configuration and trigger an online config change
* on all the regionservers in the RSGroup.
* @param groupName the group name
* @throws IOException if a remote or network exception occurs
*/
void updateConfiguration(String groupName) throws IOException;

/**
* Get the info port of the current master if one is available.
* @return master info port
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -781,6 +781,11 @@ public void updateConfiguration() throws IOException {
get(admin.updateConfiguration());
}

@Override
public void updateConfiguration(String groupName) throws IOException {
get(admin.updateConfiguration(groupName));
}

@Override
public List<SecurityCapability> getSecurityCapabilities() throws IOException {
return get(admin.getSecurityCapabilities());
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -1129,6 +1129,13 @@ default CompletableFuture<Integer> getMasterInfoPort() {
*/
CompletableFuture<Void> updateConfiguration();

/**
* Update the configuration and trigger an online config change on all the regionservers in
* the RSGroup.
* @param groupName the group name
*/
CompletableFuture<Void> updateConfiguration(String groupName);

/**
* Roll the log writer. I.e. for filesystem based write ahead logs, start writing to a new file.
* <p>
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -630,6 +630,11 @@ public CompletableFuture<Void> updateConfiguration() {
return wrap(rawAdmin.updateConfiguration());
}

@Override
public CompletableFuture<Void> updateConfiguration(String groupName) {
return wrap(rawAdmin.updateConfiguration(groupName));
}

@Override
public CompletableFuture<Void> rollWALWriter(ServerName serverName) {
return wrap(rawAdmin.rollWALWriter(serverName));
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -2905,6 +2905,42 @@ public CompletableFuture<Void> updateConfiguration() {
return future;
}

@Override
public CompletableFuture<Void> updateConfiguration(String groupName) {
CompletableFuture<Void> future = new CompletableFuture<Void>();
addListener(
getRSGroup(groupName),
(rsGroupInfo, err) -> {
if (err != null) {
future.completeExceptionally(err);
} else if (rsGroupInfo == null) {
future.completeExceptionally(
new IllegalArgumentException("Group does not exist: " + groupName));
} else {
addListener(getClusterMetrics(EnumSet.of(Option.SERVERS_NAME)), (status, err2) -> {
if (err2 != null) {
future.completeExceptionally(err2);
} else {
List<CompletableFuture<Void>> futures = new ArrayList<>();
List<ServerName> groupServers = status.getServersName().stream().filter(
s -> rsGroupInfo.containsServer(s.getAddress())).collect(Collectors.toList());
groupServers.forEach(server -> futures.add(updateConfiguration(server)));
addListener(
CompletableFuture.allOf(futures.toArray(new CompletableFuture<?>[futures.size()])),
(result, err3) -> {
if (err3 != null) {
future.completeExceptionally(err3);
} else {
future.complete(result);
}
});
}
});
}
});
return future;
}

@Override
public CompletableFuture<Void> rollWALWriter(ServerName serverName) {
return this
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -41,6 +41,7 @@
import org.apache.hadoop.hbase.ServerName;
import org.apache.hadoop.hbase.TableName;
import org.apache.hadoop.hbase.Waiter;
import org.apache.hadoop.hbase.client.AbstractTestUpdateConfiguration;
import org.apache.hadoop.hbase.client.Admin;
import org.apache.hadoop.hbase.client.RegionInfo;
import org.apache.hadoop.hbase.client.TableDescriptor;
Expand All @@ -63,7 +64,7 @@
import org.apache.hbase.thirdparty.com.google.common.collect.Maps;
import org.apache.hbase.thirdparty.com.google.common.collect.Sets;

public abstract class TestRSGroupsBase {
public abstract class TestRSGroupsBase extends AbstractTestUpdateConfiguration {
protected static final Logger LOG = LoggerFactory.getLogger(TestRSGroupsBase.class);

// shared
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,122 @@
/*
* 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.rsgroup;

import static org.junit.Assert.assertEquals;
import static org.junit.Assert.fail;

import java.util.stream.Collectors;

import org.apache.hadoop.conf.Configuration;
import org.apache.hadoop.hbase.HBaseClassTestRule;
import org.apache.hadoop.hbase.testclassification.MediumTests;
import org.apache.hadoop.hbase.util.JVMClusterUtil;
import org.junit.After;
import org.junit.AfterClass;
import org.junit.Before;
import org.junit.BeforeClass;
import org.junit.ClassRule;
import org.junit.Test;
import org.junit.experimental.categories.Category;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

@Category({MediumTests.class})
public class TestUpdateRSGroupConfiguration extends TestRSGroupsBase {
protected static final Logger LOG = LoggerFactory.getLogger(TestUpdateRSGroupConfiguration.class);

@ClassRule
public static final HBaseClassTestRule CLASS_RULE =
HBaseClassTestRule.forClass(TestUpdateRSGroupConfiguration.class);
private static final String TEST_GROUP = "test";
private static final String TEST2_GROUP = "test2";

@BeforeClass
public static void setUp() throws Exception {
setUpConfigurationFiles(TEST_UTIL);
setUpTestBeforeClass();
addResourceToRegionServerConfiguration(TEST_UTIL);
}

@AfterClass
public static void tearDown() throws Exception {
tearDownAfterClass();
}

@Before
public void beforeMethod() throws Exception {
setUpBeforeMethod();
}

@After
public void afterMethod() throws Exception {
tearDownAfterMethod();
}

@Test
public void testOnlineConfigChangeInRSGroup() throws Exception {
addGroup(TEST_GROUP, 1);
ADMIN.updateConfiguration(TEST_GROUP);
}

@Test
public void testNonexistentRSGroup() throws Exception {
try {
ADMIN.updateConfiguration(TEST2_GROUP);
fail("Group does not exist: test2");
} catch (IllegalArgumentException iae) {
// expected
}
}

@Test
public void testCustomOnlineConfigChangeInRSGroup() throws Exception {
// Check the default configuration of the RegionServers
TEST_UTIL.getMiniHBaseCluster().getRegionServerThreads().forEach(thread -> {
Configuration conf = thread.getRegionServer().getConfiguration();
assertEquals(0, conf.getInt("hbase.custom.config", 0));
});

replaceHBaseSiteXML();
RSGroupInfo testRSGroup = addGroup(TEST_GROUP, 1);
RSGroupInfo test2RSGroup = addGroup(TEST2_GROUP, 1);
ADMIN.updateConfiguration(TEST_GROUP);

// Check the configuration of the RegionServer in test rsgroup, should be update
Configuration regionServerConfiguration =
TEST_UTIL.getMiniHBaseCluster().getLiveRegionServerThreads().stream()
.map(JVMClusterUtil.RegionServerThread::getRegionServer)
.filter(regionServer ->
(regionServer.getServerName().getAddress().equals(testRSGroup.getServers().first())))
.collect(Collectors.toList()).get(0).getConfiguration();
int custom = regionServerConfiguration.getInt("hbase.custom.config", 0);
assertEquals(1000, custom);

// Check the configuration of the RegionServer in test2 rsgroup, should not be update
regionServerConfiguration =
TEST_UTIL.getMiniHBaseCluster().getLiveRegionServerThreads().stream()
.map(JVMClusterUtil.RegionServerThread::getRegionServer)
.filter(regionServer ->
(regionServer.getServerName().getAddress().equals(test2RSGroup.getServers().first())))
.collect(Collectors.toList()).get(0).getConfiguration();
custom = regionServerConfiguration.getInt("hbase.custom.config", 0);
assertEquals(0, custom);

restoreHBaseSiteXML();
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -617,6 +617,10 @@ public void updateConfiguration() throws IOException {
admin.updateConfiguration();
}

public void updateConfiguration(String groupName) throws IOException {
admin.updateConfiguration(groupName);
}

public List<SecurityCapability> getSecurityCapabilities() throws IOException {
return admin.getSecurityCapabilities();
}
Expand Down
6 changes: 6 additions & 0 deletions hbase-shell/src/main/ruby/hbase/admin.rb
Original file line number Diff line number Diff line change
Expand Up @@ -1379,6 +1379,12 @@ def update_all_config
@admin.updateConfiguration
end

#----------------------------------------------------------------------------------------------
# Updates the configuration of all the regionservers in the rsgroup.
def update_rsgroup_config(groupName)
@admin.updateConfiguration(groupName)
end

#----------------------------------------------------------------------------------------------
# Returns namespace's structure description
def describe_namespace(namespace_name)
Expand Down
1 change: 1 addition & 0 deletions hbase-shell/src/main/ruby/shell.rb
Original file line number Diff line number Diff line change
Expand Up @@ -553,6 +553,7 @@ def self.exception_handler(hide_traceback)
commands: %w[
update_config
update_all_config
update_rsgroup_config
]
)

Expand Down
37 changes: 37 additions & 0 deletions hbase-shell/src/main/ruby/shell/commands/update_rsgroup_config.rb
Original file line number Diff line number Diff line change
@@ -0,0 +1,37 @@
#
#
# 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.
#

module Shell
module Commands
class UpdateRsgroupConfig < Command
def help
<<-EOF
Reload a subset of configuration on all servers in the rsgroup. See
http://hbase.apache.org/book.html#dyn_config for more details. Here is how
you would run the command in the hbase shell:
hbase> update_rsgroup_config 'groupName'
EOF
end

def command(groupName)
admin.update_rsgroup_config(groupName)
end
end
end
end
Original file line number Diff line number Diff line change
Expand Up @@ -944,6 +944,11 @@ public void updateConfiguration() {
throw new NotImplementedException("updateConfiguration not supported in ThriftAdmin");
}

@Override
public void updateConfiguration(String groupName) {
throw new NotImplementedException("updateConfiguration not supported in ThriftAdmin");
}

@Override
public List<SecurityCapability> getSecurityCapabilities() {
throw new NotImplementedException("getSecurityCapabilities not supported in ThriftAdmin");
Expand Down
4 changes: 2 additions & 2 deletions src/main/asciidoc/_chapters/configuration.adoc
Original file line number Diff line number Diff line change
Expand Up @@ -1274,8 +1274,8 @@ The corresponding properties for port configuration are `master.rmi.registry.por
== Dynamic Configuration

It is possible to change a subset of the configuration without requiring a server restart. In the
HBase shell, the operations `update_config` and `update_all_config` will prompt a server or all
servers to reload configuration.
HBase shell, the operations `update_config`, `update_all_config` and `update_rsgroup_config`
will prompt a server, all servers or all servers in the RSGroup to reload configuration.

Only a subset of all configurations can currently be changed in the running server.
Here are those configurations:
Expand Down

0 comments on commit a1177b3

Please sign in to comment.