mirror of
				https://gitee.com/gitea/gitea
				synced 2025-11-04 16:40:24 +08:00 
			
		
		
		
	Support Range header end in lfs (#11314)
* Initial support of end Range header option in lfs * Allow end range option to be unspecified * Declare toByte for consistency * Factor out content encoding tests from doLfs This is so Range tests could still use doLfs but without repeating not related tests * Add Range header test * implemented extraHeader * parametrized expectedStatus * Add more test cases of Range header * Fix S1030: should use resp.Body.String() * Add more tests for edge cases * Fix tests Signed-off-by: Andrew Thornton <art27@cantab.net> Co-authored-by: zeripath <art27@cantab.net>
This commit is contained in:
		@@ -12,6 +12,7 @@ import (
 | 
				
			|||||||
	"io"
 | 
						"io"
 | 
				
			||||||
	"io/ioutil"
 | 
						"io/ioutil"
 | 
				
			||||||
	"net/http"
 | 
						"net/http"
 | 
				
			||||||
 | 
						"net/http/httptest"
 | 
				
			||||||
	"testing"
 | 
						"testing"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	"code.gitea.io/gitea/models"
 | 
						"code.gitea.io/gitea/models"
 | 
				
			||||||
@@ -56,13 +57,7 @@ func storeObjectInRepo(t *testing.T, repositoryID int64, content *[]byte) string
 | 
				
			|||||||
	return oid
 | 
						return oid
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func doLfs(t *testing.T, content *[]byte, expectGzip bool) {
 | 
					func storeAndGetLfs(t *testing.T, content *[]byte, extraHeader *http.Header, expectedStatus int) *httptest.ResponseRecorder {
 | 
				
			||||||
	defer prepareTestEnv(t)()
 | 
					 | 
				
			||||||
	setting.CheckLFSVersion()
 | 
					 | 
				
			||||||
	if !setting.LFS.StartServer {
 | 
					 | 
				
			||||||
		t.Skip()
 | 
					 | 
				
			||||||
		return
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
	repo, err := models.GetRepositoryByOwnerAndName("user2", "repo1")
 | 
						repo, err := models.GetRepositoryByOwnerAndName("user2", "repo1")
 | 
				
			||||||
	assert.NoError(t, err)
 | 
						assert.NoError(t, err)
 | 
				
			||||||
	oid := storeObjectInRepo(t, repo.ID, content)
 | 
						oid := storeObjectInRepo(t, repo.ID, content)
 | 
				
			||||||
@@ -73,8 +68,19 @@ func doLfs(t *testing.T, content *[]byte, expectGzip bool) {
 | 
				
			|||||||
	// Request OID
 | 
						// Request OID
 | 
				
			||||||
	req := NewRequest(t, "GET", "/user2/repo1.git/info/lfs/objects/"+oid+"/test")
 | 
						req := NewRequest(t, "GET", "/user2/repo1.git/info/lfs/objects/"+oid+"/test")
 | 
				
			||||||
	req.Header.Set("Accept-Encoding", "gzip")
 | 
						req.Header.Set("Accept-Encoding", "gzip")
 | 
				
			||||||
	resp := session.MakeRequest(t, req, http.StatusOK)
 | 
						if extraHeader != nil {
 | 
				
			||||||
 | 
							for key, values := range *extraHeader {
 | 
				
			||||||
 | 
								for _, value := range values {
 | 
				
			||||||
 | 
									req.Header.Add(key, value)
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						resp := session.MakeRequest(t, req, expectedStatus)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return resp
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func checkResponseTestContentEncoding(t *testing.T, content *[]byte, resp *httptest.ResponseRecorder, expectGzip bool) {
 | 
				
			||||||
	contentEncoding := resp.Header().Get("Content-Encoding")
 | 
						contentEncoding := resp.Header().Get("Content-Encoding")
 | 
				
			||||||
	if !expectGzip || !setting.EnableGzip {
 | 
						if !expectGzip || !setting.EnableGzip {
 | 
				
			||||||
		assert.NotContains(t, contentEncoding, "gzip")
 | 
							assert.NotContains(t, contentEncoding, "gzip")
 | 
				
			||||||
@@ -89,23 +95,44 @@ func doLfs(t *testing.T, content *[]byte, expectGzip bool) {
 | 
				
			|||||||
		assert.NoError(t, err)
 | 
							assert.NoError(t, err)
 | 
				
			||||||
		assert.Equal(t, *content, result)
 | 
							assert.Equal(t, *content, result)
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func TestGetLFSSmall(t *testing.T) {
 | 
					func TestGetLFSSmall(t *testing.T) {
 | 
				
			||||||
 | 
						defer prepareTestEnv(t)()
 | 
				
			||||||
 | 
						setting.CheckLFSVersion()
 | 
				
			||||||
 | 
						if !setting.LFS.StartServer {
 | 
				
			||||||
 | 
							t.Skip()
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
	content := []byte("A very small file\n")
 | 
						content := []byte("A very small file\n")
 | 
				
			||||||
	doLfs(t, &content, false)
 | 
					
 | 
				
			||||||
 | 
						resp := storeAndGetLfs(t, &content, nil, http.StatusOK)
 | 
				
			||||||
 | 
						checkResponseTestContentEncoding(t, &content, resp, false)
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func TestGetLFSLarge(t *testing.T) {
 | 
					func TestGetLFSLarge(t *testing.T) {
 | 
				
			||||||
 | 
						defer prepareTestEnv(t)()
 | 
				
			||||||
 | 
						setting.CheckLFSVersion()
 | 
				
			||||||
 | 
						if !setting.LFS.StartServer {
 | 
				
			||||||
 | 
							t.Skip()
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
	content := make([]byte, gzip.MinSize*10)
 | 
						content := make([]byte, gzip.MinSize*10)
 | 
				
			||||||
	for i := range content {
 | 
						for i := range content {
 | 
				
			||||||
		content[i] = byte(i % 256)
 | 
							content[i] = byte(i % 256)
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
	doLfs(t, &content, true)
 | 
					
 | 
				
			||||||
 | 
						resp := storeAndGetLfs(t, &content, nil, http.StatusOK)
 | 
				
			||||||
 | 
						checkResponseTestContentEncoding(t, &content, resp, true)
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func TestGetLFSGzip(t *testing.T) {
 | 
					func TestGetLFSGzip(t *testing.T) {
 | 
				
			||||||
 | 
						defer prepareTestEnv(t)()
 | 
				
			||||||
 | 
						setting.CheckLFSVersion()
 | 
				
			||||||
 | 
						if !setting.LFS.StartServer {
 | 
				
			||||||
 | 
							t.Skip()
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
	b := make([]byte, gzip.MinSize*10)
 | 
						b := make([]byte, gzip.MinSize*10)
 | 
				
			||||||
	for i := range b {
 | 
						for i := range b {
 | 
				
			||||||
		b[i] = byte(i % 256)
 | 
							b[i] = byte(i % 256)
 | 
				
			||||||
@@ -115,10 +142,18 @@ func TestGetLFSGzip(t *testing.T) {
 | 
				
			|||||||
	gzippWriter.Write(b)
 | 
						gzippWriter.Write(b)
 | 
				
			||||||
	gzippWriter.Close()
 | 
						gzippWriter.Close()
 | 
				
			||||||
	content := outputBuffer.Bytes()
 | 
						content := outputBuffer.Bytes()
 | 
				
			||||||
	doLfs(t, &content, false)
 | 
					
 | 
				
			||||||
 | 
						resp := storeAndGetLfs(t, &content, nil, http.StatusOK)
 | 
				
			||||||
 | 
						checkResponseTestContentEncoding(t, &content, resp, false)
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func TestGetLFSZip(t *testing.T) {
 | 
					func TestGetLFSZip(t *testing.T) {
 | 
				
			||||||
 | 
						defer prepareTestEnv(t)()
 | 
				
			||||||
 | 
						setting.CheckLFSVersion()
 | 
				
			||||||
 | 
						if !setting.LFS.StartServer {
 | 
				
			||||||
 | 
							t.Skip()
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
	b := make([]byte, gzip.MinSize*10)
 | 
						b := make([]byte, gzip.MinSize*10)
 | 
				
			||||||
	for i := range b {
 | 
						for i := range b {
 | 
				
			||||||
		b[i] = byte(i % 256)
 | 
							b[i] = byte(i % 256)
 | 
				
			||||||
@@ -130,5 +165,61 @@ func TestGetLFSZip(t *testing.T) {
 | 
				
			|||||||
	fileWriter.Write(b)
 | 
						fileWriter.Write(b)
 | 
				
			||||||
	zipWriter.Close()
 | 
						zipWriter.Close()
 | 
				
			||||||
	content := outputBuffer.Bytes()
 | 
						content := outputBuffer.Bytes()
 | 
				
			||||||
	doLfs(t, &content, false)
 | 
					
 | 
				
			||||||
 | 
						resp := storeAndGetLfs(t, &content, nil, http.StatusOK)
 | 
				
			||||||
 | 
						checkResponseTestContentEncoding(t, &content, resp, false)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func TestGetLFSRangeNo(t *testing.T) {
 | 
				
			||||||
 | 
						defer prepareTestEnv(t)()
 | 
				
			||||||
 | 
						setting.CheckLFSVersion()
 | 
				
			||||||
 | 
						if !setting.LFS.StartServer {
 | 
				
			||||||
 | 
							t.Skip()
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						content := []byte("123456789\n")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						resp := storeAndGetLfs(t, &content, nil, http.StatusOK)
 | 
				
			||||||
 | 
						assert.Equal(t, content, resp.Body.Bytes())
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func TestGetLFSRange(t *testing.T) {
 | 
				
			||||||
 | 
						defer prepareTestEnv(t)()
 | 
				
			||||||
 | 
						setting.CheckLFSVersion()
 | 
				
			||||||
 | 
						if !setting.LFS.StartServer {
 | 
				
			||||||
 | 
							t.Skip()
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						content := []byte("123456789\n")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						tests := []struct {
 | 
				
			||||||
 | 
							in     string
 | 
				
			||||||
 | 
							out    string
 | 
				
			||||||
 | 
							status int
 | 
				
			||||||
 | 
						}{
 | 
				
			||||||
 | 
							{"bytes=0-0", "1", http.StatusPartialContent},
 | 
				
			||||||
 | 
							{"bytes=0-1", "12", http.StatusPartialContent},
 | 
				
			||||||
 | 
							{"bytes=1-1", "2", http.StatusPartialContent},
 | 
				
			||||||
 | 
							{"bytes=1-3", "234", http.StatusPartialContent},
 | 
				
			||||||
 | 
							{"bytes=1-", "23456789\n", http.StatusPartialContent},
 | 
				
			||||||
 | 
							// end-range smaller than start-range is ignored
 | 
				
			||||||
 | 
							{"bytes=1-0", "23456789\n", http.StatusPartialContent},
 | 
				
			||||||
 | 
							{"bytes=0-10", "123456789\n", http.StatusPartialContent},
 | 
				
			||||||
 | 
							// end-range bigger than length-1 is ignored
 | 
				
			||||||
 | 
							{"bytes=0-11", "123456789\n", http.StatusPartialContent},
 | 
				
			||||||
 | 
							{"bytes=11-", "", http.StatusPartialContent},
 | 
				
			||||||
 | 
							// incorrect header value cause whole header to be ignored
 | 
				
			||||||
 | 
							{"bytes=-", "123456789\n", http.StatusOK},
 | 
				
			||||||
 | 
							{"foobar", "123456789\n", http.StatusOK},
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						for _, tt := range tests {
 | 
				
			||||||
 | 
							t.Run(tt.in, func(t *testing.T) {
 | 
				
			||||||
 | 
								h := http.Header{
 | 
				
			||||||
 | 
									"Range": []string{tt.in},
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
								resp := storeAndGetLfs(t, &content, &h, tt.status)
 | 
				
			||||||
 | 
								assert.Equal(t, tt.out, resp.Body.String())
 | 
				
			||||||
 | 
							})
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -165,15 +165,24 @@ func getContentHandler(ctx *context.Context) {
 | 
				
			|||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	// Support resume download using Range header
 | 
						// Support resume download using Range header
 | 
				
			||||||
	var fromByte int64
 | 
						var fromByte, toByte int64
 | 
				
			||||||
 | 
						toByte = meta.Size - 1
 | 
				
			||||||
	statusCode := 200
 | 
						statusCode := 200
 | 
				
			||||||
	if rangeHdr := ctx.Req.Header.Get("Range"); rangeHdr != "" {
 | 
						if rangeHdr := ctx.Req.Header.Get("Range"); rangeHdr != "" {
 | 
				
			||||||
		regex := regexp.MustCompile(`bytes=(\d+)\-.*`)
 | 
							regex := regexp.MustCompile(`bytes=(\d+)\-(\d*).*`)
 | 
				
			||||||
		match := regex.FindStringSubmatch(rangeHdr)
 | 
							match := regex.FindStringSubmatch(rangeHdr)
 | 
				
			||||||
		if len(match) > 1 {
 | 
							if len(match) > 1 {
 | 
				
			||||||
			statusCode = 206
 | 
								statusCode = 206
 | 
				
			||||||
			fromByte, _ = strconv.ParseInt(match[1], 10, 32)
 | 
								fromByte, _ = strconv.ParseInt(match[1], 10, 32)
 | 
				
			||||||
			ctx.Resp.Header().Set("Content-Range", fmt.Sprintf("bytes %d-%d/%d", fromByte, meta.Size-1, meta.Size-fromByte))
 | 
					
 | 
				
			||||||
 | 
								if match[2] != "" {
 | 
				
			||||||
 | 
									_toByte, _ := strconv.ParseInt(match[2], 10, 32)
 | 
				
			||||||
 | 
									if _toByte >= fromByte && _toByte < toByte {
 | 
				
			||||||
 | 
										toByte = _toByte
 | 
				
			||||||
 | 
									}
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								ctx.Resp.Header().Set("Content-Range", fmt.Sprintf("bytes %d-%d/%d", fromByte, toByte, meta.Size-fromByte))
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -186,7 +195,8 @@ func getContentHandler(ctx *context.Context) {
 | 
				
			|||||||
	}
 | 
						}
 | 
				
			||||||
	defer content.Close()
 | 
						defer content.Close()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	ctx.Resp.Header().Set("Content-Length", strconv.FormatInt(meta.Size-fromByte, 10))
 | 
						contentLength := toByte + 1 - fromByte
 | 
				
			||||||
 | 
						ctx.Resp.Header().Set("Content-Length", strconv.FormatInt(contentLength, 10))
 | 
				
			||||||
	ctx.Resp.Header().Set("Content-Type", "application/octet-stream")
 | 
						ctx.Resp.Header().Set("Content-Type", "application/octet-stream")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	filename := ctx.Params("filename")
 | 
						filename := ctx.Params("filename")
 | 
				
			||||||
@@ -198,7 +208,7 @@ func getContentHandler(ctx *context.Context) {
 | 
				
			|||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	ctx.Resp.WriteHeader(statusCode)
 | 
						ctx.Resp.WriteHeader(statusCode)
 | 
				
			||||||
	if written, err := io.Copy(ctx.Resp, content); err != nil {
 | 
						if written, err := io.CopyN(ctx.Resp, content, contentLength); err != nil {
 | 
				
			||||||
		log.Error("Error whilst copying LFS OID[%s] to the response after %d bytes. Error: %v", meta.Oid, written, err)
 | 
							log.Error("Error whilst copying LFS OID[%s] to the response after %d bytes. Error: %v", meta.Oid, written, err)
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
	logRequest(ctx.Req, statusCode)
 | 
						logRequest(ctx.Req, statusCode)
 | 
				
			||||||
 
 | 
				
			|||||||
		Reference in New Issue
	
	Block a user