mirror of
				https://github.com/go-gitea/gitea
				synced 2025-10-26 00:48:29 +00:00 
			
		
		
		
	Prevent hang in git cat-file if repository is not a valid repository and other fixes (#17991)
This PR contains multiple fixes. The most important of which is:
* Prevent hang in git cat-file if the repository is not a valid repository 
    
    Unfortunately it appears that if git cat-file is run in an invalid
    repository it will hang until stdin is closed. This will result in
    deadlocked /pulls pages and dangling git cat-file calls if a broken
    repository is tried to be reviewed or pulls exists for a broken
    repository.
    Fix #14734
    Fix #9271
    Fix #16113
Otherwise there are a few small other fixes included which this PR was initially intending to fix:
* Fix panic on partial compares due to missing PullRequestWorkInProgressPrefixes
* Fix links on pulls pages  due to regression from #17551 - by making most /issues routes match /pulls too - Fix #17983
* Fix links on feeds pages due to another regression from #17551 but also fix issue with syncing tags - Fix #17943
* Add missing locale entries for oauth group claims
* Prevent NPEs if ColorFormat is called on nil users, repos or teams.
			
			
This commit is contained in:
		| @@ -27,6 +27,20 @@ type WriteCloserError interface { | ||||
| 	CloseWithError(err error) error | ||||
| } | ||||
|  | ||||
| // EnsureValidGitRepository runs git rev-parse in the repository path - thus ensuring that the repository is a valid repository. | ||||
| // Run before opening git cat-file. | ||||
| // This is needed otherwise the git cat-file will hang for invalid repositories. | ||||
| func EnsureValidGitRepository(ctx context.Context, repoPath string) error { | ||||
| 	stderr := strings.Builder{} | ||||
| 	err := NewCommandContext(ctx, "rev-parse"). | ||||
| 		SetDescription(fmt.Sprintf("%s rev-parse [repo_path: %s]", GitExecutable, repoPath)). | ||||
| 		RunInDirFullPipeline(repoPath, nil, &stderr, nil) | ||||
| 	if err != nil { | ||||
| 		return ConcatenateError(err, (&stderr).String()) | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // CatFileBatchCheck opens git cat-file --batch-check in the provided repo and returns a stdin pipe, a stdout reader and cancel function | ||||
| func CatFileBatchCheck(ctx context.Context, repoPath string) (WriteCloserError, *bufio.Reader, func()) { | ||||
| 	batchStdinReader, batchStdinWriter := io.Pipe() | ||||
|   | ||||
| @@ -50,6 +50,11 @@ func OpenRepositoryCtx(ctx context.Context, repoPath string) (*Repository, error | ||||
| 		return nil, errors.New("no such file or directory") | ||||
| 	} | ||||
|  | ||||
| 	// Now because of some insanity with git cat-file not immediately failing if not run in a valid git directory we need to run git rev-parse first! | ||||
| 	if err := EnsureValidGitRepository(ctx, repoPath); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	repo := &Repository{ | ||||
| 		Path:     repoPath, | ||||
| 		tagCache: newObjectCache(), | ||||
|   | ||||
| @@ -37,7 +37,10 @@ func (repo *Repository) ResolveReference(name string) (string, error) { | ||||
| func (repo *Repository) GetRefCommitID(name string) (string, error) { | ||||
| 	wr, rd, cancel := repo.CatFileBatchCheck(repo.Ctx) | ||||
| 	defer cancel() | ||||
| 	_, _ = wr.Write([]byte(name + "\n")) | ||||
| 	_, err := wr.Write([]byte(name + "\n")) | ||||
| 	if err != nil { | ||||
| 		return "", err | ||||
| 	} | ||||
| 	shaBs, _, _, err := ReadBatchLine(rd) | ||||
| 	if IsErrNotExist(err) { | ||||
| 		return "", ErrNotExist{name, ""} | ||||
|   | ||||
		Reference in New Issue
	
	Block a user