mirror of
				https://github.com/go-gitea/gitea
				synced 2025-09-28 03:28:13 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			261 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			261 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright 2021 The Gitea Authors. All rights reserved.
 | |
| // SPDX-License-Identifier: MIT
 | |
| 
 | |
| package web
 | |
| 
 | |
| import (
 | |
| 	"bytes"
 | |
| 	"net/http"
 | |
| 	"net/http/httptest"
 | |
| 	"strings"
 | |
| 	"testing"
 | |
| 
 | |
| 	"code.gitea.io/gitea/modules/setting"
 | |
| 	"code.gitea.io/gitea/modules/test"
 | |
| 	"code.gitea.io/gitea/modules/util"
 | |
| 
 | |
| 	"github.com/go-chi/chi/v5"
 | |
| 	"github.com/stretchr/testify/assert"
 | |
| )
 | |
| 
 | |
| func chiURLParamsToMap(chiCtx *chi.Context) map[string]string {
 | |
| 	pathParams := chiCtx.URLParams
 | |
| 	m := make(map[string]string, len(pathParams.Keys))
 | |
| 	for i, key := range pathParams.Keys {
 | |
| 		if key == "*" && pathParams.Values[i] == "" {
 | |
| 			continue // chi router will add an empty "*" key if there is a "Mount"
 | |
| 		}
 | |
| 		m[key] = pathParams.Values[i]
 | |
| 	}
 | |
| 	return util.Iif(len(m) == 0, nil, m)
 | |
| }
 | |
| 
 | |
| func TestPathProcessor(t *testing.T) {
 | |
| 	testProcess := func(pattern, uri string, expectedPathParams map[string]string) {
 | |
| 		chiCtx := chi.NewRouteContext()
 | |
| 		chiCtx.RouteMethod = "GET"
 | |
| 		p := newRouterPathMatcher("GET", patternRegexp(pattern), http.NotFound)
 | |
| 		assert.True(t, p.matchPath(chiCtx, uri), "use pattern %s to process uri %s", pattern, uri)
 | |
| 		assert.Equal(t, expectedPathParams, chiURLParamsToMap(chiCtx), "use pattern %s to process uri %s", pattern, uri)
 | |
| 	}
 | |
| 
 | |
| 	// the "<...>" is intentionally designed to distinguish from chi's path parameters, because:
 | |
| 	// 1. their behaviors are totally different, we do not want to mislead developers
 | |
| 	// 2. we can write regexp in "<name:\w{3,4}>" easily and parse it easily
 | |
| 	testProcess("/<p1>/<p2>", "/a/b", map[string]string{"p1": "a", "p2": "b"})
 | |
| 	testProcess("/<p1:*>", "", map[string]string{"p1": ""}) // this is a special case, because chi router could use empty path
 | |
| 	testProcess("/<p1:*>", "/", map[string]string{"p1": ""})
 | |
| 	testProcess("/<p1:*>/<p2>", "/a", map[string]string{"p1": "", "p2": "a"})
 | |
| 	testProcess("/<p1:*>/<p2>", "/a/b", map[string]string{"p1": "a", "p2": "b"})
 | |
| 	testProcess("/<p1:*>/<p2>", "/a/b/c", map[string]string{"p1": "a/b", "p2": "c"})
 | |
| }
 | |
| 
 | |
