mirror of
				https://github.com/hibiken/asynq.git
				synced 2025-10-26 11:16:12 +08:00 
			
		
		
		
	* Added Retention Option to specify retention TTL for tasks * Added ResultWriter as a client interface to write result data for the associated task
		
			
				
	
	
		
			486 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			486 lines
		
	
	
		
			13 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 asynq
 | |
| 
 | |
| import (
 | |
| 	"context"
 | |
| 	"fmt"
 | |
| 	"math/rand"
 | |
| 	"runtime"
 | |
| 	"runtime/debug"
 | |
| 	"sort"
 | |
| 	"strings"
 | |
| 	"sync"
 | |
| 	"time"
 | |
| 
 | |
| 	"github.com/hibiken/asynq/internal/base"
 | |
| 	asynqcontext "github.com/hibiken/asynq/internal/context"
 | |
| 	"github.com/hibiken/asynq/internal/errors"
 | |
| 	"github.com/hibiken/asynq/internal/log"
 | |
| 	"golang.org/x/time/rate"
 | |
| )
 | |
| 
 | |
| type processor struct {
 | |
| 	logger *log.Logger
 | |
| 	broker base.Broker
 | |
| 
 | |
| 	handler Handler
 | |
| 
 | |
| 	queueConfig map[string]int
 | |
| 
 | |
| 	// orderedQueues is set only in strict-priority mode.
 | |
| 	orderedQueues []string
 | |
| 
 | |
| 	retryDelayFunc RetryDelayFunc
 | |
| 	isFailureFunc  func(error) bool
 | |
| 
 | |
| 	errHandler ErrorHandler
 | |
| 
 | |
| 	shutdownTimeout time.Duration
 | |
| 
 | |
| 	// channel via which to send sync requests to syncer.
 | |
| 	syncRequestCh chan<- *syncRequest
 | |
| 
 | |
| 	// rate limiter to prevent spamming logs with a bunch of errors.
 | |
| 	errLogLimiter *rate.Limiter
 | |
| 
 | |
| 	// sema is a counting semaphore to ensure the number of active workers
 | |
| 	// does not exceed the limit.
 | |
| 	sema chan struct{}
 | |
| 
 | |
| 	// channel to communicate back to the long running "processor" goroutine.
 | |
| 	// once is used to send value to the channel only once.
 | |
| 	done chan struct{}
 | |
| 	once sync.Once
 | |
| 
 | |
| 	// quit channel is closed when the shutdown of the "processor" goroutine starts.
 | |
| 	quit chan struct{}
 | |
| 
 | |
| 	// abort channel communicates to the in-flight worker goroutines to stop.
 | |
| 	abort chan struct{}
 | |
| 
 | |
| 	// cancelations is a set of cancel functions for all active tasks.
 | |
| 	cancelations *base.Cancelations
 | |
| 
 | |
| 	starting chan<- *workerInfo
 | |
| 	finished chan<- *base.TaskMessage
 | |
| }
 | |
| 
 | |
| type processorParams struct {
 | |
| 	logger          *log.Logger
 | |
| 	broker          base.Broker
 | |
| 	retryDelayFunc  RetryDelayFunc
 | |
| 	isFailureFunc   func(error) bool
 | |
| 	syncCh          chan<- *syncRequest
 | |
| 	cancelations    *base.Cancelations
 | |
| 	concurrency     int
 | |
| 	queues          map[string]int
 | |
| 	strictPriority  bool
 | |
| 	errHandler      ErrorHandler
 | |
| 	shutdownTimeout time.Duration
 | |
| 	starting        chan<- *workerInfo
 | |
| 	finished        chan<- *base.TaskMessage
 | |
| }
 | |
| 
 | |
| // newProcessor constructs a new processor.
 | |
