2015-08-15 23:07:50 +00:00
|
|
|
package nomad
|
|
|
|
|
|
|
|
import (
|
|
|
|
"testing"
|
2015-08-16 00:42:51 +00:00
|
|
|
"time"
|
2015-08-15 23:07:50 +00:00
|
|
|
|
|
|
|
"github.com/hashicorp/nomad/nomad/mock"
|
|
|
|
"github.com/hashicorp/nomad/nomad/structs"
|
|
|
|
"github.com/hashicorp/nomad/testutil"
|
|
|
|
)
|
|
|
|
|
|
|
|
func TestCoreScheduler_EvalGC(t *testing.T) {
|
|
|
|
s1 := testServer(t, nil)
|
|
|
|
defer s1.Shutdown()
|
|
|
|
testutil.WaitForLeader(t, s1.RPC)
|
|
|
|
|
|
|
|
// Insert "dead" eval
|
|
|
|
state := s1.fsm.State()
|
|
|
|
eval := mock.Eval()
|
|
|
|
eval.Status = structs.EvalStatusFailed
|
|
|
|
err := state.UpsertEvals(1000, []*structs.Evaluation{eval})
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Insert "dead" alloc
|
|
|
|
alloc := mock.Alloc()
|
|
|
|
alloc.EvalID = eval.ID
|
2015-08-26 00:36:52 +00:00
|
|
|
alloc.DesiredStatus = structs.AllocDesiredStatusFailed
|
2015-09-07 03:47:42 +00:00
|
|
|
err = state.UpsertAllocs(1001, []*structs.Allocation{alloc})
|
2015-08-15 23:07:50 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
2015-08-16 00:42:51 +00:00
|
|
|
// Update the time tables to make this work
|
|
|
|
tt := s1.fsm.TimeTable()
|
|
|
|
tt.Witness(2000, time.Now().UTC().Add(-1*s1.config.EvalGCThreshold))
|
|
|
|
|
2015-08-15 23:07:50 +00:00
|
|
|
// Create a core scheduler
|
|
|
|
snap, err := state.Snapshot()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
core := NewCoreScheduler(s1, snap)
|
|
|
|
|
|
|
|
// Attempt the GC
|
|
|
|
gc := s1.coreJobEval(structs.CoreJobEvalGC)
|
|
|
|
gc.ModifyIndex = 2000
|
|
|
|
err = core.Process(gc)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Should be gone
|
2015-09-07 03:56:38 +00:00
|
|
|
out, err := state.EvalByID(eval.ID)
|
2015-08-15 23:07:50 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
if out != nil {
|
|
|
|
t.Fatalf("bad: %v", out)
|
|
|
|
}
|
|
|
|
|
2015-09-07 03:56:38 +00:00
|
|
|
outA, err := state.AllocByID(alloc.ID)
|
2015-08-15 23:07:50 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
if outA != nil {
|
|
|
|
t.Fatalf("bad: %v", outA)
|
|
|
|
}
|
2016-03-25 23:46:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestCoreScheduler_EvalGC_Batch_NoAllocs(t *testing.T) {
|
|
|
|
s1 := testServer(t, nil)
|
|
|
|
defer s1.Shutdown()
|
|
|
|
testutil.WaitForLeader(t, s1.RPC)
|
|
|
|
|
|
|
|
// Insert "dead" eval
|
|
|
|
state := s1.fsm.State()
|
|
|
|
eval := mock.Eval()
|
|
|
|
eval.Type = structs.JobTypeBatch
|
|
|
|
eval.Status = structs.EvalStatusFailed
|
|
|
|
err := state.UpsertEvals(1000, []*structs.Evaluation{eval})
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update the time tables to make this work
|
|
|
|
tt := s1.fsm.TimeTable()
|
|
|
|
tt.Witness(2000, time.Now().UTC().Add(-1*s1.config.EvalGCThreshold))
|
|
|
|
|
|
|
|
// Create a core scheduler
|
|
|
|
snap, err := state.Snapshot()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
core := NewCoreScheduler(s1, snap)
|
|
|
|
|
|
|
|
// Attempt the GC
|
|
|
|
gc := s1.coreJobEval(structs.CoreJobEvalGC)
|
|
|
|
gc.ModifyIndex = 2000
|
|
|
|
err = core.Process(gc)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Should be gone because there is no alloc associated
|
|
|
|
out, err := state.EvalByID(eval.ID)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
if out != nil {
|
|
|
|
t.Fatalf("bad: %v", out)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestCoreScheduler_EvalGC_Batch_Allocs(t *testing.T) {
|
|
|
|
s1 := testServer(t, nil)
|
|
|
|
defer s1.Shutdown()
|
|
|
|
testutil.WaitForLeader(t, s1.RPC)
|
|
|
|
|
|
|
|
// Insert "dead" eval
|
|
|
|
state := s1.fsm.State()
|
|
|
|
eval := mock.Eval()
|
|
|
|
eval.Type = structs.JobTypeBatch
|
|
|
|
eval.Status = structs.EvalStatusFailed
|
|
|
|
err := state.UpsertEvals(1000, []*structs.Evaluation{eval})
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Insert "dead" alloc
|
|
|
|
alloc := mock.Alloc()
|
|
|
|
alloc.EvalID = eval.ID
|
|
|
|
alloc.DesiredStatus = structs.AllocDesiredStatusFailed
|
|
|
|
err = state.UpsertAllocs(1001, []*structs.Allocation{alloc})
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update the time tables to make this work
|
|
|
|
tt := s1.fsm.TimeTable()
|
|
|
|
tt.Witness(2000, time.Now().UTC().Add(-1*s1.config.EvalGCThreshold))
|
|
|
|
|
|
|
|
// Create a core scheduler
|
|
|
|
snap, err := state.Snapshot()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
core := NewCoreScheduler(s1, snap)
|
|
|
|
|
|
|
|
// Attempt the GC
|
|
|
|
gc := s1.coreJobEval(structs.CoreJobEvalGC)
|
|
|
|
gc.ModifyIndex = 2000
|
|
|
|
err = core.Process(gc)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Shouldn't be gone because there are associated allocs.
|
|
|
|
out, err := state.EvalByID(eval.ID)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
if out == nil {
|
|
|
|
t.Fatalf("bad: %v", out)
|
|
|
|
}
|
|
|
|
|
|
|
|
outA, err := state.AllocByID(alloc.ID)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
if outA == nil {
|
|
|
|
t.Fatalf("bad: %v", outA)
|
|
|
|
}
|
2015-08-15 23:07:50 +00:00
|
|
|
}
|
2015-09-07 18:01:29 +00:00
|
|
|
|
2016-02-20 23:50:41 +00:00
|
|
|
func TestCoreScheduler_EvalGC_Force(t *testing.T) {
|
|
|
|
s1 := testServer(t, nil)
|
|
|
|
defer s1.Shutdown()
|
|
|
|
testutil.WaitForLeader(t, s1.RPC)
|
|
|
|
|
|
|
|
// Insert "dead" eval
|
|
|
|
state := s1.fsm.State()
|
|
|
|
eval := mock.Eval()
|
|
|
|
eval.Status = structs.EvalStatusFailed
|
|
|
|
err := state.UpsertEvals(1000, []*structs.Evaluation{eval})
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Insert "dead" alloc
|
|
|
|
alloc := mock.Alloc()
|
|
|
|
alloc.EvalID = eval.ID
|
|
|
|
alloc.DesiredStatus = structs.AllocDesiredStatusFailed
|
|
|
|
err = state.UpsertAllocs(1001, []*structs.Allocation{alloc})
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create a core scheduler
|
|
|
|
snap, err := state.Snapshot()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
core := NewCoreScheduler(s1, snap)
|
|
|
|
|
|
|
|
// Attempt the GC
|
|
|
|
gc := s1.forceCoreJobEval(structs.CoreJobEvalGC)
|
|
|
|
err = core.Process(gc)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Should be gone
|
|
|
|
out, err := state.EvalByID(eval.ID)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
if out != nil {
|
|
|
|
t.Fatalf("bad: %v", out)
|
|
|
|
}
|
|
|
|
|
|
|
|
outA, err := state.AllocByID(alloc.ID)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
if outA != nil {
|
|
|
|
t.Fatalf("bad: %v", outA)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-09-07 18:01:29 +00:00
|
|
|
func TestCoreScheduler_NodeGC(t *testing.T) {
|
|
|
|
s1 := testServer(t, nil)
|
|
|
|
defer s1.Shutdown()
|
|
|
|
testutil.WaitForLeader(t, s1.RPC)
|
|
|
|
|
|
|
|
// Insert "dead" node
|
|
|
|
state := s1.fsm.State()
|
|
|
|
node := mock.Node()
|
|
|
|
node.Status = structs.NodeStatusDown
|
|
|
|
err := state.UpsertNode(1000, node)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update the time tables to make this work
|
|
|
|
tt := s1.fsm.TimeTable()
|
|
|
|
tt.Witness(2000, time.Now().UTC().Add(-1*s1.config.NodeGCThreshold))
|
|
|
|
|
|
|
|
// Create a core scheduler
|
|
|
|
snap, err := state.Snapshot()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
core := NewCoreScheduler(s1, snap)
|
|
|
|
|
|
|
|
// Attempt the GC
|
|
|
|
gc := s1.coreJobEval(structs.CoreJobNodeGC)
|
|
|
|
gc.ModifyIndex = 2000
|
|
|
|
err = core.Process(gc)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Should be gone
|
|
|
|
out, err := state.NodeByID(node.ID)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
if out != nil {
|
|
|
|
t.Fatalf("bad: %v", out)
|
|
|
|
}
|
|
|
|
}
|
2015-12-15 03:20:57 +00:00
|
|
|
|
2016-02-20 23:50:41 +00:00
|
|
|
func TestCoreScheduler_NodeGC_Force(t *testing.T) {
|
|
|
|
s1 := testServer(t, nil)
|
|
|
|
defer s1.Shutdown()
|
|
|
|
testutil.WaitForLeader(t, s1.RPC)
|
|
|
|
|
|
|
|
// Insert "dead" node
|
|
|
|
state := s1.fsm.State()
|
|
|
|
node := mock.Node()
|
|
|
|
node.Status = structs.NodeStatusDown
|
|
|
|
err := state.UpsertNode(1000, node)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create a core scheduler
|
|
|
|
snap, err := state.Snapshot()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
core := NewCoreScheduler(s1, snap)
|
|
|
|
|
|
|
|
// Attempt the GC
|
|
|
|
gc := s1.forceCoreJobEval(structs.CoreJobNodeGC)
|
|
|
|
err = core.Process(gc)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Should be gone
|
|
|
|
out, err := state.NodeByID(node.ID)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
if out != nil {
|
|
|
|
t.Fatalf("bad: %v", out)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-12-15 03:20:57 +00:00
|
|
|
func TestCoreScheduler_JobGC(t *testing.T) {
|
|
|
|
tests := []struct {
|
2016-03-23 22:21:46 +00:00
|
|
|
test, evalStatus, desiredAllocStatus, clientAllocStatus string
|
|
|
|
shouldExist bool
|
2015-12-15 03:20:57 +00:00
|
|
|
}{
|
|
|
|
{
|
2016-03-23 22:21:46 +00:00
|
|
|
test: "Terminal",
|
|
|
|
evalStatus: structs.EvalStatusFailed,
|
|
|
|
desiredAllocStatus: structs.AllocDesiredStatusRun,
|
|
|
|
clientAllocStatus: structs.AllocDesiredStatusFailed,
|
|
|
|
shouldExist: false,
|
2015-12-15 03:20:57 +00:00
|
|
|
},
|
|
|
|
{
|
2016-03-23 22:21:46 +00:00
|
|
|
test: "Has Failed Alloc",
|
|
|
|
evalStatus: structs.EvalStatusFailed,
|
|
|
|
desiredAllocStatus: structs.AllocDesiredStatusRun,
|
|
|
|
clientAllocStatus: structs.AllocDesiredStatusFailed,
|
|
|
|
shouldExist: false,
|
2015-12-15 03:20:57 +00:00
|
|
|
},
|
|
|
|
{
|
2016-03-23 22:21:46 +00:00
|
|
|
test: "Has Running Alloc",
|
|
|
|
evalStatus: structs.EvalStatusFailed,
|
|
|
|
desiredAllocStatus: structs.AllocDesiredStatusRun,
|
|
|
|
clientAllocStatus: structs.AllocDesiredStatusRun,
|
|
|
|
shouldExist: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
test: "Has Eval",
|
|
|
|
evalStatus: structs.EvalStatusPending,
|
|
|
|
desiredAllocStatus: structs.AllocDesiredStatusRun,
|
|
|
|
clientAllocStatus: structs.AllocDesiredStatusFailed,
|
|
|
|
shouldExist: true,
|
2015-12-15 03:20:57 +00:00
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, test := range tests {
|
|
|
|
s1 := testServer(t, nil)
|
|
|
|
defer s1.Shutdown()
|
|
|
|
testutil.WaitForLeader(t, s1.RPC)
|
|
|
|
|
|
|
|
// Insert job.
|
|
|
|
state := s1.fsm.State()
|
|
|
|
job := mock.Job()
|
2016-03-24 01:02:01 +00:00
|
|
|
job.Type = structs.JobTypeBatch
|
2015-12-15 03:20:57 +00:00
|
|
|
err := state.UpsertJob(1000, job)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("test(%s) err: %v", test.test, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Insert eval
|
|
|
|
eval := mock.Eval()
|
|
|
|
eval.JobID = job.ID
|
|
|
|
eval.Status = test.evalStatus
|
|
|
|
err = state.UpsertEvals(1001, []*structs.Evaluation{eval})
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("test(%s) err: %v", test.test, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Insert alloc
|
|
|
|
alloc := mock.Alloc()
|
|
|
|
alloc.JobID = job.ID
|
|
|
|
alloc.EvalID = eval.ID
|
2016-03-23 22:21:46 +00:00
|
|
|
alloc.DesiredStatus = test.desiredAllocStatus
|
|
|
|
alloc.ClientStatus = test.clientAllocStatus
|
2015-12-15 03:20:57 +00:00
|
|
|
err = state.UpsertAllocs(1002, []*structs.Allocation{alloc})
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("test(%s) err: %v", test.test, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update the time tables to make this work
|
|
|
|
tt := s1.fsm.TimeTable()
|
2015-12-15 21:32:31 +00:00
|
|
|
tt.Witness(2000, time.Now().UTC().Add(-1*s1.config.JobGCThreshold))
|
2015-12-15 03:20:57 +00:00
|
|
|
|
|
|
|
// Create a core scheduler
|
|
|
|
snap, err := state.Snapshot()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("test(%s) err: %v", test.test, err)
|
|
|
|
}
|
|
|
|
core := NewCoreScheduler(s1, snap)
|
|
|
|
|
|
|
|
// Attempt the GC
|
|
|
|
gc := s1.coreJobEval(structs.CoreJobJobGC)
|
|
|
|
gc.ModifyIndex = 2000
|
|
|
|
err = core.Process(gc)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("test(%s) err: %v", test.test, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Should still exist
|
|
|
|
out, err := state.JobByID(job.ID)
|
|
|
|
if err != nil {
|
2015-12-18 20:24:53 +00:00
|
|
|
t.Fatalf("test(%s) err: %v", test.test, err)
|
2015-12-15 03:20:57 +00:00
|
|
|
}
|
|
|
|
if (test.shouldExist && out == nil) || (!test.shouldExist && out != nil) {
|
|
|
|
t.Fatalf("test(%s) bad: %v", test.test, out)
|
|
|
|
}
|
|
|
|
|
|
|
|
outE, err := state.EvalByID(eval.ID)
|
|
|
|
if err != nil {
|
2015-12-18 20:24:53 +00:00
|
|
|
t.Fatalf("test(%s) err: %v", test.test, err)
|
2015-12-15 03:20:57 +00:00
|
|
|
}
|
|
|
|
if (test.shouldExist && outE == nil) || (!test.shouldExist && outE != nil) {
|
|
|
|
t.Fatalf("test(%s) bad: %v", test.test, out)
|
|
|
|
}
|
|
|
|
|
|
|
|
outA, err := state.AllocByID(alloc.ID)
|
|
|
|
if err != nil {
|
2015-12-18 20:24:53 +00:00
|
|
|
t.Fatalf("test(%s) err: %v", test.test, err)
|
2015-12-15 03:20:57 +00:00
|
|
|
}
|
|
|
|
if (test.shouldExist && outA == nil) || (!test.shouldExist && outA != nil) {
|
|
|
|
t.Fatalf("test(%s) bad: %v", test.test, outA)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-02-20 23:50:41 +00:00
|
|
|
|
|
|
|
func TestCoreScheduler_JobGC_Force(t *testing.T) {
|
|
|
|
tests := []struct {
|
2016-03-23 22:21:46 +00:00
|
|
|
test, evalStatus, desiredAllocStatus, clientAllocStatus string
|
|
|
|
shouldExist bool
|
2016-02-20 23:50:41 +00:00
|
|
|
}{
|
|
|
|
{
|
2016-03-23 22:21:46 +00:00
|
|
|
test: "Terminal",
|
|
|
|
evalStatus: structs.EvalStatusFailed,
|
|
|
|
desiredAllocStatus: structs.AllocDesiredStatusRun,
|
|
|
|
clientAllocStatus: structs.AllocDesiredStatusFailed,
|
|
|
|
shouldExist: false,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
test: "Has Failed Alloc",
|
|
|
|
evalStatus: structs.EvalStatusFailed,
|
|
|
|
desiredAllocStatus: structs.AllocDesiredStatusRun,
|
|
|
|
clientAllocStatus: structs.AllocDesiredStatusFailed,
|
|
|
|
shouldExist: false,
|
2016-02-20 23:50:41 +00:00
|
|
|
},
|
|
|
|
{
|
2016-03-23 22:21:46 +00:00
|
|
|
test: "Has Running Alloc",
|
|
|
|
evalStatus: structs.EvalStatusFailed,
|
|
|
|
desiredAllocStatus: structs.AllocDesiredStatusRun,
|
|
|
|
clientAllocStatus: structs.AllocDesiredStatusRun,
|
|
|
|
shouldExist: true,
|
2016-02-20 23:50:41 +00:00
|
|
|
},
|
|
|
|
{
|
2016-03-23 22:21:46 +00:00
|
|
|
test: "Has Eval",
|
|
|
|
evalStatus: structs.EvalStatusPending,
|
|
|
|
desiredAllocStatus: structs.AllocDesiredStatusRun,
|
|
|
|
clientAllocStatus: structs.AllocDesiredStatusFailed,
|
|
|
|
shouldExist: true,
|
2016-02-20 23:50:41 +00:00
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, test := range tests {
|
|
|
|
s1 := testServer(t, nil)
|
|
|
|
defer s1.Shutdown()
|
|
|
|
testutil.WaitForLeader(t, s1.RPC)
|
|
|
|
|
|
|
|
// Insert job.
|
|
|
|
state := s1.fsm.State()
|
|
|
|
job := mock.Job()
|
2016-03-24 01:02:01 +00:00
|
|
|
job.Type = structs.JobTypeBatch
|
2016-02-20 23:50:41 +00:00
|
|
|
err := state.UpsertJob(1000, job)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("test(%s) err: %v", test.test, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Insert eval
|
|
|
|
eval := mock.Eval()
|
|
|
|
eval.JobID = job.ID
|
|
|
|
eval.Status = test.evalStatus
|
|
|
|
err = state.UpsertEvals(1001, []*structs.Evaluation{eval})
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("test(%s) err: %v", test.test, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Insert alloc
|
|
|
|
alloc := mock.Alloc()
|
|
|
|
alloc.JobID = job.ID
|
|
|
|
alloc.EvalID = eval.ID
|
2016-03-23 22:21:46 +00:00
|
|
|
alloc.DesiredStatus = test.desiredAllocStatus
|
|
|
|
alloc.ClientStatus = test.clientAllocStatus
|
2016-02-20 23:50:41 +00:00
|
|
|
err = state.UpsertAllocs(1002, []*structs.Allocation{alloc})
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("test(%s) err: %v", test.test, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create a core scheduler
|
|
|
|
snap, err := state.Snapshot()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("test(%s) err: %v", test.test, err)
|
|
|
|
}
|
|
|
|
core := NewCoreScheduler(s1, snap)
|
|
|
|
|
|
|
|
// Attempt the GC
|
|
|
|
gc := s1.forceCoreJobEval(structs.CoreJobJobGC)
|
|
|
|
err = core.Process(gc)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("test(%s) err: %v", test.test, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Should still exist
|
|
|
|
out, err := state.JobByID(job.ID)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("test(%s) err: %v", test.test, err)
|
|
|
|
}
|
|
|
|
if (test.shouldExist && out == nil) || (!test.shouldExist && out != nil) {
|
|
|
|
t.Fatalf("test(%s) bad: %v", test.test, out)
|
|
|
|
}
|
|
|
|
|
|
|
|
outE, err := state.EvalByID(eval.ID)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("test(%s) err: %v", test.test, err)
|
|
|
|
}
|
|
|
|
if (test.shouldExist && outE == nil) || (!test.shouldExist && outE != nil) {
|
|
|
|
t.Fatalf("test(%s) bad: %v", test.test, out)
|
|
|
|
}
|
|
|
|
|
|
|
|
outA, err := state.AllocByID(alloc.ID)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("test(%s) err: %v", test.test, err)
|
|
|
|
}
|
|
|
|
if (test.shouldExist && outA == nil) || (!test.shouldExist && outA != nil) {
|
|
|
|
t.Fatalf("test(%s) bad: %v", test.test, outA)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-03-30 22:17:13 +00:00
|
|
|
|
|
|
|
func TestCoreScheduler_PartitionReap(t *testing.T) {
|
|
|
|
s1 := testServer(t, nil)
|
|
|
|
defer s1.Shutdown()
|
|
|
|
testutil.WaitForLeader(t, s1.RPC)
|
|
|
|
|
|
|
|
// Create a core scheduler
|
|
|
|
snap, err := s1.fsm.State().Snapshot()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %v", err)
|
|
|
|
}
|
|
|
|
core := NewCoreScheduler(s1, snap)
|
|
|
|
|
|
|
|
// Set the max ids per reap to something lower.
|
|
|
|
maxIdsPerReap = 2
|
|
|
|
|
|
|
|
evals := []string{"a", "b", "c"}
|
|
|
|
allocs := []string{"1", "2", "3"}
|
|
|
|
requests := core.(*CoreScheduler).partitionReap(evals, allocs)
|
|
|
|
if len(requests) != 3 {
|
|
|
|
t.Fatalf("Expected 3 requests got: %v", requests)
|
|
|
|
}
|
|
|
|
|
|
|
|
first := requests[0]
|
|
|
|
if len(first.Evals) != 2 && len(first.Allocs) != 0 {
|
|
|
|
t.Fatalf("Unexpected first request: %v", first)
|
|
|
|
}
|
|
|
|
|
|
|
|
second := requests[1]
|
|
|
|
if len(second.Evals) != 1 && len(second.Allocs) != 1 {
|
|
|
|
t.Fatalf("Unexpected second request: %v", second)
|
|
|
|
}
|
|
|
|
|
|
|
|
third := requests[2]
|
|
|
|
if len(third.Evals) != 0 && len(third.Allocs) != 2 {
|
|
|
|
t.Fatalf("Unexpected third request: %v", third)
|
|
|
|
}
|
|
|
|
}
|