mirror of
https://github.com/AdguardTeam/AdGuardHome.git
synced 2025-01-09 23:47:23 +03:00
aac36a2d2f
Updates #951. Squashed commit of the following: commit 6b840fd516f5a87fde0420e3aceb9c239b22c974 Author: Stanislav Chzhen <s.chzhen@adguard.com> Date: Tue Aug 29 19:53:03 2023 +0300 client: imp docs more commit 7fc8f0363fbe4c4266cb0f67428fe4d18c351d2d Author: Stanislav Chzhen <s.chzhen@adguard.com> Date: Tue Aug 29 19:40:00 2023 +0300 client: imp docs commit 00bc14d5760614f2797714cdc2c4c19b1a94b86e Author: Ildar Kamalov <ik@adguard.com> Date: Mon Aug 28 18:43:49 2023 +0300 try to fix lock file commit d749df74b576091e0b58928d86ea8b3b49f919da Merge: c69f9230be1f6229e5
Author: Stanislav Chzhen <s.chzhen@adguard.com> Date: Mon Aug 28 18:14:02 2023 +0300 Merge branch 'master' into 951-blocked-services-schedule-api commit c69f9230b12f7c983db06b74324b3df77d74b32b Author: Ildar Kamalov <ik@adguard.com> Date: Mon Aug 28 17:16:20 2023 +0300 revert eslintrc commit b37916c2dff0ddea5293d87570bb58e3443d2d21 Author: Ildar Kamalov <ik@adguard.com> Date: Mon Aug 28 12:02:39 2023 +0300 fix translations commit f5bb67d81506c687d0abd580049a3eee0af808e0 Author: Ildar Kamalov <ik@adguard.com> Date: Mon Aug 28 11:43:57 2023 +0300 fix helpers commit 13ec6a8b3a0acfb62762ae7e46c6e98eb7c82212 Author: Ildar Kamalov <ik@adguard.com> Date: Mon Aug 28 11:24:57 2023 +0300 remove todo commit 23724ec2fd683ed17b9f1cee841ad9aaf4c9d04f Author: Ildar Kamalov <ik@adguard.com> Date: Mon Aug 28 09:56:56 2023 +0300 add clients schedule form commit 84d29e558a329068e64e7a95ee183946aa4515b5 Author: Ildar Kamalov <ik@adguard.com> Date: Fri Aug 25 17:44:40 2023 +0300 fix schedule form commit 83e4017688082e9eb670091d5a24d98157050502 Author: Ildar Kamalov <ik@adguard.com> Date: Fri Aug 18 12:58:16 2023 +0300 remove unused commit ef2b68e138da382e3cf42586ae604e12d9493504 Author: Ildar Kamalov <ik@adguard.com> Date: Fri Aug 18 12:57:37 2023 +0300 client: fix translation string commit 32ea80c968f52f18adbc811b2f06874644cdfe20 Author: Ildar Kamalov <ik@adguard.com> Date: Fri Aug 18 12:26:26 2023 +0300 wip schedule commit 9b770873859186c9424c8d108812e32ddff33bad Author: Stanislav Chzhen <s.chzhen@adguard.com> Date: Fri Jul 21 14:29:50 2023 +0300 all: imp naming commit ea4e9514ea3b264bcce7f2a301db817de4e87059 Author: Stanislav Chzhen <s.chzhen@adguard.com> Date: Wed Jul 19 18:09:27 2023 +0300 all: imp code commit 98a705bdaa5c1e79394c73e5d75af2416fe9f297 Author: Stanislav Chzhen <s.chzhen@adguard.com> Date: Tue Jul 18 18:23:26 2023 +0300 all: imp naming commit 4f84b55c7bfc9f7b680feac0ec45f5ea9189299a Author: Stanislav Chzhen <s.chzhen@adguard.com> Date: Fri Jul 14 15:01:17 2023 +0300 all: add global schedule api commit 87cf1646869ee9138964b47a27b7493674c8854a Merge: cabb80ac12adc8624c
Author: Stanislav Chzhen <s.chzhen@adguard.com> Date: Fri Jul 14 12:09:29 2023 +0300 Merge branch 'master' into 951-blocked-services-schedule-api commit cabb80ac16de437a8118bb0166479574379c97a3 Author: Stanislav Chzhen <s.chzhen@adguard.com> Date: Thu Jul 13 13:37:23 2023 +0300 openapi: fix typo commit 2279b03acbcfc3d76216f8aaf30ae1c7894127bc Author: Stanislav Chzhen <s.chzhen@adguard.com> Date: Thu Jul 13 12:26:19 2023 +0300 all: imp docs ... and 3 more commits
360 lines
9.6 KiB
Go
360 lines
9.6 KiB
Go
// Package schedule provides types for scheduling.
|
|
package schedule
|
|
|
|
import (
|
|
"encoding/json"
|
|
"fmt"
|
|
"time"
|
|
|
|
"github.com/AdguardTeam/AdGuardHome/internal/aghhttp"
|
|
"github.com/AdguardTeam/golibs/errors"
|
|
"github.com/AdguardTeam/golibs/timeutil"
|
|
"gopkg.in/yaml.v3"
|
|
)
|
|
|
|
// Weekly is a schedule for one week. Each day of the week has one range with
|
|
// a beginning and an end.
|
|
type Weekly struct {
|
|
// location is used to calculate the offsets of the day ranges.
|
|
location *time.Location
|
|
|
|
// days are the day ranges of this schedule. The indexes of this array are
|
|
// the [time.Weekday] values.
|
|
days [7]dayRange
|
|
}
|
|
|
|
// EmptyWeekly creates empty weekly schedule with local time zone.
|
|
func EmptyWeekly() (w *Weekly) {
|
|
return &Weekly{
|
|
location: time.Local,
|
|
}
|
|
}
|
|
|
|
// FullWeekly creates full weekly schedule with local time zone.
|
|
//
|
|
// TODO(s.chzhen): Consider moving into tests.
|
|
func FullWeekly() (w *Weekly) {
|
|
fullDay := dayRange{start: 0, end: maxDayRange}
|
|
|
|
return &Weekly{
|
|
location: time.Local,
|
|
days: [7]dayRange{
|
|
time.Sunday: fullDay,
|
|
time.Monday: fullDay,
|
|
time.Tuesday: fullDay,
|
|
time.Wednesday: fullDay,
|
|
time.Thursday: fullDay,
|
|
time.Friday: fullDay,
|
|
time.Saturday: fullDay,
|
|
},
|
|
}
|
|
}
|
|
|
|
// Clone returns a deep copy of a weekly.
|
|
func (w *Weekly) Clone() (c *Weekly) {
|
|
if w == nil {
|
|
return nil
|
|
}
|
|
|
|
// NOTE: Do not use time.LoadLocation, because the results will be
|
|
// different on time zone database update.
|
|
return &Weekly{
|
|
location: w.location,
|
|
days: w.days,
|
|
}
|
|
}
|
|
|
|
// Contains returns true if t is within the corresponding day range of the
|
|
// schedule in the schedule's time zone.
|
|
func (w *Weekly) Contains(t time.Time) (ok bool) {
|
|
t = t.In(w.location)
|
|
wd := t.Weekday()
|
|
dr := w.days[wd]
|
|
|
|
// Calculate the offset of the day range.
|
|
//
|
|
// NOTE: Do not use [time.Truncate] since it requires UTC time zone.
|
|
y, m, d := t.Date()
|
|
day := time.Date(y, m, d, 0, 0, 0, 0, w.location)
|
|
offset := t.Sub(day)
|
|
|
|
return dr.contains(offset)
|
|
}
|
|
|
|
// type check
|
|
var _ json.Unmarshaler = (*Weekly)(nil)
|
|
|
|
// UnmarshalJSON implements the [json.Unmarshaler] interface for *Weekly.
|
|
func (w *Weekly) UnmarshalJSON(data []byte) (err error) {
|
|
conf := &weeklyConfigJSON{}
|
|
err = json.Unmarshal(data, conf)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
weekly := Weekly{}
|
|
|
|
weekly.location, err = time.LoadLocation(conf.TimeZone)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
days := []*dayConfigJSON{
|
|
time.Sunday: conf.Sunday,
|
|
time.Monday: conf.Monday,
|
|
time.Tuesday: conf.Tuesday,
|
|
time.Wednesday: conf.Wednesday,
|
|
time.Thursday: conf.Thursday,
|
|
time.Friday: conf.Friday,
|
|
time.Saturday: conf.Saturday,
|
|
}
|
|
for i, d := range days {
|
|
var r dayRange
|
|
|
|
if d != nil {
|
|
r = dayRange{
|
|
start: time.Duration(d.Start),
|
|
end: time.Duration(d.End),
|
|
}
|
|
}
|
|
|
|
err = w.validate(r)
|
|
if err != nil {
|
|
return fmt.Errorf("weekday %s: %w", time.Weekday(i), err)
|
|
}
|
|
|
|
weekly.days[i] = r
|
|
}
|
|
|
|
*w = weekly
|
|
|
|
return nil
|
|
}
|
|
|
|
// type check
|
|
var _ yaml.Unmarshaler = (*Weekly)(nil)
|
|
|
|
// UnmarshalYAML implements the [yaml.Unmarshaler] interface for *Weekly.
|
|
func (w *Weekly) UnmarshalYAML(value *yaml.Node) (err error) {
|
|
conf := &weeklyConfigYAML{}
|
|
|
|
err = value.Decode(conf)
|
|
if err != nil {
|
|
// Don't wrap the error since it's informative enough as is.
|
|
return err
|
|
}
|
|
|
|
weekly := Weekly{}
|
|
|
|
weekly.location, err = time.LoadLocation(conf.TimeZone)
|
|
if err != nil {
|
|
// Don't wrap the error since it's informative enough as is.
|
|
return err
|
|
}
|
|
|
|
days := []dayConfigYAML{
|
|
time.Sunday: conf.Sunday,
|
|
time.Monday: conf.Monday,
|
|
time.Tuesday: conf.Tuesday,
|
|
time.Wednesday: conf.Wednesday,
|
|
time.Thursday: conf.Thursday,
|
|
time.Friday: conf.Friday,
|
|
time.Saturday: conf.Saturday,
|
|
}
|
|
for i, d := range days {
|
|
r := dayRange{
|
|
start: d.Start.Duration,
|
|
end: d.End.Duration,
|
|
}
|
|
|
|
err = w.validate(r)
|
|
if err != nil {
|
|
return fmt.Errorf("weekday %s: %w", time.Weekday(i), err)
|
|
}
|
|
|
|
weekly.days[i] = r
|
|
}
|
|
|
|
*w = weekly
|
|
|
|
return nil
|
|
}
|
|
|
|
// weeklyConfigYAML is the YAML configuration structure of Weekly.
|
|
type weeklyConfigYAML struct {
|
|
// TimeZone is the local time zone.
|
|
TimeZone string `yaml:"time_zone"`
|
|
|
|
// Days of the week.
|
|
|
|
Sunday dayConfigYAML `yaml:"sun,omitempty"`
|
|
Monday dayConfigYAML `yaml:"mon,omitempty"`
|
|
Tuesday dayConfigYAML `yaml:"tue,omitempty"`
|
|
Wednesday dayConfigYAML `yaml:"wed,omitempty"`
|
|
Thursday dayConfigYAML `yaml:"thu,omitempty"`
|
|
Friday dayConfigYAML `yaml:"fri,omitempty"`
|
|
Saturday dayConfigYAML `yaml:"sat,omitempty"`
|
|
}
|
|
|
|
// dayConfigYAML is the YAML configuration structure of dayRange.
|
|
type dayConfigYAML struct {
|
|
Start timeutil.Duration `yaml:"start"`
|
|
End timeutil.Duration `yaml:"end"`
|
|
}
|
|
|
|
// maxDayRange is the maximum value for day range end.
|
|
const maxDayRange = 24 * time.Hour
|
|
|
|
// validate returns the day range rounding errors, if any.
|
|
func (w *Weekly) validate(r dayRange) (err error) {
|
|
defer func() { err = errors.Annotate(err, "bad day range: %w") }()
|
|
|
|
err = r.validate()
|
|
if err != nil {
|
|
// Don't wrap the error since it's informative enough as is.
|
|
return err
|
|
}
|
|
|
|
start := r.start.Truncate(time.Minute)
|
|
end := r.end.Truncate(time.Minute)
|
|
|
|
switch {
|
|
case start != r.start:
|
|
return fmt.Errorf("start %s isn't rounded to minutes", r.start)
|
|
case end != r.end:
|
|
return fmt.Errorf("end %s isn't rounded to minutes", r.end)
|
|
default:
|
|
return nil
|
|
}
|
|
}
|
|
|
|
// type check
|
|
var _ json.Marshaler = (*Weekly)(nil)
|
|
|
|
// MarshalJSON implements the [json.Marshaler] interface for *Weekly.
|
|
func (w *Weekly) MarshalJSON() (data []byte, err error) {
|
|
c := &weeklyConfigJSON{
|
|
TimeZone: w.location.String(),
|
|
Sunday: w.days[time.Sunday].toDayConfigJSON(),
|
|
Monday: w.days[time.Monday].toDayConfigJSON(),
|
|
Tuesday: w.days[time.Tuesday].toDayConfigJSON(),
|
|
Wednesday: w.days[time.Wednesday].toDayConfigJSON(),
|
|
Thursday: w.days[time.Thursday].toDayConfigJSON(),
|
|
Friday: w.days[time.Friday].toDayConfigJSON(),
|
|
Saturday: w.days[time.Saturday].toDayConfigJSON(),
|
|
}
|
|
|
|
return json.Marshal(c)
|
|
}
|
|
|
|
// type check
|
|
var _ yaml.Marshaler = (*Weekly)(nil)
|
|
|
|
// MarshalYAML implements the [yaml.Marshaler] interface for *Weekly.
|
|
func (w *Weekly) MarshalYAML() (v any, err error) {
|
|
return weeklyConfigYAML{
|
|
TimeZone: w.location.String(),
|
|
Sunday: dayConfigYAML{
|
|
Start: timeutil.Duration{Duration: w.days[time.Sunday].start},
|
|
End: timeutil.Duration{Duration: w.days[time.Sunday].end},
|
|
},
|
|
Monday: dayConfigYAML{
|
|
Start: timeutil.Duration{Duration: w.days[time.Monday].start},
|
|
End: timeutil.Duration{Duration: w.days[time.Monday].end},
|
|
},
|
|
Tuesday: dayConfigYAML{
|
|
Start: timeutil.Duration{Duration: w.days[time.Tuesday].start},
|
|
End: timeutil.Duration{Duration: w.days[time.Tuesday].end},
|
|
},
|
|
Wednesday: dayConfigYAML{
|
|
Start: timeutil.Duration{Duration: w.days[time.Wednesday].start},
|
|
End: timeutil.Duration{Duration: w.days[time.Wednesday].end},
|
|
},
|
|
Thursday: dayConfigYAML{
|
|
Start: timeutil.Duration{Duration: w.days[time.Thursday].start},
|
|
End: timeutil.Duration{Duration: w.days[time.Thursday].end},
|
|
},
|
|
Friday: dayConfigYAML{
|
|
Start: timeutil.Duration{Duration: w.days[time.Friday].start},
|
|
End: timeutil.Duration{Duration: w.days[time.Friday].end},
|
|
},
|
|
Saturday: dayConfigYAML{
|
|
Start: timeutil.Duration{Duration: w.days[time.Saturday].start},
|
|
End: timeutil.Duration{Duration: w.days[time.Saturday].end},
|
|
},
|
|
}, nil
|
|
}
|
|
|
|
// dayRange represents a single interval within a day. The interval begins at
|
|
// start and ends before end. That is, it contains a time point T if start <=
|
|
// T < end.
|
|
type dayRange struct {
|
|
// start is an offset from the beginning of the day. It must be greater
|
|
// than or equal to zero and less than 24h.
|
|
start time.Duration
|
|
|
|
// end is an offset from the beginning of the day. It must be greater than
|
|
// or equal to zero and less than or equal to 24h.
|
|
end time.Duration
|
|
}
|
|
|
|
// validate returns the day range validation errors, if any.
|
|
func (r dayRange) validate() (err error) {
|
|
switch {
|
|
case r == dayRange{}:
|
|
return nil
|
|
case r.start < 0:
|
|
return fmt.Errorf("start %s is negative", r.start)
|
|
case r.end < 0:
|
|
return fmt.Errorf("end %s is negative", r.end)
|
|
case r.start >= r.end:
|
|
return fmt.Errorf("start %s is greater or equal to end %s", r.start, r.end)
|
|
case r.start >= maxDayRange:
|
|
return fmt.Errorf("start %s is greater or equal to %s", r.start, maxDayRange)
|
|
case r.end > maxDayRange:
|
|
return fmt.Errorf("end %s is greater than %s", r.end, maxDayRange)
|
|
default:
|
|
return nil
|
|
}
|
|
}
|
|
|
|
// contains returns true if start <= offset < end, where offset is the time
|
|
// duration from the beginning of the day.
|
|
func (r *dayRange) contains(offset time.Duration) (ok bool) {
|
|
return r.start <= offset && offset < r.end
|
|
}
|
|
|
|
// toDayConfigJSON returns nil if the day range is empty, otherwise returns
|
|
// initialized JSON configuration of the day range.
|
|
func (r dayRange) toDayConfigJSON() (j *dayConfigJSON) {
|
|
if (r == dayRange{}) {
|
|
return nil
|
|
}
|
|
|
|
return &dayConfigJSON{
|
|
Start: aghhttp.JSONDuration(r.start),
|
|
End: aghhttp.JSONDuration(r.end),
|
|
}
|
|
}
|
|
|
|
// weeklyConfigJSON is the JSON configuration structure of Weekly.
|
|
type weeklyConfigJSON struct {
|
|
// TimeZone is the local time zone.
|
|
TimeZone string `json:"time_zone"`
|
|
|
|
// Days of the week.
|
|
|
|
Sunday *dayConfigJSON `json:"sun,omitempty"`
|
|
Monday *dayConfigJSON `json:"mon,omitempty"`
|
|
Tuesday *dayConfigJSON `json:"tue,omitempty"`
|
|
Wednesday *dayConfigJSON `json:"wed,omitempty"`
|
|
Thursday *dayConfigJSON `json:"thu,omitempty"`
|
|
Friday *dayConfigJSON `json:"fri,omitempty"`
|
|
Saturday *dayConfigJSON `json:"sat,omitempty"`
|
|
}
|
|
|
|
// dayConfigJSON is the JSON configuration structure of dayRange.
|
|
type dayConfigJSON struct {
|
|
Start aghhttp.JSONDuration `json:"start"`
|
|
End aghhttp.JSONDuration `json:"end"`
|
|
}
|