211 lines
5.3 KiB
Go
211 lines
5.3 KiB
Go
package delete
|
|
|
|
import (
|
|
"bytes"
|
|
"errors"
|
|
"io"
|
|
"net/http"
|
|
"regexp"
|
|
"testing"
|
|
|
|
"github.com/cli/cli/v2/internal/config"
|
|
"github.com/cli/cli/v2/internal/gh"
|
|
"github.com/cli/cli/v2/internal/ghrepo"
|
|
"github.com/cli/cli/v2/internal/prompter"
|
|
"github.com/cli/cli/v2/pkg/cmd/issue/argparsetest"
|
|
"github.com/cli/cli/v2/pkg/cmdutil"
|
|
"github.com/cli/cli/v2/pkg/httpmock"
|
|
"github.com/cli/cli/v2/pkg/iostreams"
|
|
"github.com/cli/cli/v2/test"
|
|
"github.com/google/shlex"
|
|
"github.com/stretchr/testify/assert"
|
|
)
|
|
|
|
func TestNewCmdDelete(t *testing.T) {
|
|
argparsetest.TestArgParsing(t, NewCmdDelete)
|
|
}
|
|
|
|
func runCommand(rt http.RoundTripper, pm *prompter.MockPrompter, isTTY bool, cli string) (*test.CmdOut, error) {
|
|
ios, _, stdout, stderr := iostreams.Test()
|
|
ios.SetStdoutTTY(isTTY)
|
|
ios.SetStdinTTY(isTTY)
|
|
ios.SetStderrTTY(isTTY)
|
|
|
|
factory := &cmdutil.Factory{
|
|
IOStreams: ios,
|
|
Prompter: pm,
|
|
HttpClient: func() (*http.Client, error) {
|
|
return &http.Client{Transport: rt}, nil
|
|
},
|
|
Config: func() (gh.Config, error) {
|
|
return config.NewBlankConfig(), nil
|
|
},
|
|
BaseRepo: func() (ghrepo.Interface, error) {
|
|
return ghrepo.New("OWNER", "REPO"), nil
|
|
},
|
|
}
|
|
|
|
cmd := NewCmdDelete(factory, nil)
|
|
|
|
argv, err := shlex.Split(cli)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
cmd.SetArgs(argv)
|
|
|
|
cmd.SetIn(&bytes.Buffer{})
|
|
cmd.SetOut(io.Discard)
|
|
cmd.SetErr(io.Discard)
|
|
|
|
_, err = cmd.ExecuteC()
|
|
return &test.CmdOut{
|
|
OutBuf: stdout,
|
|
ErrBuf: stderr,
|
|
}, err
|
|
}
|
|
|
|
func TestIssueDelete(t *testing.T) {
|
|
httpRegistry := &httpmock.Registry{}
|
|
defer httpRegistry.Verify(t)
|
|
|
|
httpRegistry.Register(
|
|
httpmock.GraphQL(`query IssueByNumber\b`),
|
|
httpmock.StringResponse(`
|
|
{ "data": { "repository": {
|
|
"hasIssuesEnabled": true,
|
|
"issue": { "id": "THE-ID", "number": 13, "title": "The title of the issue"}
|
|
} } }`),
|
|
)
|
|
httpRegistry.Register(
|
|
httpmock.GraphQL(`mutation IssueDelete\b`),
|
|
httpmock.GraphQLMutation(`{"id": "THE-ID"}`,
|
|
func(inputs map[string]interface{}) {
|
|
assert.Equal(t, inputs["issueId"], "THE-ID")
|
|
}),
|
|
)
|
|
|
|
pm := prompter.NewMockPrompter(t)
|
|
pm.RegisterConfirmDeletion("13", func(_ string) error { return nil })
|
|
|
|
output, err := runCommand(httpRegistry, pm, true, "13")
|
|
if err != nil {
|
|
t.Fatalf("error running command `issue delete`: %v", err)
|
|
}
|
|
|
|
r := regexp.MustCompile(`Deleted issue OWNER/REPO#13 \(The title of the issue\)`)
|
|
|
|
if !r.MatchString(output.Stderr()) {
|
|
t.Fatalf("output did not match regexp /%s/\n> output\n%q\n", r, output.Stderr())
|
|
}
|
|
}
|
|
|
|
func TestIssueDelete_confirm(t *testing.T) {
|
|
httpRegistry := &httpmock.Registry{}
|
|
defer httpRegistry.Verify(t)
|
|
|
|
httpRegistry.Register(
|
|
httpmock.GraphQL(`query IssueByNumber\b`),
|
|
httpmock.StringResponse(`
|
|
{ "data": { "repository": {
|
|
"hasIssuesEnabled": true,
|
|
"issue": { "id": "THE-ID", "number": 13, "title": "The title of the issue"}
|
|
} } }`),
|
|
)
|
|
httpRegistry.Register(
|
|
httpmock.GraphQL(`mutation IssueDelete\b`),
|
|
httpmock.GraphQLMutation(`{"id": "THE-ID"}`,
|
|
func(inputs map[string]interface{}) {
|
|
assert.Equal(t, inputs["issueId"], "THE-ID")
|
|
}),
|
|
)
|
|
|
|
output, err := runCommand(httpRegistry, nil, true, "13 --confirm")
|
|
if err != nil {
|
|
t.Fatalf("error running command `issue delete`: %v", err)
|
|
}
|
|
|
|
r := regexp.MustCompile(`Deleted issue OWNER/REPO#13 \(The title of the issue\)`)
|
|
|
|
if !r.MatchString(output.Stderr()) {
|
|
t.Fatalf("output did not match regexp /%s/\n> output\n%q\n", r, output.Stderr())
|
|
}
|
|
}
|
|
|
|
func TestIssueDelete_cancel(t *testing.T) {
|
|
httpRegistry := &httpmock.Registry{}
|
|
defer httpRegistry.Verify(t)
|
|
|
|
httpRegistry.Register(
|
|
httpmock.GraphQL(`query IssueByNumber\b`),
|
|
httpmock.StringResponse(`
|
|
{ "data": { "repository": {
|
|
"hasIssuesEnabled": true,
|
|
"issue": { "id": "THE-ID", "number": 13, "title": "The title of the issue"}
|
|
} } }`),
|
|
)
|
|
|
|
pm := prompter.NewMockPrompter(t)
|
|
pm.RegisterConfirmDeletion("13", func(_ string) error {
|
|
return errors.New("You entered 14")
|
|
})
|
|
|
|
_, err := runCommand(httpRegistry, pm, true, "13")
|
|
if err == nil {
|
|
t.Fatalf("expected error")
|
|
}
|
|
if err.Error() != "You entered 14" {
|
|
t.Fatalf("got unexpected error '%s'", err)
|
|
}
|
|
}
|
|
|
|
func TestIssueDelete_doesNotExist(t *testing.T) {
|
|
httpRegistry := &httpmock.Registry{}
|
|
defer httpRegistry.Verify(t)
|
|
|
|
httpRegistry.Register(
|
|
httpmock.GraphQL(`query IssueByNumber\b`),
|
|
httpmock.StringResponse(`
|
|
{ "errors": [
|
|
{ "message": "Could not resolve to an Issue with the number of 13." }
|
|
] }
|
|
`),
|
|
)
|
|
|
|
_, err := runCommand(httpRegistry, nil, true, "13")
|
|
if err == nil || err.Error() != "GraphQL: Could not resolve to an Issue with the number of 13." {
|
|
t.Errorf("error running command `issue delete`: %v", err)
|
|
}
|
|
}
|
|
|
|
func TestIssueDelete_issuesDisabled(t *testing.T) {
|
|
httpRegistry := &httpmock.Registry{}
|
|
defer httpRegistry.Verify(t)
|
|
|
|
httpRegistry.Register(
|
|
httpmock.GraphQL(`query IssueByNumber\b`),
|
|
httpmock.StringResponse(`
|
|
{
|
|
"data": {
|
|
"repository": {
|
|
"hasIssuesEnabled": false,
|
|
"issue": null
|
|
}
|
|
},
|
|
"errors": [
|
|
{
|
|
"type": "NOT_FOUND",
|
|
"path": [
|
|
"repository",
|
|
"issue"
|
|
],
|
|
"message": "Could not resolve to an issue or pull request with the number of 13."
|
|
}
|
|
]
|
|
}`),
|
|
)
|
|
|
|
_, err := runCommand(httpRegistry, nil, true, "13")
|
|
if err == nil || err.Error() != "the 'OWNER/REPO' repository has disabled issues" {
|
|
t.Fatalf("got error: %v", err)
|
|
}
|
|
}
|