Merge pull request #4815 from greatroar/termstatus
Termstatus refactoring
This commit is contained in:
commit
7ea508e7b8
4 changed files with 49 additions and 76 deletions
|
@ -4,7 +4,6 @@ import (
|
||||||
"context"
|
"context"
|
||||||
"sync"
|
"sync"
|
||||||
|
|
||||||
"github.com/restic/restic/internal/ui"
|
|
||||||
"github.com/restic/restic/internal/ui/termstatus"
|
"github.com/restic/restic/internal/ui/termstatus"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -31,8 +30,7 @@ func setupTermstatus() (*termstatus.Terminal, func()) {
|
||||||
|
|
||||||
// use the termstatus for stdout/stderr
|
// use the termstatus for stdout/stderr
|
||||||
prevStdout, prevStderr := globalOptions.stdout, globalOptions.stderr
|
prevStdout, prevStderr := globalOptions.stdout, globalOptions.stderr
|
||||||
stdioWrapper := ui.NewStdioWrapper(term)
|
globalOptions.stdout, globalOptions.stderr = termstatus.WrapStdio(term)
|
||||||
globalOptions.stdout, globalOptions.stderr = stdioWrapper.Stdout(), stdioWrapper.Stderr()
|
|
||||||
|
|
||||||
return term, func() {
|
return term, func() {
|
||||||
// shutdown termstatus
|
// shutdown termstatus
|
||||||
|
|
|
@ -1,72 +0,0 @@
|
||||||
package ui
|
|
||||||
|
|
||||||
import (
|
|
||||||
"bytes"
|
|
||||||
"io"
|
|
||||||
|
|
||||||
"github.com/restic/restic/internal/ui/termstatus"
|
|
||||||
)
|
|
||||||
|
|
||||||
// StdioWrapper provides stdout and stderr integration with termstatus.
|
|
||||||
type StdioWrapper struct {
|
|
||||||
stdout *lineWriter
|
|
||||||
stderr *lineWriter
|
|
||||||
}
|
|
||||||
|
|
||||||
// NewStdioWrapper initializes a new stdio wrapper that can be used in place of
|
|
||||||
// os.Stdout or os.Stderr.
|
|
||||||
func NewStdioWrapper(term *termstatus.Terminal) *StdioWrapper {
|
|
||||||
return &StdioWrapper{
|
|
||||||
stdout: newLineWriter(term.Print),
|
|
||||||
stderr: newLineWriter(term.Error),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Stdout returns a writer that is line buffered and can be used in place of
|
|
||||||
// os.Stdout. On Close(), the remaining bytes are written, followed by a line
|
|
||||||
// break.
|
|
||||||
func (w *StdioWrapper) Stdout() io.WriteCloser {
|
|
||||||
return w.stdout
|
|
||||||
}
|
|
||||||
|
|
||||||
// Stderr returns a writer that is line buffered and can be used in place of
|
|
||||||
// os.Stderr. On Close(), the remaining bytes are written, followed by a line
|
|
||||||
// break.
|
|
||||||
func (w *StdioWrapper) Stderr() io.WriteCloser {
|
|
||||||
return w.stderr
|
|
||||||
}
|
|
||||||
|
|
||||||
type lineWriter struct {
|
|
||||||
buf *bytes.Buffer
|
|
||||||
print func(string)
|
|
||||||
}
|
|
||||||
|
|
||||||
var _ io.WriteCloser = &lineWriter{}
|
|
||||||
|
|
||||||
func newLineWriter(print func(string)) *lineWriter {
|
|
||||||
return &lineWriter{buf: bytes.NewBuffer(nil), print: print}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (w *lineWriter) Write(data []byte) (n int, err error) {
|
|
||||||
n, err = w.buf.Write(data)
|
|
||||||
if err != nil {
|
|
||||||
return n, err
|
|
||||||
}
|
|
||||||
|
|
||||||
// look for line breaks
|
|
||||||
buf := w.buf.Bytes()
|
|
||||||
i := bytes.LastIndexByte(buf, '\n')
|
|
||||||
if i != -1 {
|
|
||||||
w.print(string(buf[:i+1]))
|
|
||||||
w.buf.Next(i + 1)
|
|
||||||
}
|
|
||||||
|
|
||||||
return n, err
|
|
||||||
}
|
|
||||||
|
|
||||||
func (w *lineWriter) Close() error {
|
|
||||||
if w.buf.Len() > 0 {
|
|
||||||
w.print(string(append(w.buf.Bytes(), '\n')))
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
47
internal/ui/termstatus/stdio_wrapper.go
Normal file
47
internal/ui/termstatus/stdio_wrapper.go
Normal file
|
@ -0,0 +1,47 @@
|
||||||
|
package termstatus
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"io"
|
||||||
|
)
|
||||||
|
|
||||||
|
// WrapStdio returns line-buffering replacements for os.Stdout and os.Stderr.
|
||||||
|
// On Close, the remaining bytes are written, followed by a line break.
|
||||||
|
func WrapStdio(term *Terminal) (stdout, stderr io.WriteCloser) {
|
||||||
|
return newLineWriter(term.Print), newLineWriter(term.Error)
|
||||||
|
}
|
||||||
|
|
||||||
|
type lineWriter struct {
|
||||||
|
buf bytes.Buffer
|
||||||
|
print func(string)
|
||||||
|
}
|
||||||
|
|
||||||
|
var _ io.WriteCloser = &lineWriter{}
|
||||||
|
|
||||||
|
func newLineWriter(print func(string)) *lineWriter {
|
||||||
|
return &lineWriter{print: print}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *lineWriter) Write(data []byte) (n int, err error) {
|
||||||
|
n, err = w.buf.Write(data)
|
||||||
|
if err != nil {
|
||||||
|
return n, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// look for line breaks
|
||||||
|
buf := w.buf.Bytes()
|
||||||
|
i := bytes.LastIndexByte(buf, '\n')
|
||||||
|
if i != -1 {
|
||||||
|
w.print(string(buf[:i+1]))
|
||||||
|
w.buf.Next(i + 1)
|
||||||
|
}
|
||||||
|
|
||||||
|
return n, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *lineWriter) Close() error {
|
||||||
|
if w.buf.Len() > 0 {
|
||||||
|
w.print(string(append(w.buf.Bytes(), '\n')))
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
|
@ -1,4 +1,4 @@
|
||||||
package ui
|
package termstatus
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"strings"
|
"strings"
|
Loading…
Reference in a new issue