Skip to content

Commit 819aed3

Browse files
authored
Make route middleware/handler mockable (#25766)
To mock a handler: ```go web.RouteMock(web.MockAfterMiddlewares, func(ctx *context.Context) { // ... }) defer web.RouteMockReset() ``` It helps: * Test the middleware's behavior (assert the ctx.Data, etc) * Mock the middleware's behavior (prepare some context data for handler) * Mock the handler's response for some test cases, especially for some integration tests and e2e tests.
1 parent 887a683 commit 819aed3

File tree

3 files changed

+145
-4
lines changed

3 files changed

+145
-4
lines changed

modules/web/route.go

Lines changed: 14 additions & 4 deletions
Original file line numberDiff line numberDiff line change
@@ -50,7 +50,9 @@ func NewRoute() *Route {
5050
// Use supports two middlewares
5151
func (r *Route) Use(middlewares ...any) {
5252
for _, m := range middlewares {
53-
r.R.Use(toHandlerProvider(m))
53+
if m != nil {
54+
r.R.Use(toHandlerProvider(m))
55+
}
5456
}
5557
}
5658

@@ -79,15 +81,23 @@ func (r *Route) getPattern(pattern string) string {
7981
}
8082

8183
func (r *Route) wrapMiddlewareAndHandler(h []any) ([]func(http.Handler) http.Handler, http.HandlerFunc) {
82-
handlerProviders := make([]func(http.Handler) http.Handler, 0, len(r.curMiddlewares)+len(h))
84+
handlerProviders := make([]func(http.Handler) http.Handler, 0, len(r.curMiddlewares)+len(h)+1)
8385
for _, m := range r.curMiddlewares {
84-
handlerProviders = append(handlerProviders, toHandlerProvider(m))
86+
if m != nil {
87+
handlerProviders = append(handlerProviders, toHandlerProvider(m))
88+
}
8589
}
8690
for _, m := range h {
87-
handlerProviders = append(handlerProviders, toHandlerProvider(m))
91+
if h != nil {
92+
handlerProviders = append(handlerProviders, toHandlerProvider(m))
93+
}
8894
}
8995
middlewares := handlerProviders[:len(handlerProviders)-1]
9096
handlerFunc := handlerProviders[len(handlerProviders)-1](nil).ServeHTTP
97+
mockPoint := RouteMockPoint(MockAfterMiddlewares)
98+
if mockPoint != nil {
99+
middlewares = append(middlewares, mockPoint)
100+
}
91101
return middlewares, handlerFunc
92102
}
93103

modules/web/routemock.go

Lines changed: 61 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,61 @@
1+
// Copyright 2023 The Gitea Authors. All rights reserved.
2+
// SPDX-License-Identifier: MIT
3+
4+
package web
5+
6+
import (
7+
"net/http"
8+
9+
"code.gitea.io/gitea/modules/setting"
10+
)
11+
12+
// MockAfterMiddlewares is a general mock point, it's between middlewares and the handler
13+
const MockAfterMiddlewares = "MockAfterMiddlewares"
14+
15+
var routeMockPoints = map[string]func(next http.Handler) http.Handler{}
16+
17+
// RouteMockPoint registers a mock point as a middleware for testing, example:
18+
//
19+
// r.Use(web.RouteMockPoint("my-mock-point-1"))
20+
// r.Get("/foo", middleware2, web.RouteMockPoint("my-mock-point-2"), middleware2, handler)
21+
//
22+
// Then use web.RouteMock to mock the route execution.
23+
// It only takes effect in testing mode (setting.IsInTesting == true).
24+
func RouteMockPoint(pointName string) func(next http.Handler) http.Handler {
25+
if !setting.IsInTesting {
26+
return nil
27+
}
28+
routeMockPoints[pointName] = nil
29+
return func(next http.Handler) http.Handler {
30+
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
31+
if h := routeMockPoints[pointName]; h != nil {
32+
h(next).ServeHTTP(w, r)
33+
} else {
34+
next.ServeHTTP(w, r)
35+
}
36+
})
37+
}
38+
}
39+
40+
// RouteMock uses the registered mock point to mock the route execution, example:
41+
//
42+
// defer web.RouteMockReset()
43+
// web.RouteMock(web.MockAfterMiddlewares, func(ctx *context.Context) {
44+
// ctx.WriteResponse(...)
45+
// }
46+
//
47+
// Then the mock function will be executed as a middleware at the mock point.
48+
// It only takes effect in testing mode (setting.IsInTesting == true).
49+
func RouteMock(pointName string, h any) {
50+
if _, ok := routeMockPoints[pointName]; !ok {
51+
panic("route mock point not found: " + pointName)
52+
}
53+
routeMockPoints[pointName] = toHandlerProvider(h)
54+
}
55+
56+
// RouteMockReset resets all mock points (no mock anymore)
57+
func RouteMockReset() {
58+
for k := range routeMockPoints {
59+
routeMockPoints[k] = nil // keep the keys because RouteMock will check the keys to make sure no misspelling
60+
}
61+
}

