125 lines
		
	
	
		
			2.5 KiB
		
	
	
	
		
			Go
		
	
	
	
			
		
		
	
	
			125 lines
		
	
	
		
			2.5 KiB
		
	
	
	
		
			Go
		
	
	
	
| // 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.
 | |
| 
 | |
| package models
 | |
| 
 | |
| import (
 | |
| 	"bufio"
 | |
| 	"fmt"
 | |
| 	"io"
 | |
| 	"os"
 | |
| 	"os/exec"
 | |
| 	"regexp"
 | |
| 
 | |
| 	"code.gitea.io/gitea/modules/git"
 | |
| 	"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 {
 | |
| 	cmd     *exec.Cmd
 | |
| 	pid     int64
 | |
| 	output  io.ReadCloser
 | |
| 	scanner *bufio.Scanner
 | |
| 	lastSha *string
 | |
| }
 | |
| 
 | |
| var shaLineRegex = regexp.MustCompile("^([a-z0-9]{40})")
 | |
| 
 | |
| // NextPart returns next part of blame (sequencial code lines with the same commit)
 | |
| func (r *BlameReader) NextPart() (*BlamePart, error) {
 | |
| 	var blamePart *BlamePart
 | |
| 
 | |
| 	scanner := r.scanner
 | |
| 
 | |
| 	if r.lastSha != nil {
 | |
| 		blamePart = &BlamePart{*r.lastSha, make([]string, 0, 0)}
 | |
| 	}
 | |
| 
 | |
| 	for scanner.Scan() {
 | |
| 		line := scanner.Text()
 | |
| 
 | |
| 		// Skip empty lines
 | |
| 		if len(line) == 0 {
 | |
| 			continue
 | |
| 		}
 | |
| 
 | |
| 		lines := shaLineRegex.FindStringSubmatch(line)
 | |
| 		if lines != nil {
 | |
| 			sha1 := lines[1]
 | |
| 
 | |
| 			if blamePart == nil {
 | |
| 				blamePart = &BlamePart{sha1, make([]string, 0, 0)}
 | |
| 			}
 | |
| 
 | |
| 			if blamePart.Sha != sha1 {
 | |
| 				r.lastSha = &sha1
 | |
| 				return blamePart, nil
 | |
| 			}
 | |
| 		} else if line[0] == '\t' {
 | |
| 			code := line[1:]
 | |
| 
 | |
| 			blamePart.Lines = append(blamePart.Lines, code)
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	r.lastSha = nil
 | |
| 
 | |
| 	return blamePart, nil
 | |
| }
 | |
| 
 | |
| // Close BlameReader - don't run NextPart after invoking that
 | |
| func (r *BlameReader) Close() error {
 | |
| 	process.GetManager().Remove(r.pid)
 | |
| 
 | |
| 	if err := r.cmd.Wait(); err != nil {
 | |
| 		return fmt.Errorf("Wait: %v", err)
 | |
| 	}
 | |
| 
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| // CreateBlameReader creates reader for given repository, commit and file
 | |
| func CreateBlameReader(repoPath, commitID, file string) (*BlameReader, error) {
 | |
| 	_, err := git.OpenRepository(repoPath)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	return createBlameReader(repoPath, "git", "blame", commitID, "--porcelain", "--", file)
 | |
| }
 | |
| 
 | |
| func createBlameReader(dir string, command ...string) (*BlameReader, error) {
 | |
| 	cmd := exec.Command(command[0], command[1:]...)
 | |
| 	cmd.Dir = dir
 | |
| 	cmd.Stderr = os.Stderr
 | |
| 
 | |
| 	stdout, err := cmd.StdoutPipe()
 | |
| 	if err != nil {
 | |
| 		return nil, fmt.Errorf("StdoutPipe: %v", err)
 | |
| 	}
 | |
| 
 | |
| 	if err = cmd.Start(); err != nil {
 | |
| 		return nil, fmt.Errorf("Start: %v", err)
 | |
| 	}
 | |
| 
 | |
| 	pid := process.GetManager().Add(fmt.Sprintf("GetBlame [repo_path: %s]", dir), cmd)
 | |
| 
 | |
| 	scanner := bufio.NewScanner(stdout)
 | |
| 
 | |
| 	return &BlameReader{
 | |
| 		cmd,
 | |
| 		pid,
 | |
| 		stdout,
 | |
| 		scanner,
 | |
| 		nil,
 | |
| 	}, nil
 | |
| }
 |