240 lines
6.4 KiB
Go
240 lines
6.4 KiB
Go
package delete
|
|
|
|
import (
|
|
"bytes"
|
|
"net/http"
|
|
"testing"
|
|
|
|
"github.com/cli/cli/v2/internal/ghrepo"
|
|
"github.com/cli/cli/v2/internal/prompter"
|
|
"github.com/cli/cli/v2/pkg/cmdutil"
|
|
"github.com/cli/cli/v2/pkg/httpmock"
|
|
"github.com/cli/cli/v2/pkg/iostreams"
|
|
"github.com/google/shlex"
|
|
"github.com/stretchr/testify/assert"
|
|
)
|
|
|
|
func TestNewCmdDelete(t *testing.T) {
|
|
tests := []struct {
|
|
name string
|
|
input string
|
|
tty bool
|
|
output DeleteOptions
|
|
wantErr bool
|
|
wantErrMsg string
|
|
wantStderr string
|
|
}{
|
|
{
|
|
name: "confirm flag",
|
|
tty: true,
|
|
input: "OWNER/REPO --confirm",
|
|
output: DeleteOptions{RepoArg: "OWNER/REPO", Confirmed: true},
|
|
},
|
|
{
|
|
name: "yes flag",
|
|
tty: true,
|
|
input: "OWNER/REPO --yes",
|
|
output: DeleteOptions{RepoArg: "OWNER/REPO", Confirmed: true},
|
|
},
|
|
{
|
|
name: "no confirmation notty",
|
|
input: "OWNER/REPO",
|
|
output: DeleteOptions{RepoArg: "OWNER/REPO"},
|
|
wantErr: true,
|
|
wantErrMsg: "--yes required when not running interactively",
|
|
},
|
|
{
|
|
name: "base repo resolution",
|
|
input: "",
|
|
tty: true,
|
|
output: DeleteOptions{},
|
|
},
|
|
{
|
|
name: "yes flag ignored when no argument tty",
|
|
tty: true,
|
|
input: "--yes",
|
|
output: DeleteOptions{Confirmed: false}, // --yes should be ignored
|
|
wantErr: false,
|
|
wantStderr: "Warning: `--yes` is ignored since no repository was specified\n",
|
|
},
|
|
{
|
|
name: "yes flag error when no argument notty",
|
|
input: "--yes",
|
|
wantErr: true,
|
|
wantErrMsg: "cannot non-interactively delete current repository. Please specify a repository or run interactively",
|
|
},
|
|
{
|
|
name: "confirm flag error when no argument notty",
|
|
input: "--confirm",
|
|
wantErr: true,
|
|
wantErrMsg: "cannot non-interactively delete current repository. Please specify a repository or run interactively",
|
|
},
|
|
{
|
|
name: "confirm flag also ignored when no argument tty",
|
|
tty: true,
|
|
input: "--confirm",
|
|
output: DeleteOptions{Confirmed: false}, // --confirm should also be ignored
|
|
wantStderr: "Warning: `--yes` is ignored since no repository was specified\n",
|
|
// Note: This does not confuse the user, as deprecation warnings are shown: "Flag --confirm has been deprecated, use `--yes` instead"
|
|
},
|
|
}
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
ios, _, _, stdErr := iostreams.Test()
|
|
ios.SetStdinTTY(tt.tty)
|
|
ios.SetStdoutTTY(tt.tty)
|
|
f := &cmdutil.Factory{
|
|
IOStreams: ios,
|
|
}
|
|
argv, err := shlex.Split(tt.input)
|
|
assert.NoError(t, err)
|
|
var gotOpts *DeleteOptions
|
|
cmd := NewCmdDelete(f, func(opts *DeleteOptions) error {
|
|
gotOpts = opts
|
|
return nil
|
|
})
|
|
cmd.SetArgs(argv)
|
|
cmd.SetIn(&bytes.Buffer{})
|
|
cmd.SetOut(&bytes.Buffer{})
|
|
cmd.SetErr(stdErr)
|
|
|
|
_, err = cmd.ExecuteC()
|
|
|
|
if tt.wantErr {
|
|
assert.Error(t, err)
|
|
assert.Equal(t, tt.wantErrMsg, err.Error())
|
|
return
|
|
}
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, tt.wantStderr, stdErr.String())
|
|
assert.Equal(t, tt.output.RepoArg, gotOpts.RepoArg)
|
|
})
|
|
}
|
|
}
|
|
|
|
func Test_deleteRun(t *testing.T) {
|
|
tests := []struct {
|
|
name string
|
|
tty bool
|
|
opts *DeleteOptions
|
|
httpStubs func(*httpmock.Registry)
|
|
prompterStubs func(*prompter.PrompterMock)
|
|
wantStdout string
|
|
wantStderr string
|
|
wantErr bool
|
|
errMsg string
|
|
}{
|
|
{
|
|
name: "prompting confirmation tty",
|
|
tty: true,
|
|
opts: &DeleteOptions{RepoArg: "OWNER/REPO"},
|
|
wantStdout: "✓ Deleted repository OWNER/REPO\n",
|
|
prompterStubs: func(p *prompter.PrompterMock) {
|
|
p.ConfirmDeletionFunc = func(_ string) error {
|
|
return nil
|
|
}
|
|
},
|
|
httpStubs: func(reg *httpmock.Registry) {
|
|
reg.Register(
|
|
httpmock.REST("DELETE", "repos/OWNER/REPO"),
|
|
httpmock.StatusStringResponse(204, "{}"))
|
|
},
|
|
},
|
|
{
|
|
name: "infer base repo",
|
|
tty: true,
|
|
opts: &DeleteOptions{},
|
|
wantStdout: "✓ Deleted repository OWNER/REPO\n",
|
|
prompterStubs: func(p *prompter.PrompterMock) {
|
|
p.ConfirmDeletionFunc = func(_ string) error {
|
|
return nil
|
|
}
|
|
},
|
|
httpStubs: func(reg *httpmock.Registry) {
|
|
reg.Register(
|
|
httpmock.REST("DELETE", "repos/OWNER/REPO"),
|
|
httpmock.StatusStringResponse(204, "{}"))
|
|
},
|
|
},
|
|
{
|
|
name: "confirmation no tty",
|
|
opts: &DeleteOptions{
|
|
RepoArg: "OWNER/REPO",
|
|
Confirmed: true,
|
|
},
|
|
httpStubs: func(reg *httpmock.Registry) {
|
|
reg.Register(
|
|
httpmock.REST("DELETE", "repos/OWNER/REPO"),
|
|
httpmock.StatusStringResponse(204, "{}"))
|
|
},
|
|
},
|
|
{
|
|
name: "short repo name",
|
|
opts: &DeleteOptions{RepoArg: "REPO"},
|
|
wantStdout: "✓ Deleted repository OWNER/REPO\n",
|
|
tty: true,
|
|
prompterStubs: func(p *prompter.PrompterMock) {
|
|
p.ConfirmDeletionFunc = func(_ string) error {
|
|
return nil
|
|
}
|
|
},
|
|
httpStubs: func(reg *httpmock.Registry) {
|
|
reg.Register(
|
|
httpmock.GraphQL(`query UserCurrent\b`),
|
|
httpmock.StringResponse(`{"data":{"viewer":{"login":"OWNER"}}}`))
|
|
reg.Register(
|
|
httpmock.REST("DELETE", "repos/OWNER/REPO"),
|
|
httpmock.StatusStringResponse(204, "{}"))
|
|
},
|
|
},
|
|
{
|
|
name: "repo transferred ownership",
|
|
opts: &DeleteOptions{RepoArg: "OWNER/REPO", Confirmed: true},
|
|
wantErr: true,
|
|
errMsg: "SilentError",
|
|
wantStderr: "X Failed to delete repository: OWNER/REPO has changed name or transferred ownership\n",
|
|
httpStubs: func(reg *httpmock.Registry) {
|
|
reg.Register(
|
|
httpmock.REST("DELETE", "repos/OWNER/REPO"),
|
|
httpmock.StatusStringResponse(307, "{}"))
|
|
},
|
|
},
|
|
}
|
|
for _, tt := range tests {
|
|
pm := &prompter.PrompterMock{}
|
|
if tt.prompterStubs != nil {
|
|
tt.prompterStubs(pm)
|
|
}
|
|
tt.opts.Prompter = pm
|
|
|
|
tt.opts.BaseRepo = func() (ghrepo.Interface, error) {
|
|
return ghrepo.New("OWNER", "REPO"), nil
|
|
}
|
|
|
|
reg := &httpmock.Registry{}
|
|
if tt.httpStubs != nil {
|
|
tt.httpStubs(reg)
|
|
}
|
|
tt.opts.HttpClient = func() (*http.Client, error) {
|
|
return &http.Client{Transport: reg}, nil
|
|
}
|
|
|
|
ios, _, stdout, stderr := iostreams.Test()
|
|
ios.SetStdinTTY(tt.tty)
|
|
ios.SetStdoutTTY(tt.tty)
|
|
tt.opts.IO = ios
|
|
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
defer reg.Verify(t)
|
|
err := deleteRun(tt.opts)
|
|
if tt.wantErr {
|
|
assert.Error(t, err)
|
|
assert.Equal(t, tt.errMsg, err.Error())
|
|
return
|
|
}
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, tt.wantStdout, stdout.String())
|
|
assert.Equal(t, tt.wantStderr, stderr.String())
|
|
})
|
|
}
|
|
}
|