From 527e3347f72be69d901526325700df708d21f093 Mon Sep 17 00:00:00 2001 From: Daniel Brondani Date: Mon, 16 Dec 2024 06:59:55 +0100 Subject: [PATCH] [cpackget] Handle `Cookie` headers in HTTP GET requests when required (#406) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit ## Fixes - https://github.com/Open-CMSIS-Pack/devtools/issues/1906 ## Changes - Resend HTTP GET request with a `Cookie` header if the first attempt fails with `StatusForbidden` status and a `Set-Cookie` among the response headers. ## Checklist - [x] ๐Ÿค– This change is covered by unit tests as required. - [x] ๐Ÿคน All required manual testing has been performed. - [x] ๐Ÿ›ก๏ธ Security impacts have been considered. - [x] ๐Ÿ“– All documentation updates are complete. - [x] ๐Ÿง  This change does not change third-party dependencies --- cmd/utils/utils.go | 14 ++++++++++++++ cmd/utils/utils_test.go | 29 +++++++++++++++++++++++++++++ 2 files changed, 43 insertions(+) diff --git a/cmd/utils/utils.go b/cmd/utils/utils.go index dc4ff8e..bc2471d 100644 --- a/cmd/utils/utils.go +++ b/cmd/utils/utils.go @@ -152,6 +152,20 @@ func DownloadFile(URL string, timeout int) (string, error) { } defer resp.Body.Close() + if resp.StatusCode == http.StatusForbidden { + cookie := resp.Header.Get("Set-Cookie") + if len(cookie) > 0 { + // add cookie and resend GET request + log.Debugf("Cookie: %s", cookie) + req.Header.Add("Cookie", cookie) + resp, err = client.Do(req) + if err != nil { + log.Error(err) + return "", fmt.Errorf("\"%s\": %w", URL, errs.ErrFailedDownloadingFile) + } + } + } + if resp.StatusCode != http.StatusOK { log.Debugf("bad status: %s", resp.Status) return "", fmt.Errorf("\"%s\": %w", URL, errs.ErrBadRequest) diff --git a/cmd/utils/utils_test.go b/cmd/utils/utils_test.go index 9994ba1..226ac90 100644 --- a/cmd/utils/utils_test.go +++ b/cmd/utils/utils_test.go @@ -114,6 +114,35 @@ func TestDownloadFile(t *testing.T) { assert.Equal(bytes, goodResponse) }) + t.Run("test download set cookie", func(t *testing.T) { + fileName := "file.txt" + defer os.Remove(fileName) + goodResponse := []byte("all good") + cookieContent := "cookie=test" + goodServer := httptest.NewServer( + http.HandlerFunc( + func(w http.ResponseWriter, r *http.Request) { + cookie := r.Header.Get("Cookie") + if len(cookie) > 0 { + if cookie == cookieContent { + fmt.Fprint(w, string(goodResponse)) + } + } else { + w.Header().Add("Set-Cookie", cookieContent) + w.WriteHeader(http.StatusForbidden) + } + }, + ), + ) + url := goodServer.URL + "/" + fileName + _, err1 := utils.DownloadFile(url, 0) + assert.Nil(err1) + assert.True(utils.FileExists(fileName)) + bytes, err2 := os.ReadFile(fileName) + assert.Nil(err2) + assert.Equal(bytes, goodResponse) + }) + t.Run("test download uses cache", func(t *testing.T) { fileName := "file.txt" defer os.Remove(fileName)