2020-12-17 07:00:47 -07:00
|
|
|
// Copyright 2015 The Gogs Authors. All rights reserved.
|
|
|
|
// Copyright 2017 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.
|
|
|
|
|
2021-08-24 10:47:09 -06:00
|
|
|
//go:build !gogit
|
2020-12-17 07:00:47 -07:00
|
|
|
// +build !gogit
|
|
|
|
|
|
|
|
package git
|
|
|
|
|
|
|
|
import (
|
2021-05-09 19:27:03 -06:00
|
|
|
"bufio"
|
|
|
|
"context"
|
2020-12-17 07:00:47 -07:00
|
|
|
"errors"
|
|
|
|
"path/filepath"
|
2022-05-02 06:44:45 -06:00
|
|
|
"runtime"
|
|
|
|
"sync"
|
2021-06-25 10:54:08 -06:00
|
|
|
|
|
|
|
"code.gitea.io/gitea/modules/log"
|
2022-05-02 06:44:45 -06:00
|
|
|
"code.gitea.io/gitea/modules/process"
|
2020-12-17 07:00:47 -07:00
|
|
|
)
|
|
|
|
|
|
|
|
// Repository represents a Git repository.
|
|
|
|
type Repository struct {
|
|
|
|
Path string
|
|
|
|
|
|
|
|
tagCache *ObjectCache
|
|
|
|
|
|
|
|
gpgSettings *GPGSettings
|
2021-05-09 19:27:03 -06:00
|
|
|
|
2022-05-02 06:44:45 -06:00
|
|
|
lock sync.Mutex
|
|
|
|
|
|
|
|
closed bool
|
|
|
|
|
2021-05-09 19:27:03 -06:00
|
|
|
batchCancel context.CancelFunc
|
|
|
|
batchReader *bufio.Reader
|
|
|
|
batchWriter WriteCloserError
|
|
|
|
|
|
|
|
checkCancel context.CancelFunc
|
|
|
|
checkReader *bufio.Reader
|
|
|
|
checkWriter WriteCloserError
|
2021-11-30 13:06:32 -07:00
|
|
|
|
|
|
|
Ctx context.Context
|
2020-12-17 07:00:47 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// OpenRepository opens the repository at the given path.
|
|
|
|
func OpenRepository(repoPath string) (*Repository, error) {
|
2021-11-30 13:06:32 -07:00
|
|
|
return OpenRepositoryCtx(DefaultContext, repoPath)
|
|
|
|
}
|
|
|
|
|
|
|
|
// OpenRepositoryCtx opens the repository at the given path with the provided context.
|
|
|
|
func OpenRepositoryCtx(ctx context.Context, repoPath string) (*Repository, error) {
|
2020-12-17 07:00:47 -07:00
|
|
|
repoPath, err := filepath.Abs(repoPath)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
} else if !isDir(repoPath) {
|
|
|
|
return nil, errors.New("no such file or directory")
|
|
|
|
}
|
2021-05-09 19:27:03 -06:00
|
|
|
|
2021-12-16 12:01:14 -07:00
|
|
|
// 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
|
|
|
|
}
|
|
|
|
|
2021-05-09 19:27:03 -06:00
|
|
|
repo := &Repository{
|
2020-12-17 07:00:47 -07:00
|
|
|
Path: repoPath,
|
|
|
|
tagCache: newObjectCache(),
|
2021-11-30 13:06:32 -07:00
|
|
|
Ctx: ctx,
|
2021-05-09 19:27:03 -06:00
|
|
|
}
|
|
|
|
|
2021-11-30 13:06:32 -07:00
|
|
|
repo.batchWriter, repo.batchReader, repo.batchCancel = CatFileBatch(ctx, repoPath)
|
|
|
|
repo.checkWriter, repo.checkReader, repo.checkCancel = CatFileBatchCheck(ctx, repo.Path)
|
2021-05-09 19:27:03 -06:00
|
|
|
|
2022-05-02 06:44:45 -06:00
|
|
|
runtime.SetFinalizer(repo, (*Repository).finalizer)
|
|
|
|
|
2021-05-09 19:27:03 -06:00
|
|
|
return repo, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// CatFileBatch obtains a CatFileBatch for this repository
|
2021-11-30 13:06:32 -07:00
|
|
|
func (repo *Repository) CatFileBatch(ctx context.Context) (WriteCloserError, *bufio.Reader, func()) {
|
2022-05-02 06:44:45 -06:00
|
|
|
repo.lock.Lock()
|
|
|
|
defer repo.lock.Unlock()
|
|
|
|
|
|
|
|
if repo.closed || repo.batchReader.Buffered() > 0 {
|
2021-06-25 10:54:08 -06:00
|
|
|
log.Debug("Opening temporary cat file batch for: %s", repo.Path)
|
2021-11-30 13:06:32 -07:00
|
|
|
return CatFileBatch(ctx, repo.Path)
|
2021-05-09 19:27:03 -06:00
|
|
|
}
|
2022-05-02 06:44:45 -06:00
|
|
|
|
|
|
|
if repo.batchCancel == nil {
|
|
|
|
repo.batchWriter, repo.batchReader, repo.batchCancel = CatFileBatch(ctx, repo.Path)
|
|
|
|
}
|
|
|
|
|
2021-05-09 19:27:03 -06:00
|
|
|
return repo.batchWriter, repo.batchReader, func() {}
|
|
|
|
}
|
|
|
|
|
|
|
|
// CatFileBatchCheck obtains a CatFileBatchCheck for this repository
|
2021-11-30 13:06:32 -07:00
|
|
|
func (repo *Repository) CatFileBatchCheck(ctx context.Context) (WriteCloserError, *bufio.Reader, func()) {
|
2022-05-02 06:44:45 -06:00
|
|
|
repo.lock.Lock()
|
|
|
|
defer repo.lock.Unlock()
|
|
|
|
|
|
|
|
if repo.closed || repo.checkReader.Buffered() > 0 {
|
2021-06-25 10:54:08 -06:00
|
|
|
log.Debug("Opening temporary cat file batch-check: %s", repo.Path)
|
2021-11-30 13:06:32 -07:00
|
|
|
return CatFileBatchCheck(ctx, repo.Path)
|
2021-05-09 19:27:03 -06:00
|
|
|
}
|
2022-05-02 06:44:45 -06:00
|
|
|
|
|
|
|
if repo.checkCancel == nil {
|
|
|
|
repo.checkWriter, repo.checkReader, repo.checkCancel = CatFileBatchCheck(ctx, repo.Path)
|
|
|
|
}
|
|
|
|
|
2021-05-09 19:27:03 -06:00
|
|
|
return repo.checkWriter, repo.checkReader, func() {}
|
2020-12-17 07:00:47 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// Close this repository, in particular close the underlying gogitStorage if this is not nil
|
2022-05-02 06:44:45 -06:00
|
|
|
func (repo *Repository) Close() (err error) {
|
|
|
|
if repo == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
repo.lock.Lock()
|
|
|
|
defer repo.lock.Unlock()
|
|
|
|
|
|
|
|
return repo.close()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (repo *Repository) close() (err error) {
|
2021-05-09 19:27:03 -06:00
|
|
|
if repo == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if repo.batchCancel != nil {
|
|
|
|
repo.batchCancel()
|
|
|
|
repo.batchReader = nil
|
|
|
|
repo.batchWriter = nil
|
|
|
|
repo.batchCancel = nil
|
|
|
|
}
|
|
|
|
if repo.checkCancel != nil {
|
|
|
|
repo.checkCancel()
|
|
|
|
repo.checkCancel = nil
|
|
|
|
repo.checkReader = nil
|
|
|
|
repo.checkWriter = nil
|
|
|
|
}
|
2022-05-02 06:44:45 -06:00
|
|
|
repo.closed = true
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
func (repo *Repository) finalizer() (err error) {
|
|
|
|
if repo == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
repo.lock.Lock()
|
|
|
|
defer repo.lock.Unlock()
|
|
|
|
if repo.closed {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
if repo.batchCancel != nil || repo.checkCancel != nil {
|
|
|
|
pid := ""
|
|
|
|
if repo.Ctx != nil {
|
|
|
|
pid = " from PID: " + string(process.GetPID(repo.Ctx))
|
|
|
|
}
|
|
|
|
log.Error("Finalizer running on unclosed repository%s: %s%s", pid, repo.Path)
|
|
|
|
}
|
|
|
|
return repo.close()
|
2020-12-17 07:00:47 -07:00
|
|
|
}
|