mirror of
				https://gitee.com/gitea/gitea
				synced 2025-11-04 08:30:25 +08:00 
			
		
		
		
	Fix http path bug (#16117)
* Fix http path bug * Add missed request * add tests Co-authored-by: 6543 <6543@obermui.de>
This commit is contained in:
		
							
								
								
									
										69
									
								
								integrations/git_smart_http_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										69
									
								
								integrations/git_smart_http_test.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,69 @@
 | 
			
		||||
// Copyright 2021 The Gitea Authors. All rights reserved.
 | 
			
		||||
// Use of this source code is governed by a MIT-style
 | 
			
		||||
// license that can be found in the LICENSE file.
 | 
			
		||||
 | 
			
		||||
package integrations
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"io/ioutil"
 | 
			
		||||
	"net/http"
 | 
			
		||||
	"net/url"
 | 
			
		||||
	"testing"
 | 
			
		||||
 | 
			
		||||
	"github.com/stretchr/testify/assert"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
func TestGitSmartHTTP(t *testing.T) {
 | 
			
		||||
	onGiteaRun(t, testGitSmartHTTP)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func testGitSmartHTTP(t *testing.T, u *url.URL) {
 | 
			
		||||
	var kases = []struct {
 | 
			
		||||
		p    string
 | 
			
		||||
		code int
 | 
			
		||||
	}{
 | 
			
		||||
		{
 | 
			
		||||
			p:    "user2/repo1/info/refs",
 | 
			
		||||
			code: 200,
 | 
			
		||||
		},
 | 
			
		||||
		{
 | 
			
		||||
			p:    "user2/repo1/HEAD",
 | 
			
		||||
			code: 200,
 | 
			
		||||
		},
 | 
			
		||||
		{
 | 
			
		||||
			p:    "user2/repo1/objects/info/alternates",
 | 
			
		||||
			code: 404,
 | 
			
		||||
		},
 | 
			
		||||
		{
 | 
			
		||||
			p:    "user2/repo1/objects/info/http-alternates",
 | 
			
		||||
			code: 404,
 | 
			
		||||
		},
 | 
			
		||||
		{
 | 
			
		||||
			p:    "user2/repo1/../../custom/conf/app.ini",
 | 
			
		||||
			code: 404,
 | 
			
		||||
		},
 | 
			
		||||
		{
 | 
			
		||||
			p:    "user2/repo1/objects/info/../../../../custom/conf/app.ini",
 | 
			
		||||
			code: 404,
 | 
			
		||||
		},
 | 
			
		||||
		{
 | 
			
		||||
			p:    `user2/repo1/objects/info/..\..\..\..\custom\conf\app.ini`,
 | 
			
		||||
			code: 400,
 | 
			
		||||
		},
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	for _, kase := range kases {
 | 
			
		||||
		t.Run(kase.p, func(t *testing.T) {
 | 
			
		||||
			p := u.String() + kase.p
 | 
			
		||||
			req, err := http.NewRequest("GET", p, nil)
 | 
			
		||||
			assert.NoError(t, err)
 | 
			
		||||
			req.SetBasicAuth("user2", userPassword)
 | 
			
		||||
			resp, err := http.DefaultClient.Do(req)
 | 
			
		||||
			assert.NoError(t, err)
 | 
			
		||||
			defer resp.Body.Close()
 | 
			
		||||
			assert.EqualValues(t, kase.code, resp.StatusCode)
 | 
			
		||||
			_, err = ioutil.ReadAll(resp.Body)
 | 
			
		||||
			assert.NoError(t, err)
 | 
			
		||||
		})
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
@@ -366,7 +366,26 @@ func (h *serviceHandler) setHeaderCacheForever() {
 | 
			
		||||
	h.w.Header().Set("Cache-Control", "public, max-age=31536000")
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func containsParentDirectorySeparator(v string) bool {
 | 
			
		||||
	if !strings.Contains(v, "..") {
 | 
			
		||||
		return false
 | 
			
		||||
	}
 | 
			
		||||
	for _, ent := range strings.FieldsFunc(v, isSlashRune) {
 | 
			
		||||
		if ent == ".." {
 | 
			
		||||
			return true
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
	return false
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func isSlashRune(r rune) bool { return r == '/' || r == '\\' }
 | 
			
		||||
 | 
			
		||||
func (h *serviceHandler) sendFile(contentType, file string) {
 | 
			
		||||
	if containsParentDirectorySeparator(file) {
 | 
			
		||||
		log.Error("request file path contains invalid path: %v", file)
 | 
			
		||||
		h.w.WriteHeader(http.StatusBadRequest)
 | 
			
		||||
		return
 | 
			
		||||
	}
 | 
			
		||||
	reqFile := path.Join(h.dir, file)
 | 
			
		||||
 | 
			
		||||
	fi, err := os.Stat(reqFile)
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										43
									
								
								routers/web/repo/http_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										43
									
								
								routers/web/repo/http_test.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,43 @@
 | 
			
		||||
// Copyright 2021 The Gitea Authors. All rights reserved.
 | 
			
		||||
// Use of this source code is governed by a MIT-style
 | 
			
		||||
// license that can be found in the LICENSE file.
 | 
			
		||||
 | 
			
		||||
package repo
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"testing"
 | 
			
		||||
 | 
			
		||||
	"github.com/stretchr/testify/assert"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
func TestContainsParentDirectorySeparator(t *testing.T) {
 | 
			
		||||
	tests := []struct {
 | 
			
		||||
		v string
 | 
			
		||||
		b bool
 | 
			
		||||
	}{
 | 
			
		||||
		{
 | 
			
		||||
			v: `user2/repo1/info/refs`,
 | 
			
		||||
			b: false,
 | 
			
		||||
		},
 | 
			
		||||
		{
 | 
			
		||||
			v: `user2/repo1/HEAD`,
 | 
			
		||||
			b: false,
 | 
			
		||||
		},
 | 
			
		||||
		{
 | 
			
		||||
			v: `user2/repo1/some.../strange_file...mp3`,
 | 
			
		||||
			b: false,
 | 
			
		||||
		},
 | 
			
		||||
		{
 | 
			
		||||
			v: `user2/repo1/../../custom/conf/app.ini`,
 | 
			
		||||
			b: true,
 | 
			
		||||
		},
 | 
			
		||||
		{
 | 
			
		||||
			v: `user2/repo1/objects/info/..\..\..\..\custom\conf\app.ini`,
 | 
			
		||||
			b: true,
 | 
			
		||||
		},
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	for i := range tests {
 | 
			
		||||
		assert.EqualValues(t, tests[i].b, containsParentDirectorySeparator(tests[i].v))
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
		Reference in New Issue
	
	Block a user