2023-11-14 16:33:16 -08:00
|
|
|
package progress
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2023-11-19 11:00:43 -05:00
|
|
|
"math"
|
2023-11-14 16:33:16 -08:00
|
|
|
"os"
|
|
|
|
"strings"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/jmorganca/ollama/format"
|
|
|
|
"golang.org/x/term"
|
|
|
|
)
|
|
|
|
|
|
|
|
type Bar struct {
|
|
|
|
message string
|
|
|
|
messageWidth int
|
|
|
|
|
|
|
|
maxValue int64
|
|
|
|
initialValue int64
|
|
|
|
currentValue int64
|
|
|
|
|
|
|
|
started time.Time
|
|
|
|
stopped time.Time
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewBar(message string, maxValue, initialValue int64) *Bar {
|
|
|
|
return &Bar{
|
|
|
|
message: message,
|
|
|
|
messageWidth: -1,
|
|
|
|
maxValue: maxValue,
|
|
|
|
initialValue: initialValue,
|
|
|
|
currentValue: initialValue,
|
|
|
|
started: time.Now(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *Bar) String() string {
|
|
|
|
termWidth, _, err := term.GetSize(int(os.Stderr.Fd()))
|
|
|
|
if err != nil {
|
2023-11-19 23:50:45 -05:00
|
|
|
termWidth = 80
|
2023-11-14 16:33:16 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
var pre, mid, suf strings.Builder
|
|
|
|
|
|
|
|
if b.message != "" {
|
|
|
|
message := strings.TrimSpace(b.message)
|
|
|
|
if b.messageWidth > 0 && len(message) > b.messageWidth {
|
|
|
|
message = message[:b.messageWidth]
|
|
|
|
}
|
|
|
|
|
|
|
|
fmt.Fprintf(&pre, "%s", message)
|
|
|
|
if b.messageWidth-pre.Len() >= 0 {
|
|
|
|
pre.WriteString(strings.Repeat(" ", b.messageWidth-pre.Len()))
|
|
|
|
}
|
|
|
|
|
|
|
|
pre.WriteString(" ")
|
|
|
|
}
|
|
|
|
|
2023-11-19 11:00:43 -05:00
|
|
|
fmt.Fprintf(&pre, "%3.0f%% ", math.Floor(b.percent()))
|
2023-11-14 16:33:16 -08:00
|
|
|
|
|
|
|
fmt.Fprintf(&suf, "(%s/%s, %s/s, %s)",
|
|
|
|
format.HumanBytes(b.currentValue),
|
|
|
|
format.HumanBytes(b.maxValue),
|
|
|
|
format.HumanBytes(int64(b.rate())),
|
|
|
|
b.elapsed())
|
|
|
|
|
2023-11-19 09:54:33 -05:00
|
|
|
mid.WriteString("▕")
|
2023-11-14 16:33:16 -08:00
|
|
|
|
2023-11-19 13:43:21 -05:00
|
|
|
// add 3 extra spaces: 2 boundary characters and 1 space at the end
|
|
|
|
f := termWidth - pre.Len() - suf.Len() - 3
|
2023-11-14 16:33:16 -08:00
|
|
|
n := int(float64(f) * b.percent() / 100)
|
|
|
|
|
2023-11-19 09:54:33 -05:00
|
|
|
if n > 0 {
|
|
|
|
mid.WriteString(strings.Repeat("█", n))
|
2023-11-14 16:33:16 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
if f-n > 0 {
|
|
|
|
mid.WriteString(strings.Repeat(" ", f-n))
|
|
|
|
}
|
|
|
|
|
2023-11-19 09:54:33 -05:00
|
|
|
mid.WriteString("▏")
|
2023-11-14 16:33:16 -08:00
|
|
|
|
|
|
|
return pre.String() + mid.String() + suf.String()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *Bar) Set(value int64) {
|
|
|
|
if value >= b.maxValue {
|
|
|
|
value = b.maxValue
|
|
|
|
b.stopped = time.Now()
|
|
|
|
}
|
|
|
|
|
|
|
|
b.currentValue = value
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *Bar) percent() float64 {
|
|
|
|
if b.maxValue > 0 {
|
|
|
|
return float64(b.currentValue) / float64(b.maxValue) * 100
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *Bar) rate() float64 {
|
2023-11-15 14:52:21 -08:00
|
|
|
elapsed := b.elapsed()
|
|
|
|
if elapsed.Seconds() > 0 {
|
|
|
|
return (float64(b.currentValue) - float64(b.initialValue)) / elapsed.Seconds()
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0
|
2023-11-14 16:33:16 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
func (b *Bar) elapsed() time.Duration {
|
|
|
|
stopped := b.stopped
|
|
|
|
if stopped.IsZero() {
|
|
|
|
stopped = time.Now()
|
|
|
|
}
|
|
|
|
|
|
|
|
return stopped.Sub(b.started).Round(time.Second)
|
|
|
|
}
|