From 0749bb54847e17fb7a2376e392871049f44793e7 Mon Sep 17 00:00:00 2001 From: Lunny Xiao Date: Thu, 12 Dec 2024 15:31:58 -0800 Subject: [PATCH 1/3] Refactor getpatch/getdiff functions and remove fallback automatically --- modules/git/repo_compare.go | 90 ++++++++++++-------------------- modules/git/repo_compare_test.go | 27 +++++++++- services/pull/patch.go | 18 +++++-- 3 files changed, 73 insertions(+), 62 deletions(-) diff --git a/modules/git/repo_compare.go b/modules/git/repo_compare.go index 16fcdcf4c8f96..5d7fbe6df64c6 100644 --- a/modules/git/repo_compare.go +++ b/modules/git/repo_compare.go @@ -233,72 +233,63 @@ func parseDiffStat(stdout string) (numFiles, totalAdditions, totalDeletions int, return numFiles, totalAdditions, totalDeletions, err } -// GetDiffOrPatch generates either diff or formatted patch data between given revisions -func (repo *Repository) GetDiffOrPatch(base, head string, w io.Writer, patch, binary bool) error { - if patch { - return repo.GetPatch(base, head, w) +func parseCompareArgs(compareArgs string) (args []string) { + parts := strings.Split(compareArgs, "...") + if len(parts) == 2 { + return []string{compareArgs} } - if binary { - return repo.GetDiffBinary(base, head, w) + parts = strings.Split(compareArgs, "..") + if len(parts) == 2 { + return parts } - return repo.GetDiff(base, head, w) + parts = strings.Fields(compareArgs) + if len(parts) == 2 { + return parts + } + + return nil } // GetDiff generates and returns patch data between given revisions, optimized for human readability -func (repo *Repository) GetDiff(base, head string, w io.Writer) error { +func (repo *Repository) GetDiff(compareArgs string, w io.Writer) error { + args := parseCompareArgs(compareArgs) + if len(args) == 0 { + return fmt.Errorf("invalid compareArgs: %s", compareArgs) + } stderr := new(bytes.Buffer) - err := NewCommand(repo.Ctx, "diff", "-p").AddDynamicArguments(base + "..." + head). + return NewCommand(repo.Ctx, "diff", "-p").AddDynamicArguments(args...). Run(&RunOpts{ Dir: repo.Path, Stdout: w, Stderr: stderr, }) - if err != nil && bytes.Contains(stderr.Bytes(), []byte("no merge base")) { - return NewCommand(repo.Ctx, "diff", "-p").AddDynamicArguments(base, head). - Run(&RunOpts{ - Dir: repo.Path, - Stdout: w, - }) - } - return err } // GetDiffBinary generates and returns patch data between given revisions, including binary diffs. -func (repo *Repository) GetDiffBinary(base, head string, w io.Writer) error { - stderr := new(bytes.Buffer) - err := NewCommand(repo.Ctx, "diff", "-p", "--binary", "--histogram").AddDynamicArguments(base + "..." + head). - Run(&RunOpts{ - Dir: repo.Path, - Stdout: w, - Stderr: stderr, - }) - if err != nil && bytes.Contains(stderr.Bytes(), []byte("no merge base")) { - return NewCommand(repo.Ctx, "diff", "-p", "--binary", "--histogram").AddDynamicArguments(base, head). - Run(&RunOpts{ - Dir: repo.Path, - Stdout: w, - }) +func (repo *Repository) GetDiffBinary(compareArgs string, w io.Writer) error { + args := parseCompareArgs(compareArgs) + if len(args) == 0 { + return fmt.Errorf("invalid compareArgs: %s", compareArgs) } - return err + return NewCommand(repo.Ctx, "diff", "-p", "--binary", "--histogram").AddDynamicArguments(args...).Run(&RunOpts{ + Dir: repo.Path, + Stdout: w, + }) } // GetPatch generates and returns format-patch data between given revisions, able to be used with `git apply` -func (repo *Repository) GetPatch(base, head string, w io.Writer) error { +func (repo *Repository) GetPatch(compareArgs string, w io.Writer) error { + args := parseCompareArgs(compareArgs) + if len(args) == 0 { + return fmt.Errorf("invalid compareArgs: %s", compareArgs) + } stderr := new(bytes.Buffer) - err := NewCommand(repo.Ctx, "format-patch", "--binary", "--stdout").AddDynamicArguments(base + "..." + head). + return NewCommand(repo.Ctx, "format-patch", "--binary", "--stdout").AddDynamicArguments(args...). Run(&RunOpts{ Dir: repo.Path, Stdout: w, Stderr: stderr, }) - if err != nil && bytes.Contains(stderr.Bytes(), []byte("no merge base")) { - return NewCommand(repo.Ctx, "format-patch", "--binary", "--stdout").AddDynamicArguments(base, head). - Run(&RunOpts{ - Dir: repo.Path, - Stdout: w, - }) - } - return err } // GetFilesChangedBetween returns a list of all files that have been changed between the given commits @@ -329,21 +320,6 @@ func (repo *Repository) GetFilesChangedBetween(base, head string) ([]string, err return split, err } -// GetDiffFromMergeBase generates and return patch data from merge base to head -func (repo *Repository) GetDiffFromMergeBase(base, head string, w io.Writer) error { - stderr := new(bytes.Buffer) - err := NewCommand(repo.Ctx, "diff", "-p", "--binary").AddDynamicArguments(base + "..." + head). - Run(&RunOpts{ - Dir: repo.Path, - Stdout: w, - Stderr: stderr, - }) - if err != nil && bytes.Contains(stderr.Bytes(), []byte("no merge base")) { - return repo.GetDiffBinary(base, head, w) - } - return err -} - // ReadPatchCommit will check if a diff patch exists and return stats func (repo *Repository) ReadPatchCommit(prID int64) (commitSHA string, err error) { // Migrated repositories download patches to "pulls" location diff --git a/modules/git/repo_compare_test.go b/modules/git/repo_compare_test.go index 9983873186720..d4597bd948682 100644 --- a/modules/git/repo_compare_test.go +++ b/modules/git/repo_compare_test.go @@ -12,6 +12,31 @@ import ( "github.com/stretchr/testify/assert" ) +func Test_parseCompareArgs(t *testing.T) { + testCases := []struct { + compareString string + expected []string + }{ + { + "master..develop", + []string{"master", "develop"}, + }, + { + "master HEAD", + []string{"master", "HEAD"}, + }, + { + "HEAD...develop", + []string{"HEAD...develop"}, + }, + } + + for _, tc := range testCases { + args := parseCompareArgs(tc.compareString) + assert.Equal(t, tc.expected, args) + } +} + func TestGetFormatPatch(t *testing.T) { bareRepo1Path := filepath.Join(testReposDir, "repo1_bare") clonedPath, err := cloneRepo(t, bareRepo1Path) @@ -28,7 +53,7 @@ func TestGetFormatPatch(t *testing.T) { defer repo.Close() rd := &bytes.Buffer{} - err = repo.GetPatch("8d92fc95^", "8d92fc95", rd) + err = repo.GetPatch("8d92fc95^...8d92fc95", rd) if err != nil { assert.NoError(t, err) return diff --git a/services/pull/patch.go b/services/pull/patch.go index 0934a86c89a65..296a84bb429bd 100644 --- a/services/pull/patch.go +++ b/services/pull/patch.go @@ -42,9 +42,19 @@ func DownloadDiffOrPatch(ctx context.Context, pr *issues_model.PullRequest, w io } defer closer.Close() - if err := gitRepo.GetDiffOrPatch(pr.MergeBase, pr.GetGitRefName(), w, patch, binary); err != nil { - log.Error("Unable to get patch file from %s to %s in %s Error: %v", pr.MergeBase, pr.HeadBranch, pr.BaseRepo.FullName(), err) - return fmt.Errorf("Unable to get patch file from %s to %s in %s Error: %w", pr.MergeBase, pr.HeadBranch, pr.BaseRepo.FullName(), err) + compareString := pr.MergeBase + "..." + pr.GetGitRefName() + switch { + case patch: + err = gitRepo.GetPatch(compareString, w) + case binary: + err = gitRepo.GetDiffBinary(compareString, w) + default: + err = gitRepo.GetDiff(compareString, w) + } + + if err != nil { + log.Error("unable to get patch file from %s to %s in %s Error: %v", pr.MergeBase, pr.HeadBranch, pr.BaseRepo.FullName(), err) + return fmt.Errorf("unable to get patch file from %s to %s in %s Error: %w", pr.MergeBase, pr.HeadBranch, pr.BaseRepo.FullName(), err) } return nil } @@ -355,7 +365,7 @@ func checkConflicts(ctx context.Context, pr *issues_model.PullRequest, gitRepo * _ = util.Remove(tmpPatchFile.Name()) }() - if err := gitRepo.GetDiffBinary(pr.MergeBase, "tracking", tmpPatchFile); err != nil { + if err := gitRepo.GetDiffBinary(pr.MergeBase+"...tracking", tmpPatchFile); err != nil { tmpPatchFile.Close() log.Error("Unable to get patch file from %s to %s in %s Error: %v", pr.MergeBase, pr.HeadBranch, pr.BaseRepo.FullName(), err) return false, fmt.Errorf("unable to get patch file from %s to %s in %s Error: %w", pr.MergeBase, pr.HeadBranch, pr.BaseRepo.FullName(), err) From b30075a9c7855528b0f1d11af9b9900ceba9f683 Mon Sep 17 00:00:00 2001 From: Lunny Xiao Date: Mon, 23 Dec 2024 21:04:09 -0800 Subject: [PATCH 2/3] Just allow a...b and a..b two kinds of parameter --- modules/git/repo_compare.go | 8 ++++---- modules/git/repo_compare_test.go | 4 ---- 2 files changed, 4 insertions(+), 8 deletions(-) diff --git a/modules/git/repo_compare.go b/modules/git/repo_compare.go index 5d7fbe6df64c6..85aef50ff2552 100644 --- a/modules/git/repo_compare.go +++ b/modules/git/repo_compare.go @@ -233,6 +233,10 @@ func parseDiffStat(stdout string) (numFiles, totalAdditions, totalDeletions int, return numFiles, totalAdditions, totalDeletions, err } +// parseCompareArgs parses the compareArgs string into a slice of arguments +// Only supports the following formats: +// - "base...head" +// - "base..head" func parseCompareArgs(compareArgs string) (args []string) { parts := strings.Split(compareArgs, "...") if len(parts) == 2 { @@ -242,10 +246,6 @@ func parseCompareArgs(compareArgs string) (args []string) { if len(parts) == 2 { return parts } - parts = strings.Fields(compareArgs) - if len(parts) == 2 { - return parts - } return nil } diff --git a/modules/git/repo_compare_test.go b/modules/git/repo_compare_test.go index d4597bd948682..252d08f426f7c 100644 --- a/modules/git/repo_compare_test.go +++ b/modules/git/repo_compare_test.go @@ -21,10 +21,6 @@ func Test_parseCompareArgs(t *testing.T) { "master..develop", []string{"master", "develop"}, }, - { - "master HEAD", - []string{"master", "HEAD"}, - }, { "HEAD...develop", []string{"HEAD...develop"}, From db99fae0d0560b46190a18ebe0ebed9fef95c325 Mon Sep 17 00:00:00 2001 From: wxiaoguang Date: Tue, 24 Dec 2024 13:37:59 +0800 Subject: [PATCH 3/3] remove all unnecessary code --- modules/git/repo_compare.go | 41 +++++--------------------------- modules/git/repo_compare_test.go | 21 ---------------- services/pull/patch.go | 8 +++---- 3 files changed, 10 insertions(+), 60 deletions(-) diff --git a/modules/git/repo_compare.go b/modules/git/repo_compare.go index 85aef50ff2552..877a7ff3b86b0 100644 --- a/modules/git/repo_compare.go +++ b/modules/git/repo_compare.go @@ -233,31 +233,10 @@ func parseDiffStat(stdout string) (numFiles, totalAdditions, totalDeletions int, return numFiles, totalAdditions, totalDeletions, err } -// parseCompareArgs parses the compareArgs string into a slice of arguments -// Only supports the following formats: -// - "base...head" -// - "base..head" -func parseCompareArgs(compareArgs string) (args []string) { - parts := strings.Split(compareArgs, "...") - if len(parts) == 2 { - return []string{compareArgs} - } - parts = strings.Split(compareArgs, "..") - if len(parts) == 2 { - return parts - } - - return nil -} - // GetDiff generates and returns patch data between given revisions, optimized for human readability -func (repo *Repository) GetDiff(compareArgs string, w io.Writer) error { - args := parseCompareArgs(compareArgs) - if len(args) == 0 { - return fmt.Errorf("invalid compareArgs: %s", compareArgs) - } +func (repo *Repository) GetDiff(compareArg string, w io.Writer) error { stderr := new(bytes.Buffer) - return NewCommand(repo.Ctx, "diff", "-p").AddDynamicArguments(args...). + return NewCommand(repo.Ctx, "diff", "-p").AddDynamicArguments(compareArg). Run(&RunOpts{ Dir: repo.Path, Stdout: w, @@ -266,25 +245,17 @@ func (repo *Repository) GetDiff(compareArgs string, w io.Writer) error { } // GetDiffBinary generates and returns patch data between given revisions, including binary diffs. -func (repo *Repository) GetDiffBinary(compareArgs string, w io.Writer) error { - args := parseCompareArgs(compareArgs) - if len(args) == 0 { - return fmt.Errorf("invalid compareArgs: %s", compareArgs) - } - return NewCommand(repo.Ctx, "diff", "-p", "--binary", "--histogram").AddDynamicArguments(args...).Run(&RunOpts{ +func (repo *Repository) GetDiffBinary(compareArg string, w io.Writer) error { + return NewCommand(repo.Ctx, "diff", "-p", "--binary", "--histogram").AddDynamicArguments(compareArg).Run(&RunOpts{ Dir: repo.Path, Stdout: w, }) } // GetPatch generates and returns format-patch data between given revisions, able to be used with `git apply` -func (repo *Repository) GetPatch(compareArgs string, w io.Writer) error { - args := parseCompareArgs(compareArgs) - if len(args) == 0 { - return fmt.Errorf("invalid compareArgs: %s", compareArgs) - } +func (repo *Repository) GetPatch(compareArg string, w io.Writer) error { stderr := new(bytes.Buffer) - return NewCommand(repo.Ctx, "format-patch", "--binary", "--stdout").AddDynamicArguments(args...). + return NewCommand(repo.Ctx, "format-patch", "--binary", "--stdout").AddDynamicArguments(compareArg). Run(&RunOpts{ Dir: repo.Path, Stdout: w, diff --git a/modules/git/repo_compare_test.go b/modules/git/repo_compare_test.go index be9585b8efe8b..25ee4c5198568 100644 --- a/modules/git/repo_compare_test.go +++ b/modules/git/repo_compare_test.go @@ -12,27 +12,6 @@ import ( "github.com/stretchr/testify/assert" ) -func Test_parseCompareArgs(t *testing.T) { - testCases := []struct { - compareString string - expected []string - }{ - { - "master..develop", - []string{"master", "develop"}, - }, - { - "HEAD...develop", - []string{"HEAD...develop"}, - }, - } - - for _, tc := range testCases { - args := parseCompareArgs(tc.compareString) - assert.Equal(t, tc.expected, args) - } -} - func TestGetFormatPatch(t *testing.T) { bareRepo1Path := filepath.Join(testReposDir, "repo1_bare") clonedPath, err := cloneRepo(t, bareRepo1Path) diff --git a/services/pull/patch.go b/services/pull/patch.go index bc0dc579a230a..13623d73c6699 100644 --- a/services/pull/patch.go +++ b/services/pull/patch.go @@ -41,14 +41,14 @@ func DownloadDiffOrPatch(ctx context.Context, pr *issues_model.PullRequest, w io } defer closer.Close() - compareString := pr.MergeBase + "..." + pr.GetGitRefName() + compareArg := pr.MergeBase + "..." + pr.GetGitRefName() switch { case patch: - err = gitRepo.GetPatch(compareString, w) + err = gitRepo.GetPatch(compareArg, w) case binary: - err = gitRepo.GetDiffBinary(compareString, w) + err = gitRepo.GetDiffBinary(compareArg, w) default: - err = gitRepo.GetDiff(compareString, w) + err = gitRepo.GetDiff(compareArg, w) } if err != nil {