HDFS-13653. Make dfs.client.failover.random.order a per nameservice configuration. Contributed by Ekanth Sethuramalingam.
(cherry picked from commit 784dbbd2d1fc0c361fee851d3f9b3aa50be5481d)
This commit is contained in:
parent
e3c96354a7
commit
f0150f024f
|
@ -94,9 +94,7 @@ public class ConfiguredFailoverProxyProvider<T> extends
|
||||||
proxies.add(new AddressRpcProxyPair<T>(address));
|
proxies.add(new AddressRpcProxyPair<T>(address));
|
||||||
}
|
}
|
||||||
// Randomize the list to prevent all clients pointing to the same one
|
// Randomize the list to prevent all clients pointing to the same one
|
||||||
boolean randomized = conf.getBoolean(
|
boolean randomized = getRandomOrder(conf, uri);
|
||||||
HdfsClientConfigKeys.Failover.RANDOM_ORDER,
|
|
||||||
HdfsClientConfigKeys.Failover.RANDOM_ORDER_DEFAULT);
|
|
||||||
if (randomized) {
|
if (randomized) {
|
||||||
Collections.shuffle(proxies);
|
Collections.shuffle(proxies);
|
||||||
}
|
}
|
||||||
|
@ -111,6 +109,31 @@ public class ConfiguredFailoverProxyProvider<T> extends
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Check whether random order is configured for failover proxy provider
|
||||||
|
* for the namenode/nameservice.
|
||||||
|
*
|
||||||
|
* @param conf Configuration
|
||||||
|
* @param nameNodeUri The URI of namenode/nameservice
|
||||||
|
* @return random order configuration
|
||||||
|
*/
|
||||||
|
private static boolean getRandomOrder(
|
||||||
|
Configuration conf, URI nameNodeUri) {
|
||||||
|
String host = nameNodeUri.getHost();
|
||||||
|
String configKeyWithHost = HdfsClientConfigKeys.Failover.RANDOM_ORDER
|
||||||
|
+ "." + host;
|
||||||
|
|
||||||
|
if (conf.get(configKeyWithHost) != null) {
|
||||||
|
return conf.getBoolean(
|
||||||
|
configKeyWithHost,
|
||||||
|
HdfsClientConfigKeys.Failover.RANDOM_ORDER_DEFAULT);
|
||||||
|
}
|
||||||
|
|
||||||
|
return conf.getBoolean(
|
||||||
|
HdfsClientConfigKeys.Failover.RANDOM_ORDER,
|
||||||
|
HdfsClientConfigKeys.Failover.RANDOM_ORDER_DEFAULT);
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Class<T> getInterface() {
|
public Class<T> getInterface() {
|
||||||
return xface;
|
return xface;
|
||||||
|
|
|
@ -0,0 +1,264 @@
|
||||||
|
/**
|
||||||
|
* 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.hdfs.server.namenode.ha;
|
||||||
|
|
||||||
|
import org.apache.hadoop.conf.Configuration;
|
||||||
|
import org.apache.hadoop.hdfs.client.HdfsClientConfigKeys;
|
||||||
|
import org.apache.hadoop.hdfs.protocol.ClientProtocol;
|
||||||
|
import org.apache.hadoop.security.UserGroupInformation;
|
||||||
|
import org.apache.hadoop.test.GenericTestUtils;
|
||||||
|
import org.apache.hadoop.util.Time;
|
||||||
|
import org.junit.Before;
|
||||||
|
import org.junit.BeforeClass;
|
||||||
|
import org.junit.Test;
|
||||||
|
import org.mockito.invocation.InvocationOnMock;
|
||||||
|
import org.mockito.stubbing.Answer;
|
||||||
|
import org.slf4j.event.Level;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.net.InetSocketAddress;
|
||||||
|
import java.net.URI;
|
||||||
|
import java.net.URISyntaxException;
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.concurrent.atomic.AtomicBoolean;
|
||||||
|
import java.util.concurrent.atomic.AtomicInteger;
|
||||||
|
|
||||||
|
import static org.junit.Assert.assertEquals;
|
||||||
|
import static org.junit.Assert.assertTrue;
|
||||||
|
import static org.mockito.Mockito.mock;
|
||||||
|
import static org.mockito.Mockito.when;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test {@link ConfiguredFailoverProxyProvider}.
|
||||||
|
* This manages failover logic for a given set of nameservices/namenodes
|
||||||
|
* (aka proxies).
|
||||||
|
*/
|
||||||
|
public class TestConfiguredFailoverProxyProvider {
|
||||||
|
private Configuration conf;
|
||||||
|
private int rpcPort = 8020;
|
||||||
|
private URI ns1Uri;
|
||||||
|
private URI ns2Uri;
|
||||||
|
private String ns1;
|
||||||
|
private String ns1nn1Hostname = "machine1.foo.bar";
|
||||||
|
private InetSocketAddress ns1nn1 =
|
||||||
|
new InetSocketAddress(ns1nn1Hostname, rpcPort);
|
||||||
|
private String ns1nn2Hostname = "machine2.foo.bar";
|
||||||
|
private InetSocketAddress ns1nn2 =
|
||||||
|
new InetSocketAddress(ns1nn2Hostname, rpcPort);
|
||||||
|
private String ns2;
|
||||||
|
private String ns2nn1Hostname = "router1.foo.bar";
|
||||||
|
private InetSocketAddress ns2nn1 =
|
||||||
|
new InetSocketAddress(ns2nn1Hostname, rpcPort);
|
||||||
|
private String ns2nn2Hostname = "router2.foo.bar";
|
||||||
|
private InetSocketAddress ns2nn2 =
|
||||||
|
new InetSocketAddress(ns2nn2Hostname, rpcPort);
|
||||||
|
private String ns2nn3Hostname = "router3.foo.bar";
|
||||||
|
private InetSocketAddress ns2nn3 =
|
||||||
|
new InetSocketAddress(ns2nn3Hostname, rpcPort);
|
||||||
|
private static final int NUM_ITERATIONS = 50;
|
||||||
|
|
||||||
|
@BeforeClass
|
||||||
|
public static void setupClass() throws Exception {
|
||||||
|
GenericTestUtils.setLogLevel(RequestHedgingProxyProvider.LOG, Level.TRACE);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Before
|
||||||
|
public void setup() throws URISyntaxException {
|
||||||
|
ns1 = "mycluster-1-" + Time.monotonicNow();
|
||||||
|
ns1Uri = new URI("hdfs://" + ns1);
|
||||||
|
conf = new Configuration();
|
||||||
|
conf.set(
|
||||||
|
HdfsClientConfigKeys.DFS_HA_NAMENODES_KEY_PREFIX + "." + ns1,
|
||||||
|
"nn1,nn2,nn3");
|
||||||
|
conf.set(
|
||||||
|
HdfsClientConfigKeys.DFS_NAMENODE_RPC_ADDRESS_KEY + "." + ns1 + ".nn1",
|
||||||
|
ns1nn1Hostname + ":" + rpcPort);
|
||||||
|
conf.set(
|
||||||
|
HdfsClientConfigKeys.DFS_NAMENODE_RPC_ADDRESS_KEY + "." + ns1 + ".nn2",
|
||||||
|
ns1nn2Hostname + ":" + rpcPort);
|
||||||
|
conf.set(
|
||||||
|
HdfsClientConfigKeys.Failover.PROXY_PROVIDER_KEY_PREFIX + "." + ns1,
|
||||||
|
ConfiguredFailoverProxyProvider.class.getName());
|
||||||
|
conf.setBoolean(
|
||||||
|
HdfsClientConfigKeys.Failover.RANDOM_ORDER + "." + ns1,
|
||||||
|
false);
|
||||||
|
|
||||||
|
ns2 = "myroutercluster-2-" + Time.monotonicNow();
|
||||||
|
ns2Uri = new URI("hdfs://" + ns2);
|
||||||
|
conf.set(
|
||||||
|
HdfsClientConfigKeys.DFS_HA_NAMENODES_KEY_PREFIX + "." + ns2,
|
||||||
|
"nn1,nn2,nn3");
|
||||||
|
conf.set(
|
||||||
|
HdfsClientConfigKeys.DFS_NAMENODE_RPC_ADDRESS_KEY + "." + ns2 + ".nn1",
|
||||||
|
ns2nn1Hostname + ":" + rpcPort);
|
||||||
|
conf.set(
|
||||||
|
HdfsClientConfigKeys.DFS_NAMENODE_RPC_ADDRESS_KEY + "." + ns2 + ".nn2",
|
||||||
|
ns2nn2Hostname + ":" + rpcPort);
|
||||||
|
conf.set(
|
||||||
|
HdfsClientConfigKeys.DFS_NAMENODE_RPC_ADDRESS_KEY + "." + ns2 + ".nn3",
|
||||||
|
ns2nn3Hostname + ":" + rpcPort);
|
||||||
|
conf.set(
|
||||||
|
HdfsClientConfigKeys.Failover.PROXY_PROVIDER_KEY_PREFIX + "." + ns2,
|
||||||
|
ConfiguredFailoverProxyProvider.class.getName());
|
||||||
|
conf.setBoolean(
|
||||||
|
HdfsClientConfigKeys.Failover.RANDOM_ORDER + "." + ns2,
|
||||||
|
true);
|
||||||
|
|
||||||
|
conf.set(HdfsClientConfigKeys.DFS_NAMESERVICES, ns1 + "," + ns2);
|
||||||
|
conf.set("fs.defaultFS", "hdfs://" + ns1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests getProxy with random.order configuration set to false.
|
||||||
|
* This expects the proxy order to be consistent every time a new
|
||||||
|
* ConfiguredFailoverProxyProvider is created.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testNonRandomGetProxy() throws Exception {
|
||||||
|
final AtomicInteger nn1Count = new AtomicInteger(0);
|
||||||
|
final AtomicInteger nn2Count = new AtomicInteger(0);
|
||||||
|
|
||||||
|
Map<InetSocketAddress, ClientProtocol> proxyMap = new HashMap<>();
|
||||||
|
|
||||||
|
final ClientProtocol nn1Mock = mock(ClientProtocol.class);
|
||||||
|
when(nn1Mock.getStats()).thenAnswer(createAnswer(nn1Count, 1));
|
||||||
|
proxyMap.put(ns1nn1, nn1Mock);
|
||||||
|
|
||||||
|
final ClientProtocol nn2Mock = mock(ClientProtocol.class);
|
||||||
|
when(nn2Mock.getStats()).thenAnswer(createAnswer(nn2Count, 2));
|
||||||
|
proxyMap.put(ns1nn2, nn2Mock);
|
||||||
|
|
||||||
|
ConfiguredFailoverProxyProvider<ClientProtocol> provider1 =
|
||||||
|
new ConfiguredFailoverProxyProvider<>(conf, ns1Uri,
|
||||||
|
ClientProtocol.class, createFactory(proxyMap));
|
||||||
|
ClientProtocol proxy1 = provider1.getProxy().proxy;
|
||||||
|
proxy1.getStats();
|
||||||
|
assertEquals(1, nn1Count.get());
|
||||||
|
assertEquals(0, nn2Count.get());
|
||||||
|
proxy1.getStats();
|
||||||
|
assertEquals(2, nn1Count.get());
|
||||||
|
assertEquals(0, nn2Count.get());
|
||||||
|
nn1Count.set(0);
|
||||||
|
nn2Count.set(0);
|
||||||
|
|
||||||
|
for (int i = 0; i < NUM_ITERATIONS; i++) {
|
||||||
|
ConfiguredFailoverProxyProvider<ClientProtocol> provider2 =
|
||||||
|
new ConfiguredFailoverProxyProvider<>(conf, ns1Uri,
|
||||||
|
ClientProtocol.class, createFactory(proxyMap));
|
||||||
|
ClientProtocol proxy2 = provider2.getProxy().proxy;
|
||||||
|
proxy2.getStats();
|
||||||
|
}
|
||||||
|
assertEquals(NUM_ITERATIONS, nn1Count.get());
|
||||||
|
assertEquals(0, nn2Count.get());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests getProxy with random.order configuration set to true.
|
||||||
|
* This expects the proxy order to be random every time a new
|
||||||
|
* ConfiguredFailoverProxyProvider is created.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testRandomGetProxy() throws Exception {
|
||||||
|
final AtomicInteger nn1Count = new AtomicInteger(0);
|
||||||
|
final AtomicInteger nn2Count = new AtomicInteger(0);
|
||||||
|
final AtomicInteger nn3Count = new AtomicInteger(0);
|
||||||
|
|
||||||
|
Map<InetSocketAddress, ClientProtocol> proxyMap = new HashMap<>();
|
||||||
|
|
||||||
|
final ClientProtocol nn1Mock = mock(ClientProtocol.class);
|
||||||
|
when(nn1Mock.getStats()).thenAnswer(createAnswer(nn1Count, 1));
|
||||||
|
proxyMap.put(ns2nn1, nn1Mock);
|
||||||
|
|
||||||
|
final ClientProtocol nn2Mock = mock(ClientProtocol.class);
|
||||||
|
when(nn2Mock.getStats()).thenAnswer(createAnswer(nn2Count, 2));
|
||||||
|
proxyMap.put(ns2nn2, nn2Mock);
|
||||||
|
|
||||||
|
final ClientProtocol nn3Mock = mock(ClientProtocol.class);
|
||||||
|
when(nn3Mock.getStats()).thenAnswer(createAnswer(nn3Count, 3));
|
||||||
|
proxyMap.put(ns2nn3, nn3Mock);
|
||||||
|
|
||||||
|
|
||||||
|
for (int i = 0; i < NUM_ITERATIONS; i++) {
|
||||||
|
ConfiguredFailoverProxyProvider<ClientProtocol> provider =
|
||||||
|
new ConfiguredFailoverProxyProvider<>(conf, ns2Uri,
|
||||||
|
ClientProtocol.class, createFactory(proxyMap));
|
||||||
|
ClientProtocol proxy = provider.getProxy().proxy;
|
||||||
|
proxy.getStats();
|
||||||
|
}
|
||||||
|
|
||||||
|
assertTrue(nn1Count.get() < NUM_ITERATIONS && nn1Count.get() > 0);
|
||||||
|
assertTrue(nn2Count.get() < NUM_ITERATIONS && nn2Count.get() > 0);
|
||||||
|
assertTrue(nn3Count.get() < NUM_ITERATIONS && nn3Count.get() > 0);
|
||||||
|
assertEquals(NUM_ITERATIONS,
|
||||||
|
nn1Count.get() + nn2Count.get() + nn3Count.get());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* createAnswer creates an Answer for using with the ClientProtocol mocks.
|
||||||
|
* @param counter counter to increment
|
||||||
|
* @param retVal return value from answer
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
|
private Answer<long[]> createAnswer(final AtomicInteger counter,
|
||||||
|
final long retVal) {
|
||||||
|
return new Answer<long[]>() {
|
||||||
|
@Override
|
||||||
|
public long[] answer(InvocationOnMock invocation) throws Throwable {
|
||||||
|
counter.incrementAndGet();
|
||||||
|
return new long[]{retVal};
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* createFactory returns a HAProxyFactory for tests.
|
||||||
|
* This uses a map of name node address to ClientProtocol to route calls to
|
||||||
|
* different ClientProtocol objects. The tests could create ClientProtocol
|
||||||
|
* mocks and create name node mappings to use with
|
||||||
|
* ConfiguredFailoverProxyProvider.
|
||||||
|
*/
|
||||||
|
private HAProxyFactory<ClientProtocol> createFactory(
|
||||||
|
final Map<InetSocketAddress, ClientProtocol> proxies) {
|
||||||
|
final Map<InetSocketAddress, ClientProtocol> proxyMap = proxies;
|
||||||
|
return new HAProxyFactory<ClientProtocol>() {
|
||||||
|
@Override
|
||||||
|
public ClientProtocol createProxy(Configuration cfg,
|
||||||
|
InetSocketAddress nnAddr, Class<ClientProtocol> xface,
|
||||||
|
UserGroupInformation ugi, boolean withRetries,
|
||||||
|
AtomicBoolean fallbackToSimpleAuth) throws IOException {
|
||||||
|
if (proxyMap.containsKey(nnAddr)) {
|
||||||
|
return proxyMap.get(nnAddr);
|
||||||
|
} else {
|
||||||
|
throw new IOException("Name node address not found");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public ClientProtocol createProxy(Configuration cfg,
|
||||||
|
InetSocketAddress nnAddr, Class<ClientProtocol> xface,
|
||||||
|
UserGroupInformation ugi, boolean withRetries) throws IOException {
|
||||||
|
if (proxyMap.containsKey(nnAddr)) {
|
||||||
|
return proxyMap.get(nnAddr);
|
||||||
|
} else {
|
||||||
|
throw new IOException("Name node address not found");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
|
@ -3631,6 +3631,18 @@
|
||||||
</description>
|
</description>
|
||||||
</property>
|
</property>
|
||||||
|
|
||||||
|
<property>
|
||||||
|
<name>dfs.client.failover.random.order</name>
|
||||||
|
<value>false</value>
|
||||||
|
<description>
|
||||||
|
Determines if the failover proxies are picked in random order instead of the
|
||||||
|
configured order. The prefix can be used with an optional nameservice ID
|
||||||
|
(of form dfs.client.failover.random.order[.nameservice]) in case multiple
|
||||||
|
nameservices exist and random order should be enabled for specific
|
||||||
|
nameservices.
|
||||||
|
</description>
|
||||||
|
</property>
|
||||||
|
|
||||||
<property>
|
<property>
|
||||||
<name>dfs.client.key.provider.cache.expiry</name>
|
<name>dfs.client.key.provider.cache.expiry</name>
|
||||||
<value>864000000</value>
|
<value>864000000</value>
|
||||||
|
|
|
@ -41,6 +41,7 @@ public class TestHdfsConfigFields extends TestConfigurationFieldsBase {
|
||||||
public void initializeMemberVariables() {
|
public void initializeMemberVariables() {
|
||||||
xmlFilename = new String("hdfs-default.xml");
|
xmlFilename = new String("hdfs-default.xml");
|
||||||
configurationClasses = new Class[] { HdfsClientConfigKeys.class,
|
configurationClasses = new Class[] { HdfsClientConfigKeys.class,
|
||||||
|
HdfsClientConfigKeys.Failover.class,
|
||||||
HdfsClientConfigKeys.StripedRead.class, DFSConfigKeys.class,
|
HdfsClientConfigKeys.StripedRead.class, DFSConfigKeys.class,
|
||||||
HdfsClientConfigKeys.BlockWrite.ReplaceDatanodeOnFailure.class };
|
HdfsClientConfigKeys.BlockWrite.ReplaceDatanodeOnFailure.class };
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue