Skip to content

Commit

Permalink
feat(httpclient): support appending the same header multiple times
Browse files Browse the repository at this point in the history
See #829
  • Loading branch information
sathieu committed Dec 5, 2023
1 parent fdcf6a0 commit 7067bef
Show file tree
Hide file tree
Showing 5 changed files with 24 additions and 20 deletions.
6 changes: 3 additions & 3 deletions backend/http_settings.go
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@ type HTTPSettings struct {
BasicAuthEnabled bool
BasicAuthUser string
BasicAuthPassword string
Headers map[string]string
Headers map[string][]string

Timeout time.Duration
DialTimeout time.Duration
Expand Down Expand Up @@ -104,7 +104,7 @@ func (s *HTTPSettings) HTTPClientOptions() httpclient.Options {
//gocyclo:ignore
func parseHTTPSettings(jsonData json.RawMessage, secureJSONData map[string]string) (*HTTPSettings, error) {
s := &HTTPSettings{
Headers: map[string]string{},
Headers: map[string][]string{},
}

var dat map[string]interface{}
Expand Down Expand Up @@ -273,7 +273,7 @@ func parseHTTPSettings(jsonData json.RawMessage, secureJSONData map[string]strin

if key, exists := dat[headerNameSuffix]; exists {
if value, exists := secureJSONData[headerValueSuffix]; exists {
s.Headers[key.(string)] = value
s.Headers[key.(string)] = append(s.Headers[key.(string)], value)
}
} else {
// No (more) header values are available
Expand Down
16 changes: 9 additions & 7 deletions backend/http_settings_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -44,7 +44,8 @@ func TestParseHTTPSettings(t *testing.T) {
"sigV4ExternalId": "ext123",
"sigV4Profile": "ghi",
"httpHeaderName1": "X-HeaderOne",
"httpHeaderName2": "X-HeaderTwo"
"httpHeaderName2": "X-HeaderTwo",
"httpHeaderName3": "X-HeaderTwo"
}`
secureData := map[string]string{
"basicAuthPassword": "pwd",
Expand All @@ -55,6 +56,7 @@ func TestParseHTTPSettings(t *testing.T) {
"sigV4SecretKey": "sigV4SecretKey5",
"httpHeaderValue1": "SecretOne",
"httpHeaderValue2": "SecretTwo",
"httpHeaderValue3": "SecretThree",
}
var jsonMap map[string]interface{}
err := json.Unmarshal([]byte(jsonStr), &jsonMap)
Expand All @@ -68,9 +70,9 @@ func TestParseHTTPSettings(t *testing.T) {
BasicAuthEnabled: true,
BasicAuthUser: "user",
BasicAuthPassword: "pwd",
Headers: map[string]string{
"X-HeaderOne": "SecretOne",
"X-HeaderTwo": "SecretTwo",
Headers: map[string][]string{
"X-HeaderOne": {"SecretOne"},
"X-HeaderTwo": {"SecretTwo", "SecretThree"},
},
Timeout: 10 * time.Second,
DialTimeout: 10 * time.Second,
Expand Down Expand Up @@ -108,9 +110,9 @@ func TestParseHTTPSettings(t *testing.T) {
User: "user",
Password: "pwd",
},
Headers: map[string]string{
"X-HeaderOne": "SecretOne",
"X-HeaderTwo": "SecretTwo",
Headers: map[string][]string{
"X-HeaderOne": {"SecretOne"},
"X-HeaderTwo": {"SecretTwo", "SecretThree"},
},
Timeouts: &httpclient.TimeoutOptions{
Timeout: 10 * time.Second,
Expand Down
6 changes: 4 additions & 2 deletions backend/httpclient/custom_headers_middleware.go
Original file line number Diff line number Diff line change
Expand Up @@ -17,8 +17,10 @@ func CustomHeadersMiddleware() Middleware {
}

return RoundTripperFunc(func(req *http.Request) (*http.Response, error) {
for key, value := range opts.Headers {
req.Header.Set(key, value)
for key, values := range opts.Headers {
for _, value := range values {
req.Header.Add(key, value)
}
}

return next.RoundTrip(req)
Expand Down
14 changes: 7 additions & 7 deletions backend/httpclient/custom_headers_middleware_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -34,10 +34,10 @@ func TestCustomHeadersMiddleware(t *testing.T) {
ctx := &testContext{}
finalRoundTripper := ctx.createRoundTripper("final")
customHeaders := CustomHeadersMiddleware()
rt := customHeaders.CreateMiddleware(Options{Headers: map[string]string{
"X-HeaderOne": "ValueOne",
"X-HeaderTwo": "ValueTwo",
"X-HeaderThree": "ValueThree",
rt := customHeaders.CreateMiddleware(Options{Headers: map[string][]string{
"X-HeaderOne": {"ValueOne"},
"X-HeaderTwo": {"ValueTwo"},
"X-HeaderThree": {"ValueThree", "ValueThreeAgain"},
}}, finalRoundTripper)
require.NotNil(t, rt)
middlewareName, ok := customHeaders.(MiddlewareName)
Expand All @@ -55,8 +55,8 @@ func TestCustomHeadersMiddleware(t *testing.T) {
require.Len(t, ctx.callChain, 1)
require.ElementsMatch(t, []string{"final"}, ctx.callChain)

require.Equal(t, "ValueOne", req.Header.Get("X-HeaderOne"))
require.Equal(t, "ValueTwo", req.Header.Get("X-HeaderTwo"))
require.Equal(t, "ValueThree", req.Header.Get("X-HeaderThree"))
require.Equal(t, []string{"ValueOne"}, req.Header.Values("X-HeaderOne"))
require.Equal(t, []string{"ValueTwo"}, req.Header.Values("X-HeaderTwo"))
require.Equal(t, []string{"ValueThree", "ValueThreeAgain"}, req.Header.Values("X-HeaderThree"))
})
}
2 changes: 1 addition & 1 deletion backend/httpclient/options.go
Original file line number Diff line number Diff line change
Expand Up @@ -38,7 +38,7 @@ type Options struct {
ProxyOptions *proxy.Options

// Headers custom headers.
Headers map[string]string
Headers map[string][]string

// CustomOptions allows custom options to be provided.
CustomOptions map[string]interface{}
Expand Down

0 comments on commit 7067bef

Please sign in to comment.