cli/commands/issue/list/issue_list_test.go

329 lines
9.1 KiB
Go

package list
import (
"bytes"
"fmt"
"io/ioutil"
"net/http"
"regexp"
"strings"
"testing"
"gitlab.com/gitlab-org/cli/pkg/iostreams"
"gitlab.com/gitlab-org/cli/commands/cmdtest"
"github.com/MakeNowJust/heredoc"
"github.com/alecthomas/assert"
"github.com/google/shlex"
"github.com/xanzy/go-gitlab"
"gitlab.com/gitlab-org/cli/api"
"gitlab.com/gitlab-org/cli/commands/cmdutils"
"gitlab.com/gitlab-org/cli/internal/config"
"gitlab.com/gitlab-org/cli/internal/glrepo"
"gitlab.com/gitlab-org/cli/pkg/httpmock"
"gitlab.com/gitlab-org/cli/test"
)
func runCommand(rt http.RoundTripper, isTTY bool, cli string, runE func(opts *ListOptions) error, doHyperlinks string) (*test.CmdOut, error) {
io, _, stdout, stderr := iostreams.Test()
io.IsaTTY = isTTY
io.IsInTTY = isTTY
io.IsErrTTY = isTTY
if doHyperlinks != "" {
io.SetDisplayHyperlinks(doHyperlinks)
}
factory := &cmdutils.Factory{
IO: io,
HttpClient: func() (*gitlab.Client, error) {
a, err := api.TestClient(&http.Client{Transport: rt}, "", "", false)
if err != nil {
return nil, err
}
return a.Lab(), err
},
Config: func() (config.Config, error) {
return config.NewBlankConfig(), nil
},
BaseRepo: func() (glrepo.Interface, error) {
return glrepo.New("OWNER", "REPO"), nil
},
}
// TODO: shouldn't be there but the stub doesn't work without it
_, _ = factory.HttpClient()
cmd := NewCmdList(factory, runE)
argv, err := shlex.Split(cli)
if err != nil {
return nil, err
}
cmd.SetArgs(argv)
cmd.SetIn(&bytes.Buffer{})
cmd.SetOut(ioutil.Discard)
cmd.SetErr(ioutil.Discard)
_, err = cmd.ExecuteC()
return &test.CmdOut{
OutBuf: stdout,
ErrBuf: stderr,
}, err
}
func TestNewCmdList(t *testing.T) {
io, _, _, _ := iostreams.Test()
io.IsaTTY = true
io.IsInTTY = true
io.IsErrTTY = true
fakeHTTP := httpmock.New()
defer fakeHTTP.Verify(t)
factory := &cmdutils.Factory{
IO: io,
HttpClient: func() (*gitlab.Client, error) {
a, err := api.TestClient(&http.Client{Transport: fakeHTTP}, "", "", false)
if err != nil {
return nil, err
}
return a.Lab(), err
},
Config: func() (config.Config, error) {
return config.NewBlankConfig(), nil
},
BaseRepo: func() (glrepo.Interface, error) {
return glrepo.New("OWNER", "REPO"), nil
},
}
t.Run("Issue_NewCmdList", func(t *testing.T) {
gotOpts := &ListOptions{}
err := NewCmdList(factory, func(opts *ListOptions) error {
gotOpts = opts
return nil
}).Execute()
assert.Nil(t, err)
assert.Equal(t, factory.IO, gotOpts.IO)
gotBaseRepo, _ := gotOpts.BaseRepo()
expectedBaseRepo, _ := factory.BaseRepo()
assert.Equal(t, gotBaseRepo, expectedBaseRepo)
})
}
func TestIssueList_tty(t *testing.T) {
fakeHTTP := httpmock.New()
defer fakeHTTP.Verify(t)
fakeHTTP.RegisterResponder("GET", "/projects/OWNER/REPO/issues",
httpmock.NewFileResponse(200, "./fixtures/issueList.json"))
output, err := runCommand(fakeHTTP, true, "", nil, "")
if err != nil {
t.Errorf("error running command `issue list`: %v", err)
}
out := output.String()
timeRE := regexp.MustCompile(`\d+ years`)
out = timeRE.ReplaceAllString(out, "X years")
assert.Equal(t, heredoc.Doc(`
Showing 2 open issues in OWNER/REPO that match your search (Page 1)
#6 OWNER/REPO/issues/6 Issue one (foo, bar) about X years ago
#7 OWNER/REPO/issues/7 Issue two (fooz, baz) about X years ago
`), out)
assert.Equal(t, ``, output.Stderr())
}
func TestIssueList_tty_withFlags(t *testing.T) {
t.Run("project", func(t *testing.T) {
fakeHTTP := httpmock.New()
defer fakeHTTP.Verify(t)
fakeHTTP.RegisterResponder("GET", "/projects/OWNER/REPO/issues",
httpmock.NewStringResponse(200, `[]`))
output, err := runCommand(fakeHTTP, true, "--opened -P1 -p100 --confidential -a someuser -l bug -m1", nil, "")
if err != nil {
t.Errorf("error running command `issue list`: %v", err)
}
cmdtest.Eq(t, output.Stderr(), "")
cmdtest.Eq(t, output.String(), `No open issues match your search in OWNER/REPO
`)
})
t.Run("group", func(t *testing.T) {
fakeHTTP := httpmock.New()
defer fakeHTTP.Verify(t)
fakeHTTP.RegisterResponder("GET", "/groups/GROUP/issues",
httpmock.NewStringResponse(200, `[]`))
output, err := runCommand(fakeHTTP, true, "--group GROUP", nil, "")
if err != nil {
t.Errorf("error running command `issue list`: %v", err)
}
cmdtest.Eq(t, output.Stderr(), "")
cmdtest.Eq(t, output.String(), `No open issues match your search in GROUP
`)
})
}
func TestIssueList_tty_withIssueType(t *testing.T) {
fakeHTTP := httpmock.New()
defer fakeHTTP.Verify(t)
fakeHTTP.RegisterResponder("GET", "/projects/OWNER/REPO/issues",
httpmock.NewFileResponse(200, "./fixtures/incidentList.json"))
output, err := runCommand(fakeHTTP, true, "--issue-type=incident", nil, "")
if err != nil {
t.Errorf("error running command `issue list`: %v", err)
}
out := output.String()
timeRE := regexp.MustCompile(`\d+ years`)
out = timeRE.ReplaceAllString(out, "X years")
assert.Equal(t, heredoc.Doc(`
Showing 1 open issue in OWNER/REPO that match your search (Page 1)
#8 OWNER/REPO/issues/8 Incident (foo, baz) about X years ago
`), out)
assert.Equal(t, ``, output.Stderr())
}
func TestIssueList_tty_mine(t *testing.T) {
t.Run("mine with all flag and user exists", func(t *testing.T) {
fakeHTTP := httpmock.New()
defer fakeHTTP.Verify(t)
fakeHTTP.RegisterResponder("GET", "/projects/OWNER/REPO/issues",
httpmock.NewStringResponse(200, `[]`))
fakeHTTP.RegisterResponder("GET", "/user",
httpmock.NewStringResponse(200, `{"username": "john_smith"}`))
output, err := runCommand(fakeHTTP, true, "--mine -A", nil, "")
if err != nil {
t.Errorf("error running command `issue list`: %v", err)
}
cmdtest.Eq(t, output.Stderr(), "")
cmdtest.Eq(t, output.String(), `No issues match your search in OWNER/REPO
`)
})
t.Run("user does not exists", func(t *testing.T) {
fakeHTTP := httpmock.New()
defer fakeHTTP.Verify(t)
fakeHTTP.RegisterResponder("GET", "/user",
httpmock.NewStringResponse(404, `{message: 404 Not found}`))
output, err := runCommand(fakeHTTP, true, "--mine -A", nil, "")
assert.NotNil(t, err)
cmdtest.Eq(t, output.Stderr(), "")
cmdtest.Eq(t, output.String(), "")
})
}
func makeHyperlink(linkText, targetURL string) string {
return fmt.Sprintf("\x1b]8;;%s\x1b\\%s\x1b]8;;\x1b\\", targetURL, linkText)
}
func TestIssueList_hyperlinks(t *testing.T) {
noHyperlinkCells := [][]string{
{"#6", "OWNER/REPO/issues/6", "Issue one", "(foo, bar)", "about X years ago"},
{"#7", "OWNER/REPO/issues/7", "Issue two", "(fooz, baz)", "about X years ago"},
}
hyperlinkCells := [][]string{
{makeHyperlink("#6", "http://gitlab.com/OWNER/REPO/issues/6"), "OWNER/REPO/issues/6", "Issue one", "(foo, bar)", "about X years ago"},
{makeHyperlink("#7", "http://gitlab.com/OWNER/REPO/issues/7"), "OWNER/REPO/issues/7", "Issue two", "(fooz, baz)", "about X years ago"},
}
type hyperlinkTest struct {
forceHyperlinksEnv string
displayHyperlinksConfig string
isTTY bool
expectedCells [][]string
}
tests := []hyperlinkTest{
// FORCE_HYPERLINKS causes hyperlinks to be output, whether or not we're talking to a TTY
{forceHyperlinksEnv: "1", isTTY: true, expectedCells: hyperlinkCells},
{forceHyperlinksEnv: "1", isTTY: false, expectedCells: hyperlinkCells},
// empty/missing display_hyperlinks in config defaults to *not* outputting hyperlinks
{displayHyperlinksConfig: "", isTTY: true, expectedCells: noHyperlinkCells},
{displayHyperlinksConfig: "", isTTY: false, expectedCells: noHyperlinkCells},
// display_hyperlinks: false in config prevents outputting hyperlinks
{displayHyperlinksConfig: "false", isTTY: true, expectedCells: noHyperlinkCells},
{displayHyperlinksConfig: "false", isTTY: false, expectedCells: noHyperlinkCells},
// display_hyperlinks: true in config only outputs hyperlinks if we're talking to a TTY
{displayHyperlinksConfig: "true", isTTY: true, expectedCells: hyperlinkCells},
{displayHyperlinksConfig: "true", isTTY: false, expectedCells: noHyperlinkCells},
}
for _, test := range tests {
t.Run("", func(t *testing.T) {
fakeHTTP := httpmock.New()
defer fakeHTTP.Verify(t)
fakeHTTP.RegisterResponder("GET", "/projects/OWNER/REPO/issues",
httpmock.NewFileResponse(200, "./fixtures/issueList.json"))
doHyperlinks := "never"
if test.forceHyperlinksEnv == "1" {
doHyperlinks = "always"
} else if test.displayHyperlinksConfig == "true" {
doHyperlinks = "auto"
}
output, err := runCommand(fakeHTTP, test.isTTY, "", nil, doHyperlinks)
if err != nil {
t.Errorf("error running command `issue list`: %v", err)
}
out := output.String()
timeRE := regexp.MustCompile(`\d+ years`)
out = timeRE.ReplaceAllString(out, "X years")
lines := strings.Split(strings.TrimRight(out, "\n"), "\n")
// first two lines have the header and some separating whitespace, so skip those
for lineNum, line := range lines[2:] {
gotCells := strings.Split(line, "\t")
expectedCells := test.expectedCells[lineNum]
assert.Equal(t, len(expectedCells), len(gotCells))
for cellNum, gotCell := range gotCells {
expectedCell := expectedCells[cellNum]
assert.Equal(t, expectedCell, strings.Trim(gotCell, " "))
}
}
})
}
}