Skip to content

Commit

Permalink
Adjust tests and minor implementation details
Browse files Browse the repository at this point in the history
  • Loading branch information
williammartin committed May 23, 2024
1 parent 14196b4 commit b77eef3
Show file tree
Hide file tree
Showing 2 changed files with 113 additions and 92 deletions.
28 changes: 16 additions & 12 deletions pkg/cmd/variable/get/get.go
Original file line number Diff line number Diff line change
@@ -1,9 +1,9 @@
package get

Check failure on line 1 in pkg/cmd/variable/get/get.go

View workflow job for this annotation

GitHub Actions / lint

: # github.com/cli/cli/v2/pkg/cmd/variable/get [github.com/cli/cli/v2/pkg/cmd/variable/get.test]

import (
"errors"
"fmt"
"net/http"
"time"

"github.com/MakeNowJust/heredoc"
"github.com/cli/cli/v2/api"
Expand All @@ -26,13 +26,14 @@ type GetOptions struct {
EnvName string
}

type Variable struct {
Name string `json:"name"`
Value string `json:"value"`
UpdatedAt time.Time `json:"updated_at"`
Visibility shared.Visibility `json:"visibility"`
SelectedReposURL string `json:"selected_repositories_url"`
NumSelectedRepos int `json:"num_selected_repos"`
type GetVariableResponse struct {
Value string `json:"value"`
// Other available but unused fields
// Name string `json:"name"`
// UpdatedAt time.Time `json:"updated_at"`
// Visibility shared.Visibility `json:"visibility"`
// SelectedReposURL string `json:"selected_repositories_url"`
// NumSelectedRepos int `json:"num_selected_repos"`
}

func NewCmdGet(f *cmdutil.Factory, runF func(*GetOptions) error) *cobra.Command {
Expand Down Expand Up @@ -115,14 +116,17 @@ func getRun(opts *GetOptions) error {

host, _ := cfg.Authentication().DefaultHost()

response := &Variable{}
var response GetVariableResponse
if err = client.REST(host, "GET", path, nil, &response); err != nil {
var httpErr api.HTTPError
if errors.As(err, &httpErr) && httpErr.StatusCode == http.StatusNotFound {
return fmt.Errorf("variable %s was not found", opts.VariableName)
}

err = client.REST(host, "GET", path, nil, &response)
if err != nil {
return fmt.Errorf("failed to get variable %s: %w", opts.VariableName, err)
}

fmt.Fprintf(opts.IO.Out, "%s", response.Value)
fmt.Fprintf(opts.IO.Out, "%s\n", response.Value)

return nil
}
177 changes: 97 additions & 80 deletions pkg/cmd/variable/get/get_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -14,13 +14,15 @@ import (
"github.com/cli/cli/v2/pkg/iostreams"
"github.com/google/shlex"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
)

func TestNewCmdList(t *testing.T) {
func TestNewCmdGet(t *testing.T) {
tests := []struct {
name string
cli string
wants GetOptions
name string
cli string
wants GetOptions
wantErr error
}{
{
name: "repo",
Expand All @@ -32,20 +34,25 @@ func TestNewCmdList(t *testing.T) {
},
{
name: "org",
cli: "-oUmbrellaCorporation BAR",
cli: "-o TestOrg BAR",
wants: GetOptions{
OrgName: "UmbrellaCorporation",
OrgName: "TestOrg",
VariableName: "BAR",
},
},
{
name: "env",
cli: "-eDevelopment BAZ",
cli: "-e Development BAZ",
wants: GetOptions{
EnvName: "Development",
VariableName: "BAZ",
},
},
{
name: "org and env",
cli: "-o TestOrg -e Development QUX",
wantErr: cmdutil.FlagErrorf("%s", "specify only one of `--org` or `--env`"),
},
}

for _, tt := range tests {
Expand All @@ -69,120 +76,130 @@ func TestNewCmdList(t *testing.T) {
cmd.SetErr(&bytes.Buffer{})

_, err = cmd.ExecuteC()
assert.NoError(t, err)
if tt.wantErr != nil {
require.Equal(t, err, tt.wantErr)
return
}
require.NoError(t, err)

assert.Equal(t, tt.wants.OrgName, gotOpts.OrgName)
assert.Equal(t, tt.wants.EnvName, gotOpts.EnvName)
assert.Equal(t, tt.wants.VariableName, gotOpts.VariableName)
require.Equal(t, tt.wants.OrgName, gotOpts.OrgName)
require.Equal(t, tt.wants.EnvName, gotOpts.EnvName)
require.Equal(t, tt.wants.VariableName, gotOpts.VariableName)
})
}
}

func Test_getRun(t *testing.T) {
tests := []struct {
name string
tty bool
opts *GetOptions
wantOut string
name string
opts *GetOptions
httpStubs func(*httpmock.Registry)
wantOut string
wantErr error
}{
{
name: "repo tty",
tty: true,
name: "getting repo variable",
opts: &GetOptions{
VariableName: "VARIABLE_ONE",
},
wantOut: "one",
},
{
name: "repo not tty",
tty: false,
opts: &GetOptions{
VariableName: "VARIABLE_ONE",
httpStubs: func(reg *httpmock.Registry) {
reg.Register(httpmock.REST("GET", "repos/owner/repo/actions/variables/VARIABLE_ONE"),
httpmock.JSONResponse(GetVariableResponse{
Name: "VARIABLE_ONE",

Check failure on line 108 in pkg/cmd/variable/get/get_test.go

View workflow job for this annotation

GitHub Actions / lint

unknown field Name in struct literal of type GetVariableResponse

Check failure on line 108 in pkg/cmd/variable/get/get_test.go

View workflow job for this annotation

GitHub Actions / build (ubuntu-latest)

unknown field Name in struct literal of type GetVariableResponse

Check failure on line 108 in pkg/cmd/variable/get/get_test.go

View workflow job for this annotation

GitHub Actions / build (windows-latest)

unknown field Name in struct literal of type GetVariableResponse

Check failure on line 108 in pkg/cmd/variable/get/get_test.go

View workflow job for this annotation

GitHub Actions / build (macos-latest)

unknown field Name in struct literal of type GetVariableResponse
Value: "repo_var",
}))
},
wantOut: "one",
wantOut: "repo_var\n",
},
{
name: "org tty",
tty: true,
name: "getting org variable",
opts: &GetOptions{
OrgName: "UmbrellaCorporation",
OrgName: "TestOrg",
VariableName: "VARIABLE_ONE",
},
wantOut: "org_one",
httpStubs: func(reg *httpmock.Registry) {
reg.Register(httpmock.REST("GET", "orgs/TestOrg/actions/variables/VARIABLE_ONE"),
httpmock.JSONResponse(GetVariableResponse{
Name: "VARIABLE_ONE",

Check failure on line 123 in pkg/cmd/variable/get/get_test.go

View workflow job for this annotation

GitHub Actions / lint

unknown field Name in struct literal of type GetVariableResponse

Check failure on line 123 in pkg/cmd/variable/get/get_test.go

View workflow job for this annotation

GitHub Actions / build (ubuntu-latest)

unknown field Name in struct literal of type GetVariableResponse

Check failure on line 123 in pkg/cmd/variable/get/get_test.go

View workflow job for this annotation

GitHub Actions / build (windows-latest)

unknown field Name in struct literal of type GetVariableResponse

Check failure on line 123 in pkg/cmd/variable/get/get_test.go

View workflow job for this annotation

GitHub Actions / build (macos-latest)

unknown field Name in struct literal of type GetVariableResponse
Value: "org_var",
}))
},
wantOut: "org_var\n",
},
{
name: "org not tty",
tty: false,
name: "getting env variable",
opts: &GetOptions{
OrgName: "UmbrellaCorporation",
EnvName: "Development",
VariableName: "VARIABLE_ONE",
},
wantOut: "org_one",
httpStubs: func(reg *httpmock.Registry) {
reg.Register(httpmock.REST("GET", "repos/owner/repo/environments/Development/variables/VARIABLE_ONE"),
httpmock.JSONResponse(GetVariableResponse{
Name: "VARIABLE_ONE",

Check failure on line 138 in pkg/cmd/variable/get/get_test.go

View workflow job for this annotation

GitHub Actions / lint

unknown field Name in struct literal of type GetVariableResponse (typecheck)

Check failure on line 138 in pkg/cmd/variable/get/get_test.go

View workflow job for this annotation

GitHub Actions / build (ubuntu-latest)

unknown field Name in struct literal of type GetVariableResponse

Check failure on line 138 in pkg/cmd/variable/get/get_test.go

View workflow job for this annotation

GitHub Actions / build (windows-latest)

unknown field Name in struct literal of type GetVariableResponse

Check failure on line 138 in pkg/cmd/variable/get/get_test.go

View workflow job for this annotation

GitHub Actions / build (macos-latest)

unknown field Name in struct literal of type GetVariableResponse
Value: "env_var",
}))
},
wantOut: "env_var\n",
},
{
name: "env tty",
tty: true,
name: "when the variable is not found, an error is returned",
opts: &GetOptions{
EnvName: "Development",
VariableName: "VARIABLE_ONE",
},
wantOut: "one",
httpStubs: func(reg *httpmock.Registry) {
reg.Register(httpmock.REST("GET", "repos/owner/repo/actions/variables/VARIABLE_ONE"),
httpmock.StatusStringResponse(404, "not found"),
)
},
wantErr: fmt.Errorf("variable VARIABLE_ONE was not found"),
},
{
name: "env not tty",
tty: false,
name: "when getting any variable from API fails, the error is bubbled with context",
opts: &GetOptions{
EnvName: "Development",
VariableName: "VARIABLE_ONE",
},
wantOut: "one",
httpStubs: func(reg *httpmock.Registry) {
reg.Register(httpmock.REST("GET", "repos/owner/repo/actions/variables/VARIABLE_ONE"),
httpmock.StatusStringResponse(400, "not found"),
)
},
wantErr: fmt.Errorf("failed to get variable VARIABLE_ONE: HTTP 400 (https://api.github.com/repos/owner/repo/actions/variables/VARIABLE_ONE)"),
},
}

for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
reg := &httpmock.Registry{}
defer reg.Verify(t)

path := fmt.Sprintf("repos/owner/repo/actions/variables/%s", tt.opts.VariableName)
if tt.opts.EnvName != "" {
path = fmt.Sprintf("repos/owner/repo/environments/%s/variables/%s", tt.opts.EnvName, tt.opts.VariableName)
} else if tt.opts.OrgName != "" {
path = fmt.Sprintf("orgs/%s/actions/variables/%s", tt.opts.OrgName, tt.opts.VariableName)
}

payload := Variable{
Name: "VARIABLE_ONE",
Value: "one",
}
if tt.opts.OrgName != "" {
payload = Variable{
Name: "VARIABLE_ONE",
Value: "org_one",
var runTest = func(tty bool) func(t *testing.T) {
return func(t *testing.T) {
reg := &httpmock.Registry{}
tt.httpStubs(reg)
defer reg.Verify(t)

ios, _, stdout, _ := iostreams.Test()
ios.SetStdoutTTY(tty)

tt.opts.IO = ios
tt.opts.BaseRepo = func() (ghrepo.Interface, error) {
return ghrepo.FromFullName("owner/repo")
}
tt.opts.HttpClient = func() (*http.Client, error) {
return &http.Client{Transport: reg}, nil
}
tt.opts.Config = func() (gh.Config, error) {
return config.NewBlankConfig(), nil
}
}

reg.Register(httpmock.REST("GET", path), httpmock.JSONResponse(payload))

ios, _, stdout, _ := iostreams.Test()

ios.SetStdoutTTY(tt.tty)
err := getRun(tt.opts)
if err != nil {
require.EqualError(t, tt.wantErr, err.Error())
return
}

tt.opts.IO = ios
tt.opts.BaseRepo = func() (ghrepo.Interface, error) {
return ghrepo.FromFullName("owner/repo")
}
tt.opts.HttpClient = func() (*http.Client, error) {
return &http.Client{Transport: reg}, nil
require.NoError(t, err)
require.Equal(t, tt.wantOut, stdout.String())
}
tt.opts.Config = func() (gh.Config, error) {
return config.NewBlankConfig(), nil
}

err := getRun(tt.opts)
assert.NoError(t, err)
}

assert.Equal(t, tt.wantOut, stdout.String())
})
t.Run(tt.name+" tty", runTest(true))
t.Run(tt.name+" no-tty", runTest(false))
}
}

0 comments on commit b77eef3

Please sign in to comment.