mirror of
				https://codeberg.org/forgejo/forgejo.git
				synced 2025-10-31 14:31:02 +00:00 
			
		
		
		
	- The parser of `git grep`'s output uses `bufio.Scanner`, which is a good
choice overall, however it does have a limit that's usually not noticed,
it will not read more than `64 * 1024` bytes at once which can be hit in
practical scenarios.
- Use `bufio.Reader` instead which doesn't have this limitation, but is
a bit harder to work with as it's a more lower level primitive.
- Adds unit test.
- Resolves https://codeberg.org/forgejo/forgejo/issues/3149
(cherry picked from commit 668709a33f)
		
	
			
		
			
				
	
	
		
			78 lines
		
	
	
	
		
			2.1 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			78 lines
		
	
	
	
		
			2.1 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright 2024 The Gitea Authors. All rights reserved.
 | |
| // SPDX-License-Identifier: MIT
 | |
| 
 | |
| package git
 | |
| 
 | |
| import (
 | |
| 	"bytes"
 | |
| 	"context"
 | |
| 	"os"
 | |
| 	"path"
 | |
| 	"path/filepath"
 | |
| 	"testing"
 | |
| 
 | |
| 	"github.com/stretchr/testify/assert"
 | |
| )
 | |
| 
 | |
| func TestGrepSearch(t *testing.T) {
 | |
| 	repo, err := openRepositoryWithDefaultContext(filepath.Join(testReposDir, "language_stats_repo"))
 | |
| 	assert.NoError(t, err)
 | |
| 	defer repo.Close()
 | |
| 
 | |
| 	res, err := GrepSearch(context.Background(), repo, "void", GrepOptions{})
 | |
| 	assert.NoError(t, err)
 | |
| 	assert.Equal(t, []*GrepResult{
 | |
| 		{
 | |
| 			Filename:    "java-hello/main.java",
 | |
| 			LineNumbers: []int{3},
 | |
| 			LineCodes:   []string{" public static void main(String[] args)"},
 | |
| 		},
 | |
| 		{
 | |
| 			Filename:    "main.vendor.java",
 | |
| 			LineNumbers: []int{3},
 | |
| 			LineCodes:   []string{" public static void main(String[] args)"},
 | |
| 		},
 | |
| 	}, res)
 | |
| 
 | |
| 	res, err = GrepSearch(context.Background(), repo, "void", GrepOptions{MaxResultLimit: 1})
 | |
| 	assert.NoError(t, err)
 | |
| 	assert.Equal(t, []*GrepResult{
 | |
| 		{
 | |
| 			Filename:    "java-hello/main.java",
 | |
| 			LineNumbers: []int{3},
 | |
| 			LineCodes:   []string{" public static void main(String[] args)"},
 | |
| 		},
 | |
| 	}, res)
 | |
| 
 | |
| 	res, err = GrepSearch(context.Background(), repo, "no-such-content", GrepOptions{})
 | |
| 	assert.NoError(t, err)
 | |
| 	assert.Len(t, res, 0)
 | |
| 
 | |
| 	res, err = GrepSearch(context.Background(), &Repository{Path: "no-such-git-repo"}, "no-such-content", GrepOptions{})
 | |
| 	assert.Error(t, err)
 | |
| 	assert.Len(t, res, 0)
 | |
| }
 | |
| 
 | |
| func TestGrepLongFiles(t *testing.T) {
 | |
| 	tmpDir := t.TempDir()
 | |
| 
 | |
| 	err := InitRepository(DefaultContext, tmpDir, false, Sha1ObjectFormat.Name())
 | |
| 	assert.NoError(t, err)
 | |
| 
 | |
| 	gitRepo, err := openRepositoryWithDefaultContext(tmpDir)
 | |
| 	assert.NoError(t, err)
 | |
| 	defer gitRepo.Close()
 | |
| 
 | |
| 	assert.NoError(t, os.WriteFile(path.Join(tmpDir, "README.md"), bytes.Repeat([]byte{'a'}, 65*1024), 0o666))
 | |
| 
 | |
| 	err = AddChanges(tmpDir, true)
 | |
| 	assert.NoError(t, err)
 | |
| 
 | |
| 	err = CommitChanges(tmpDir, CommitChangesOptions{Message: "Long file"})
 | |
| 	assert.NoError(t, err)
 | |
| 
 | |
| 	res, err := GrepSearch(context.Background(), gitRepo, "a", GrepOptions{})
 | |
| 	assert.NoError(t, err)
 | |
| 	assert.Len(t, res, 1)
 | |
| 	assert.Len(t, res[0].LineCodes[0], 65*1024)
 | |
| }
 |