ollama/progress/progress.go
2025-02-17 20:01:03 -08:00

120 lines
1.9 KiB
Go

package progress
import (
"bytes"
"fmt"
"io"
"sync"
"time"
)
type State interface {
String() string
}
type Progress struct {
mu sync.Mutex
w io.Writer
buf bytes.Buffer
pos int
ticker *time.Ticker
states []State
}
func NewProgress(w io.Writer) *Progress {
p := &Progress{w: w}
go p.start()
return p
}
func (p *Progress) stop() bool {
for _, state := range p.states {
if spinner, ok := state.(*Spinner); ok {
spinner.Stop()
}
}
if p.ticker != nil {
p.ticker.Stop()
p.ticker = nil
p.render()
return true
}
return false
}
func (p *Progress) Stop() bool {
stopped := p.stop()
if stopped {
fmt.Fprint(p.w, "\n")
}
return stopped
}
func (p *Progress) StopAndClear() bool {
fmt.Fprint(p.w, "\033[?25l")
defer fmt.Fprint(p.w, "\033[?25h")
stopped := p.stop()
if stopped {
// clear all progress lines
for i := range p.pos {
if i > 0 {
fmt.Fprint(p.w, "\033[A")
}
fmt.Fprint(p.w, "\033[2K\033[1G")
}
}
return stopped
}
func (p *Progress) Add(key string, state State) {
p.mu.Lock()
defer p.mu.Unlock()
p.states = append(p.states, state)
}
func (p *Progress) render() {
p.mu.Lock()
defer p.mu.Unlock()
// buffer output to minimize flickering on all terminals
p.buf.Reset()
defer p.buf.WriteTo(p.w)
// eliminate flickering on terminals that support synchronized output
fmt.Fprint(&p.buf, "\033[?2026h")
defer fmt.Fprint(&p.buf, "\033[?2026l")
fmt.Fprint(&p.buf, "\033[?25l")
defer fmt.Fprint(&p.buf, "\033[?25h")
// move the cursor back to the beginning
for range p.pos - 1 {
fmt.Fprint(&p.buf, "\033[A")
}
fmt.Fprint(&p.buf, "\033[1G")
// render progress lines
for i, state := range p.states {
fmt.Fprint(&p.buf, state.String())
fmt.Fprintf(&p.buf, "\033[K")
if i < len(p.states)-1 {
fmt.Fprint(&p.buf, "\n")
}
}
p.pos = len(p.states)
}
func (p *Progress) start() {
p.ticker = time.NewTicker(100 * time.Millisecond)
for range p.ticker.C {
p.render()
}
}