mirror of
https://github.com/navidrome/navidrome.git
synced 2026-04-26 10:30:46 +00:00
Some checks are pending
Pipeline: Test, Lint, Build / Get version info (push) Waiting to run
Pipeline: Test, Lint, Build / Push to Docker Hub (push) Blocked by required conditions
Pipeline: Test, Lint, Build / Lint Go code (push) Waiting to run
Pipeline: Test, Lint, Build / Test Go code (push) Waiting to run
Pipeline: Test, Lint, Build / Test JS code (push) Waiting to run
Pipeline: Test, Lint, Build / Lint i18n files (push) Waiting to run
Pipeline: Test, Lint, Build / Check Docker configuration (push) Waiting to run
Pipeline: Test, Lint, Build / Cleanup digest artifacts (push) Blocked by required conditions
Pipeline: Test, Lint, Build / Build (push) Blocked by required conditions
Pipeline: Test, Lint, Build / Build-1 (push) Blocked by required conditions
Pipeline: Test, Lint, Build / Build-2 (push) Blocked by required conditions
Pipeline: Test, Lint, Build / Build-3 (push) Blocked by required conditions
Pipeline: Test, Lint, Build / Build-4 (push) Blocked by required conditions
Pipeline: Test, Lint, Build / Build-5 (push) Blocked by required conditions
Pipeline: Test, Lint, Build / Build-6 (push) Blocked by required conditions
Pipeline: Test, Lint, Build / Build-7 (push) Blocked by required conditions
Pipeline: Test, Lint, Build / Build-8 (push) Blocked by required conditions
Pipeline: Test, Lint, Build / Build-9 (push) Blocked by required conditions
Pipeline: Test, Lint, Build / Build-10 (push) Blocked by required conditions
Pipeline: Test, Lint, Build / Push to GHCR (push) Blocked by required conditions
Pipeline: Test, Lint, Build / Build Windows installers (push) Blocked by required conditions
Pipeline: Test, Lint, Build / Package/Release (push) Blocked by required conditions
Pipeline: Test, Lint, Build / Upload Linux PKG (push) Blocked by required conditions
* feat(scheduler): add CrontabSchedule with crontab(5) random ~ syntax Implement ParseCrontab() that extends robfig/cron with support for the crontab(5) random ~ operator (e.g., 0~30 * * * *). Random values are resolved fresh on each Next() call for load spreading. Supports A~B, ~B, A~, and bare ~ forms in all 6 fields (including seconds). Expressions without ~ delegate to robfig's standard parser with zero overhead. Integrates into scheduler.Add() and conf.validateSchedule() so that scanner.schedule and backup.schedule config values accept ~ syntax. * refactor(scheduler): resolve random ~ values once at parse time Change from per-Next() randomization to per-parse randomization, matching crontab(5) semantics. This prevents double-firing within the same period when random values land after the current time. ParseCrontab now resolves ~ fields to concrete values, substitutes them into the spec string, and delegates to robfig's parser. This eliminates CrontabSchedule, randomField, and resolveField entirely. * test(scheduler): replace WaitGroup with channel for job execution synchronization Signed-off-by: Deluan <deluan@navidrome.org> --------- Signed-off-by: Deluan <deluan@navidrome.org>
46 lines
831 B
Go
46 lines
831 B
Go
package scheduler
|
|
|
|
import (
|
|
"context"
|
|
|
|
"github.com/navidrome/navidrome/utils/singleton"
|
|
"github.com/robfig/cron/v3"
|
|
)
|
|
|
|
type Scheduler interface {
|
|
Run(ctx context.Context)
|
|
Add(crontab string, cmd func()) (int, error)
|
|
Remove(id int)
|
|
}
|
|
|
|
func GetInstance() Scheduler {
|
|
return singleton.GetInstance(func() *scheduler {
|
|
c := cron.New(cron.WithLogger(&logger{}))
|
|
return &scheduler{
|
|
c: c,
|
|
}
|
|
})
|
|
}
|
|
|
|
type scheduler struct {
|
|
c *cron.Cron
|
|
}
|
|
|
|
func (s *scheduler) Run(ctx context.Context) {
|
|
s.c.Start()
|
|
<-ctx.Done()
|
|
s.c.Stop()
|
|
}
|
|
|
|
func (s *scheduler) Add(crontab string, cmd func()) (int, error) {
|
|
schedule, err := ParseCrontab(crontab)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
entryID := s.c.Schedule(schedule, cron.FuncJob(cmd))
|
|
return int(entryID), nil
|
|
}
|
|
|
|
func (s *scheduler) Remove(id int) {
|
|
s.c.Remove(cron.EntryID(id))
|
|
}
|