2019-04-20 02:47:00 +00:00
|
|
|
// Copyright 2019 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.
|
|
|
|
|
2019-06-26 18:15:26 +00:00
|
|
|
package git
|
2019-04-20 02:47:00 +00:00
|
|
|
|
|
|
|
import (
|
|
|
|
"bufio"
|
2019-11-30 14:40:22 +00:00
|
|
|
"context"
|
2019-04-20 02:47:00 +00:00
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"os"
|
|
|
|
"os/exec"
|
|
|
|
"regexp"
|
|
|
|
|
|
|
|
"code.gitea.io/gitea/modules/process"
|
|
|
|
)
|
|
|
|
|
|
|
|
// BlamePart represents block of blame - continuous lines with one sha
|
|
|
|
type BlamePart struct {
|
|
|
|
Sha string
|
|
|
|
Lines []string
|
|
|
|
}
|
|
|
|
|
|
|
|
// BlameReader returns part of file blame one by one
|
|
|
|
type BlameReader struct {
|
2021-11-30 20:06:32 +00:00
|
|
|
cmd *exec.Cmd
|
|
|
|
output io.ReadCloser
|
|
|
|
reader *bufio.Reader
|
|
|
|
lastSha *string
|
|
|
|
cancel context.CancelFunc // Cancels the context that this reader runs in
|
|
|
|
finished process.FinishedFunc // Tells the process manager we're finished and it can remove the associated process from the process table
|
2019-04-20 02:47:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
var shaLineRegex = regexp.MustCompile("^([a-z0-9]{40})")
|
|
|
|
|
2021-07-08 11:38:13 +00:00
|
|
|
// NextPart returns next part of blame (sequential code lines with the same commit)
|
2019-04-20 02:47:00 +00:00
|
|
|
func (r *BlameReader) NextPart() (*BlamePart, error) {
|
|
|
|
var blamePart *BlamePart
|
|
|
|
|
2020-11-10 02:14:02 +00:00
|
|
|
reader := r.reader
|
2019-04-20 02:47:00 +00:00
|
|
|
|
|
|
|
if r.lastSha != nil {
|
2019-06-12 19:41:28 +00:00
|
|
|
blamePart = &BlamePart{*r.lastSha, make([]string, 0)}
|
2019-04-20 02:47:00 +00:00
|
|
|
}
|
|
|
|
|
2020-11-10 02:14:02 +00:00
|
|
|
var line []byte
|
|
|
|
var isPrefix bool
|
|
|
|
var err error
|
|
|
|
|
|
|
|
for err != io.EOF {
|
|
|
|
line, isPrefix, err = reader.ReadLine()
|
|
|
|
if err != nil && err != io.EOF {
|
|
|
|
return blamePart, err
|
|
|
|
}
|
2019-04-20 02:47:00 +00:00
|
|
|
|
|
|
|
if len(line) == 0 {
|
2020-11-10 02:14:02 +00:00
|
|
|
// isPrefix will be false
|
2019-04-20 02:47:00 +00:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2020-11-10 02:14:02 +00:00
|
|
|
lines := shaLineRegex.FindSubmatch(line)
|
2019-04-20 02:47:00 +00:00
|
|
|
if lines != nil {
|
2020-11-10 02:14:02 +00:00
|
|
|
sha1 := string(lines[1])
|
2019-04-20 02:47:00 +00:00
|
|
|
|
|
|
|
if blamePart == nil {
|
2019-06-12 19:41:28 +00:00
|
|
|
blamePart = &BlamePart{sha1, make([]string, 0)}
|
2019-04-20 02:47:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if blamePart.Sha != sha1 {
|
|
|
|
r.lastSha = &sha1
|
2020-11-10 02:14:02 +00:00
|
|
|
// need to munch to end of line...
|
|
|
|
for isPrefix {
|
|
|
|
_, isPrefix, err = reader.ReadLine()
|
|
|
|
if err != nil && err != io.EOF {
|
|
|
|
return blamePart, err
|
|
|
|
}
|
|
|
|
}
|
2019-04-20 02:47:00 +00:00
|
|
|
return blamePart, nil
|
|
|
|
}
|
|
|
|
} else if line[0] == '\t' {
|
|
|
|
code := line[1:]
|
|
|
|
|
2020-11-10 02:14:02 +00:00
|
|
|
blamePart.Lines = append(blamePart.Lines, string(code))
|
|
|
|
}
|
|
|
|
|
|
|
|
// need to munch to end of line...
|
|
|
|
for isPrefix {
|
|
|
|
_, isPrefix, err = reader.ReadLine()
|
|
|
|
if err != nil && err != io.EOF {
|
|
|
|
return blamePart, err
|
|
|
|
}
|
2019-04-20 02:47:00 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
r.lastSha = nil
|
|
|
|
|
|
|
|
return blamePart, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Close BlameReader - don't run NextPart after invoking that
|
|
|
|
func (r *BlameReader) Close() error {
|
2021-11-30 20:06:32 +00:00
|
|
|
defer r.finished() // Only remove the process from the process table when the underlying command is closed
|
|
|
|
r.cancel() // However, first cancel our own context early
|
2020-07-01 13:01:17 +00:00
|
|
|
|
|
|
|
_ = r.output.Close()
|
2019-04-20 02:47:00 +00:00
|
|
|
|
|
|
|
if err := r.cmd.Wait(); err != nil {
|
|
|
|
return fmt.Errorf("Wait: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// CreateBlameReader creates reader for given repository, commit and file
|
2020-07-01 13:01:17 +00:00
|
|
|
func CreateBlameReader(ctx context.Context, repoPath, commitID, file string) (*BlameReader, error) {
|
2021-11-30 20:06:32 +00:00
|
|
|
gitRepo, err := OpenRepositoryCtx(ctx, repoPath)
|
2019-04-20 02:47:00 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2019-11-13 07:01:19 +00:00
|
|
|
gitRepo.Close()
|
2019-04-20 02:47:00 +00:00
|
|
|
|
2020-07-01 13:01:17 +00:00
|
|
|
return createBlameReader(ctx, repoPath, GitExecutable, "blame", commitID, "--porcelain", "--", file)
|
2019-04-20 02:47:00 +00:00
|
|
|
}
|
|
|
|
|
2020-07-01 13:01:17 +00:00
|
|
|
func createBlameReader(ctx context.Context, dir string, command ...string) (*BlameReader, error) {
|
|
|
|
// Here we use the provided context - this should be tied to the request performing the blame so that it does not hang around.
|
2021-11-30 20:06:32 +00:00
|
|
|
ctx, cancel, finished := process.GetManager().AddContext(ctx, fmt.Sprintf("GetBlame [repo_path: %s]", dir))
|
|
|
|
|
2019-11-30 14:40:22 +00:00
|
|
|
cmd := exec.CommandContext(ctx, command[0], command[1:]...)
|
2019-04-20 02:47:00 +00:00
|
|
|
cmd.Dir = dir
|
|
|
|
cmd.Stderr = os.Stderr
|
|
|
|
|
|
|
|
stdout, err := cmd.StdoutPipe()
|
|
|
|
if err != nil {
|
2021-11-30 20:06:32 +00:00
|
|
|
defer finished()
|
2019-04-20 02:47:00 +00:00
|
|
|
return nil, fmt.Errorf("StdoutPipe: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err = cmd.Start(); err != nil {
|
2021-11-30 20:06:32 +00:00
|
|
|
defer finished()
|
|
|
|
_ = stdout.Close()
|
2019-04-20 02:47:00 +00:00
|
|
|
return nil, fmt.Errorf("Start: %v", err)
|
|
|
|
}
|
|
|
|
|
2020-11-10 02:14:02 +00:00
|
|
|
reader := bufio.NewReader(stdout)
|
2019-04-20 02:47:00 +00:00
|
|
|
|
|
|
|
return &BlameReader{
|
2021-11-30 20:06:32 +00:00
|
|
|
cmd: cmd,
|
|
|
|
output: stdout,
|
|
|
|
reader: reader,
|
|
|
|
cancel: cancel,
|
|
|
|
finished: finished,
|
2019-04-20 02:47:00 +00:00
|
|
|
}, nil
|
|
|
|
}
|