2022-10-21 17:34:14 +02:00
|
|
|
package ui
|
|
|
|
|
|
|
|
import (
|
2023-05-01 12:01:03 +02:00
|
|
|
"bytes"
|
|
|
|
"encoding/json"
|
2023-07-02 20:09:57 +02:00
|
|
|
"errors"
|
2022-10-21 17:34:14 +02:00
|
|
|
"fmt"
|
2023-07-02 20:09:57 +02:00
|
|
|
"math/bits"
|
|
|
|
"strconv"
|
2022-10-21 17:34:14 +02:00
|
|
|
"time"
|
|
|
|
)
|
|
|
|
|
|
|
|
func FormatBytes(c uint64) string {
|
|
|
|
b := float64(c)
|
|
|
|
switch {
|
2022-10-22 10:19:00 +02:00
|
|
|
case c >= 1<<40:
|
2022-10-21 17:34:14 +02:00
|
|
|
return fmt.Sprintf("%.3f TiB", b/(1<<40))
|
2022-10-22 10:19:00 +02:00
|
|
|
case c >= 1<<30:
|
2022-10-21 17:34:14 +02:00
|
|
|
return fmt.Sprintf("%.3f GiB", b/(1<<30))
|
2022-10-22 10:19:00 +02:00
|
|
|
case c >= 1<<20:
|
2022-10-21 17:34:14 +02:00
|
|
|
return fmt.Sprintf("%.3f MiB", b/(1<<20))
|
2022-10-22 10:19:00 +02:00
|
|
|
case c >= 1<<10:
|
2022-10-21 17:34:14 +02:00
|
|
|
return fmt.Sprintf("%.3f KiB", b/(1<<10))
|
|
|
|
default:
|
|
|
|
return fmt.Sprintf("%d B", c)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// FormatPercent formats numerator/denominator as a percentage.
|
|
|
|
func FormatPercent(numerator uint64, denominator uint64) string {
|
|
|
|
if denominator == 0 {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
percent := 100.0 * float64(numerator) / float64(denominator)
|
|
|
|
if percent > 100 {
|
|
|
|
percent = 100
|
|
|
|
}
|
|
|
|
|
|
|
|
return fmt.Sprintf("%3.2f%%", percent)
|
|
|
|
}
|
|
|
|
|
|
|
|
// FormatDuration formats d as FormatSeconds would.
|
|
|
|
func FormatDuration(d time.Duration) string {
|
|
|
|
sec := uint64(d / time.Second)
|
|
|
|
return FormatSeconds(sec)
|
|
|
|
}
|
|
|
|
|
|
|
|
// FormatSeconds formats sec as MM:SS, or HH:MM:SS if sec seconds
|
|
|
|
// is at least an hour.
|
|
|
|
func FormatSeconds(sec uint64) string {
|
|
|
|
hours := sec / 3600
|
|
|
|
sec -= hours * 3600
|
|
|
|
min := sec / 60
|
|
|
|
sec -= min * 60
|
|
|
|
if hours > 0 {
|
|
|
|
return fmt.Sprintf("%d:%02d:%02d", hours, min, sec)
|
|
|
|
}
|
|
|
|
return fmt.Sprintf("%d:%02d", min, sec)
|
|
|
|
}
|
2023-05-01 12:01:03 +02:00
|
|
|
|
2023-07-02 20:09:57 +02:00
|
|
|
// ParseBytes parses a size in bytes from s. It understands the suffixes
|
|
|
|
// B, K, M, G and T for powers of 1024.
|
|
|
|
func ParseBytes(s string) (int64, error) {
|
|
|
|
if s == "" {
|
|
|
|
return 0, errors.New("expected size, got empty string")
|
|
|
|
}
|
|
|
|
|
|
|
|
numStr := s[:len(s)-1]
|
|
|
|
var unit uint64 = 1
|
|
|
|
|
|
|
|
switch s[len(s)-1] {
|
|
|
|
case 'b', 'B':
|
|
|
|
// use initialized values, do nothing here
|
|
|
|
case 'k', 'K':
|
|
|
|
unit = 1024
|
|
|
|
case 'm', 'M':
|
|
|
|
unit = 1024 * 1024
|
|
|
|
case 'g', 'G':
|
|
|
|
unit = 1024 * 1024 * 1024
|
|
|
|
case 't', 'T':
|
|
|
|
unit = 1024 * 1024 * 1024 * 1024
|
|
|
|
default:
|
|
|
|
numStr = s
|
|
|
|
}
|
|
|
|
value, err := strconv.ParseInt(numStr, 10, 64)
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
|
|
|
|
hi, lo := bits.Mul64(uint64(value), unit)
|
|
|
|
value = int64(lo)
|
|
|
|
if hi != 0 || value < 0 {
|
|
|
|
return 0, fmt.Errorf("ParseSize: %q: %w", numStr, strconv.ErrRange)
|
|
|
|
}
|
|
|
|
|
|
|
|
return value, nil
|
|
|
|
}
|
|
|
|
|
2023-05-01 12:01:03 +02:00
|
|
|
func ToJSONString(status interface{}) string {
|
|
|
|
buf := new(bytes.Buffer)
|
|
|
|
err := json.NewEncoder(buf).Encode(status)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
return buf.String()
|
|
|
|
}
|