mirror of
https://github.com/hibiken/asynq.git
synced 2024-12-25 23:32:17 +08:00
Add tests for rdb list methods
This commit is contained in:
parent
437fb03bb3
commit
f3b9745228
223
internal/rdb/inspect.go
Normal file
223
internal/rdb/inspect.go
Normal file
@ -0,0 +1,223 @@
|
|||||||
|
package rdb
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/google/uuid"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Stats represents a state of queues at a certain time.
|
||||||
|
type Stats struct {
|
||||||
|
Enqueued int
|
||||||
|
InProgress int
|
||||||
|
Scheduled int
|
||||||
|
Retry int
|
||||||
|
Dead int
|
||||||
|
Timestamp time.Time
|
||||||
|
}
|
||||||
|
|
||||||
|
// EnqueuedTask is a task in a queue and is ready to be processed.
|
||||||
|
// Note: This is read only and used for monitoring purpose.
|
||||||
|
type EnqueuedTask struct {
|
||||||
|
ID uuid.UUID
|
||||||
|
Type string
|
||||||
|
Payload map[string]interface{}
|
||||||
|
}
|
||||||
|
|
||||||
|
// InProgressTask is a task that's currently being processed.
|
||||||
|
// Note: This is read only and used for monitoring purpose.
|
||||||
|
type InProgressTask struct {
|
||||||
|
ID uuid.UUID
|
||||||
|
Type string
|
||||||
|
Payload map[string]interface{}
|
||||||
|
}
|
||||||
|
|
||||||
|
// ScheduledTask is a task that's scheduled to be processed in the future.
|
||||||
|
// Note: This is read only and used for monitoring purpose.
|
||||||
|
type ScheduledTask struct {
|
||||||
|
ID uuid.UUID
|
||||||
|
Type string
|
||||||
|
Payload map[string]interface{}
|
||||||
|
ProcessAt time.Time
|
||||||
|
}
|
||||||
|
|
||||||
|
// RetryTask is a task that's in retry queue because worker failed to process the task.
|
||||||
|
// Note: This is read only and used for monitoring purpose.
|
||||||
|
type RetryTask struct {
|
||||||
|
ID uuid.UUID
|
||||||
|
Type string
|
||||||
|
Payload map[string]interface{}
|
||||||
|
// TODO(hibiken): add LastFailedAt time.Time
|
||||||
|
ProcessAt time.Time
|
||||||
|
ErrorMsg string
|
||||||
|
Retried int
|
||||||
|
Retry int
|
||||||
|
}
|
||||||
|
|
||||||
|
// DeadTask is a task in that has exhausted all retries.
|
||||||
|
// Note: This is read only and used for monitoring purpose.
|
||||||
|
type DeadTask struct {
|
||||||
|
ID uuid.UUID
|
||||||
|
Type string
|
||||||
|
Payload map[string]interface{}
|
||||||
|
LastFailedAt time.Time
|
||||||
|
ErrorMsg string
|
||||||
|
}
|
||||||
|
|
||||||
|
// CurrentStats returns a current state of the queues.
|
||||||
|
func (r *RDB) CurrentStats() (*Stats, error) {
|
||||||
|
pipe := r.client.Pipeline()
|
||||||
|
qlen := pipe.LLen(defaultQ)
|
||||||
|
plen := pipe.LLen(inProgressQ)
|
||||||
|
slen := pipe.ZCard(scheduledQ)
|
||||||
|
rlen := pipe.ZCard(retryQ)
|
||||||
|
dlen := pipe.ZCard(deadQ)
|
||||||
|
_, err := pipe.Exec()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return &Stats{
|
||||||
|
Enqueued: int(qlen.Val()),
|
||||||
|
InProgress: int(plen.Val()),
|
||||||
|
Scheduled: int(slen.Val()),
|
||||||
|
Retry: int(rlen.Val()),
|
||||||
|
Dead: int(dlen.Val()),
|
||||||
|
Timestamp: time.Now(),
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// ListEnqueued returns all enqueued tasks that are ready to be processed.
|
||||||
|
func (r *RDB) ListEnqueued() ([]*EnqueuedTask, error) {
|
||||||
|
data, err := r.client.LRange(defaultQ, 0, -1).Result()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
var tasks []*EnqueuedTask
|
||||||
|
for _, s := range data {
|
||||||
|
var msg TaskMessage
|
||||||
|
err := json.Unmarshal([]byte(s), &msg)
|
||||||
|
if err != nil {
|
||||||
|
// continue // bad data, ignore and continue
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
tasks = append(tasks, &EnqueuedTask{
|
||||||
|
ID: msg.ID,
|
||||||
|
Type: msg.Type,
|
||||||
|
Payload: msg.Payload,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
return tasks, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// ListInProgress returns all tasks that are currently being processed.
|
||||||
|
func (r *RDB) ListInProgress() ([]*InProgressTask, error) {
|
||||||
|
data, err := r.client.LRange(inProgressQ, 0, -1).Result()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
var tasks []*InProgressTask
|
||||||
|
for _, s := range data {
|
||||||
|
var msg TaskMessage
|
||||||
|
err := json.Unmarshal([]byte(s), &msg)
|
||||||
|
if err != nil {
|
||||||
|
continue // bad data, ignore and continue
|
||||||
|
}
|
||||||
|
tasks = append(tasks, &InProgressTask{
|
||||||
|
ID: msg.ID,
|
||||||
|
Type: msg.Type,
|
||||||
|
Payload: msg.Payload,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
return tasks, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// ListScheduled returns all tasks that are scheduled to be processed
|
||||||
|
// in the future.
|
||||||
|
func (r *RDB) ListScheduled() ([]*ScheduledTask, error) {
|
||||||
|
data, err := r.client.ZRangeWithScores(scheduledQ, 0, -1).Result()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
var tasks []*ScheduledTask
|
||||||
|
for _, z := range data {
|
||||||
|
s, ok := z.Member.(string)
|
||||||
|
if !ok {
|
||||||
|
continue // bad data, ignore and continue
|
||||||
|
}
|
||||||
|
var msg TaskMessage
|
||||||
|
err := json.Unmarshal([]byte(s), &msg)
|
||||||
|
if err != nil {
|
||||||
|
continue // bad data, ignore and continue
|
||||||
|
}
|
||||||
|
processAt := time.Unix(int64(z.Score), 0)
|
||||||
|
tasks = append(tasks, &ScheduledTask{
|
||||||
|
ID: msg.ID,
|
||||||
|
Type: msg.Type,
|
||||||
|
Payload: msg.Payload,
|
||||||
|
ProcessAt: processAt,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
return tasks, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// ListRetry returns all tasks that have failed before and willl be retried
|
||||||
|
// in the future.
|
||||||
|
func (r *RDB) ListRetry() ([]*RetryTask, error) {
|
||||||
|
data, err := r.client.ZRangeWithScores(retryQ, 0, -1).Result()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
var tasks []*RetryTask
|
||||||
|
for _, z := range data {
|
||||||
|
s, ok := z.Member.(string)
|
||||||
|
if !ok {
|
||||||
|
continue // bad data, ignore and continue
|
||||||
|
}
|
||||||
|
var msg TaskMessage
|
||||||
|
err := json.Unmarshal([]byte(s), &msg)
|
||||||
|
if err != nil {
|
||||||
|
continue // bad data, ignore and continue
|
||||||
|
}
|
||||||
|
processAt := time.Unix(int64(z.Score), 0)
|
||||||
|
tasks = append(tasks, &RetryTask{
|
||||||
|
ID: msg.ID,
|
||||||
|
Type: msg.Type,
|
||||||
|
Payload: msg.Payload,
|
||||||
|
ErrorMsg: msg.ErrorMsg,
|
||||||
|
Retry: msg.Retry,
|
||||||
|
Retried: msg.Retried,
|
||||||
|
ProcessAt: processAt,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
return tasks, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// ListDead returns all tasks that have exhausted its retry limit.
|
||||||
|
func (r *RDB) ListDead() ([]*DeadTask, error) {
|
||||||
|
data, err := r.client.ZRangeWithScores(deadQ, 0, -1).Result()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
var tasks []*DeadTask
|
||||||
|
for _, z := range data {
|
||||||
|
s, ok := z.Member.(string)
|
||||||
|
if !ok {
|
||||||
|
continue // bad data, ignore and continue
|
||||||
|
}
|
||||||
|
var msg TaskMessage
|
||||||
|
err := json.Unmarshal([]byte(s), &msg)
|
||||||
|
if err != nil {
|
||||||
|
continue // bad data, ignore and continue
|
||||||
|
}
|
||||||
|
lastFailedAt := time.Unix(int64(z.Score), 0)
|
||||||
|
tasks = append(tasks, &DeadTask{
|
||||||
|
ID: msg.ID,
|
||||||
|
Type: msg.Type,
|
||||||
|
Payload: msg.Payload,
|
||||||
|
ErrorMsg: msg.ErrorMsg,
|
||||||
|
LastFailedAt: lastFailedAt,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
return tasks, nil
|
||||||
|
}
|
468
internal/rdb/inspect_test.go
Normal file
468
internal/rdb/inspect_test.go
Normal file
@ -0,0 +1,468 @@
|
|||||||
|
package rdb
|
||||||
|
|
||||||
|
import (
|
||||||
|
"sort"
|
||||||
|
"testing"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/go-redis/redis/v7"
|
||||||
|
"github.com/google/go-cmp/cmp"
|
||||||
|
"github.com/google/uuid"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestCurrentStats(t *testing.T) {
|
||||||
|
r := setup(t)
|
||||||
|
m1 := randomTask("send_email", "default", map[string]interface{}{"subject": "hello"})
|
||||||
|
m2 := randomTask("reindex", "default", nil)
|
||||||
|
m3 := randomTask("gen_thumbnail", "default", map[string]interface{}{"src": "some/path/to/img"})
|
||||||
|
m4 := randomTask("sync", "default", nil)
|
||||||
|
|
||||||
|
tests := []struct {
|
||||||
|
enqueued []*TaskMessage
|
||||||
|
inProgress []*TaskMessage
|
||||||
|
scheduled []*TaskMessage
|
||||||
|
retry []*TaskMessage
|
||||||
|
dead []*TaskMessage
|
||||||
|
want *Stats
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
enqueued: []*TaskMessage{m1},
|
||||||
|
inProgress: []*TaskMessage{m2},
|
||||||
|
scheduled: []*TaskMessage{m3, m4},
|
||||||
|
retry: []*TaskMessage{},
|
||||||
|
dead: []*TaskMessage{},
|
||||||
|
want: &Stats{
|
||||||
|
Enqueued: 1,
|
||||||
|
InProgress: 1,
|
||||||
|
Scheduled: 2,
|
||||||
|
Retry: 0,
|
||||||
|
Dead: 0,
|
||||||
|
Timestamp: time.Now(),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
enqueued: []*TaskMessage{},
|
||||||
|
inProgress: []*TaskMessage{},
|
||||||
|
scheduled: []*TaskMessage{m3, m4},
|
||||||
|
retry: []*TaskMessage{m1},
|
||||||
|
dead: []*TaskMessage{m2},
|
||||||
|
want: &Stats{
|
||||||
|
Enqueued: 0,
|
||||||
|
InProgress: 0,
|
||||||
|
Scheduled: 2,
|
||||||
|
Retry: 1,
|
||||||
|
Dead: 1,
|
||||||
|
Timestamp: time.Now(),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, tc := range tests {
|
||||||
|
// clean up db before each test case.
|
||||||
|
if err := r.client.FlushDB().Err(); err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
// initialize the queues.
|
||||||
|
for _, msg := range tc.enqueued {
|
||||||
|
if err := r.Enqueue(msg); err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for _, msg := range tc.inProgress {
|
||||||
|
if err := r.client.LPush(inProgressQ, mustMarshal(t, msg)).Err(); err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for _, msg := range tc.scheduled {
|
||||||
|
if err := r.Schedule(msg, time.Now().Add(time.Hour)); err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for _, msg := range tc.retry {
|
||||||
|
if err := r.RetryLater(msg, time.Now().Add(time.Hour)); err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for _, msg := range tc.dead {
|
||||||
|
if err := r.Kill(msg); err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
got, err := r.CurrentStats()
|
||||||
|
if err != nil {
|
||||||
|
t.Errorf("r.CurrentStats() = %v, %v, want %v, nil", got, err, tc.want)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
if diff := cmp.Diff(tc.want, got, timeCmpOpt); diff != "" {
|
||||||
|
t.Errorf("r.CurrentStats() = %v, %v, want %v, nil; (-want, +got)\n%s", got, err, tc.want, diff)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestListEnqueued(t *testing.T) {
|
||||||
|
r := setup(t)
|
||||||
|
|
||||||
|
m1 := randomTask("send_email", "default", map[string]interface{}{"subject": "hello"})
|
||||||
|
m2 := randomTask("reindex", "default", nil)
|
||||||
|
t1 := &EnqueuedTask{ID: m1.ID, Type: m1.Type, Payload: m1.Payload}
|
||||||
|
t2 := &EnqueuedTask{ID: m2.ID, Type: m2.Type, Payload: m2.Payload}
|
||||||
|
tests := []struct {
|
||||||
|
enqueued []*TaskMessage
|
||||||
|
want []*EnqueuedTask
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
enqueued: []*TaskMessage{m1, m2},
|
||||||
|
want: []*EnqueuedTask{t1, t2},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
enqueued: []*TaskMessage{},
|
||||||
|
want: []*EnqueuedTask{},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, tc := range tests {
|
||||||
|
// clean up db before each test case.
|
||||||
|
if err := r.client.FlushDB().Err(); err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
// initialize the list
|
||||||
|
for _, msg := range tc.enqueued {
|
||||||
|
if err := r.Enqueue(msg); err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
got, err := r.ListEnqueued()
|
||||||
|
if err != nil {
|
||||||
|
t.Errorf("r.ListEnqueued() = %v, %v, want %v, nil", got, err, tc.want)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
sortOpt := cmp.Transformer("SortMsg", func(in []*EnqueuedTask) []*EnqueuedTask {
|
||||||
|
out := append([]*EnqueuedTask(nil), in...) // Copy input to avoid mutating it
|
||||||
|
sort.Slice(out, func(i, j int) bool {
|
||||||
|
return out[i].ID.String() < out[j].ID.String()
|
||||||
|
})
|
||||||
|
return out
|
||||||
|
})
|
||||||
|
if diff := cmp.Diff(tc.want, got, sortOpt); diff != "" {
|
||||||
|
t.Errorf("r.ListEnqueued() = %v, %v, want %v, nil; (-want, +got)\n%s", got, err, tc.want, diff)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestListInProgress(t *testing.T) {
|
||||||
|
r := setup(t)
|
||||||
|
|
||||||
|
m1 := randomTask("send_email", "default", map[string]interface{}{"subject": "hello"})
|
||||||
|
m2 := randomTask("reindex", "default", nil)
|
||||||
|
t1 := &InProgressTask{ID: m1.ID, Type: m1.Type, Payload: m1.Payload}
|
||||||
|
t2 := &InProgressTask{ID: m2.ID, Type: m2.Type, Payload: m2.Payload}
|
||||||
|
tests := []struct {
|
||||||
|
enqueued []*TaskMessage
|
||||||
|
want []*InProgressTask
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
enqueued: []*TaskMessage{m1, m2},
|
||||||
|
want: []*InProgressTask{t1, t2},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
enqueued: []*TaskMessage{},
|
||||||
|
want: []*InProgressTask{},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, tc := range tests {
|
||||||
|
// clean up db before each test case.
|
||||||
|
if err := r.client.FlushDB().Err(); err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
// initialize the list
|
||||||
|
for _, msg := range tc.enqueued {
|
||||||
|
if err := r.client.LPush(inProgressQ, mustMarshal(t, msg)).Err(); err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
got, err := r.ListInProgress()
|
||||||
|
if err != nil {
|
||||||
|
t.Errorf("r.ListInProgress() = %v, %v, want %v, nil", got, err, tc.want)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
sortOpt := cmp.Transformer("SortMsg", func(in []*InProgressTask) []*InProgressTask {
|
||||||
|
out := append([]*InProgressTask(nil), in...) // Copy input to avoid mutating it
|
||||||
|
sort.Slice(out, func(i, j int) bool {
|
||||||
|
return out[i].ID.String() < out[j].ID.String()
|
||||||
|
})
|
||||||
|
return out
|
||||||
|
})
|
||||||
|
if diff := cmp.Diff(tc.want, got, sortOpt); diff != "" {
|
||||||
|
t.Errorf("r.ListInProgress() = %v, %v, want %v, nil; (-want, +got)\n%s", got, err, tc.want, diff)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestListScheduled(t *testing.T) {
|
||||||
|
r := setup(t)
|
||||||
|
m1 := randomTask("send_email", "default", map[string]interface{}{"subject": "hello"})
|
||||||
|
m2 := randomTask("reindex", "default", nil)
|
||||||
|
p1 := time.Now().Add(30 * time.Minute)
|
||||||
|
p2 := time.Now().Add(24 * time.Hour)
|
||||||
|
t1 := &ScheduledTask{ID: m1.ID, Type: m1.Type, Payload: m1.Payload, ProcessAt: p1}
|
||||||
|
t2 := &ScheduledTask{ID: m2.ID, Type: m2.Type, Payload: m2.Payload, ProcessAt: p2}
|
||||||
|
|
||||||
|
type scheduledEntry struct {
|
||||||
|
msg *TaskMessage
|
||||||
|
processAt time.Time
|
||||||
|
}
|
||||||
|
|
||||||
|
tests := []struct {
|
||||||
|
scheduled []scheduledEntry
|
||||||
|
want []*ScheduledTask
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
scheduled: []scheduledEntry{
|
||||||
|
{m1, p1},
|
||||||
|
{m2, p2},
|
||||||
|
},
|
||||||
|
want: []*ScheduledTask{t1, t2},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
scheduled: []scheduledEntry{},
|
||||||
|
want: []*ScheduledTask{},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, tc := range tests {
|
||||||
|
// clean up db before each test case.
|
||||||
|
if err := r.client.FlushDB().Err(); err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
// initialize the scheduled queue
|
||||||
|
for _, s := range tc.scheduled {
|
||||||
|
err := r.Schedule(s.msg, s.processAt)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
got, err := r.ListScheduled()
|
||||||
|
if err != nil {
|
||||||
|
t.Errorf("r.ListScheduled() = %v, %v, want %v, nil", got, err, tc.want)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
sortOpt := cmp.Transformer("SortMsg", func(in []*ScheduledTask) []*ScheduledTask {
|
||||||
|
out := append([]*ScheduledTask(nil), in...) // Copy input to avoid mutating it
|
||||||
|
sort.Slice(out, func(i, j int) bool {
|
||||||
|
return out[i].ID.String() < out[j].ID.String()
|
||||||
|
})
|
||||||
|
return out
|
||||||
|
})
|
||||||
|
|
||||||
|
if diff := cmp.Diff(tc.want, got, sortOpt, timeCmpOpt); diff != "" {
|
||||||
|
t.Errorf("r.ListScheduled() = %v, %v, want %v, nil; (-want, +got)\n%s", got, err, tc.want, diff)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestListRetry(t *testing.T) {
|
||||||
|
r := setup(t)
|
||||||
|
m1 := &TaskMessage{
|
||||||
|
ID: uuid.New(),
|
||||||
|
Type: "send_email",
|
||||||
|
Queue: "default",
|
||||||
|
Payload: map[string]interface{}{"subject": "hello"},
|
||||||
|
ErrorMsg: "email server not responding",
|
||||||
|
Retry: 25,
|
||||||
|
Retried: 10,
|
||||||
|
}
|
||||||
|
m2 := &TaskMessage{
|
||||||
|
ID: uuid.New(),
|
||||||
|
Type: "reindex",
|
||||||
|
Queue: "default",
|
||||||
|
Payload: nil,
|
||||||
|
ErrorMsg: "search engine not responding",
|
||||||
|
Retry: 25,
|
||||||
|
Retried: 2,
|
||||||
|
}
|
||||||
|
p1 := time.Now().Add(5 * time.Minute)
|
||||||
|
p2 := time.Now().Add(24 * time.Hour)
|
||||||
|
t1 := &RetryTask{ID: m1.ID, Type: m1.Type, Payload: m1.Payload,
|
||||||
|
ProcessAt: p1, ErrorMsg: m1.ErrorMsg, Retried: m1.Retried, Retry: m1.Retry}
|
||||||
|
t2 := &RetryTask{ID: m2.ID, Type: m2.Type, Payload: m2.Payload,
|
||||||
|
ProcessAt: p2, ErrorMsg: m2.ErrorMsg, Retried: m2.Retried, Retry: m2.Retry}
|
||||||
|
|
||||||
|
type retryEntry struct {
|
||||||
|
msg *TaskMessage
|
||||||
|
processAt time.Time
|
||||||
|
}
|
||||||
|
|
||||||
|
tests := []struct {
|
||||||
|
dead []retryEntry
|
||||||
|
want []*RetryTask
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
dead: []retryEntry{
|
||||||
|
{m1, p1},
|
||||||
|
{m2, p2},
|
||||||
|
},
|
||||||
|
want: []*RetryTask{t1, t2},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
dead: []retryEntry{},
|
||||||
|
want: []*RetryTask{},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, tc := range tests {
|
||||||
|
// clean up db before each test case.
|
||||||
|
if err := r.client.FlushDB().Err(); err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
// initialize the scheduled queue
|
||||||
|
for _, d := range tc.dead {
|
||||||
|
r.client.ZAdd(retryQ, &redis.Z{
|
||||||
|
Member: mustMarshal(t, d.msg),
|
||||||
|
Score: float64(d.processAt.Unix()),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
got, err := r.ListRetry()
|
||||||
|
if err != nil {
|
||||||
|
t.Errorf("r.ListRetry() = %v, %v, want %v, nil", got, err, tc.want)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
sortOpt := cmp.Transformer("SortMsg", func(in []*RetryTask) []*RetryTask {
|
||||||
|
out := append([]*RetryTask(nil), in...) // Copy input to avoid mutating it
|
||||||
|
sort.Slice(out, func(i, j int) bool {
|
||||||
|
return out[i].ID.String() < out[j].ID.String()
|
||||||
|
})
|
||||||
|
return out
|
||||||
|
})
|
||||||
|
|
||||||
|
if diff := cmp.Diff(tc.want, got, sortOpt, timeCmpOpt); diff != "" {
|
||||||
|
t.Errorf("r.ListRetry() = %v, %v, want %v, nil; (-want, +got)\n%s", got, err, tc.want, diff)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestListDead(t *testing.T) {
|
||||||
|
r := setup(t)
|
||||||
|
m1 := &TaskMessage{
|
||||||
|
ID: uuid.New(),
|
||||||
|
Type: "send_email",
|
||||||
|
Queue: "default",
|
||||||
|
Payload: map[string]interface{}{"subject": "hello"},
|
||||||
|
ErrorMsg: "email server not responding",
|
||||||
|
}
|
||||||
|
m2 := &TaskMessage{
|
||||||
|
ID: uuid.New(),
|
||||||
|
Type: "reindex",
|
||||||
|
Queue: "default",
|
||||||
|
Payload: nil,
|
||||||
|
ErrorMsg: "search engine not responding",
|
||||||
|
}
|
||||||
|
f1 := time.Now().Add(-5 * time.Minute)
|
||||||
|
f2 := time.Now().Add(-24 * time.Hour)
|
||||||
|
t1 := &DeadTask{ID: m1.ID, Type: m1.Type, Payload: m1.Payload, LastFailedAt: f1, ErrorMsg: m1.ErrorMsg}
|
||||||
|
t2 := &DeadTask{ID: m2.ID, Type: m2.Type, Payload: m2.Payload, LastFailedAt: f2, ErrorMsg: m2.ErrorMsg}
|
||||||
|
|
||||||
|
type deadEntry struct {
|
||||||
|
msg *TaskMessage
|
||||||
|
lastFailedAt time.Time
|
||||||
|
}
|
||||||
|
|
||||||
|
tests := []struct {
|
||||||
|
dead []deadEntry
|
||||||
|
want []*DeadTask
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
dead: []deadEntry{
|
||||||
|
{m1, f1},
|
||||||
|
{m2, f2},
|
||||||
|
},
|
||||||
|
want: []*DeadTask{t1, t2},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
dead: []deadEntry{},
|
||||||
|
want: []*DeadTask{},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, tc := range tests {
|
||||||
|
// clean up db before each test case.
|
||||||
|
if err := r.client.FlushDB().Err(); err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
// initialize the scheduled queue
|
||||||
|
for _, d := range tc.dead {
|
||||||
|
r.client.ZAdd(deadQ, &redis.Z{
|
||||||
|
Member: mustMarshal(t, d.msg),
|
||||||
|
Score: float64(d.lastFailedAt.Unix()),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
got, err := r.ListDead()
|
||||||
|
if err != nil {
|
||||||
|
t.Errorf("r.ListDead() = %v, %v, want %v, nil", got, err, tc.want)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
sortOpt := cmp.Transformer("SortMsg", func(in []*DeadTask) []*DeadTask {
|
||||||
|
out := append([]*DeadTask(nil), in...) // Copy input to avoid mutating it
|
||||||
|
sort.Slice(out, func(i, j int) bool {
|
||||||
|
return out[i].ID.String() < out[j].ID.String()
|
||||||
|
})
|
||||||
|
return out
|
||||||
|
})
|
||||||
|
|
||||||
|
if diff := cmp.Diff(tc.want, got, sortOpt, timeCmpOpt); diff != "" {
|
||||||
|
t.Errorf("r.ListDead() = %v, %v, want %v, nil; (-want, +got)\n%s", got, err, tc.want, diff)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var timeCmpOpt = EquateApproxTime(time.Second)
|
||||||
|
|
||||||
|
// TODO(hibiken): Replace this with cmpopts.EquateApproxTime once it becomes availalble.
|
||||||
|
// https://github.com/google/go-cmp/issues/166
|
||||||
|
//
|
||||||
|
// EquateApproxTime returns a Comparer options that
|
||||||
|
// determine two time.Time values to be equal if they
|
||||||
|
// are within the given time interval of one another.
|
||||||
|
// Note that if both times have a monotonic clock reading,
|
||||||
|
// the monotonic time difference will be used.
|
||||||
|
//
|
||||||
|
// The zero time is treated specially: it is only considered
|
||||||
|
// equal to another zero time value.
|
||||||
|
//
|
||||||
|
// It will panic if margin is negative.
|
||||||
|
func EquateApproxTime(margin time.Duration) cmp.Option {
|
||||||
|
if margin < 0 {
|
||||||
|
panic("negative duration in EquateApproxTime")
|
||||||
|
}
|
||||||
|
return cmp.FilterValues(func(x, y time.Time) bool {
|
||||||
|
return !x.IsZero() && !y.IsZero()
|
||||||
|
}, cmp.Comparer(timeApproximator{margin}.compare))
|
||||||
|
}
|
||||||
|
|
||||||
|
type timeApproximator struct {
|
||||||
|
margin time.Duration
|
||||||
|
}
|
||||||
|
|
||||||
|
func (a timeApproximator) compare(x, y time.Time) bool {
|
||||||
|
// Avoid subtracting times to avoid overflow when the
|
||||||
|
// difference is larger than the largest representible duration.
|
||||||
|
if x.After(y) {
|
||||||
|
// Ensure x is always before y
|
||||||
|
x, y = y, x
|
||||||
|
}
|
||||||
|
// We're within the margin if x+margin >= y.
|
||||||
|
// Note: time.Time doesn't have AfterOrEqual method hence the negation.
|
||||||
|
return !x.Add(a.margin).Before(y)
|
||||||
|
}
|
@ -58,64 +58,6 @@ type TaskMessage struct {
|
|||||||
ErrorMsg string
|
ErrorMsg string
|
||||||
}
|
}
|
||||||
|
|
||||||
// Stats represents a state of queues at a certain time.
|
|
||||||
type Stats struct {
|
|
||||||
Enqueued int
|
|
||||||
InProgress int
|
|
||||||
Scheduled int
|
|
||||||
Retry int
|
|
||||||
Dead int
|
|
||||||
Timestamp time.Time
|
|
||||||
}
|
|
||||||
|
|
||||||
// EnqueuedTask is a task in a queue and is ready to be processed.
|
|
||||||
// Note: This is read only and used for monitoring purpose.
|
|
||||||
type EnqueuedTask struct {
|
|
||||||
ID uuid.UUID
|
|
||||||
Type string
|
|
||||||
Payload map[string]interface{}
|
|
||||||
}
|
|
||||||
|
|
||||||
// InProgressTask is a task that's currently being processed.
|
|
||||||
// Note: This is read only and used for monitoring purpose.
|
|
||||||
type InProgressTask struct {
|
|
||||||
ID uuid.UUID
|
|
||||||
Type string
|
|
||||||
Payload map[string]interface{}
|
|
||||||
}
|
|
||||||
|
|
||||||
// ScheduledTask is a task that's scheduled to be processed in the future.
|
|
||||||
// Note: This is read only and used for monitoring purpose.
|
|
||||||
type ScheduledTask struct {
|
|
||||||
ID uuid.UUID
|
|
||||||
Type string
|
|
||||||
Payload map[string]interface{}
|
|
||||||
ProcessAt time.Time
|
|
||||||
}
|
|
||||||
|
|
||||||
// RetryTask is a task that's in retry queue because worker failed to process the task.
|
|
||||||
// Note: This is read only and used for monitoring purpose.
|
|
||||||
type RetryTask struct {
|
|
||||||
ID uuid.UUID
|
|
||||||
Type string
|
|
||||||
Payload map[string]interface{}
|
|
||||||
// TODO(hibiken): add LastFailedAt time.Time
|
|
||||||
ProcessAt time.Time
|
|
||||||
ErrorMsg string
|
|
||||||
Retried int
|
|
||||||
Retry int
|
|
||||||
}
|
|
||||||
|
|
||||||
// DeadTask is a task in that has exhausted all retries.
|
|
||||||
// Note: This is read only and used for monitoring purpose.
|
|
||||||
type DeadTask struct {
|
|
||||||
ID uuid.UUID
|
|
||||||
Type string
|
|
||||||
Payload map[string]interface{}
|
|
||||||
LastFailedAt time.Time
|
|
||||||
ErrorMsg string
|
|
||||||
}
|
|
||||||
|
|
||||||
// Close closes the connection with redis server.
|
// Close closes the connection with redis server.
|
||||||
func (r *RDB) Close() error {
|
func (r *RDB) Close() error {
|
||||||
return r.client.Close()
|
return r.client.Close()
|
||||||
@ -257,160 +199,4 @@ func (r *RDB) forward(from string) error {
|
|||||||
res, err := script.Run(r.client, []string{from, allQueues, defaultQ}, now).Result()
|
res, err := script.Run(r.client, []string{from, allQueues, defaultQ}, now).Result()
|
||||||
fmt.Printf("[DEBUG] got %d tasks from %q\n", len(res.([]interface{})), from)
|
fmt.Printf("[DEBUG] got %d tasks from %q\n", len(res.([]interface{})), from)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
// CurrentStats returns a current state of the queues.
|
|
||||||
func (r *RDB) CurrentStats() (*Stats, error) {
|
|
||||||
pipe := r.client.Pipeline()
|
|
||||||
qlen := pipe.LLen(defaultQ)
|
|
||||||
plen := pipe.LLen(inProgressQ)
|
|
||||||
slen := pipe.ZCard(scheduledQ)
|
|
||||||
rlen := pipe.ZCard(retryQ)
|
|
||||||
dlen := pipe.ZCard(deadQ)
|
|
||||||
_, err := pipe.Exec()
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
return &Stats{
|
|
||||||
Enqueued: int(qlen.Val()),
|
|
||||||
InProgress: int(plen.Val()),
|
|
||||||
Scheduled: int(slen.Val()),
|
|
||||||
Retry: int(rlen.Val()),
|
|
||||||
Dead: int(dlen.Val()),
|
|
||||||
Timestamp: time.Now(),
|
|
||||||
}, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// ListEnqueued returns all enqueued tasks that are ready to be processed.
|
|
||||||
func (r *RDB) ListEnqueued() ([]*EnqueuedTask, error) {
|
|
||||||
data, err := r.client.LRange(defaultQ, 0, -1).Result()
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
var tasks []*EnqueuedTask
|
|
||||||
for _, s := range data {
|
|
||||||
var msg TaskMessage
|
|
||||||
err := json.Unmarshal([]byte(s), &msg)
|
|
||||||
if err != nil {
|
|
||||||
continue // bad data, ignore and continue
|
|
||||||
}
|
|
||||||
tasks = append(tasks, &EnqueuedTask{
|
|
||||||
ID: msg.ID,
|
|
||||||
Type: msg.Type,
|
|
||||||
Payload: msg.Payload,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
return tasks, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// ListInProgress returns all tasks that are currently being processed.
|
|
||||||
func (r *RDB) ListInProgress() ([]*InProgressTask, error) {
|
|
||||||
data, err := r.client.LRange(defaultQ, 0, -1).Result()
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
var tasks []*InProgressTask
|
|
||||||
for _, s := range data {
|
|
||||||
var msg TaskMessage
|
|
||||||
err := json.Unmarshal([]byte(s), &msg)
|
|
||||||
if err != nil {
|
|
||||||
continue // bad data, ignore and continue
|
|
||||||
}
|
|
||||||
tasks = append(tasks, &InProgressTask{
|
|
||||||
ID: msg.ID,
|
|
||||||
Type: msg.Type,
|
|
||||||
Payload: msg.Payload,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
return tasks, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// ListScheduled returns all tasks that are scheduled to be processed
|
|
||||||
// in the future.
|
|
||||||
func (r *RDB) ListScheduled() ([]*ScheduledTask, error) {
|
|
||||||
data, err := r.client.ZRangeWithScores(scheduledQ, 0, -1).Result()
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
var tasks []*ScheduledTask
|
|
||||||
for _, z := range data {
|
|
||||||
s, ok := z.Member.(string)
|
|
||||||
if !ok {
|
|
||||||
continue // bad data, ignore and continue
|
|
||||||
}
|
|
||||||
var msg TaskMessage
|
|
||||||
err := json.Unmarshal([]byte(s), &msg)
|
|
||||||
if err != nil {
|
|
||||||
continue // bad data, ignore and continue
|
|
||||||
}
|
|
||||||
processAt := time.Unix(int64(z.Score), 0)
|
|
||||||
tasks = append(tasks, &ScheduledTask{
|
|
||||||
ID: msg.ID,
|
|
||||||
Type: msg.Type,
|
|
||||||
Payload: msg.Payload,
|
|
||||||
ProcessAt: processAt,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
return tasks, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// ListRetry returns all tasks that have failed before and willl be retried
|
|
||||||
// in the future.
|
|
||||||
func (r *RDB) ListRetry() ([]*RetryTask, error) {
|
|
||||||
data, err := r.client.ZRangeWithScores(retryQ, 0, -1).Result()
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
var tasks []*RetryTask
|
|
||||||
for _, z := range data {
|
|
||||||
s, ok := z.Member.(string)
|
|
||||||
if !ok {
|
|
||||||
continue // bad data, ignore and continue
|
|
||||||
}
|
|
||||||
var msg TaskMessage
|
|
||||||
err := json.Unmarshal([]byte(s), &msg)
|
|
||||||
if err != nil {
|
|
||||||
continue // bad data, ignore and continue
|
|
||||||
}
|
|
||||||
processAt := time.Unix(int64(z.Score), 0)
|
|
||||||
tasks = append(tasks, &RetryTask{
|
|
||||||
ID: msg.ID,
|
|
||||||
Type: msg.Type,
|
|
||||||
Payload: msg.Payload,
|
|
||||||
ErrorMsg: msg.ErrorMsg,
|
|
||||||
Retry: msg.Retry,
|
|
||||||
Retried: msg.Retried,
|
|
||||||
ProcessAt: processAt,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
return tasks, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// ListDead returns all tasks that have exhausted its retry limit.
|
|
||||||
func (r *RDB) ListDead() ([]*DeadTask, error) {
|
|
||||||
data, err := r.client.ZRangeWithScores(deadQ, 0, -1).Result()
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
var tasks []*DeadTask
|
|
||||||
for _, z := range data {
|
|
||||||
s, ok := z.Member.(string)
|
|
||||||
if !ok {
|
|
||||||
continue // bad data, ignore and continue
|
|
||||||
}
|
|
||||||
var msg TaskMessage
|
|
||||||
err := json.Unmarshal([]byte(s), &msg)
|
|
||||||
if err != nil {
|
|
||||||
continue // bad data, ignore and continue
|
|
||||||
}
|
|
||||||
lastFailedAt := time.Unix(int64(z.Score), 0)
|
|
||||||
tasks = append(tasks, &DeadTask{
|
|
||||||
ID: msg.ID,
|
|
||||||
Type: msg.Type,
|
|
||||||
Payload: msg.Payload,
|
|
||||||
ErrorMsg: msg.ErrorMsg,
|
|
||||||
LastFailedAt: lastFailedAt,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
return tasks, nil
|
|
||||||
}
|
|
Loading…
Reference in New Issue
Block a user