From c4d473a2132d1c27203529382863b5b413bed220 Mon Sep 17 00:00:00 2001 From: yinheli Date: Sat, 14 Sep 2024 23:04:15 +0800 Subject: [PATCH] =?UTF-8?q?=F0=9F=A9=B9=20Fix:=20static=20server=20in=20su?= =?UTF-8?q?b=20app=20with=20mount=20(#3104)?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- mount.go | 11 +++++++++++ router.go | 7 +++++++ router_test.go | 52 ++++++++++++++++++++++++++++++++++++++++++++++++-- 3 files changed, 68 insertions(+), 2 deletions(-) diff --git a/mount.go b/mount.go index abb5695e9f..c154f08ca2 100644 --- a/mount.go +++ b/mount.go @@ -23,6 +23,8 @@ type mountFields struct { subAppsProcessed sync.Once // Prefix of app if it was mounted mountPath string + // Parent app of the current app + parentApp *App } // Create empty mountFields instance @@ -50,6 +52,7 @@ func (app *App) Mount(prefix string, subApp *App) Router { subApp.mountFields.mountPath = path app.mountFields.appList[path] = subApp + subApp.mountFields.parentApp = app } // register mounted group @@ -99,6 +102,14 @@ func (app *App) MountPath() string { return app.mountFields.mountPath } +// FullMountPath returns the full mount path of the app, including the parent app's mount path. +func (app *App) FullMountPath() string { + if app.mountFields.parentApp == nil { + return app.mountFields.mountPath + } + return getGroupPath(app.mountFields.parentApp.FullMountPath(), app.mountFields.mountPath) +} + // hasMountedApps Checks if there are any mounted apps in the current application. func (app *App) hasMountedApps() bool { return len(app.mountFields.appList) > 1 diff --git a/router.go b/router.go index 4afa741537..7c6a90d0ad 100644 --- a/router.go +++ b/router.go @@ -5,6 +5,7 @@ package fiber import ( + "bytes" "fmt" "html" "sort" @@ -357,6 +358,12 @@ func (app *App) registerStatic(prefix, root string, config ...Static) { IndexNames: []string{"index.html"}, PathRewrite: func(fctx *fasthttp.RequestCtx) []byte { path := fctx.Path() + mountPath := app.FullMountPath() + if n := len(mountPath); n > 0 { + if bytes.Equal(path[:n], utils.UnsafeBytes(mountPath)) { + path = path[n:] + } + } if len(path) >= prefixLen { if isStar && app.getString(path[0:prefixLen]) == prefix { path = append(path[0:0], '/') diff --git a/router_test.go b/router_test.go index 6a43db5937..5c257bc44c 100644 --- a/router_test.go +++ b/router_test.go @@ -3,6 +3,7 @@ // 📌 API Documentation: https://docs.gofiber.io //nolint:bodyclose // Much easier to just ignore memory leaks in tests +//nolint:goconst // Much easier to just ignore constant variables in tests package fiber import ( @@ -354,7 +355,7 @@ func Test_Router_Handler_Catch_Error(t *testing.T) { func Test_Route_Static_Root(t *testing.T) { t.Parallel() - dir := "./.github/testdata/fs/css" + dir := "./.github/testdata/fs/css" //nolint:goconst // Repeated for test clarity app := New() app.Static("/", dir, Static{ Browse: true, @@ -391,7 +392,7 @@ func Test_Route_Static_Root(t *testing.T) { func Test_Route_Static_HasPrefix(t *testing.T) { t.Parallel() - dir := "./.github/testdata/fs/css" + dir := "./.github/testdata/fs/css" //nolint:goconst // Repeated for test clarity app := New() app.Static("/static", dir, Static{ Browse: true, @@ -471,6 +472,53 @@ func Test_Route_Static_HasPrefix(t *testing.T) { body, err = io.ReadAll(resp.Body) utils.AssertEqual(t, nil, err, "app.Test(req)") utils.AssertEqual(t, true, strings.Contains(app.getString(body), "color")) + + app = New() + app.Static("/css", dir) + + resp, err = app.Test(httptest.NewRequest(MethodGet, "/css/style.css", nil)) + utils.AssertEqual(t, nil, err, "app.Test(req)") + utils.AssertEqual(t, 200, resp.StatusCode, "Status code") + + body, err = io.ReadAll(resp.Body) + utils.AssertEqual(t, nil, err, "app.Test(req)") + utils.AssertEqual(t, true, strings.Contains(app.getString(body), "color")) +} + +func Test_Route_Static_SubApp(t *testing.T) { + t.Parallel() + + dir := "./.github/testdata/fs/css" //nolint:goconst // Repeated for test clarity + app := New() + + // subapp + subApp := New() + subApp.Static("/css", dir) + app.Mount("/sub", subApp) + + // nested subapp + nestApp := New() + nestApp.Static("/css", dir) + subApp.Mount("/nest", nestApp) + + // test subapp + resp, err := app.Test(httptest.NewRequest(MethodGet, "/sub/css/style.css", nil)) + utils.AssertEqual(t, nil, err, "app.Test(req)") + utils.AssertEqual(t, 200, resp.StatusCode, "Status code") + + body, err := io.ReadAll(resp.Body) + utils.AssertEqual(t, nil, err, "app.Test(req)") + utils.AssertEqual(t, true, strings.Contains(app.getString(body), "color")) + + // test nested subapp + + resp, err = app.Test(httptest.NewRequest(MethodGet, "/sub/nest/css/style.css", nil)) + utils.AssertEqual(t, nil, err, "app.Test(req)") + utils.AssertEqual(t, 200, resp.StatusCode, "Status code") + + body, err = io.ReadAll(resp.Body) + utils.AssertEqual(t, nil, err, "app.Test(req)") + utils.AssertEqual(t, true, strings.Contains(app.getString(body), "color")) } func Test_Router_NotFound(t *testing.T) {