Fix numbr of files, total additions, and deletions (#11614)
Signed-off-by: Andrew Thornton <art27@cantab.net> Co-authored-by: techknowlogick <techknowlogick@gitea.io>
This commit is contained in:
parent
b97917a6e7
commit
5cb201dc93
8 changed files with 106 additions and 15 deletions
|
@ -6,9 +6,11 @@
|
||||||
package git
|
package git
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"bytes"
|
||||||
"container/list"
|
"container/list"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
|
"regexp"
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
@ -84,14 +86,97 @@ func (repo *Repository) GetCompareInfo(basePath, baseBranch, headBranch string)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Count number of changed files.
|
// Count number of changed files.
|
||||||
stdout, err := NewCommand("diff", "--name-only", remoteBranch+"..."+headBranch).RunInDir(repo.Path)
|
// This probably should be removed as we need to use shortstat elsewhere
|
||||||
|
// Now there is git diff --shortstat but this appears to be slower than simply iterating with --nameonly
|
||||||
|
compareInfo.NumFiles, err = repo.GetDiffNumChangedFiles(remoteBranch, headBranch)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
compareInfo.NumFiles = len(strings.Split(stdout, "\n")) - 1
|
|
||||||
return compareInfo, nil
|
return compareInfo, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type lineCountWriter struct {
|
||||||
|
numLines int
|
||||||
|
}
|
||||||
|
|
||||||
|
// Write counts the number of newlines in the provided bytestream
|
||||||
|
func (l *lineCountWriter) Write(p []byte) (n int, err error) {
|
||||||
|
n = len(p)
|
||||||
|
l.numLines += bytes.Count(p, []byte{'\000'})
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetDiffNumChangedFiles counts the number of changed files
|
||||||
|
// This is substantially quicker than shortstat but...
|
||||||
|
func (repo *Repository) GetDiffNumChangedFiles(base, head string) (int, error) {
|
||||||
|
// Now there is git diff --shortstat but this appears to be slower than simply iterating with --nameonly
|
||||||
|
w := &lineCountWriter{}
|
||||||
|
stderr := new(bytes.Buffer)
|
||||||
|
|
||||||
|
if err := NewCommand("diff", "-z", "--name-only", base+"..."+head).
|
||||||
|
RunInDirPipeline(repo.Path, w, stderr); err != nil {
|
||||||
|
return 0, fmt.Errorf("%v: Stderr: %s", err, stderr)
|
||||||
|
}
|
||||||
|
return w.numLines, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetDiffShortStat counts number of changed files, number of additions and deletions
|
||||||
|
func (repo *Repository) GetDiffShortStat(base, head string) (numFiles, totalAdditions, totalDeletions int, err error) {
|
||||||
|
return GetDiffShortStat(repo.Path, base+"..."+head)
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetDiffShortStat counts number of changed files, number of additions and deletions
|
||||||
|
func GetDiffShortStat(repoPath string, args ...string) (numFiles, totalAdditions, totalDeletions int, err error) {
|
||||||
|
// Now if we call:
|
||||||
|
// $ git diff --shortstat 1ebb35b98889ff77299f24d82da426b434b0cca0...788b8b1440462d477f45b0088875
|
||||||
|
// we get:
|
||||||
|
// " 9902 files changed, 2034198 insertions(+), 298800 deletions(-)\n"
|
||||||
|
args = append([]string{
|
||||||
|
"diff",
|
||||||
|
"--shortstat",
|
||||||
|
}, args...)
|
||||||
|
|
||||||
|
stdout, err := NewCommand(args...).RunInDir(repoPath)
|
||||||
|
if err != nil {
|
||||||
|
return 0, 0, 0, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return parseDiffStat(stdout)
|
||||||
|
}
|
||||||
|
|
||||||
|
var shortStatFormat = regexp.MustCompile(
|
||||||
|
`\s*(\d+) files? changed(?:, (\d+) insertions?\(\+\))?(?:, (\d+) deletions?\(-\))?`)
|
||||||
|
|
||||||
|
func parseDiffStat(stdout string) (numFiles, totalAdditions, totalDeletions int, err error) {
|
||||||
|
if len(stdout) == 0 || stdout == "\n" {
|
||||||
|
return 0, 0, 0, nil
|
||||||
|
}
|
||||||
|
groups := shortStatFormat.FindStringSubmatch(stdout)
|
||||||
|
if len(groups) != 4 {
|
||||||
|
return 0, 0, 0, fmt.Errorf("unable to parse shortstat: %s groups: %s", stdout, groups)
|
||||||
|
}
|
||||||
|
|
||||||
|
numFiles, err = strconv.Atoi(groups[1])
|
||||||
|
if err != nil {
|
||||||
|
return 0, 0, 0, fmt.Errorf("unable to parse shortstat: %s. Error parsing NumFiles %v", stdout, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(groups[2]) != 0 {
|
||||||
|
totalAdditions, err = strconv.Atoi(groups[2])
|
||||||
|
if err != nil {
|
||||||
|
return 0, 0, 0, fmt.Errorf("unable to parse shortstat: %s. Error parsing NumAdditions %v", stdout, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(groups[3]) != 0 {
|
||||||
|
totalDeletions, err = strconv.Atoi(groups[3])
|
||||||
|
if err != nil {
|
||||||
|
return 0, 0, 0, fmt.Errorf("unable to parse shortstat: %s. Error parsing NumDeletions %v", stdout, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
// GetDiffOrPatch generates either diff or formatted patch data between given revisions
|
// GetDiffOrPatch generates either diff or formatted patch data between given revisions
|
||||||
func (repo *Repository) GetDiffOrPatch(base, head string, w io.Writer, formatted bool) error {
|
func (repo *Repository) GetDiffOrPatch(base, head string, w io.Writer, formatted bool) error {
|
||||||
if formatted {
|
if formatted {
|
||||||
|
|
|
@ -108,6 +108,7 @@ func TestGetDiffPreview(t *testing.T) {
|
||||||
},
|
},
|
||||||
IsIncomplete: false,
|
IsIncomplete: false,
|
||||||
}
|
}
|
||||||
|
expectedDiff.NumFiles = len(expectedDiff.Files)
|
||||||
|
|
||||||
t.Run("with given branch", func(t *testing.T) {
|
t.Run("with given branch", func(t *testing.T) {
|
||||||
diff, err := GetDiffPreview(ctx.Repo.Repository, branch, treePath, content)
|
diff, err := GetDiffPreview(ctx.Repo.Repository, branch, treePath, content)
|
||||||
|
|
|
@ -299,6 +299,11 @@ func (t *TemporaryUploadRepository) DiffIndex() (*gitdiff.Diff, error) {
|
||||||
t.repo.FullName(), err, stderr)
|
t.repo.FullName(), err, stderr)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
diff.NumFiles, diff.TotalAddition, diff.TotalDeletion, err = git.GetDiffShortStat(t.basePath, "--cached", "HEAD")
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
return diff, nil
|
return diff, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -291,7 +291,7 @@ func Diff(ctx *context.Context) {
|
||||||
ctx.Data["Author"] = models.ValidateCommitWithEmail(commit)
|
ctx.Data["Author"] = models.ValidateCommitWithEmail(commit)
|
||||||
ctx.Data["Diff"] = diff
|
ctx.Data["Diff"] = diff
|
||||||
ctx.Data["Parents"] = parents
|
ctx.Data["Parents"] = parents
|
||||||
ctx.Data["DiffNotAvailable"] = diff.NumFiles() == 0
|
ctx.Data["DiffNotAvailable"] = diff.NumFiles == 0
|
||||||
|
|
||||||
if err := models.CalculateTrustStatus(verification, ctx.Repo.Repository, nil); err != nil {
|
if err := models.CalculateTrustStatus(verification, ctx.Repo.Repository, nil); err != nil {
|
||||||
ctx.ServerError("CalculateTrustStatus", err)
|
ctx.ServerError("CalculateTrustStatus", err)
|
||||||
|
|
|
@ -437,7 +437,7 @@ func PrepareCompareDiff(
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
ctx.Data["Diff"] = diff
|
ctx.Data["Diff"] = diff
|
||||||
ctx.Data["DiffNotAvailable"] = diff.NumFiles() == 0
|
ctx.Data["DiffNotAvailable"] = diff.NumFiles == 0
|
||||||
|
|
||||||
headCommit, err := headGitRepo.GetCommit(headCommitID)
|
headCommit, err := headGitRepo.GetCommit(headCommitID)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
|
|
@ -339,7 +339,7 @@ func DiffPreviewPost(ctx *context.Context, form auth.EditPreviewDiffForm) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
if diff.NumFiles() == 0 {
|
if diff.NumFiles == 0 {
|
||||||
ctx.PlainText(200, []byte(ctx.Tr("repo.editor.no_changes_to_show")))
|
ctx.PlainText(200, []byte(ctx.Tr("repo.editor.no_changes_to_show")))
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
|
@ -611,7 +611,7 @@ func ViewPullFiles(ctx *context.Context) {
|
||||||
}
|
}
|
||||||
|
|
||||||
ctx.Data["Diff"] = diff
|
ctx.Data["Diff"] = diff
|
||||||
ctx.Data["DiffNotAvailable"] = diff.NumFiles() == 0
|
ctx.Data["DiffNotAvailable"] = diff.NumFiles == 0
|
||||||
|
|
||||||
baseCommit, err := ctx.Repo.GitRepo.GetCommit(startCommitID)
|
baseCommit, err := ctx.Repo.GitRepo.GetCommit(startCommitID)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
|
|
@ -367,9 +367,9 @@ func getCommitFileLineCount(commit *git.Commit, filePath string) int {
|
||||||
|
|
||||||
// Diff represents a difference between two git trees.
|
// Diff represents a difference between two git trees.
|
||||||
type Diff struct {
|
type Diff struct {
|
||||||
TotalAddition, TotalDeletion int
|
NumFiles, TotalAddition, TotalDeletion int
|
||||||
Files []*DiffFile
|
Files []*DiffFile
|
||||||
IsIncomplete bool
|
IsIncomplete bool
|
||||||
}
|
}
|
||||||
|
|
||||||
// LoadComments loads comments into each line
|
// LoadComments loads comments into each line
|
||||||
|
@ -398,11 +398,6 @@ func (diff *Diff) LoadComments(issue *models.Issue, currentUser *models.User) er
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// NumFiles returns number of files changes in a diff.
|
|
||||||
func (diff *Diff) NumFiles() int {
|
|
||||||
return len(diff.Files)
|
|
||||||
}
|
|
||||||
|
|
||||||
const cmdDiffHead = "diff --git "
|
const cmdDiffHead = "diff --git "
|
||||||
|
|
||||||
// ParsePatch builds a Diff object from a io.Reader and some
|
// ParsePatch builds a Diff object from a io.Reader and some
|
||||||
|
@ -639,7 +634,7 @@ func ParsePatch(maxLines, maxLineCharacters, maxFiles int, reader io.Reader) (*D
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
diff.NumFiles = len(diff.Files)
|
||||||
return diff, nil
|
return diff, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -716,6 +711,11 @@ func GetDiffRangeWithWhitespaceBehavior(repoPath, beforeCommitID, afterCommitID
|
||||||
return nil, fmt.Errorf("Wait: %v", err)
|
return nil, fmt.Errorf("Wait: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
diff.NumFiles, diff.TotalAddition, diff.TotalDeletion, err = git.GetDiffShortStat(repoPath, beforeCommitID+"..."+afterCommitID)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
return diff, nil
|
return diff, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue