mirror of
				https://github.com/go-gitea/gitea.git
				synced 2025-10-27 00:23:41 +09:00 
			
		
		
		
	Fix wrong display of recently pushed notification (#25812)
There's a bug in #25715: If user pushed a commit into another repo with same branch name, the no-related repo will display the recently pushed notification incorrectly. It is simple to fix this, we should match the repo id in the sql query.  The latest commit is 2 weeks ago.  The notification comes from another repo with same branch name:  After: In forked repo:  New PR Link will redirect to the original repo:  In the original repo:  New PR Link:  In the same repo:  New PR Link:  08/15 Update: Follow #26257, added permission check and logic fix mentioned in https://github.com/go-gitea/gitea/pull/26257#discussion_r1294085203 2024/04/25 Update: Fix #30611 --------- Co-authored-by: silverwind <me@silverwind.io> Co-authored-by: Lunny Xiao <xiaolunwen@gmail.com> Co-authored-by: wxiaoguang <wxiaoguang@gmail.com>
This commit is contained in:
		| @@ -4,26 +4,37 @@ | ||||
| package integration | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"net/http" | ||||
| 	"net/url" | ||||
| 	"path" | ||||
| 	"strings" | ||||
| 	"testing" | ||||
|  | ||||
| 	auth_model "code.gitea.io/gitea/models/auth" | ||||
| 	org_model "code.gitea.io/gitea/models/organization" | ||||
| 	"code.gitea.io/gitea/models/perm" | ||||
| 	repo_model "code.gitea.io/gitea/models/repo" | ||||
| 	"code.gitea.io/gitea/models/unit" | ||||
| 	"code.gitea.io/gitea/models/unittest" | ||||
| 	"code.gitea.io/gitea/modules/setting" | ||||
| 	api "code.gitea.io/gitea/modules/structs" | ||||
| 	"code.gitea.io/gitea/modules/test" | ||||
| 	"code.gitea.io/gitea/modules/translation" | ||||
| 	"code.gitea.io/gitea/tests" | ||||
|  | ||||
| 	"github.com/PuerkitoBio/goquery" | ||||
| 	"github.com/stretchr/testify/assert" | ||||
| ) | ||||
|  | ||||
| func testCreateBranch(t testing.TB, session *TestSession, user, repo, oldRefSubURL, newBranchName string, expectedStatus int) string { | ||||
| 	var csrf string | ||||
| 	if expectedStatus == http.StatusNotFound { | ||||
| 		csrf = GetCSRF(t, session, path.Join(user, repo, "src/branch/master")) | ||||
| 		// src/branch/branch_name may not container "_csrf" input, | ||||
| 		// so we need to get it from cookies not from body | ||||
| 		csrf = GetCSRFFromCookie(t, session, path.Join(user, repo, "src/branch/master")) | ||||
| 	} else { | ||||
| 		csrf = GetCSRF(t, session, path.Join(user, repo, "src", oldRefSubURL)) | ||||
| 		csrf = GetCSRFFromCookie(t, session, path.Join(user, repo, "src", oldRefSubURL)) | ||||
| 	} | ||||
| 	req := NewRequestWithValues(t, "POST", path.Join(user, repo, "branches/_new", oldRefSubURL), map[string]string{ | ||||
| 		"_csrf":           csrf, | ||||
| @@ -145,3 +156,136 @@ func TestCreateBranchInvalidCSRF(t *testing.T) { | ||||
| 		strings.TrimSpace(htmlDoc.doc.Find(".ui.message").Text()), | ||||
| 	) | ||||
| } | ||||
|  | ||||
| func prepareBranch(t *testing.T, session *TestSession, repo *repo_model.Repository) { | ||||
| 	baseRefSubURL := fmt.Sprintf("branch/%s", repo.DefaultBranch) | ||||
|  | ||||
| 	// create branch with no new commit | ||||
| 	testCreateBranch(t, session, repo.OwnerName, repo.Name, baseRefSubURL, "no-commit", http.StatusSeeOther) | ||||
|  | ||||
| 	// create branch with commit | ||||
| 	testCreateBranch(t, session, repo.OwnerName, repo.Name, baseRefSubURL, "new-commit", http.StatusSeeOther) | ||||
| 	testAPINewFile(t, session, repo.OwnerName, repo.Name, "new-commit", "new-commit.txt", "new-commit") | ||||
|  | ||||
| 	// create deleted branch | ||||
| 	testCreateBranch(t, session, repo.OwnerName, repo.Name, "branch/new-commit", "deleted-branch", http.StatusSeeOther) | ||||
| 	testUIDeleteBranch(t, session, repo.OwnerName, repo.Name, "deleted-branch") | ||||
| } | ||||
|  | ||||
| func testCreatePullToDefaultBranch(t *testing.T, session *TestSession, baseRepo, headRepo *repo_model.Repository, headBranch, title string) string { | ||||
| 	srcRef := headBranch | ||||
| 	if baseRepo.ID != headRepo.ID { | ||||
| 		srcRef = fmt.Sprintf("%s/%s:%s", headRepo.OwnerName, headRepo.Name, headBranch) | ||||
| 	} | ||||
| 	resp := testPullCreate(t, session, baseRepo.OwnerName, baseRepo.Name, false, baseRepo.DefaultBranch, srcRef, title) | ||||
| 	elem := strings.Split(test.RedirectURL(resp), "/") | ||||
| 	// return pull request ID | ||||
| 	return elem[4] | ||||
| } | ||||
|  | ||||
| func prepareRepoPR(t *testing.T, baseSession, headSession *TestSession, baseRepo, headRepo *repo_model.Repository) { | ||||
| 	// create opening PR | ||||
| 	testCreateBranch(t, headSession, headRepo.OwnerName, headRepo.Name, "branch/new-commit", "opening-pr", http.StatusSeeOther) | ||||
| 	testCreatePullToDefaultBranch(t, baseSession, baseRepo, headRepo, "opening-pr", "opening pr") | ||||
|  | ||||
| 	// create closed PR | ||||
| 	testCreateBranch(t, headSession, headRepo.OwnerName, headRepo.Name, "branch/new-commit", "closed-pr", http.StatusSeeOther) | ||||
| 	prID := testCreatePullToDefaultBranch(t, baseSession, baseRepo, headRepo, "closed-pr", "closed pr") | ||||
| 	testIssueClose(t, baseSession, baseRepo.OwnerName, baseRepo.Name, prID) | ||||
|  | ||||
| 	// create closed PR with deleted branch | ||||
| 	testCreateBranch(t, headSession, headRepo.OwnerName, headRepo.Name, "branch/new-commit", "closed-pr-deleted", http.StatusSeeOther) | ||||
| 	prID = testCreatePullToDefaultBranch(t, baseSession, baseRepo, headRepo, "closed-pr-deleted", "closed pr with deleted branch") | ||||
| 	testIssueClose(t, baseSession, baseRepo.OwnerName, baseRepo.Name, prID) | ||||
| 	testUIDeleteBranch(t, headSession, headRepo.OwnerName, headRepo.Name, "closed-pr-deleted") | ||||
|  | ||||
| 	// create merged PR | ||||
| 	testCreateBranch(t, headSession, headRepo.OwnerName, headRepo.Name, "branch/new-commit", "merged-pr", http.StatusSeeOther) | ||||
| 	prID = testCreatePullToDefaultBranch(t, baseSession, baseRepo, headRepo, "merged-pr", "merged pr") | ||||
| 	testAPINewFile(t, headSession, headRepo.OwnerName, headRepo.Name, "merged-pr", fmt.Sprintf("new-commit-%s.txt", headRepo.Name), "new-commit") | ||||
| 	testPullMerge(t, baseSession, baseRepo.OwnerName, baseRepo.Name, prID, repo_model.MergeStyleRebaseMerge, false) | ||||
|  | ||||
| 	// create merged PR with deleted branch | ||||
| 	testCreateBranch(t, headSession, headRepo.OwnerName, headRepo.Name, "branch/new-commit", "merged-pr-deleted", http.StatusSeeOther) | ||||
| 	prID = testCreatePullToDefaultBranch(t, baseSession, baseRepo, headRepo, "merged-pr-deleted", "merged pr with deleted branch") | ||||
| 	testAPINewFile(t, headSession, headRepo.OwnerName, headRepo.Name, "merged-pr-deleted", fmt.Sprintf("new-commit-%s-2.txt", headRepo.Name), "new-commit") | ||||
| 	testPullMerge(t, baseSession, baseRepo.OwnerName, baseRepo.Name, prID, repo_model.MergeStyleRebaseMerge, true) | ||||
| } | ||||
|  | ||||
| func checkRecentlyPushedNewBranches(t *testing.T, session *TestSession, repoPath string, expected []string) { | ||||
| 	branches := make([]string, 0, 2) | ||||
| 	req := NewRequest(t, "GET", repoPath) | ||||
| 	resp := session.MakeRequest(t, req, http.StatusOK) | ||||
| 	doc := NewHTMLParser(t, resp.Body) | ||||
| 	doc.doc.Find(".ui.positive.message div a").Each(func(index int, branch *goquery.Selection) { | ||||
| 		branches = append(branches, branch.Text()) | ||||
| 	}) | ||||
| 	assert.Equal(t, expected, branches) | ||||
| } | ||||
|  | ||||
| func TestRecentlyPushedNewBranches(t *testing.T) { | ||||
| 	defer tests.PrepareTestEnv(t)() | ||||
|  | ||||
| 	onGiteaRun(t, func(t *testing.T, u *url.URL) { | ||||
| 		user1Session := loginUser(t, "user1") | ||||
| 		user2Session := loginUser(t, "user2") | ||||
| 		user12Session := loginUser(t, "user12") | ||||
| 		user13Session := loginUser(t, "user13") | ||||
|  | ||||
| 		// prepare branch and PRs in original repo | ||||
| 		repo10 := unittest.AssertExistsAndLoadBean(t, &repo_model.Repository{ID: 10}) | ||||
| 		prepareBranch(t, user12Session, repo10) | ||||
| 		prepareRepoPR(t, user12Session, user12Session, repo10, repo10) | ||||
|  | ||||
| 		// outdated new branch should not be displayed | ||||
| 		checkRecentlyPushedNewBranches(t, user12Session, "user12/repo10", []string{"new-commit"}) | ||||
|  | ||||
| 		// create a fork repo in public org | ||||
| 		testRepoFork(t, user12Session, repo10.OwnerName, repo10.Name, "org25", "org25_fork_repo10", "new-commit") | ||||
| 		orgPublicForkRepo := unittest.AssertExistsAndLoadBean(t, &repo_model.Repository{OwnerID: 25, Name: "org25_fork_repo10"}) | ||||
| 		prepareRepoPR(t, user12Session, user12Session, repo10, orgPublicForkRepo) | ||||
|  | ||||
| 		// user12 is the owner of the repo10 and the organization org25 | ||||
| 		// in repo10, user12 has opening/closed/merged pr and closed/merged pr with deleted branch | ||||
| 		checkRecentlyPushedNewBranches(t, user12Session, "user12/repo10", []string{"org25/org25_fork_repo10:new-commit", "new-commit"}) | ||||
|  | ||||
| 		userForkRepo := unittest.AssertExistsAndLoadBean(t, &repo_model.Repository{ID: 11}) | ||||
| 		testCtx := NewAPITestContext(t, repo10.OwnerName, repo10.Name, auth_model.AccessTokenScopeWriteRepository) | ||||
| 		t.Run("AddUser13AsCollaborator", doAPIAddCollaborator(testCtx, "user13", perm.AccessModeWrite)) | ||||
| 		prepareBranch(t, user13Session, userForkRepo) | ||||
| 		prepareRepoPR(t, user13Session, user13Session, repo10, userForkRepo) | ||||
|  | ||||
| 		// create branch with same name in different repo by user13 | ||||
| 		testCreateBranch(t, user13Session, repo10.OwnerName, repo10.Name, "branch/new-commit", "same-name-branch", http.StatusSeeOther) | ||||
| 		testCreateBranch(t, user13Session, userForkRepo.OwnerName, userForkRepo.Name, "branch/new-commit", "same-name-branch", http.StatusSeeOther) | ||||
| 		testCreatePullToDefaultBranch(t, user13Session, repo10, userForkRepo, "same-name-branch", "same name branch pr") | ||||
|  | ||||
| 		// user13 pushed 2 branches with the same name in repo10 and repo11 | ||||
| 		// and repo11's branch has a pr, but repo10's branch doesn't | ||||
| 		// in this case, we should get repo10's branch but not repo11's branch | ||||
| 		checkRecentlyPushedNewBranches(t, user13Session, "user12/repo10", []string{"same-name-branch", "user13/repo11:new-commit"}) | ||||
|  | ||||
| 		// create a fork repo in private org | ||||
| 		testRepoFork(t, user1Session, repo10.OwnerName, repo10.Name, "private_org35", "org35_fork_repo10", "new-commit") | ||||
| 		orgPrivateForkRepo := unittest.AssertExistsAndLoadBean(t, &repo_model.Repository{OwnerID: 35, Name: "org35_fork_repo10"}) | ||||
| 		prepareRepoPR(t, user1Session, user1Session, repo10, orgPrivateForkRepo) | ||||
|  | ||||
| 		// user1 is the owner of private_org35 and no write permission to repo10 | ||||
| 		// so user1 can only see the branch in org35_fork_repo10 | ||||
| 		checkRecentlyPushedNewBranches(t, user1Session, "user12/repo10", []string{"private_org35/org35_fork_repo10:new-commit"}) | ||||
|  | ||||
| 		// user2 push a branch in private_org35 | ||||
| 		testCreateBranch(t, user2Session, orgPrivateForkRepo.OwnerName, orgPrivateForkRepo.Name, "branch/new-commit", "user-read-permission", http.StatusSeeOther) | ||||
| 		// convert write permission to read permission for code unit | ||||
| 		token := getTokenForLoggedInUser(t, user1Session, auth_model.AccessTokenScopeWriteOrganization) | ||||
| 		req := NewRequestWithJSON(t, "PATCH", fmt.Sprintf("/api/v1/teams/%d", 24), &api.EditTeamOption{ | ||||
| 			Name:     "team24", | ||||
| 			UnitsMap: map[string]string{"repo.code": "read"}, | ||||
| 		}).AddTokenAuth(token) | ||||
| 		MakeRequest(t, req, http.StatusOK) | ||||
| 		teamUnit := unittest.AssertExistsAndLoadBean(t, &org_model.TeamUnit{TeamID: 24, Type: unit.TypeCode}) | ||||
| 		assert.Equal(t, perm.AccessModeRead, teamUnit.AccessMode) | ||||
| 		// user2 can see the branch as it is created by user2 | ||||
| 		checkRecentlyPushedNewBranches(t, user2Session, "user12/repo10", []string{"private_org35/org35_fork_repo10:user-read-permission"}) | ||||
| 	}) | ||||
| } | ||||
|   | ||||
		Reference in New Issue
	
	Block a user