// Copyright 2018 The Gitea Authors. All rights reserved. // SPDX-License-Identifier: MIT package integration import ( "fmt" "net/http" "net/url" "path" "testing" "code.gitea.io/gitea/tests" "github.com/stretchr/testify/require" ) func setDefaultBranch(t *testing.T, session *TestSession, user, repo, branch string) { location := path.Join("/", user, repo, "settings/branches") csrf := GetUserCSRFToken(t, session) req := NewRequestWithValues(t, "POST", location, map[string]string{ "_csrf": csrf, "action": "default_branch", "branch": branch, }) session.MakeRequest(t, req, http.StatusSeeOther) } func TestNonAsciiBranches(t *testing.T) { testRedirects := []struct { from string to string status int }{ // Branches { from: "master", to: "branch/master", status: http.StatusOK, }, { from: "master/README.md", to: "branch/master/README.md", status: http.StatusOK, }, { from: "master/badfile", to: "branch/master/badfile", status: http.StatusNotFound, // it does not exists }, { from: "ГлавнаяВетка", to: "branch/%D0%93%D0%BB%D0%B0%D0%B2%D0%BD%D0%B0%D1%8F%D0%92%D0%B5%D1%82%D0%BA%D0%B0", status: http.StatusOK, }, { from: "а/б/в", to: "branch/%D0%B0/%D0%B1/%D0%B2", status: http.StatusOK, }, { from: "Grüßen/README.md", to: "branch/Gr%C3%BC%C3%9Fen/README.md", status: http.StatusOK, }, { from: "Plus+Is+Not+Space", to: "branch/Plus+Is+Not+Space", status: http.StatusOK, }, { from: "Plus+Is+Not+Space/Файл.md", to: "branch/Plus+Is+Not+Space/%D0%A4%D0%B0%D0%B9%D0%BB.md", status: http.StatusOK, }, { from: "Plus+Is+Not+Space/and+it+is+valid.md", to: "branch/Plus+Is+Not+Space/and+it+is+valid.md", status: http.StatusOK, }, { from: "ブランチ", to: "branch/%E3%83%96%E3%83%A9%E3%83%B3%E3%83%81", status: http.StatusOK, }, // Tags { from: "Тэг", to: "tag/%D0%A2%D1%8D%D0%B3", status: http.StatusOK, }, { from: "Ё/人", to: "tag/%D0%81/%E4%BA%BA", status: http.StatusOK, }, { from: "タグ", to: "tag/%E3%82%BF%E3%82%B0", status: http.StatusOK, }, { from: "タグ/ファイル.md", to: "tag/%E3%82%BF%E3%82%B0/%E3%83%95%E3%82%A1%E3%82%A4%E3%83%AB.md", status: http.StatusOK, }, // Files { from: "README.md", to: "branch/Plus+Is+Not+Space/README.md", status: http.StatusOK, }, { from: "Файл.md", to: "branch/Plus+Is+Not+Space/%D0%A4%D0%B0%D0%B9%D0%BB.md", status: http.StatusOK, }, { from: "ファイル.md", to: "branch/Plus+Is+Not+Space/%E3%83%95%E3%82%A1%E3%82%A4%E3%83%AB.md", status: http.StatusNotFound, // it's not on default branch }, // Same but url-encoded (few tests) { from: "%E3%83%96%E3%83%A9%E3%83%B3%E3%83%81", to: "branch/%E3%83%96%E3%83%A9%E3%83%B3%E3%83%81", status: http.StatusOK, }, { from: "%E3%82%BF%E3%82%b0", to: "tag/%E3%82%BF%E3%82%B0", status: http.StatusOK, }, { from: "%D0%A4%D0%B0%D0%B9%D0%BB.md", to: "branch/Plus+Is+Not+Space/%D0%A4%D0%B0%D0%B9%D0%BB.md", status: http.StatusOK, }, { from: "%D0%81%2F%E4%BA%BA", to: "tag/%D0%81/%E4%BA%BA", status: http.StatusOK, }, { from: "Ё%2F%E4%BA%BA", to: "tag/%D0%81/%E4%BA%BA", status: http.StatusOK, }, { from: "Plus+Is+Not+Space/%25%252525mightnotplaywell", to: "branch/Plus+Is+Not+Space/%25%252525mightnotplaywell", status: http.StatusOK, }, { from: "Plus+Is+Not+Space/%25253Fisnotaquestion%25253F", to: "branch/Plus+Is+Not+Space/%25253Fisnotaquestion%25253F", status: http.StatusOK, }, { from: "Plus+Is+Not+Space/" + url.PathEscape("%3Fis?and#afile"), to: "branch/Plus+Is+Not+Space/" + url.PathEscape("%3Fis?and#afile"), status: http.StatusOK, }, { from: "Plus+Is+Not+Space/10%25.md", to: "branch/Plus+Is+Not+Space/10%25.md", status: http.StatusOK, }, { from: "Plus+Is+Not+Space/" + url.PathEscape("This+file%20has 1space"), to: "branch/Plus+Is+Not+Space/" + url.PathEscape("This+file%20has 1space"), status: http.StatusOK, }, { from: "Plus+Is+Not+Space/" + url.PathEscape("This+file%2520has 2 spaces"), to: "branch/Plus+Is+Not+Space/" + url.PathEscape("This+file%2520has 2 spaces"), status: http.StatusOK, }, { from: "Plus+Is+Not+Space/" + url.PathEscape("£15&$6.txt"), to: "branch/Plus+Is+Not+Space/" + url.PathEscape("£15&$6.txt"), status: http.StatusOK, }, } defer tests.PrepareTestEnv(t)() user := "user2" repo := "utf8" session := loginUser(t, user) setDefaultBranch(t, session, user, repo, "Plus+Is+Not+Space") defer setDefaultBranch(t, session, user, repo, "master") for _, test := range testRedirects { t.Run(test.from, func(t *testing.T) { req := NewRequest(t, "GET", fmt.Sprintf("/%s/%s/src/%s", user, repo, test.from)) resp := session.MakeRequest(t, req, http.StatusSeeOther) require.Equal(t, http.StatusSeeOther, resp.Code) redirectLocation := resp.Header().Get("Location") require.Equal(t, fmt.Sprintf("/%s/%s/src/%s", user, repo, test.to), redirectLocation) req = NewRequest(t, "GET", redirectLocation) session.MakeRequest(t, req, test.status) }) } }