2018-01-12 18:30:54 +02:00
|
|
|
package fs
|
|
|
|
|
|
|
|
import (
|
2024-03-13 20:08:59 +02:00
|
|
|
"encoding/json"
|
2018-05-14 17:32:27 +02:00
|
|
|
"fmt"
|
|
|
|
"math"
|
2018-01-12 18:30:54 +02:00
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
"time"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Duration is a time.Duration with some more parsing options
|
|
|
|
type Duration time.Duration
|
|
|
|
|
2018-03-12 22:52:42 +02:00
|
|
|
// DurationOff is the default value for flags which can be turned off
|
|
|
|
const DurationOff = Duration((1 << 63) - 1)
|
|
|
|
|
2018-01-12 18:30:54 +02:00
|
|
|
// Turn Duration into a string
|
|
|
|
func (d Duration) String() string {
|
2018-03-12 22:52:42 +02:00
|
|
|
if d == DurationOff {
|
|
|
|
return "off"
|
|
|
|
}
|
2018-05-14 17:32:27 +02:00
|
|
|
for i := len(ageSuffixes) - 2; i >= 0; i-- {
|
|
|
|
ageSuffix := &ageSuffixes[i]
|
|
|
|
if math.Abs(float64(d)) >= float64(ageSuffix.Multiplier) {
|
|
|
|
timeUnits := float64(d) / float64(ageSuffix.Multiplier)
|
|
|
|
return strconv.FormatFloat(timeUnits, 'f', -1, 64) + ageSuffix.Suffix
|
|
|
|
}
|
|
|
|
}
|
2018-01-12 18:30:54 +02:00
|
|
|
return time.Duration(d).String()
|
|
|
|
}
|
|
|
|
|
2018-03-12 22:52:42 +02:00
|
|
|
// IsSet returns if the duration is != DurationOff
|
|
|
|
func (d Duration) IsSet() bool {
|
|
|
|
return d != DurationOff
|
|
|
|
}
|
|
|
|
|
2018-01-12 18:30:54 +02:00
|
|
|
// We use time conventions
|
|
|
|
var ageSuffixes = []struct {
|
|
|
|
Suffix string
|
|
|
|
Multiplier time.Duration
|
|
|
|
}{
|
|
|
|
{Suffix: "d", Multiplier: time.Hour * 24},
|
|
|
|
{Suffix: "w", Multiplier: time.Hour * 24 * 7},
|
|
|
|
{Suffix: "M", Multiplier: time.Hour * 24 * 30},
|
|
|
|
{Suffix: "y", Multiplier: time.Hour * 24 * 365},
|
|
|
|
|
|
|
|
// Default to second
|
|
|
|
{Suffix: "", Multiplier: time.Second},
|
|
|
|
}
|
|
|
|
|
2020-05-11 14:25:39 +02:00
|
|
|
// parse the age as suffixed ages
|
|
|
|
func parseDurationSuffixes(age string) (time.Duration, error) {
|
2018-01-12 18:30:54 +02:00
|
|
|
var period float64
|
|
|
|
|
|
|
|
for _, ageSuffix := range ageSuffixes {
|
|
|
|
if strings.HasSuffix(age, ageSuffix.Suffix) {
|
|
|
|
numberString := age[:len(age)-len(ageSuffix.Suffix)]
|
|
|
|
var err error
|
|
|
|
period, err = strconv.ParseFloat(numberString, 64)
|
|
|
|
if err != nil {
|
|
|
|
return time.Duration(0), err
|
|
|
|
}
|
|
|
|
period *= float64(ageSuffix.Multiplier)
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return time.Duration(period), nil
|
|
|
|
}
|
|
|
|
|
2020-05-11 14:25:39 +02:00
|
|
|
// time formats to try parsing ages as - in order
|
|
|
|
var timeFormats = []string{
|
|
|
|
time.RFC3339,
|
|
|
|
"2006-01-02T15:04:05",
|
|
|
|
"2006-01-02 15:04:05",
|
|
|
|
"2006-01-02",
|
|
|
|
}
|
|
|
|
|
2022-03-26 20:49:04 +02:00
|
|
|
// parse the date as time in various date formats
|
|
|
|
func parseTimeDates(date string) (t time.Time, err error) {
|
2020-05-11 14:25:39 +02:00
|
|
|
var instant time.Time
|
|
|
|
for _, timeFormat := range timeFormats {
|
2022-03-26 20:49:04 +02:00
|
|
|
instant, err = time.ParseInLocation(timeFormat, date, time.Local)
|
2020-05-11 14:25:39 +02:00
|
|
|
if err == nil {
|
2022-03-26 20:49:04 +02:00
|
|
|
return instant, nil
|
2020-05-11 14:25:39 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return t, err
|
|
|
|
}
|
|
|
|
|
2022-03-26 20:49:04 +02:00
|
|
|
// parse the age as time before the epoch in various date formats
|
|
|
|
func parseDurationDates(age string, epoch time.Time) (d time.Duration, err error) {
|
|
|
|
instant, err := parseTimeDates(age)
|
|
|
|
if err != nil {
|
|
|
|
return d, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return epoch.Sub(instant), nil
|
|
|
|
}
|
|
|
|
|
2020-09-07 01:50:16 +02:00
|
|
|
// parseDurationFromNow parses a duration string. Allows ParseDuration to match the time
|
|
|
|
// package and easier testing within the fs package.
|
|
|
|
func parseDurationFromNow(age string, getNow func() time.Time) (d time.Duration, err error) {
|
2020-05-11 14:25:39 +02:00
|
|
|
if age == "off" {
|
|
|
|
return time.Duration(DurationOff), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Attempt to parse as a time.Duration first
|
|
|
|
d, err = time.ParseDuration(age)
|
|
|
|
if err == nil {
|
|
|
|
return d, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
d, err = parseDurationSuffixes(age)
|
|
|
|
if err == nil {
|
|
|
|
return d, nil
|
|
|
|
}
|
|
|
|
|
2020-09-07 01:50:16 +02:00
|
|
|
d, err = parseDurationDates(age, getNow())
|
2020-05-11 14:25:39 +02:00
|
|
|
if err == nil {
|
|
|
|
return d, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return d, err
|
|
|
|
}
|
|
|
|
|
2020-09-07 01:50:16 +02:00
|
|
|
// ParseDuration parses a duration string. Accept ms|s|m|h|d|w|M|y suffixes. Defaults to second if not provided
|
|
|
|
func ParseDuration(age string) (time.Duration, error) {
|
2022-08-05 16:23:44 +02:00
|
|
|
return parseDurationFromNow(age, timeNowFunc)
|
2020-09-07 01:50:16 +02:00
|
|
|
}
|
|
|
|
|
2022-08-19 11:40:43 +02:00
|
|
|
// ReadableString parses d into a human-readable duration with units.
|
|
|
|
// Examples: "3s", "1d2h23m20s", "292y24w3d23h47m16s".
|
2019-07-01 13:09:19 +02:00
|
|
|
func (d Duration) ReadableString() string {
|
2022-08-19 11:40:43 +02:00
|
|
|
return d.readableString(0)
|
|
|
|
}
|
|
|
|
|
|
|
|
// ShortReadableString parses d into a human-readable duration with units.
|
|
|
|
// This method returns it in short format, including the 3 most significant
|
|
|
|
// units only, sacrificing precision if necessary. E.g. returns "292y24w3d"
|
|
|
|
// instead of "292y24w3d23h47m16s", and "3d23h47m" instead of "3d23h47m16s".
|
|
|
|
func (d Duration) ShortReadableString() string {
|
|
|
|
return d.readableString(3)
|
|
|
|
}
|
|
|
|
|
|
|
|
// readableString parses d into a human-readable duration with units.
|
|
|
|
// Parameter maxNumberOfUnits limits number of significant units to include,
|
|
|
|
// sacrificing precision. E.g. with argument 3 it returns "292y24w3d" instead
|
|
|
|
// of "292y24w3d23h47m16s", and "3d23h47m" instead of "3d23h47m16s". Zero or
|
|
|
|
// negative argument means include all.
|
|
|
|
// Based on https://github.com/hako/durafmt
|
|
|
|
func (d Duration) readableString(maxNumberOfUnits int) string {
|
2019-07-01 13:09:19 +02:00
|
|
|
switch d {
|
|
|
|
case DurationOff:
|
|
|
|
return "off"
|
|
|
|
case 0:
|
|
|
|
return "0s"
|
|
|
|
}
|
|
|
|
|
|
|
|
readableString := ""
|
|
|
|
|
|
|
|
// Check for minus durations.
|
|
|
|
if d < 0 {
|
|
|
|
readableString += "-"
|
|
|
|
}
|
|
|
|
|
|
|
|
duration := time.Duration(math.Abs(float64(d)))
|
|
|
|
|
|
|
|
// Convert duration.
|
|
|
|
seconds := int64(duration.Seconds()) % 60
|
|
|
|
minutes := int64(duration.Minutes()) % 60
|
|
|
|
hours := int64(duration.Hours()) % 24
|
|
|
|
days := int64(duration/(24*time.Hour)) % 365 % 7
|
|
|
|
|
|
|
|
// Edge case between 364 and 365 days.
|
|
|
|
// We need to calculate weeks from what is left from years
|
|
|
|
leftYearDays := int64(duration/(24*time.Hour)) % 365
|
|
|
|
weeks := leftYearDays / 7
|
|
|
|
if leftYearDays >= 364 && leftYearDays < 365 {
|
|
|
|
weeks = 52
|
|
|
|
}
|
|
|
|
|
|
|
|
years := int64(duration/(24*time.Hour)) / 365
|
|
|
|
milliseconds := int64(duration/time.Millisecond) -
|
|
|
|
(seconds * 1000) - (minutes * 60000) - (hours * 3600000) -
|
|
|
|
(days * 86400000) - (weeks * 604800000) - (years * 31536000000)
|
|
|
|
|
|
|
|
// Create a map of the converted duration time.
|
|
|
|
durationMap := map[string]int64{
|
|
|
|
"ms": milliseconds,
|
|
|
|
"s": seconds,
|
|
|
|
"m": minutes,
|
|
|
|
"h": hours,
|
|
|
|
"d": days,
|
|
|
|
"w": weeks,
|
|
|
|
"y": years,
|
|
|
|
}
|
|
|
|
|
|
|
|
// Construct duration string.
|
2022-08-19 11:40:43 +02:00
|
|
|
numberOfUnits := 0
|
2019-07-01 13:09:19 +02:00
|
|
|
for _, u := range [...]string{"y", "w", "d", "h", "m", "s", "ms"} {
|
|
|
|
v := durationMap[u]
|
|
|
|
strval := strconv.FormatInt(v, 10)
|
|
|
|
if v == 0 {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
readableString += strval + u
|
2022-08-19 11:40:43 +02:00
|
|
|
numberOfUnits++
|
|
|
|
if maxNumberOfUnits > 0 && numberOfUnits >= maxNumberOfUnits {
|
|
|
|
break
|
|
|
|
}
|
2019-07-01 13:09:19 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return readableString
|
|
|
|
}
|
|
|
|
|
2018-01-12 18:30:54 +02:00
|
|
|
// Set a Duration
|
|
|
|
func (d *Duration) Set(s string) error {
|
|
|
|
duration, err := ParseDuration(s)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
*d = Duration(duration)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Type of the value
|
|
|
|
func (d Duration) Type() string {
|
2019-02-07 13:57:26 +02:00
|
|
|
return "Duration"
|
2018-01-12 18:30:54 +02:00
|
|
|
}
|
2018-05-14 17:32:27 +02:00
|
|
|
|
2020-12-11 19:48:09 +02:00
|
|
|
// UnmarshalJSON makes sure the value can be parsed as a string or integer in JSON
|
|
|
|
func (d *Duration) UnmarshalJSON(in []byte) error {
|
2024-03-13 20:08:59 +02:00
|
|
|
// Check if the input is a string value.
|
|
|
|
if len(in) >= 2 && in[0] == '"' && in[len(in)-1] == '"' {
|
|
|
|
strVal := string(in[1 : len(in)-1]) // Remove the quotes
|
|
|
|
|
|
|
|
// Attempt to parse the string as a duration.
|
|
|
|
parsedDuration, err := ParseDuration(strVal)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
*d = Duration(parsedDuration)
|
2020-12-11 19:48:09 +02:00
|
|
|
return nil
|
2024-03-13 20:08:59 +02:00
|
|
|
}
|
|
|
|
// Handle numeric values.
|
|
|
|
var i int64
|
|
|
|
err := json.Unmarshal(in, &i)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
*d = Duration(i)
|
|
|
|
return nil
|
2020-12-11 19:48:09 +02:00
|
|
|
}
|
|
|
|
|
2018-05-14 17:32:27 +02:00
|
|
|
// Scan implements the fmt.Scanner interface
|
|
|
|
func (d *Duration) Scan(s fmt.ScanState, ch rune) error {
|
2022-08-05 16:23:44 +02:00
|
|
|
token, err := s.Token(true, func(rune) bool { return true })
|
2018-05-14 17:32:27 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return d.Set(string(token))
|
|
|
|
}
|