mirror of
https://github.com/hibiken/asynq.git
synced 2024-11-10 11:31:58 +08:00
Update enqueue methods in RDB
This commit is contained in:
parent
dbf140a767
commit
47e9ba4eba
@ -5,8 +5,6 @@
|
|||||||
package rdb
|
package rdb
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/json"
|
|
||||||
"errors"
|
|
||||||
"fmt"
|
"fmt"
|
||||||
"strings"
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
@ -57,8 +55,6 @@ type DailyStats struct {
|
|||||||
Time time.Time
|
Time time.Time
|
||||||
}
|
}
|
||||||
|
|
||||||
var ErrQueueNotFound = errors.New("rdb: queue does not exist")
|
|
||||||
|
|
||||||
// KEYS[1] -> asynq:<qname>
|
// KEYS[1] -> asynq:<qname>
|
||||||
// KEYS[2] -> asynq:<qname>:in_progress
|
// KEYS[2] -> asynq:<qname>:in_progress
|
||||||
// KEYS[3] -> asynq:<qname>:scheduled
|
// KEYS[3] -> asynq:<qname>:scheduled
|
||||||
@ -104,7 +100,7 @@ func (r *RDB) CurrentStats(qname string) (*Stats, error) {
|
|||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
if !exists {
|
if !exists {
|
||||||
return nil, ErrQueueNotFound
|
return nil, &ErrQueueNotFound{qname}
|
||||||
}
|
}
|
||||||
now := time.Now()
|
now := time.Now()
|
||||||
res, err := currentStatsCmd.Run(r.client, []string{
|
res, err := currentStatsCmd.Run(r.client, []string{
|
||||||
@ -138,7 +134,7 @@ func (r *RDB) CurrentStats(qname string) (*Stats, error) {
|
|||||||
stats.InProgress = val
|
stats.InProgress = val
|
||||||
case base.ScheduledKey(qname):
|
case base.ScheduledKey(qname):
|
||||||
stats.Scheduled = val
|
stats.Scheduled = val
|
||||||
case ase.RetryKey(qname):
|
case base.RetryKey(qname):
|
||||||
stats.Retry = val
|
stats.Retry = val
|
||||||
case base.DeadKey(qname):
|
case base.DeadKey(qname):
|
||||||
stats.Dead = val
|
stats.Dead = val
|
||||||
@ -181,7 +177,7 @@ func (r *RDB) HistoricalStats(qname string, n int) ([]*DailyStats, error) {
|
|||||||
ts := now.Add(-time.Duration(i) * day)
|
ts := now.Add(-time.Duration(i) * day)
|
||||||
days = append(days, ts)
|
days = append(days, ts)
|
||||||
keys = append(keys, base.ProcessedKey(qname, ts))
|
keys = append(keys, base.ProcessedKey(qname, ts))
|
||||||
keys = append(keys, base.FailureKey(qname, ts))
|
keys = append(keys, base.FailedKey(qname, ts))
|
||||||
}
|
}
|
||||||
res, err := historicalStatsCmd.Run(r.client, keys).Result()
|
res, err := historicalStatsCmd.Run(r.client, keys).Result()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -247,11 +243,10 @@ func (p Pagination) stop() int64 {
|
|||||||
|
|
||||||
// ListEnqueued returns enqueued tasks that are ready to be processed.
|
// ListEnqueued returns enqueued tasks that are ready to be processed.
|
||||||
func (r *RDB) ListEnqueued(qname string, pgn Pagination) ([]*base.TaskMessage, error) {
|
func (r *RDB) ListEnqueued(qname string, pgn Pagination) ([]*base.TaskMessage, error) {
|
||||||
qkey := base.QueueKey(qname)
|
if !r.client.SIsMember(base.AllQueues, qname).Val() {
|
||||||
if !r.client.SIsMember(base.AllQueues, qkey).Val() {
|
|
||||||
return nil, fmt.Errorf("queue %q does not exist", qname)
|
return nil, fmt.Errorf("queue %q does not exist", qname)
|
||||||
}
|
}
|
||||||
return r.listMessages(qkey, pgn)
|
return r.listMessages(base.QueueKey(qname), pgn)
|
||||||
}
|
}
|
||||||
|
|
||||||
// ListInProgress returns all tasks that are currently being processed for the given queue.
|
// ListInProgress returns all tasks that are currently being processed for the given queue.
|
||||||
@ -321,11 +316,11 @@ func (r *RDB) listZSetEntries(key string, pgn Pagination) ([]base.Z, error) {
|
|||||||
return res, nil
|
return res, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// EnqueueDeadTask finds a task that matches the given id and score from dead queue
|
// EnqueueDeadTask finds a dead task that matches the given id and score from
|
||||||
// and enqueues it for processing. If a task that matches the id and score
|
// the given queue and enqueues it for processing.
|
||||||
// does not exist, it returns ErrTaskNotFound.
|
//If a task that matches the id and score does not exist, it returns ErrTaskNotFound.
|
||||||
func (r *RDB) EnqueueDeadTask(id uuid.UUID, score int64) error {
|
func (r *RDB) EnqueueDeadTask(qname string, id uuid.UUID, score int64) error {
|
||||||
n, err := r.removeAndEnqueue(base.DeadQueue, id.String(), float64(score))
|
n, err := r.removeAndEnqueue(base.DeadKey(qname), base.QueueKey(qname), id.String(), float64(score))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
@ -335,11 +330,11 @@ func (r *RDB) EnqueueDeadTask(id uuid.UUID, score int64) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// EnqueueRetryTask finds a task that matches the given id and score from retry queue
|
// EnqueueRetryTask finds a retry task that matches the given id and score from
|
||||||
// and enqueues it for processing. If a task that matches the id and score
|
// the given queue and enqueues it for processing.
|
||||||
// does not exist, it returns ErrTaskNotFound.
|
// If a task that matches the id and score does not exist, it returns ErrTaskNotFound.
|
||||||
func (r *RDB) EnqueueRetryTask(id uuid.UUID, score int64) error {
|
func (r *RDB) EnqueueRetryTask(qname string, id uuid.UUID, score int64) error {
|
||||||
n, err := r.removeAndEnqueue(base.RetryQueue, id.String(), float64(score))
|
n, err := r.removeAndEnqueue(base.RetryKey(qname), base.QueueKey(qname), id.String(), float64(score))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
@ -349,11 +344,11 @@ func (r *RDB) EnqueueRetryTask(id uuid.UUID, score int64) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// EnqueueScheduledTask finds a task that matches the given id and score from scheduled queue
|
// EnqueueScheduledTask finds a scheduled task that matches the given id and score from
|
||||||
// and enqueues it for processing. If a task that matches the id and score does not
|
// from the given queue and enqueues it for processing.
|
||||||
// exist, it returns ErrTaskNotFound.
|
// If a task that matches the id and score does not exist, it returns ErrTaskNotFound.
|
||||||
func (r *RDB) EnqueueScheduledTask(id uuid.UUID, score int64) error {
|
func (r *RDB) EnqueueScheduledTask(qname string, id uuid.UUID, score int64) error {
|
||||||
n, err := r.removeAndEnqueue(base.ScheduledQueue, id.String(), float64(score))
|
n, err := r.removeAndEnqueue(base.ScheduledKey(qname), base.QueueKey(qname), id.String(), float64(score))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
@ -363,22 +358,22 @@ func (r *RDB) EnqueueScheduledTask(id uuid.UUID, score int64) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// EnqueueAllScheduledTasks enqueues all tasks from scheduled queue
|
// EnqueueAllScheduledTasks enqueues all scheduled tasks from the given queue
|
||||||
// and returns the number of tasks enqueued.
|
// and returns the number of tasks enqueued.
|
||||||
func (r *RDB) EnqueueAllScheduledTasks() (int64, error) {
|
func (r *RDB) EnqueueAllScheduledTasks(qname string) (int64, error) {
|
||||||
return r.removeAndEnqueueAll(base.ScheduledQueue)
|
return r.removeAndEnqueueAll(base.ScheduledKey(qname), base.QueueKey(qname))
|
||||||
}
|
}
|
||||||
|
|
||||||
// EnqueueAllRetryTasks enqueues all tasks from retry queue
|
// EnqueueAllRetryTasks enqueues all retry tasks from the given queue
|
||||||
// and returns the number of tasks enqueued.
|
// and returns the number of tasks enqueued.
|
||||||
func (r *RDB) EnqueueAllRetryTasks() (int64, error) {
|
func (r *RDB) EnqueueAllRetryTasks(qname string) (int64, error) {
|
||||||
return r.removeAndEnqueueAll(base.RetryQueue)
|
return r.removeAndEnqueueAll(base.RetryKey(qname), base.QueueKey(qname))
|
||||||
}
|
}
|
||||||
|
|
||||||
// EnqueueAllDeadTasks enqueues all tasks from dead queue
|
// EnqueueAllDeadTasks enqueues all tasks from dead queue
|
||||||
// and returns the number of tasks enqueued.
|
// and returns the number of tasks enqueued.
|
||||||
func (r *RDB) EnqueueAllDeadTasks() (int64, error) {
|
func (r *RDB) EnqueueAllDeadTasks(qname string) (int64, error) {
|
||||||
return r.removeAndEnqueueAll(base.DeadQueue)
|
return r.removeAndEnqueueAll(base.DeadKey(qname), base.QueueKey(qname))
|
||||||
}
|
}
|
||||||
|
|
||||||
var removeAndEnqueueCmd = redis.NewScript(`
|
var removeAndEnqueueCmd = redis.NewScript(`
|
||||||
@ -386,16 +381,15 @@ local msgs = redis.call("ZRANGEBYSCORE", KEYS[1], ARGV[1], ARGV[1])
|
|||||||
for _, msg in ipairs(msgs) do
|
for _, msg in ipairs(msgs) do
|
||||||
local decoded = cjson.decode(msg)
|
local decoded = cjson.decode(msg)
|
||||||
if decoded["ID"] == ARGV[2] then
|
if decoded["ID"] == ARGV[2] then
|
||||||
local qkey = ARGV[3] .. decoded["Queue"]
|
redis.call("LPUSH", KEYS[2], msg)
|
||||||
redis.call("LPUSH", qkey, msg)
|
|
||||||
redis.call("ZREM", KEYS[1], msg)
|
redis.call("ZREM", KEYS[1], msg)
|
||||||
return 1
|
return 1
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
return 0`)
|
return 0`)
|
||||||
|
|
||||||
func (r *RDB) removeAndEnqueue(zset, id string, score float64) (int64, error) {
|
func (r *RDB) removeAndEnqueue(zset, qkey, id string, score float64) (int64, error) {
|
||||||
res, err := removeAndEnqueueCmd.Run(r.client, []string{zset}, score, id, base.QueuePrefix).Result()
|
res, err := removeAndEnqueueCmd.Run(r.client, []string{zset, qkey}, score, id).Result()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return 0, err
|
return 0, err
|
||||||
}
|
}
|
||||||
@ -409,15 +403,13 @@ func (r *RDB) removeAndEnqueue(zset, id string, score float64) (int64, error) {
|
|||||||
var removeAndEnqueueAllCmd = redis.NewScript(`
|
var removeAndEnqueueAllCmd = redis.NewScript(`
|
||||||
local msgs = redis.call("ZRANGE", KEYS[1], 0, -1)
|
local msgs = redis.call("ZRANGE", KEYS[1], 0, -1)
|
||||||
for _, msg in ipairs(msgs) do
|
for _, msg in ipairs(msgs) do
|
||||||
local decoded = cjson.decode(msg)
|
redis.call("LPUSH", KEYS[2], msg)
|
||||||
local qkey = ARGV[1] .. decoded["Queue"]
|
|
||||||
redis.call("LPUSH", qkey, msg)
|
|
||||||
redis.call("ZREM", KEYS[1], msg)
|
redis.call("ZREM", KEYS[1], msg)
|
||||||
end
|
end
|
||||||
return table.getn(msgs)`)
|
return table.getn(msgs)`)
|
||||||
|
|
||||||
func (r *RDB) removeAndEnqueueAll(zset string) (int64, error) {
|
func (r *RDB) removeAndEnqueueAll(zset, qkey string) (int64, error) {
|
||||||
res, err := removeAndEnqueueAllCmd.Run(r.client, []string{zset}, base.QueuePrefix).Result()
|
res, err := removeAndEnqueueAllCmd.Run(r.client, []string{zset, qkey}).Result()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return 0, err
|
return 0, err
|
||||||
}
|
}
|
||||||
@ -428,6 +420,7 @@ func (r *RDB) removeAndEnqueueAll(zset string) (int64, error) {
|
|||||||
return n, nil
|
return n, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
// KillRetryTask finds a task that matches the given id and score from retry queue
|
// KillRetryTask finds a task that matches the given id and score from retry queue
|
||||||
// and moves it to dead queue. If a task that maches the id and score does not exist,
|
// and moves it to dead queue. If a task that maches the id and score does not exist,
|
||||||
// it returns ErrTaskNotFound.
|
// it returns ErrTaskNotFound.
|
||||||
@ -617,6 +610,7 @@ func (r *RDB) deleteAll(key string) (int64, error) {
|
|||||||
}
|
}
|
||||||
return n, nil
|
return n, nil
|
||||||
}
|
}
|
||||||
|
*/
|
||||||
|
|
||||||
// ErrQueueNotFound indicates specified queue does not exist.
|
// ErrQueueNotFound indicates specified queue does not exist.
|
||||||
type ErrQueueNotFound struct {
|
type ErrQueueNotFound struct {
|
||||||
@ -636,6 +630,7 @@ func (e *ErrQueueNotEmpty) Error() string {
|
|||||||
return fmt.Sprintf("queue %q is not empty", e.qname)
|
return fmt.Sprintf("queue %q is not empty", e.qname)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
// Skip checking whether queue is empty before removing.
|
// Skip checking whether queue is empty before removing.
|
||||||
var removeQueueForceCmd = redis.NewScript(`
|
var removeQueueForceCmd = redis.NewScript(`
|
||||||
local n = redis.call("SREM", KEYS[1], KEYS[2])
|
local n = redis.call("SREM", KEYS[1], KEYS[2])
|
||||||
@ -758,15 +753,16 @@ func (r *RDB) ListWorkers() ([]*base.WorkerInfo, error) {
|
|||||||
}
|
}
|
||||||
return workers, nil
|
return workers, nil
|
||||||
}
|
}
|
||||||
|
*/
|
||||||
|
|
||||||
// Pause pauses processing of tasks from the given queue.
|
// Pause pauses processing of tasks from the given queue.
|
||||||
func (r *RDB) Pause(qname string) error {
|
func (r *RDB) Pause(qname string) error {
|
||||||
key := base.PauseKey(qname)
|
key := base.PausedKey(qname)
|
||||||
exists, err := r.client.SetNX(key, time.Now().Unix(), 0).Result()
|
ok, err := r.client.SetNX(key, time.Now().Unix(), 0).Result()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
if exists {
|
if !ok {
|
||||||
return fmt.Errorf("queue %q is already paused", qname)
|
return fmt.Errorf("queue %q is already paused", qname)
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
@ -774,7 +770,7 @@ func (r *RDB) Pause(qname string) error {
|
|||||||
|
|
||||||
// Unpause resumes processing of tasks from the given queue.
|
// Unpause resumes processing of tasks from the given queue.
|
||||||
func (r *RDB) Unpause(qname string) error {
|
func (r *RDB) Unpause(qname string) error {
|
||||||
key := base.PauseKey(qname)
|
key := base.PausedKey(qname)
|
||||||
deleted, err := r.client.Del(key).Result()
|
deleted, err := r.client.Del(key).Result()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
|
@ -30,15 +30,17 @@ func TestAllQueues(t *testing.T) {
|
|||||||
|
|
||||||
for _, tc := range tests {
|
for _, tc := range tests {
|
||||||
h.FlushDB(t, r.client)
|
h.FlushDB(t, r.client)
|
||||||
if err := r.client.SAdd(base.AllQueues, tc.queues...).Err(); err != nil {
|
for _, qname := range tc.queues {
|
||||||
t.Fatal("could not initialize all queue set")
|
if err := r.client.SAdd(base.AllQueues, qname).Err(); err != nil {
|
||||||
|
t.Fatalf("could not initialize all queue set: %v", err)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
got, err := r.AllQueues()
|
got, err := r.AllQueues()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Errorf("AllQueues() returned an error: %v", err)
|
t.Errorf("AllQueues() returned an error: %v", err)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
if diff := cmp.Diff(tc.queues, got, h.SortStringSliceOpt); diff != nil {
|
if diff := cmp.Diff(tc.queues, got, h.SortStringSliceOpt); diff != "" {
|
||||||
t.Errorf("AllQueues() = %v, want %v; (-want, +got)\n%s", got, tc.queues, diff)
|
t.Errorf("AllQueues() = %v, want %v; (-want, +got)\n%s", got, tc.queues, diff)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -193,7 +195,7 @@ func TestCurrentStats(t *testing.T) {
|
|||||||
r.client.Set(processedKey, n, 0)
|
r.client.Set(processedKey, n, 0)
|
||||||
}
|
}
|
||||||
for qname, n := range tc.failed {
|
for qname, n := range tc.failed {
|
||||||
failedKey := base.FailedKey(now)
|
failedKey := base.FailedKey(qname, now)
|
||||||
r.client.Set(failedKey, n, 0)
|
r.client.Set(failedKey, n, 0)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -215,8 +217,8 @@ func TestCurrentStatsWithNonExistentQueue(t *testing.T) {
|
|||||||
|
|
||||||
qname := "non-existent"
|
qname := "non-existent"
|
||||||
got, err := r.CurrentStats(qname)
|
got, err := r.CurrentStats(qname)
|
||||||
if err != ErrQueueNotFound {
|
if err == nil {
|
||||||
t.Fatalf("r.CurrentStats(%q) = %v, %v, want nil, %v", qname, got, err, ErrQueueNotFound)
|
t.Fatalf("r.CurrentStats(%q) = %v, %v, want nil, %v", qname, got, err, &ErrQueueNotFound{qname})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -225,7 +227,7 @@ func TestHistoricalStats(t *testing.T) {
|
|||||||
now := time.Now().UTC()
|
now := time.Now().UTC()
|
||||||
|
|
||||||
tests := []struct {
|
tests := []struct {
|
||||||
qname // queue of interest
|
qname string // queue of interest
|
||||||
n int // number of days
|
n int // number of days
|
||||||
}{
|
}{
|
||||||
{"default", 90},
|
{"default", 90},
|
||||||
@ -240,7 +242,7 @@ func TestHistoricalStats(t *testing.T) {
|
|||||||
for i := 0; i < tc.n; i++ {
|
for i := 0; i < tc.n; i++ {
|
||||||
ts := now.Add(-time.Duration(i) * 24 * time.Hour)
|
ts := now.Add(-time.Duration(i) * 24 * time.Hour)
|
||||||
processedKey := base.ProcessedKey(tc.qname, ts)
|
processedKey := base.ProcessedKey(tc.qname, ts)
|
||||||
failedKey := base.FailureKey(tc.qname, ts)
|
failedKey := base.FailedKey(tc.qname, ts)
|
||||||
r.client.Set(processedKey, (i+1)*1000, 0)
|
r.client.Set(processedKey, (i+1)*1000, 0)
|
||||||
r.client.Set(failedKey, (i+1)*10, 0)
|
r.client.Set(failedKey, (i+1)*10, 0)
|
||||||
}
|
}
|
||||||
@ -344,9 +346,7 @@ func TestListEnqueued(t *testing.T) {
|
|||||||
|
|
||||||
for _, tc := range tests {
|
for _, tc := range tests {
|
||||||
h.FlushDB(t, r.client) // clean up db before each test case
|
h.FlushDB(t, r.client) // clean up db before each test case
|
||||||
for qname, msgs := range tc.enqueued {
|
h.SeedAllEnqueuedQueues(t, r.client, tc.enqueued)
|
||||||
h.SeedEnqueuedQueue(t, r.client, msgs, qname)
|
|
||||||
}
|
|
||||||
|
|
||||||
got, err := r.ListEnqueued(tc.qname, Pagination{Size: 20, Page: 0})
|
got, err := r.ListEnqueued(tc.qname, Pagination{Size: 20, Page: 0})
|
||||||
op := fmt.Sprintf("r.ListEnqueued(%q, Pagination{Size: 20, Page: 0})", tc.qname)
|
op := fmt.Sprintf("r.ListEnqueued(%q, Pagination{Size: 20, Page: 0})", tc.qname)
|
||||||
@ -454,7 +454,7 @@ func TestListInProgress(t *testing.T) {
|
|||||||
"default": {},
|
"default": {},
|
||||||
},
|
},
|
||||||
qname: "default",
|
qname: "default",
|
||||||
want: []*base.TaskMessage{},
|
want: []*base.TaskMessage(nil),
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -747,7 +747,7 @@ func TestListRetry(t *testing.T) {
|
|||||||
h.FlushDB(t, r.client) // clean up db before each test case
|
h.FlushDB(t, r.client) // clean up db before each test case
|
||||||
h.SeedAllRetryQueues(t, r.client, tc.retry)
|
h.SeedAllRetryQueues(t, r.client, tc.retry)
|
||||||
|
|
||||||
got, err := r.ListRetry(Pagination{Size: 20, Page: 0})
|
got, err := r.ListRetry(tc.qname, Pagination{Size: 20, Page: 0})
|
||||||
op := fmt.Sprintf("r.ListRetry(%q, Pagination{Size: 20, Page: 0})", tc.qname)
|
op := fmt.Sprintf("r.ListRetry(%q, Pagination{Size: 20, Page: 0})", tc.qname)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Errorf("%s = %v, %v, want %v, nil", op, got, err, tc.want)
|
t.Errorf("%s = %v, %v, want %v, nil", op, got, err, tc.want)
|
||||||
@ -898,7 +898,7 @@ func TestListDead(t *testing.T) {
|
|||||||
h.FlushDB(t, r.client) // clean up db before each test case
|
h.FlushDB(t, r.client) // clean up db before each test case
|
||||||
h.SeedAllDeadQueues(t, r.client, tc.dead)
|
h.SeedAllDeadQueues(t, r.client, tc.dead)
|
||||||
|
|
||||||
got, err := r.ListDead(Pagination{Size: 20, Page: 0})
|
got, err := r.ListDead(tc.qname, Pagination{Size: 20, Page: 0})
|
||||||
op := fmt.Sprintf("r.ListDead(%q, Pagination{Size: 20, Page: 0})", tc.qname)
|
op := fmt.Sprintf("r.ListDead(%q, Pagination{Size: 20, Page: 0})", tc.qname)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Errorf("%s = %v, %v, want %v, nil", op, got, err, tc.want)
|
t.Errorf("%s = %v, %v, want %v, nil", op, got, err, tc.want)
|
||||||
@ -976,57 +976,75 @@ func TestEnqueueDeadTask(t *testing.T) {
|
|||||||
r := setup(t)
|
r := setup(t)
|
||||||
t1 := h.NewTaskMessage("send_email", nil)
|
t1 := h.NewTaskMessage("send_email", nil)
|
||||||
t2 := h.NewTaskMessage("gen_thumbnail", nil)
|
t2 := h.NewTaskMessage("gen_thumbnail", nil)
|
||||||
t3 := h.NewTaskMessage("send_notification", nil)
|
t3 := h.NewTaskMessageWithQueue("send_notification", nil, "critical")
|
||||||
t3.Queue = "critical"
|
|
||||||
s1 := time.Now().Add(-5 * time.Minute).Unix()
|
s1 := time.Now().Add(-5 * time.Minute).Unix()
|
||||||
s2 := time.Now().Add(-time.Hour).Unix()
|
s2 := time.Now().Add(-time.Hour).Unix()
|
||||||
|
|
||||||
tests := []struct {
|
tests := []struct {
|
||||||
dead []base.Z
|
dead map[string][]base.Z
|
||||||
|
qname string
|
||||||
score int64
|
score int64
|
||||||
id uuid.UUID
|
id uuid.UUID
|
||||||
want error // expected return value from calling EnqueueDeadTask
|
want error // expected return value from calling EnqueueDeadTask
|
||||||
wantDead []*base.TaskMessage
|
wantDead map[string][]*base.TaskMessage
|
||||||
wantEnqueued map[string][]*base.TaskMessage
|
wantEnqueued map[string][]*base.TaskMessage
|
||||||
}{
|
}{
|
||||||
{
|
{
|
||||||
dead: []base.Z{
|
dead: map[string][]base.Z{
|
||||||
|
"default": {
|
||||||
{Message: t1, Score: s1},
|
{Message: t1, Score: s1},
|
||||||
{Message: t2, Score: s2},
|
{Message: t2, Score: s2},
|
||||||
},
|
},
|
||||||
|
},
|
||||||
|
qname: "default",
|
||||||
score: s2,
|
score: s2,
|
||||||
id: t2.ID,
|
id: t2.ID,
|
||||||
want: nil,
|
want: nil,
|
||||||
wantDead: []*base.TaskMessage{t1},
|
wantDead: map[string][]*base.TaskMessage{
|
||||||
|
"default": {t1},
|
||||||
|
},
|
||||||
wantEnqueued: map[string][]*base.TaskMessage{
|
wantEnqueued: map[string][]*base.TaskMessage{
|
||||||
base.DefaultQueueName: {t2},
|
"default": {t2},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
dead: []base.Z{
|
dead: map[string][]base.Z{
|
||||||
|
"default": {
|
||||||
{Message: t1, Score: s1},
|
{Message: t1, Score: s1},
|
||||||
{Message: t2, Score: s2},
|
{Message: t2, Score: s2},
|
||||||
},
|
},
|
||||||
|
},
|
||||||
|
qname: "default",
|
||||||
score: 123,
|
score: 123,
|
||||||
id: t2.ID,
|
id: t2.ID,
|
||||||
want: ErrTaskNotFound,
|
want: ErrTaskNotFound,
|
||||||
wantDead: []*base.TaskMessage{t1, t2},
|
wantDead: map[string][]*base.TaskMessage{
|
||||||
|
"default": {t1, t2},
|
||||||
|
},
|
||||||
wantEnqueued: map[string][]*base.TaskMessage{
|
wantEnqueued: map[string][]*base.TaskMessage{
|
||||||
base.DefaultQueueName: {},
|
"default": {},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
dead: []base.Z{
|
dead: map[string][]base.Z{
|
||||||
|
"default": {
|
||||||
{Message: t1, Score: s1},
|
{Message: t1, Score: s1},
|
||||||
{Message: t2, Score: s2},
|
{Message: t2, Score: s2},
|
||||||
|
},
|
||||||
|
"critical": {
|
||||||
{Message: t3, Score: s1},
|
{Message: t3, Score: s1},
|
||||||
},
|
},
|
||||||
|
},
|
||||||
|
qname: "critical",
|
||||||
score: s1,
|
score: s1,
|
||||||
id: t3.ID,
|
id: t3.ID,
|
||||||
want: nil,
|
want: nil,
|
||||||
wantDead: []*base.TaskMessage{t1, t2},
|
wantDead: map[string][]*base.TaskMessage{
|
||||||
|
"default": {t1, t2},
|
||||||
|
"critical": {},
|
||||||
|
},
|
||||||
wantEnqueued: map[string][]*base.TaskMessage{
|
wantEnqueued: map[string][]*base.TaskMessage{
|
||||||
base.DefaultQueueName: {},
|
"default": {},
|
||||||
"critical": {t3},
|
"critical": {t3},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
@ -1034,11 +1052,11 @@ func TestEnqueueDeadTask(t *testing.T) {
|
|||||||
|
|
||||||
for _, tc := range tests {
|
for _, tc := range tests {
|
||||||
h.FlushDB(t, r.client) // clean up db before each test case
|
h.FlushDB(t, r.client) // clean up db before each test case
|
||||||
h.SeedDeadQueue(t, r.client, tc.dead)
|
h.SeedAllDeadQueues(t, r.client, tc.dead)
|
||||||
|
|
||||||
got := r.EnqueueDeadTask(tc.id, tc.score)
|
got := r.EnqueueDeadTask(tc.qname, tc.id, tc.score)
|
||||||
if got != tc.want {
|
if got != tc.want {
|
||||||
t.Errorf("r.EnqueueDeadTask(%s, %d) = %v, want %v", tc.id, tc.score, got, tc.want)
|
t.Errorf("r.EnqueueDeadTask(%q, %s, %d) = %v, want %v", tc.qname, tc.id, tc.score, got, tc.want)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1049,9 +1067,11 @@ func TestEnqueueDeadTask(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
gotDead := h.GetDeadMessages(t, r.client)
|
for qname, want := range tc.wantDead {
|
||||||
if diff := cmp.Diff(tc.wantDead, gotDead, h.SortMsgOpt); diff != "" {
|
gotDead := h.GetDeadMessages(t, r.client, qname)
|
||||||
t.Errorf("mismatch found in %q, (-want, +got)\n%s", base.DeadQueue, diff)
|
if diff := cmp.Diff(want, gotDead, h.SortMsgOpt); diff != "" {
|
||||||
|
t.Errorf("mismatch found in %q, (-want, +got)\n%s", base.DeadKey(qname), diff)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1061,56 +1081,74 @@ func TestEnqueueRetryTask(t *testing.T) {
|
|||||||
|
|
||||||
t1 := h.NewTaskMessage("send_email", nil)
|
t1 := h.NewTaskMessage("send_email", nil)
|
||||||
t2 := h.NewTaskMessage("gen_thumbnail", nil)
|
t2 := h.NewTaskMessage("gen_thumbnail", nil)
|
||||||
t3 := h.NewTaskMessage("send_notification", nil)
|
t3 := h.NewTaskMessageWithQueue("send_notification", nil, "low")
|
||||||
t3.Queue = "low"
|
|
||||||
s1 := time.Now().Add(-5 * time.Minute).Unix()
|
s1 := time.Now().Add(-5 * time.Minute).Unix()
|
||||||
s2 := time.Now().Add(-time.Hour).Unix()
|
s2 := time.Now().Add(-time.Hour).Unix()
|
||||||
tests := []struct {
|
tests := []struct {
|
||||||
retry []base.Z
|
retry map[string][]base.Z
|
||||||
|
qname string
|
||||||
score int64
|
score int64
|
||||||
id uuid.UUID
|
id uuid.UUID
|
||||||
want error // expected return value from calling EnqueueRetryTask
|
want error // expected return value from calling EnqueueRetryTask
|
||||||
wantRetry []*base.TaskMessage
|
wantRetry map[string][]*base.TaskMessage
|
||||||
wantEnqueued map[string][]*base.TaskMessage
|
wantEnqueued map[string][]*base.TaskMessage
|
||||||
}{
|
}{
|
||||||
{
|
{
|
||||||
retry: []base.Z{
|
retry: map[string][]base.Z{
|
||||||
|
"default": {
|
||||||
{Message: t1, Score: s1},
|
{Message: t1, Score: s1},
|
||||||
{Message: t2, Score: s2},
|
{Message: t2, Score: s2},
|
||||||
},
|
},
|
||||||
|
},
|
||||||
|
qname: "default",
|
||||||
score: s2,
|
score: s2,
|
||||||
id: t2.ID,
|
id: t2.ID,
|
||||||
want: nil,
|
want: nil,
|
||||||
wantRetry: []*base.TaskMessage{t1},
|
wantRetry: map[string][]*base.TaskMessage{
|
||||||
|
"default": {t1},
|
||||||
|
},
|
||||||
wantEnqueued: map[string][]*base.TaskMessage{
|
wantEnqueued: map[string][]*base.TaskMessage{
|
||||||
base.DefaultQueueName: {t2},
|
"default": {t2},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
retry: []base.Z{
|
retry: map[string][]base.Z{
|
||||||
|
"default": {
|
||||||
{Message: t1, Score: s1},
|
{Message: t1, Score: s1},
|
||||||
{Message: t2, Score: s2},
|
{Message: t2, Score: s2},
|
||||||
},
|
},
|
||||||
|
},
|
||||||
|
qname: "default",
|
||||||
score: 123,
|
score: 123,
|
||||||
id: t2.ID,
|
id: t2.ID,
|
||||||
want: ErrTaskNotFound,
|
want: ErrTaskNotFound,
|
||||||
wantRetry: []*base.TaskMessage{t1, t2},
|
wantRetry: map[string][]*base.TaskMessage{
|
||||||
|
"default": {t1, t2},
|
||||||
|
},
|
||||||
wantEnqueued: map[string][]*base.TaskMessage{
|
wantEnqueued: map[string][]*base.TaskMessage{
|
||||||
base.DefaultQueueName: {},
|
"default": {},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
retry: []base.Z{
|
retry: map[string][]base.Z{
|
||||||
|
"default": {
|
||||||
{Message: t1, Score: s1},
|
{Message: t1, Score: s1},
|
||||||
{Message: t2, Score: s2},
|
{Message: t2, Score: s2},
|
||||||
|
},
|
||||||
|
"low": {
|
||||||
{Message: t3, Score: s2},
|
{Message: t3, Score: s2},
|
||||||
},
|
},
|
||||||
|
},
|
||||||
|
qname: "low",
|
||||||
score: s2,
|
score: s2,
|
||||||
id: t3.ID,
|
id: t3.ID,
|
||||||
want: nil,
|
want: nil,
|
||||||
wantRetry: []*base.TaskMessage{t1, t2},
|
wantRetry: map[string][]*base.TaskMessage{
|
||||||
|
"default": {t1, t2},
|
||||||
|
"low": {},
|
||||||
|
},
|
||||||
wantEnqueued: map[string][]*base.TaskMessage{
|
wantEnqueued: map[string][]*base.TaskMessage{
|
||||||
base.DefaultQueueName: {},
|
"default": {},
|
||||||
"low": {t3},
|
"low": {t3},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
@ -1118,11 +1156,11 @@ func TestEnqueueRetryTask(t *testing.T) {
|
|||||||
|
|
||||||
for _, tc := range tests {
|
for _, tc := range tests {
|
||||||
h.FlushDB(t, r.client) // clean up db before each test case
|
h.FlushDB(t, r.client) // clean up db before each test case
|
||||||
h.SeedRetryQueue(t, r.client, tc.retry) // initialize retry queue
|
h.SeedAllRetryQueues(t, r.client, tc.retry) // initialize retry queue
|
||||||
|
|
||||||
got := r.EnqueueRetryTask(tc.id, tc.score)
|
got := r.EnqueueRetryTask(tc.qname, tc.id, tc.score)
|
||||||
if got != tc.want {
|
if got != tc.want {
|
||||||
t.Errorf("r.EnqueueRetryTask(%s, %d) = %v, want %v", tc.id, tc.score, got, tc.want)
|
t.Errorf("r.EnqueueRetryTask(%q, %s, %d) = %v, want %v", tc.qname, tc.id, tc.score, got, tc.want)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1133,9 +1171,11 @@ func TestEnqueueRetryTask(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
gotRetry := h.GetRetryMessages(t, r.client)
|
for qname, want := range tc.wantRetry {
|
||||||
if diff := cmp.Diff(tc.wantRetry, gotRetry, h.SortMsgOpt); diff != "" {
|
gotRetry := h.GetRetryMessages(t, r.client, qname)
|
||||||
t.Errorf("mismatch found in %q, (-want, +got)\n%s", base.RetryQueue, diff)
|
if diff := cmp.Diff(want, gotRetry, h.SortMsgOpt); diff != "" {
|
||||||
|
t.Errorf("mismatch found in %q, (-want, +got)\n%s", base.RetryKey(qname), diff)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1144,57 +1184,75 @@ func TestEnqueueScheduledTask(t *testing.T) {
|
|||||||
r := setup(t)
|
r := setup(t)
|
||||||
t1 := h.NewTaskMessage("send_email", nil)
|
t1 := h.NewTaskMessage("send_email", nil)
|
||||||
t2 := h.NewTaskMessage("gen_thumbnail", nil)
|
t2 := h.NewTaskMessage("gen_thumbnail", nil)
|
||||||
t3 := h.NewTaskMessage("send_notification", nil)
|
t3 := h.NewTaskMessageWithQueue("send_notification", nil, "notifications")
|
||||||
t3.Queue = "notifications"
|
|
||||||
s1 := time.Now().Add(-5 * time.Minute).Unix()
|
s1 := time.Now().Add(-5 * time.Minute).Unix()
|
||||||
s2 := time.Now().Add(-time.Hour).Unix()
|
s2 := time.Now().Add(-time.Hour).Unix()
|
||||||
|
|
||||||
tests := []struct {
|
tests := []struct {
|
||||||
scheduled []base.Z
|
scheduled map[string][]base.Z
|
||||||
|
qname string
|
||||||
score int64
|
score int64
|
||||||
id uuid.UUID
|
id uuid.UUID
|
||||||
want error // expected return value from calling EnqueueScheduledTask
|
want error // expected return value from calling EnqueueScheduledTask
|
||||||
wantScheduled []*base.TaskMessage
|
wantScheduled map[string][]*base.TaskMessage
|
||||||
wantEnqueued map[string][]*base.TaskMessage
|
wantEnqueued map[string][]*base.TaskMessage
|
||||||
}{
|
}{
|
||||||
{
|
{
|
||||||
scheduled: []base.Z{
|
scheduled: map[string][]base.Z{
|
||||||
|
"default": {
|
||||||
{Message: t1, Score: s1},
|
{Message: t1, Score: s1},
|
||||||
{Message: t2, Score: s2},
|
{Message: t2, Score: s2},
|
||||||
},
|
},
|
||||||
|
},
|
||||||
|
qname: "default",
|
||||||
score: s2,
|
score: s2,
|
||||||
id: t2.ID,
|
id: t2.ID,
|
||||||
want: nil,
|
want: nil,
|
||||||
wantScheduled: []*base.TaskMessage{t1},
|
wantScheduled: map[string][]*base.TaskMessage{
|
||||||
|
"default": {t1},
|
||||||
|
},
|
||||||
wantEnqueued: map[string][]*base.TaskMessage{
|
wantEnqueued: map[string][]*base.TaskMessage{
|
||||||
base.DefaultQueueName: {t2},
|
"default": {t2},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
scheduled: []base.Z{
|
scheduled: map[string][]base.Z{
|
||||||
|
"default": {
|
||||||
{Message: t1, Score: s1},
|
{Message: t1, Score: s1},
|
||||||
{Message: t2, Score: s2},
|
{Message: t2, Score: s2},
|
||||||
},
|
},
|
||||||
|
},
|
||||||
|
qname: "default",
|
||||||
score: 123,
|
score: 123,
|
||||||
id: t2.ID,
|
id: t2.ID,
|
||||||
want: ErrTaskNotFound,
|
want: ErrTaskNotFound,
|
||||||
wantScheduled: []*base.TaskMessage{t1, t2},
|
wantScheduled: map[string][]*base.TaskMessage{
|
||||||
|
"default": {t1, t2},
|
||||||
|
},
|
||||||
wantEnqueued: map[string][]*base.TaskMessage{
|
wantEnqueued: map[string][]*base.TaskMessage{
|
||||||
base.DefaultQueueName: {},
|
"default": {},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
scheduled: []base.Z{
|
scheduled: map[string][]base.Z{
|
||||||
|
"default": {
|
||||||
{Message: t1, Score: s1},
|
{Message: t1, Score: s1},
|
||||||
{Message: t2, Score: s2},
|
{Message: t2, Score: s2},
|
||||||
|
},
|
||||||
|
"notifications": {
|
||||||
{Message: t3, Score: s1},
|
{Message: t3, Score: s1},
|
||||||
},
|
},
|
||||||
|
},
|
||||||
|
qname: "notifications",
|
||||||
score: s1,
|
score: s1,
|
||||||
id: t3.ID,
|
id: t3.ID,
|
||||||
want: nil,
|
want: nil,
|
||||||
wantScheduled: []*base.TaskMessage{t1, t2},
|
wantScheduled: map[string][]*base.TaskMessage{
|
||||||
|
"default": {t1, t2},
|
||||||
|
"notifications": {},
|
||||||
|
},
|
||||||
wantEnqueued: map[string][]*base.TaskMessage{
|
wantEnqueued: map[string][]*base.TaskMessage{
|
||||||
base.DefaultQueueName: {},
|
"default": {},
|
||||||
"notifications": {t3},
|
"notifications": {t3},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
@ -1202,11 +1260,11 @@ func TestEnqueueScheduledTask(t *testing.T) {
|
|||||||
|
|
||||||
for _, tc := range tests {
|
for _, tc := range tests {
|
||||||
h.FlushDB(t, r.client) // clean up db before each test case
|
h.FlushDB(t, r.client) // clean up db before each test case
|
||||||
h.SeedScheduledQueue(t, r.client, tc.scheduled)
|
h.SeedAllScheduledQueues(t, r.client, tc.scheduled)
|
||||||
|
|
||||||
got := r.EnqueueScheduledTask(tc.id, tc.score)
|
got := r.EnqueueScheduledTask(tc.qname, tc.id, tc.score)
|
||||||
if got != tc.want {
|
if got != tc.want {
|
||||||
t.Errorf("r.EnqueueRetryTask(%s, %d) = %v, want %v", tc.id, tc.score, got, tc.want)
|
t.Errorf("r.EnqueueRetryTask(%q, %s, %d) = %v, want %v", tc.qname, tc.id, tc.score, got, tc.want)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1217,9 +1275,11 @@ func TestEnqueueScheduledTask(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
gotScheduled := h.GetScheduledMessages(t, r.client)
|
for qname, want := range tc.wantScheduled {
|
||||||
if diff := cmp.Diff(tc.wantScheduled, gotScheduled, h.SortMsgOpt); diff != "" {
|
gotScheduled := h.GetScheduledMessages(t, r.client, qname)
|
||||||
t.Errorf("mismatch found in %q, (-want, +got)\n%s", base.ScheduledQueue, diff)
|
if diff := cmp.Diff(want, gotScheduled, h.SortMsgOpt); diff != "" {
|
||||||
|
t.Errorf("mismatch found in %q, (-want, +got)\n%s", base.ScheduledKey(qname), diff)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1229,69 +1289,89 @@ func TestEnqueueAllScheduledTasks(t *testing.T) {
|
|||||||
t1 := h.NewTaskMessage("send_email", nil)
|
t1 := h.NewTaskMessage("send_email", nil)
|
||||||
t2 := h.NewTaskMessage("gen_thumbnail", nil)
|
t2 := h.NewTaskMessage("gen_thumbnail", nil)
|
||||||
t3 := h.NewTaskMessage("reindex", nil)
|
t3 := h.NewTaskMessage("reindex", nil)
|
||||||
t4 := h.NewTaskMessage("important_notification", nil)
|
t4 := h.NewTaskMessageWithQueue("important_notification", nil, "custom")
|
||||||
t4.Queue = "critical"
|
t5 := h.NewTaskMessageWithQueue("minor_notification", nil, "custom")
|
||||||
t5 := h.NewTaskMessage("minor_notification", nil)
|
|
||||||
t5.Queue = "low"
|
|
||||||
|
|
||||||
tests := []struct {
|
tests := []struct {
|
||||||
desc string
|
desc string
|
||||||
scheduled []base.Z
|
scheduled map[string][]base.Z
|
||||||
|
qname string
|
||||||
want int64
|
want int64
|
||||||
wantEnqueued map[string][]*base.TaskMessage
|
wantEnqueued map[string][]*base.TaskMessage
|
||||||
|
wantScheduled map[string][]*base.TaskMessage
|
||||||
}{
|
}{
|
||||||
{
|
{
|
||||||
desc: "with tasks in scheduled queue",
|
desc: "with tasks in scheduled queue",
|
||||||
scheduled: []base.Z{
|
scheduled: map[string][]base.Z{
|
||||||
|
"default": {
|
||||||
{Message: t1, Score: time.Now().Add(time.Hour).Unix()},
|
{Message: t1, Score: time.Now().Add(time.Hour).Unix()},
|
||||||
{Message: t2, Score: time.Now().Add(time.Hour).Unix()},
|
{Message: t2, Score: time.Now().Add(time.Hour).Unix()},
|
||||||
{Message: t3, Score: time.Now().Add(time.Hour).Unix()},
|
{Message: t3, Score: time.Now().Add(time.Hour).Unix()},
|
||||||
},
|
},
|
||||||
|
},
|
||||||
|
qname: "default",
|
||||||
want: 3,
|
want: 3,
|
||||||
wantEnqueued: map[string][]*base.TaskMessage{
|
wantEnqueued: map[string][]*base.TaskMessage{
|
||||||
base.DefaultQueueName: {t1, t2, t3},
|
"default": {t1, t2, t3},
|
||||||
|
},
|
||||||
|
wantScheduled: map[string][]*base.TaskMessage{
|
||||||
|
"default": {},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
desc: "with empty scheduled queue",
|
desc: "with empty scheduled queue",
|
||||||
scheduled: []base.Z{},
|
scheduled: map[string][]base.Z{
|
||||||
|
"default": {},
|
||||||
|
},
|
||||||
|
qname: "default",
|
||||||
want: 0,
|
want: 0,
|
||||||
wantEnqueued: map[string][]*base.TaskMessage{
|
wantEnqueued: map[string][]*base.TaskMessage{
|
||||||
base.DefaultQueueName: {},
|
"default": {},
|
||||||
|
},
|
||||||
|
wantScheduled: map[string][]*base.TaskMessage{
|
||||||
|
"default": {},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
desc: "with custom queues",
|
desc: "with custom queues",
|
||||||
scheduled: []base.Z{
|
scheduled: map[string][]base.Z{
|
||||||
|
"default": {
|
||||||
{Message: t1, Score: time.Now().Add(time.Hour).Unix()},
|
{Message: t1, Score: time.Now().Add(time.Hour).Unix()},
|
||||||
{Message: t2, Score: time.Now().Add(time.Hour).Unix()},
|
{Message: t2, Score: time.Now().Add(time.Hour).Unix()},
|
||||||
{Message: t3, Score: time.Now().Add(time.Hour).Unix()},
|
{Message: t3, Score: time.Now().Add(time.Hour).Unix()},
|
||||||
|
},
|
||||||
|
"custom": {
|
||||||
{Message: t4, Score: time.Now().Add(time.Hour).Unix()},
|
{Message: t4, Score: time.Now().Add(time.Hour).Unix()},
|
||||||
{Message: t5, Score: time.Now().Add(time.Hour).Unix()},
|
{Message: t5, Score: time.Now().Add(time.Hour).Unix()},
|
||||||
},
|
},
|
||||||
want: 5,
|
},
|
||||||
|
qname: "custom",
|
||||||
|
want: 2,
|
||||||
wantEnqueued: map[string][]*base.TaskMessage{
|
wantEnqueued: map[string][]*base.TaskMessage{
|
||||||
base.DefaultQueueName: {t1, t2, t3},
|
"default": {},
|
||||||
"critical": {t4},
|
"custom": {t4, t5},
|
||||||
"low": {t5},
|
},
|
||||||
|
wantScheduled: map[string][]*base.TaskMessage{
|
||||||
|
"default": {t1, t2, t3},
|
||||||
|
"custom": {},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, tc := range tests {
|
for _, tc := range tests {
|
||||||
h.FlushDB(t, r.client) // clean up db before each test case
|
h.FlushDB(t, r.client) // clean up db before each test case
|
||||||
h.SeedScheduledQueue(t, r.client, tc.scheduled)
|
h.SeedAllScheduledQueues(t, r.client, tc.scheduled)
|
||||||
|
|
||||||
got, err := r.EnqueueAllScheduledTasks()
|
got, err := r.EnqueueAllScheduledTasks(tc.qname)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Errorf("%s; r.EnqueueAllScheduledTasks = %v, %v; want %v, nil",
|
t.Errorf("%s; r.EnqueueAllScheduledTasks(%q) = %v, %v; want %v, nil",
|
||||||
tc.desc, got, err, tc.want)
|
tc.desc, tc.qname, got, err, tc.want)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
if got != tc.want {
|
if got != tc.want {
|
||||||
t.Errorf("%s; r.EnqueueAllScheduledTasks = %v, %v; want %v, nil",
|
t.Errorf("%s; r.EnqueueAllScheduledTasks(%q) = %v, %v; want %v, nil",
|
||||||
tc.desc, got, err, tc.want)
|
tc.desc, tc.qname, got, err, tc.want)
|
||||||
}
|
}
|
||||||
|
|
||||||
for qname, want := range tc.wantEnqueued {
|
for qname, want := range tc.wantEnqueued {
|
||||||
@ -1300,6 +1380,12 @@ func TestEnqueueAllScheduledTasks(t *testing.T) {
|
|||||||
t.Errorf("%s; mismatch found in %q; (-want, +got)\n%s", tc.desc, base.QueueKey(qname), diff)
|
t.Errorf("%s; mismatch found in %q; (-want, +got)\n%s", tc.desc, base.QueueKey(qname), diff)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
for qname, want := range tc.wantScheduled {
|
||||||
|
gotScheduled := h.GetScheduledMessages(t, r.client, qname)
|
||||||
|
if diff := cmp.Diff(want, gotScheduled, h.SortMsgOpt); diff != "" {
|
||||||
|
t.Errorf("%s; mismatch found in %q; (-want, +got)\n%s", tc.desc, base.ScheduledKey(qname), diff)
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1308,69 +1394,89 @@ func TestEnqueueAllRetryTasks(t *testing.T) {
|
|||||||
t1 := h.NewTaskMessage("send_email", nil)
|
t1 := h.NewTaskMessage("send_email", nil)
|
||||||
t2 := h.NewTaskMessage("gen_thumbnail", nil)
|
t2 := h.NewTaskMessage("gen_thumbnail", nil)
|
||||||
t3 := h.NewTaskMessage("reindex", nil)
|
t3 := h.NewTaskMessage("reindex", nil)
|
||||||
t4 := h.NewTaskMessage("important_notification", nil)
|
t4 := h.NewTaskMessageWithQueue("important_notification", nil, "custom")
|
||||||
t4.Queue = "critical"
|
t5 := h.NewTaskMessageWithQueue("minor_notification", nil, "custom")
|
||||||
t5 := h.NewTaskMessage("minor_notification", nil)
|
|
||||||
t5.Queue = "low"
|
|
||||||
|
|
||||||
tests := []struct {
|
tests := []struct {
|
||||||
desc string
|
desc string
|
||||||
retry []base.Z
|
retry map[string][]base.Z
|
||||||
|
qname string
|
||||||
want int64
|
want int64
|
||||||
wantEnqueued map[string][]*base.TaskMessage
|
wantEnqueued map[string][]*base.TaskMessage
|
||||||
|
wantRetry map[string][]*base.TaskMessage
|
||||||
}{
|
}{
|
||||||
{
|
{
|
||||||
desc: "with tasks in retry queue",
|
desc: "with tasks in retry queue",
|
||||||
retry: []base.Z{
|
retry: map[string][]base.Z{
|
||||||
|
"default": {
|
||||||
{Message: t1, Score: time.Now().Add(time.Hour).Unix()},
|
{Message: t1, Score: time.Now().Add(time.Hour).Unix()},
|
||||||
{Message: t2, Score: time.Now().Add(time.Hour).Unix()},
|
{Message: t2, Score: time.Now().Add(time.Hour).Unix()},
|
||||||
{Message: t3, Score: time.Now().Add(time.Hour).Unix()},
|
{Message: t3, Score: time.Now().Add(time.Hour).Unix()},
|
||||||
},
|
},
|
||||||
|
},
|
||||||
|
qname: "default",
|
||||||
want: 3,
|
want: 3,
|
||||||
wantEnqueued: map[string][]*base.TaskMessage{
|
wantEnqueued: map[string][]*base.TaskMessage{
|
||||||
base.DefaultQueueName: {t1, t2, t3},
|
"default": {t1, t2, t3},
|
||||||
|
},
|
||||||
|
wantRetry: map[string][]*base.TaskMessage{
|
||||||
|
"default": {},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
desc: "with empty retry queue",
|
desc: "with empty retry queue",
|
||||||
retry: []base.Z{},
|
retry: map[string][]base.Z{
|
||||||
|
"default": {},
|
||||||
|
},
|
||||||
|
qname: "default",
|
||||||
want: 0,
|
want: 0,
|
||||||
wantEnqueued: map[string][]*base.TaskMessage{
|
wantEnqueued: map[string][]*base.TaskMessage{
|
||||||
base.DefaultQueueName: {},
|
"default": {},
|
||||||
|
},
|
||||||
|
wantRetry: map[string][]*base.TaskMessage{
|
||||||
|
"default": {},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
desc: "with custom queues",
|
desc: "with custom queues",
|
||||||
retry: []base.Z{
|
retry: map[string][]base.Z{
|
||||||
|
"default": {
|
||||||
{Message: t1, Score: time.Now().Add(time.Hour).Unix()},
|
{Message: t1, Score: time.Now().Add(time.Hour).Unix()},
|
||||||
{Message: t2, Score: time.Now().Add(time.Hour).Unix()},
|
{Message: t2, Score: time.Now().Add(time.Hour).Unix()},
|
||||||
{Message: t3, Score: time.Now().Add(time.Hour).Unix()},
|
{Message: t3, Score: time.Now().Add(time.Hour).Unix()},
|
||||||
|
},
|
||||||
|
"custom": {
|
||||||
{Message: t4, Score: time.Now().Add(time.Hour).Unix()},
|
{Message: t4, Score: time.Now().Add(time.Hour).Unix()},
|
||||||
{Message: t5, Score: time.Now().Add(time.Hour).Unix()},
|
{Message: t5, Score: time.Now().Add(time.Hour).Unix()},
|
||||||
},
|
},
|
||||||
want: 5,
|
},
|
||||||
|
qname: "custom",
|
||||||
|
want: 2,
|
||||||
wantEnqueued: map[string][]*base.TaskMessage{
|
wantEnqueued: map[string][]*base.TaskMessage{
|
||||||
base.DefaultQueueName: {t1, t2, t3},
|
"default": {},
|
||||||
"critical": {t4},
|
"custom": {t4, t5},
|
||||||
"low": {t5},
|
},
|
||||||
|
wantRetry: map[string][]*base.TaskMessage{
|
||||||
|
"default": {t1, t2, t3},
|
||||||
|
"custom": {},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, tc := range tests {
|
for _, tc := range tests {
|
||||||
h.FlushDB(t, r.client) // clean up db before each test case
|
h.FlushDB(t, r.client) // clean up db before each test case
|
||||||
h.SeedRetryQueue(t, r.client, tc.retry)
|
h.SeedAllRetryQueues(t, r.client, tc.retry)
|
||||||
|
|
||||||
got, err := r.EnqueueAllRetryTasks()
|
got, err := r.EnqueueAllRetryTasks(tc.qname)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Errorf("%s; r.EnqueueAllRetryTasks = %v, %v; want %v, nil",
|
t.Errorf("%s; r.EnqueueAllRetryTasks(%q) = %v, %v; want %v, nil",
|
||||||
tc.desc, got, err, tc.want)
|
tc.desc, tc.qname, got, err, tc.want)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
if got != tc.want {
|
if got != tc.want {
|
||||||
t.Errorf("%s; r.EnqueueAllRetryTasks = %v, %v; want %v, nil",
|
t.Errorf("%s; r.EnqueueAllRetryTasks(%q) = %v, %v; want %v, nil",
|
||||||
tc.desc, got, err, tc.want)
|
tc.desc, tc.qname, got, err, tc.want)
|
||||||
}
|
}
|
||||||
|
|
||||||
for qname, want := range tc.wantEnqueued {
|
for qname, want := range tc.wantEnqueued {
|
||||||
@ -1379,6 +1485,12 @@ func TestEnqueueAllRetryTasks(t *testing.T) {
|
|||||||
t.Errorf("%s; mismatch found in %q; (-want, +got)\n%s", tc.desc, base.QueueKey(qname), diff)
|
t.Errorf("%s; mismatch found in %q; (-want, +got)\n%s", tc.desc, base.QueueKey(qname), diff)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
for qname, want := range tc.wantRetry {
|
||||||
|
gotRetry := h.GetRetryMessages(t, r.client, qname)
|
||||||
|
if diff := cmp.Diff(want, gotRetry, h.SortMsgOpt); diff != "" {
|
||||||
|
t.Errorf("%s; mismatch found in %q; (-want, +got)\n%s", tc.desc, base.RetryKey(qname), diff)
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1387,69 +1499,89 @@ func TestEnqueueAllDeadTasks(t *testing.T) {
|
|||||||
t1 := h.NewTaskMessage("send_email", nil)
|
t1 := h.NewTaskMessage("send_email", nil)
|
||||||
t2 := h.NewTaskMessage("gen_thumbnail", nil)
|
t2 := h.NewTaskMessage("gen_thumbnail", nil)
|
||||||
t3 := h.NewTaskMessage("reindex", nil)
|
t3 := h.NewTaskMessage("reindex", nil)
|
||||||
t4 := h.NewTaskMessage("important_notification", nil)
|
t4 := h.NewTaskMessageWithQueue("important_notification", nil, "custom")
|
||||||
t4.Queue = "critical"
|
t5 := h.NewTaskMessageWithQueue("minor_notification", nil, "custom")
|
||||||
t5 := h.NewTaskMessage("minor_notification", nil)
|
|
||||||
t5.Queue = "low"
|
|
||||||
|
|
||||||
tests := []struct {
|
tests := []struct {
|
||||||
desc string
|
desc string
|
||||||
dead []base.Z
|
dead map[string][]base.Z
|
||||||
|
qname string
|
||||||
want int64
|
want int64
|
||||||
wantEnqueued map[string][]*base.TaskMessage
|
wantEnqueued map[string][]*base.TaskMessage
|
||||||
|
wantDead map[string][]*base.TaskMessage
|
||||||
}{
|
}{
|
||||||
{
|
{
|
||||||
desc: "with tasks in dead queue",
|
desc: "with tasks in dead queue",
|
||||||
dead: []base.Z{
|
dead: map[string][]base.Z{
|
||||||
|
"default": {
|
||||||
{Message: t1, Score: time.Now().Add(-time.Minute).Unix()},
|
{Message: t1, Score: time.Now().Add(-time.Minute).Unix()},
|
||||||
{Message: t2, Score: time.Now().Add(-time.Minute).Unix()},
|
{Message: t2, Score: time.Now().Add(-time.Minute).Unix()},
|
||||||
{Message: t3, Score: time.Now().Add(-time.Minute).Unix()},
|
{Message: t3, Score: time.Now().Add(-time.Minute).Unix()},
|
||||||
},
|
},
|
||||||
|
},
|
||||||
|
qname: "default",
|
||||||
want: 3,
|
want: 3,
|
||||||
wantEnqueued: map[string][]*base.TaskMessage{
|
wantEnqueued: map[string][]*base.TaskMessage{
|
||||||
base.DefaultQueueName: {t1, t2, t3},
|
"default": {t1, t2, t3},
|
||||||
|
},
|
||||||
|
wantDead: map[string][]*base.TaskMessage{
|
||||||
|
"default": {},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
desc: "with empty dead queue",
|
desc: "with empty dead queue",
|
||||||
dead: []base.Z{},
|
dead: map[string][]base.Z{
|
||||||
|
"default": {},
|
||||||
|
},
|
||||||
|
qname: "default",
|
||||||
want: 0,
|
want: 0,
|
||||||
wantEnqueued: map[string][]*base.TaskMessage{
|
wantEnqueued: map[string][]*base.TaskMessage{
|
||||||
base.DefaultQueueName: {},
|
"default": {},
|
||||||
|
},
|
||||||
|
wantDead: map[string][]*base.TaskMessage{
|
||||||
|
"default": {},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
desc: "with custom queues",
|
desc: "with custom queues",
|
||||||
dead: []base.Z{
|
dead: map[string][]base.Z{
|
||||||
|
"default": {
|
||||||
{Message: t1, Score: time.Now().Add(-time.Minute).Unix()},
|
{Message: t1, Score: time.Now().Add(-time.Minute).Unix()},
|
||||||
{Message: t2, Score: time.Now().Add(-time.Minute).Unix()},
|
{Message: t2, Score: time.Now().Add(-time.Minute).Unix()},
|
||||||
{Message: t3, Score: time.Now().Add(-time.Minute).Unix()},
|
{Message: t3, Score: time.Now().Add(-time.Minute).Unix()},
|
||||||
|
},
|
||||||
|
"custom": {
|
||||||
{Message: t4, Score: time.Now().Add(-time.Minute).Unix()},
|
{Message: t4, Score: time.Now().Add(-time.Minute).Unix()},
|
||||||
{Message: t5, Score: time.Now().Add(-time.Minute).Unix()},
|
{Message: t5, Score: time.Now().Add(-time.Minute).Unix()},
|
||||||
},
|
},
|
||||||
want: 5,
|
},
|
||||||
|
qname: "custom",
|
||||||
|
want: 2,
|
||||||
wantEnqueued: map[string][]*base.TaskMessage{
|
wantEnqueued: map[string][]*base.TaskMessage{
|
||||||
base.DefaultQueueName: {t1, t2, t3},
|
"default": {},
|
||||||
"critical": {t4},
|
"custom": {t4, t5},
|
||||||
"low": {t5},
|
},
|
||||||
|
wantDead: map[string][]*base.TaskMessage{
|
||||||
|
"default": {t1, t2, t3},
|
||||||
|
"custom": {},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, tc := range tests {
|
for _, tc := range tests {
|
||||||
h.FlushDB(t, r.client) // clean up db before each test case
|
h.FlushDB(t, r.client) // clean up db before each test case
|
||||||
h.SeedDeadQueue(t, r.client, tc.dead)
|
h.SeedAllDeadQueues(t, r.client, tc.dead)
|
||||||
|
|
||||||
got, err := r.EnqueueAllDeadTasks()
|
got, err := r.EnqueueAllDeadTasks(tc.qname)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Errorf("%s; r.EnqueueAllDeadTasks = %v, %v; want %v, nil",
|
t.Errorf("%s; r.EnqueueAllDeadTasks(%q) = %v, %v; want %v, nil",
|
||||||
tc.desc, got, err, tc.want)
|
tc.desc, tc.qname, got, err, tc.want)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
if got != tc.want {
|
if got != tc.want {
|
||||||
t.Errorf("%s; r.EnqueueAllDeadTasks = %v, %v; want %v, nil",
|
t.Errorf("%s; r.EnqueueAllDeadTasks(%q) = %v, %v; want %v, nil",
|
||||||
tc.desc, got, err, tc.want)
|
tc.desc, tc.qname, got, err, tc.want)
|
||||||
}
|
}
|
||||||
|
|
||||||
for qname, want := range tc.wantEnqueued {
|
for qname, want := range tc.wantEnqueued {
|
||||||
@ -1458,9 +1590,16 @@ func TestEnqueueAllDeadTasks(t *testing.T) {
|
|||||||
t.Errorf("%s; mismatch found in %q; (-want, +got)\n%s", tc.desc, base.QueueKey(qname), diff)
|
t.Errorf("%s; mismatch found in %q; (-want, +got)\n%s", tc.desc, base.QueueKey(qname), diff)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
for qname, want := range tc.wantDead {
|
||||||
|
gotDead := h.GetDeadMessages(t, r.client, qname)
|
||||||
|
if diff := cmp.Diff(want, gotDead, h.SortMsgOpt); diff != "" {
|
||||||
|
t.Errorf("%s; mismatch found in %q; (-want, +got)\n%s", tc.desc, base.DeadKey(qname), diff)
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
func TestKillRetryTask(t *testing.T) {
|
func TestKillRetryTask(t *testing.T) {
|
||||||
r := setup(t)
|
r := setup(t)
|
||||||
m1 := h.NewTaskMessage("send_email", nil)
|
m1 := h.NewTaskMessage("send_email", nil)
|
||||||
@ -2309,6 +2448,7 @@ func TestListWorkers(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
*/
|
||||||
|
|
||||||
func TestPause(t *testing.T) {
|
func TestPause(t *testing.T) {
|
||||||
r := setup(t)
|
r := setup(t)
|
||||||
@ -2327,7 +2467,7 @@ func TestPause(t *testing.T) {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
t.Errorf("Pause(%q) returned error: %v", tc.qname, err)
|
t.Errorf("Pause(%q) returned error: %v", tc.qname, err)
|
||||||
}
|
}
|
||||||
key := base.PauseKey(tc.qname)
|
key := base.PausedKey(tc.qname)
|
||||||
if r.client.Exists(key).Val() == 0 {
|
if r.client.Exists(key).Val() == 0 {
|
||||||
t.Errorf("key %q does not exist", key)
|
t.Errorf("key %q does not exist", key)
|
||||||
}
|
}
|
||||||
@ -2382,8 +2522,8 @@ func TestUnpause(t *testing.T) {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
t.Errorf("Unpause(%q) returned error: %v", tc.qname, err)
|
t.Errorf("Unpause(%q) returned error: %v", tc.qname, err)
|
||||||
}
|
}
|
||||||
key := base.PauseKey(tc.qname)
|
key := base.PausedKey(tc.qname)
|
||||||
if r.client.Exists(key) == 1 {
|
if r.client.Exists(key).Val() == 1 {
|
||||||
t.Errorf("key %q exists", key)
|
t.Errorf("key %q exists", key)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -309,8 +309,22 @@ func TestDequeue(t *testing.T) {
|
|||||||
|
|
||||||
func TestDequeueIgnoresPausedQueues(t *testing.T) {
|
func TestDequeueIgnoresPausedQueues(t *testing.T) {
|
||||||
r := setup(t)
|
r := setup(t)
|
||||||
t1 := h.NewTaskMessageWithQueue("send_email", map[string]interface{}{"subject": "hello!"}, "default")
|
t1 := &base.TaskMessage{
|
||||||
t2 := h.NewTaskMessageWithQueue("export_csv", nil, "critical")
|
ID: uuid.New(),
|
||||||
|
Type: "send_email",
|
||||||
|
Payload: map[string]interface{}{"subject": "hello!"},
|
||||||
|
Queue: "default",
|
||||||
|
Timeout: 1800,
|
||||||
|
Deadline: 0,
|
||||||
|
}
|
||||||
|
t2 := &base.TaskMessage{
|
||||||
|
ID: uuid.New(),
|
||||||
|
Type: "export_csv",
|
||||||
|
Payload: nil,
|
||||||
|
Queue: "critical",
|
||||||
|
Timeout: 1800,
|
||||||
|
Deadline: 0,
|
||||||
|
}
|
||||||
|
|
||||||
tests := []struct {
|
tests := []struct {
|
||||||
paused []string // list of paused queues
|
paused []string // list of paused queues
|
||||||
@ -381,7 +395,7 @@ func TestDequeueIgnoresPausedQueues(t *testing.T) {
|
|||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
h.SeedAllEnqueuedQueues(t, r.client, msgs, queue, tc.enqueued)
|
h.SeedAllEnqueuedQueues(t, r.client, tc.enqueued)
|
||||||
|
|
||||||
got, _, err := r.Dequeue(tc.args...)
|
got, _, err := r.Dequeue(tc.args...)
|
||||||
if !cmp.Equal(got, tc.wantMsg) || err != tc.err {
|
if !cmp.Equal(got, tc.wantMsg) || err != tc.err {
|
||||||
|
Loading…
Reference in New Issue
Block a user