From 772979fa51c50da0c1d4922f7196cfda9ea570aa Mon Sep 17 00:00:00 2001 From: John Roesler Date: Thu, 4 Aug 2022 08:00:48 -0500 Subject: [PATCH] fix deadlock when starting / stopping the scheduler (#369) * fix deadlock when starting / stopping the scheduler * add mutex to stop chan --- scheduler.go | 19 +++++++++++++++---- 1 file changed, 15 insertions(+), 4 deletions(-) diff --git a/scheduler.go b/scheduler.go index 4f909ea0..1fa8407c 100644 --- a/scheduler.go +++ b/scheduler.go @@ -37,7 +37,8 @@ type Scheduler struct { singletonMode bool // defaults all jobs to use SingletonMode() jobCreated bool // so the scheduler knows a job was created prior to calling Every or Cron - stopChan chan struct{} // stops the scheduler + startBlockingStopChanMutex sync.Mutex + startBlockingStopChan chan struct{} // stops the scheduler } // days in a week @@ -54,7 +55,6 @@ func NewScheduler(loc *time.Location) *Scheduler { time: &trueTime{}, executor: &executor, tagsUnique: false, - stopChan: make(chan struct{}, 1), } } @@ -69,7 +69,10 @@ func (s *Scheduler) SetMaxConcurrentJobs(n int, mode limitMode) { // This blocking method can be stopped with Stop() from a separate goroutine. func (s *Scheduler) StartBlocking() { s.StartAsync() - <-s.stopChan + s.startBlockingStopChanMutex.Lock() + s.startBlockingStopChan = make(chan struct{}, 1) + s.startBlockingStopChanMutex.Unlock() + <-s.startBlockingStopChan } // StartAsync starts all jobs without blocking the current thread @@ -815,7 +818,7 @@ func (s *Scheduler) stop() { s.setRunning(false) s.stopJobs(s.jobs) s.executor.stop() - s.stopChan <- struct{}{} + s.StopBlockingChan() } func (s *Scheduler) stopJobs(jobs []*Job) { @@ -1299,3 +1302,11 @@ func (s *Scheduler) StartImmediately() *Scheduler { func (s *Scheduler) CustomTime(customTimeWrapper TimeWrapper) { s.time = customTimeWrapper } + +func (s *Scheduler) StopBlockingChan() { + s.startBlockingStopChanMutex.Lock() + if s.startBlockingStopChan != nil { + s.startBlockingStopChan <- struct{}{} + } + s.startBlockingStopChanMutex.Unlock() +}