-
Notifications
You must be signed in to change notification settings - Fork 120
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Refresh token and retry request on 401, in :curl commands
- Loading branch information
1 parent
b8245f5
commit 813c704
Showing
7 changed files
with
195 additions
and
58 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -19,3 +19,6 @@ indent_size = 2 | |
|
||
[*.yml] | ||
indent_size = 2 | ||
|
||
[*.go] | ||
indent_style = tab |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,63 @@ | ||
package tests | ||
|
||
import ( | ||
"encoding/json" | ||
"net/http" | ||
"net/http/httptest" | ||
"strings" | ||
"testing" | ||
|
||
"github.com/go-chi/chi/v5" | ||
"github.com/go-chi/chi/v5/middleware" | ||
"github.com/stretchr/testify/assert" | ||
) | ||
|
||
func TestApiCurlCommand(t *testing.T) { | ||
validToken := "valid-token" | ||
|
||
mux := chi.NewMux() | ||
if testing.Verbose() { | ||
mux.Use(middleware.DefaultLogger) | ||
} | ||
mux.Use(func(next http.Handler) http.Handler { | ||
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { | ||
if !strings.HasPrefix(r.URL.Path, "/oauth2") { | ||
if r.Header.Get("Authorization") != "Bearer "+validToken { | ||
w.WriteHeader(http.StatusUnauthorized) | ||
return | ||
} | ||
} | ||
next.ServeHTTP(w, r) | ||
}) | ||
}) | ||
var tokenFetches int | ||
mux.Post("/oauth2/token", func(w http.ResponseWriter, r *http.Request) { | ||
w.WriteHeader(http.StatusOK) | ||
tokenFetches++ | ||
_ = json.NewEncoder(w).Encode(map[string]any{"access_token": validToken, "expires_in": 900, "token_type": "bearer"}) | ||
}) | ||
mux.Get("/users/me", func(w http.ResponseWriter, r *http.Request) { | ||
_ = json.NewEncoder(w).Encode(map[string]any{"id": "userID", "email": "[email protected]"}) | ||
}) | ||
mux.Get("/fake-api-path", func(w http.ResponseWriter, r *http.Request) { | ||
_, _ = w.Write([]byte("success")) | ||
}) | ||
mockServer := httptest.NewServer(mux) | ||
defer mockServer.Close() | ||
|
||
run := runnerWithAuth(t, mockServer.URL, mockServer.URL) | ||
|
||
// Load the first token. | ||
assert.Equal(t, "success", run("api:curl", "/fake-api-path")) | ||
assert.Equal(t, 1, tokenFetches) | ||
|
||
// Revoke the access token and try the command again. | ||
// The old token should be considered invalid, so the API call should return 401, | ||
// and then the CLI should refresh the token and retry. | ||
validToken = "new-valid-token" | ||
assert.Equal(t, "success", run("api:curl", "/fake-api-path")) | ||
assert.Equal(t, 2, tokenFetches) | ||
|
||
assert.Equal(t, "success", run("api:curl", "/fake-api-path")) | ||
assert.Equal(t, 2, tokenFetches) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,20 +1,32 @@ | ||
github.com/creack/pty v1.1.9/go.mod h1:oKZEueFk5CKHvIhNR5MUki03XCEU+Q6VDXinZuGJ33E= | ||
github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc h1:U9qPSI2PIWSS1VwoXQT9A3Wy9MM3WgvqSxFWenqJduM= | ||
github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | ||
github.com/go-chi/chi/v5 v5.1.0 h1:acVI1TYaD+hhedDJ3r54HyA6sExp3HfXq7QWEEY/xMw= | ||
github.com/go-chi/chi/v5 v5.1.0/go.mod h1:DslCQbL2OYiznFReuXYUmQ2hGd1aDpCnlMNITLSKoi8= | ||
github.com/kr/pretty v0.3.1 h1:flRD4NNwYAUpkphVc1HcthR4KEIFJ65n8Mw5qdRn3LE= | ||
github.com/kr/pretty v0.3.1/go.mod h1:hoEshYVHaxMs3cyo3Yncou5ZscifuDolrwPKZanG3xk= | ||
github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= | ||
github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= | ||
github.com/oklog/ulid/v2 v2.1.0 h1:+9lhoxAP56we25tyYETBBY1YLA2SaoLvUFgrP2miPJU= | ||
github.com/oklog/ulid/v2 v2.1.0/go.mod h1:rcEKHmBBKfef9DhnvX7y1HZBYxjXb0cP5ExxNsTT1QQ= | ||
github.com/pborman/getopt v0.0.0-20170112200414-7148bc3a4c30/go.mod h1:85jBQOZwpVEaDAr341tbn15RS4fCAsIst0qp7i8ex1o= | ||
github.com/pkg/diff v0.0.0-20210226163009-20ebb0f2a09e/go.mod h1:pJLUxLENpZxwdsKMEsNbx1VGcRFpLqf3715MtcvvzbA= | ||
github.com/platformsh/cli v0.0.0-20241126124927-2e901f7c6a3b h1:Pbyjf2FNzShe71EJnG/8ezhUl63RjkC8y7VGmBGmIXM= | ||
github.com/platformsh/cli v0.0.0-20241126124927-2e901f7c6a3b/go.mod h1:j9Aj8DxVGyn+Jm3ntopLnk6p0XtOeLWVBpF3zhqHh7M= | ||
github.com/pmezard/go-difflib v1.0.1-0.20181226105442-5d4384ee4fb2 h1:Jamvg5psRIccs7FGNTlIRMkT8wgtp5eCXdBlqhYGL6U= | ||
github.com/pmezard/go-difflib v1.0.1-0.20181226105442-5d4384ee4fb2/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= | ||
github.com/rogpeppe/go-internal v1.9.0 h1:73kH8U+JUqXU8lRuOHeVHaa/SZPifC7BkcraZVejAe8= | ||
github.com/rogpeppe/go-internal v1.9.0/go.mod h1:WtVeX8xhTBvf0smdhujwtBcq4Qrzq/fJaraNFVN+nFs= | ||
github.com/stretchr/testify v1.9.0 h1:HtqpIVDClZ4nwg75+f6Lvsy/wHu+3BoSGCbBAcpTsTg= | ||
github.com/stretchr/testify v1.9.0/go.mod h1:r2ic/lqez/lEtzL7wO/rwa5dbSLXVDPFyf8C91i36aY= | ||
golang.org/x/crypto v0.24.0 h1:mnl8DM0o513X8fdIkmyFE/5hTYxbwYOjDS/+rK6qpRI= | ||
golang.org/x/crypto v0.24.0/go.mod h1:Z1PMYSOR5nyMcyAVAIQSKCDwalqy85Aqn1x3Ws4L5DM= | ||
golang.org/x/sys v0.21.0 h1:rF+pYz3DAGSQAxAu1CbC7catZg4ebC4UIeIhKxBZvws= | ||
golang.org/x/sys v0.21.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= | ||
golang.org/x/term v0.21.0 h1:WVXCp+/EBEHOj53Rvu+7KiT/iElMrO8ACK16SMZ3jaA= | ||
golang.org/x/term v0.21.0/go.mod h1:ooXLefLobQVslOqselCNF4SxFAaoS6KujMbsGzSDmX0= | ||
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||
gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c h1:Hei/4ADfdWqJk1ZMxUNpqntNwaWcugrBjAiHlqqRiVk= | ||
gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c/go.mod h1:JHkPIbrfpd72SG/EVd6muEfDQjcINNoR0C8j2r3qZ4Q= | ||
gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA= | ||
gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,36 +1,36 @@ | ||
package tests | ||
|
||
import ( | ||
"net/http/httptest" | ||
"net/url" | ||
"strings" | ||
"testing" | ||
"net/http/httptest" | ||
"net/url" | ||
"strings" | ||
"testing" | ||
|
||
"github.com/stretchr/testify/assert" | ||
"github.com/stretchr/testify/assert" | ||
|
||
"github.com/platformsh/cli/pkg/mockapi" | ||
"github.com/platformsh/cli/pkg/mockapi" | ||
) | ||
|
||
func TestOrgList(t *testing.T) { | ||
authServer := mockapi.NewAuthServer(t) | ||
defer authServer.Close() | ||
authServer := mockapi.NewAuthServer(t) | ||
defer authServer.Close() | ||
|
||
myUserID := "user-id-1" | ||
myUserID := "user-id-1" | ||
|
||
apiHandler := mockapi.NewHandler(t) | ||
apiHandler.SetMyUser(&mockapi.User{ID: myUserID}) | ||
apiHandler.SetOrgs([]*mockapi.Org{ | ||
makeOrg("org-id-1", "acme", "ACME Inc.", myUserID), | ||
makeOrg("org-id-2", "four-seasons", "Four Seasons Total Landscaping", myUserID), | ||
makeOrg("org-id-3", "duff", "Duff Beer", "user-id-2"), | ||
}) | ||
apiHandler := mockapi.NewHandler(t) | ||
apiHandler.SetMyUser(&mockapi.User{ID: myUserID}) | ||
apiHandler.SetOrgs([]*mockapi.Org{ | ||
makeOrg("org-id-1", "acme", "ACME Inc.", myUserID), | ||
makeOrg("org-id-2", "four-seasons", "Four Seasons Total Landscaping", myUserID), | ||
makeOrg("org-id-3", "duff", "Duff Beer", "user-id-2"), | ||
}) | ||
|
||
apiServer := httptest.NewServer(apiHandler) | ||
defer apiServer.Close() | ||
apiServer := httptest.NewServer(apiHandler) | ||
defer apiServer.Close() | ||
|
||
run := runnerWithAuth(t, apiServer.URL, authServer.URL) | ||
run := runnerWithAuth(t, apiServer.URL, authServer.URL) | ||
|
||
assert.Equal(t, strings.TrimLeft(` | ||
assert.Equal(t, strings.TrimLeft(` | ||
+--------------+--------------------------------+-----------------------+ | ||
| Name | Label | Owner email | | ||
+--------------+--------------------------------+-----------------------+ | ||
|
@@ -40,27 +40,27 @@ func TestOrgList(t *testing.T) { | |
+--------------+--------------------------------+-----------------------+ | ||
`, "\n"), run("orgs")) | ||
|
||
assert.Equal(t, strings.TrimLeft(` | ||
assert.Equal(t, strings.TrimLeft(` | ||
Name Label Owner email | ||
acme ACME Inc. [email protected] | ||
duff Duff Beer [email protected] | ||
four-seasons Four Seasons Total Landscaping [email protected] | ||
`, "\n"), run("orgs", "--format", "plain")) | ||
|
||
assert.Equal(t, strings.TrimLeft(` | ||
assert.Equal(t, strings.TrimLeft(` | ||
org-id-1,acme | ||
org-id-3,duff | ||
org-id-2,four-seasons | ||
`, "\n"), run("orgs", "--format", "csv", "--columns", "id,name", "--no-header")) | ||
} | ||
|
||
func makeOrg(id, name, label, owner string) *mockapi.Org { | ||
return &mockapi.Org{ | ||
ID: id, | ||
Name: name, | ||
Label: label, | ||
Owner: owner, | ||
Capabilities: []string{}, | ||
Links: mockapi.MakeHALLinks("self=/organizations/" + url.PathEscape(id)), | ||
} | ||
return &mockapi.Org{ | ||
ID: id, | ||
Name: name, | ||
Label: label, | ||
Owner: owner, | ||
Capabilities: []string{}, | ||
Links: mockapi.MakeHALLinks("self=/organizations/" + url.PathEscape(id)), | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters