mirror of
https://github.com/yusing/godoxy.git
synced 2025-05-19 20:32:35 +02:00
221 lines
5.5 KiB
Go
221 lines
5.5 KiB
Go
package strutils
|
|
|
|
import (
|
|
"fmt"
|
|
"math"
|
|
"strconv"
|
|
"time"
|
|
)
|
|
|
|
// AppendDuration appends a duration to a buffer with the following format:
|
|
// - 1 ns
|
|
// - 1 ms
|
|
// - 1 seconds
|
|
// - 1 minutes and 1 seconds
|
|
// - 1 hours, 1 minutes and 1 seconds
|
|
// - 1 days, 1 hours and 1 minutes (ignore seconds if days >= 1)
|
|
func AppendDuration(d time.Duration, buf []byte) []byte {
|
|
if d < 0 {
|
|
buf = append(buf, '-')
|
|
d = -d
|
|
}
|
|
|
|
if d == 0 {
|
|
return append(buf, []byte("0 Seconds")...)
|
|
}
|
|
|
|
switch {
|
|
case d < time.Millisecond:
|
|
buf = strconv.AppendInt(buf, d.Nanoseconds(), 10)
|
|
buf = append(buf, []byte(" ns")...)
|
|
return buf
|
|
case d < time.Second:
|
|
buf = strconv.AppendInt(buf, d.Milliseconds(), 10)
|
|
buf = append(buf, []byte(" ms")...)
|
|
return buf
|
|
}
|
|
|
|
// Get total seconds from duration
|
|
totalSeconds := int64(d.Seconds())
|
|
|
|
// Calculate days, hours, minutes, and seconds
|
|
days := totalSeconds / (24 * 3600)
|
|
hours := (totalSeconds % (24 * 3600)) / 3600
|
|
minutes := (totalSeconds % 3600) / 60
|
|
seconds := totalSeconds % 60
|
|
|
|
idxPartBeg := 0
|
|
if days > 0 {
|
|
buf = strconv.AppendInt(buf, days, 10)
|
|
buf = fmt.Appendf(buf, " day%s, ", Pluralize(days))
|
|
}
|
|
if hours > 0 {
|
|
idxPartBeg = len(buf) - 2
|
|
buf = strconv.AppendInt(buf, hours, 10)
|
|
buf = fmt.Appendf(buf, " hour%s, ", Pluralize(hours))
|
|
}
|
|
if minutes > 0 {
|
|
idxPartBeg = len(buf) - 2
|
|
buf = strconv.AppendInt(buf, minutes, 10)
|
|
buf = fmt.Appendf(buf, " minute%s, ", Pluralize(minutes))
|
|
}
|
|
if seconds > 0 && totalSeconds < 3600 {
|
|
idxPartBeg = len(buf) - 2
|
|
buf = strconv.AppendInt(buf, seconds, 10)
|
|
buf = fmt.Appendf(buf, " second%s, ", Pluralize(seconds))
|
|
}
|
|
// remove last comma and space
|
|
buf = buf[:len(buf)-2]
|
|
if idxPartBeg > 0 && idxPartBeg < len(buf) {
|
|
// replace last part ', ' with ' and ' in-place, alloc-free
|
|
// ', ' is 2 bytes, ' and ' is 5 bytes, so we need to make room for 3 more bytes
|
|
tailLen := len(buf) - (idxPartBeg + 2)
|
|
buf = append(buf, "000"...) // append 3 bytes for ' and '
|
|
copy(buf[idxPartBeg+5:], buf[idxPartBeg+2:idxPartBeg+2+tailLen]) // shift tail right by 3
|
|
copy(buf[idxPartBeg:], " and ") // overwrite ', ' with ' and '
|
|
}
|
|
return buf
|
|
}
|
|
|
|
func FormatDuration(d time.Duration) string {
|
|
return string(AppendDuration(d, nil))
|
|
}
|
|
|
|
func FormatLastSeen(t time.Time) string {
|
|
if t.IsZero() {
|
|
return "never"
|
|
}
|
|
return FormatTime(t)
|
|
}
|
|
|
|
func appendRound(f float64, buf []byte) []byte {
|
|
return strconv.AppendInt(buf, int64(math.Round(f)), 10)
|
|
}
|
|
|
|
func appendFloat(f float64, buf []byte) []byte {
|
|
f = math.Round(f*100) / 100
|
|
if f == 0 {
|
|
return buf
|
|
}
|
|
return strconv.AppendFloat(buf, f, 'f', -1, 64)
|
|
}
|
|
|
|
func AppendTime(t time.Time, buf []byte) []byte {
|
|
if t.IsZero() {
|
|
return append(buf, []byte("never")...)
|
|
}
|
|
return AppendTimeWithReference(t, time.Now(), buf)
|
|
}
|
|
|
|
func FormatTime(t time.Time) string {
|
|
return string(AppendTime(t, nil))
|
|
}
|
|
|
|
func FormatUnixTime(t int64) string {
|
|
return FormatTime(time.Unix(t, 0))
|
|
}
|
|
|
|
func FormatTimeWithReference(t, ref time.Time) string {
|
|
return string(AppendTimeWithReference(t, ref, nil))
|
|
}
|
|
|
|
func AppendTimeWithReference(t, ref time.Time, buf []byte) []byte {
|
|
if t.IsZero() {
|
|
return append(buf, []byte("never")...)
|
|
}
|
|
diff := t.Sub(ref)
|
|
absDiff := diff.Abs()
|
|
switch {
|
|
case absDiff < time.Second:
|
|
return append(buf, []byte("now")...)
|
|
case absDiff < 3*time.Second:
|
|
if diff < 0 {
|
|
return append(buf, []byte("just now")...)
|
|
}
|
|
fallthrough
|
|
case absDiff < 60*time.Second:
|
|
if diff < 0 {
|
|
buf = appendRound(absDiff.Seconds(), buf)
|
|
buf = append(buf, []byte(" seconds ago")...)
|
|
} else {
|
|
buf = append(buf, []byte("in ")...)
|
|
buf = appendRound(absDiff.Seconds(), buf)
|
|
buf = append(buf, []byte(" seconds")...)
|
|
}
|
|
return buf
|
|
case absDiff < 60*time.Minute:
|
|
if diff < 0 {
|
|
buf = appendRound(absDiff.Minutes(), buf)
|
|
buf = append(buf, []byte(" minutes ago")...)
|
|
} else {
|
|
buf = append(buf, []byte("in ")...)
|
|
buf = appendRound(absDiff.Minutes(), buf)
|
|
buf = append(buf, []byte(" minutes")...)
|
|
}
|
|
return buf
|
|
case absDiff < 24*time.Hour:
|
|
if diff < 0 {
|
|
buf = appendRound(absDiff.Hours(), buf)
|
|
buf = append(buf, []byte(" hours ago")...)
|
|
} else {
|
|
buf = append(buf, []byte("in ")...)
|
|
buf = appendRound(absDiff.Hours(), buf)
|
|
buf = append(buf, []byte(" hours")...)
|
|
}
|
|
return buf
|
|
case t.Year() == ref.Year():
|
|
return t.AppendFormat(buf, "01-02 15:04:05")
|
|
default:
|
|
return t.AppendFormat(buf, "2006-01-02 15:04:05")
|
|
}
|
|
}
|
|
|
|
func FormatByteSize[T ~int | ~uint | ~int64 | ~uint64 | ~float64](size T) string {
|
|
return string(AppendByteSize(size, nil))
|
|
}
|
|
|
|
func AppendByteSize[T ~int | ~uint | ~int64 | ~uint64 | ~float64](size T, buf []byte) []byte {
|
|
const (
|
|
_ = (1 << (10 * iota))
|
|
kb
|
|
mb
|
|
gb
|
|
tb
|
|
pb
|
|
)
|
|
switch {
|
|
case size < kb:
|
|
switch any(size).(type) {
|
|
case int, int64:
|
|
buf = strconv.AppendInt(buf, int64(size), 10)
|
|
case uint, uint64:
|
|
buf = strconv.AppendUint(buf, uint64(size), 10)
|
|
case float64:
|
|
buf = appendFloat(float64(size), buf)
|
|
}
|
|
buf = append(buf, []byte(" B")...)
|
|
case size < mb:
|
|
buf = appendFloat(float64(size)/kb, buf)
|
|
buf = append(buf, []byte(" KiB")...)
|
|
case size < gb:
|
|
buf = appendFloat(float64(size)/mb, buf)
|
|
buf = append(buf, []byte(" MiB")...)
|
|
case size < tb:
|
|
buf = appendFloat(float64(size)/gb, buf)
|
|
buf = append(buf, []byte(" GiB")...)
|
|
case size < pb:
|
|
buf = appendFloat(float64(size/gb)/kb, buf)
|
|
buf = append(buf, []byte(" TiB")...)
|
|
default:
|
|
buf = appendFloat(float64(size/tb)/kb, buf)
|
|
buf = append(buf, []byte(" PiB")...)
|
|
}
|
|
return buf
|
|
}
|
|
|
|
func Pluralize(n int64) string {
|
|
if n > 1 {
|
|
return "s"
|
|
}
|
|
return ""
|
|
}
|