mirror of
https://github.com/restic/restic.git
synced 2025-10-19 15:43:21 +00:00
187 lines
4.2 KiB
Go
187 lines
4.2 KiB
Go
package ui
|
|
|
|
import (
|
|
"bytes"
|
|
"encoding/json"
|
|
"errors"
|
|
"fmt"
|
|
"math/bits"
|
|
"strconv"
|
|
"time"
|
|
"unicode"
|
|
|
|
"golang.org/x/text/width"
|
|
)
|
|
|
|
func FormatBytes(c uint64) string {
|
|
b := float64(c)
|
|
switch {
|
|
case c >= 1<<40:
|
|
return fmt.Sprintf("%.3f TiB", b/(1<<40))
|
|
case c >= 1<<30:
|
|
return fmt.Sprintf("%.3f GiB", b/(1<<30))
|
|
case c >= 1<<20:
|
|
return fmt.Sprintf("%.3f MiB", b/(1<<20))
|
|
case c >= 1<<10:
|
|
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
|
|
mins := sec / 60
|
|
sec -= mins * 60
|
|
if hours > 0 {
|
|
return fmt.Sprintf("%d:%02d:%02d", hours, mins, sec)
|
|
}
|
|
return fmt.Sprintf("%d:%02d", mins, sec)
|
|
}
|
|
|
|
// 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
|
|
}
|
|
|
|
func ToJSONString(status interface{}) string {
|
|
buf := new(bytes.Buffer)
|
|
err := json.NewEncoder(buf).Encode(status)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
return buf.String()
|
|
}
|
|
|
|
// DisplayWidth returns the number of terminal cells needed to display s
|
|
func DisplayWidth(s string) int {
|
|
width := 0
|
|
for _, r := range s {
|
|
width += displayRuneWidth(r)
|
|
}
|
|
|
|
return width
|
|
}
|
|
|
|
func displayRuneWidth(r rune) int {
|
|
switch width.LookupRune(r).Kind() {
|
|
case width.EastAsianWide, width.EastAsianFullwidth:
|
|
return 2
|
|
case width.EastAsianNarrow, width.EastAsianHalfwidth, width.EastAsianAmbiguous, width.Neutral:
|
|
return 1
|
|
default:
|
|
return 0
|
|
}
|
|
}
|
|
|
|
// Quote lines with funny characters in them, meaning control chars, newlines,
|
|
// tabs, anything else non-printable and invalid UTF-8.
|
|
//
|
|
// This is intended to produce a string that does not mess up the terminal
|
|
// rather than produce an unambiguous quoted string.
|
|
func Quote(line string) string {
|
|
for _, r := range line {
|
|
// The replacement character usually means the input is not UTF-8.
|
|
if r == unicode.ReplacementChar || !unicode.IsPrint(r) {
|
|
return strconv.Quote(line)
|
|
}
|
|
}
|
|
return line
|
|
}
|
|
|
|
// Truncate s to fit in width (number of terminal cells) w.
|
|
// If w is negative, returns the empty string.
|
|
func Truncate(s string, w int) string {
|
|
if len(s) < w {
|
|
// Since the display width of a character is at most 2
|
|
// and all of ASCII (single byte per rune) has width 1,
|
|
// no character takes more bytes to encode than its width.
|
|
return s
|
|
}
|
|
|
|
for i := uint(0); i < uint(len(s)); {
|
|
utfsize := uint(1) // UTF-8 encoding size of first rune in s.
|
|
w--
|
|
|
|
if s[i] > unicode.MaxASCII {
|
|
var wide bool
|
|
if wide, utfsize = wideRune(s[i:]); wide {
|
|
w--
|
|
}
|
|
}
|
|
|
|
if w < 0 {
|
|
return s[:i]
|
|
}
|
|
i += utfsize
|
|
}
|
|
|
|
return s
|
|
}
|
|
|
|
// Guess whether the first rune in s would occupy two terminal cells
|
|
// instead of one. This cannot be determined exactly without knowing
|
|
// the terminal font, so we treat all ambiguous runes as full-width,
|
|
// i.e., two cells.
|
|
func wideRune(s string) (wide bool, utfsize uint) {
|
|
prop, size := width.LookupString(s)
|
|
kind := prop.Kind()
|
|
wide = kind != width.Neutral && kind != width.EastAsianNarrow
|
|
return wide, uint(size)
|
|
}
|