1
0
Fork 0
mirror of https://github.com/diamondburned/arikawa.git synced 2024-11-27 09:12:53 +00:00
arikawa/discord/time.go
2021-10-30 16:43:28 -07:00

194 lines
4.2 KiB
Go

package discord
import (
"encoding/json"
"strconv"
"strings"
"time"
)
// Timestamp has a valid zero-value, which can be checked using the IsValid()
// method. This is useful for optional timestamps such as EditedTimestamp.
type Timestamp time.Time
const TimestampFormat = time.RFC3339 // same as ISO8601
var (
_ json.Unmarshaler = (*Timestamp)(nil)
_ json.Marshaler = (*Timestamp)(nil)
)
func NewTimestamp(t time.Time) Timestamp {
return Timestamp(t)
}
func NowTimestamp() Timestamp {
return NewTimestamp(time.Now())
}
// UnmarshalJSON parses a nullable RFC3339 string into time.
func (t *Timestamp) UnmarshalJSON(v []byte) error {
str := strings.Trim(string(v), `"`)
if str == "null" {
*t = Timestamp{}
return nil
}
r, err := time.Parse(TimestampFormat, str)
if err != nil {
return err
}
*t = Timestamp(r)
return nil
}
// MarshalJSON returns null if Timestamp is not valid (zero). It returns the
// time formatted in RFC3339 otherwise.
func (t Timestamp) MarshalJSON() ([]byte, error) {
if !t.IsValid() {
return []byte("null"), nil
}
return []byte(`"` + t.Format(TimestampFormat) + `"`), nil
}
func (t Timestamp) IsValid() bool {
return !t.Time().IsZero()
}
func (t Timestamp) Format(fmt string) string {
return t.Time().Format(fmt)
}
func (t Timestamp) Time() time.Time {
return time.Time(t)
}
//
type UnixTimestamp int64
func (t UnixTimestamp) String() string {
return t.Time().String()
}
func (t UnixTimestamp) Time() time.Time {
return time.Unix(int64(t), 0)
}
//
type UnixMsTimestamp int64
func TimeToMilliseconds(t time.Time) UnixMsTimestamp {
return UnixMsTimestamp(t.UnixNano() / int64(time.Millisecond))
}
func (t UnixMsTimestamp) String() string {
return t.Time().String()
}
func (t UnixMsTimestamp) Time() time.Time {
return time.Unix(0, int64(t)*int64(time.Millisecond))
}
//
type Seconds int
// NullSecond is used in cases where null should be used instead of a number or
// omitted. This is similar to NullSnowflake.
const NullSecond = -1
func DurationToSeconds(dura time.Duration) Seconds {
return Seconds(dura.Seconds())
}
func (s Seconds) MarshalJSON() ([]byte, error) {
if s < 1 {
return []byte("null"), nil
}
return []byte(strconv.Itoa(int(s))), nil
}
func (s *Seconds) UnmarshalJSON(data []byte) error {
if string(data) == "null" {
*s = NullSecond
return nil
}
return json.Unmarshal(data, (*int)(s))
}
func (s Seconds) String() string {
return s.Duration().String()
}
func (s Seconds) Duration() time.Duration {
return time.Duration(s) * time.Second
}
//
// OptionalSeconds is the option type for Seconds.
type OptionalSeconds = *Seconds
// ZeroOptionalSeconds are 0 OptionalSeconds.
var ZeroOptionalSeconds = NewOptionalSeconds(0)
// NewOptionalSeconds creates a new OptionalSeconds using the value of the
// passed Seconds.
func NewOptionalSeconds(s Seconds) OptionalSeconds { return &s }
//
// Milliseconds is in float64 because some Discord events return time with a
// trailing decimal.
type Milliseconds float64
func DurationToMilliseconds(dura time.Duration) Milliseconds {
return Milliseconds(dura.Milliseconds())
}
func (ms Milliseconds) String() string {
return ms.Duration().String()
}
func (ms Milliseconds) Duration() time.Duration {
const f64ms = Milliseconds(time.Millisecond)
return time.Duration(ms * f64ms)
}
//
// ArchiveDuration is the duration after which a thread without activity will
// be archived.
//
// The duration's unit is minutes.
type ArchiveDuration int
const (
OneHourArchive ArchiveDuration = 60
OneDayArchive ArchiveDuration = 24 * OneHourArchive
// ThreeDaysArchive archives a thread after three days.
//
// This duration is only available to nitro boosted guilds. The Features
// field of a Guild will indicate whether this is the case.
ThreeDaysArchive ArchiveDuration = 3 * OneDayArchive
// SevenDaysArchive archives a thread after seven days.
//
// This duration is only available to nitro boosted guilds. The Features
// field of a Guild will indicate whether this is the case.
SevenDaysArchive ArchiveDuration = 7 * OneDayArchive
)
func (m ArchiveDuration) String() string {
return m.Duration().String()
}
func (m ArchiveDuration) Duration() time.Duration {
return time.Duration(m) * time.Minute
}