001/** 002 * Licensed to the Apache Software Foundation (ASF) under one 003 * or more contributor license agreements. See the NOTICE file 004 * distributed with this work for additional information 005 * regarding copyright ownership. The ASF licenses this file 006 * to you under the Apache License, Version 2.0 (the 007 * "License"); you may not use this file except in compliance 008 * with the License. You may obtain a copy of the License at 009 * 010 * http://www.apache.org/licenses/LICENSE-2.0 011 * 012 * Unless required by applicable law or agreed to in writing, software 013 * distributed under the License is distributed on an "AS IS" BASIS, 014 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 015 * See the License for the specific language governing permissions and 016 * limitations under the License. 017 */ 018package org.apache.hadoop.hbase.master.balancer; 019 020import static org.junit.Assert.assertEquals; 021import static org.junit.Assert.assertNull; 022import static org.junit.Assert.assertTrue; 023import static org.mockito.Mockito.mock; 024import static org.mockito.Mockito.when; 025 026import java.util.ArrayList; 027import java.util.Collections; 028import java.util.HashMap; 029import java.util.LinkedHashMap; 030import java.util.List; 031import java.util.Map; 032import java.util.Set; 033import java.util.TreeMap; 034import java.util.TreeSet; 035import java.util.stream.Collectors; 036import org.apache.commons.lang3.ArrayUtils; 037import org.apache.hadoop.conf.Configuration; 038import org.apache.hadoop.hbase.HBaseClassTestRule; 039import org.apache.hadoop.hbase.HBaseConfiguration; 040import org.apache.hadoop.hbase.ServerName; 041import org.apache.hadoop.hbase.TableName; 042import org.apache.hadoop.hbase.client.RegionInfo; 043import org.apache.hadoop.hbase.client.RegionInfoBuilder; 044import org.apache.hadoop.hbase.client.RegionReplicaUtil; 045import org.apache.hadoop.hbase.master.LoadBalancer; 046import org.apache.hadoop.hbase.master.MasterServices; 047import org.apache.hadoop.hbase.master.RackManager; 048import org.apache.hadoop.hbase.master.RegionPlan; 049import org.apache.hadoop.hbase.master.ServerManager; 050import org.apache.hadoop.hbase.master.balancer.BaseLoadBalancer.Cluster; 051import org.apache.hadoop.hbase.master.balancer.BaseLoadBalancer.Cluster.MoveRegionAction; 052import org.apache.hadoop.hbase.testclassification.MasterTests; 053import org.apache.hadoop.hbase.testclassification.MediumTests; 054import org.apache.hadoop.net.DNSToSwitchMapping; 055import org.junit.BeforeClass; 056import org.junit.ClassRule; 057import org.junit.Rule; 058import org.junit.Test; 059import org.junit.experimental.categories.Category; 060import org.junit.rules.TestName; 061import org.mockito.Mockito; 062import org.slf4j.Logger; 063import org.slf4j.LoggerFactory; 064 065import org.apache.hbase.thirdparty.com.google.common.collect.Lists; 066 067@Category({MasterTests.class, MediumTests.class}) 068public class TestBaseLoadBalancer extends BalancerTestBase { 069 070 @ClassRule 071 public static final HBaseClassTestRule CLASS_RULE = 072 HBaseClassTestRule.forClass(TestBaseLoadBalancer.class); 073 074 private static LoadBalancer loadBalancer; 075 private static final Logger LOG = LoggerFactory.getLogger(TestBaseLoadBalancer.class); 076 private static final ServerName master = ServerName.valueOf("fake-master", 0, 1L); 077 private static RackManager rackManager; 078 private static final int NUM_SERVERS = 15; 079 private static ServerName[] servers = new ServerName[NUM_SERVERS]; 080 081 int[][] regionsAndServersMocks = new int[][] { 082 // { num regions, num servers } 083 new int[] { 0, 0 }, new int[] { 0, 1 }, new int[] { 1, 1 }, new int[] { 2, 1 }, 084 new int[] { 10, 1 }, new int[] { 1, 2 }, new int[] { 2, 2 }, new int[] { 3, 2 }, 085 new int[] { 1, 3 }, new int[] { 2, 3 }, new int[] { 3, 3 }, new int[] { 25, 3 }, 086 new int[] { 2, 10 }, new int[] { 2, 100 }, new int[] { 12, 10 }, new int[] { 12, 100 }, }; 087 088 @Rule 089 public TestName name = new TestName(); 090 091 @BeforeClass 092 public static void beforeAllTests() throws Exception { 093 Configuration conf = HBaseConfiguration.create(); 094 conf.setClass("hbase.util.ip.to.rack.determiner", MockMapping.class, DNSToSwitchMapping.class); 095 loadBalancer = new MockBalancer(); 096 loadBalancer.setConf(conf); 097 MasterServices st = Mockito.mock(MasterServices.class); 098 Mockito.when(st.getServerName()).thenReturn(master); 099 loadBalancer.setMasterServices(st); 100 101 // Set up the rack topologies (5 machines per rack) 102 rackManager = Mockito.mock(RackManager.class); 103 for (int i = 0; i < NUM_SERVERS; i++) { 104 servers[i] = ServerName.valueOf("foo"+i+":1234",-1); 105 if (i < 5) { 106 Mockito.when(rackManager.getRack(servers[i])).thenReturn("rack1"); 107 } 108 if (i >= 5 && i < 10) { 109 Mockito.when(rackManager.getRack(servers[i])).thenReturn("rack2"); 110 } 111 if (i >= 10) { 112 Mockito.when(rackManager.getRack(servers[i])).thenReturn("rack3"); 113 } 114 } 115 } 116 117 public static class MockBalancer extends BaseLoadBalancer { 118 @Override 119 public List<RegionPlan> 120 balanceCluster(Map<TableName, Map<ServerName, List<RegionInfo>>> loadOfAllTable) { 121 return null; 122 } 123 124 @Override 125 public List<RegionPlan> balanceTable(TableName tableName, 126 Map<ServerName, List<RegionInfo>> loadOfOneTable) { 127 return null; 128 } 129 } 130 131 /** 132 * All regions have an assignment. 133 * @param regions 134 * @param servers 135 * @param assignments 136 */ 137 private void assertImmediateAssignment(List<RegionInfo> regions, List<ServerName> servers, 138 Map<RegionInfo, ServerName> assignments) { 139 for (RegionInfo region : regions) { 140 assertTrue(assignments.containsKey(region)); 141 } 142 } 143 144 /** 145 * Tests the bulk assignment used during cluster startup. 146 * 147 * Round-robin. Should yield a balanced cluster so same invariant as the load 148 * balancer holds, all servers holding either floor(avg) or ceiling(avg). 149 * 150 * @throws Exception 151 */ 152 @Test 153 public void testBulkAssignment() throws Exception { 154 List<ServerName> tmp = getListOfServerNames(randomServers(5, 0)); 155 List<RegionInfo> hris = randomRegions(20); 156 hris.add(RegionInfoBuilder.FIRST_META_REGIONINFO); 157 tmp.add(master); 158 Map<ServerName, List<RegionInfo>> plans = loadBalancer.roundRobinAssignment(hris, tmp); 159 if (LoadBalancer.isTablesOnMaster(loadBalancer.getConf())) { 160 assertTrue(plans.get(master).contains(RegionInfoBuilder.FIRST_META_REGIONINFO)); 161 assertEquals(1, plans.get(master).size()); 162 } 163 int totalRegion = 0; 164 for (List<RegionInfo> regions: plans.values()) { 165 totalRegion += regions.size(); 166 } 167 assertEquals(hris.size(), totalRegion); 168 for (int[] mock : regionsAndServersMocks) { 169 LOG.debug("testBulkAssignment with " + mock[0] + " regions and " + mock[1] + " servers"); 170 List<RegionInfo> regions = randomRegions(mock[0]); 171 List<ServerAndLoad> servers = randomServers(mock[1], 0); 172 List<ServerName> list = getListOfServerNames(servers); 173 Map<ServerName, List<RegionInfo>> assignments = 174 loadBalancer.roundRobinAssignment(regions, list); 175 float average = (float) regions.size() / servers.size(); 176 int min = (int) Math.floor(average); 177 int max = (int) Math.ceil(average); 178 if (assignments != null && !assignments.isEmpty()) { 179 for (List<RegionInfo> regionList : assignments.values()) { 180 assertTrue(regionList.size() == min || regionList.size() == max); 181 } 182 } 183 returnRegions(regions); 184 returnServers(list); 185 } 186 } 187 188 /** 189 * Test the cluster startup bulk assignment which attempts to retain 190 * assignment info. 191 * @throws Exception 192 */ 193 @Test 194 public void testRetainAssignment() throws Exception { 195 // Test simple case where all same servers are there 196 List<ServerAndLoad> servers = randomServers(10, 10); 197 List<RegionInfo> regions = randomRegions(100); 198 Map<RegionInfo, ServerName> existing = new TreeMap<>(RegionInfo.COMPARATOR); 199 for (int i = 0; i < regions.size(); i++) { 200 ServerName sn = servers.get(i % servers.size()).getServerName(); 201 // The old server would have had same host and port, but different 202 // start code! 203 ServerName snWithOldStartCode = 204 ServerName.valueOf(sn.getHostname(), sn.getPort(), sn.getStartcode() - 10); 205 existing.put(regions.get(i), snWithOldStartCode); 206 } 207 List<ServerName> listOfServerNames = getListOfServerNames(servers); 208 Map<ServerName, List<RegionInfo>> assignment = 209 loadBalancer.retainAssignment(existing, listOfServerNames); 210 assertRetainedAssignment(existing, listOfServerNames, assignment); 211 212 // Include two new servers that were not there before 213 List<ServerAndLoad> servers2 = new ArrayList<>(servers); 214 servers2.add(randomServer(10)); 215 servers2.add(randomServer(10)); 216 listOfServerNames = getListOfServerNames(servers2); 217 assignment = loadBalancer.retainAssignment(existing, listOfServerNames); 218 assertRetainedAssignment(existing, listOfServerNames, assignment); 219 220 // Remove two of the servers that were previously there 221 List<ServerAndLoad> servers3 = new ArrayList<>(servers); 222 servers3.remove(0); 223 servers3.remove(0); 224 listOfServerNames = getListOfServerNames(servers3); 225 assignment = loadBalancer.retainAssignment(existing, listOfServerNames); 226 assertRetainedAssignment(existing, listOfServerNames, assignment); 227 } 228 229 @Test 230 public void testRandomAssignment() throws Exception { 231 for (int i = 1; i != 5; ++i) { 232 LOG.info("run testRandomAssignment() with idle servers:" + i); 233 testRandomAssignment(i); 234 } 235 } 236 237 private void testRandomAssignment(int numberOfIdleServers) throws Exception { 238 assert numberOfIdleServers > 0; 239 List<ServerName> idleServers = new ArrayList<>(numberOfIdleServers); 240 for (int i = 0; i != numberOfIdleServers; ++i) { 241 idleServers.add(ServerName.valueOf("server-" + i, 1000, 1L)); 242 } 243 List<ServerName> allServers = new ArrayList<>(idleServers.size() + 1); 244 allServers.add(ServerName.valueOf("server-" + numberOfIdleServers, 1000, 1L)); 245 allServers.addAll(idleServers); 246 LoadBalancer balancer = new MockBalancer() { 247 @Override 248 public boolean shouldBeOnMaster(RegionInfo region) { 249 return false; 250 } 251 }; 252 Configuration conf = HBaseConfiguration.create(); 253 conf.setClass("hbase.util.ip.to.rack.determiner", MockMapping.class, DNSToSwitchMapping.class); 254 balancer.setConf(conf); 255 ServerManager sm = Mockito.mock(ServerManager.class); 256 Mockito.when(sm.getOnlineServersListWithPredicator(allServers, BaseLoadBalancer.IDLE_SERVER_PREDICATOR)) 257 .thenReturn(idleServers); 258 MasterServices services = Mockito.mock(MasterServices.class); 259 Mockito.when(services.getServerManager()).thenReturn(sm); 260 balancer.setMasterServices(services); 261 RegionInfo hri1 = RegionInfoBuilder.newBuilder(TableName.valueOf(name.getMethodName())) 262 .setStartKey("key1".getBytes()) 263 .setEndKey("key2".getBytes()) 264 .setSplit(false) 265 .setRegionId(100) 266 .build(); 267 assertNull(balancer.randomAssignment(hri1, Collections.EMPTY_LIST)); 268 assertNull(balancer.randomAssignment(hri1, null)); 269 for (int i = 0; i != 3; ++i) { 270 ServerName sn = balancer.randomAssignment(hri1, allServers); 271 assertTrue("actual:" + sn + ", except:" + idleServers, idleServers.contains(sn)); 272 } 273 } 274 275 @Test 276 public void testRegionAvailability() throws Exception { 277 // Create a cluster with a few servers, assign them to specific racks 278 // then assign some regions. The tests should check whether moving a 279 // replica from one node to a specific other node or rack lowers the 280 // availability of the region or not 281 282 List<RegionInfo> list0 = new ArrayList<>(); 283 List<RegionInfo> list1 = new ArrayList<>(); 284 List<RegionInfo> list2 = new ArrayList<>(); 285 // create a region (region1) 286 RegionInfo hri1 = RegionInfoBuilder.newBuilder(TableName.valueOf(name.getMethodName())) 287 .setStartKey("key1".getBytes()) 288 .setEndKey("key2".getBytes()) 289 .setSplit(false) 290 .setRegionId(100) 291 .build(); 292 // create a replica of the region (replica_of_region1) 293 RegionInfo hri2 = RegionReplicaUtil.getRegionInfoForReplica(hri1, 1); 294 // create a second region (region2) 295 RegionInfo hri3 = RegionInfoBuilder.newBuilder(TableName.valueOf(name.getMethodName())) 296 .setStartKey("key2".getBytes()) 297 .setEndKey("key3".getBytes()) 298 .setSplit(false) 299 .setRegionId(101) 300 .build(); 301 list0.add(hri1); //only region1 302 list1.add(hri2); //only replica_of_region1 303 list2.add(hri3); //only region2 304 Map<ServerName, List<RegionInfo>> clusterState = new LinkedHashMap<>(); 305 clusterState.put(servers[0], list0); //servers[0] hosts region1 306 clusterState.put(servers[1], list1); //servers[1] hosts replica_of_region1 307 clusterState.put(servers[2], list2); //servers[2] hosts region2 308 // create a cluster with the above clusterState. The way in which the 309 // cluster is created (constructor code) would make sure the indices of 310 // the servers are in the order in which it is inserted in the clusterState 311 // map (linkedhashmap is important). A similar thing applies to the region lists 312 Cluster cluster = new Cluster(clusterState, null, null, rackManager); 313 // check whether a move of region1 from servers[0] to servers[1] would lower 314 // the availability of region1 315 assertTrue(cluster.wouldLowerAvailability(hri1, servers[1])); 316 // check whether a move of region1 from servers[0] to servers[2] would lower 317 // the availability of region1 318 assertTrue(!cluster.wouldLowerAvailability(hri1, servers[2])); 319 // check whether a move of replica_of_region1 from servers[0] to servers[2] would lower 320 // the availability of replica_of_region1 321 assertTrue(!cluster.wouldLowerAvailability(hri2, servers[2])); 322 // check whether a move of region2 from servers[0] to servers[1] would lower 323 // the availability of region2 324 assertTrue(!cluster.wouldLowerAvailability(hri3, servers[1])); 325 326 // now lets have servers[1] host replica_of_region2 327 list1.add(RegionReplicaUtil.getRegionInfoForReplica(hri3, 1)); 328 // create a new clusterState with the above change 329 cluster = new Cluster(clusterState, null, null, rackManager); 330 // now check whether a move of a replica from servers[0] to servers[1] would lower 331 // the availability of region2 332 assertTrue(cluster.wouldLowerAvailability(hri3, servers[1])); 333 334 // start over again 335 clusterState.clear(); 336 clusterState.put(servers[0], list0); //servers[0], rack1 hosts region1 337 clusterState.put(servers[5], list1); //servers[5], rack2 hosts replica_of_region1 and replica_of_region2 338 clusterState.put(servers[6], list2); //servers[6], rack2 hosts region2 339 clusterState.put(servers[10], new ArrayList<>()); //servers[10], rack3 hosts no region 340 // create a cluster with the above clusterState 341 cluster = new Cluster(clusterState, null, null, rackManager); 342 // check whether a move of region1 from servers[0],rack1 to servers[6],rack2 would 343 // lower the availability 344 345 assertTrue(cluster.wouldLowerAvailability(hri1, servers[0])); 346 347 // now create a cluster without the rack manager 348 cluster = new Cluster(clusterState, null, null, null); 349 // now repeat check whether a move of region1 from servers[0] to servers[6] would 350 // lower the availability 351 assertTrue(!cluster.wouldLowerAvailability(hri1, servers[6])); 352 } 353 354 @Test 355 public void testRegionAvailabilityWithRegionMoves() throws Exception { 356 List<RegionInfo> list0 = new ArrayList<>(); 357 List<RegionInfo> list1 = new ArrayList<>(); 358 List<RegionInfo> list2 = new ArrayList<>(); 359 // create a region (region1) 360 RegionInfo hri1 = RegionInfoBuilder.newBuilder(TableName.valueOf(name.getMethodName())) 361 .setStartKey("key1".getBytes()) 362 .setEndKey("key2".getBytes()) 363 .setSplit(false) 364 .setRegionId(100) 365 .build(); 366 // create a replica of the region (replica_of_region1) 367 RegionInfo hri2 = RegionReplicaUtil.getRegionInfoForReplica(hri1, 1); 368 // create a second region (region2) 369 RegionInfo hri3 = RegionInfoBuilder.newBuilder(TableName.valueOf(name.getMethodName())) 370 .setStartKey("key2".getBytes()) 371 .setEndKey("key3".getBytes()) 372 .setSplit(false) 373 .setRegionId(101) 374 .build(); 375 list0.add(hri1); //only region1 376 list1.add(hri2); //only replica_of_region1 377 list2.add(hri3); //only region2 378 Map<ServerName, List<RegionInfo>> clusterState = new LinkedHashMap<>(); 379 clusterState.put(servers[0], list0); //servers[0] hosts region1 380 clusterState.put(servers[1], list1); //servers[1] hosts replica_of_region1 381 clusterState.put(servers[2], list2); //servers[2] hosts region2 382 // create a cluster with the above clusterState. The way in which the 383 // cluster is created (constructor code) would make sure the indices of 384 // the servers are in the order in which it is inserted in the clusterState 385 // map (linkedhashmap is important). 386 Cluster cluster = new Cluster(clusterState, null, null, rackManager); 387 // check whether moving region1 from servers[1] to servers[2] would lower availability 388 assertTrue(!cluster.wouldLowerAvailability(hri1, servers[2])); 389 390 // now move region1 from servers[0] to servers[2] 391 cluster.doAction(new MoveRegionAction(0, 0, 2)); 392 // check that the numMaxRegionsPerTable for "table" has increased to 2 393 assertEquals(2, cluster.numMaxRegionsPerTable[0]); 394 // now repeat check whether moving region1 from servers[1] to servers[2] 395 // would lower availability 396 assertTrue(cluster.wouldLowerAvailability(hri1, servers[2])); 397 398 // start over again 399 clusterState.clear(); 400 List<RegionInfo> list3 = new ArrayList<>(); 401 RegionInfo hri4 = RegionReplicaUtil.getRegionInfoForReplica(hri3, 1); 402 list3.add(hri4); 403 clusterState.put(servers[0], list0); //servers[0], rack1 hosts region1 404 clusterState.put(servers[5], list1); //servers[5], rack2 hosts replica_of_region1 405 clusterState.put(servers[6], list2); //servers[6], rack2 hosts region2 406 clusterState.put(servers[12], list3); //servers[12], rack3 hosts replica_of_region2 407 // create a cluster with the above clusterState 408 cluster = new Cluster(clusterState, null, null, rackManager); 409 // check whether a move of replica_of_region2 from servers[12],rack3 to servers[0],rack1 would 410 // lower the availability 411 assertTrue(!cluster.wouldLowerAvailability(hri4, servers[0])); 412 // now move region2 from servers[6],rack2 to servers[0],rack1 413 cluster.doAction(new MoveRegionAction(2, 2, 0)); 414 // now repeat check if replica_of_region2 from servers[12],rack3 to servers[0],rack1 would 415 // lower the availability 416 assertTrue(cluster.wouldLowerAvailability(hri3, servers[0])); 417 } 418 419 private List<ServerName> getListOfServerNames(final List<ServerAndLoad> sals) { 420 return sals.stream().map(ServerAndLoad::getServerName).collect(Collectors.toList()); 421 } 422 423 /** 424 * Asserts a valid retained assignment plan. 425 * <p> 426 * Must meet the following conditions: 427 * <ul> 428 * <li>Every input region has an assignment, and to an online server 429 * <li>If a region had an existing assignment to a server with the same 430 * address a a currently online server, it will be assigned to it 431 * </ul> 432 * @param existing 433 * @param servers 434 * @param assignment 435 */ 436 private void assertRetainedAssignment(Map<RegionInfo, ServerName> existing, 437 List<ServerName> servers, Map<ServerName, List<RegionInfo>> assignment) { 438 // Verify condition 1, every region assigned, and to online server 439 Set<ServerName> onlineServerSet = new TreeSet<>(servers); 440 Set<RegionInfo> assignedRegions = new TreeSet<>(RegionInfo.COMPARATOR); 441 for (Map.Entry<ServerName, List<RegionInfo>> a : assignment.entrySet()) { 442 assertTrue("Region assigned to server that was not listed as online", 443 onlineServerSet.contains(a.getKey())); 444 for (RegionInfo r : a.getValue()) 445 assignedRegions.add(r); 446 } 447 assertEquals(existing.size(), assignedRegions.size()); 448 449 // Verify condition 2, if server had existing assignment, must have same 450 Set<String> onlineHostNames = new TreeSet<>(); 451 for (ServerName s : servers) { 452 onlineHostNames.add(s.getHostname()); 453 } 454 455 for (Map.Entry<ServerName, List<RegionInfo>> a : assignment.entrySet()) { 456 ServerName assignedTo = a.getKey(); 457 for (RegionInfo r : a.getValue()) { 458 ServerName address = existing.get(r); 459 if (address != null && onlineHostNames.contains(address.getHostname())) { 460 // this region was prevously assigned somewhere, and that 461 // host is still around, then it should be re-assigned on the 462 // same host 463 assertEquals(address.getHostname(), assignedTo.getHostname()); 464 } 465 } 466 } 467 } 468 469 @Test 470 public void testClusterServersWithSameHostPort() { 471 // tests whether the BaseLoadBalancer.Cluster can be constructed with servers 472 // sharing same host and port 473 List<ServerName> servers = getListOfServerNames(randomServers(10, 10)); 474 List<RegionInfo> regions = randomRegions(101); 475 Map<ServerName, List<RegionInfo>> clusterState = new TreeMap<>(); 476 477 assignRegions(regions, servers, clusterState); 478 479 // construct another list of servers, but sharing same hosts and ports 480 List<ServerName> oldServers = new ArrayList<>(servers.size()); 481 for (ServerName sn : servers) { 482 // The old server would have had same host and port, but different start code! 483 oldServers.add(ServerName.valueOf(sn.getHostname(), sn.getPort(), sn.getStartcode() - 10)); 484 } 485 486 regions = randomRegions(9); // some more regions 487 assignRegions(regions, oldServers, clusterState); 488 489 // should not throw exception: 490 BaseLoadBalancer.Cluster cluster = new Cluster(clusterState, null, null, null); 491 assertEquals(101 + 9, cluster.numRegions); 492 assertEquals(10, cluster.numServers); // only 10 servers because they share the same host + port 493 494 // test move 495 ServerName sn = oldServers.get(0); 496 int r0 = ArrayUtils.indexOf(cluster.regions, clusterState.get(sn).get(0)); 497 int f0 = cluster.serversToIndex.get(sn.getHostAndPort()); 498 int t0 = cluster.serversToIndex.get(servers.get(1).getHostAndPort()); 499 cluster.doAction(new MoveRegionAction(r0, f0, t0)); 500 } 501 502 private void assignRegions(List<RegionInfo> regions, List<ServerName> servers, 503 Map<ServerName, List<RegionInfo>> clusterState) { 504 for (int i = 0; i < regions.size(); i++) { 505 ServerName sn = servers.get(i % servers.size()); 506 List<RegionInfo> regionsOfServer = clusterState.get(sn); 507 if (regionsOfServer == null) { 508 regionsOfServer = new ArrayList<>(10); 509 clusterState.put(sn, regionsOfServer); 510 } 511 512 regionsOfServer.add(regions.get(i)); 513 } 514 } 515 516 @Test 517 public void testClusterRegionLocations() { 518 // tests whether region locations are handled correctly in Cluster 519 List<ServerName> servers = getListOfServerNames(randomServers(10, 10)); 520 List<RegionInfo> regions = randomRegions(101); 521 Map<ServerName, List<RegionInfo>> clusterState = new HashMap<>(); 522 523 assignRegions(regions, servers, clusterState); 524 525 // mock block locality for some regions 526 RegionLocationFinder locationFinder = mock(RegionLocationFinder.class); 527 // block locality: region:0 => {server:0} 528 // region:1 => {server:0, server:1} 529 // region:42 => {server:4, server:9, server:5} 530 when(locationFinder.getTopBlockLocations(regions.get(0))).thenReturn( 531 Lists.newArrayList(servers.get(0))); 532 when(locationFinder.getTopBlockLocations(regions.get(1))).thenReturn( 533 Lists.newArrayList(servers.get(0), servers.get(1))); 534 when(locationFinder.getTopBlockLocations(regions.get(42))).thenReturn( 535 Lists.newArrayList(servers.get(4), servers.get(9), servers.get(5))); 536 when(locationFinder.getTopBlockLocations(regions.get(43))).thenReturn( 537 Lists.newArrayList(ServerName.valueOf("foo", 0, 0))); // this server does not exists in clusterStatus 538 539 BaseLoadBalancer.Cluster cluster = new Cluster(clusterState, null, locationFinder, null); 540 541 int r0 = ArrayUtils.indexOf(cluster.regions, regions.get(0)); // this is ok, it is just a test 542 int r1 = ArrayUtils.indexOf(cluster.regions, regions.get(1)); 543 int r10 = ArrayUtils.indexOf(cluster.regions, regions.get(10)); 544 int r42 = ArrayUtils.indexOf(cluster.regions, regions.get(42)); 545 int r43 = ArrayUtils.indexOf(cluster.regions, regions.get(43)); 546 547 int s0 = cluster.serversToIndex.get(servers.get(0).getHostAndPort()); 548 int s1 = cluster.serversToIndex.get(servers.get(1).getHostAndPort()); 549 int s4 = cluster.serversToIndex.get(servers.get(4).getHostAndPort()); 550 int s5 = cluster.serversToIndex.get(servers.get(5).getHostAndPort()); 551 int s9 = cluster.serversToIndex.get(servers.get(9).getHostAndPort()); 552 553 // region 0 locations 554 assertEquals(1, cluster.regionLocations[r0].length); 555 assertEquals(s0, cluster.regionLocations[r0][0]); 556 557 // region 1 locations 558 assertEquals(2, cluster.regionLocations[r1].length); 559 assertEquals(s0, cluster.regionLocations[r1][0]); 560 assertEquals(s1, cluster.regionLocations[r1][1]); 561 562 // region 10 locations 563 assertEquals(0, cluster.regionLocations[r10].length); 564 565 // region 42 locations 566 assertEquals(3, cluster.regionLocations[r42].length); 567 assertEquals(s4, cluster.regionLocations[r42][0]); 568 assertEquals(s9, cluster.regionLocations[r42][1]); 569 assertEquals(s5, cluster.regionLocations[r42][2]); 570 571 // region 43 locations 572 assertEquals(1, cluster.regionLocations[r43].length); 573 assertEquals(-1, cluster.regionLocations[r43][0]); 574 } 575}