From 3672f70e9fb99b73d9c3a6102ecab06cd87d3da5 Mon Sep 17 00:00:00 2001 From: shroman Date: Wed, 12 Jul 2017 22:04:31 +0900 Subject: [PATCH] [ROCKETMQ-243] Avoid picking the 1st element in the list of addresses for BrokerData#selectBrokerAddr(). Signed-off-by: shroman --- .../common/protocol/route/BrokerData.java | 25 +++++--- .../common/protocol/route/BrokerDataTest.java | 57 +++++++++++++++++++ 2 files changed, 74 insertions(+), 8 deletions(-) create mode 100644 common/src/test/java/org/apache/rocketmq/common/protocol/route/BrokerDataTest.java diff --git a/common/src/main/java/org/apache/rocketmq/common/protocol/route/BrokerData.java b/common/src/main/java/org/apache/rocketmq/common/protocol/route/BrokerData.java index 9d868ae03c0..36599fbc874 100644 --- a/common/src/main/java/org/apache/rocketmq/common/protocol/route/BrokerData.java +++ b/common/src/main/java/org/apache/rocketmq/common/protocol/route/BrokerData.java @@ -15,11 +15,12 @@ * limitations under the License. */ - package org.apache.rocketmq.common.protocol.route; +import java.util.ArrayList; import java.util.HashMap; -import java.util.Map; +import java.util.List; +import java.util.Random; import org.apache.rocketmq.common.MixAll; public class BrokerData implements Comparable { @@ -27,6 +28,8 @@ public class BrokerData implements Comparable { private String brokerName; private HashMap brokerAddrs; + private final Random random = new Random(); + public BrokerData() { } @@ -37,15 +40,21 @@ public BrokerData(String cluster, String brokerName, HashMap broke this.brokerAddrs = brokerAddrs; } + /** + * Selects a (preferably master) broker address from the registered list. + * If the master's address cannot be found, a slave broker address is selected in a random manner. + * + * @return Broker address. + */ public String selectBrokerAddr() { - String value = this.brokerAddrs.get(MixAll.MASTER_ID); - if (null == value) { - for (Map.Entry entry : this.brokerAddrs.entrySet()) { - return entry.getValue(); - } + String addr = this.brokerAddrs.get(MixAll.MASTER_ID); + + if (addr == null) { + List addrs = new ArrayList(brokerAddrs.values()); + return addrs.get(random.nextInt(addrs.size())); } - return value; + return addr; } public HashMap getBrokerAddrs() { diff --git a/common/src/test/java/org/apache/rocketmq/common/protocol/route/BrokerDataTest.java b/common/src/test/java/org/apache/rocketmq/common/protocol/route/BrokerDataTest.java new file mode 100644 index 00000000000..97da458da77 --- /dev/null +++ b/common/src/test/java/org/apache/rocketmq/common/protocol/route/BrokerDataTest.java @@ -0,0 +1,57 @@ +/* + * 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.rocketmq.common.protocol.route; + +import java.util.ArrayList; +import java.util.HashMap; +import java.util.List; +import org.junit.Assert; +import org.junit.BeforeClass; +import org.junit.Test; + +/** + * BrokerData tests. + */ +public class BrokerDataTest { + private static BrokerData brokerData; + + @BeforeClass + public static void prepare() { + brokerData = new BrokerData("testCluster", "testBroker", + new HashMap() {{ + put(1L, "addr1"); + put(2L, "addr2"); + put(3L, "addr3"); + }}); + } + + @Test + public void selectBrokerAddr() throws Exception { + List selectedAddr = new ArrayList(); + + for (int i = 0; i < 5; i++) + selectedAddr.add(brokerData.selectBrokerAddr()); + + List firstElemList = new ArrayList(); + + for (int i = 0; i < 5; i++) + firstElemList.add(selectedAddr.get(0)); + + Assert.assertFalse("Contains same addresses", selectedAddr.equals(firstElemList)); + } +}