}
func getLastCommitForPathsByCache(commitID, treePath string, paths []string, cache *LastCommitCache) (map[string]*Commit, []string, error) {
+ wr, rd, cancel := CatFileBatch(cache.repo.Path)
+ defer cancel()
+
var unHitEntryPaths []string
var results = make(map[string]*Commit)
for _, p := range paths {
- lastCommit, err := cache.Get(commitID, path.Join(treePath, p))
+ lastCommit, err := cache.Get(commitID, path.Join(treePath, p), wr, rd)
if err != nil {
return nil, nil, err
}
package git
import (
+ "bufio"
+ "io"
"path"
)
}
// Get get the last commit information by commit id and entry path
-func (c *LastCommitCache) Get(ref, entryPath string) (interface{}, error) {
+func (c *LastCommitCache) Get(ref, entryPath string, wr *io.PipeWriter, rd *bufio.Reader) (interface{}, error) {
v := c.cache.Get(c.getCacheKey(c.repoPath, ref, entryPath))
if vs, ok := v.(string); ok {
log("LastCommitCache hit level 1: [%s:%s:%s]", ref, entryPath, vs)
if err != nil {
return nil, err
}
- commit, err := c.repo.getCommit(id)
+ if _, err := wr.Write([]byte(vs + "\n")); err != nil {
+ return nil, err
+ }
+ commit, err := c.repo.getCommitFromBatchReader(rd, id)
if err != nil {
return nil, err
}
import (
"bufio"
"errors"
- "fmt"
"io"
"io/ioutil"
+ "os"
+ "path/filepath"
"strings"
)
// GetRefCommitID returns the last commit ID string of given reference (branch or tag).
func (repo *Repository) GetRefCommitID(name string) (string, error) {
+ if strings.HasPrefix(name, "refs/") {
+ // We're gonna try just reading the ref file as this is likely to be quicker than other options
+ fileInfo, err := os.Lstat(filepath.Join(repo.Path, name))
+ if err == nil && fileInfo.Mode().IsRegular() && fileInfo.Size() == 41 {
+ ref, err := ioutil.ReadFile(filepath.Join(repo.Path, name))
+
+ if err == nil && SHAPattern.Match(ref[:40]) && ref[40] == '\n' {
+ return string(ref[:40]), nil
+ }
+ }
+ }
+
stdout, err := NewCommand("show-ref", "--verify", "--hash", name).RunInDir(repo.Path)
if err != nil {
if strings.Contains(err.Error(), "not a valid ref") {
}()
bufReader := bufio.NewReader(stdoutReader)
+
+ return repo.getCommitFromBatchReader(bufReader, id)
+}
+
+func (repo *Repository) getCommitFromBatchReader(bufReader *bufio.Reader, id SHA1) (*Commit, error) {
_, typ, size, err := ReadBatchLine(bufReader)
if err != nil {
if errors.Is(err, io.EOF) {
case "commit":
return CommitFromReader(repo, id, io.LimitReader(bufReader, size))
default:
- _ = stdoutReader.CloseWithError(fmt.Errorf("unknown typ: %s", typ))
log("Unknown typ: %s", typ)
return nil, ErrNotExist{
ID: id.String(),