Files
restic/internal/ui/termstatus/status_test.go

160 lines
3.9 KiB
Go
Raw Normal View History

package termstatus
import (
"bytes"
"context"
"errors"
"fmt"
"io"
2025-09-21 20:16:29 +02:00
"strings"
"testing"
"github.com/restic/restic/internal/terminal"
rtest "github.com/restic/restic/internal/test"
)
func TestSetStatus(t *testing.T) {
2025-09-21 20:16:29 +02:00
buf, term, cancel := setupStatusTest()
const (
cl = terminal.PosixControlClearLine
home = terminal.PosixControlMoveCursorHome
up = terminal.PosixControlMoveCursorUp
)
term.SetStatus([]string{"first"})
exp := home + cl + "first" + home
term.SetStatus([]string{""})
exp += home + cl + "" + home
term.SetStatus([]string{})
exp += home + cl + "" + home
// already empty status
term.SetStatus([]string{})
term.SetStatus([]string{"foo", "bar", "baz"})
exp += home + cl + "foo\n" + home + cl + "bar\n" +
home + cl + "baz" + home + up + up
term.SetStatus([]string{"quux", "needs\nquote"})
exp += home + cl + "quux\n" +
home + cl + "\"needs\\nquote\"\n" +
home + cl + home + up + up // Clear third line
cancel()
exp += home + cl + "\n" + home + cl + home + up // Status cleared
<-term.closed
rtest.Equals(t, exp, buf.String())
}
2025-09-21 20:16:29 +02:00
func setupStatusTest() (*bytes.Buffer, *Terminal, context.CancelFunc) {
buf := &bytes.Buffer{}
term := New(nil, buf, buf, false)
term.canUpdateStatus = true
term.fd = ^uintptr(0)
term.clearCurrentLine = terminal.PosixClearCurrentLine
term.moveCursorUp = terminal.PosixMoveCursorUp
ctx, cancel := context.WithCancel(context.Background())
go term.Run(ctx)
return buf, term, cancel
}
func TestPrint(t *testing.T) {
buf, term, cancel := setupStatusTest()
const (
cl = terminal.PosixControlClearLine
home = terminal.PosixControlMoveCursorHome
)
term.Print("test")
exp := home + cl + "test\n"
term.Error("error")
exp += home + cl + "error\n"
cancel()
<-term.closed
rtest.Equals(t, exp, buf.String())
}
func TestSanitizeLines(t *testing.T) {
var tests = []struct {
input []string
width int
output []string
}{
{[]string{""}, 80, []string{""}},
{[]string{"too long test line"}, 10, []string{"too long"}},
{[]string{"too long test line", "text"}, 10, []string{"too long\n", "text"}},
{[]string{"too long test line", "second long test line"}, 10, []string{"too long\n", "second l"}},
}
for _, test := range tests {
t.Run(fmt.Sprintf("%s %d", test.input, test.width), func(t *testing.T) {
out := sanitizeLines(test.input, test.width)
rtest.Equals(t, test.output, out)
})
}
}
type errorReader struct{ err error }
func (r *errorReader) Read([]byte) (int, error) { return 0, r.err }
func TestReadPassword(t *testing.T) {
want := errors.New("foo")
_, err := readPassword(&errorReader{want})
rtest.Assert(t, errors.Is(err, want), "wrong error %v", err)
}
2025-09-21 20:16:29 +02:00
func TestReadPasswordTerminal(t *testing.T) {
expected := "password"
term := New(io.NopCloser(strings.NewReader(expected)), io.Discard, io.Discard, false)
pw, err := term.ReadPassword(context.Background(), "test")
rtest.OK(t, err)
rtest.Equals(t, expected, pw)
}
func TestRawInputOutput(t *testing.T) {
input := io.NopCloser(strings.NewReader("password"))
var output bytes.Buffer
term := New(input, &output, io.Discard, false)
rtest.Equals(t, input, term.InputRaw())
rtest.Equals(t, false, term.InputIsTerminal())
rtest.Equals(t, &output, term.OutputRaw())
rtest.Equals(t, false, term.OutputIsTerminal())
rtest.Equals(t, false, term.CanUpdateStatus())
}
func TestDisableStatus(t *testing.T) {
var output bytes.Buffer
term, cancel := Setup(nil, &output, &output, true)
rtest.Equals(t, false, term.CanUpdateStatus())
term.Print("test")
term.Error("error")
term.SetStatus([]string{"status"})
cancel()
rtest.Equals(t, "test\nerror\nstatus\n", output.String())
}
func TestOutputWriter(t *testing.T) {
var output bytes.Buffer
term, cancel := Setup(nil, &output, &output, true)
_, err := term.OutputWriter().Write([]byte("output\npartial"))
rtest.OK(t, err)
term.Print("test")
term.Error("error")
cancel()
rtest.Equals(t, "output\ntest\nerror\npartial\n", output.String())
}