mirror of
https://github.com/hibiken/asynq.git
synced 2024-11-10 11:31:58 +08:00
7af3981929
Changes: - Task messages are stored under "asynq:{<qname>}:t:<task_id>" key in redis, value is a HASH type and message are stored under "msg" key in the hash. The hash also stores "deadline", "timeout". - Redis LIST and ZSET stores task message IDs - Task messages are serialized using protocol buffer
267 lines
6.7 KiB
Go
267 lines
6.7 KiB
Go
// Copyright 2020 Kentaro Hibino. All rights reserved.
|
|
// Use of this source code is governed by a MIT license
|
|
// that can be found in the LICENSE file.
|
|
|
|
package rdb
|
|
|
|
import (
|
|
"fmt"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/hibiken/asynq/internal/asynqtest"
|
|
"github.com/hibiken/asynq/internal/base"
|
|
)
|
|
|
|
func BenchmarkEnqueue(b *testing.B) {
|
|
r := setup(b)
|
|
msg := asynqtest.NewTaskMessage("task1", nil)
|
|
b.ResetTimer()
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
b.StopTimer()
|
|
asynqtest.FlushDB(b, r.client)
|
|
b.StartTimer()
|
|
|
|
if err := r.Enqueue(msg); err != nil {
|
|
b.Fatalf("Enqueue failed: %v", err)
|
|
}
|
|
}
|
|
}
|
|
|
|
func BenchmarkEnqueueUnique(b *testing.B) {
|
|
r := setup(b)
|
|
msg := &base.TaskMessage{
|
|
Type: "task1",
|
|
Payload: nil,
|
|
Queue: base.DefaultQueueName,
|
|
UniqueKey: base.UniqueKey("default", "task1", nil),
|
|
}
|
|
uniqueTTL := 5 * time.Minute
|
|
b.ResetTimer()
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
b.StopTimer()
|
|
asynqtest.FlushDB(b, r.client)
|
|
b.StartTimer()
|
|
|
|
if err := r.EnqueueUnique(msg, uniqueTTL); err != nil {
|
|
b.Fatalf("EnqueueUnique failed: %v", err)
|
|
}
|
|
}
|
|
}
|
|
|
|
func BenchmarkSchedule(b *testing.B) {
|
|
r := setup(b)
|
|
msg := asynqtest.NewTaskMessage("task1", nil)
|
|
processAt := time.Now().Add(3 * time.Minute)
|
|
b.ResetTimer()
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
b.StopTimer()
|
|
asynqtest.FlushDB(b, r.client)
|
|
b.StartTimer()
|
|
|
|
if err := r.Schedule(msg, processAt); err != nil {
|
|
b.Fatalf("Schedule failed: %v", err)
|
|
}
|
|
}
|
|
}
|
|
|
|
func BenchmarkScheduleUnique(b *testing.B) {
|
|
r := setup(b)
|
|
msg := &base.TaskMessage{
|
|
Type: "task1",
|
|
Payload: nil,
|
|
Queue: base.DefaultQueueName,
|
|
UniqueKey: base.UniqueKey("default", "task1", nil),
|
|
}
|
|
processAt := time.Now().Add(3 * time.Minute)
|
|
uniqueTTL := 5 * time.Minute
|
|
b.ResetTimer()
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
b.StopTimer()
|
|
asynqtest.FlushDB(b, r.client)
|
|
b.StartTimer()
|
|
|
|
if err := r.ScheduleUnique(msg, processAt, uniqueTTL); err != nil {
|
|
b.Fatalf("EnqueueUnique failed: %v", err)
|
|
}
|
|
}
|
|
}
|
|
|
|
func BenchmarkDequeueSingleQueue(b *testing.B) {
|
|
r := setup(b)
|
|
b.ResetTimer()
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
b.StopTimer()
|
|
asynqtest.FlushDB(b, r.client)
|
|
for i := 0; i < 10; i++ {
|
|
m := asynqtest.NewTaskMessageWithQueue(
|
|
fmt.Sprintf("task%d", i), nil, base.DefaultQueueName)
|
|
if err := r.Enqueue(m); err != nil {
|
|
b.Fatalf("Enqueue failed: %v", err)
|
|
}
|
|
}
|
|
b.StartTimer()
|
|
|
|
if _, _, err := r.Dequeue(base.DefaultQueueName); err != nil {
|
|
b.Fatalf("Dequeue failed: %v", err)
|
|
}
|
|
}
|
|
}
|
|
|
|
func BenchmarkDequeueMultipleQueues(b *testing.B) {
|
|
qnames := []string{"critical", "default", "low"}
|
|
r := setup(b)
|
|
b.ResetTimer()
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
b.StopTimer()
|
|
asynqtest.FlushDB(b, r.client)
|
|
for i := 0; i < 10; i++ {
|
|
for _, qname := range qnames {
|
|
m := asynqtest.NewTaskMessageWithQueue(
|
|
fmt.Sprintf("%s_task%d", qname, i), nil, qname)
|
|
if err := r.Enqueue(m); err != nil {
|
|
b.Fatalf("Enqueue failed: %v", err)
|
|
}
|
|
}
|
|
}
|
|
b.StartTimer()
|
|
|
|
if _, _, err := r.Dequeue(qnames...); err != nil {
|
|
b.Fatalf("Dequeue failed: %v", err)
|
|
}
|
|
}
|
|
}
|
|
|
|
func BenchmarkDone(b *testing.B) {
|
|
r := setup(b)
|
|
m1 := asynqtest.NewTaskMessage("task1", nil)
|
|
m2 := asynqtest.NewTaskMessage("task2", nil)
|
|
m3 := asynqtest.NewTaskMessage("task3", nil)
|
|
msgs := []*base.TaskMessage{m1, m2, m3}
|
|
zs := []base.Z{
|
|
{Message: m1, Score: time.Now().Add(10 * time.Second).Unix()},
|
|
{Message: m2, Score: time.Now().Add(20 * time.Second).Unix()},
|
|
{Message: m3, Score: time.Now().Add(30 * time.Second).Unix()},
|
|
}
|
|
b.ResetTimer()
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
b.StopTimer()
|
|
asynqtest.FlushDB(b, r.client)
|
|
asynqtest.SeedActiveQueue(b, r.client, msgs, base.DefaultQueueName)
|
|
asynqtest.SeedDeadlines(b, r.client, zs, base.DefaultQueueName)
|
|
b.StartTimer()
|
|
|
|
if err := r.Done(msgs[0]); err != nil {
|
|
b.Fatalf("Done failed: %v", err)
|
|
}
|
|
}
|
|
}
|
|
|
|
func BenchmarkRetry(b *testing.B) {
|
|
r := setup(b)
|
|
m1 := asynqtest.NewTaskMessage("task1", nil)
|
|
m2 := asynqtest.NewTaskMessage("task2", nil)
|
|
m3 := asynqtest.NewTaskMessage("task3", nil)
|
|
msgs := []*base.TaskMessage{m1, m2, m3}
|
|
zs := []base.Z{
|
|
{Message: m1, Score: time.Now().Add(10 * time.Second).Unix()},
|
|
{Message: m2, Score: time.Now().Add(20 * time.Second).Unix()},
|
|
{Message: m3, Score: time.Now().Add(30 * time.Second).Unix()},
|
|
}
|
|
b.ResetTimer()
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
b.StopTimer()
|
|
asynqtest.FlushDB(b, r.client)
|
|
asynqtest.SeedActiveQueue(b, r.client, msgs, base.DefaultQueueName)
|
|
asynqtest.SeedDeadlines(b, r.client, zs, base.DefaultQueueName)
|
|
b.StartTimer()
|
|
|
|
if err := r.Retry(msgs[0], time.Now().Add(1*time.Minute), "error"); err != nil {
|
|
b.Fatalf("Retry failed: %v", err)
|
|
}
|
|
}
|
|
}
|
|
|
|
func BenchmarkArchive(b *testing.B) {
|
|
r := setup(b)
|
|
m1 := asynqtest.NewTaskMessage("task1", nil)
|
|
m2 := asynqtest.NewTaskMessage("task2", nil)
|
|
m3 := asynqtest.NewTaskMessage("task3", nil)
|
|
msgs := []*base.TaskMessage{m1, m2, m3}
|
|
zs := []base.Z{
|
|
{Message: m1, Score: time.Now().Add(10 * time.Second).Unix()},
|
|
{Message: m2, Score: time.Now().Add(20 * time.Second).Unix()},
|
|
{Message: m3, Score: time.Now().Add(30 * time.Second).Unix()},
|
|
}
|
|
b.ResetTimer()
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
b.StopTimer()
|
|
asynqtest.FlushDB(b, r.client)
|
|
asynqtest.SeedActiveQueue(b, r.client, msgs, base.DefaultQueueName)
|
|
asynqtest.SeedDeadlines(b, r.client, zs, base.DefaultQueueName)
|
|
b.StartTimer()
|
|
|
|
if err := r.Archive(msgs[0], "error"); err != nil {
|
|
b.Fatalf("Archive failed: %v", err)
|
|
}
|
|
}
|
|
}
|
|
|
|
func BenchmarkRequeue(b *testing.B) {
|
|
r := setup(b)
|
|
m1 := asynqtest.NewTaskMessage("task1", nil)
|
|
m2 := asynqtest.NewTaskMessage("task2", nil)
|
|
m3 := asynqtest.NewTaskMessage("task3", nil)
|
|
msgs := []*base.TaskMessage{m1, m2, m3}
|
|
zs := []base.Z{
|
|
{Message: m1, Score: time.Now().Add(10 * time.Second).Unix()},
|
|
{Message: m2, Score: time.Now().Add(20 * time.Second).Unix()},
|
|
{Message: m3, Score: time.Now().Add(30 * time.Second).Unix()},
|
|
}
|
|
b.ResetTimer()
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
b.StopTimer()
|
|
asynqtest.FlushDB(b, r.client)
|
|
asynqtest.SeedActiveQueue(b, r.client, msgs, base.DefaultQueueName)
|
|
asynqtest.SeedDeadlines(b, r.client, zs, base.DefaultQueueName)
|
|
b.StartTimer()
|
|
|
|
if err := r.Requeue(msgs[0]); err != nil {
|
|
b.Fatalf("Requeue failed: %v", err)
|
|
}
|
|
}
|
|
}
|
|
|
|
func BenchmarkCheckAndEnqueue(b *testing.B) {
|
|
r := setup(b)
|
|
now := time.Now()
|
|
var zs []base.Z
|
|
for i := -100; i < 100; i++ {
|
|
msg := asynqtest.NewTaskMessage(fmt.Sprintf("task%d", i), nil)
|
|
score := now.Add(time.Duration(i) * time.Second).Unix()
|
|
zs = append(zs, base.Z{Message: msg, Score: score})
|
|
}
|
|
b.ResetTimer()
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
b.StopTimer()
|
|
asynqtest.FlushDB(b, r.client)
|
|
asynqtest.SeedScheduledQueue(b, r.client, zs, base.DefaultQueueName)
|
|
b.StartTimer()
|
|
|
|
if err := r.ForwardIfReady(base.DefaultQueueName); err != nil {
|
|
b.Fatalf("ForwardIfReady failed: %v", err)
|
|
}
|
|
}
|
|
}
|