Skip to content

Commit

Permalink
net/http: make Request.Clone create fresh copies for matches and othe…
Browse files Browse the repository at this point in the history
…rValues

This change fixes Request.Clone to correctly work with SetPathValue
by creating fresh copies for matches and otherValues so that
SetPathValue for cloned requests doesn't pollute the original request.

While here, also added a doc for Request.SetPathValue.

Fixes golang#64911

Change-Id: I2831b38e135935dfaea2b939bb9db554c75b65ef
GitHub-Last-Rev: 1981db1
GitHub-Pull-Request: golang#64913
Reviewed-on: https://go-review.googlesource.com/c/go/+/553375
Reviewed-by: Emmanuel Odeke <[email protected]>
Run-TryBot: Jes Cok <[email protected]>
Auto-Submit: Dmitri Shuralyov <[email protected]>
Reviewed-by: Jonathan Amsterdam <[email protected]>
TryBot-Result: Gopher Robot <[email protected]>
Reviewed-by: Dmitri Shuralyov <[email protected]>
  • Loading branch information
callthingsoff authored and ezz-no committed Feb 17, 2024
1 parent 90284fc commit 23a5fef
Show file tree
Hide file tree
Showing 2 changed files with 43 additions and 0 deletions.
16 changes: 16 additions & 0 deletions src/net/http/request.go
Original file line number Diff line number Diff line change
Expand Up @@ -397,6 +397,20 @@ func (r *Request) Clone(ctx context.Context) *Request {
r2.Form = cloneURLValues(r.Form)
r2.PostForm = cloneURLValues(r.PostForm)
r2.MultipartForm = cloneMultipartForm(r.MultipartForm)

// Copy matches and otherValues. See issue 61410.
if s := r.matches; s != nil {
s2 := make([]string, len(s))
copy(s2, s)
r2.matches = s2
}
if s := r.otherValues; s != nil {
s2 := make(map[string]string, len(s))
for k, v := range s {
s2[k] = v
}
r2.otherValues = s2
}
return r2
}

Expand Down Expand Up @@ -1427,6 +1441,8 @@ func (r *Request) PathValue(name string) string {
return r.otherValues[name]
}

// SetPathValue sets name to value, so that subsequent calls to r.PathValue(name)
// return value.
func (r *Request) SetPathValue(name, value string) {
if i := r.patIndex(name); i >= 0 {
r.matches[i] = value
Expand Down
27 changes: 27 additions & 0 deletions src/net/http/request_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -1053,6 +1053,33 @@ func TestRequestCloneTransferEncoding(t *testing.T) {
}
}

// Ensure that Request.Clone works correctly with PathValue.
// See issue 64911.
func TestRequestClonePathValue(t *testing.T) {
req, _ := http.NewRequest("GET", "https://example.org/", nil)
req.SetPathValue("p1", "orig")

clonedReq := req.Clone(context.Background())
clonedReq.SetPathValue("p2", "copy")

// Ensure that any modifications to the cloned
// request do not pollute the original request.
if g, w := req.PathValue("p2"), ""; g != w {
t.Fatalf("p2 mismatch got %q, want %q", g, w)
}
if g, w := req.PathValue("p1"), "orig"; g != w {
t.Fatalf("p1 mismatch got %q, want %q", g, w)
}

// Assert on the changes to the cloned request.
if g, w := clonedReq.PathValue("p1"), "orig"; g != w {
t.Fatalf("p1 mismatch got %q, want %q", g, w)
}
if g, w := clonedReq.PathValue("p2"), "copy"; g != w {
t.Fatalf("p2 mismatch got %q, want %q", g, w)
}
}

// Issue 34878: verify we don't panic when including basic auth (Go 1.13 regression)
func TestNoPanicOnRoundTripWithBasicAuth(t *testing.T) { run(t, testNoPanicWithBasicAuth) }
func testNoPanicWithBasicAuth(t *testing.T, mode testMode) {
Expand Down

0 comments on commit 23a5fef

Please sign in to comment.