From f374357cc31b1e9be7ddc8b608a756382bb92806 Mon Sep 17 00:00:00 2001
From: Duo Zhang
Date: Fri, 23 Apr 2021 15:20:27 +0800
Subject: [PATCH] HBASE-25802 Miscellaneous style improvements for load
balancer related classes (#3192)
Signed-off-by: Yulin Niu
---
.../master/balancer/BaseLoadBalancer.java | 65 ++++-----
.../master/balancer/CandidateGenerator.java | 9 +-
.../balancer/FavoredStochasticBalancer.java | 9 +-
.../balancer/RandomCandidateGenerator.java | 34 +++++
.../RegionReplicaCandidateGenerator.java | 6 +-
.../RegionReplicaRackCandidateGenerator.java | 54 +++++++
.../master/balancer/SimpleLoadBalancer.java | 133 +++++++++++-------
.../balancer/StochasticLoadBalancer.java | 59 +-------
.../master/balancer/TestBaseLoadBalancer.java | 39 ++---
...ochasticLoadBalancerHeterogeneousCost.java | 2 +-
10 files changed, 233 insertions(+), 177 deletions(-)
create mode 100644 hbase-server/src/main/java/org/apache/hadoop/hbase/master/balancer/RandomCandidateGenerator.java
create mode 100644 hbase-server/src/main/java/org/apache/hadoop/hbase/master/balancer/RegionReplicaRackCandidateGenerator.java
diff --git a/hbase-server/src/main/java/org/apache/hadoop/hbase/master/balancer/BaseLoadBalancer.java b/hbase-server/src/main/java/org/apache/hadoop/hbase/master/balancer/BaseLoadBalancer.java
index 9840a0e223d..f5c462f1150 100644
--- a/hbase-server/src/main/java/org/apache/hadoop/hbase/master/balancer/BaseLoadBalancer.java
+++ b/hbase-server/src/main/java/org/apache/hadoop/hbase/master/balancer/BaseLoadBalancer.java
@@ -35,7 +35,6 @@ import java.util.function.Predicate;
import java.util.stream.Collectors;
import org.apache.hadoop.conf.Configuration;
import org.apache.hadoop.hbase.ClusterMetrics;
-import org.apache.hadoop.hbase.HBaseConfiguration;
import org.apache.hadoop.hbase.HBaseIOException;
import org.apache.hadoop.hbase.HConstants;
import org.apache.hadoop.hbase.ServerMetrics;
@@ -66,6 +65,8 @@ import org.apache.hbase.thirdparty.com.google.common.collect.Sets;
@InterfaceAudience.Private
public abstract class BaseLoadBalancer implements LoadBalancer {
+ private static final Logger LOG = LoggerFactory.getLogger(BaseLoadBalancer.class);
+
public static final String BALANCER_DECISION_BUFFER_ENABLED =
"hbase.master.balancer.decision.buffer.enabled";
public static final boolean DEFAULT_BALANCER_DECISION_BUFFER_ENABLED = false;
@@ -73,46 +74,19 @@ public abstract class BaseLoadBalancer implements LoadBalancer {
protected static final int MIN_SERVER_BALANCE = 2;
private volatile boolean stopped = false;
- static final List EMPTY_REGION_LIST = Collections.emptyList();
-
- static final Predicate IDLE_SERVER_PREDICATOR
+ private static final Predicate IDLE_SERVER_PREDICATOR
= load -> load.getRegionMetrics().isEmpty();
protected RegionLocationFinder regionFinder;
protected boolean useRegionFinder;
protected boolean isByTable = false;
- /**
- * The constructor that uses the basic MetricsBalancer
- */
- protected BaseLoadBalancer() {
- metricsBalancer = new MetricsBalancer();
- createRegionFinder();
- }
-
- /**
- * This Constructor accepts an instance of MetricsBalancer,
- * which will be used instead of creating a new one
- */
- protected BaseLoadBalancer(MetricsBalancer metricsBalancer) {
- this.metricsBalancer = (metricsBalancer != null) ? metricsBalancer : new MetricsBalancer();
- createRegionFinder();
- }
-
- private void createRegionFinder() {
- useRegionFinder = config.getBoolean("hbase.master.balancer.uselocality", true);
- if (useRegionFinder) {
- regionFinder = new RegionLocationFinder();
- }
- }
-
// slop for regions
protected float slop;
// overallSlop to control simpleLoadBalancer's cluster level threshold
protected float overallSlop;
- protected Configuration config = HBaseConfiguration.create();
+ protected Configuration config;
protected RackManager rackManager;
- static final Logger LOG = LoggerFactory.getLogger(BaseLoadBalancer.class);
protected MetricsBalancer metricsBalancer = null;
protected ClusterMetrics clusterStatus = null;
protected ServerName masterServerName;
@@ -125,6 +99,21 @@ public abstract class BaseLoadBalancer implements LoadBalancer {
@Deprecated
protected boolean onlySystemTablesOnMaster;
+ /**
+ * The constructor that uses the basic MetricsBalancer
+ */
+ protected BaseLoadBalancer() {
+ this(null);
+ }
+
+ /**
+ * This Constructor accepts an instance of MetricsBalancer,
+ * which will be used instead of creating a new one
+ */
+ protected BaseLoadBalancer(MetricsBalancer metricsBalancer) {
+ this.metricsBalancer = (metricsBalancer != null) ? metricsBalancer : new MetricsBalancer();
+ }
+
@Override
public void setConf(Configuration conf) {
this.config = conf;
@@ -144,7 +133,9 @@ public abstract class BaseLoadBalancer implements LoadBalancer {
this.onlySystemTablesOnMaster = LoadBalancer.isSystemTablesOnlyOnMaster(this.config);
this.rackManager = new RackManager(getConf());
+ useRegionFinder = config.getBoolean("hbase.master.balancer.uselocality", true);
if (useRegionFinder) {
+ regionFinder = new RegionLocationFinder();
regionFinder.setConf(conf);
}
this.isByTable = conf.getBoolean(HConstants.HBASE_MASTER_LOADBALANCE_BYTABLE, isByTable);
@@ -295,7 +286,9 @@ public abstract class BaseLoadBalancer implements LoadBalancer {
}
return false;
}
- if(areSomeRegionReplicasColocated(c)) return true;
+ if (areSomeRegionReplicasColocated(c)) {
+ return true;
+ }
if(idleRegionServerExist(c)) {
return true;
}
@@ -348,10 +341,10 @@ public abstract class BaseLoadBalancer implements LoadBalancer {
/**
* Generates a bulk assignment plan to be used on cluster startup using a
* simple round-robin assignment.
- *
+ *
* Takes a list of all the regions and all the servers in the cluster and
* returns a map of each server to the regions that it should be assigned.
- *
+ *
* Currently implemented as a round-robin assignment. Same invariant as load
* balancing, all servers holding floor(avg) or ceiling(avg).
*
@@ -407,7 +400,7 @@ public abstract class BaseLoadBalancer implements LoadBalancer {
return assignments;
}
- protected BalancerClusterState createCluster(List servers,
+ private BalancerClusterState createCluster(List servers,
Collection regions) throws HBaseIOException {
boolean hasRegionReplica = false;
try {
@@ -439,7 +432,7 @@ public abstract class BaseLoadBalancer implements LoadBalancer {
for (ServerName server : servers) {
if (!clusterState.containsKey(server)) {
- clusterState.put(server, EMPTY_REGION_LIST);
+ clusterState.put(server, Collections.emptyList());
}
}
return new BalancerClusterState(regions, clusterState, null, this.regionFinder,
@@ -758,7 +751,7 @@ public abstract class BaseLoadBalancer implements LoadBalancer {
}
}
- protected Map> getRegionAssignmentsByServer(
+ private Map> getRegionAssignmentsByServer(
Collection regions) {
if (this.services != null && this.services.getAssignmentManager() != null) {
return this.services.getAssignmentManager().getSnapShotOfAssignment(regions);
diff --git a/hbase-server/src/main/java/org/apache/hadoop/hbase/master/balancer/CandidateGenerator.java b/hbase-server/src/main/java/org/apache/hadoop/hbase/master/balancer/CandidateGenerator.java
index 06b5623d1b5..cf43269eb45 100644
--- a/hbase-server/src/main/java/org/apache/hadoop/hbase/master/balancer/CandidateGenerator.java
+++ b/hbase-server/src/main/java/org/apache/hadoop/hbase/master/balancer/CandidateGenerator.java
@@ -19,6 +19,7 @@
package org.apache.hadoop.hbase.master.balancer;
import java.util.Map;
+import java.util.concurrent.ThreadLocalRandom;
import org.apache.hadoop.hbase.client.RegionInfo;
import org.apache.yetus.audience.InterfaceAudience;
@@ -46,11 +47,11 @@ abstract class CandidateGenerator {
double chanceOfNoSwap) {
// Check to see if this is just a move.
if (cluster.regionsPerServer[server].length == 0
- || StochasticLoadBalancer.RANDOM.nextFloat() < chanceOfNoSwap) {
+ || ThreadLocalRandom.current().nextFloat() < chanceOfNoSwap) {
// signal a move only.
return -1;
}
- int rand = StochasticLoadBalancer.RANDOM.nextInt(cluster.regionsPerServer[server].length);
+ int rand = ThreadLocalRandom.current().nextInt(cluster.regionsPerServer[server].length);
return cluster.regionsPerServer[server][rand];
}
@@ -59,7 +60,7 @@ abstract class CandidateGenerator {
return -1;
}
- return StochasticLoadBalancer.RANDOM.nextInt(cluster.numServers);
+ return ThreadLocalRandom.current().nextInt(cluster.numServers);
}
int pickRandomRack(BalancerClusterState cluster) {
@@ -67,7 +68,7 @@ abstract class CandidateGenerator {
return -1;
}
- return StochasticLoadBalancer.RANDOM.nextInt(cluster.numRacks);
+ return ThreadLocalRandom.current().nextInt(cluster.numRacks);
}
int pickOtherRandomServer(BalancerClusterState cluster, int serverIndex) {
diff --git a/hbase-server/src/main/java/org/apache/hadoop/hbase/master/balancer/FavoredStochasticBalancer.java b/hbase-server/src/main/java/org/apache/hadoop/hbase/master/balancer/FavoredStochasticBalancer.java
index 8bf0fa54239..df047dc6f2f 100644
--- a/hbase-server/src/main/java/org/apache/hadoop/hbase/master/balancer/FavoredStochasticBalancer.java
+++ b/hbase-server/src/main/java/org/apache/hadoop/hbase/master/balancer/FavoredStochasticBalancer.java
@@ -32,7 +32,7 @@ import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
import java.util.Set;
-
+import java.util.concurrent.ThreadLocalRandom;
import org.apache.hadoop.hbase.HBaseIOException;
import org.apache.hadoop.hbase.ServerMetrics;
import org.apache.hadoop.hbase.ServerName;
@@ -290,7 +290,7 @@ public class FavoredStochasticBalancer extends StochasticLoadBalancer implements
}
} else {
// We don't have one/more load, lets just choose a random node
- s = RANDOM.nextBoolean() ? secondaryHost : tertiaryHost;
+ s = ThreadLocalRandom.current().nextBoolean() ? secondaryHost : tertiaryHost;
}
addRegionToMap(assignmentMapForFavoredNodes, region, s);
} else if (secondaryHost != null) {
@@ -348,7 +348,7 @@ public class FavoredStochasticBalancer extends StochasticLoadBalancer implements
List onlineServers = getOnlineFavoredNodes(servers, favoredNodes);
if (onlineServers.size() > 0) {
- destination = onlineServers.get(RANDOM.nextInt(onlineServers.size()));
+ destination = onlineServers.get(ThreadLocalRandom.current().nextInt(onlineServers.size()));
}
boolean alwaysAssign = getConf().getBoolean(FAVORED_ALWAYS_ASSIGN_REGIONS, true);
@@ -433,7 +433,8 @@ public class FavoredStochasticBalancer extends StochasticLoadBalancer implements
if (FavoredNodesPlan.getFavoredServerPosition(favoredNodes, sn) != null) {
addRegionToMap(assignmentMap, hri, sn);
} else {
- ServerName destination = onlineFN.get(RANDOM.nextInt(onlineFN.size()));
+ ServerName destination =
+ onlineFN.get(ThreadLocalRandom.current().nextInt(onlineFN.size()));
LOG.warn("Region: " + hri + " not hosted on favored nodes: " + favoredNodes
+ " current: " + sn + " moving to: " + destination);
addRegionToMap(assignmentMap, hri, destination);
diff --git a/hbase-server/src/main/java/org/apache/hadoop/hbase/master/balancer/RandomCandidateGenerator.java b/hbase-server/src/main/java/org/apache/hadoop/hbase/master/balancer/RandomCandidateGenerator.java
new file mode 100644
index 00000000000..c2cd998524f
--- /dev/null
+++ b/hbase-server/src/main/java/org/apache/hadoop/hbase/master/balancer/RandomCandidateGenerator.java
@@ -0,0 +1,34 @@
+/*
+ * 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.master.balancer;
+
+import org.apache.yetus.audience.InterfaceAudience;
+
+@InterfaceAudience.Private
+class RandomCandidateGenerator extends CandidateGenerator {
+
+ @Override
+ BalanceAction generate(BalancerClusterState cluster) {
+ int thisServer = pickRandomServer(cluster);
+
+ // Pick the other server
+ int otherServer = pickOtherRandomServer(cluster, thisServer);
+
+ return pickRandomRegions(cluster, thisServer, otherServer);
+ }
+}
\ No newline at end of file
diff --git a/hbase-server/src/main/java/org/apache/hadoop/hbase/master/balancer/RegionReplicaCandidateGenerator.java b/hbase-server/src/main/java/org/apache/hadoop/hbase/master/balancer/RegionReplicaCandidateGenerator.java
index 4badd4d2902..f3b69554aca 100644
--- a/hbase-server/src/main/java/org/apache/hadoop/hbase/master/balancer/RegionReplicaCandidateGenerator.java
+++ b/hbase-server/src/main/java/org/apache/hadoop/hbase/master/balancer/RegionReplicaCandidateGenerator.java
@@ -18,6 +18,7 @@
package org.apache.hadoop.hbase.master.balancer;
+import java.util.concurrent.ThreadLocalRandom;
import org.apache.yetus.audience.InterfaceAudience;
/**
@@ -27,8 +28,7 @@ import org.apache.yetus.audience.InterfaceAudience;
@InterfaceAudience.Private
class RegionReplicaCandidateGenerator extends CandidateGenerator {
- StochasticLoadBalancer.RandomCandidateGenerator randomGenerator =
- new StochasticLoadBalancer.RandomCandidateGenerator();
+ protected final RandomCandidateGenerator randomGenerator = new RandomCandidateGenerator();
/**
* Randomly select one regionIndex out of all region replicas co-hosted in the same group
@@ -56,7 +56,7 @@ class RegionReplicaCandidateGenerator extends CandidateGenerator {
int numReplicas = j - currentPrimaryIndex;
if (numReplicas > 1) { // means consecutive primaries, indicating co-location
// decide to select this primary region id or not
- double currentRandom = StochasticLoadBalancer.RANDOM.nextDouble();
+ double currentRandom = ThreadLocalRandom.current().nextDouble();
// we don't know how many region replicas are co-hosted, we will randomly select one
// using reservoir sampling (http://gregable.com/2007/10/reservoir-sampling.html)
if (currentRandom > currentLargestRandom) {
diff --git a/hbase-server/src/main/java/org/apache/hadoop/hbase/master/balancer/RegionReplicaRackCandidateGenerator.java b/hbase-server/src/main/java/org/apache/hadoop/hbase/master/balancer/RegionReplicaRackCandidateGenerator.java
new file mode 100644
index 00000000000..a67dc1dbef3
--- /dev/null
+++ b/hbase-server/src/main/java/org/apache/hadoop/hbase/master/balancer/RegionReplicaRackCandidateGenerator.java
@@ -0,0 +1,54 @@
+/*
+ * 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.master.balancer;
+
+import java.util.concurrent.ThreadLocalRandom;
+import org.apache.yetus.audience.InterfaceAudience;
+
+/**
+ * Generates candidates which moves the replicas out of the rack for co-hosted region replicas in
+ * the same rack
+ */
+@InterfaceAudience.Private
+class RegionReplicaRackCandidateGenerator extends RegionReplicaCandidateGenerator {
+
+ @Override
+ BalanceAction generate(BalancerClusterState cluster) {
+ int rackIndex = pickRandomRack(cluster);
+ if (cluster.numRacks <= 1 || rackIndex == -1) {
+ return super.generate(cluster);
+ }
+
+ int regionIndex = selectCoHostedRegionPerGroup(cluster.primariesOfRegionsPerRack[rackIndex],
+ cluster.regionsPerRack[rackIndex], cluster.regionIndexToPrimaryIndex);
+
+ // if there are no pairs of region replicas co-hosted, default to random generator
+ if (regionIndex == -1) {
+ // default to randompicker
+ return randomGenerator.generate(cluster);
+ }
+
+ int serverIndex = cluster.regionIndexToServerIndex[regionIndex];
+ int toRackIndex = pickOtherRandomRack(cluster, rackIndex);
+
+ int rand = ThreadLocalRandom.current().nextInt(cluster.serversPerRack[toRackIndex].length);
+ int toServerIndex = cluster.serversPerRack[toRackIndex][rand];
+ int toRegionIndex = pickRandomRegion(cluster, toServerIndex, 0.9f);
+ return getAction(serverIndex, regionIndex, toServerIndex, toRegionIndex);
+ }
+}
\ No newline at end of file
diff --git a/hbase-server/src/main/java/org/apache/hadoop/hbase/master/balancer/SimpleLoadBalancer.java b/hbase-server/src/main/java/org/apache/hadoop/hbase/master/balancer/SimpleLoadBalancer.java
index a0b61391592..f25585493d7 100644
--- a/hbase-server/src/main/java/org/apache/hadoop/hbase/master/balancer/SimpleLoadBalancer.java
+++ b/hbase-server/src/main/java/org/apache/hadoop/hbase/master/balancer/SimpleLoadBalancer.java
@@ -25,7 +25,6 @@ import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.NavigableMap;
-import java.util.Random;
import java.util.TreeMap;
import org.apache.hadoop.conf.Configuration;
@@ -41,22 +40,20 @@ import org.slf4j.LoggerFactory;
import org.apache.hbase.thirdparty.com.google.common.collect.MinMaxPriorityQueue;
/**
- * Makes decisions about the placement and movement of Regions across
- * RegionServers.
- *
- * Cluster-wide load balancing will occur only when there are no regions in
- * transition and according to a fixed period of a time using {@link #balanceCluster(Map)}.
- *
- *
On cluster startup, bulk assignment can be used to determine
- * locations for all Regions in a cluster.
- *
- *
This classes produces plans for the
+ * Makes decisions about the placement and movement of Regions across RegionServers.
+ *
+ * Cluster-wide load balancing will occur only when there are no regions in transition and according
+ * to a fixed period of a time using {@link #balanceCluster(Map)}.
+ *
+ * On cluster startup, bulk assignment can be used to determine locations for all Regions in a
+ * cluster.
+ *
+ * This classes produces plans for the
* {@link org.apache.hadoop.hbase.master.assignment.AssignmentManager} to execute.
*/
@InterfaceAudience.LimitedPrivate(HBaseInterfaceAudience.CONFIG)
public class SimpleLoadBalancer extends BaseLoadBalancer {
private static final Logger LOG = LoggerFactory.getLogger(SimpleLoadBalancer.class);
- private static final Random RANDOM = new Random(System.currentTimeMillis());
private RegionInfoComparator riComparator = new RegionInfoComparator();
private RegionPlan.RegionPlanComparator rpComparator = new RegionPlan.RegionPlanComparator();
@@ -66,12 +63,12 @@ public class SimpleLoadBalancer extends BaseLoadBalancer {
/**
* Stores additional per-server information about the regions added/removed
* during the run of the balancing algorithm.
- *
+ *
* For servers that shed regions, we need to track which regions we have already
* shed. nextRegionForUnload contains the index in the list of regions on
* the server that is the next to be shed.
*/
- static class BalanceInfo {
+ private static final class BalanceInfo {
private int nextRegionForUnload;
private int numRegionsAdded;
@@ -139,8 +136,9 @@ public class SimpleLoadBalancer extends BaseLoadBalancer {
originSlop, slop, originOverallSlop, overallSlop);
}
- private void setLoad(List slList, int i, int loadChange){
- ServerAndLoad newsl = new ServerAndLoad(slList.get(i).getServerName(),slList.get(i).getLoad() + loadChange);
+ private void setLoad(List slList, int i, int loadChange) {
+ ServerAndLoad newsl =
+ new ServerAndLoad(slList.get(i).getServerName(), slList.get(i).getLoad() + loadChange);
slList.set(i, newsl);
}
@@ -328,7 +326,9 @@ public class SimpleLoadBalancer extends BaseLoadBalancer {
&& masterServerName.equals(sal.getServerName())) continue;
regionsToMove.add(new RegionPlan(hri, sal.getServerName(), null));
numTaken++;
- if (numTaken >= numToOffload) break;
+ if (numTaken >= numToOffload) {
+ break;
+ }
}
serverBalanceInfo.put(sal.getServerName(),
new BalanceInfo(numToOffload, (-1)*numTaken, server.getValue()));
@@ -343,7 +343,9 @@ public class SimpleLoadBalancer extends BaseLoadBalancer {
int maxToTake = numRegions - min;
for (Map.Entry> server:
serversByLoad.entrySet()) {
- if (maxToTake == 0) break; // no more to take
+ if (maxToTake == 0) {
+ break; // no more to take
+ }
int load = server.getKey().getLoad();
if (load >= min) {
continue; // look for other servers which haven't reached min
@@ -357,15 +359,19 @@ public class SimpleLoadBalancer extends BaseLoadBalancer {
int incr = 1;
List sns =
Arrays.asList(underloadedServers.keySet().toArray(new ServerName[serversUnderloaded]));
- Collections.shuffle(sns, RANDOM);
+ Collections.shuffle(sns);
while (regionsToMove.size() > 0) {
int cnt = 0;
int i = incr > 0 ? 0 : underloadedServers.size()-1;
for (; i >= 0 && i < underloadedServers.size(); i += incr) {
- if (regionsToMove.isEmpty()) break;
+ if (regionsToMove.isEmpty()) {
+ break;
+ }
ServerName si = sns.get(i);
int numToTake = underloadedServers.get(si);
- if (numToTake == 0) continue;
+ if (numToTake == 0) {
+ continue;
+ }
addRegionPlan(regionsToMove, fetchFromTail, si, regionsToReturn);
@@ -374,7 +380,9 @@ public class SimpleLoadBalancer extends BaseLoadBalancer {
BalanceInfo bi = serverBalanceInfo.get(si);
bi.setNumRegionsAdded(bi.getNumRegionsAdded()+1);
}
- if (cnt == 0) break;
+ if (cnt == 0) {
+ break;
+ }
// iterates underloadedServers in the other direction
incr = -incr;
}
@@ -395,9 +403,13 @@ public class SimpleLoadBalancer extends BaseLoadBalancer {
serverBalanceInfo.get(server.getKey().getServerName());
int idx =
balanceInfo == null ? 0 : balanceInfo.getNextRegionForUnload();
- if (idx >= server.getValue().size()) break;
+ if (idx >= server.getValue().size()) {
+ break;
+ }
RegionInfo region = server.getValue().get(idx);
- if (region.isMetaRegion()) continue; // Don't move meta regions.
+ if (region.isMetaRegion()) {
+ continue; // Don't move meta regions.
+ }
regionsToMove.add(new RegionPlan(region, server.getKey().getServerName(), null));
balanceInfo.setNumRegionsAdded(balanceInfo.getNumRegionsAdded() - 1);
balanceInfo.setNextRegionForUnload(balanceInfo.getNextRegionForUnload() + 1);
@@ -416,7 +428,9 @@ public class SimpleLoadBalancer extends BaseLoadBalancer {
for (Map.Entry> server :
serversByLoad.entrySet()) {
int regionCount = server.getKey().getLoad();
- if (regionCount >= min) break;
+ if (regionCount >= min) {
+ break;
+ }
BalanceInfo balanceInfo = serverBalanceInfo.get(server.getKey().getServerName());
if(balanceInfo != null) {
regionCount += balanceInfo.getNumRegionsAdded();
@@ -447,7 +461,9 @@ public class SimpleLoadBalancer extends BaseLoadBalancer {
", serversUnderloaded=" + serversUnderloaded);
StringBuilder sb = new StringBuilder();
for (Map.Entry> e: loadOfOneTable.entrySet()) {
- if (sb.length() > 0) sb.append(", ");
+ if (sb.length() > 0) {
+ sb.append(", ");
+ }
sb.append(e.getKey().toString());
sb.append(" ");
sb.append(e.getValue().size());
@@ -465,15 +481,16 @@ public class SimpleLoadBalancer extends BaseLoadBalancer {
}
/**
- * If we need to balanceoverall, we need to add one more round to peel off one region from each max.
- * Together with other regions left to be assigned, we distribute all regionToMove, to the RS
+ * If we need to balanceoverall, we need to add one more round to peel off one region from each
+ * max. Together with other regions left to be assigned, we distribute all regionToMove, to the RS
* that have less regions in whole cluster scope.
*/
public void balanceOverall(List regionsToReturn,
- Map serverBalanceInfo, boolean fetchFromTail,
- MinMaxPriorityQueue regionsToMove, int max, int min ){
+ Map serverBalanceInfo, boolean fetchFromTail,
+ MinMaxPriorityQueue regionsToMove, int max, int min) {
// Step 1.
- // A map to record the plan we have already got as status quo, in order to resolve a cyclic assignment pair,
+ // A map to record the plan we have already got as status quo, in order to resolve a cyclic
+ // assignment pair,
// e.g. plan 1: A -> B, plan 2: B ->C => resolve plan1 to A -> C, remove plan2
Map> returnMap = new HashMap<>();
for (int i = 0; i < regionsToReturn.size(); i++) {
@@ -509,12 +526,14 @@ public class SimpleLoadBalancer extends BaseLoadBalancer {
RegionPlan maxPlan = new RegionPlan(hriToPlan, serverload.getServerName(), null);
regionsToMove.add(maxPlan);
setLoad(serverLoadList, i, -1);
- }else if(balanceInfo.getHriList().size() + balanceInfo.getNumRegionsAdded() > max
- || balanceInfo.getHriList().size() + balanceInfo.getNumRegionsAdded() < min){
- LOG.warn("Encounter incorrect region numbers after calculating move plan during balanceOverall, " +
- "for this table, " + serverload.getServerName() + " originally has " + balanceInfo.getHriList().size() +
- " regions and " + balanceInfo.getNumRegionsAdded() + " regions have been added. Yet, max =" +
- max + ", min =" + min + ". Thus stop balance for this table"); // should not happen
+ } else if (balanceInfo.getHriList().size() + balanceInfo.getNumRegionsAdded() > max ||
+ balanceInfo.getHriList().size() + balanceInfo.getNumRegionsAdded() < min) {
+ LOG.warn(
+ "Encounter incorrect region numbers after calculating move plan during balanceOverall, " +
+ "for this table, " + serverload.getServerName() + " originally has " +
+ balanceInfo.getHriList().size() + " regions and " + balanceInfo.getNumRegionsAdded() +
+ " regions have been added. Yet, max =" + max + ", min =" + min +
+ ". Thus stop balance for this table"); // should not happen
return;
}
}
@@ -522,12 +541,16 @@ public class SimpleLoadBalancer extends BaseLoadBalancer {
// Step 3. sort the ServerLoadList, the ArrayList hold overall load for each server.
// We only need to assign the regionsToMove to
// the first n = regionsToMove.size() RS that has least load.
- Collections.sort(serverLoadList,new Comparator(){
+ Collections.sort(serverLoadList, new Comparator() {
@Override
public int compare(ServerAndLoad s1, ServerAndLoad s2) {
- if(s1.getLoad() == s2.getLoad()) return 0;
- else return (s1.getLoad() > s2.getLoad())? 1 : -1;
- }});
+ if (s1.getLoad() == s2.getLoad()) {
+ return 0;
+ } else {
+ return (s1.getLoad() > s2.getLoad()) ? 1 : -1;
+ }
+ }
+ });
// Step 4.
// Preparation before assign out all regionsToMove.
@@ -542,16 +565,19 @@ public class SimpleLoadBalancer extends BaseLoadBalancer {
Pair shredLoad;
// A List to help mark the plan in regionsToMove that should be removed
List planToRemoveList = new ArrayList<>();
- // A structure to record how many times a server becomes the source of a plan, from regionsToMove.
+ // A structure to record how many times a server becomes the source of a plan, from
+ // regionsToMove.
Map sourceMap = new HashMap<>();
// We remove one of the plan which would cause source RS equals destination RS.
// But we should keep in mind that the second plan from such RS should be kept.
for(RegionPlan plan: regionsToMove){
// the source RS's load and index in ServerLoadList
shredLoad = SnLoadMap.get(plan.getSource());
- if(!sourceMap.containsKey(plan.getSource())) sourceMap.put(plan.getSource(), 0);
+ if (!sourceMap.containsKey(plan.getSource())) {
+ sourceMap.put(plan.getSource(), 0);
+ }
sourceMap.put(plan.getSource(), sourceMap.get(plan.getSource()) + 1);
- if(shredLoad.getSecond() < assignLength && sourceMap.get(plan.getSource()) == 1) {
+ if (shredLoad.getSecond() < assignLength && sourceMap.get(plan.getSource()) == 1) {
planToRemoveList.add(plan);
// While marked as to be removed, the count should be add back to the source RS
setLoad(serverLoadList, shredLoad.getSecond(), 1);
@@ -570,14 +596,18 @@ public class SimpleLoadBalancer extends BaseLoadBalancer {
// With this strategy adopted, we can gradually achieve the overall balance,
// while keeping table level balanced.
for(int i = 0; i < assignLength; i++){
- // skip the RS that is also the source, we have removed them from regionsToMove in previous step
- if(sourceMap.containsKey(serverLoadList.get(i).getServerName())) continue;
+ // skip the RS that is also the source, we have removed them from regionsToMove in previous
+ // step
+ if (sourceMap.containsKey(serverLoadList.get(i).getServerName())) {
+ continue;
+ }
addRegionPlan(regionsToMove, fetchFromTail,
serverLoadList.get(i).getServerName(), regionsToReturn);
setLoad(serverLoadList, i, 1);
// resolve a possible cyclic assignment pair if we just produced one:
// e.g. plan1: A -> B, plan2: B -> C => resolve plan1 to A -> C and remove plan2
- List pos = returnMap.get(regionsToReturn.get(regionsToReturn.size() - 1).getSource());
+ List pos =
+ returnMap.get(regionsToReturn.get(regionsToReturn.size() - 1).getSource());
if (pos != null && pos.size() != 0) {
regionsToReturn.get(pos.get(pos.size() - 1)).setDestination(
regionsToReturn.get(regionsToReturn.size() - 1).getDestination());
@@ -592,10 +622,13 @@ public class SimpleLoadBalancer extends BaseLoadBalancer {
* Add a region from the head or tail to the List of regions to return.
*/
private void addRegionPlan(final MinMaxPriorityQueue regionsToMove,
- final boolean fetchFromTail, final ServerName sn, List regionsToReturn) {
+ final boolean fetchFromTail, final ServerName sn, List regionsToReturn) {
RegionPlan rp = null;
- if (!fetchFromTail) rp = regionsToMove.remove();
- else rp = regionsToMove.removeLast();
+ if (!fetchFromTail) {
+ rp = regionsToMove.remove();
+ } else {
+ rp = regionsToMove.removeLast();
+ }
rp.setDestination(sn);
regionsToReturn.add(rp);
}
diff --git a/hbase-server/src/main/java/org/apache/hadoop/hbase/master/balancer/StochasticLoadBalancer.java b/hbase-server/src/main/java/org/apache/hadoop/hbase/master/balancer/StochasticLoadBalancer.java
index 48594943346..2dba9084265 100644
--- a/hbase-server/src/main/java/org/apache/hadoop/hbase/master/balancer/StochasticLoadBalancer.java
+++ b/hbase-server/src/main/java/org/apache/hadoop/hbase/master/balancer/StochasticLoadBalancer.java
@@ -27,7 +27,7 @@ import java.util.LinkedList;
import java.util.List;
import java.util.Map;
import java.util.Objects;
-import java.util.Random;
+import java.util.concurrent.ThreadLocalRandom;
import java.util.stream.Collectors;
import org.apache.hadoop.conf.Configuration;
import org.apache.hadoop.hbase.ClusterMetrics;
@@ -106,6 +106,8 @@ import org.apache.hbase.thirdparty.com.google.common.collect.Lists;
justification="Complaint is about costFunctions not being synchronized; not end of the world")
public class StochasticLoadBalancer extends BaseLoadBalancer {
+ private static final Logger LOG = LoggerFactory.getLogger(StochasticLoadBalancer.class);
+
protected static final String STEPS_PER_REGION_KEY =
"hbase.master.balancer.stochastic.stepsPerRegion";
protected static final String MAX_STEPS_KEY =
@@ -122,9 +124,6 @@ public class StochasticLoadBalancer extends BaseLoadBalancer {
protected static final String COST_FUNCTIONS_COST_FUNCTIONS_KEY =
"hbase.master.balancer.stochastic.additionalCostFunctions";
- protected static final Random RANDOM = new Random(System.currentTimeMillis());
- private static final Logger LOG = LoggerFactory.getLogger(StochasticLoadBalancer.class);
-
Map> loads = new HashMap<>();
// values are defaults
@@ -363,8 +362,8 @@ public class StochasticLoadBalancer extends BaseLoadBalancer {
}
BalanceAction nextAction(BalancerClusterState cluster) {
- return candidateGenerators.get(RANDOM.nextInt(candidateGenerators.size()))
- .generate(cluster);
+ return candidateGenerators.get(ThreadLocalRandom.current().nextInt(candidateGenerators.size()))
+ .generate(cluster);
}
/**
@@ -694,53 +693,6 @@ public class StochasticLoadBalancer extends BaseLoadBalancer {
return total;
}
- static class RandomCandidateGenerator extends CandidateGenerator {
-
- @Override
- BalanceAction generate(BalancerClusterState cluster) {
-
- int thisServer = pickRandomServer(cluster);
-
- // Pick the other server
- int otherServer = pickOtherRandomServer(cluster, thisServer);
-
- return pickRandomRegions(cluster, thisServer, otherServer);
- }
- }
-
- /**
- * Generates candidates which moves the replicas out of the rack for
- * co-hosted region replicas in the same rack
- */
- static class RegionReplicaRackCandidateGenerator extends RegionReplicaCandidateGenerator {
- @Override
- BalanceAction generate(BalancerClusterState cluster) {
- int rackIndex = pickRandomRack(cluster);
- if (cluster.numRacks <= 1 || rackIndex == -1) {
- return super.generate(cluster);
- }
-
- int regionIndex = selectCoHostedRegionPerGroup(
- cluster.primariesOfRegionsPerRack[rackIndex],
- cluster.regionsPerRack[rackIndex],
- cluster.regionIndexToPrimaryIndex);
-
- // if there are no pairs of region replicas co-hosted, default to random generator
- if (regionIndex == -1) {
- // default to randompicker
- return randomGenerator.generate(cluster);
- }
-
- int serverIndex = cluster.regionIndexToServerIndex[regionIndex];
- int toRackIndex = pickOtherRandomRack(cluster, rackIndex);
-
- int rand = RANDOM.nextInt(cluster.serversPerRack[toRackIndex].length);
- int toServerIndex = cluster.serversPerRack[toRackIndex][rand];
- int toRegionIndex = pickRandomRegion(cluster, toServerIndex, 0.9f);
- return getAction(serverIndex, regionIndex, toServerIndex, toRegionIndex);
- }
- }
-
/**
* Base class of StochasticLoadBalancer's Cost Functions.
*/
@@ -756,6 +708,7 @@ public class StochasticLoadBalancer extends BaseLoadBalancer {
boolean isNeeded() {
return true;
}
+
float getMultiplier() {
return multiplier;
}
diff --git a/hbase-server/src/test/java/org/apache/hadoop/hbase/master/balancer/TestBaseLoadBalancer.java b/hbase-server/src/test/java/org/apache/hadoop/hbase/master/balancer/TestBaseLoadBalancer.java
index 288a121aa5a..54078a5b70a 100644
--- a/hbase-server/src/test/java/org/apache/hadoop/hbase/master/balancer/TestBaseLoadBalancer.java
+++ b/hbase-server/src/test/java/org/apache/hadoop/hbase/master/balancer/TestBaseLoadBalancer.java
@@ -20,6 +20,8 @@ package org.apache.hadoop.hbase.master.balancer;
import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertNull;
import static org.junit.Assert.assertTrue;
+import static org.mockito.ArgumentMatchers.any;
+import static org.mockito.ArgumentMatchers.anyList;
import static org.mockito.Mockito.mock;
import static org.mockito.Mockito.when;
@@ -56,7 +58,6 @@ import org.junit.Rule;
import org.junit.Test;
import org.junit.experimental.categories.Category;
import org.junit.rules.TestName;
-import org.mockito.Mockito;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
@@ -92,22 +93,22 @@ public class TestBaseLoadBalancer extends BalancerTestBase {
conf.setClass("hbase.util.ip.to.rack.determiner", MockMapping.class, DNSToSwitchMapping.class);
loadBalancer = new MockBalancer();
loadBalancer.setConf(conf);
- MasterServices st = Mockito.mock(MasterServices.class);
- Mockito.when(st.getServerName()).thenReturn(master);
+ MasterServices st = mock(MasterServices.class);
+ when(st.getServerName()).thenReturn(master);
loadBalancer.setMasterServices(st);
// Set up the rack topologies (5 machines per rack)
- rackManager = Mockito.mock(RackManager.class);
+ rackManager = mock(RackManager.class);
for (int i = 0; i < NUM_SERVERS; i++) {
servers[i] = ServerName.valueOf("foo"+i+":1234",-1);
if (i < 5) {
- Mockito.when(rackManager.getRack(servers[i])).thenReturn("rack1");
+ when(rackManager.getRack(servers[i])).thenReturn("rack1");
}
if (i >= 5 && i < 10) {
- Mockito.when(rackManager.getRack(servers[i])).thenReturn("rack2");
+ when(rackManager.getRack(servers[i])).thenReturn("rack2");
}
if (i >= 10) {
- Mockito.when(rackManager.getRack(servers[i])).thenReturn("rack3");
+ when(rackManager.getRack(servers[i])).thenReturn("rack3");
}
}
}
@@ -126,19 +127,6 @@ public class TestBaseLoadBalancer extends BalancerTestBase {
}
}
- /**
- * All regions have an assignment.
- * @param regions
- * @param servers
- * @param assignments
- */
- private void assertImmediateAssignment(List regions, List servers,
- Map assignments) {
- for (RegionInfo region : regions) {
- assertTrue(assignments.containsKey(region));
- }
- }
-
/**
* Tests the bulk assignment used during cluster startup.
*
@@ -250,11 +238,10 @@ public class TestBaseLoadBalancer extends BalancerTestBase {
Configuration conf = HBaseConfiguration.create();
conf.setClass("hbase.util.ip.to.rack.determiner", MockMapping.class, DNSToSwitchMapping.class);
balancer.setConf(conf);
- ServerManager sm = Mockito.mock(ServerManager.class);
- Mockito.when(sm.getOnlineServersListWithPredicator(allServers, BaseLoadBalancer.IDLE_SERVER_PREDICATOR))
- .thenReturn(idleServers);
- MasterServices services = Mockito.mock(MasterServices.class);
- Mockito.when(services.getServerManager()).thenReturn(sm);
+ ServerManager sm = mock(ServerManager.class);
+ when(sm.getOnlineServersListWithPredicator(anyList(), any())).thenReturn(idleServers);
+ MasterServices services = mock(MasterServices.class);
+ when(services.getServerManager()).thenReturn(sm);
balancer.setMasterServices(services);
RegionInfo hri1 = RegionInfoBuilder.newBuilder(TableName.valueOf(name.getMethodName()))
.setStartKey("key1".getBytes())
@@ -262,7 +249,7 @@ public class TestBaseLoadBalancer extends BalancerTestBase {
.setSplit(false)
.setRegionId(100)
.build();
- assertNull(balancer.randomAssignment(hri1, Collections.EMPTY_LIST));
+ assertNull(balancer.randomAssignment(hri1, Collections.emptyList()));
assertNull(balancer.randomAssignment(hri1, null));
for (int i = 0; i != 3; ++i) {
ServerName sn = balancer.randomAssignment(hri1, allServers);
diff --git a/hbase-server/src/test/java/org/apache/hadoop/hbase/master/balancer/TestStochasticLoadBalancerHeterogeneousCost.java b/hbase-server/src/test/java/org/apache/hadoop/hbase/master/balancer/TestStochasticLoadBalancerHeterogeneousCost.java
index 8a2ec848a75..8ea71183b86 100644
--- a/hbase-server/src/test/java/org/apache/hadoop/hbase/master/balancer/TestStochasticLoadBalancerHeterogeneousCost.java
+++ b/hbase-server/src/test/java/org/apache/hadoop/hbase/master/balancer/TestStochasticLoadBalancerHeterogeneousCost.java
@@ -282,7 +282,7 @@ public class TestStochasticLoadBalancerHeterogeneousCost extends BalancerTestBas
}
static class FairRandomCandidateGenerator extends
- StochasticLoadBalancer.RandomCandidateGenerator {
+ RandomCandidateGenerator {
@Override
public BalanceAction pickRandomRegions(BalancerClusterState cluster,