YARN-8233. NPE in CapacityScheduler#tryCommit when handling allocate/reserve proposal whose allocatedOrReservedContainer is null. Contributed by Tao Yang.
This commit is contained in:
parent
3929465c2e
commit
daad077121
|
@ -2627,7 +2627,11 @@ public class CapacityScheduler extends
|
||||||
.getContainersToKill().isEmpty()) {
|
.getContainersToKill().isEmpty()) {
|
||||||
list = new ArrayList<>();
|
list = new ArrayList<>();
|
||||||
for (RMContainer rmContainer : csAssignment.getContainersToKill()) {
|
for (RMContainer rmContainer : csAssignment.getContainersToKill()) {
|
||||||
list.add(getSchedulerContainer(rmContainer, false));
|
SchedulerContainer schedulerContainer =
|
||||||
|
getSchedulerContainer(rmContainer, false);
|
||||||
|
if (schedulerContainer != null) {
|
||||||
|
list.add(schedulerContainer);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2635,10 +2639,16 @@ public class CapacityScheduler extends
|
||||||
if (null == list) {
|
if (null == list) {
|
||||||
list = new ArrayList<>();
|
list = new ArrayList<>();
|
||||||
}
|
}
|
||||||
list.add(
|
SchedulerContainer schedulerContainer =
|
||||||
getSchedulerContainer(csAssignment.getExcessReservation(), false));
|
getSchedulerContainer(csAssignment.getExcessReservation(), false);
|
||||||
|
if (schedulerContainer != null) {
|
||||||
|
list.add(schedulerContainer);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (list != null && list.isEmpty()) {
|
||||||
|
list = null;
|
||||||
|
}
|
||||||
return list;
|
return list;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2723,11 +2733,15 @@ public class CapacityScheduler extends
|
||||||
((RMContainerImpl)rmContainer).setAllocationTags(
|
((RMContainerImpl)rmContainer).setAllocationTags(
|
||||||
new HashSet<>(schedulingRequest.getAllocationTags()));
|
new HashSet<>(schedulingRequest.getAllocationTags()));
|
||||||
|
|
||||||
allocated = new ContainerAllocationProposal<>(
|
SchedulerContainer<FiCaSchedulerApp, FiCaSchedulerNode>
|
||||||
getSchedulerContainer(rmContainer, true),
|
schedulerContainer = getSchedulerContainer(rmContainer, true);
|
||||||
|
if (schedulerContainer == null) {
|
||||||
|
allocated = null;
|
||||||
|
} else {
|
||||||
|
allocated = new ContainerAllocationProposal<>(schedulerContainer,
|
||||||
null, null, NodeType.NODE_LOCAL, NodeType.NODE_LOCAL,
|
null, null, NodeType.NODE_LOCAL, NodeType.NODE_LOCAL,
|
||||||
SchedulingMode.RESPECT_PARTITION_EXCLUSIVITY,
|
SchedulingMode.RESPECT_PARTITION_EXCLUSIVITY, resource);
|
||||||
resource);
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (null != allocated) {
|
if (null != allocated) {
|
||||||
|
@ -2757,35 +2771,51 @@ public class CapacityScheduler extends
|
||||||
csAssignment.getAssignmentInformation().getAllocationDetails();
|
csAssignment.getAssignmentInformation().getAllocationDetails();
|
||||||
if (!allocations.isEmpty()) {
|
if (!allocations.isEmpty()) {
|
||||||
RMContainer rmContainer = allocations.get(0).rmContainer;
|
RMContainer rmContainer = allocations.get(0).rmContainer;
|
||||||
allocated = new ContainerAllocationProposal<>(
|
SchedulerContainer<FiCaSchedulerApp, FiCaSchedulerNode>
|
||||||
getSchedulerContainer(rmContainer, true),
|
schedulerContainer = getSchedulerContainer(rmContainer, true);
|
||||||
|
if (schedulerContainer == null) {
|
||||||
|
allocated = null;
|
||||||
|
// Decrease unconfirmed resource if app is alive
|
||||||
|
FiCaSchedulerApp app = getApplicationAttempt(
|
||||||
|
rmContainer.getApplicationAttemptId());
|
||||||
|
if (app != null) {
|
||||||
|
app.decUnconfirmedRes(rmContainer.getAllocatedResource());
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
allocated = new ContainerAllocationProposal<>(schedulerContainer,
|
||||||
getSchedulerContainersToRelease(csAssignment),
|
getSchedulerContainersToRelease(csAssignment),
|
||||||
getSchedulerContainer(csAssignment.getFulfilledReservedContainer(),
|
getSchedulerContainer(
|
||||||
false), csAssignment.getType(),
|
csAssignment.getFulfilledReservedContainer(), false),
|
||||||
csAssignment.getRequestLocalityType(),
|
csAssignment.getType(), csAssignment.getRequestLocalityType(),
|
||||||
csAssignment.getSchedulingMode() != null ?
|
csAssignment.getSchedulingMode() != null ?
|
||||||
csAssignment.getSchedulingMode() :
|
csAssignment.getSchedulingMode() :
|
||||||
SchedulingMode.RESPECT_PARTITION_EXCLUSIVITY,
|
SchedulingMode.RESPECT_PARTITION_EXCLUSIVITY,
|
||||||
csAssignment.getResource());
|
csAssignment.getResource());
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Reserved something
|
// Reserved something
|
||||||
List<AssignmentInformation.AssignmentDetails> reservation =
|
List<AssignmentInformation.AssignmentDetails> reservation =
|
||||||
csAssignment.getAssignmentInformation().getReservationDetails();
|
csAssignment.getAssignmentInformation().getReservationDetails();
|
||||||
if (!reservation.isEmpty()) {
|
if (!reservation.isEmpty()) {
|
||||||
RMContainer rmContainer = reservation.get(0).rmContainer;
|
RMContainer rmContainer = reservation.get(0).rmContainer;
|
||||||
reserved = new ContainerAllocationProposal<>(
|
SchedulerContainer<FiCaSchedulerApp, FiCaSchedulerNode>
|
||||||
getSchedulerContainer(rmContainer, false),
|
schedulerContainer = getSchedulerContainer(rmContainer, false);
|
||||||
|
if (schedulerContainer == null) {
|
||||||
|
reserved = null;
|
||||||
|
} else {
|
||||||
|
reserved = new ContainerAllocationProposal<>(schedulerContainer,
|
||||||
getSchedulerContainersToRelease(csAssignment),
|
getSchedulerContainersToRelease(csAssignment),
|
||||||
getSchedulerContainer(csAssignment.getFulfilledReservedContainer(),
|
getSchedulerContainer(
|
||||||
false), csAssignment.getType(),
|
csAssignment.getFulfilledReservedContainer(), false),
|
||||||
csAssignment.getRequestLocalityType(),
|
csAssignment.getType(), csAssignment.getRequestLocalityType(),
|
||||||
csAssignment.getSchedulingMode() != null ?
|
csAssignment.getSchedulingMode() != null ?
|
||||||
csAssignment.getSchedulingMode() :
|
csAssignment.getSchedulingMode() :
|
||||||
SchedulingMode.RESPECT_PARTITION_EXCLUSIVITY,
|
SchedulingMode.RESPECT_PARTITION_EXCLUSIVITY,
|
||||||
csAssignment.getResource());
|
csAssignment.getResource());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// When we don't need to allocate/reserve anything, we can feel free to
|
// When we don't need to allocate/reserve anything, we can feel free to
|
||||||
// kill all to-release containers in the request.
|
// kill all to-release containers in the request.
|
||||||
|
|
|
@ -56,8 +56,11 @@ import org.apache.hadoop.yarn.server.resourcemanager.scheduler.common.ContainerA
|
||||||
import org.apache.hadoop.yarn.server.resourcemanager.scheduler.common.ResourceCommitRequest;
|
import org.apache.hadoop.yarn.server.resourcemanager.scheduler.common.ResourceCommitRequest;
|
||||||
import org.apache.hadoop.yarn.server.resourcemanager.scheduler.common.SchedulerContainer;
|
import org.apache.hadoop.yarn.server.resourcemanager.scheduler.common.SchedulerContainer;
|
||||||
import org.apache.hadoop.yarn.server.resourcemanager.scheduler.common.fica.FiCaSchedulerApp;
|
import org.apache.hadoop.yarn.server.resourcemanager.scheduler.common.fica.FiCaSchedulerApp;
|
||||||
|
import org.apache.hadoop.yarn.server.resourcemanager.scheduler.common.fica.FiCaSchedulerNode;
|
||||||
import org.apache.hadoop.yarn.server.resourcemanager.scheduler.event.AppAttemptRemovedSchedulerEvent;
|
import org.apache.hadoop.yarn.server.resourcemanager.scheduler.event.AppAttemptRemovedSchedulerEvent;
|
||||||
import org.apache.hadoop.yarn.server.resourcemanager.scheduler.event.NodeUpdateSchedulerEvent;
|
import org.apache.hadoop.yarn.server.resourcemanager.scheduler.event.NodeUpdateSchedulerEvent;
|
||||||
|
import org.apache.hadoop.yarn.server.resourcemanager.scheduler.placement.CandidateNodeSet;
|
||||||
|
import org.apache.hadoop.yarn.server.resourcemanager.scheduler.placement.SimpleCandidateNodeSet;
|
||||||
import org.apache.hadoop.yarn.server.scheduler.SchedulerRequestKey;
|
import org.apache.hadoop.yarn.server.scheduler.SchedulerRequestKey;
|
||||||
import org.apache.hadoop.yarn.util.resource.Resources;
|
import org.apache.hadoop.yarn.util.resource.Resources;
|
||||||
import org.junit.Assert;
|
import org.junit.Assert;
|
||||||
|
@ -843,6 +846,92 @@ public class TestCapacitySchedulerAsyncScheduling {
|
||||||
return new ResourceCommitRequest(allocateProposals, null, null);
|
return new ResourceCommitRequest(allocateProposals, null, null);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test(timeout = 30000)
|
||||||
|
public void testReturnNullWhenGetSchedulerContainer() throws Exception {
|
||||||
|
// disable async-scheduling for simulating complex scenario
|
||||||
|
Configuration disableAsyncConf = new Configuration(conf);
|
||||||
|
disableAsyncConf.setBoolean(
|
||||||
|
CapacitySchedulerConfiguration.SCHEDULE_ASYNCHRONOUSLY_ENABLE, false);
|
||||||
|
|
||||||
|
// init RM & NMs
|
||||||
|
final MockRM rm = new MockRM(disableAsyncConf);
|
||||||
|
rm.start();
|
||||||
|
final MockNM nm1 = rm.registerNode("192.168.0.1:1234", 8 * GB);
|
||||||
|
final MockNM nm2 = rm.registerNode("192.168.0.2:2234", 8 * GB);
|
||||||
|
rm.drainEvents();
|
||||||
|
CapacityScheduler cs = (CapacityScheduler) rm.getRMContext().getScheduler();
|
||||||
|
SchedulerNode sn1 = cs.getSchedulerNode(nm1.getNodeId());
|
||||||
|
RMNode rmNode1 = cs.getNode(nm1.getNodeId()).getRMNode();
|
||||||
|
SchedulerNode sn2 = cs.getSchedulerNode(nm2.getNodeId());
|
||||||
|
|
||||||
|
// launch app1-am on nm1
|
||||||
|
RMApp app1 = rm.submitApp(1 * GB, "app1", "user", null, false, "default",
|
||||||
|
YarnConfiguration.DEFAULT_RM_AM_MAX_ATTEMPTS, null, null, true, true);
|
||||||
|
MockAM am1 = MockRM.launchAndRegisterAM(app1, rm, nm1);
|
||||||
|
|
||||||
|
// app2 asks 1 * 1G container
|
||||||
|
am1.allocate(ImmutableList.of(ResourceRequest
|
||||||
|
.newInstance(Priority.newInstance(0), "*",
|
||||||
|
Resources.createResource(1 * GB), 1)), null);
|
||||||
|
RMContainer amContainer = cs.getRMContainer(
|
||||||
|
ContainerId.newContainerId(am1.getApplicationAttemptId(), 1));
|
||||||
|
|
||||||
|
// spy CapacityScheduler
|
||||||
|
final CapacityScheduler spyCs = Mockito.spy(cs);
|
||||||
|
// hook CapacityScheduler#submitResourceCommitRequest
|
||||||
|
List<CSAssignment> assignmentSnapshots = new ArrayList<>();
|
||||||
|
Mockito.doAnswer(new Answer<Object>() {
|
||||||
|
public Boolean answer(InvocationOnMock invocation) throws Exception {
|
||||||
|
CSAssignment assignment = (CSAssignment) invocation.getArguments()[1];
|
||||||
|
if (cs.getNode(nm1.getNodeId()) != null) {
|
||||||
|
// decommission nm1 for first allocation on nm1
|
||||||
|
cs.getRMContext().getDispatcher().getEventHandler().handle(
|
||||||
|
new RMNodeEvent(nm1.getNodeId(), RMNodeEventType.DECOMMISSION));
|
||||||
|
rm.drainEvents();
|
||||||
|
Assert.assertEquals(NodeState.DECOMMISSIONED, rmNode1.getState());
|
||||||
|
Assert.assertNull(cs.getNode(nm1.getNodeId()));
|
||||||
|
assignmentSnapshots.add(assignment);
|
||||||
|
} else {
|
||||||
|
// add am container on nm1 to containersToKill
|
||||||
|
// for second allocation on nm2
|
||||||
|
assignment.setContainersToKill(ImmutableList.of(amContainer));
|
||||||
|
}
|
||||||
|
// check no NPE in actual submit, before YARN-8233 will throw NPE
|
||||||
|
cs.submitResourceCommitRequest((Resource) invocation.getArguments()[0],
|
||||||
|
assignment);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}).when(spyCs).submitResourceCommitRequest(Mockito.any(Resource.class),
|
||||||
|
Mockito.any(CSAssignment.class));
|
||||||
|
|
||||||
|
// allocation on nm1, test return null when get scheduler container
|
||||||
|
CandidateNodeSet<FiCaSchedulerNode> candidateNodeSet =
|
||||||
|
new SimpleCandidateNodeSet(sn1);
|
||||||
|
spyCs.allocateContainersToNode(candidateNodeSet, false);
|
||||||
|
// make sure unconfirmed resource is decreased correctly
|
||||||
|
Assert.assertTrue(spyCs.getApplicationAttempt(am1.getApplicationAttemptId())
|
||||||
|
.hasPendingResourceRequest(
|
||||||
|
rm.getResourceScheduler().getResourceCalculator(),
|
||||||
|
RMNodeLabelsManager.NO_LABEL,
|
||||||
|
rm.getResourceScheduler().getClusterResource(),
|
||||||
|
SchedulingMode.RESPECT_PARTITION_EXCLUSIVITY));
|
||||||
|
|
||||||
|
// allocation on nm2,
|
||||||
|
// test return null when get scheduler container to release
|
||||||
|
candidateNodeSet =
|
||||||
|
new SimpleCandidateNodeSet(sn2);
|
||||||
|
spyCs.allocateContainersToNode(candidateNodeSet, false);
|
||||||
|
// make sure unconfirmed resource is decreased correctly
|
||||||
|
Assert.assertTrue(spyCs.getApplicationAttempt(am1.getApplicationAttemptId())
|
||||||
|
.hasPendingResourceRequest(
|
||||||
|
rm.getResourceScheduler().getResourceCalculator(),
|
||||||
|
RMNodeLabelsManager.NO_LABEL,
|
||||||
|
rm.getResourceScheduler().getClusterResource(),
|
||||||
|
SchedulingMode.RESPECT_PARTITION_EXCLUSIVITY));
|
||||||
|
|
||||||
|
rm.stop();
|
||||||
|
}
|
||||||
|
|
||||||
private void keepNMHeartbeat(List<MockNM> mockNMs, int interval) {
|
private void keepNMHeartbeat(List<MockNM> mockNMs, int interval) {
|
||||||
if (nmHeartbeatThread != null) {
|
if (nmHeartbeatThread != null) {
|
||||||
nmHeartbeatThread.setShouldStop();
|
nmHeartbeatThread.setShouldStop();
|
||||||
|
|
Loading…
Reference in New Issue