2
0
mirror of https://github.com/hibiken/asynq.git synced 2024-11-10 11:31:58 +08:00

Change internal representation of task data written to redis

This commit is contained in:
Ken Hibino 2019-11-17 13:25:01 -08:00
parent d2d0d1fde5
commit e75756937e

View File

@ -17,7 +17,6 @@ import (
"time" "time"
"github.com/go-redis/redis/v7" "github.com/go-redis/redis/v7"
"github.com/google/uuid"
) )
// Redis keys // Redis keys
@ -27,6 +26,9 @@ const (
scheduled = "asynq:scheduled" scheduled = "asynq:scheduled"
) )
// Max retry count by default
const defaultMaxRetry = 25
// Client is an interface for scheduling tasks. // Client is an interface for scheduling tasks.
type Client struct { type Client struct {
rdb *redis.Client rdb *redis.Client
@ -42,10 +44,25 @@ type Task struct {
Payload map[string]interface{} Payload map[string]interface{}
} }
type delayedTask struct { // taskMessage is an internal representation of a task with additional metadata fields.
ID string // This data gets written in redis.
type taskMessage struct {
// fields from type Task
Type string
Payload map[string]interface{}
//------- metadata fields ----------
// queue name this message should be enqueued to
Queue string Queue string
Task *Task
// remainig retry count
Retry int
// number of times we've retried so far
Retried int
// error message from the last failure
ErrorMsg string
} }
// RedisOpt specifies redis options. // RedisOpt specifies redis options.
@ -67,15 +84,16 @@ func (c *Client) Process(task *Task, executeAt time.Time) error {
// enqueue pushes a given task to the specified queue. // enqueue pushes a given task to the specified queue.
func (c *Client) enqueue(queue string, task *Task, executeAt time.Time) error { func (c *Client) enqueue(queue string, task *Task, executeAt time.Time) error {
msg := &taskMessage{
Type: task.Type,
Payload: task.Payload,
Queue: queue,
Retry: defaultMaxRetry,
}
if time.Now().After(executeAt) { if time.Now().After(executeAt) {
return push(c.rdb, queue, task) return push(c.rdb, msg)
} }
dt := &delayedTask{ bytes, err := json.Marshal(msg)
ID: uuid.New().String(),
Queue: queue,
Task: task,
}
bytes, err := json.Marshal(dt)
if err != nil { if err != nil {
return err return err
} }
@ -120,19 +138,25 @@ func (w *Workers) Run(handler TaskHandler) {
continue continue
} }
q, msg := res[0], res[1] q, data := res[0], res[1]
fmt.Printf("perform task %v from %s\n", msg, q) fmt.Printf("perform task %v from %s\n", data, q)
var task Task var msg taskMessage
err = json.Unmarshal([]byte(msg), &task) err = json.Unmarshal([]byte(data), &msg)
if err != nil { if err != nil {
log.Printf("[Servere Error] could not parse json encoded message %s: %v", msg, err) log.Printf("[Servere Error] could not parse json encoded message %s: %v", data, err)
continue continue
} }
w.poolTokens <- struct{}{} // acquire a token w.poolTokens <- struct{}{} // acquire a token
t := &Task{Type: msg.Type, Payload: msg.Payload}
go func(task *Task) { go func(task *Task) {
handler(task) err := handler(task)
<-w.poolTokens if err != nil {
}(&task) fmt.Println("RETRY!!!")
//timeout := 10 * time.Second // TODO(hibiken): Implement exponential backoff.
// TODO(hibiken): Enqueue the task to "retry" ZSET with some timeout
}
<-w.poolTokens // release the token
}(t)
} }
} }
@ -156,17 +180,17 @@ func (w *Workers) pollScheduledTasks() {
} }
for _, j := range jobs { for _, j := range jobs {
var job delayedTask var msg taskMessage
err = json.Unmarshal([]byte(j), &job) err = json.Unmarshal([]byte(j), &msg)
if err != nil { if err != nil {
fmt.Println("unmarshal failed") fmt.Println("unmarshal failed")
continue continue
} }
if w.rdb.ZRem(scheduled, j).Val() > 0 { if w.rdb.ZRem(scheduled, j).Val() > 0 {
err = push(w.rdb, job.Queue, job.Task) err = push(w.rdb, &msg)
if err != nil { if err != nil {
log.Printf("could not push task to queue %q: %v", job.Queue, err) log.Printf("could not push task to queue %q: %v", msg.Queue, err)
// TODO(hibiken): Handle this error properly. Add back to scheduled ZSET? // TODO(hibiken): Handle this error properly. Add back to scheduled ZSET?
continue continue
} }
@ -176,12 +200,12 @@ func (w *Workers) pollScheduledTasks() {
} }
// push pushes the task to the specified queue to get picked up by a worker. // push pushes the task to the specified queue to get picked up by a worker.
func push(rdb *redis.Client, queue string, t *Task) error { func push(rdb *redis.Client, msg *taskMessage) error {
bytes, err := json.Marshal(t) bytes, err := json.Marshal(msg)
if err != nil { if err != nil {
return fmt.Errorf("could not encode task into JSON: %v", err) return fmt.Errorf("could not encode task into JSON: %v", err)
} }
qname := queuePrefix + queue qname := queuePrefix + msg.Queue
err = rdb.SAdd(allQueues, qname).Err() err = rdb.SAdd(allQueues, qname).Err()
if err != nil { if err != nil {
return fmt.Errorf("could not execute command SADD %q %q: %v", return fmt.Errorf("could not execute command SADD %q %q: %v",