forked from TrueCloudLab/restic
restore: reorganize progress bar code
The structure is now much more similar to that of the backup command.
This commit is contained in:
parent
bb20078641
commit
a9aff885d6
5 changed files with 94 additions and 80 deletions
|
@ -177,7 +177,7 @@ func runRestore(ctx context.Context, opts RestoreOptions, gopts GlobalOptions,
|
||||||
|
|
||||||
var progress *restoreui.Progress
|
var progress *restoreui.Progress
|
||||||
if !gopts.Quiet && !gopts.JSON {
|
if !gopts.Quiet && !gopts.JSON {
|
||||||
progress = restoreui.NewProgress(restoreui.NewProgressPrinter(term), calculateProgressInterval(!gopts.Quiet, gopts.JSON))
|
progress = restoreui.NewProgress(restoreui.NewTextPrinter(term), calculateProgressInterval(!gopts.Quiet, gopts.JSON))
|
||||||
}
|
}
|
||||||
|
|
||||||
res := restorer.NewRestorer(repo, sn, opts.Sparse, progress)
|
res := restorer.NewRestorer(repo, sn, opts.Sparse, progress)
|
||||||
|
|
|
@ -1,11 +1,9 @@
|
||||||
package restore
|
package restore
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
|
||||||
"sync"
|
"sync"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/restic/restic/internal/ui"
|
|
||||||
"github.com/restic/restic/internal/ui/progress"
|
"github.com/restic/restic/internal/ui/progress"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -28,6 +26,11 @@ type progressInfoEntry struct {
|
||||||
bytesTotal uint64
|
bytesTotal uint64
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type term interface {
|
||||||
|
Print(line string)
|
||||||
|
SetStatus(lines []string)
|
||||||
|
}
|
||||||
|
|
||||||
type ProgressPrinter interface {
|
type ProgressPrinter interface {
|
||||||
Update(filesFinished, filesTotal, allBytesWritten, allBytesTotal uint64, duration time.Duration)
|
Update(filesFinished, filesTotal, allBytesWritten, allBytesTotal uint64, duration time.Duration)
|
||||||
Finish(filesFinished, filesTotal, allBytesWritten, allBytesTotal uint64, duration time.Duration)
|
Finish(filesFinished, filesTotal, allBytesWritten, allBytesTotal uint64, duration time.Duration)
|
||||||
|
@ -85,47 +88,3 @@ func (p *Progress) AddProgress(name string, bytesWrittenPortion uint64, bytesTot
|
||||||
func (p *Progress) Finish() {
|
func (p *Progress) Finish() {
|
||||||
p.updater.Done()
|
p.updater.Done()
|
||||||
}
|
}
|
||||||
|
|
||||||
type term interface {
|
|
||||||
Print(line string)
|
|
||||||
SetStatus(lines []string)
|
|
||||||
}
|
|
||||||
|
|
||||||
type textPrinter struct {
|
|
||||||
terminal term
|
|
||||||
}
|
|
||||||
|
|
||||||
func NewProgressPrinter(terminal term) ProgressPrinter {
|
|
||||||
return &textPrinter{
|
|
||||||
terminal: terminal,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (t *textPrinter) Update(filesFinished, filesTotal, allBytesWritten, allBytesTotal uint64, duration time.Duration) {
|
|
||||||
timeLeft := ui.FormatDuration(duration)
|
|
||||||
formattedAllBytesWritten := ui.FormatBytes(allBytesWritten)
|
|
||||||
formattedAllBytesTotal := ui.FormatBytes(allBytesTotal)
|
|
||||||
allPercent := ui.FormatPercent(allBytesWritten, allBytesTotal)
|
|
||||||
progress := fmt.Sprintf("[%s] %s %v files %s, total %v files %v",
|
|
||||||
timeLeft, allPercent, filesFinished, formattedAllBytesWritten, filesTotal, formattedAllBytesTotal)
|
|
||||||
|
|
||||||
t.terminal.SetStatus([]string{progress})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (t *textPrinter) Finish(filesFinished, filesTotal, allBytesWritten, allBytesTotal uint64, duration time.Duration) {
|
|
||||||
t.terminal.SetStatus([]string{})
|
|
||||||
|
|
||||||
timeLeft := ui.FormatDuration(duration)
|
|
||||||
formattedAllBytesTotal := ui.FormatBytes(allBytesTotal)
|
|
||||||
|
|
||||||
var summary string
|
|
||||||
if filesFinished == filesTotal && allBytesWritten == allBytesTotal {
|
|
||||||
summary = fmt.Sprintf("Summary: Restored %d Files (%s) in %s", filesTotal, formattedAllBytesTotal, timeLeft)
|
|
||||||
} else {
|
|
||||||
formattedAllBytesWritten := ui.FormatBytes(allBytesWritten)
|
|
||||||
summary = fmt.Sprintf("Summary: Restored %d / %d Files (%s / %s) in %s",
|
|
||||||
filesFinished, filesTotal, formattedAllBytesWritten, formattedAllBytesTotal, timeLeft)
|
|
||||||
}
|
|
||||||
|
|
||||||
t.terminal.Print(summary)
|
|
||||||
}
|
|
|
@ -135,36 +135,3 @@ func TestSummaryOnErrors(t *testing.T) {
|
||||||
printerTraceEntry{1, 2, 50 + fileSize/2, 50 + fileSize, mockFinishDuration, true},
|
printerTraceEntry{1, 2, 50 + fileSize/2, 50 + fileSize, mockFinishDuration, true},
|
||||||
}, result)
|
}, result)
|
||||||
}
|
}
|
||||||
|
|
||||||
type mockTerm struct {
|
|
||||||
output []string
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *mockTerm) Print(line string) {
|
|
||||||
m.output = append(m.output, line)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *mockTerm) SetStatus(lines []string) {
|
|
||||||
m.output = append([]string{}, lines...)
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestPrintUpdate(t *testing.T) {
|
|
||||||
term := &mockTerm{}
|
|
||||||
printer := NewProgressPrinter(term)
|
|
||||||
printer.Update(3, 11, 29, 47, 5*time.Second)
|
|
||||||
test.Equals(t, []string{"[0:05] 61.70% 3 files 29 B, total 11 files 47 B"}, term.output)
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestPrintSummaryOnSuccess(t *testing.T) {
|
|
||||||
term := &mockTerm{}
|
|
||||||
printer := NewProgressPrinter(term)
|
|
||||||
printer.Finish(11, 11, 47, 47, 5*time.Second)
|
|
||||||
test.Equals(t, []string{"Summary: Restored 11 Files (47 B) in 0:05"}, term.output)
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestPrintSummaryOnErrors(t *testing.T) {
|
|
||||||
term := &mockTerm{}
|
|
||||||
printer := NewProgressPrinter(term)
|
|
||||||
printer.Finish(3, 11, 29, 47, 5*time.Second)
|
|
||||||
test.Equals(t, []string{"Summary: Restored 3 / 11 Files (29 B / 47 B) in 0:05"}, term.output)
|
|
||||||
}
|
|
47
internal/ui/restore/text.go
Normal file
47
internal/ui/restore/text.go
Normal file
|
@ -0,0 +1,47 @@
|
||||||
|
package restore
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/restic/restic/internal/ui"
|
||||||
|
)
|
||||||
|
|
||||||
|
type textPrinter struct {
|
||||||
|
terminal term
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewTextPrinter(terminal term) ProgressPrinter {
|
||||||
|
return &textPrinter{
|
||||||
|
terminal: terminal,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *textPrinter) Update(filesFinished, filesTotal, allBytesWritten, allBytesTotal uint64, duration time.Duration) {
|
||||||
|
timeLeft := ui.FormatDuration(duration)
|
||||||
|
formattedAllBytesWritten := ui.FormatBytes(allBytesWritten)
|
||||||
|
formattedAllBytesTotal := ui.FormatBytes(allBytesTotal)
|
||||||
|
allPercent := ui.FormatPercent(allBytesWritten, allBytesTotal)
|
||||||
|
progress := fmt.Sprintf("[%s] %s %v files %s, total %v files %v",
|
||||||
|
timeLeft, allPercent, filesFinished, formattedAllBytesWritten, filesTotal, formattedAllBytesTotal)
|
||||||
|
|
||||||
|
t.terminal.SetStatus([]string{progress})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *textPrinter) Finish(filesFinished, filesTotal, allBytesWritten, allBytesTotal uint64, duration time.Duration) {
|
||||||
|
t.terminal.SetStatus([]string{})
|
||||||
|
|
||||||
|
timeLeft := ui.FormatDuration(duration)
|
||||||
|
formattedAllBytesTotal := ui.FormatBytes(allBytesTotal)
|
||||||
|
|
||||||
|
var summary string
|
||||||
|
if filesFinished == filesTotal && allBytesWritten == allBytesTotal {
|
||||||
|
summary = fmt.Sprintf("Summary: Restored %d Files (%s) in %s", filesTotal, formattedAllBytesTotal, timeLeft)
|
||||||
|
} else {
|
||||||
|
formattedAllBytesWritten := ui.FormatBytes(allBytesWritten)
|
||||||
|
summary = fmt.Sprintf("Summary: Restored %d / %d Files (%s / %s) in %s",
|
||||||
|
filesFinished, filesTotal, formattedAllBytesWritten, formattedAllBytesTotal, timeLeft)
|
||||||
|
}
|
||||||
|
|
||||||
|
t.terminal.Print(summary)
|
||||||
|
}
|
41
internal/ui/restore/text_test.go
Normal file
41
internal/ui/restore/text_test.go
Normal file
|
@ -0,0 +1,41 @@
|
||||||
|
package restore
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/restic/restic/internal/test"
|
||||||
|
)
|
||||||
|
|
||||||
|
type mockTerm struct {
|
||||||
|
output []string
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *mockTerm) Print(line string) {
|
||||||
|
m.output = append(m.output, line)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *mockTerm) SetStatus(lines []string) {
|
||||||
|
m.output = append([]string{}, lines...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestPrintUpdate(t *testing.T) {
|
||||||
|
term := &mockTerm{}
|
||||||
|
printer := NewTextPrinter(term)
|
||||||
|
printer.Update(3, 11, 29, 47, 5*time.Second)
|
||||||
|
test.Equals(t, []string{"[0:05] 61.70% 3 files 29 B, total 11 files 47 B"}, term.output)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestPrintSummaryOnSuccess(t *testing.T) {
|
||||||
|
term := &mockTerm{}
|
||||||
|
printer := NewTextPrinter(term)
|
||||||
|
printer.Finish(11, 11, 47, 47, 5*time.Second)
|
||||||
|
test.Equals(t, []string{"Summary: Restored 11 Files (47 B) in 0:05"}, term.output)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestPrintSummaryOnErrors(t *testing.T) {
|
||||||
|
term := &mockTerm{}
|
||||||
|
printer := NewTextPrinter(term)
|
||||||
|
printer.Finish(3, 11, 29, 47, 5*time.Second)
|
||||||
|
test.Equals(t, []string{"Summary: Restored 3 / 11 Files (29 B / 47 B) in 0:05"}, term.output)
|
||||||
|
}
|
Loading…
Reference in a new issue