mirror of https://github.com/portainer/portainer
				
				
				
			
		
			
				
	
	
		
			100 lines
		
	
	
		
			2.0 KiB
		
	
	
	
		
			Go
		
	
	
			
		
		
	
	
			100 lines
		
	
	
		
			2.0 KiB
		
	
	
	
		
			Go
		
	
	
| package scheduler
 | |
| 
 | |
| import (
 | |
| 	"context"
 | |
| 	"log"
 | |
| 	"strconv"
 | |
| 	"time"
 | |
| 
 | |
| 	"github.com/pkg/errors"
 | |
| 	"github.com/robfig/cron/v3"
 | |
| 	"github.com/sirupsen/logrus"
 | |
| )
 | |
| 
 | |
| type Scheduler struct {
 | |
| 	crontab    *cron.Cron
 | |
| 	activeJobs map[cron.EntryID]context.CancelFunc
 | |
| }
 | |
| 
 | |
| func NewScheduler(ctx context.Context) *Scheduler {
 | |
| 	crontab := cron.New(cron.WithChain(cron.Recover(cron.DefaultLogger)))
 | |
| 	crontab.Start()
 | |
| 
 | |
| 	s := &Scheduler{
 | |
| 		crontab:    crontab,
 | |
| 		activeJobs: make(map[cron.EntryID]context.CancelFunc),
 | |
| 	}
 | |
| 
 | |
| 	if ctx != nil {
 | |
| 		go func() {
 | |
| 			<-ctx.Done()
 | |
| 			s.Shutdown()
 | |
| 		}()
 | |
| 	}
 | |
| 
 | |
| 	return s
 | |
| }
 | |
| 
 | |
| // Shutdown stops the scheduler and waits for it to stop if it is running; otherwise does nothing.
 | |
| func (s *Scheduler) Shutdown() error {
 | |
| 	if s.crontab == nil {
 | |
| 		return nil
 | |
| 	}
 | |
| 
 | |
| 	log.Println("[DEBUG] Stopping scheduler")
 | |
| 	ctx := s.crontab.Stop()
 | |
| 	<-ctx.Done()
 | |
| 
 | |
| 	for _, job := range s.crontab.Entries() {
 | |
| 		if cancel, ok := s.activeJobs[job.ID]; ok {
 | |
| 			cancel()
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	err := ctx.Err()
 | |
| 	if err == context.Canceled {
 | |
| 		return nil
 | |
| 	}
 | |
| 	return err
 | |
| }
 | |
| 
 | |
| // StopJob stops the job from being run in the future
 | |
| func (s *Scheduler) StopJob(jobID string) error {
 | |
| 	id, err := strconv.Atoi(jobID)
 | |
| 	if err != nil {
 | |
| 		return errors.Wrapf(err, "failed convert jobID %q to int", jobID)
 | |
| 	}
 | |
| 	entryID := cron.EntryID(id)
 | |
| 	if cancel, ok := s.activeJobs[entryID]; ok {
 | |
| 		cancel()
 | |
| 	}
 | |
| 
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| // StartJobEvery schedules a new periodic job with a given duration.
 | |
| // Returns job id that could be used to stop the given job.
 | |
| // When job run returns an error, that job won't be run again.
 | |
| func (s *Scheduler) StartJobEvery(duration time.Duration, job func() error) string {
 | |
| 	ctx, cancel := context.WithCancel(context.Background())
 | |
| 
 | |
| 	j := cron.FuncJob(func() {
 | |
| 		if err := job(); err != nil {
 | |
| 			logrus.Debug("job returned an error")
 | |
| 			cancel()
 | |
| 		}
 | |
| 	})
 | |
| 
 | |
| 	entryID := s.crontab.Schedule(cron.Every(duration), j)
 | |
| 
 | |
| 	s.activeJobs[entryID] = cancel
 | |
| 
 | |
| 	go func(entryID cron.EntryID) {
 | |
| 		<-ctx.Done()
 | |
| 		logrus.Debug("job cancelled, stopping")
 | |
| 		s.crontab.Remove(entryID)
 | |
| 	}(entryID)
 | |
| 
 | |
| 	return strconv.Itoa(int(entryID))
 | |
| }
 |