ollama/progress/spinner.go

80 lines
1.4 KiB
Go
Raw Normal View History

2023-11-14 16:33:16 -08:00
package progress
import (
"fmt"
"strings"
"sync/atomic"
2023-11-14 16:33:16 -08:00
"time"
)
type Spinner struct {
message atomic.Value
2023-11-14 16:33:16 -08:00
messageWidth int
parts []string
value int
ticker *time.Ticker
started time.Time
stopped time.Time
}
func NewSpinner(message string) *Spinner {
s := &Spinner{
parts: []string{
"⠋", "⠙", "⠹", "⠸", "⠼", "⠴", "⠦", "⠧", "⠇", "⠏",
},
started: time.Now(),
}
s.SetMessage(message)
2023-11-14 16:33:16 -08:00
go s.start()
return s
}
func (s *Spinner) SetMessage(message string) {
s.message.Store(message)
}
2023-11-14 16:33:16 -08:00
func (s *Spinner) String() string {
var sb strings.Builder
if message, ok := s.message.Load().(string); ok && len(message) > 0 {
message := strings.TrimSpace(message)
2023-11-14 16:33:16 -08:00
if s.messageWidth > 0 && len(message) > s.messageWidth {
message = message[:s.messageWidth]
}
fmt.Fprintf(&sb, "%s", message)
2023-11-18 16:23:03 -08:00
if padding := s.messageWidth - sb.Len(); padding > 0 {
sb.WriteString(strings.Repeat(" ", padding))
2023-11-14 16:33:16 -08:00
}
sb.WriteString(" ")
2023-11-14 16:33:16 -08:00
}
if s.stopped.IsZero() {
spinner := s.parts[s.value]
sb.WriteString(spinner)
sb.WriteString(" ")
2023-11-14 16:33:16 -08:00
}
return sb.String()
2023-11-14 16:33:16 -08:00
}
func (s *Spinner) start() {
s.ticker = time.NewTicker(100 * time.Millisecond)
for range s.ticker.C {
s.value = (s.value + 1) % len(s.parts)
if !s.stopped.IsZero() {
return
}
}
}
func (s *Spinner) Stop() {
if s.stopped.IsZero() {
s.stopped = time.Now()
}
}