2016-11-03 23:16:01 +01:00
// Copyright 2015 The Gogs Authors. All rights reserved.
2019-06-07 22:29:29 +02:00
// Copyright 2019 The Gitea Authors. All rights reserved.
2016-11-03 23:16:01 +01:00
// Use of this source code is governed by a MIT-style
// license that can be found in the LICENSE file.
package git
import (
2020-05-26 06:58:07 +01:00
"bytes"
2016-11-03 23:16:01 +01:00
"container/list"
"fmt"
2018-01-07 14:10:20 +01:00
"io"
2020-05-26 06:58:07 +01:00
"regexp"
2016-11-03 23:16:01 +01:00
"strconv"
"strings"
"time"
2019-06-12 21:41:28 +02:00
logger "code.gitea.io/gitea/modules/log"
2016-11-03 23:16:01 +01:00
)
2019-06-07 22:29:29 +02:00
// CompareInfo represents needed information for comparing references.
type CompareInfo struct {
2021-02-16 15:39:45 +00:00
MergeBase string
BaseCommitID string
HeadCommitID string
Commits * list . List
NumFiles int
2016-11-03 23:16:01 +01:00
}
2019-06-12 01:32:08 +02:00
// GetMergeBase checks and returns merge base of two branches and the reference used as base.
func ( repo * Repository ) GetMergeBase ( tmpRemote string , base , head string ) ( string , string , error ) {
2019-06-07 22:29:29 +02:00
if tmpRemote == "" {
tmpRemote = "origin"
}
if tmpRemote != "origin" {
tmpBaseName := "refs/remotes/" + tmpRemote + "/tmp_" + base
// Fetch commit into a temporary branch in order to be able to handle commits and tags
_ , err := NewCommand ( "fetch" , tmpRemote , base + ":" + tmpBaseName ) . RunInDir ( repo . Path )
if err == nil {
base = tmpBaseName
}
}
2019-08-05 21:39:39 +01:00
stdout , err := NewCommand ( "merge-base" , "--" , base , head ) . RunInDir ( repo . Path )
2019-06-12 01:32:08 +02:00
return strings . TrimSpace ( stdout ) , base , err
2016-11-03 23:16:01 +01:00
}
2019-06-07 22:29:29 +02:00
// GetCompareInfo generates and returns compare information between base and head branches of repositories.
func ( repo * Repository ) GetCompareInfo ( basePath , baseBranch , headBranch string ) ( _ * CompareInfo , err error ) {
var (
remoteBranch string
tmpRemote string
)
2016-11-03 23:16:01 +01:00
// We don't need a temporary remote for same repository.
if repo . Path != basePath {
// Add a temporary remote
2019-06-07 22:29:29 +02:00
tmpRemote = strconv . FormatInt ( time . Now ( ) . UnixNano ( ) , 10 )
2019-08-13 16:30:44 +08:00
if err = repo . AddRemote ( tmpRemote , basePath , false ) ; err != nil {
2016-11-03 23:16:01 +01:00
return nil , fmt . Errorf ( "AddRemote: %v" , err )
}
2019-06-12 21:41:28 +02:00
defer func ( ) {
if err := repo . RemoveRemote ( tmpRemote ) ; err != nil {
logger . Error ( "GetPullRequestInfo: RemoveRemote: %v" , err )
}
} ( )
2016-11-03 23:16:01 +01:00
}
2019-06-07 22:29:29 +02:00
compareInfo := new ( CompareInfo )
2021-02-16 15:39:45 +00:00
compareInfo . HeadCommitID , err = GetFullCommitID ( repo . Path , headBranch )
if err != nil {
compareInfo . HeadCommitID = headBranch
}
2019-06-12 01:32:08 +02:00
compareInfo . MergeBase , remoteBranch , err = repo . GetMergeBase ( tmpRemote , baseBranch , headBranch )
2019-04-09 21:45:58 +01:00
if err == nil {
2021-02-16 15:39:45 +00:00
compareInfo . BaseCommitID , err = GetFullCommitID ( repo . Path , remoteBranch )
if err != nil {
compareInfo . BaseCommitID = remoteBranch
}
2020-07-29 18:53:04 +01:00
// We have a common base - therefore we know that ... should work
2019-06-07 22:29:29 +02:00
logs , err := NewCommand ( "log" , compareInfo . MergeBase + "..." + headBranch , prettyLogFormat ) . RunInDirBytes ( repo . Path )
2019-04-09 21:45:58 +01:00
if err != nil {
return nil , err
}
2019-06-07 22:29:29 +02:00
compareInfo . Commits , err = repo . parsePrettyFormatLogToList ( logs )
2019-04-09 21:45:58 +01:00
if err != nil {
return nil , fmt . Errorf ( "parsePrettyFormatLogToList: %v" , err )
}
} else {
2019-06-07 22:29:29 +02:00
compareInfo . Commits = list . New ( )
compareInfo . MergeBase , err = GetFullCommitID ( repo . Path , remoteBranch )
2019-04-09 21:45:58 +01:00
if err != nil {
2019-06-07 22:29:29 +02:00
compareInfo . MergeBase = remoteBranch
2019-04-09 21:45:58 +01:00
}
2021-02-16 15:39:45 +00:00
compareInfo . BaseCommitID = compareInfo . MergeBase
2016-11-03 23:16:01 +01:00
}
// Count number of changed files.
2020-05-26 06:58:07 +01:00
// 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 )
2016-11-03 23:16:01 +01:00
if err != nil {
return nil , err
}
2019-06-07 22:29:29 +02:00
return compareInfo , nil
2016-11-03 23:16:01 +01:00
}
2020-05-26 06:58:07 +01:00
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 {
2020-07-29 18:53:04 +01:00
if strings . Contains ( stderr . String ( ) , "no merge base" ) {
// git >= 2.28 now returns an error if base and head have become unrelated.
// previously it would return the results of git diff -z --name-only base head so let's try that...
w = & lineCountWriter { }
stderr . Reset ( )
if err = NewCommand ( "diff" , "-z" , "--name-only" , base , head ) . RunInDirPipeline ( repo . Path , w , stderr ) ; err == nil {
return w . numLines , nil
}
}
2020-05-26 06:58:07 +01:00
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 ) {
2020-07-29 18:53:04 +01:00
numFiles , totalAdditions , totalDeletions , err = GetDiffShortStat ( repo . Path , base + "..." + head )
if err != nil && strings . Contains ( err . Error ( ) , "no merge base" ) {
return GetDiffShortStat ( repo . Path , base , head )
}
return
2020-05-26 06:58:07 +01:00
}
// 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
}
2019-12-13 22:21:06 +00:00
// GetDiffOrPatch generates either diff or formatted patch data between given revisions
func ( repo * Repository ) GetDiffOrPatch ( base , head string , w io . Writer , formatted bool ) error {
if formatted {
return repo . GetPatch ( base , head , w )
}
return repo . GetDiff ( base , head , w )
2016-11-03 23:16:01 +01:00
}
2018-01-07 14:10:20 +01:00
2019-12-13 22:21:06 +00:00
// GetDiff generates and returns patch data between given revisions.
func ( repo * Repository ) GetDiff ( base , head string , w io . Writer ) error {
return NewCommand ( "diff" , "-p" , "--binary" , base , head ) .
RunInDirPipeline ( repo . Path , w , nil )
}
2018-01-07 14:10:20 +01:00
2019-12-13 22:21:06 +00:00
// GetPatch generates and returns format-patch data between given revisions.
func ( repo * Repository ) GetPatch ( base , head string , w io . Writer ) error {
2020-07-29 18:53:04 +01:00
stderr := new ( bytes . Buffer )
err := NewCommand ( "format-patch" , "--binary" , "--stdout" , base + "..." + head ) .
RunInDirPipeline ( repo . Path , w , stderr )
if err != nil && bytes . Contains ( stderr . Bytes ( ) , [ ] byte ( "no merge base" ) ) {
return NewCommand ( "format-patch" , "--binary" , "--stdout" , base , head ) .
RunInDirPipeline ( repo . Path , w , nil )
}
return err
2018-01-07 14:10:20 +01:00
}
2020-01-09 02:47:45 +01:00
// GetDiffFromMergeBase generates and return patch data from merge base to head
func ( repo * Repository ) GetDiffFromMergeBase ( base , head string , w io . Writer ) error {
2020-07-29 18:53:04 +01:00
stderr := new ( bytes . Buffer )
err := NewCommand ( "diff" , "-p" , "--binary" , base + "..." + head ) .
RunInDirPipeline ( repo . Path , w , stderr )
if err != nil && bytes . Contains ( stderr . Bytes ( ) , [ ] byte ( "no merge base" ) ) {
return NewCommand ( "diff" , "-p" , "--binary" , base , head ) .
RunInDirPipeline ( repo . Path , w , nil )
}
return err
2020-01-09 02:47:45 +01:00
}