forked from albrow/jobs
-
Notifications
You must be signed in to change notification settings - Fork 0
/
worker.go
124 lines (118 loc) · 3.49 KB
/
worker.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
// Copyright 2015 Alex Browne. All rights reserved.
// Use of this source code is governed by the MIT
// license, which can be found in the LICENSE file.
package jobs
import (
"fmt"
"github.com/garyburd/redigo/redis"
"reflect"
"sync"
"time"
)
// worker continuously executes jobs within its own goroutine.
// The jobs chan is shared between all jobs. To stop the worker,
// simply close the jobs channel.
type worker struct {
jobs chan *Job
wg *sync.WaitGroup
}
// start starts a goroutine in which the worker will continuously
// execute jobs until the jobs channel is closed.
func (w *worker) start() {
go func() {
for job := range w.jobs {
w.doJob(job)
}
w.wg.Done()
}()
}
// doJob executes the given job. It also sets the status and timestamps for
// the job appropriately depending on the outcome of the execution.
func (w *worker) doJob(job *Job) {
defer func() {
if r := recover(); r != nil {
// Get a reasonable error message from the panic
msg := ""
if err, ok := r.(error); ok {
msg = err.Error()
} else {
msg = fmt.Sprint(r)
}
if err := setJobError(job, msg); err != nil {
// Nothing left to do but panic
panic(err)
}
}
}()
// Set the started field and save the job
job.started = time.Now().UTC().UnixNano()
t0 := newTransaction()
t0.command("HSET", redis.Args{job.key(), "started", job.started}, nil)
if err := t0.exec(); err != nil {
if err := setJobError(job, err.Error()); err != nil {
// NOTE: panics will be caught by the recover statment above
panic(err)
}
return
}
// Use reflection to instantiate arguments for the handler
handlerArgs := []reflect.Value{}
if job.typ.dataType != nil {
// Instantiate a new variable to hold this argument
dataVal := reflect.New(job.typ.dataType)
if err := decode(job.data, dataVal.Interface()); err != nil {
if err := setJobError(job, err.Error()); err != nil {
// NOTE: panics will be caught by the recover statment above
panic(err)
}
return
}
handlerArgs = append(handlerArgs, dataVal.Elem())
}
// Call the handler using the arguments we just instantiated
handlerVal := reflect.ValueOf(job.typ.handler)
returnVals := handlerVal.Call(handlerArgs)
// Check if the error return value was nil
if !returnVals[0].IsNil() {
err := returnVals[0].Interface().(error)
if err := setJobError(job, err.Error()); err != nil {
// NOTE: panics will be caught by the recover statment above
panic(err)
}
return
}
// Set the finished timestamp
job.finished = time.Now().UTC().UnixNano()
t1 := newTransaction()
t1.command("HSET", redis.Args{job.key(), "finished", job.finished}, nil)
if job.isRecurring() {
// If the job is recurring, reschedule and set status to queued
job.time = job.nextTime()
t1.command("HSET", redis.Args{job.key(), "time", job.time}, nil)
t1.addJobToTimeIndex(job)
t1.setStatus(job, StatusQueued)
} else {
// Otherwise, set status to finished
t1.setStatus(job, StatusFinished)
}
if err := t1.exec(); err != nil {
if err := setJobError(job, err.Error()); err != nil {
// NOTE: panics will be caught by the recover statment above
panic(err)
}
return
}
}
func setJobError(job *Job, msg string) error {
// Start a new transaction
t := newTransaction()
// Set the job error field
t.command("HSET", redis.Args{job.key(), "error", msg}, nil)
// Either queue the job for retry or mark it as failed depending
// on how many retries the job has left
t.retryOrFailJob(job, nil)
if err := t.exec(); err != nil {
return err
}
return nil
}