[code-953] diff merge base with annotated tags (#809)

This commit is contained in:
Enver Bisevac 2023-11-16 17:51:08 +00:00 committed by Harness
parent ab9089d613
commit 6f83d8d07b
5 changed files with 428 additions and 3 deletions

View File

@ -39,6 +39,17 @@ func (a Adapter) RawDiff(
if repoPath == "" {
return ErrRepositoryPathEmpty
}
baseTag, err := a.GetAnnotatedTag(ctx, repoPath, baseRef)
if err == nil {
baseRef = baseTag.TargetSha
}
headTag, err := a.GetAnnotatedTag(ctx, repoPath, headRef)
if err == nil {
headRef = headTag.TargetSha
}
args := make([]string, 0, 8)
args = append(args, "diff", "-M", "--full-index")
if mergeBase {
@ -57,7 +68,7 @@ func (a Adapter) RawDiff(
if errbuf.Len() > 0 {
err = &runStdError{err: err, stderr: errbuf.String()}
}
return processGiteaErrorf(err, "git diff failed between '%s' and '%s' with err: %v", baseRef, headRef, err)
return processGiteaErrorf(err, "git diff failed between '%s' and '%s'", baseRef, headRef)
}
return nil
}

158
git/adapter/diff_test.go Normal file
View File

@ -0,0 +1,158 @@
// Copyright 2023 Harness, Inc.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package adapter_test
import (
"bytes"
"context"
"testing"
"github.com/harness/gitness/git/adapter"
)
func TestAdapter_RawDiff(t *testing.T) {
git := setupGit(t)
repo, teardown := setupRepo(t, git, "testrawdiff")
defer teardown()
baseBranch := "main"
// write file to main branch
parentSHA := writeFile(t, repo, "file.txt", "some content", nil)
err := repo.SetReference("refs/heads/"+baseBranch, parentSHA.String())
if err != nil {
t.Fatalf("failed updating reference '%s': %v", baseBranch, err)
}
baseTag := "0.0.1"
err = repo.CreateAnnotatedTag(baseTag, "test tag 1", parentSHA.String())
if err != nil {
t.Fatalf("error creating annotated tag '%s': %v", baseTag, err)
}
headBranch := "dev"
// create branch
err = repo.CreateBranch(headBranch, baseBranch)
if err != nil {
t.Fatalf("failed creating a branch '%s': %v", headBranch, err)
}
// write file to main branch
sha := writeFile(t, repo, "file.txt", "new content", []string{parentSHA.String()})
err = repo.SetReference("refs/heads/"+headBranch, sha.String())
if err != nil {
t.Fatalf("failed updating reference '%s': %v", headBranch, err)
}
headTag := "0.0.2"
err = repo.CreateAnnotatedTag(headTag, "test tag 2", sha.String())
if err != nil {
t.Fatalf("error creating annotated tag '%s': %v", headTag, err)
}
want := `diff --git a/file.txt b/file.txt
index f0eec86f614944a81f87d879ebdc9a79aea0d7ea..47d2739ba2c34690248c8f91b84bb54e8936899a 100644
--- a/file.txt
+++ b/file.txt
@@ -1 +1 @@
-some content
\ No newline at end of file
+new content
\ No newline at end of file
`
type args struct {
ctx context.Context
repoPath string
baseRef string
headRef string
mergeBase bool
}
tests := []struct {
name string
adapter adapter.Adapter
args args
wantW string
wantErr bool
}{
{
name: "test branches",
adapter: git,
args: args{
ctx: context.Background(),
repoPath: repo.Path,
baseRef: baseBranch,
headRef: headBranch,
mergeBase: false,
},
wantW: want,
wantErr: false,
},
{
name: "test annotated tag",
adapter: git,
args: args{
ctx: context.Background(),
repoPath: repo.Path,
baseRef: baseTag,
headRef: headTag,
mergeBase: false,
},
wantW: want,
wantErr: false,
},
{
name: "test branches using merge-base",
adapter: git,
args: args{
ctx: context.Background(),
repoPath: repo.Path,
baseRef: baseBranch,
headRef: headBranch,
mergeBase: true,
},
wantW: want,
wantErr: false,
},
{
name: "test annotated tag using merge-base",
adapter: git,
args: args{
ctx: context.Background(),
repoPath: repo.Path,
baseRef: baseTag,
headRef: headTag,
mergeBase: true,
},
wantW: want,
wantErr: false,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
w := &bytes.Buffer{}
err := tt.adapter.RawDiff(tt.args.ctx, tt.args.repoPath, tt.args.baseRef, tt.args.headRef, tt.args.mergeBase, w)
if (err != nil) != tt.wantErr {
t.Errorf("RawDiff() error = %v, wantErr %v", err, tt.wantErr)
return
}
if gotW := w.String(); gotW != tt.wantW {
t.Errorf("RawDiff() gotW = %v, want %v", gotW, tt.wantW)
}
})
}
}

View File

@ -547,9 +547,9 @@ func (a Adapter) GetMergeBase(
}
}
stdout, _, err := git.NewCommand(ctx, "merge-base", "--", base, head).RunStdString(&git.RunOpts{Dir: repoPath})
stdout, stderr, err := git.NewCommand(ctx, "merge-base", "--", base, head).RunStdString(&git.RunOpts{Dir: repoPath})
if err != nil {
return "", "", processGiteaErrorf(err, "failed to get merge-base")
return "", "", processGiteaErrorf(err, "failed to get merge-base: %v", stderr)
}
return strings.TrimSpace(stdout), base, nil

123
git/adapter/merge_test.go Normal file
View File

