mirror of
https://github.com/coder/coder.git
synced 2025-07-06 15:41:45 +00:00
126 lines
3.9 KiB
Go
126 lines
3.9 KiB
Go
package schedule
|
|
|
|
import (
|
|
"context"
|
|
"strings"
|
|
|
|
"github.com/google/uuid"
|
|
"golang.org/x/xerrors"
|
|
|
|
"github.com/coder/coder/v2/coderd/database"
|
|
agpl "github.com/coder/coder/v2/coderd/schedule"
|
|
"github.com/coder/coder/v2/coderd/schedule/cron"
|
|
"github.com/coder/coder/v2/coderd/tracing"
|
|
)
|
|
|
|
// enterpriseUserQuietHoursScheduleStore provides an
|
|
// agpl.UserQuietHoursScheduleStore that has all fields implemented for
|
|
// enterprise customers.
|
|
type enterpriseUserQuietHoursScheduleStore struct {
|
|
defaultSchedule string
|
|
userCanSet bool
|
|
}
|
|
|
|
var _ agpl.UserQuietHoursScheduleStore = &enterpriseUserQuietHoursScheduleStore{}
|
|
|
|
func NewEnterpriseUserQuietHoursScheduleStore(defaultSchedule string, userCanSet bool) (agpl.UserQuietHoursScheduleStore, error) {
|
|
if defaultSchedule == "" {
|
|
return nil, xerrors.Errorf("default schedule must be set")
|
|
}
|
|
|
|
s := &enterpriseUserQuietHoursScheduleStore{
|
|
defaultSchedule: defaultSchedule,
|
|
userCanSet: userCanSet,
|
|
}
|
|
|
|
// The context is only used for tracing so using a background ctx is fine.
|
|
_, err := s.parseSchedule(context.Background(), defaultSchedule)
|
|
if err != nil {
|
|
return nil, xerrors.Errorf("parse default schedule: %w", err)
|
|
}
|
|
|
|
return s, nil
|
|
}
|
|
|
|
func (s *enterpriseUserQuietHoursScheduleStore) parseSchedule(ctx context.Context, rawSchedule string) (agpl.UserQuietHoursScheduleOptions, error) {
|
|
_, span := tracing.StartSpan(ctx)
|
|
defer span.End()
|
|
|
|
userSet := true
|
|
if strings.TrimSpace(rawSchedule) == "" {
|
|
userSet = false
|
|
rawSchedule = s.defaultSchedule
|
|
}
|
|
|
|
sched, err := cron.Daily(rawSchedule)
|
|
if err != nil {
|
|
// This shouldn't get hit during Gets, only Sets.
|
|
return agpl.UserQuietHoursScheduleOptions{}, xerrors.Errorf("parse daily schedule %q: %w", rawSchedule, err)
|
|
}
|
|
if strings.HasPrefix(sched.Time(), "cron(") {
|
|
// Times starting with "cron(" mean it isn't a single time and probably
|
|
// a range or a list of times as a cron expression. We only support
|
|
// single times for user quiet hours schedules.
|
|
// This shouldn't get hit during Gets, only Sets.
|
|
return agpl.UserQuietHoursScheduleOptions{}, xerrors.Errorf("daily schedule %q has more than one time: %v", rawSchedule, sched.Time())
|
|
}
|
|
|
|
return agpl.UserQuietHoursScheduleOptions{
|
|
Schedule: sched,
|
|
UserSet: userSet,
|
|
UserCanSet: s.userCanSet,
|
|
}, nil
|
|
}
|
|
|
|
func (s *enterpriseUserQuietHoursScheduleStore) Get(ctx context.Context, db database.Store, userID uuid.UUID) (agpl.UserQuietHoursScheduleOptions, error) {
|
|
ctx, span := tracing.StartSpan(ctx)
|
|
defer span.End()
|
|
|
|
if !s.userCanSet {
|
|
return s.parseSchedule(ctx, "")
|
|
}
|
|
|
|
user, err := db.GetUserByID(ctx, userID)
|
|
if err != nil {
|
|
return agpl.UserQuietHoursScheduleOptions{}, xerrors.Errorf("get user by ID: %w", err)
|
|
}
|
|
|
|
return s.parseSchedule(ctx, user.QuietHoursSchedule)
|
|
}
|
|
|
|
func (s *enterpriseUserQuietHoursScheduleStore) Set(ctx context.Context, db database.Store, userID uuid.UUID, rawSchedule string) (agpl.UserQuietHoursScheduleOptions, error) {
|
|
ctx, span := tracing.StartSpan(ctx)
|
|
defer span.End()
|
|
|
|
if !s.userCanSet {
|
|
return agpl.UserQuietHoursScheduleOptions{}, agpl.ErrUserCannotSetQuietHoursSchedule
|
|
}
|
|
|
|
opts, err := s.parseSchedule(ctx, rawSchedule)
|
|
if err != nil {
|
|
return opts, err
|
|
}
|
|
|
|
// Use the tidy version when storing in the database.
|
|
rawSchedule = ""
|
|
if opts.UserSet {
|
|
rawSchedule = opts.Schedule.String()
|
|
}
|
|
_, err = db.UpdateUserQuietHoursSchedule(ctx, database.UpdateUserQuietHoursScheduleParams{
|
|
ID: userID,
|
|
QuietHoursSchedule: rawSchedule,
|
|
})
|
|
if err != nil {
|
|
return agpl.UserQuietHoursScheduleOptions{}, xerrors.Errorf("update user quiet hours schedule: %w", err)
|
|
}
|
|
|
|
// We don't update workspace build deadlines when the user changes their own
|
|
// quiet hours schedule, because they could potentially keep their workspace
|
|
// running forever.
|
|
//
|
|
// Workspace build deadlines are updated when the template admin changes the
|
|
// template's settings however.
|
|
|
|
return opts, nil
|
|
}
|