- Config: try ENV_FILE, .env, ../.env for loading; trim trailing slash from BaseURL - Log BASE_URL at server startup for verification - .env.example: document BASE_URL - Tasks, projects, tags, migrations and related API/handlers Made-with: Cursor
161 lines
4.3 KiB
Go
161 lines
4.3 KiB
Go
package scheduler
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"fmt"
|
|
"log"
|
|
"time"
|
|
|
|
"github.com/calendarapi/internal/repository"
|
|
"github.com/calendarapi/internal/utils"
|
|
"github.com/google/uuid"
|
|
"github.com/hibiken/asynq"
|
|
)
|
|
|
|
const TypeReminder = "reminder:send"
|
|
const TypeSubscriptionSync = "subscription:sync"
|
|
const TypeTaskReminder = "task_reminder:send"
|
|
const TypeRecurringTask = "recurring_task:generate"
|
|
|
|
type ReminderPayload struct {
|
|
EventID uuid.UUID `json:"event_id"`
|
|
ReminderID uuid.UUID `json:"reminder_id"`
|
|
UserID uuid.UUID `json:"user_id"`
|
|
}
|
|
|
|
type Scheduler struct {
|
|
client *asynq.Client
|
|
}
|
|
|
|
func NewScheduler(redisAddr string) *Scheduler {
|
|
client := asynq.NewClient(asynq.RedisClientOpt{Addr: redisAddr})
|
|
return &Scheduler{client: client}
|
|
}
|
|
|
|
func (s *Scheduler) ScheduleReminder(_ context.Context, eventID, reminderID, userID uuid.UUID, triggerAt time.Time) error {
|
|
payload, err := json.Marshal(ReminderPayload{
|
|
EventID: eventID,
|
|
ReminderID: reminderID,
|
|
UserID: userID,
|
|
})
|
|
if err != nil {
|
|
return fmt.Errorf("marshal reminder payload: %w", err)
|
|
}
|
|
|
|
task := asynq.NewTask(TypeReminder, payload)
|
|
_, err = s.client.Enqueue(task,
|
|
asynq.ProcessAt(triggerAt),
|
|
asynq.MaxRetry(5),
|
|
asynq.Queue("reminders"),
|
|
)
|
|
if err != nil {
|
|
return fmt.Errorf("enqueue reminder: %w", err)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
type TaskReminderPayload struct {
|
|
TaskReminderID uuid.UUID `json:"task_reminder_id"`
|
|
TaskID uuid.UUID `json:"task_id"`
|
|
OwnerID uuid.UUID `json:"owner_id"`
|
|
}
|
|
|
|
type RecurringTaskPayload struct {
|
|
TaskID uuid.UUID `json:"task_id"`
|
|
OwnerID uuid.UUID `json:"owner_id"`
|
|
}
|
|
|
|
type SubscriptionSyncPayload struct {
|
|
SubscriptionID string `json:"subscription_id"`
|
|
}
|
|
|
|
func (s *Scheduler) ScheduleTaskReminder(_ context.Context, reminderID, taskID, ownerID uuid.UUID, triggerAt time.Time) error {
|
|
payload, err := json.Marshal(TaskReminderPayload{
|
|
TaskReminderID: reminderID,
|
|
TaskID: taskID,
|
|
OwnerID: ownerID,
|
|
})
|
|
if err != nil {
|
|
return fmt.Errorf("marshal task reminder payload: %w", err)
|
|
}
|
|
task := asynq.NewTask(TypeTaskReminder, payload)
|
|
_, err = s.client.Enqueue(task,
|
|
asynq.ProcessAt(triggerAt),
|
|
asynq.MaxRetry(3),
|
|
asynq.Queue("reminders"),
|
|
)
|
|
return err
|
|
}
|
|
|
|
func (s *Scheduler) EnqueueRecurringTask(ctx context.Context, taskID, ownerID uuid.UUID) error {
|
|
payload, err := json.Marshal(RecurringTaskPayload{TaskID: taskID, OwnerID: ownerID})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
task := asynq.NewTask(TypeRecurringTask, payload)
|
|
_, err = s.client.Enqueue(task, asynq.MaxRetry(3), asynq.Queue("default"))
|
|
return err
|
|
}
|
|
|
|
func (s *Scheduler) EnqueueSubscriptionSync(ctx context.Context, subscriptionID string) error {
|
|
payload, err := json.Marshal(SubscriptionSyncPayload{SubscriptionID: subscriptionID})
|
|
if err != nil {
|
|
return fmt.Errorf("marshal subscription sync payload: %w", err)
|
|
}
|
|
task := asynq.NewTask(TypeSubscriptionSync, payload)
|
|
_, err = s.client.Enqueue(task,
|
|
asynq.MaxRetry(3),
|
|
asynq.Queue("subscriptions"),
|
|
)
|
|
if err != nil {
|
|
return fmt.Errorf("enqueue subscription sync: %w", err)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (s *Scheduler) Close() error {
|
|
return s.client.Close()
|
|
}
|
|
|
|
// StartSubscriptionEnqueuer runs a goroutine that every interval lists subscriptions due for sync
|
|
// and enqueues a sync task for each. Call with a cancellable context to stop.
|
|
func StartSubscriptionEnqueuer(ctx context.Context, queries *repository.Queries, sched *Scheduler, interval time.Duration) {
|
|
ticker := time.NewTicker(interval)
|
|
defer ticker.Stop()
|
|
for {
|
|
select {
|
|
case <-ctx.Done():
|
|
return
|
|
case <-ticker.C:
|
|
subs, err := queries.ListSubscriptionsDueForSync(ctx)
|
|
if err != nil {
|
|
log.Printf("subscription enqueuer: list due for sync: %v", err)
|
|
continue
|
|
}
|
|
for _, sub := range subs {
|
|
subID := utils.FromPgUUID(sub.ID)
|
|
if err := sched.EnqueueSubscriptionSync(ctx, subID.String()); err != nil {
|
|
log.Printf("subscription enqueuer: enqueue %s: %v", subID, err)
|
|
}
|
|
}
|
|
if len(subs) > 0 {
|
|
log.Printf("subscription enqueuer: enqueued %d subscriptions", len(subs))
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
type NoopScheduler struct{}
|
|
|
|
func (NoopScheduler) ScheduleReminder(_ context.Context, _, _, _ uuid.UUID, _ time.Time) error {
|
|
return nil
|
|
}
|
|
|
|
func (NoopScheduler) ScheduleTaskReminder(_ context.Context, _, _, _ uuid.UUID, _ time.Time) error {
|
|
return nil
|
|
}
|
|
|
|
func (NoopScheduler) Close() error { return nil }
|