GoDoxy/internal/utils/strutils/format.go

170 lines
3.7 KiB
Go

package strutils
import (
"fmt"
"math"
"strconv"
"strings"
"time"
"github.com/yusing/go-proxy/internal/utils/strutils/ansi"
)
func FormatDuration(d time.Duration) string {
// 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
// Create a slice to hold parts of the duration
var parts []string
if days > 0 {
parts = append(parts, fmt.Sprintf("%d day%s", days, pluralize(days)))
}
if hours > 0 {
parts = append(parts, fmt.Sprintf("%d hour%s", hours, pluralize(hours)))
}
if minutes > 0 {
parts = append(parts, fmt.Sprintf("%d minute%s", minutes, pluralize(minutes)))
}
if seconds > 0 && totalSeconds < 3600 {
parts = append(parts, fmt.Sprintf("%d second%s", seconds, pluralize(seconds)))
}
// Join the parts with appropriate connectors
if len(parts) == 0 {
return "0 Seconds"
}
if len(parts) == 1 {
return parts[0]
}
return strings.Join(parts[:len(parts)-1], ", ") + " and " + parts[len(parts)-1]
}
func FormatLastSeen(t time.Time) string {
if t.IsZero() {
return "never"
}
return FormatTime(t)
}
func roundString(f float64) string {
return strconv.Itoa(int(math.Round(f)))
}
func FormatTime(t time.Time) string {
if t.IsZero() {
return "never"
}
return FormatTimeWithReference(t, time.Now())
}
func FormatUnixTime(t int64) string {
return FormatTime(time.Unix(t, 0))
}
func FormatTimeWithReference(t, ref time.Time) string {
if t.IsZero() {
return "never"
}
diff := t.Sub(ref)
absDiff := diff.Abs()
switch {
case absDiff < time.Second:
return "now"
case absDiff < 3*time.Second:
if diff < 0 {
return "just now"
}
fallthrough
case absDiff < 60*time.Second:
if diff < 0 {
return roundString(absDiff.Seconds()) + " seconds ago"
}
return "in " + roundString(absDiff.Seconds()) + " seconds"
case absDiff < 60*time.Minute:
if diff < 0 {
return roundString(absDiff.Minutes()) + " minutes ago"
}
return "in " + roundString(absDiff.Minutes()) + " minutes"
case absDiff < 24*time.Hour:
if diff < 0 {
return roundString(absDiff.Hours()) + " hours ago"
}
return "in " + roundString(absDiff.Hours()) + " hours"
case t.Year() == ref.Year():
return t.Format("01-02 15:04:05")
default:
return t.Format("2006-01-02 15:04:05")
}
}
func ParseBool(s string) bool {
switch strings.ToLower(s) {
case "1", "true", "yes", "on":
return true
default:
return false
}
}
func formatFloat(f float64) string {
f = math.Round(f*100) / 100
if f == 0 {
return "0"
}
return strconv.FormatFloat(f, 'f', -1, 64)
}
func FormatByteSize[T ~int64 | ~uint64 | ~float64](size T) (value, unit string) {
const (
_ = (1 << (10 * iota))
kb
mb
gb
tb
pb
)
switch {
case size < kb:
return fmt.Sprintf("%v", size), "B"
case size < mb:
return formatFloat(float64(size) / kb), "KiB"
case size < gb:
return formatFloat(float64(size) / mb), "MiB"
case size < tb:
return formatFloat(float64(size) / gb), "GiB"
case size < pb:
return formatFloat(float64(size/gb) / kb), "TiB" // prevent overflow
default:
return formatFloat(float64(size/tb) / kb), "PiB" // prevent overflow
}
}
func FormatByteSizeWithUnit[T ~int64 | ~uint64 | ~float64](size T) string {
value, unit := FormatByteSize(size)
return value + " " + unit
}
func PortString(port uint16) string {
return strconv.FormatUint(uint64(port), 10)
}
func DoYouMean(s string) string {
if s == "" {
return ""
}
return "Did you mean " + ansi.HighlightGreen + s + ansi.Reset + "?"
}
func pluralize(n int64) string {
if n > 1 {
return "s"
}
return ""
}