141 lines
3.5 KiB
Go
141 lines
3.5 KiB
Go
package provisioner
|
|
|
|
import (
|
|
"encoding/json"
|
|
"time"
|
|
|
|
"github.com/pkg/errors"
|
|
)
|
|
|
|
var now = func() time.Time {
|
|
return time.Now().UTC()
|
|
}
|
|
|
|
// TimeDuration is a type that represents a time but the JSON unmarshaling can
|
|
// use a time using the RFC 3339 format or a time.Duration string. If a duration
|
|
// is used, the time will be set on the first call to TimeDuration.Time.
|
|
type TimeDuration struct {
|
|
t time.Time
|
|
d time.Duration
|
|
}
|
|
|
|
// NewTimeDuration returns a TimeDuration with the defined time.
|
|
func NewTimeDuration(t time.Time) TimeDuration {
|
|
return TimeDuration{t: t}
|
|
}
|
|
|
|
// ParseTimeDuration returns a new TimeDuration parsing the RFC 3339 time or
|
|
// time.Duration string.
|
|
func ParseTimeDuration(s string) (TimeDuration, error) {
|
|
if s == "" {
|
|
return TimeDuration{}, nil
|
|
}
|
|
|
|
// Try to use the unquoted RFC 3339 format
|
|
var t time.Time
|
|
if err := t.UnmarshalText([]byte(s)); err == nil {
|
|
return TimeDuration{t: t.UTC()}, nil
|
|
}
|
|
|
|
// Try to use the time.Duration string format
|
|
if d, err := time.ParseDuration(s); err == nil {
|
|
return TimeDuration{d: d}, nil
|
|
}
|
|
|
|
return TimeDuration{}, errors.Errorf("failed to parse %s", s)
|
|
}
|
|
|
|
// SetDuration initializes the TimeDuration with the given duration string. If
|
|
// the time was set it will re-set to zero.
|
|
func (t *TimeDuration) SetDuration(d time.Duration) {
|
|
t.t, t.d = time.Time{}, d
|
|
}
|
|
|
|
// SetTime initializes the TimeDuration with the given time. If the duration is
|
|
// set it will be re-set to zero.
|
|
func (t *TimeDuration) SetTime(tt time.Time) {
|
|
t.t, t.d = tt, 0
|
|
}
|
|
|
|
// IsZero returns true the TimeDuration represents the zero value, false
|
|
// otherwise.
|
|
func (t *TimeDuration) IsZero() bool {
|
|
return t.t.IsZero() && t.d == 0
|
|
}
|
|
|
|
// Equal returns if t and other are equal.
|
|
func (t *TimeDuration) Equal(other *TimeDuration) bool {
|
|
return t.t.Equal(other.t) && t.d == other.d
|
|
}
|
|
|
|
// MarshalJSON implements the json.Marshaler interface. If the time is set it
|
|
// will return the time in RFC 3339 format if not it will return the duration
|
|
// string.
|
|
func (t TimeDuration) MarshalJSON() ([]byte, error) {
|
|
switch {
|
|
case t.t.IsZero():
|
|
if t.d == 0 {
|
|
return []byte("null"), nil
|
|
}
|
|
return json.Marshal(t.d.String())
|
|
default:
|
|
return t.t.MarshalJSON()
|
|
}
|
|
}
|
|
|
|
// UnmarshalJSON implements the json.Unmarshaler interface. The time is expected
|
|
// to be a quoted string in RFC 3339 format or a quoted time.Duration string.
|
|
func (t *TimeDuration) UnmarshalJSON(data []byte) error {
|
|
var s string
|
|
if err := json.Unmarshal(data, &s); err != nil {
|
|
return errors.Wrapf(err, "error unmarshaling %s", data)
|
|
}
|
|
|
|
// Empty TimeDuration
|
|
if s == "" {
|
|
*t = TimeDuration{}
|
|
return nil
|
|
}
|
|
|
|
// Try to use the unquoted RFC 3339 format
|
|
var tt time.Time
|
|
if err := tt.UnmarshalText([]byte(s)); err == nil {
|
|
*t = TimeDuration{t: tt}
|
|
return nil
|
|
}
|
|
|
|
// Try to use the time.Duration string format
|
|
if d, err := time.ParseDuration(s); err == nil {
|
|
*t = TimeDuration{d: d}
|
|
return nil
|
|
}
|
|
|
|
return errors.Errorf("failed to parse %s", data)
|
|
}
|
|
|
|
// Time calculates the embedded time.Time, sets it if necessary, and returns it.
|
|
func (t *TimeDuration) Time() time.Time {
|
|
return t.RelativeTime(now())
|
|
}
|
|
|
|
// RelativeTime returns the embedded time.Time or the base time plus the
|
|
// duration if this is not zero.
|
|
func (t *TimeDuration) RelativeTime(base time.Time) time.Time {
|
|
switch {
|
|
case t == nil:
|
|
return time.Time{}
|
|
case t.t.IsZero():
|
|
if t.d == 0 {
|
|
return time.Time{}
|
|
}
|
|
t.t = base.Add(t.d)
|
|
return t.t.UTC()
|
|
default:
|
|
return t.t.UTC()
|
|
}
|
|
}
|
|
|
|
// String implements the fmt.Stringer interface.
|
|
func (t *TimeDuration) String() string {
|
|
return t.Time().String()
|
|
}
|