2017-05-23 00:14:38 +00:00
|
|
|
package scheduler
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2017-06-02 23:11:29 +00:00
|
|
|
"sort"
|
|
|
|
"strings"
|
2017-05-23 00:14:38 +00:00
|
|
|
|
|
|
|
"github.com/hashicorp/nomad/nomad/structs"
|
|
|
|
)
|
|
|
|
|
|
|
|
// allocMatrix is a mapping of task groups to their allocation set.
|
|
|
|
type allocMatrix map[string]allocSet
|
|
|
|
|
|
|
|
// newAllocMatrix takes a job and the existing allocations for the job and
|
|
|
|
// creates an allocMatrix
|
|
|
|
func newAllocMatrix(job *structs.Job, allocs []*structs.Allocation) allocMatrix {
|
|
|
|
m := allocMatrix(make(map[string]allocSet))
|
|
|
|
for _, a := range allocs {
|
|
|
|
s, ok := m[a.TaskGroup]
|
|
|
|
if !ok {
|
|
|
|
s = make(map[string]*structs.Allocation)
|
|
|
|
m[a.TaskGroup] = s
|
|
|
|
}
|
|
|
|
s[a.ID] = a
|
|
|
|
}
|
2017-06-01 22:16:24 +00:00
|
|
|
|
|
|
|
if job != nil {
|
|
|
|
for _, tg := range job.TaskGroups {
|
|
|
|
s, ok := m[tg.Name]
|
|
|
|
if !ok {
|
|
|
|
s = make(map[string]*structs.Allocation)
|
|
|
|
m[tg.Name] = s
|
|
|
|
}
|
2017-05-23 00:14:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return m
|
|
|
|
}
|
|
|
|
|
|
|
|
// allocSet is a set of allocations with a series of helper functions defined
|
|
|
|
// that help reconcile state.
|
|
|
|
type allocSet map[string]*structs.Allocation
|
|
|
|
|
|
|
|
// newAllocSet creates an allocation set given a set of allocations
|
|
|
|
func newAllocSet(allocs []*structs.Allocation) allocSet {
|
|
|
|
s := make(map[string]*structs.Allocation, len(allocs))
|
|
|
|
for _, a := range allocs {
|
|
|
|
s[a.ID] = a
|
|
|
|
}
|
|
|
|
return s
|
|
|
|
}
|
|
|
|
|
|
|
|
// GoString provides a human readable view of the set
|
|
|
|
func (a allocSet) GoString() string {
|
|
|
|
if len(a) == 0 {
|
|
|
|
return "[]"
|
|
|
|
}
|
|
|
|
|
|
|
|
start := fmt.Sprintf("len(%d) [\n", len(a))
|
2017-06-02 23:11:29 +00:00
|
|
|
var s []string
|
|
|
|
for k, v := range a {
|
|
|
|
s = append(s, fmt.Sprintf("%q: %v", k, v.Name))
|
|
|
|
}
|
|
|
|
return start + strings.Join(s, "\n") + "]"
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a allocSet) nameSet() map[string]struct{} {
|
|
|
|
names := make(map[string]struct{}, len(a))
|
|
|
|
for _, alloc := range a {
|
|
|
|
names[alloc.Name] = struct{}{}
|
|
|
|
}
|
|
|
|
return names
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a allocSet) nameOrder() []*structs.Allocation {
|
|
|
|
allocs := make([]*structs.Allocation, 0, len(a))
|
|
|
|
for _, alloc := range a {
|
|
|
|
allocs = append(allocs, alloc)
|
2017-05-23 00:14:38 +00:00
|
|
|
}
|
2017-06-02 23:11:29 +00:00
|
|
|
sort.Slice(allocs, func(i, j int) bool {
|
|
|
|
return allocs[i].Index() < allocs[j].Index()
|
|
|
|
})
|
|
|
|
return allocs
|
2017-05-23 00:14:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// difference returns a new allocSet that has all the existing item except those
|
|
|
|
// contained within the other allocation sets
|
|
|
|
func (a allocSet) difference(others ...allocSet) allocSet {
|
|
|
|
diff := make(map[string]*structs.Allocation)
|
|
|
|
OUTER:
|
|
|
|
for k, v := range a {
|
|
|
|
for _, other := range others {
|
|
|
|
if _, ok := other[k]; ok {
|
|
|
|
continue OUTER
|
|
|
|
}
|
|
|
|
}
|
|
|
|
diff[k] = v
|
|
|
|
}
|
|
|
|
return diff
|
|
|
|
}
|
|
|
|
|
2017-05-31 18:34:46 +00:00
|
|
|
// union returns a new allocSet that has the union of the two allocSets.
|
|
|
|
// Conflicts prefer the last passed allocSet containing the value
|
|
|
|
func (a allocSet) union(others ...allocSet) allocSet {
|
|
|
|
union := make(map[string]*structs.Allocation, len(a))
|
|
|
|
order := []allocSet{a}
|
|
|
|
order = append(order, others...)
|
|
|
|
|
|
|
|
for _, set := range order {
|
|
|
|
for k, v := range set {
|
|
|
|
union[k] = v
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return union
|
|
|
|
}
|
|
|
|
|
2017-05-23 00:14:38 +00:00
|
|
|
// fitlerByTainted takes a set of tainted nodes and filters the allocation set
|
|
|
|
// into three groups:
|
|
|
|
// 1. Those that exist on untainted nodes
|
|
|
|
// 2. Those exist on nodes that are draining
|
|
|
|
// 3. Those that exist on lost nodes
|
|
|
|
func (a allocSet) filterByTainted(nodes map[string]*structs.Node) (untainted, migrate, lost allocSet) {
|
|
|
|
untainted = make(map[string]*structs.Allocation)
|
|
|
|
migrate = make(map[string]*structs.Allocation)
|
|
|
|
lost = make(map[string]*structs.Allocation)
|
|
|
|
for _, alloc := range a {
|
|
|
|
n, ok := nodes[alloc.NodeID]
|
2017-05-24 20:43:01 +00:00
|
|
|
if !ok {
|
|
|
|
untainted[alloc.ID] = alloc
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the job is batch and finished successfully, the fact that the
|
|
|
|
// node is tainted does not mean it should be migrated or marked as
|
|
|
|
// lost as the work was already successfully finished. However for
|
|
|
|
// service/system jobs, tasks should never complete. The check of
|
|
|
|
// batch type, defends against client bugs.
|
|
|
|
if alloc.Job.Type == structs.JobTypeBatch && alloc.RanSuccessfully() {
|
2017-05-23 00:14:38 +00:00
|
|
|
untainted[alloc.ID] = alloc
|
2017-05-24 20:43:01 +00:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
if n == nil || n.TerminalStatus() {
|
2017-05-23 00:14:38 +00:00
|
|
|
lost[alloc.ID] = alloc
|
2017-05-24 20:43:01 +00:00
|
|
|
} else {
|
2017-05-23 00:14:38 +00:00
|
|
|
migrate[alloc.ID] = alloc
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// filterByCanary returns a new allocation set that contains only canaries
|
|
|
|
func (a allocSet) filterByCanary() allocSet {
|
|
|
|
canaries := make(map[string]*structs.Allocation)
|
|
|
|
for _, alloc := range a {
|
|
|
|
if alloc.Canary {
|
|
|
|
canaries[alloc.ID] = alloc
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return canaries
|
|
|
|
}
|
|
|
|
|
|
|
|
// filterByDeployment filters allocations into two sets, those that match the
|
|
|
|
// given deployment ID and those that don't
|
|
|
|
func (a allocSet) filterByDeployment(id string) (match, nonmatch allocSet) {
|
|
|
|
match = make(map[string]*structs.Allocation)
|
|
|
|
nonmatch = make(map[string]*structs.Allocation)
|
|
|
|
for _, alloc := range a {
|
|
|
|
if alloc.DeploymentID == id {
|
|
|
|
match[alloc.ID] = alloc
|
|
|
|
} else {
|
|
|
|
nonmatch[alloc.ID] = alloc
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|