| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240 |
- // Copyright 2024 The Gitea Authors. All rights reserved.
- // SPDX-License-Identifier: MIT
-
- package integration
-
- import (
- "bytes"
- "io"
- "net/url"
- "sync"
- "testing"
-
- auth_model "code.gitea.io/gitea/models/auth"
- issues_model "code.gitea.io/gitea/models/issues"
- repo_model "code.gitea.io/gitea/models/repo"
- "code.gitea.io/gitea/models/unittest"
- user_model "code.gitea.io/gitea/models/user"
- "code.gitea.io/gitea/modules/git"
- "code.gitea.io/gitea/modules/git/gitcmd"
- "code.gitea.io/gitea/modules/gitrepo"
- files_service "code.gitea.io/gitea/services/repository/files"
-
- "github.com/stretchr/testify/assert"
- )
-
- func TestDataAsyncDoubleRead_Issue29101(t *testing.T) {
- onGiteaRun(t, func(t *testing.T, u *url.URL) {
- user := unittest.AssertExistsAndLoadBean(t, &user_model.User{ID: 2})
- repo := unittest.AssertExistsAndLoadBean(t, &repo_model.Repository{ID: 1})
-
- testContent := bytes.Repeat([]byte{'a'}, 10000)
- resp, err := files_service.ChangeRepoFiles(t.Context(), repo, user, &files_service.ChangeRepoFilesOptions{
- Files: []*files_service.ChangeRepoFile{
- {
- Operation: "create",
- TreePath: "test.txt",
- ContentReader: bytes.NewReader(testContent),
- },
- },
- OldBranch: repo.DefaultBranch,
- NewBranch: repo.DefaultBranch,
- })
- assert.NoError(t, err)
-
- sha := resp.Commit.SHA
-
- gitRepo, err := gitrepo.OpenRepository(t.Context(), repo)
- assert.NoError(t, err)
-
- commit, err := gitRepo.GetCommit(sha)
- assert.NoError(t, err)
-
- entry, err := commit.GetTreeEntryByPath("test.txt")
- assert.NoError(t, err)
-
- b := entry.Blob()
- r1, err := b.DataAsync()
- assert.NoError(t, err)
- defer r1.Close()
- r2, err := b.DataAsync()
- assert.NoError(t, err)
- defer r2.Close()
-
- var data1, data2 []byte
- wg := sync.WaitGroup{}
- wg.Add(2)
- go func() {
- data1, _ = io.ReadAll(r1)
- assert.NoError(t, err)
- wg.Done()
- }()
- go func() {
- data2, _ = io.ReadAll(r2)
- assert.NoError(t, err)
- wg.Done()
- }()
- wg.Wait()
- assert.Equal(t, testContent, data1)
- assert.Equal(t, testContent, data2)
- })
- }
-
- func TestAgitPullPush(t *testing.T) {
- onGiteaRun(t, func(t *testing.T, u *url.URL) {
- baseAPITestContext := NewAPITestContext(t, "user2", "repo1", auth_model.AccessTokenScopeWriteRepository, auth_model.AccessTokenScopeWriteUser)
-
- u.Path = baseAPITestContext.GitPath()
- u.User = url.UserPassword("user2", userPassword)
-
- dstPath := t.TempDir()
- doGitClone(dstPath, u)(t)
-
- gitRepo, err := git.OpenRepository(t.Context(), dstPath)
- assert.NoError(t, err)
- defer gitRepo.Close()
-
- doGitCreateBranch(dstPath, "test-agit-push")
-
- // commit 1
- _, err = generateCommitWithNewData(t.Context(), testFileSizeSmall, dstPath, "user2@example.com", "User Two", "branch-data-file-")
- assert.NoError(t, err)
-
- // push to create an agit pull request
- err = gitcmd.NewCommand("push", "origin",
- "-o", "title=test-title", "-o", "description=test-description",
- "HEAD:refs/for/master/test-agit-push",
- ).Run(t.Context(), &gitcmd.RunOpts{Dir: dstPath})
- assert.NoError(t, err)
-
- // check pull request exist
- pr := unittest.AssertExistsAndLoadBean(t, &issues_model.PullRequest{BaseRepoID: 1, Flow: issues_model.PullRequestFlowAGit, HeadBranch: "user2/test-agit-push"})
- assert.NoError(t, pr.LoadIssue(t.Context()))
- assert.Equal(t, "test-title", pr.Issue.Title)
- assert.Equal(t, "test-description", pr.Issue.Content)
-
- // commit 2
- _, err = generateCommitWithNewData(t.Context(), testFileSizeSmall, dstPath, "user2@example.com", "User Two", "branch-data-file-2-")
- assert.NoError(t, err)
-
- // push 2
- err = gitcmd.NewCommand("push", "origin", "HEAD:refs/for/master/test-agit-push").Run(t.Context(), &gitcmd.RunOpts{Dir: dstPath})
- assert.NoError(t, err)
-
- // reset to first commit
- err = gitcmd.NewCommand("reset", "--hard", "HEAD~1").Run(t.Context(), &gitcmd.RunOpts{Dir: dstPath})
- assert.NoError(t, err)
-
- // test force push without confirm
- _, stderr, err := gitcmd.NewCommand("push", "origin", "HEAD:refs/for/master/test-agit-push").RunStdString(t.Context(), &gitcmd.RunOpts{Dir: dstPath})
- assert.Error(t, err)
- assert.Contains(t, stderr, "[remote rejected] HEAD -> refs/for/master/test-agit-push (request `force-push` push option)")
-
- // test force push with confirm
- err = gitcmd.NewCommand("push", "origin", "HEAD:refs/for/master/test-agit-push", "-o", "force-push").Run(t.Context(), &gitcmd.RunOpts{Dir: dstPath})
- assert.NoError(t, err)
- })
- }
-
- func TestAgitReviewStaleness(t *testing.T) {
- onGiteaRun(t, func(t *testing.T, u *url.URL) {
- baseAPITestContext := NewAPITestContext(t, "user2", "repo1", auth_model.AccessTokenScopeWriteRepository, auth_model.AccessTokenScopeWriteUser)
-
- u.Path = baseAPITestContext.GitPath()
- u.User = url.UserPassword("user2", userPassword)
-
- dstPath := t.TempDir()
- doGitClone(dstPath, u)(t)
-
- gitRepo, err := git.OpenRepository(t.Context(), dstPath)
- assert.NoError(t, err)
- defer gitRepo.Close()
-
- doGitCreateBranch(dstPath, "test-agit-review")
-
- // Create initial commit
- _, err = generateCommitWithNewData(t.Context(), testFileSizeSmall, dstPath, "user2@example.com", "User Two", "initial-")
- assert.NoError(t, err)
-
- // create PR via agit
- err = gitcmd.NewCommand("push", "origin",
- "-o", "title=Test agit Review Staleness", "-o", "description=Testing review staleness",
- "HEAD:refs/for/master/test-agit-review",
- ).Run(t.Context(), &gitcmd.RunOpts{Dir: dstPath})
- assert.NoError(t, err)
-
- pr := unittest.AssertExistsAndLoadBean(t, &issues_model.PullRequest{
- BaseRepoID: 1,
- Flow: issues_model.PullRequestFlowAGit,
- HeadBranch: "user2/test-agit-review",
- })
- assert.NoError(t, pr.LoadIssue(t.Context()))
-
- // Get initial commit ID for the review
- initialCommitID := pr.HeadCommitID
- t.Logf("Initial commit ID: %s", initialCommitID)
-
- // Create a review on the PR (as user1 reviewing user2's PR)
- reviewer := unittest.AssertExistsAndLoadBean(t, &user_model.User{ID: 1})
- review, err := issues_model.CreateReview(t.Context(), issues_model.CreateReviewOptions{
- Type: issues_model.ReviewTypeApprove,
- Reviewer: reviewer,
- Issue: pr.Issue,
- CommitID: initialCommitID,
- Content: "LGTM! Looks good to merge.",
- Official: false,
- })
- assert.NoError(t, err)
- assert.False(t, review.Stale, "New review should not be stale")
-
- // Verify review exists and is not stale
- reviews, err := issues_model.FindReviews(t.Context(), issues_model.FindReviewOptions{
- IssueID: pr.IssueID,
- })
- assert.NoError(t, err)
- assert.Len(t, reviews, 1)
- assert.Equal(t, initialCommitID, reviews[0].CommitID)
- assert.False(t, reviews[0].Stale, "Review should not be stale initially")
-
- // Create a new commit and update the agit PR
- _, err = generateCommitWithNewData(t.Context(), testFileSizeSmall, dstPath, "user2@example.com", "User Two", "updated-")
- assert.NoError(t, err)
-
- err = gitcmd.NewCommand("push", "origin", "HEAD:refs/for/master/test-agit-review").Run(t.Context(), &gitcmd.RunOpts{Dir: dstPath})
- assert.NoError(t, err)
-
- // Reload PR to get updated commit ID
- pr = unittest.AssertExistsAndLoadBean(t, &issues_model.PullRequest{
- BaseRepoID: 1,
- Flow: issues_model.PullRequestFlowAGit,
- HeadBranch: "user2/test-agit-review",
- })
- assert.NoError(t, pr.LoadIssue(t.Context()))
-
- // For AGit PRs, HeadCommitID must be loaded from git references
- baseRepo := unittest.AssertExistsAndLoadBean(t, &repo_model.Repository{ID: 1})
- baseGitRepo, err := gitrepo.OpenRepository(t.Context(), baseRepo)
- assert.NoError(t, err)
- defer baseGitRepo.Close()
-
- updatedCommitID, err := baseGitRepo.GetRefCommitID(pr.GetGitHeadRefName())
- assert.NoError(t, err)
- t.Logf("Updated commit ID: %s", updatedCommitID)
-
- // Verify the PR was updated with new commit
- assert.NotEqual(t, initialCommitID, updatedCommitID, "PR should have new commit ID after update")
-
- // Check that the review is now marked as stale
- reviews, err = issues_model.FindReviews(t.Context(), issues_model.FindReviewOptions{
- IssueID: pr.IssueID,
- })
- assert.NoError(t, err)
- assert.Len(t, reviews, 1)
-
- assert.True(t, reviews[0].Stale, "Review should be marked as stale after AGit PR update")
-
- // The review commit ID should remain the same (pointing to the original commit)
- assert.Equal(t, initialCommitID, reviews[0].CommitID, "Review commit ID should remain unchanged and point to original commit")
- })
- }
|