reqs = new ArrayList<>();
+ ResourceRequest nodeReq = ResourceRequest.newInstance(
+ Priority.newInstance(0), nm2.getNodeId().getHost(),
+ Resource.newInstance(200, 1), 1, true);
+ ResourceRequest rackReq = ResourceRequest.newInstance(
+ Priority.newInstance(0), "/default-rack",
+ Resource.newInstance(200, 1), 1, true);
+ ResourceRequest anyReq = ResourceRequest.newInstance(
+ Priority.newInstance(0), ResourceRequest.ANY,
+ Resource.newInstance(200, 1), 1, true);
+ reqs.add(anyReq);
+ reqs.add(rackReq);
+ reqs.add(nodeReq);
+ RMApp app = rm.submitApp(reqs);
+
+ MockAM am1 = MockRM.launchAndRegisterAM(app, rm, nm2);
+ ContainerId amContainerId =
+ ContainerId.newContainerId(am1.getApplicationAttemptId(), 1);
+ RMContainer rmContainer = scheduler.getRMContainer(amContainerId);
+ NodeId nodeWhereAMRan = rmContainer.getAllocatedNode();
+ Assert.assertEquals(nm2.getNodeId(), nodeWhereAMRan);
+
+ // Set the exist status to INVALID so that we can verify that the system
+ // automatically blacklisting the node
+ makeAMContainerExit(rm, amContainerId, nm2, ContainerExitStatus.INVALID);
+
+ // restart the am
+ RMAppAttempt attempt = MockRM.waitForAttemptScheduled(app, rm);
+ System.out.println("New AppAttempt launched " + attempt.getAppAttemptId());
+
+ nm2.nodeHeartbeat(true);
+ nm1.nodeHeartbeat(true);
+ nm3.nodeHeartbeat(true);
+ nm4.nodeHeartbeat(true);
+ nm5.nodeHeartbeat(true);
+ dispatcher.await();
+
+ // Now the AM container should be allocated
+ MockRM.waitForState(attempt, RMAppAttemptState.ALLOCATED, 20000);
+
+ MockAM am2 = rm.sendAMLaunched(attempt.getAppAttemptId());
+ rm.waitForState(attempt.getAppAttemptId(), RMAppAttemptState.LAUNCHED);
+ amContainerId =
+ ContainerId.newContainerId(am2.getApplicationAttemptId(), 1);
+ rmContainer = scheduler.getRMContainer(amContainerId);
+ nodeWhereAMRan = rmContainer.getAllocatedNode();
+
+ // The second AM should be on a different node because the relaxed locality
+ // made the app schedulable on other nodes and nm2 is blacklisted
+ System.out.println("AM ran on " + nodeWhereAMRan);
+ Assert.assertNotEquals(nm2.getNodeId(), nodeWhereAMRan);
+
+ am2.registerAppAttempt();
+ rm.waitForState(app.getApplicationId(), RMAppState.RUNNING);
+ }
+
@Test(timeout = 100000)
public void testNoBlacklistingForNonSystemErrors() throws Exception {
diff --git a/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/TestRMServerUtils.java b/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/TestRMServerUtils.java
new file mode 100644
index 00000000000..078b8fd3290
--- /dev/null
+++ b/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/TestRMServerUtils.java
@@ -0,0 +1,297 @@
+/**
+ * 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.yarn.server.resourcemanager;
+
+import org.apache.hadoop.yarn.api.records.NodeId;
+import org.apache.hadoop.yarn.api.records.Priority;
+import org.apache.hadoop.yarn.api.records.Resource;
+import org.apache.hadoop.yarn.api.records.ResourceRequest;
+import org.apache.hadoop.yarn.conf.YarnConfiguration;
+import org.apache.hadoop.yarn.server.resourcemanager.nodelabels.RMNodeLabelsManager;
+import org.apache.hadoop.yarn.server.resourcemanager.scheduler.ResourceScheduler;
+import org.junit.Assert;
+import org.junit.Test;
+import org.mockito.Mockito;
+
+import java.util.ArrayList;
+import java.util.Collections;
+import java.util.HashMap;
+import java.util.HashSet;
+import java.util.List;
+import java.util.Map;
+import java.util.Set;
+
+public class TestRMServerUtils {
+ @Test
+ public void testGetApplicableNodeCountForAMLocality() throws Exception {
+ List rack1Nodes = new ArrayList<>();
+ for (int i = 0; i < 29; i++) {
+ rack1Nodes.add(NodeId.newInstance("host" + i, 1234));
+ }
+ NodeId node1 = NodeId.newInstance("node1", 1234);
+ NodeId node2 = NodeId.newInstance("node2", 1234);
+ rack1Nodes.add(node2);
+
+ YarnConfiguration conf = new YarnConfiguration();
+ conf.setBoolean(YarnConfiguration.NODE_LABELS_ENABLED, false);
+ ResourceScheduler scheduler = Mockito.mock(ResourceScheduler.class);
+ Mockito.when(scheduler.getNumClusterNodes()).thenReturn(100);
+ Mockito.when(scheduler.getNodeIds("/rack1")).thenReturn(rack1Nodes);
+ Mockito.when(scheduler.getNodeIds("node1"))
+ .thenReturn(Collections.singletonList(node1));
+ Mockito.when(scheduler.getNodeIds("node2"))
+ .thenReturn(Collections.singletonList(node2));
+ RMContext rmContext = Mockito.mock(RMContext.class);
+ Mockito.when(rmContext.getScheduler()).thenReturn(scheduler);
+
+ ResourceRequest anyReq = createResourceRequest(ResourceRequest.ANY,
+ true, null);
+ List reqs = new ArrayList<>();
+ reqs.add(anyReq);
+ Assert.assertEquals(100,
+ RMServerUtils.getApplicableNodeCountForAM(rmContext, conf, reqs));
+
+ ResourceRequest rackReq = createResourceRequest("/rack1", true, null);
+ reqs.add(rackReq);
+ Assert.assertEquals(30,
+ RMServerUtils.getApplicableNodeCountForAM(rmContext, conf, reqs));
+ anyReq.setRelaxLocality(false);
+ Assert.assertEquals(30,
+ RMServerUtils.getApplicableNodeCountForAM(rmContext, conf, reqs));
+ rackReq.setRelaxLocality(false);
+ Assert.assertEquals(100,
+ RMServerUtils.getApplicableNodeCountForAM(rmContext, conf, reqs));
+
+ ResourceRequest node1Req = createResourceRequest("node1", false, null);
+ reqs.add(node1Req);
+ Assert.assertEquals(100,
+ RMServerUtils.getApplicableNodeCountForAM(rmContext, conf, reqs));
+ node1Req.setRelaxLocality(true);
+ Assert.assertEquals(1,
+ RMServerUtils.getApplicableNodeCountForAM(rmContext, conf, reqs));
+ rackReq.setRelaxLocality(true);
+ Assert.assertEquals(31,
+ RMServerUtils.getApplicableNodeCountForAM(rmContext, conf, reqs));
+
+ ResourceRequest node2Req = createResourceRequest("node2", false, null);
+ reqs.add(node2Req);
+ Assert.assertEquals(31,
+ RMServerUtils.getApplicableNodeCountForAM(rmContext, conf, reqs));
+ node2Req.setRelaxLocality(true);
+ Assert.assertEquals(31,
+ RMServerUtils.getApplicableNodeCountForAM(rmContext, conf, reqs));
+ rackReq.setRelaxLocality(false);
+ Assert.assertEquals(2,
+ RMServerUtils.getApplicableNodeCountForAM(rmContext, conf, reqs));
+ node1Req.setRelaxLocality(false);
+ Assert.assertEquals(1,
+ RMServerUtils.getApplicableNodeCountForAM(rmContext, conf, reqs));
+ node2Req.setRelaxLocality(false);
+ Assert.assertEquals(100,
+ RMServerUtils.getApplicableNodeCountForAM(rmContext, conf, reqs));
+ }
+
+ @Test
+ public void testGetApplicableNodeCountForAMLabels() throws Exception {
+ Set noLabelNodes = new HashSet<>();
+ for (int i = 0; i < 80; i++) {
+ noLabelNodes.add(NodeId.newInstance("host" + i, 1234));
+ }
+ Set label1Nodes = new HashSet<>();
+ for (int i = 80; i < 90; i++) {
+ label1Nodes.add(NodeId.newInstance("host" + i, 1234));
+ }
+ label1Nodes.add(NodeId.newInstance("host101", 0));
+ label1Nodes.add(NodeId.newInstance("host102", 0));
+ Map> label1NodesMap = new HashMap<>();
+ label1NodesMap.put("label1", label1Nodes);
+
+ YarnConfiguration conf = new YarnConfiguration();
+ conf.setBoolean(YarnConfiguration.NODE_LABELS_ENABLED, true);
+ ResourceScheduler scheduler = Mockito.mock(ResourceScheduler.class);
+ Mockito.when(scheduler.getNumClusterNodes()).thenReturn(100);
+ RMContext rmContext = Mockito.mock(RMContext.class);
+ Mockito.when(rmContext.getScheduler()).thenReturn(scheduler);
+ RMNodeLabelsManager labMan = Mockito.mock(RMNodeLabelsManager.class);
+ Mockito.when(labMan.getNodesWithoutALabel()).thenReturn(noLabelNodes);
+ Mockito.when(labMan.getLabelsToNodes(Collections.singleton("label1")))
+ .thenReturn(label1NodesMap);
+ Mockito.when(rmContext.getNodeLabelManager()).thenReturn(labMan);
+
+ ResourceRequest anyReq = createResourceRequest(ResourceRequest.ANY,
+ true, null);
+ List reqs = new ArrayList<>();
+ reqs.add(anyReq);
+ Assert.assertEquals(80,
+ RMServerUtils.getApplicableNodeCountForAM(rmContext, conf, reqs));
+ anyReq.setNodeLabelExpression("label1");
+ Assert.assertEquals(10,
+ RMServerUtils.getApplicableNodeCountForAM(rmContext, conf, reqs));
+ }
+
+ @Test
+ public void testGetApplicableNodeCountForAMLocalityAndLabels()
+ throws Exception {
+ List rack1Nodes = new ArrayList<>();
+ for (int i = 0; i < 29; i++) {
+ rack1Nodes.add(NodeId.newInstance("host" + i, 1234));
+ }
+ NodeId node1 = NodeId.newInstance("node1", 1234);
+ NodeId node2 = NodeId.newInstance("node2", 1234);
+ rack1Nodes.add(node2);
+ Set noLabelNodes = new HashSet<>();
+ for (int i = 0; i < 19; i++) {
+ noLabelNodes.add(rack1Nodes.get(i));
+ }
+ noLabelNodes.add(node2);
+ for (int i = 29; i < 89; i++) {
+ noLabelNodes.add(NodeId.newInstance("host" + i, 1234));
+ }
+ Set label1Nodes = new HashSet<>();
+ label1Nodes.add(node1);
+ for (int i = 89; i < 93; i++) {
+ label1Nodes.add(NodeId.newInstance("host" + i, 1234));
+ }
+ for (int i = 19; i < 29; i++) {
+ label1Nodes.add(rack1Nodes.get(i));
+ }
+ label1Nodes.add(NodeId.newInstance("host101", 0));
+ label1Nodes.add(NodeId.newInstance("host102", 0));
+ Map> label1NodesMap = new HashMap<>();
+ label1NodesMap.put("label1", label1Nodes);
+
+ YarnConfiguration conf = new YarnConfiguration();
+ conf.setBoolean(YarnConfiguration.NODE_LABELS_ENABLED, true);
+ ResourceScheduler scheduler = Mockito.mock(ResourceScheduler.class);
+ Mockito.when(scheduler.getNumClusterNodes()).thenReturn(100);
+ Mockito.when(scheduler.getNodeIds("/rack1")).thenReturn(rack1Nodes);
+ Mockito.when(scheduler.getNodeIds("node1"))
+ .thenReturn(Collections.singletonList(node1));
+ Mockito.when(scheduler.getNodeIds("node2"))
+ .thenReturn(Collections.singletonList(node2));
+ RMContext rmContext = Mockito.mock(RMContext.class);
+ Mockito.when(rmContext.getScheduler()).thenReturn(scheduler);
+ RMNodeLabelsManager labMan = Mockito.mock(RMNodeLabelsManager.class);
+ Mockito.when(labMan.getNodesWithoutALabel()).thenReturn(noLabelNodes);
+ Mockito.when(labMan.getLabelsToNodes(Collections.singleton("label1")))
+ .thenReturn(label1NodesMap);
+ Mockito.when(rmContext.getNodeLabelManager()).thenReturn(labMan);
+
+ ResourceRequest anyReq = createResourceRequest(ResourceRequest.ANY,
+ true, null);
+ List reqs = new ArrayList<>();
+ reqs.add(anyReq);
+ Assert.assertEquals(80,
+ RMServerUtils.getApplicableNodeCountForAM(rmContext, conf, reqs));
+
+ ResourceRequest rackReq = createResourceRequest("/rack1", true, null);
+ reqs.add(rackReq);
+ Assert.assertEquals(20,
+ RMServerUtils.getApplicableNodeCountForAM(rmContext, conf, reqs));
+ anyReq.setRelaxLocality(false);
+ Assert.assertEquals(20,
+ RMServerUtils.getApplicableNodeCountForAM(rmContext, conf, reqs));
+ rackReq.setRelaxLocality(false);
+ Assert.assertEquals(80,
+ RMServerUtils.getApplicableNodeCountForAM(rmContext, conf, reqs));
+
+ ResourceRequest node1Req = createResourceRequest("node1", false, null);
+ reqs.add(node1Req);
+ Assert.assertEquals(80,
+ RMServerUtils.getApplicableNodeCountForAM(rmContext, conf, reqs));
+ node1Req.setRelaxLocality(true);
+ Assert.assertEquals(0,
+ RMServerUtils.getApplicableNodeCountForAM(rmContext, conf, reqs));
+ rackReq.setRelaxLocality(true);
+ Assert.assertEquals(20,
+ RMServerUtils.getApplicableNodeCountForAM(rmContext, conf, reqs));
+
+ ResourceRequest node2Req = createResourceRequest("node2", false, null);
+ reqs.add(node2Req);
+ Assert.assertEquals(20,
+ RMServerUtils.getApplicableNodeCountForAM(rmContext, conf, reqs));
+ node2Req.setRelaxLocality(true);
+ Assert.assertEquals(20,
+ RMServerUtils.getApplicableNodeCountForAM(rmContext, conf, reqs));
+ rackReq.setRelaxLocality(false);
+ Assert.assertEquals(1,
+ RMServerUtils.getApplicableNodeCountForAM(rmContext, conf, reqs));
+ node1Req.setRelaxLocality(false);
+ Assert.assertEquals(1,
+ RMServerUtils.getApplicableNodeCountForAM(rmContext, conf, reqs));
+ node2Req.setRelaxLocality(false);
+ Assert.assertEquals(80,
+ RMServerUtils.getApplicableNodeCountForAM(rmContext, conf, reqs));
+
+ anyReq.setNodeLabelExpression("label1");
+ rackReq.setNodeLabelExpression("label1");
+ node1Req.setNodeLabelExpression("label1");
+ node2Req.setNodeLabelExpression("label1");
+ anyReq.setRelaxLocality(true);
+ reqs = new ArrayList<>();
+ reqs.add(anyReq);
+ Assert.assertEquals(15,
+ RMServerUtils.getApplicableNodeCountForAM(rmContext, conf, reqs));
+
+ rackReq.setRelaxLocality(true);
+ reqs.add(rackReq);
+ Assert.assertEquals(10,
+ RMServerUtils.getApplicableNodeCountForAM(rmContext, conf, reqs));
+ anyReq.setRelaxLocality(false);
+ Assert.assertEquals(10,
+ RMServerUtils.getApplicableNodeCountForAM(rmContext, conf, reqs));
+ rackReq.setRelaxLocality(false);
+ Assert.assertEquals(15,
+ RMServerUtils.getApplicableNodeCountForAM(rmContext, conf, reqs));
+
+ node1Req.setRelaxLocality(false);
+ reqs.add(node1Req);
+ Assert.assertEquals(15,
+ RMServerUtils.getApplicableNodeCountForAM(rmContext, conf, reqs));
+ node1Req.setRelaxLocality(true);
+ Assert.assertEquals(1,
+ RMServerUtils.getApplicableNodeCountForAM(rmContext, conf, reqs));
+ rackReq.setRelaxLocality(true);
+ Assert.assertEquals(11,
+ RMServerUtils.getApplicableNodeCountForAM(rmContext, conf, reqs));
+
+ node2Req.setRelaxLocality(false);
+ reqs.add(node2Req);
+ Assert.assertEquals(11,
+ RMServerUtils.getApplicableNodeCountForAM(rmContext, conf, reqs));
+ node2Req.setRelaxLocality(true);
+ Assert.assertEquals(11,
+ RMServerUtils.getApplicableNodeCountForAM(rmContext, conf, reqs));
+ rackReq.setRelaxLocality(false);
+ Assert.assertEquals(1,
+ RMServerUtils.getApplicableNodeCountForAM(rmContext, conf, reqs));
+ node1Req.setRelaxLocality(false);
+ Assert.assertEquals(0,
+ RMServerUtils.getApplicableNodeCountForAM(rmContext, conf, reqs));
+ node2Req.setRelaxLocality(false);
+ Assert.assertEquals(15,
+ RMServerUtils.getApplicableNodeCountForAM(rmContext, conf, reqs));
+ }
+
+ private ResourceRequest createResourceRequest(String resource,
+ boolean relaxLocality, String nodeLabel) {
+ return ResourceRequest.newInstance(Priority.newInstance(0),
+ resource, Resource.newInstance(1, 1), 1, relaxLocality, nodeLabel);
+ }
+}
diff --git a/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/applicationsmanager/MockAsm.java b/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/applicationsmanager/MockAsm.java
index 3d3aeaa41b7..b9adf78e32d 100644
--- a/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/applicationsmanager/MockAsm.java
+++ b/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/applicationsmanager/MockAsm.java
@@ -57,7 +57,7 @@ import com.google.common.collect.Lists;
public abstract class MockAsm extends MockApps {
public static class ApplicationBase implements RMApp {
- ResourceRequest amReq;
+ List amReqs;
@Override
public String getUser() {
throw new UnsupportedOperationException("Not supported yet.");
@@ -192,8 +192,8 @@ public abstract class MockAsm extends MockApps {
}
@Override
- public ResourceRequest getAMResourceRequest() {
- return this.amReq;
+ public List getAMResourceRequests() {
+ return this.amReqs;
}
@Override
diff --git a/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/metrics/TestSystemMetricsPublisher.java b/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/metrics/TestSystemMetricsPublisher.java
index f6290ebda4f..56dfe6eb0f0 100644
--- a/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/metrics/TestSystemMetricsPublisher.java
+++ b/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/metrics/TestSystemMetricsPublisher.java
@@ -525,7 +525,8 @@ public class TestSystemMetricsPublisher {
when(app.getAppNodeLabelExpression()).thenCallRealMethod();
ResourceRequest amReq = mock(ResourceRequest.class);
when(amReq.getNodeLabelExpression()).thenReturn("high-mem");
- when(app.getAMResourceRequest()).thenReturn(amReq);
+ when(app.getAMResourceRequests())
+ .thenReturn(Collections.singletonList(amReq));
when(app.getAmNodeLabelExpression()).thenCallRealMethod();
when(app.getApplicationPriority()).thenReturn(Priority.newInstance(10));
when(app.getCallerContext())
diff --git a/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/rmapp/MockRMApp.java b/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/rmapp/MockRMApp.java
index bcc4be1ca5b..d3639085977 100644
--- a/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/rmapp/MockRMApp.java
+++ b/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/rmapp/MockRMApp.java
@@ -21,6 +21,7 @@ package org.apache.hadoop.yarn.server.resourcemanager.rmapp;
import java.util.Collection;
import java.util.Collections;
import java.util.LinkedHashMap;
+import java.util.List;
import java.util.Map;
import java.util.Set;
@@ -62,14 +63,14 @@ public class MockRMApp implements RMApp {
StringBuilder diagnostics = new StringBuilder();
RMAppAttempt attempt;
int maxAppAttempts = 1;
- ResourceRequest amReq;
+ List amReqs;
public MockRMApp(int newid, long time, RMAppState newState) {
finish = time;
id = MockApps.newAppID(newid);
state = newState;
- amReq = ResourceRequest.newInstance(Priority.UNDEFINED, "0.0.0.0",
- Resource.newInstance(0, 0), 1);
+ amReqs = Collections.singletonList(ResourceRequest.newInstance(
+ Priority.UNDEFINED, "0.0.0.0", Resource.newInstance(0, 0), 1));
}
public MockRMApp(int newid, long time, RMAppState newState, String userName) {
@@ -276,8 +277,8 @@ public class MockRMApp implements RMApp {
}
@Override
- public ResourceRequest getAMResourceRequest() {
- return this.amReq;
+ public List getAMResourceRequests() {
+ return this.amReqs;
}
@Override
diff --git a/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/rmapp/TestRMAppTransitions.java b/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/rmapp/TestRMAppTransitions.java
index 9977683898f..1ec951023fd 100644
--- a/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/rmapp/TestRMAppTransitions.java
+++ b/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/rmapp/TestRMAppTransitions.java
@@ -30,8 +30,10 @@ import static org.mockito.Mockito.verify;
import java.io.IOException;
import java.nio.ByteBuffer;
+import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
+import java.util.Collections;
import java.util.Map;
import org.apache.commons.logging.Log;
@@ -267,7 +269,8 @@ public class TestRMAppTransitions {
submissionContext.setAMContainerSpec(mock(ContainerLaunchContext.class));
RMApp application = new RMAppImpl(applicationId, rmContext, conf, name,
user, queue, submissionContext, scheduler, masterService,
- System.currentTimeMillis(), "YARN", null, mock(ResourceRequest.class));
+ System.currentTimeMillis(), "YARN", null,
+ new ArrayList());
testAppStartState(applicationId, user, name, queue, application);
this.rmContext.getRMApps().putIfAbsent(application.getApplicationId(),
@@ -1020,9 +1023,9 @@ public class TestRMAppTransitions {
submissionContext.getQueue(), submissionContext, scheduler, null,
appState.getSubmitTime(), submissionContext.getApplicationType(),
submissionContext.getApplicationTags(),
- BuilderUtils.newResourceRequest(
+ Collections.singletonList(BuilderUtils.newResourceRequest(
RMAppAttemptImpl.AM_CONTAINER_PRIORITY, ResourceRequest.ANY,
- submissionContext.getResource(), 1));
+ submissionContext.getResource(), 1)));
Assert.assertEquals(RMAppState.NEW, application.getState());
RMAppEvent recoverEvent =
diff --git a/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/rmapp/attempt/TestRMAppAttemptTransitions.java b/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/rmapp/attempt/TestRMAppAttemptTransitions.java
index ced5bd9b8d8..9a4b6dc7f72 100644
--- a/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/rmapp/attempt/TestRMAppAttemptTransitions.java
+++ b/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/rmapp/attempt/TestRMAppAttemptTransitions.java
@@ -328,9 +328,9 @@ public class TestRMAppAttemptTransitions {
applicationAttempt =
new RMAppAttemptImpl(applicationAttemptId, spyRMContext, scheduler,
masterService, submissionContext, new Configuration(),
- BuilderUtils.newResourceRequest(
+ Collections.singletonList(BuilderUtils.newResourceRequest(
RMAppAttemptImpl.AM_CONTAINER_PRIORITY, ResourceRequest.ANY,
- submissionContext.getResource(), 1), application);
+ submissionContext.getResource(), 1)), application);
when(application.getCurrentAppAttempt()).thenReturn(applicationAttempt);
when(application.getApplicationId()).thenReturn(applicationId);
@@ -1108,9 +1108,9 @@ public class TestRMAppAttemptTransitions {
new RMAppAttemptImpl(applicationAttempt.getAppAttemptId(),
spyRMContext, scheduler,masterService,
submissionContext, myConf,
- BuilderUtils.newResourceRequest(
+ Collections.singletonList(BuilderUtils.newResourceRequest(
RMAppAttemptImpl.AM_CONTAINER_PRIORITY, ResourceRequest.ANY,
- submissionContext.getResource(), 1), application);
+ submissionContext.getResource(), 1)), application);
//submit, schedule and allocate app attempt
myApplicationAttempt.handle(
@@ -1584,9 +1584,9 @@ public class TestRMAppAttemptTransitions {
applicationAttempt =
new RMAppAttemptImpl(applicationAttempt.getAppAttemptId(), spyRMContext,
scheduler, masterService, submissionContext, new Configuration(),
- BuilderUtils.newResourceRequest(
+ Collections.singletonList(BuilderUtils.newResourceRequest(
RMAppAttemptImpl.AM_CONTAINER_PRIORITY, ResourceRequest.ANY,
- submissionContext.getResource(), 1), application);
+ submissionContext.getResource(), 1)), application);
when(submissionContext.getKeepContainersAcrossApplicationAttempts())
.thenReturn(true);
when(submissionContext.getMaxAppAttempts()).thenReturn(1);
@@ -1645,9 +1645,10 @@ public class TestRMAppAttemptTransitions {
applicationAttempt =
new RMAppAttemptImpl(applicationAttempt.getAppAttemptId(),
spyRMContext, scheduler, masterService, submissionContext,
- new Configuration(), ResourceRequest.newInstance(
- Priority.UNDEFINED, "host1", Resource.newInstance(3333, 1), 3,
- false, "label-expression"), application);
+ new Configuration(), Collections.singletonList(
+ ResourceRequest.newInstance(Priority.UNDEFINED, "host1",
+ Resource.newInstance(3333, 1), 3,
+ false, "label-expression")), application);
new RMAppAttemptImpl.ScheduleTransition().transition(
(RMAppAttemptImpl) applicationAttempt, null);
}
diff --git a/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/scheduler/capacity/TestApplicationLimits.java b/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/scheduler/capacity/TestApplicationLimits.java
index 2884f6708aa..37b0da8d6d8 100644
--- a/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/scheduler/capacity/TestApplicationLimits.java
+++ b/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/scheduler/capacity/TestApplicationLimits.java
@@ -28,6 +28,7 @@ import static org.mockito.Mockito.when;
import java.io.IOException;
import java.util.ArrayList;
+import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
@@ -613,7 +614,8 @@ public class TestApplicationLimits {
ResourceRequest amResourceRequest = mock(ResourceRequest.class);
Resource amResource = Resources.createResource(0, 0);
when(amResourceRequest.getCapability()).thenReturn(amResource);
- when(rmApp.getAMResourceRequest()).thenReturn(amResourceRequest);
+ when(rmApp.getAMResourceRequests()).thenReturn(
+ Collections.singletonList(amResourceRequest));
Mockito.doReturn(rmApp).when(spyApps).get((ApplicationId)Matchers.any());
when(spyRMContext.getRMApps()).thenReturn(spyApps);
RMAppAttempt rmAppAttempt = mock(RMAppAttempt.class);
diff --git a/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/scheduler/capacity/TestApplicationLimitsByPartition.java b/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/scheduler/capacity/TestApplicationLimitsByPartition.java
index 2fa06e84788..547571e3a5b 100644
--- a/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/scheduler/capacity/TestApplicationLimitsByPartition.java
+++ b/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/scheduler/capacity/TestApplicationLimitsByPartition.java
@@ -25,6 +25,7 @@ import static org.mockito.Mockito.when;
import java.io.IOException;
import java.util.ArrayList;
+import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
@@ -639,7 +640,8 @@ public class TestApplicationLimitsByPartition {
ResourceRequest amResourceRequest = mock(ResourceRequest.class);
Resource amResource = Resources.createResource(0, 0);
when(amResourceRequest.getCapability()).thenReturn(amResource);
- when(rmApp.getAMResourceRequest()).thenReturn(amResourceRequest);
+ when(rmApp.getAMResourceRequests()).thenReturn(
+ Collections.singletonList(amResourceRequest));
Mockito.doReturn(rmApp).when(spyApps).get((ApplicationId) Matchers.any());
when(spyRMContext.getRMApps()).thenReturn(spyApps);
RMAppAttempt rmAppAttempt = mock(RMAppAttempt.class);
diff --git a/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/scheduler/capacity/TestCapacityScheduler.java b/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/scheduler/capacity/TestCapacityScheduler.java
index ba64ed799c9..37ebefd7e53 100644
--- a/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/scheduler/capacity/TestCapacityScheduler.java
+++ b/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/scheduler/capacity/TestCapacityScheduler.java
@@ -3203,8 +3203,8 @@ public class TestCapacityScheduler {
RMApp rmApp = rm.submitApp(amMemory, "app-1", "user_0", null, queueName);
assertEquals("RMApp does not containes minimum allocation",
- minAllocResource, rmApp.getAMResourceRequest().getCapability());
-
+ minAllocResource, rmApp.getAMResourceRequests().get(0).getCapability());
+
ResourceScheduler scheduler = rm.getRMContext().getScheduler();
LeafQueue queueA =
(LeafQueue) ((CapacityScheduler) scheduler).getQueue(queueName);
diff --git a/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/scheduler/capacity/TestLeafQueue.java b/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/scheduler/capacity/TestLeafQueue.java
index 50dc92ed897..a59a3cc2f89 100644
--- a/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/scheduler/capacity/TestLeafQueue.java
+++ b/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/scheduler/capacity/TestLeafQueue.java
@@ -153,7 +153,8 @@ public class TestLeafQueue {
amResourceRequest = mock(ResourceRequest.class);
when(amResourceRequest.getCapability()).thenReturn(
Resources.createResource(0, 0));
- when(rmApp.getAMResourceRequest()).thenReturn(amResourceRequest);
+ when(rmApp.getAMResourceRequests()).thenReturn(
+ Collections.singletonList(amResourceRequest));
Mockito.doReturn(rmApp).when(spyApps).get((ApplicationId)Matchers.any());
when(spyRMContext.getRMApps()).thenReturn(spyApps);
diff --git a/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/scheduler/fair/TestFairScheduler.java b/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/scheduler/fair/TestFairScheduler.java
index 670793d36e3..3bbb5771308 100644
--- a/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/scheduler/fair/TestFairScheduler.java
+++ b/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/scheduler/fair/TestFairScheduler.java
@@ -3206,6 +3206,84 @@ public class TestFairScheduler extends FairSchedulerTestBase {
assertEquals(1, app.getLiveContainers().size());
}
+ @Test
+ public void testAMStrictLocalityRack() throws IOException {
+ testAMStrictLocality(false, false);
+ }
+
+ @Test
+ public void testAMStrictLocalityNode() throws IOException {
+ testAMStrictLocality(true, false);
+ }
+
+ @Test
+ public void testAMStrictLocalityRackInvalid() throws IOException {
+ testAMStrictLocality(false, true);
+ }
+
+ @Test
+ public void testAMStrictLocalityNodeInvalid() throws IOException {
+ testAMStrictLocality(true, true);
+ }
+
+ private void testAMStrictLocality(boolean node, boolean invalid)
+ throws IOException {
+ scheduler.init(conf);
+ scheduler.start();
+ scheduler.reinitialize(conf, resourceManager.getRMContext());
+
+ RMNode node1 = MockNodes.newNodeInfo(1, Resources.createResource(1024), 1,
+ "127.0.0.1");
+ NodeAddedSchedulerEvent nodeEvent1 = new NodeAddedSchedulerEvent(node1);
+ scheduler.handle(nodeEvent1);
+
+ RMNode node2 = MockNodes.newNodeInfo(2, Resources.createResource(1024), 2,
+ "127.0.0.2");
+ NodeAddedSchedulerEvent nodeEvent2 = new NodeAddedSchedulerEvent(node2);
+ scheduler.handle(nodeEvent2);
+
+ List reqs = new ArrayList<>();
+ ResourceRequest nodeRequest = createResourceRequest(1024,
+ node2.getHostName(), 1, 1, true);
+ if (node && invalid) {
+ nodeRequest.setResourceName("invalid");
+ }
+ ResourceRequest rackRequest = createResourceRequest(1024,
+ node2.getRackName(), 1, 1, !node);
+ if (!node && invalid) {
+ rackRequest.setResourceName("invalid");
+ }
+ ResourceRequest anyRequest = createResourceRequest(1024,
+ ResourceRequest.ANY, 1, 1, false);
+ reqs.add(anyRequest);
+ reqs.add(rackRequest);
+ if (node) {
+ reqs.add(nodeRequest);
+ }
+
+ ApplicationAttemptId attId1 =
+ createSchedulingRequest("queue1", "user1", reqs);
+
+ scheduler.update();
+
+ NodeUpdateSchedulerEvent node2UpdateEvent =
+ new NodeUpdateSchedulerEvent(node2);
+
+ FSAppAttempt app = scheduler.getSchedulerApp(attId1);
+
+ // node2 should get the container
+ scheduler.handle(node2UpdateEvent);
+ if (invalid) {
+ assertEquals(0, app.getLiveContainers().size());
+ assertEquals(0, scheduler.getNode(node2.getNodeID()).getNumContainers());
+ assertEquals(0, scheduler.getNode(node1.getNodeID()).getNumContainers());
+ } else {
+ assertEquals(1, app.getLiveContainers().size());
+ assertEquals(1, scheduler.getNode(node2.getNodeID()).getNumContainers());
+ assertEquals(0, scheduler.getNode(node1.getNodeID()).getNumContainers());
+ }
+ }
+
/**
* Strict locality requests shouldn't reserve resources on another node.
*/
diff --git a/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/webapp/TestRMWebServicesApps.java b/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/webapp/TestRMWebServicesApps.java
index c1815bf2b33..dadb0753da6 100644
--- a/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/webapp/TestRMWebServicesApps.java
+++ b/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-server/hadoop-yarn-server-resourcemanager/src/test/java/org/apache/hadoop/yarn/server/resourcemanager/webapp/TestRMWebServicesApps.java
@@ -1168,8 +1168,8 @@ public class TestRMWebServicesApps extends JerseyTestBase {
assertEquals(app1.getApplicationId().toString(), appInfo.getAppId());
assertEquals(app1.getName(), appInfo.getName());
assertEquals(app1.createApplicationState(), appInfo.getState());
- assertEquals(app1.getAMResourceRequest().getCapability().getMemorySize(),
- appInfo.getAllocatedMB());
+ assertEquals(app1.getAMResourceRequests().get(0).getCapability()
+ .getMemorySize(), appInfo.getAllocatedMB());
rm.stop();
}
@@ -1378,7 +1378,7 @@ public class TestRMWebServicesApps extends JerseyTestBase {
expectedNumberOfElements++;
appNodeLabelExpression = info.getString("appNodeLabelExpression");
}
- if (app.getAMResourceRequest().getNodeLabelExpression() != null) {
+ if (app.getAMResourceRequests().get(0).getNodeLabelExpression() != null) {
expectedNumberOfElements++;
amNodeLabelExpression = info.getString("amNodeLabelExpression");
}
@@ -1485,7 +1485,7 @@ public class TestRMWebServicesApps extends JerseyTestBase {
app.getApplicationSubmissionContext().getNodeLabelExpression(),
appNodeLabelExpression);
assertEquals("unmanagedApplication doesn't match",
- app.getAMResourceRequest().getNodeLabelExpression(),
+ app.getAMResourceRequests().get(0).getNodeLabelExpression(),
amNodeLabelExpression);
assertEquals("amRPCAddress",
AppInfo.getAmRPCAddressFromRMAppAttempt(app.getCurrentAppAttempt()),
@@ -1512,7 +1512,7 @@ public class TestRMWebServicesApps extends JerseyTestBase {
String nodeLabelExpression, int numContainers, boolean relaxLocality,
int priority, String resourceName, long memory, long vCores,
String executionType, boolean enforceExecutionType) {
- ResourceRequest request = app.getAMResourceRequest();
+ ResourceRequest request = app.getAMResourceRequests().get(0);
assertEquals("nodeLabelExpression doesn't match",
request.getNodeLabelExpression(), nodeLabelExpression);
assertEquals("numContainers doesn't match", request.getNumContainers(),