| func newProcessor(params processorParams) *processor {
 | |
| 	queues := normalizeQueues(params.queues)
 | |
| 	orderedQueues := []string(nil)
 | |
| 	if params.strictPriority {
 | |
| 		orderedQueues = sortByPriority(queues)
 | |
| 	}
 | |
| 	return &processor{
 | |
| 		logger:          params.logger,
 | |
| 		broker:          params.broker,
 | |
| 		queueConfig:     queues,
 | |
| 		orderedQueues:   orderedQueues,
 | |
| 		retryDelayFunc:  params.retryDelayFunc,
 | |
| 		isFailureFunc:   params.isFailureFunc,
 | |
| 		syncRequestCh:   params.syncCh,
 | |
| 		cancelations:    params.cancelations,
 | |
| 		errLogLimiter:   rate.NewLimiter(rate.Every(3*time.Second), 1),
 | |
| 		sema:            make(chan struct{}, params.concurrency),
 | |
| 		done:            make(chan struct{}),
 | |
| 		quit:            make(chan struct{}),
 | |
| 		abort:           make(chan struct{}),
 | |
| 		errHandler:      params.errHandler,
 | |
| 		handler:         HandlerFunc(func(ctx context.Context, t *Task) error { return fmt.Errorf("handler not set") }),
 | |
| 		shutdownTimeout: params.shutdownTimeout,
 | |
| 		starting:        params.starting,
 | |
| 		finished:        params.finished,
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // Note: stops only the "processor" goroutine, does not stop workers.
 | |
| // It's safe to call this method multiple times.
 | |
| func (p *processor) stop() {
 | |
| 	p.once.Do(func() {
 | |
| 		p.logger.Debug("Processor shutting down...")
 | |
| 		// Unblock if processor is waiting for sema token.
 | |
| 		close(p.quit)
 | |
| 		// Signal the processor goroutine to stop processing tasks
 | |
| 		// from the queue.
 | |
| 		p.done <- struct{}{}
 | |
| 	})
 | |
| }
 | |
| 
 | |
| // NOTE: once shutdown, processor cannot be re-started.
 | |
| func (p *processor) shutdown() {
 | |
| 	p.stop()
 | |
| 
 | |
| 	time.AfterFunc(p.shutdownTimeout, func() { close(p.abort) })
 | |
| 
 | |
| 	p.logger.Info("Waiting for all workers to finish...")
 | |
| 	// block until all workers have released the token
 | |
| 	for i := 0; i < cap(p.sema); i++ {
 | |
| 		p.sema <- struct{}{}
 | |
| 	}
 | |
| 	p.logger.Info("All workers have finished")
 | |
| }
 | |
| 
 | |
| func (p *processor) start(wg *sync.WaitGroup) {
 | |
| 	wg.Add(1)
 | |
| 	go func() {
 | |
| 		defer wg.Done()
 | |
| 		for {
 | |
| 			select {
 | |
| 			case <-p.done:
 | |
| 				p.logger.Debug("Processor done")
 | |
| 				return
 | |
| 			default:
 | |
| 				p.exec()
 | |
| 			}
 | |
| 		}
 | |
| 	}()
 | |
| }
 | |
| 
 | |
| // exec pulls a task out of the queue and starts a worker goroutine to
 | |
| // process the task.
 | |
| func (p *processor) exec() {
 | |
| 	select {
 | |
| 	case <-p.quit:
 | |
| 		return
 | |
| 	case p.sema <- struct{}{}: // acquire token
 | |
| 		qnames := p.queues()
 | |
| 		msg, deadline, err := p.broker.Dequeue(qnames...)
 | |
| 		switch {
 | |
| 		case errors.Is(err, errors.ErrNoProcessableTask):
 | |
| 			p.logger.Debug("All queues are empty")
 | |
| 			// Queues are empty, this is a normal behavior.
 | |
| 			// Sleep to avoid slamming redis and let scheduler move tasks into queues.
 | |
| 			// Note: We are not using blocking pop operation and polling queues instead.
 | |
| 			// This adds significant load to redis.
 | |
| 			time.Sleep(time.Second)
 | |
| 			<-p.sema // release token
 | |
| 			return
 | |
| 		case err != nil:
 | |
| 			if p.errLogLimiter.Allow() {
 | |
| 				p.logger.Errorf("Dequeue error: %v", err)
 | |
| 			}
 | |
| 			<-p.sema // release token
 | |
| 			return
 | |
| 		}
 | |
| 
 | |
| 		p.starting <- &workerInfo{msg, time.Now(), deadline}
 | |
| 		go func() {
 | |
| 			defer func() {
 | |
| 				p.finished <- msg
 | |
| 				<-p.sema // release token
 | |
| 			}()
 | |
| 
 | |
| 			ctx, cancel := asynqcontext.New(msg, deadline)
 | |
| 			p.cancelations.Add(msg.ID, cancel)
 | |
| 			defer func() {
 | |
| 				cancel()
 | |
| 				p.cancelations.Delete(msg.ID)
 | |
| 			}()
 | |
| 
 | |
| 			// check context before starting a worker goroutine.
 | |
| 			select {
 | |
| 			case <-ctx.Done():
 | |
| 				// already canceled (e.g. deadline exceeded).
 | |
| 				p.handleFailedMessage(ctx, msg, ctx.Err())
 | |
| 				return
 | |
| 			default:
 | |
| 			}
 | |
| 
 | |
| 			resCh := make(chan error, 1)
 | |
| 			go func() {
 | |
| 				task := newTask(
 | |
| 					msg.Type,
 | |
| 					msg.Payload,
 | |
| 					&ResultWriter{
 | |
| 						id:     msg.ID,
 | |
| 						qname:  msg.Queue,
 | |
| 						broker: p.broker,
 | |
| 						ctx:    ctx,
 | |
| 					},
 | |
| 				)
 | |
| 				resCh <- p.perform(ctx, task)
 | |
| 			}()
 | |
| 
 | |
| 			select {
 | |
| 			case <-p.abort:
 | |
| 				// time is up, push the message back to queue and quit this worker goroutine.
 | |
| 				p.logger.Warnf("Quitting worker. task id=%s", msg.ID)
 | |
| 				p.requeue(msg)
 | |
| 				return
 | |
| 			case <-ctx.Done():
 | |
| 				p.handleFailedMessage(ctx, msg, ctx.Err())
 | |
| 				return
 | |
| 			case resErr := <-resCh:
 | |
| 				if resErr != nil {
 | |
| 					p.handleFailedMessage(ctx, msg, resErr)
 | |
| 					return
 | |
| 				}
 | |
| 				p.handleSucceededMessage(ctx, msg)
 | |
| 			}
 | |
| 		}()
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (p *processor) requeue(msg *base.TaskMessage) {
 | |
| 	err := p.broker.Requeue(msg)
 | |
| 	if err != nil {
 | |
| 		p.logger.Errorf("Could not push task id=%s back to queue: %v", msg.ID, err)
 | |
| 	} else {
 | |
| 		p.logger.Infof("Pushed task id=%s back to queue", msg.ID)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (p *processor) handleSucceededMessage(ctx context.Context, msg *base.TaskMessage) {
 | |
| 	if msg.Retention > 0 {
 | |
| 		p.markAsComplete(ctx, msg)
 | |
| 	} else {
 | |
| 		p.markAsDone(ctx, msg)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (p *processor) markAsComplete(ctx context.Context, msg *base.TaskMessage) {
 | |
| 	err := p.broker.MarkAsComplete(msg)
 | |
| 	if err != nil {
 | |
| 		errMsg := fmt.Sprintf("Could not move task id=%s type=%q from %q to %q:  %+v",
 | |
| 			msg.ID, msg.Type, base.ActiveKey(msg.Queue), base.CompletedKey(msg.Queue), err)
 | |
| 		deadline, ok := ctx.Deadline()
 | |
| 		if !ok {
 | |
| 			panic("asynq: internal error: missing deadline in context")
 | |
| 		}
 | |
| 		p.logger.Warnf("%s; Will retry syncing", errMsg)
 | |
| 		p.syncRequestCh <- &syncRequest{
 | |
| 			fn: func() error {
 | |
| 				return p.broker.MarkAsComplete(msg)
 | |
| 			},
 | |
| 			errMsg:   errMsg,
 | |
| 			deadline: deadline,
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (p *processor) markAsDone(ctx context.Context, msg *base.TaskMessage) {
 | |
| 	err := p.broker.Done(msg)
 | |
| 	if err != nil {
 | |
| 		errMsg := fmt.Sprintf("Could not remove task id=%s type=%q from %q err: %+v", msg.ID, msg.Type, base.ActiveKey(msg.Queue), err)
 | |
| 		deadline, ok := ctx.Deadline()
 | |
| 		if !ok {
 | |
| 			panic("asynq: internal error: missing deadline in context")
 | |
| 		}
 | |
| 		p.logger.Warnf("%s; Will retry syncing", errMsg)
 | |
| 		p.syncRequestCh <- &syncRequest{
 | |
| 			fn: func() error {
 | |
| 				return p.broker.Done(msg)
 | |
| 			},
 | |
| 			errMsg:   errMsg,
 | |
| 			deadline: deadline,
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // SkipRetry is used as a return value from Handler.ProcessTask to indicate that
 | |
| // the task should not be retried and should be archived instead.
 | |
| var SkipRetry = errors.New("skip retry for the task")
 | |
| 
 | |
| func (p *processor) handleFailedMessage(ctx context.Context, msg *base.TaskMessage, err error) {
 | |
| 	if p.errHandler != nil {
 | |
| 		p.errHandler.HandleError(ctx, NewTask(msg.Type, msg.Payload), err)
 | |
| 	}
 | |
| 	if !p.isFailureFunc(err) {
 | |
| 		// retry the task without marking it as failed
 | |
| 		p.retry(ctx, msg, err, false /*isFailure*/)
 | |
| 		return
 | |
| 	}
 | |
| 	if msg.Retried >= msg.Retry || errors.Is(err, SkipRetry) {
 | |
| 		p.logger.Warnf("Retry exhausted for task id=%s", msg.ID)
 | |
| 		p.archive(ctx, msg, err)
 | |
| 	} else {
 | |
| 		p.retry(ctx, msg, err, true /*isFailure*/)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (p *processor) retry(ctx context.Context, msg *base.TaskMessage, e error, isFailure bool) {
 | |
| 	d := p.retryDelayFunc(msg.Retried, e, NewTask(msg.Type, msg.Payload))
 | |
| 	retryAt := time.Now().Add(d)
 | |
| 	err := p.broker.Retry(msg, retryAt, e.Error(), isFailure)
 | |
| 	if err != nil {
 | |
| 		errMsg := fmt.Sprintf("Could not move task id=%s from %q to %q", msg.ID, base.ActiveKey(msg.Queue), base.RetryKey(msg.Queue))
 | |
| 		deadline, ok := ctx.Deadline()
 | |
| 		if !ok {
 | |
| 			panic("asynq: internal error: missing deadline in context")
 | |
| 		}
 | |
| 		p.logger.Warnf("%s; Will retry syncing", errMsg)
 | |
| 		p.syncRequestCh <- &syncRequest{
 | |
| 			fn: func() error {
 | |
| 				return p.broker.Retry(msg, retryAt, e.Error(), isFailure)
 | |
| 			},
 | |
| 			errMsg:   errMsg,
 | |
| 			deadline: deadline,
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (p *processor) archive(ctx context.Context, msg *base.TaskMessage, e error) {
 | |
| 	err := p.broker.Archive(msg, e.Error())
 | |
| 	if err != nil {
 | |
| 		errMsg := fmt.Sprintf("Could not move task id=%s from %q to %q", msg.ID, base.ActiveKey(msg.Queue), base.ArchivedKey(msg.Queue))
 | |
| 		deadline, ok := ctx.Deadline()
 | |
| 		if !ok {
 | |
| 			panic("asynq: internal error: missing deadline in context")
 | |
| 		}
 | |
| 		p.logger.Warnf("%s; Will retry syncing", errMsg)
 | |
| 		p.syncRequestCh <- &syncRequest{
 | |
| 			fn: func() error {
 | |
| 				return p.broker.Archive(msg, e.Error())
 | |
| 			},
 | |
| 			errMsg:   errMsg,
 | |
| 			deadline: deadline,
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // queues returns a list of queues to query.
 | |
| // Order of the queue names is based on the priority of each queue.
 | |
| // Queue names is sorted by their priority level if strict-priority is true.
 | |
| // If strict-priority is false, then the order of queue names are roughly based on
 | |
| // the priority level but randomized in order to avoid starving low priority queues.
 | |
| func (p *processor) queues() []string {
 | |
| 	// skip the overhead of generating a list of queue names
 | |
| 	// if we are processing one queue.
 | |
| 	if len(p.queueConfig) == 1 {
 | |
| 		for qname := range p.queueConfig {
 | |
| 			return []string{qname}
 | |
| 		}
 | |
| 	}
 | |
| 	if p.orderedQueues != nil {
 | |
| 		return p.orderedQueues
 | |
| 	}
 | |
| 	var names []string
 | |
| 	for qname, priority := range p.queueConfig {
 | |
| 		for i := 0; i < priority; i++ {
 | |
| 			names = append(names, qname)
 | |
| 		}
 | |
| 	}
 | |
| 	r := rand.New(rand.NewSource(time.Now().UnixNano()))
 | |
| 	r.Shuffle(len(names), func(i, j int) { names[i], names[j] = names[j], names[i] })
 | |
| 	return uniq(names, len(p.queueConfig))
 | |
| }
 | |
| 
 | |
| // perform calls the handler with the given task.
 | |
| // If the call returns without panic, it simply returns the value,
 | |
| // otherwise, it recovers from panic and returns an error.
 | |
| func (p *processor) perform(ctx context.Context, task *Task) (err error) {
 | |
| 	defer func() {
 | |
| 		if x := recover(); x != nil {
 | |
| 			p.logger.Errorf("recovering from panic. See the stack trace below for details:\n%s", string(debug.Stack()))
 | |
| 			_, file, line, ok := runtime.Caller(1) // skip the first frame (panic itself)
 | |
| 			if ok && strings.Contains(file, "runtime/") {
 | |
| 				// The panic came from the runtime, most likely due to incorrect
 | |
| 				// map/slice usage. The parent frame should have the real trigger.
 | |
| 				_, file, line, ok = runtime.Caller(2)
 | |
| 			}
 | |
| 
 | |
| 			// Include the file and line number info in the error, if runtime.Caller returned ok.
 | |
| 			if ok {
 | |
| 				err = fmt.Errorf("panic [%s:%d]: %v", file, line, x)
 | |
| 			} else {
 | |
| 				err = fmt.Errorf("panic: %v", x)
 | |
| 			}
 | |
| 		}
 | |
| 	}()
 | |
| 	return p.handler.ProcessTask(ctx, task)
 | |
| }
 | |
| 
 | |
| // uniq dedupes elements and returns a slice of unique names of length l.
 | |
| // Order of the output slice is based on the input list.
 | |
| func uniq(names []string, l int) []string {
 | |
| 	var res []string
 | |
| 	seen := make(map[string]struct{})
 | |
| 	for _, s := range names {
 | |
| 		if _, ok := seen[s]; !ok {
 | |
| 			seen[s] = struct{}{}
 | |
| 			res = append(res, s)
 | |
| 		}
 | |
| 		if len(res) == l {
 | |
| 			break
 | |
| 		}
 | |
| 	}
 | |
| 	return res
 | |
| }
 | |
| 
 | |
| // sortByPriority returns a list of queue names sorted by
 | |
| // their priority level in descending order.
 | |
| func sortByPriority(qcfg map[string]int) []string {
 | |
| 	var queues []*queue
 | |
| 	for qname, n := range qcfg {
 | |
| 		queues = append(queues, &queue{qname, n})
 | |
| 	}
 | |
| 	sort.Sort(sort.Reverse(byPriority(queues)))
 | |
| 	var res []string
 | |
| 	for _, q := range queues {
 | |
| 		res = append(res, q.name)
 | |
| 	}
 | |
| 	return res
 | |
| }
 | |
| 
 | |
| type queue struct {
 | |
| 	name     string
 | |
| 	priority int
 | |
| }
 | |
| 
 | |
| type byPriority []*queue
 | |
| 
 | |
| func (x byPriority) Len() int           { return len(x) }
 | |
| func (x byPriority) Less(i, j int) bool { return x[i].priority < x[j].priority }
 | |
| func (x byPriority) Swap(i, j int)      { x[i], x[j] = x[j], x[i] }
 | |
| 
 | |
| // normalizeQueues divides priority numbers by their greatest common divisor.
 | |
| func normalizeQueues(queues map[string]int) map[string]int {
 | |
| 	var xs []int
 | |
| 	for _, x := range queues {
 | |
| 		xs = append(xs, x)
 | |
| 	}
 | |
| 	d := gcd(xs...)
 | |
| 	res := make(map[string]int)
 | |
| 	for q, x := range queues {
 | |
| 		res[q] = x / d
 | |
| 	}
 | |
| 	return res
 | |
| }
 | |
| 
 | |
| func gcd(xs ...int) int {
 | |
| 	fn := func(x, y int) int {
 | |
| 		for y > 0 {
 | |
| 			x, y = y, x%y
 | |
| 		}
 | |
| 		return x
 | |
| 	}
 | |
| 	res := xs[0]
 | |
| 	for i := 0; i < len(xs); i++ {
 | |
| 		res = fn(xs[i], res)
 | |
| 		if res == 1 {
 | |
| 			return 1
 | |
| 		}
 | |
| 	}
 | |
| 	return res
 | |
| }
 |