Skip to content

Commit

Permalink
http.go: Request.SetURI() (Fix valyala#1141) (valyala#1148)
Browse files Browse the repository at this point in the history
Currently, the only way to set URI for a request is to call SetRequestURI(string).
Then when a request performed the string will be parsed into a fasthttp.URI struct.
If there are many requests with the same URI then we'll waste CPU for a parsing of the same URI string.
With the new SetURI(*URI) method we can once parse a URI string into a fasthttp.URI struct and then reuse it for many requests.
Unfortunately the URI will be copied because may be modified inside the request.
But anyway this will be more lightweight than parsing.
  • Loading branch information
stokito authored Nov 8, 2021
1 parent 2ca01c7 commit 8febad0
Show file tree
Hide file tree
Showing 2 changed files with 39 additions and 0 deletions.
14 changes: 14 additions & 0 deletions http.go
Original file line number Diff line number Diff line change
Expand Up @@ -774,6 +774,20 @@ func (req *Request) URI() *URI {
return &req.uri
}

// SetURI initializes request URI
// Use this method if a single URI may be reused across multiple requests.
// Otherwise, you can just use SetRequestURI() and it will be parsed as new URI.
// The URI is copied and can be safely modified later.
func (req *Request) SetURI(newUri *URI) {
if newUri != nil {
newUri.CopyTo(&req.uri)
req.parsedURI = true
return
}
req.uri.Reset()
req.parsedURI = false
}

func (req *Request) parseURI() error {
if req.parsedURI {
return nil
Expand Down
25 changes: 25 additions & 0 deletions http_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -515,6 +515,31 @@ tailfoobar`
}
}

func TestRequestSetURI(t *testing.T) {
t.Parallel()

var r Request

uri := "/foo/bar?baz"
u := &URI{}
u.Parse(nil, []byte(uri)) //nolint:errcheck
// Set request uri via SetURI()
r.SetURI(u) // copies URI
// modifying an original URI struct doesn't affect stored URI inside of request
u.SetPath("newPath")
if string(r.RequestURI()) != uri {
t.Fatalf("unexpected request uri %q. Expecting %q", r.RequestURI(), uri)
}

// Set request uri to nil just resets the URI
r.Reset()
uri = "/"
r.SetURI(nil)
if string(r.RequestURI()) != uri {
t.Fatalf("unexpected request uri %q. Expecting %q", r.RequestURI(), uri)
}
}

func TestRequestRequestURI(t *testing.T) {
t.Parallel()

Expand Down

0 comments on commit 8febad0

Please sign in to comment.