Skip to content

Commit

Permalink
Fix signer receiving a drained body on retries
Browse files Browse the repository at this point in the history
  • Loading branch information
aouji committed Sep 21, 2024
1 parent bd5d233 commit 5aafd47
Show file tree
Hide file tree
Showing 2 changed files with 47 additions and 4 deletions.
8 changes: 4 additions & 4 deletions opensearchtransport/opensearchtransport.go
Original file line number Diff line number Diff line change
Expand Up @@ -292,10 +292,6 @@ func (c *Client) Perform(req *http.Request) (*http.Response, error) {
c.setReqURL(conn.URL, req)
c.setReqAuth(conn.URL, req)

if err = c.signRequest(req); err != nil {
return nil, fmt.Errorf("failed to sign request: %w", err)
}

if !c.disableRetry && i > 0 && req.Body != nil && req.Body != http.NoBody {
body, err := req.GetBody()
if err != nil {
Expand All @@ -304,6 +300,10 @@ func (c *Client) Perform(req *http.Request) (*http.Response, error) {
req.Body = body
}

if err = c.signRequest(req); err != nil {
return nil, fmt.Errorf("failed to sign request: %w", err)
}

// Set up time measures and execute the request
start := time.Now().UTC()
res, err = c.transport.RoundTrip(req)
Expand Down
43 changes: 43 additions & 0 deletions opensearchtransport/opensearchtransport_internal_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -64,9 +64,11 @@ type mockSigner struct {
SampleKey string
SampleValue string
ReturnError bool
testHook func(*http.Request)
}

func (m *mockSigner) SignRequest(req *http.Request) error {
m.testHook(req)
if m.ReturnError {
return fmt.Errorf("invalid data")
}
Expand Down Expand Up @@ -732,6 +734,47 @@ func TestTransportPerformRetries(t *testing.T) {
}
})

t.Run("Signer can sign correctly during retry", func(t *testing.T) {
u, _ := url.Parse("https://foo.com/bar")
signer := mockSigner{}
callsToSigner := 0
expectedBody := "FOOBAR"

signer.testHook = func(req *http.Request) {
callsToSigner++
body, err := io.ReadAll(req.Body)
if err != nil {
panic(err)
}
if string(body) != expectedBody {
t.Fatalf("request %d body: expected %q, got %q", callsToSigner, expectedBody, body)
}
}

tp, _ := New(
Config{
URLs: []*url.URL{u},
Signer: &signer,
Transport: &mockTransp{
RoundTripFunc: func(req *http.Request) (*http.Response, error) {
return &http.Response{Status: "MOCK", StatusCode: http.StatusBadGateway}, nil
},
},
},
)

req, _ := http.NewRequest(http.MethodPost, "/abc", strings.NewReader(expectedBody))
//nolint:bodyclose // Mock response does not have a body to close
_, err := tp.Perform(req)
if err != nil {
t.Fatalf("Unexpected error: %s", err)
}

if callsToSigner != 4 {
t.Fatalf("expected 4 requests, got %d", callsToSigner)
}
})

t.Run("Don't retry request on regular error", func(t *testing.T) {
var i int

Expand Down

0 comments on commit 5aafd47

Please sign in to comment.