mirror of
				https://github.com/hibiken/asynq.git
				synced 2025-10-26 11:16:12 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			178 lines
		
	
	
		
			4.2 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			178 lines
		
	
	
		
			4.2 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package asynq
 | |
| 
 | |
| import (
 | |
| 	"encoding/json"
 | |
| 	"fmt"
 | |
| 	"log"
 | |
| 	"strconv"
 | |
| 	"strings"
 | |
| 	"time"
 | |
| 
 | |
| 	"github.com/go-redis/redis/v7"
 | |
| 	"github.com/google/uuid"
 | |
| )
 | |
| 
 | |
| // Redis keys
 | |
| const (
 | |
| 	queuePrefix = "asynq:queues:"
 | |
| 	allQueues   = "asynq:queues"
 | |
| 	scheduled   = "asynq:scheduled"
 | |
| )
 | |
| 
 | |
| // Client is an interface for scheduling tasks.
 | |
| type Client struct {
 | |
| 	rdb *redis.Client
 | |
| }
 | |
| 
 | |
| // Task represents a task to be performed.
 | |
| type Task struct {
 | |
| 	Handler string
 | |
| 	Args    []interface{}
 | |
| }
 | |
| 
 | |
| type delayedTask struct {
 | |
| 	ID    string
 | |
| 	Queue string
 | |
| 	Task  *Task
 | |
| }
 | |
| 
 | |
| // RedisOpt specifies redis options.
 | |
| type RedisOpt struct {
 | |
| 	Addr     string
 | |
| 	Password string
 | |
| }
 | |
| 
 | |
| // NewClient creates and returns a new client.
 | |
| func NewClient(opt *RedisOpt) *Client {
 | |
| 	rdb := redis.NewClient(&redis.Options{Addr: opt.Addr, Password: opt.Password})
 | |
| 	return &Client{rdb: rdb}
 | |
| }
 | |
| 
 | |
| // Enqueue pushes a given task to the specified queue.
 | |
| func (c *Client) Enqueue(queue string, task *Task, executeAt time.Time) error {
 | |
| 	if time.Now().After(executeAt) {
 | |
| 		return push(c.rdb, queue, task)
 | |
| 	}
 | |
| 	dt := &delayedTask{
 | |
| 		ID:    uuid.New().String(),
 | |
| 		Queue: queue,
 | |
| 		Task:  task,
 | |
| 	}
 | |
| 	bytes, err := json.Marshal(dt)
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 	return c.rdb.ZAdd(scheduled, &redis.Z{Member: string(bytes), Score: float64(executeAt.Unix())}).Err()
 | |
| }
 | |
| 
 | |
| //-------------------- Workers --------------------
 | |
| 
 | |
| // Workers represents a pool of workers.
 | |
| type Workers struct {
 | |
| 	rdb *redis.Client
 | |
| 
 | |
| 	// poolTokens is a counting semaphore to ensure the number of active workers
 | |
| 	// does not exceed the limit.
 | |
| 	poolTokens chan struct{}
 | |
| 
 | |
| 	// handlers maps queue name to a handler for that queue's messages.
 | |
| 	handlers map[string]func(msg string)
 | |
| }
 | |
| 
 | |
| // NewWorkers creates and returns a new Workers.
 | |
| func NewWorkers(poolSize int, opt *RedisOpt) *Workers {
 | |
| 	rdb := redis.NewClient(&redis.Options{Addr: opt.Addr, Password: opt.Password})
 | |
| 	return &Workers{
 | |
| 		rdb:        rdb,
 | |
| 		poolTokens: make(chan struct{}, poolSize),
 | |
| 		handlers:   make(map[string]func(string)),
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // Handle registers a handler function for a given queue.
 | |
| func (w *Workers) Handle(q string, fn func(msg string)) {
 | |
| 	w.handlers[q] = fn
 | |
| }
 | |
| 
 | |
| // Run starts the workers and scheduler.
 | |
| func (w *Workers) Run() {
 | |
| 	go w.pollScheduledTasks()
 | |
| 
 | |
| 	for {
 | |
| 		// pull message out of the queue and process it
 | |
| 		// TODO(hibiken): get a list of queues in order of priority
 | |
| 		res, err := w.rdb.BLPop(0, "asynq:queues:test").Result() // A timeout of zero means block indefinitely.
 | |
| 		if err != nil {
 | |
| 			if err != redis.Nil {
 | |
| 				log.Printf("error when BLPOP from %s: %v\n", "aysnq:queues:test", err)
 | |
| 			}
 | |
| 			continue
 | |
| 		}
 | |
| 
 | |
| 		q, msg := res[0], res[1]
 | |
| 		fmt.Printf("perform task %v from %s\n", msg, q)
 | |
| 		handler, ok := w.handlers[strings.TrimPrefix(q, queuePrefix)]
 | |
| 		if !ok {
 | |
| 			log.Printf("no handler found for queue %q\n", strings.TrimPrefix(q, queuePrefix))
 | |
| 			continue
 | |
| 		}
 | |
| 		w.poolTokens <- struct{}{} // acquire a token
 | |
| 		go func(msg string) {
 | |
| 			handler(msg)
 | |
| 			<-w.poolTokens
 | |
| 		}(msg)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (w *Workers) pollScheduledTasks() {
 | |
| 	for {
 | |
| 		// Get next items in the queue with scores (time to execute) <= now.
 | |
| 		now := time.Now().Unix()
 | |
| 		jobs, err := w.rdb.ZRangeByScore(scheduled,
 | |
| 			&redis.ZRangeBy{
 | |
| 				Min: "-inf",
 | |
| 				Max: strconv.Itoa(int(now))}).Result()
 | |
| 		fmt.Printf("len(jobs) = %d\n", len(jobs))
 | |
| 		if err != nil {
 | |
| 			log.Printf("radis command ZRANGEBYSCORE failed: %v\n", err)
 | |
| 			continue
 | |
| 		}
 | |
| 		if len(jobs) == 0 {
 | |
| 			fmt.Println("jobs empty")
 | |
| 			time.Sleep(5 * time.Second)
 | |
| 			continue
 | |
| 		}
 | |
| 
 | |
| 		for _, j := range jobs {
 | |
| 			var job delayedTask
 | |
| 			err = json.Unmarshal([]byte(j), &job)
 | |
| 			if err != nil {
 | |
| 				fmt.Println("unmarshal failed")
 | |
| 				continue
 | |
| 			}
 | |
| 
 | |
| 			if w.rdb.ZRem(scheduled, j).Val() > 0 {
 | |
| 				err = push(w.rdb, job.Queue, job.Task)
 | |
| 				if err != nil {
 | |
| 					log.Printf("could not push task to queue %q: %v", job.Queue, err)
 | |
| 					// TODO(hibiken): Handle this error properly. Add back to scheduled ZSET?
 | |
| 					continue
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func push(rdb *redis.Client, queue string, t *Task) error {
 | |
| 	bytes, err := json.Marshal(t)
 | |
| 	if err != nil {
 | |
| 		return fmt.Errorf("could not encode task into JSON: %v", err)
 | |
| 	}
 | |
| 	err = rdb.SAdd(allQueues, queue).Err()
 | |
| 	if err != nil {
 | |
| 		return fmt.Errorf("could not execute command SADD %q %q: %v",
 | |
| 			allQueues, queue, err)
 | |
| 	}
 | |
| 	return rdb.RPush(queuePrefix+queue, string(bytes)).Err()
 | |
| }
 |