mirror of
https://github.com/pocket-id/pocket-id.git
synced 2026-03-22 22:00:08 +00:00
136 lines
3.2 KiB
Go
136 lines
3.2 KiB
Go
package job
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"log/slog"
|
|
"time"
|
|
|
|
backoff "github.com/cenkalti/backoff/v5"
|
|
"github.com/go-co-op/gocron/v2"
|
|
"github.com/google/uuid"
|
|
|
|
"github.com/pocket-id/pocket-id/backend/internal/service"
|
|
)
|
|
|
|
type Scheduler struct {
|
|
scheduler gocron.Scheduler
|
|
}
|
|
|
|
func NewScheduler() (*Scheduler, error) {
|
|
scheduler, err := gocron.NewScheduler()
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to create a new scheduler: %w", err)
|
|
}
|
|
|
|
return &Scheduler{
|
|
scheduler: scheduler,
|
|
}, nil
|
|
}
|
|
|
|
func (s *Scheduler) RemoveJob(name string) error {
|
|
jobs := s.scheduler.Jobs()
|
|
|
|
var errs []error
|
|
for _, job := range jobs {
|
|
if job.Name() == name {
|
|
err := s.scheduler.RemoveJob(job.ID())
|
|
if err != nil {
|
|
errs = append(errs, fmt.Errorf("failed to dequeue job %q with ID %q: %w", name, job.ID().String(), err))
|
|
}
|
|
}
|
|
}
|
|
|
|
return errors.Join(errs...)
|
|
}
|
|
|
|
// Run the scheduler.
|
|
// This function blocks until the context is canceled.
|
|
func (s *Scheduler) Run(ctx context.Context) error {
|
|
slog.Info("Starting job scheduler")
|
|
s.scheduler.Start()
|
|
|
|
// Block until context is canceled
|
|
<-ctx.Done()
|
|
|
|
err := s.scheduler.Shutdown()
|
|
if err != nil {
|
|
slog.Error("Error shutting down job scheduler", slog.Any("error", err))
|
|
} else {
|
|
slog.Info("Job scheduler shut down")
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (s *Scheduler) RegisterJob(ctx context.Context, name string, def gocron.JobDefinition, jobFn func(ctx context.Context) error, opts service.RegisterJobOpts) error {
|
|
// If a BackOff strategy is provided, wrap the job with retry logic.
|
|
if opts.BackOff != nil {
|
|
origJob := jobFn
|
|
jobFn = func(ctx context.Context) error {
|
|
_, err := backoff.Retry(
|
|
ctx,
|
|
func() (struct{}, error) {
|
|
return struct{}{}, origJob(ctx)
|
|
},
|
|
backoff.WithBackOff(opts.BackOff),
|
|
backoff.WithNotify(func(err error, d time.Duration) {
|
|
slog.WarnContext(ctx, "Job failed, retrying",
|
|
slog.String("name", name),
|
|
slog.Any("error", err),
|
|
slog.Duration("retryIn", d),
|
|
)
|
|
}),
|
|
)
|
|
return err
|
|
}
|
|
}
|
|
|
|
jobOptions := []gocron.JobOption{
|
|
gocron.WithContext(ctx),
|
|
gocron.WithName(name),
|
|
gocron.WithEventListeners(
|
|
gocron.BeforeJobRuns(func(jobID uuid.UUID, jobName string) {
|
|
slog.Info("Starting job",
|
|
slog.String("name", name),
|
|
slog.String("id", jobID.String()),
|
|
)
|
|
}),
|
|
gocron.AfterJobRuns(func(jobID uuid.UUID, jobName string) {
|
|
slog.Info("Job run successfully",
|
|
slog.String("name", name),
|
|
slog.String("id", jobID.String()),
|
|
)
|
|
}),
|
|
gocron.AfterJobRunsWithError(func(jobID uuid.UUID, jobName string, err error) {
|
|
slog.Error("Job failed with error",
|
|
slog.String("name", name),
|
|
slog.String("id", jobID.String()),
|
|
slog.Any("error", err),
|
|
)
|
|
}),
|
|
),
|
|
}
|
|
|
|
if opts.RunImmediately {
|
|
jobOptions = append(jobOptions, gocron.JobOption(gocron.WithStartImmediately()))
|
|
}
|
|
|
|
jobOptions = append(jobOptions, opts.ExtraOptions...)
|
|
|
|
_, err := s.scheduler.NewJob(def, gocron.NewTask(jobFn), jobOptions...)
|
|
|
|
if err != nil {
|
|
return fmt.Errorf("failed to register job %q: %w", name, err)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func jobDefWithJitter(interval time.Duration) gocron.JobDefinition {
|
|
const jitter = 5 * time.Minute
|
|
|
|
return gocron.DurationRandomJob(interval-jitter, interval+jitter)
|
|
}
|