modules/web/routemock_test.go

Lines changed: 70 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,70 @@
1+
// Copyright 2023 The Gitea Authors. All rights reserved.
2+
// SPDX-License-Identifier: MIT
3+
4+
package web
5+
6+
import (
7+
"net/http"
8+
"net/http/httptest"
9+
"testing"
10+
11+
"code.gitea.io/gitea/modules/setting"
12+
13+
"github.com/stretchr/testify/assert"
14+
)
15+
16+
func TestRouteMock(t *testing.T) {
17+
setting.IsInTesting = true
18+
19+
r := NewRoute()
20+
middleware1 := func(resp http.ResponseWriter, req *http.Request) {
21+
resp.Header().Set("X-Test-Middleware1", "m1")
22+
}
23+
middleware2 := func(resp http.ResponseWriter, req *http.Request) {
24+
resp.Header().Set("X-Test-Middleware2", "m2")
25+
}
26+
handler := func(resp http.ResponseWriter, req *http.Request) {
27+
resp.Header().Set("X-Test-Handler", "h")
28+
}
29+
r.Get("/foo", middleware1, RouteMockPoint("mock-point"), middleware2, handler)
30+
31+
// normal request
32+
recorder := httptest.NewRecorder()
33+
req, err := http.NewRequest("GET", "http://localhost:8000/foo", nil)
34+
assert.NoError(t, err)
35+
r.ServeHTTP(recorder, req)
36+
assert.Len(t, recorder.Header(), 3)
37+
assert.EqualValues(t, "m1", recorder.Header().Get("X-Test-Middleware1"))
38+
assert.EqualValues(t, "m2", recorder.Header().Get("X-Test-Middleware2"))
39+
assert.EqualValues(t, "h", recorder.Header().Get("X-Test-Handler"))
40+
RouteMockReset()
41+
42+
// mock at "mock-point"
43+
RouteMock("mock-point", func(resp http.ResponseWriter, req *http.Request) {
44+
resp.Header().Set("X-Test-MockPoint", "a")
45+
resp.WriteHeader(http.StatusOK)
46+
})
47+
recorder = httptest.NewRecorder()
48+
req, err = http.NewRequest("GET", "http://localhost:8000/foo", nil)
49+
assert.NoError(t, err)
50+
r.ServeHTTP(recorder, req)
51+
assert.Len(t, recorder.Header(), 2)
52+
assert.EqualValues(t, "m1", recorder.Header().Get("X-Test-Middleware1"))
53+
assert.EqualValues(t, "a", recorder.Header().Get("X-Test-MockPoint"))
54+
RouteMockReset()
55+
56+
// mock at MockAfterMiddlewares
57+
RouteMock(MockAfterMiddlewares, func(resp http.ResponseWriter, req *http.Request) {
58+
resp.Header().Set("X-Test-MockPoint", "b")
59+
resp.WriteHeader(http.StatusOK)
60+
})
61+
recorder = httptest.NewRecorder()
62+
req, err = http.NewRequest("GET", "http://localhost:8000/foo", nil)
63+
assert.NoError(t, err)
64+
r.ServeHTTP(recorder, req)
65+
assert.Len(t, recorder.Header(), 3)
66+
assert.EqualValues(t, "m1", recorder.Header().Get("X-Test-Middleware1"))
67+
assert.EqualValues(t, "m2", recorder.Header().Get("X-Test-Middleware2"))
68+
assert.EqualValues(t, "b", recorder.Header().Get("X-Test-MockPoint"))
69+
RouteMockReset()
70+
}

0 commit comments

Comments
 (0)