@ -0,0 +1,123 @@
// Copyright 2023 Harness, Inc.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package adapter_test
import (
"context"
"testing"
"github.com/harness/gitness/git/adapter"
)
func TestAdapter_GetMergeBase(t *testing.T) {
git := setupGit(t)
repo, teardown := setupRepo(t, git, "testmergebase")
defer teardown()
baseBranch := "main"
// write file to main branch
parentSHA := writeFile(t, repo, "file1.txt", "some content", nil)
err := repo.SetReference("refs/heads/"+baseBranch, parentSHA.String())
if err != nil {
t.Fatalf("failed updating reference '%s': %v", baseBranch, err)
}
baseTag := "0.0.1"
err = repo.CreateAnnotatedTag(baseTag, "test tag 1", parentSHA.String())
if err != nil {
t.Fatalf("error creating annotated tag '%s': %v", baseTag, err)
}
headBranch := "dev"
// create branch
err = repo.CreateBranch(headBranch, baseBranch)
if err != nil {
t.Fatalf("failed creating a branch '%s': %v", headBranch, err)
}
// write file to main branch
sha := writeFile(t, repo, "file1.txt", "new content", []string{parentSHA.String()})
err = repo.SetReference("refs/heads/"+headBranch, sha.String())
if err != nil {
t.Fatalf("failed updating reference '%s': %v", headBranch, err)
}
headTag := "0.0.2"
err = repo.CreateAnnotatedTag(headTag, "test tag 2", sha.String())
if err != nil {
t.Fatalf("error creating annotated tag '%s': %v", headTag, err)
}
type args struct {
ctx context.Context
repoPath string
remote string
base string
head string
}
tests := []struct {
name string
git adapter.Adapter
args args
want string
want1 string
wantErr bool
}{
{
name: "git merge base using branch names",
git: git,
args: args{
ctx: context.Background(),
repoPath: repo.Path,
remote: "",
base: baseBranch,
head: headBranch,
},
want: parentSHA.String(),
want1: baseBranch,
},
{
name: "git merge base using annotated tags",
git: git,
args: args{
ctx: context.Background(),
repoPath: repo.Path,
remote: "",
base: baseTag,
head: headTag,
},
want: parentSHA.String(),
want1: baseTag,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
got, got1, err := tt.git.GetMergeBase(tt.args.ctx, tt.args.repoPath, tt.args.remote, tt.args.base, tt.args.head)
if (err != nil) != tt.wantErr {
t.Errorf("GetMergeBase() error = %v, wantErr %v", err, tt.wantErr)
return
}
if got != tt.want {
t.Errorf("GetMergeBase() got = %v, want %v", got, tt.want)
}
if got1 != tt.want1 {
t.Errorf("GetMergeBase() got1 = %v, want %v", got1, tt.want1)
}
})
}
}

133
git/adapter/setup_test.go Normal file
View File

@ -0,0 +1,133 @@
// Copyright 2023 Harness, Inc.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package adapter_test
import (
"context"
"os"
"path"
"strings"
"testing"
"github.com/harness/gitness/git/adapter"
"github.com/harness/gitness/types"
gitea "code.gitea.io/gitea/modules/git"
)
type teardown func()
var (
testAuthor = &gitea.Signature{
Name: "test",
Email: "test@test.com",
}
testCommitter = &gitea.Signature{
Name: "test",
Email: "test@test.com",
}
)
func setupGit(t *testing.T) adapter.Adapter {
t.Helper()
config := &types.Config{}
gogitProvider := adapter.ProvideGoGitRepoProvider()
git, err := adapter.New(
gogitProvider,
adapter.ProvideLastCommitCache(
config,
nil,
gogitProvider,
),
)
if err != nil {
t.Fatalf("error initializing repository: %v", err)
}
return git
}
func setupRepo(t *testing.T, git adapter.Adapter, name string) (*gitea.Repository, teardown) {
t.Helper()
ctx := context.Background()
tmpdir := os.TempDir()
repoPath := path.Join(tmpdir, "test_repos", name)
err := git.InitRepository(ctx, repoPath, true)
if err != nil {
t.Fatalf("error initializing repository: %v", err)
}
repo, err := git.OpenRepository(ctx, repoPath)
if err != nil {
t.Fatalf("error opening repository '%s': %v", name, err)
}
err = repo.SetDefaultBranch("main")
if err != nil {
t.Fatalf("error setting default branch 'main': %v", err)
}
err = git.Config(ctx, repoPath, "user.email", testCommitter.Email)
if err != nil {
t.Fatalf("error setting config user.email %s: %v", testCommitter.Email, err)
}
err = git.Config(ctx, repoPath, "user.name", testCommitter.Name)
if err != nil {
t.Fatalf("error setting config user.name %s: %v", testCommitter.Name, err)
}
return repo, func() {
if err := os.RemoveAll(repoPath); err != nil {
t.Errorf("error while removeng the repository '%s'", repoPath)
}
}
}
func writeFile(
t *testing.T,
repo *gitea.Repository,
path string,
content string,
parents []string,
) gitea.SHA1 {
t.Helper()
sha, err := repo.HashObject(strings.NewReader(content))
if err != nil {
t.Fatalf("failed to hash object: %v", err)
}
err = repo.AddObjectToIndex("100644", sha, path)
if err != nil {
t.Fatalf("failed to add object to index: %v", err)
}
tree, err := repo.WriteTree()
if err != nil {
t.Fatalf("failed to write tree: %v", err)
}
sha, err = repo.CommitTree(testAuthor, testCommitter, tree, gitea.CommitTreeOpts{
Message: "write file operation",
Parents: parents,
})
if err != nil {
t.Fatalf("failed to commit tree: %v", err)
}
return sha
}