Skip to content

Commit e3466ad

Browse files
author
hidu
committed
can use the http.Request.PathValue method to read route variables when using go1.22 and above
1 parent db9d1d0 commit e3466ad

File tree

7 files changed

+50
-7
lines changed

7 files changed

+50
-7
lines changed

.github/workflows/security.yml

+1-1
Original file line numberDiff line numberDiff line change
@@ -12,7 +12,7 @@ jobs:
1212
scan:
1313
strategy:
1414
matrix:
15-
go: ['1.20','1.21']
15+
go: ['1.20','1.21','1.22']
1616
fail-fast: true
1717
runs-on: ubuntu-latest
1818
steps:

.github/workflows/test.yml

+1-1
Original file line numberDiff line numberDiff line change
@@ -12,7 +12,7 @@ jobs:
1212
unit:
1313
strategy:
1414
matrix:
15-
go: ['1.20','1.21']
15+
go: ['1.20','1.21','1.22']
1616
os: [ubuntu-latest, macos-latest, windows-latest]
1717
fail-fast: true
1818
runs-on: ${{ matrix.os }}

.github/workflows/verify.yml

+1-1
Original file line numberDiff line numberDiff line change
@@ -12,7 +12,7 @@ jobs:
1212
lint:
1313
strategy:
1414
matrix:
15-
go: ['1.20','1.21']
15+
go: ['1.20','1.21','1.22']
1616
fail-fast: true
1717
runs-on: ubuntu-latest
1818
steps:

README.md

+8
Original file line numberDiff line numberDiff line change
@@ -79,6 +79,14 @@ func ArticlesCategoryHandler(w http.ResponseWriter, r *http.Request) {
7979
}
8080
```
8181

82+
When using Go 1.22 and above, you also can use the `http.Request.PathValue` method to read route variables:
83+
```go
84+
func ArticlesCategoryHandler(w http.ResponseWriter, r *http.Request) {
85+
w.WriteHeader(http.StatusOK)
86+
fmt.Fprintf(w, "Category: %v\n", r.PathValue("category"))
87+
}
88+
```
89+
8290
And this is all you need to know about the basic usage. More advanced options are explained below.
8391

8492
### Matching Routes

doc.go

+5
Original file line numberDiff line numberDiff line change
@@ -62,6 +62,11 @@ this, convert any capturing groups to non-capturing, e.g. change "/{sort:(asc|de
6262
"/{sort:(?:asc|desc)}". This is a change from prior versions which behaved unpredictably
6363
when capturing groups were present.
6464
65+
When using Go 1.22 and above, you also can use the `http.Request.PathValue` method to
66+
read route variables:
67+
68+
r.PathValue("category")
69+
6570
And this is all you need to know about the basic usage. More advanced options
6671
are explained below.
6772

mux.go

+19
Original file line numberDiff line numberDiff line change
@@ -494,17 +494,36 @@ func requestWithVars(r *http.Request, vars map[string]string) *http.Request {
494494
if len(vars) == 0 {
495495
return r
496496
}
497+
var ro any = r
498+
if rr, ok := ro.(canSetPathValue); ok {
499+
for k, v := range vars {
500+
rr.SetPathValue(k, v)
501+
}
502+
}
497503
ctx := context.WithValue(r.Context(), varsKey, vars)
498504
return r.WithContext(ctx)
499505
}
500506

507+
// canSetPathValue is an interface introduced in Go 1.22.0.
508+
// The http.Request type added methods PathValue and SetPathValue,
509+
// which are used to handle named path wildcards in URL routes.
510+
type canSetPathValue interface {
511+
SetPathValue(name, value string)
512+
}
513+
501514
// requestWithRouteAndVars adds the matched route and vars to the request ctx.
502515
// It saves extra allocations in cloning the request once and skipping the
503516
//
504517
// population of empty vars, which in turn mux.Vars can handle gracefully.
505518
func requestWithRouteAndVars(r *http.Request, route *Route, vars map[string]string) *http.Request {
506519
ctx := context.WithValue(r.Context(), routeKey, route)
507520
if len(vars) > 0 {
521+
var ro any = r
522+
if rr, ok := ro.(canSetPathValue); ok {
523+
for k, v := range vars {
524+
rr.SetPathValue(k, v)
525+
}
526+
}
508527
ctx = context.WithValue(ctx, varsKey, vars)
509528
}
510529
return r.WithContext(ctx)

mux_test.go

+15-4
Original file line numberDiff line numberDiff line change
@@ -2970,17 +2970,28 @@ func TestContextMiddleware(t *testing.T) {
29702970

29712971
r := NewRouter()
29722972
r.Handle("/path/{foo}", withTimeout(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
2973-
vars := Vars(r)
2974-
if vars["foo"] != "bar" {
2975-
t.Fatal("Expected foo var to be set")
2976-
}
2973+
checkRouteVar(t, r, "foo", "bar")
29772974
})))
29782975

29792976
rec := NewRecorder()
29802977
req := newRequest("GET", "/path/bar")
29812978
r.ServeHTTP(rec, req)
29822979
}
29832980

2981+
func checkRouteVar(t *testing.T, r *http.Request, name string, want string) {
2982+
t.Helper()
2983+
vars := Vars(r)
2984+
if vars[name] != want {
2985+
t.Fatalf("Expected var %s = %q, but got %q", name, want, vars[name])
2986+
}
2987+
var ro any = r
2988+
if rr, ok := ro.(interface{ PathValue(string) string }); ok {
2989+
if val := rr.PathValue(name); val != want {
2990+
t.Fatalf("Expected r.PathValue(%s) = %q, but got %q", name, want, val)
2991+
}
2992+
}
2993+
}
2994+
29842995
func TestGetVarNames(t *testing.T) {
29852996
r := NewRouter()
29862997

0 commit comments

Comments
 (0)