in pkg/scheduler/objects/application.go [981:1064]
func (sa *Application) tryAllocate(headRoom *resources.Resource, allowPreemption bool, preemptionDelay time.Duration, preemptAttemptsRemaining *int, nodeIterator func() NodeIterator, fullNodeIterator func() NodeIterator, getNodeFn func(string) *Node) *AllocationResult {
sa.Lock()
defer sa.Unlock()
if sa.sortedRequests == nil {
return nil
}
// calculate the users' headroom, includes group check which requires the applicationID
userHeadroom := ugm.GetUserManager().Headroom(sa.queuePath, sa.ApplicationID, sa.user)
// get all the requests from the app sorted in order
for _, request := range sa.sortedRequests {
if request.IsAllocated() {
continue
}
// check if there is a replacement possible
if sa.canReplace(request) {
continue
}
// check if this fits in the users' headroom first, if that fits check the queues' headroom
// NOTE: preemption most likely will not help in this case. The chance that preemption helps is mall
// as the preempted allocation must be for the same user in a different queue in the hierarchy...
if !userHeadroom.FitInMaxUndef(request.GetAllocatedResource()) {
request.LogAllocationFailure(NotEnoughUserQuota, true) // error message MUST be constant!
request.setUserQuotaCheckFailed(userHeadroom)
continue
}
request.setUserQuotaCheckPassed()
request.SetSchedulingAttempted(true)
// resource must fit in headroom otherwise skip the request (unless preemption could help)
if !headRoom.FitInMaxUndef(request.GetAllocatedResource()) {
// attempt preemption
if allowPreemption && *preemptAttemptsRemaining > 0 {
*preemptAttemptsRemaining--
fullIterator := fullNodeIterator()
if fullIterator != nil {
if result, ok := sa.tryPreemption(headRoom, preemptionDelay, request, fullIterator, false); ok {
// preemption occurred, and possibly reservation
return result
}
request.LogAllocationFailure(common.PreemptionDoesNotHelp, true)
}
}
request.LogAllocationFailure(NotEnoughQueueQuota, true) // error message MUST be constant!
request.setHeadroomCheckFailed(headRoom, sa.queuePath)
continue
}
request.setHeadroomCheckPassed(sa.queuePath)
requiredNode := request.GetRequiredNode()
// does request have any constraint to run on specific node?
if requiredNode != "" {
result := sa.tryRequiredNode(request, getNodeFn)
if result != nil {
return result
}
// it did not allocate or reserve: should only happen if the node is not registered yet
// just continue with the next request
continue
}
iterator := nodeIterator()
if iterator != nil {
if result := sa.tryNodes(request, iterator); result != nil {
// have a candidate return it
return result
}
// no nodes qualify, attempt preemption
if allowPreemption && *preemptAttemptsRemaining > 0 {
*preemptAttemptsRemaining--
fullIterator := fullNodeIterator()
if fullIterator != nil {
if result, ok := sa.tryPreemption(headRoom, preemptionDelay, request, fullIterator, true); ok {
// preemption occurred, and possibly reservation
return result
}
}
request.LogAllocationFailure(common.PreemptionDoesNotHelp, true)
}
}
}
// no requests fit, skip to next app
return nil
}