Some checks failed
Pipeline: Test, Lint, Build / Get version info (push) Has been cancelled
Pipeline: Test, Lint, Build / Lint Go code (push) Has been cancelled
Pipeline: Test, Lint, Build / Test Go code (push) Has been cancelled
Pipeline: Test, Lint, Build / Test JS code (push) Has been cancelled
Pipeline: Test, Lint, Build / Lint i18n files (push) Has been cancelled
Pipeline: Test, Lint, Build / Check Docker configuration (push) Has been cancelled
Pipeline: Test, Lint, Build / Build (darwin/amd64) (push) Has been cancelled
Pipeline: Test, Lint, Build / Build (darwin/arm64) (push) Has been cancelled
Pipeline: Test, Lint, Build / Build (linux/386) (push) Has been cancelled
Pipeline: Test, Lint, Build / Build (linux/amd64) (push) Has been cancelled
Pipeline: Test, Lint, Build / Build (linux/arm/v5) (push) Has been cancelled
Pipeline: Test, Lint, Build / Build (linux/arm/v6) (push) Has been cancelled
Pipeline: Test, Lint, Build / Build (linux/arm/v7) (push) Has been cancelled
Pipeline: Test, Lint, Build / Build (linux/arm64) (push) Has been cancelled
Pipeline: Test, Lint, Build / Build (windows/386) (push) Has been cancelled
Pipeline: Test, Lint, Build / Build (windows/amd64) (push) Has been cancelled
Pipeline: Test, Lint, Build / Push to GHCR (push) Has been cancelled
Pipeline: Test, Lint, Build / Push to Docker Hub (push) Has been cancelled
Pipeline: Test, Lint, Build / Cleanup digest artifacts (push) Has been cancelled
Pipeline: Test, Lint, Build / Build Windows installers (push) Has been cancelled
Pipeline: Test, Lint, Build / Package/Release (push) Has been cancelled
Pipeline: Test, Lint, Build / Upload Linux PKG (push) Has been cancelled
Close stale issues and PRs / stale (push) Has been cancelled
POEditor import / update-translations (push) Has been cancelled
75 lines
1.4 KiB
Go
75 lines
1.4 KiB
Go
package log
|
|
|
|
import (
|
|
"fmt"
|
|
"io"
|
|
"iter"
|
|
"reflect"
|
|
"slices"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/navidrome/navidrome/utils/slice"
|
|
)
|
|
|
|
func ShortDur(d time.Duration) string {
|
|
var s string
|
|
switch {
|
|
case d > time.Hour:
|
|
s = d.Round(time.Minute).String()
|
|
case d > time.Minute:
|
|
s = d.Round(time.Second).String()
|
|
case d > time.Second:
|
|
s = d.Round(10 * time.Millisecond).String()
|
|
case d > time.Millisecond:
|
|
s = d.Round(100 * time.Microsecond).String()
|
|
default:
|
|
s = d.String()
|
|
}
|
|
s = strings.TrimSuffix(s, "0s")
|
|
return strings.TrimSuffix(s, "0m")
|
|
}
|
|
|
|
func StringerValue(s fmt.Stringer) string {
|
|
v := reflect.ValueOf(s)
|
|
if v.Kind() == reflect.Pointer && v.IsNil() {
|
|
return "nil"
|
|
}
|
|
return s.String()
|
|
}
|
|
|
|
func formatSeq[T any](v iter.Seq[T]) string {
|
|
return formatSlice(slices.Collect(v))
|
|
}
|
|
|
|
func formatSlice[T any](v []T) string {
|
|
s := slice.Map(v, func(x T) string { return fmt.Sprintf("%v", x) })
|
|
return fmt.Sprintf("[`%s`]", strings.Join(s, "`,`"))
|
|
}
|
|
|
|
func CRLFWriter(w io.Writer) io.Writer {
|
|
return &crlfWriter{w: w}
|
|
}
|
|
|
|
type crlfWriter struct {
|
|
w io.Writer
|
|
lastByte byte
|
|
}
|
|
|
|
func (cw *crlfWriter) Write(p []byte) (int, error) {
|
|
var written int
|
|
for _, b := range p {
|
|
if b == '\n' && cw.lastByte != '\r' {
|
|
if _, err := cw.w.Write([]byte{'\r'}); err != nil {
|
|
return written, err
|
|
}
|
|
}
|
|
if _, err := cw.w.Write([]byte{b}); err != nil {
|
|
return written, err
|
|
}
|
|
written++
|
|
cw.lastByte = b
|
|
}
|
|
return written, nil
|
|
}
|