| func TestRouter(t *testing.T) {
 | |
| 	buff := &bytes.Buffer{}
 | |
| 	recorder := httptest.NewRecorder()
 | |
| 	recorder.Body = buff
 | |
| 
 | |
| 	type resultStruct struct {
 | |
| 		method       string
 | |
| 		pathParams   map[string]string
 | |
| 		handlerMarks []string
 | |
| 	}
 | |
| 
 | |
| 	var res resultStruct
 | |
| 	h := func(optMark ...string) func(resp http.ResponseWriter, req *http.Request) {
 | |
| 		mark := util.OptionalArg(optMark, "")
 | |
| 		return func(resp http.ResponseWriter, req *http.Request) {
 | |
| 			res.method = req.Method
 | |
| 			res.pathParams = chiURLParamsToMap(chi.RouteContext(req.Context()))
 | |
| 			if mark != "" {
 | |
| 				res.handlerMarks = append(res.handlerMarks, mark)
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	stopMark := func(optMark ...string) func(resp http.ResponseWriter, req *http.Request) {
 | |
| 		mark := util.OptionalArg(optMark, "")
 | |
| 		return func(resp http.ResponseWriter, req *http.Request) {
 | |
| 			if stop := req.FormValue("stop"); stop != "" && (mark == "" || mark == stop) {
 | |
| 				h(stop)(resp, req)
 | |
| 				resp.WriteHeader(http.StatusOK)
 | |
| 			} else if mark != "" {
 | |
| 				res.handlerMarks = append(res.handlerMarks, mark)
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	r := NewRouter()
 | |
| 	r.NotFound(h("not-found:/"))
 | |
| 	r.Get("/{username}/{reponame}/{type:issues|pulls}", h("list-issues-a")) // this one will never be called
 | |
| 	r.Group("/{username}/{reponame}", func() {
 | |
| 		r.Get("/{type:issues|pulls}", h("list-issues-b"))
 | |
| 		r.Group("", func() {
 | |
| 			r.Get("/{type:issues|pulls}/{index}", h("view-issue"))
 | |
| 		}, stopMark())
 | |
| 		r.Group("/issues/{index}", func() {
 | |
| 			r.Post("/update", h("update-issue"))
 | |
| 		})
 | |
| 	})
 | |
| 
 | |
| 	m := NewRouter()
 | |
| 	m.NotFound(h("not-found:/api/v1"))
 | |
| 	r.Mount("/api/v1", m)
 | |
| 	m.Group("/repos", func() {
 | |
| 		m.Group("/{username}/{reponame}", func() {
 | |
| 			m.Group("/branches", func() {
 | |
| 				m.Get("", h())
 | |
| 				m.Post("", h())
 | |
| 				m.Group("/{name}", func() {
 | |
| 					m.Get("", h())
 | |
| 					m.Patch("", h())
 | |
| 					m.Delete("", h())
 | |
| 				})
 | |
| 				m.PathGroup("/*", func(g *RouterPathGroup) {
 | |
| 					g.MatchPattern("GET", g.PatternRegexp(`/<dir:*>/<file:[a-z]{1,2}>`, stopMark("s2")), stopMark("s3"), h("match-path"))
 | |
| 				}, stopMark("s1"))
 | |
| 			})
 | |
| 		})
 | |
| 	})
 | |
| 
 | |
| 	testRoute := func(t *testing.T, methodPath string, expected resultStruct) {
 | |
| 		t.Run(methodPath, func(t *testing.T) {
 | |
| 			res = resultStruct{}
 | |
| 			methodPathFields := strings.Fields(methodPath)
 | |
| 			req, err := http.NewRequest(methodPathFields[0], methodPathFields[1], nil)
 | |
| 			assert.NoError(t, err)
 | |
| 			r.ServeHTTP(recorder, req)
 | |
| 			assert.Equal(t, expected, res)
 | |
| 		})
 | |
| 	}
 | |
| 
 | |
| 	t.Run("RootRouter", func(t *testing.T) {
 | |
| 		testRoute(t, "GET /the-user/the-repo/other", resultStruct{method: "GET", handlerMarks: []string{"not-found:/"}})
 | |
| 		testRoute(t, "GET /the-user/the-repo/pulls", resultStruct{
 | |
| 			method:       "GET",
 | |
| 			pathParams:   map[string]string{"username": "the-user", "reponame": "the-repo", "type": "pulls"},
 | |
| 			handlerMarks: []string{"list-issues-b"},
 | |
| 		})
 | |
| 		testRoute(t, "GET /the-user/the-repo/issues/123", resultStruct{
 | |
| 			method:       "GET",
 | |
| 			pathParams:   map[string]string{"username": "the-user", "reponame": "the-repo", "type": "issues", "index": "123"},
 | |
| 			handlerMarks: []string{"view-issue"},
 | |
| 		})
 | |
| 		testRoute(t, "GET /the-user/the-repo/issues/123?stop=hijack", resultStruct{
 | |
| 			method:       "GET",
 | |
| 			pathParams:   map[string]string{"username": "the-user", "reponame": "the-repo", "type": "issues", "index": "123"},
 | |
| 			handlerMarks: []string{"hijack"},
 | |
| 		})
 | |
| 		testRoute(t, "POST /the-user/the-repo/issues/123/update", resultStruct{
 | |
| 			method:       "POST",
 | |
| 			pathParams:   map[string]string{"username": "the-user", "reponame": "the-repo", "index": "123"},
 | |
| 			handlerMarks: []string{"update-issue"},
 | |
| 		})
 | |
| 	})
 | |
| 
 | |
| 	t.Run("Sub Router", func(t *testing.T) {
 | |
| 		testRoute(t, "GET /api/v1/other", resultStruct{method: "GET", handlerMarks: []string{"not-found:/api/v1"}})
 | |
| 		testRoute(t, "GET /api/v1/repos/the-user/the-repo/branches", resultStruct{
 | |
| 			method:     "GET",
 | |
| 			pathParams: map[string]string{"username": "the-user", "reponame": "the-repo"},
 | |
| 		})
 | |
| 
 | |
| 		testRoute(t, "POST /api/v1/repos/the-user/the-repo/branches", resultStruct{
 | |
| 			method:     "POST",
 | |
| 			pathParams: map[string]string{"username": "the-user", "reponame": "the-repo"},
 | |
| 		})
 | |
| 
 | |
| 		testRoute(t, "GET /api/v1/repos/the-user/the-repo/branches/master", resultStruct{
 | |
| 			method:     "GET",
 | |
| 			pathParams: map[string]string{"username": "the-user", "reponame": "the-repo", "name": "master"},
 | |
| 		})
 | |
| 
 | |
| 		testRoute(t, "PATCH /api/v1/repos/the-user/the-repo/branches/master", resultStruct{
 | |
| 			method:     "PATCH",
 | |
| 			pathParams: map[string]string{"username": "the-user", "reponame": "the-repo", "name": "master"},
 | |
| 		})
 | |
| 
 | |
| 		testRoute(t, "DELETE /api/v1/repos/the-user/the-repo/branches/master", resultStruct{
 | |
| 			method:     "DELETE",
 | |
| 			pathParams: map[string]string{"username": "the-user", "reponame": "the-repo", "name": "master"},
 | |
| 		})
 | |
| 	})
 | |
| 
 | |
| 	t.Run("MatchPath", func(t *testing.T) {
 | |
| 		testRoute(t, "GET /api/v1/repos/the-user/the-repo/branches/d1/d2/fn", resultStruct{
 | |
| 			method:       "GET",
 | |
| 			pathParams:   map[string]string{"username": "the-user", "reponame": "the-repo", "*": "d1/d2/fn", "dir": "d1/d2", "file": "fn"},
 | |
| 			handlerMarks: []string{"s1", "s2", "s3", "match-path"},
 | |
| 		})
 | |
| 		testRoute(t, "GET /api/v1/repos/the-user/the-repo/branches/d1%2fd2/fn", resultStruct{
 | |
| 			method:       "GET",
 | |
| 			pathParams:   map[string]string{"username": "the-user", "reponame": "the-repo", "*": "d1%2fd2/fn", "dir": "d1%2fd2", "file": "fn"},
 | |
| 			handlerMarks: []string{"s1", "s2", "s3", "match-path"},
 | |
| 		})
 | |
| 		testRoute(t, "GET /api/v1/repos/the-user/the-repo/branches/d1/d2/000", resultStruct{
 | |
| 			method:       "GET",
 | |
| 			pathParams:   map[string]string{"reponame": "the-repo", "username": "the-user", "*": "d1/d2/000"},
 | |
| 			handlerMarks: []string{"s1", "not-found:/api/v1"},
 | |
| 		})
 | |
| 
 | |
| 		testRoute(t, "GET /api/v1/repos/the-user/the-repo/branches/d1/d2/fn?stop=s1", resultStruct{
 | |
| 			method:       "GET",
 | |
| 			pathParams:   map[string]string{"username": "the-user", "reponame": "the-repo", "*": "d1/d2/fn"},
 | |
| 			handlerMarks: []string{"s1"},
 | |
| 		})
 | |
| 
 | |
| 		testRoute(t, "GET /api/v1/repos/the-user/the-repo/branches/d1/d2/fn?stop=s2", resultStruct{
 | |
| 			method:       "GET",
 | |
| 			pathParams:   map[string]string{"username": "the-user", "reponame": "the-repo", "*": "d1/d2/fn", "dir": "d1/d2", "file": "fn"},
 | |
| 			handlerMarks: []string{"s1", "s2"},
 | |
| 		})
 | |
| 
 | |
| 		testRoute(t, "GET /api/v1/repos/the-user/the-repo/branches/d1/d2/fn?stop=s3", resultStruct{
 | |
| 			method:       "GET",
 | |
| 			pathParams:   map[string]string{"username": "the-user", "reponame": "the-repo", "*": "d1/d2/fn", "dir": "d1/d2", "file": "fn"},
 | |
| 			handlerMarks: []string{"s1", "s2", "s3"},
 | |
| 		})
 | |
| 	})
 | |
| }
 | |
| 
 | |
| func TestRouteNormalizePath(t *testing.T) {
 | |
| 	type paths struct {
 | |
| 		EscapedPath, RawPath, Path string
 | |
| 	}
 | |
| 	testPath := func(reqPath string, expectedPaths paths) {
 | |
| 		recorder := httptest.NewRecorder()
 | |
| 		recorder.Body = bytes.NewBuffer(nil)
 | |
| 
 | |
| 		actualPaths := paths{EscapedPath: "(none)", RawPath: "(none)", Path: "(none)"}
 | |
| 		r := NewRouter()
 | |
| 		r.Get("/*", func(resp http.ResponseWriter, req *http.Request) {
 | |
| 			actualPaths.EscapedPath = req.URL.EscapedPath()
 | |
| 			actualPaths.RawPath = req.URL.RawPath
 | |
| 			actualPaths.Path = req.URL.Path
 | |
| 		})
 | |
| 
 | |
| 		req, err := http.NewRequest(http.MethodGet, reqPath, nil)
 | |
| 		assert.NoError(t, err)
 | |
| 		r.ServeHTTP(recorder, req)
 | |
| 		assert.Equal(t, expectedPaths, actualPaths, "req path = %q", reqPath)
 | |
| 	}
 | |
| 
 | |
| 	// RawPath could be empty if the EscapedPath is the same as escape(Path) and it is already normalized
 | |
| 	testPath("/", paths{EscapedPath: "/", RawPath: "", Path: "/"})
 | |
| 	testPath("//", paths{EscapedPath: "/", RawPath: "/", Path: "/"})
 | |
| 	testPath("/%2f", paths{EscapedPath: "/%2f", RawPath: "/%2f", Path: "//"})
 | |
| 	testPath("///a//b/", paths{EscapedPath: "/a/b", RawPath: "/a/b", Path: "/a/b"})
 | |
| 
 | |
| 	defer test.MockVariableValue(&setting.UseSubURLPath, true)()
 | |
| 	defer test.MockVariableValue(&setting.AppSubURL, "/sub-path")()
 | |
| 	testPath("/", paths{EscapedPath: "(none)", RawPath: "(none)", Path: "(none)"}) // 404
 | |
| 	testPath("/sub-path", paths{EscapedPath: "/", RawPath: "/", Path: "/"})
 | |
| 	testPath("/sub-path/", paths{EscapedPath: "/", RawPath: "/", Path: "/"})
 | |
| 	testPath("/sub-path//a/b///", paths{EscapedPath: "/a/b", RawPath: "/a/b", Path: "/a/b"})
 | |
| 	testPath("/sub-path/%2f/", paths{EscapedPath: "/%2f", RawPath: "/%2f", Path: "//"})
 | |
| 	// "/v2" is special for OCI container registry, it should always be in the root of the site
 | |
| 	testPath("/v2", paths{EscapedPath: "/v2", RawPath: "/v2", Path: "/v2"})
 | |
| 	testPath("/v2/", paths{EscapedPath: "/v2", RawPath: "/v2", Path: "/v2"})
 | |
| 	testPath("/v2/%2f", paths{EscapedPath: "/v2/%2f", RawPath: "/v2/%2f", Path: "/v2//"})
 | |
| }
 |