-
Notifications
You must be signed in to change notification settings - Fork 57
/
Copy pathrunner_test.go
159 lines (130 loc) · 3.57 KB
/
runner_test.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
package runner_test
import (
"errors"
"runtime"
"syscall"
"testing"
"time"
"github.com/ardanlabs/kit/runner"
)
// Success and failure markers.
var (
success = "\u2713"
failed = "\u2717"
)
//==============================================================================
// task represents a test task.
type task struct {
kill chan bool
err error
}
// Job is the implementation of the Jobber interface.
func (t *task) Job(traceID string) error {
// Pretend you are doing work for the specified
// amount of time.
<-t.kill
// Report we received the signal to keep things in
// sync between test functions.
t.kill <- true
return t.err
}
// Kill will kill the Job method immediately.
func (t *task) Kill() {
select {
case t.kill <- true:
// If we were able to send the message, wait
// for the response to keep things in sync.
<-t.kill
default:
}
}
// KillAfter will kill the Job method after the specified duration.
func (t *task) KillAfter(dur time.Duration) {
t.kill = make(chan bool)
go func() {
time.Sleep(dur)
t.Kill()
}()
runtime.Gosched()
}
//==============================================================================
// TestCompleted tests when jobs complete properly.
func TestCompleted(t *testing.T) {
t.Log("Given the need to test a successful task run.")
{
t.Log("\tWhen using a task that will complete in time.")
{
var job task
job.KillAfter(time.Millisecond)
r := runner.New(time.Second)
if err := r.Run("traceID", &job); err != nil {
t.Fatalf("\t%s\tShould not receive an error : %v", failed, err)
}
t.Logf("\t%s\tShould not receive an error.", success)
}
}
}
// TestError tests when jobs complete properly but with errors.
func TestError(t *testing.T) {
t.Log("Given the need to test a successful task run with error.")
{
t.Log("\tWhen using a task that will complete in time.")
{
job := task{
err: errors.New("Error"),
}
job.KillAfter(time.Millisecond)
r := runner.New(time.Second)
if err := r.Run("traceID", &job); err == nil {
t.Fatalf("\t%s\tShould receive our error : %v", failed, err)
}
t.Logf("\t%s\tShould receive our error.", success)
}
}
}
// TestTimeout tests when jobs timeout.
func TestTimeout(t *testing.T) {
t.Log("Given the need to test a task that timesout.")
{
t.Log("\tWhen using a task that will timeout.")
{
var job task
job.KillAfter(time.Second)
// Need the job method to quit as soon as we are done.
defer job.Kill()
r := runner.New(time.Millisecond)
if err := r.Run("traceID", &job); err != runner.ErrTimeout {
t.Fatalf("\t%s\tShould receive a timeout error : %v", failed, err)
}
t.Logf("\t%s\tShould receive a timeout error.", success)
}
}
}
// TestSignaled tests when jobs is requested to shutdown.
func TestSignaled(t *testing.T) {
t.Log("Given the need to test a task that is requested to shutdown.")
{
t.Log("\tWhen using a task that should see the signal.")
{
var job task
job.KillAfter(100 * time.Millisecond)
// Need the job method to quit as soon as we are done.
defer job.Kill()
go func() {
time.Sleep(50 * time.Millisecond)
syscall.Kill(syscall.Getpid(), syscall.SIGINT)
}()
r := runner.New(3 * time.Second)
if err := r.Run("traceID", &job); err != nil {
t.Errorf("\t%s\tShould receive no error : %v", failed, err)
} else {
t.Logf("\t%s\tShould receive no error.", success)
}
if !r.CheckShutdown() {
t.Errorf("\t%s\tShould show the check shutdown flag is set.", failed)
} else {
t.Logf("\t%s\tShould show the check shutdown flag is set.", success)
}
}
}
}