mirror of
				https://github.com/go-gitea/gitea.git
				synced 2025-10-26 17:14:23 +01:00 
			
		
		
		
	Fix so that user can still fork his own repository to owned organizations (#2699)
* Fix so that user can still fork his own repository to his organizations * Fix to only use owned organizations * Add integration test for forking own repository to owned organization
This commit is contained in:
		
							parent
							
								
									32ca299650
								
							
						
					
					
						commit
						1ec4dc6c1d
					
				| @ -46,7 +46,7 @@ func testPullCreate(t *testing.T, session *TestSession, user, repo, branch strin | |||||||
| func TestPullCreate(t *testing.T) { | func TestPullCreate(t *testing.T) { | ||||||
| 	prepareTestEnv(t) | 	prepareTestEnv(t) | ||||||
| 	session := loginUser(t, "user1") | 	session := loginUser(t, "user1") | ||||||
| 	testRepoFork(t, session) | 	testRepoFork(t, session, "user2", "repo1", "user1", "repo1") | ||||||
| 	testEditFile(t, session, "user1", "repo1", "master", "README.md") | 	testEditFile(t, session, "user1", "repo1", "master", "README.md") | ||||||
| 	testPullCreate(t, session, "user1", "repo1", "master") | 	testPullCreate(t, session, "user1", "repo1", "master") | ||||||
| } | } | ||||||
|  | |||||||
| @ -48,7 +48,7 @@ func testPullCleanUp(t *testing.T, session *TestSession, user, repo, pullnum str | |||||||
| func TestPullMerge(t *testing.T) { | func TestPullMerge(t *testing.T) { | ||||||
| 	prepareTestEnv(t) | 	prepareTestEnv(t) | ||||||
| 	session := loginUser(t, "user1") | 	session := loginUser(t, "user1") | ||||||
| 	testRepoFork(t, session) | 	testRepoFork(t, session, "user2", "repo1", "user1", "repo1") | ||||||
| 	testEditFile(t, session, "user1", "repo1", "master", "README.md") | 	testEditFile(t, session, "user1", "repo1", "master", "README.md") | ||||||
| 
 | 
 | ||||||
| 	resp := testPullCreate(t, session, "user1", "repo1", "master") | 	resp := testPullCreate(t, session, "user1", "repo1", "master") | ||||||
| @ -61,7 +61,7 @@ func TestPullMerge(t *testing.T) { | |||||||
| func TestPullCleanUpAfterMerge(t *testing.T) { | func TestPullCleanUpAfterMerge(t *testing.T) { | ||||||
| 	prepareTestEnv(t) | 	prepareTestEnv(t) | ||||||
| 	session := loginUser(t, "user1") | 	session := loginUser(t, "user1") | ||||||
| 	testRepoFork(t, session) | 	testRepoFork(t, session, "user2", "repo1", "user1", "repo1") | ||||||
| 	testEditFileToNewBranch(t, session, "user1", "repo1", "master", "feature/test", "README.md") | 	testEditFileToNewBranch(t, session, "user1", "repo1", "master", "feature/test", "README.md") | ||||||
| 
 | 
 | ||||||
| 	resp := testPullCreate(t, session, "user1", "repo1", "feature/test") | 	resp := testPullCreate(t, session, "user1", "repo1", "feature/test") | ||||||
|  | |||||||
| @ -5,19 +5,24 @@ | |||||||
| package integrations | package integrations | ||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
|  | 	"fmt" | ||||||
| 	"net/http" | 	"net/http" | ||||||
| 	"testing" | 	"testing" | ||||||
| 
 | 
 | ||||||
|  | 	"code.gitea.io/gitea/models" | ||||||
|  | 
 | ||||||
| 	"github.com/stretchr/testify/assert" | 	"github.com/stretchr/testify/assert" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| func testRepoFork(t *testing.T, session *TestSession) *TestResponse { | func testRepoFork(t *testing.T, session *TestSession, ownerName, repoName, forkOwnerName, forkRepoName string) *TestResponse { | ||||||
|  | 	forkOwner := models.AssertExistsAndLoadBean(t, &models.User{Name: forkOwnerName}).(*models.User) | ||||||
|  | 
 | ||||||
| 	// Step0: check the existence of the to-fork repo | 	// Step0: check the existence of the to-fork repo | ||||||
| 	req := NewRequest(t, "GET", "/user1/repo1") | 	req := NewRequestf(t, "GET", "/%s/%s", forkOwnerName, forkRepoName) | ||||||
| 	resp := session.MakeRequest(t, req, http.StatusNotFound) | 	resp := session.MakeRequest(t, req, http.StatusNotFound) | ||||||
| 
 | 
 | ||||||
| 	// Step1: go to the main page of repo | 	// Step1: go to the main page of repo | ||||||
| 	req = NewRequest(t, "GET", "/user2/repo1") | 	req = NewRequestf(t, "GET", "/%s/%s", ownerName, repoName) | ||||||
| 	resp = session.MakeRequest(t, req, http.StatusOK) | 	resp = session.MakeRequest(t, req, http.StatusOK) | ||||||
| 
 | 
 | ||||||
| 	// Step2: click the fork button | 	// Step2: click the fork button | ||||||
| @ -31,15 +36,17 @@ func testRepoFork(t *testing.T, session *TestSession) *TestResponse { | |||||||
| 	htmlDoc = NewHTMLParser(t, resp.Body) | 	htmlDoc = NewHTMLParser(t, resp.Body) | ||||||
| 	link, exists = htmlDoc.doc.Find("form.ui.form[action^=\"/repo/fork/\"]").Attr("action") | 	link, exists = htmlDoc.doc.Find("form.ui.form[action^=\"/repo/fork/\"]").Attr("action") | ||||||
| 	assert.True(t, exists, "The template has changed") | 	assert.True(t, exists, "The template has changed") | ||||||
|  | 	_, exists = htmlDoc.doc.Find(fmt.Sprintf(".owner.dropdown .item[data-value=\"%d\"]", forkOwner.ID)).Attr("data-value") | ||||||
|  | 	assert.True(t, exists, fmt.Sprintf("Fork owner '%s' is not present in select box", forkOwnerName)) | ||||||
| 	req = NewRequestWithValues(t, "POST", link, map[string]string{ | 	req = NewRequestWithValues(t, "POST", link, map[string]string{ | ||||||
| 		"_csrf":     htmlDoc.GetCSRF(), | 		"_csrf":     htmlDoc.GetCSRF(), | ||||||
| 		"uid":       "1", | 		"uid":       fmt.Sprintf("%d", forkOwner.ID), | ||||||
| 		"repo_name": "repo1", | 		"repo_name": forkRepoName, | ||||||
| 	}) | 	}) | ||||||
| 	resp = session.MakeRequest(t, req, http.StatusFound) | 	resp = session.MakeRequest(t, req, http.StatusFound) | ||||||
| 
 | 
 | ||||||
| 	// Step4: check the existence of the forked repo | 	// Step4: check the existence of the forked repo | ||||||
| 	req = NewRequest(t, "GET", "/user1/repo1") | 	req = NewRequestf(t, "GET", "/%s/%s", forkOwnerName, forkRepoName) | ||||||
| 	resp = session.MakeRequest(t, req, http.StatusOK) | 	resp = session.MakeRequest(t, req, http.StatusOK) | ||||||
| 
 | 
 | ||||||
| 	return resp | 	return resp | ||||||
| @ -48,5 +55,19 @@ func testRepoFork(t *testing.T, session *TestSession) *TestResponse { | |||||||
| func TestRepoFork(t *testing.T) { | func TestRepoFork(t *testing.T) { | ||||||
| 	prepareTestEnv(t) | 	prepareTestEnv(t) | ||||||
| 	session := loginUser(t, "user1") | 	session := loginUser(t, "user1") | ||||||
| 	testRepoFork(t, session) | 	testRepoFork(t, session, "user2", "repo1", "user1", "repo1") | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func TestRepoForkToOrg(t *testing.T) { | ||||||
|  | 	prepareTestEnv(t) | ||||||
|  | 	session := loginUser(t, "user2") | ||||||
|  | 	testRepoFork(t, session, "user2", "repo1", "user3", "repo1") | ||||||
|  | 
 | ||||||
|  | 	// Check that no more forking is allowed as user2 owns repository | ||||||
|  | 	//  and user3 organization that owner user2 is also now has forked this repository | ||||||
|  | 	req := NewRequest(t, "GET", "/user2/repo1") | ||||||
|  | 	resp := session.MakeRequest(t, req, http.StatusOK) | ||||||
|  | 	htmlDoc := NewHTMLParser(t, resp.Body) | ||||||
|  | 	_, exists := htmlDoc.doc.Find("a.ui.button[href^=\"/repo/fork/\"]").Attr("href") | ||||||
|  | 	assert.False(t, exists, "Forking should not be allowed anymore") | ||||||
| } | } | ||||||
|  | |||||||
| @ -651,6 +651,25 @@ func (repo *Repository) CanBeForked() bool { | |||||||
| 	return !repo.IsBare && repo.UnitEnabled(UnitTypeCode) | 	return !repo.IsBare && repo.UnitEnabled(UnitTypeCode) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // CanUserFork returns true if specified user can fork repository. | ||||||
|  | func (repo *Repository) CanUserFork(user *User) (bool, error) { | ||||||
|  | 	if user == nil { | ||||||
|  | 		return false, nil | ||||||
|  | 	} | ||||||
|  | 	if repo.OwnerID != user.ID && !user.HasForkedRepo(repo.ID) { | ||||||
|  | 		return true, nil | ||||||
|  | 	} | ||||||
|  | 	if err := user.GetOwnedOrganizations(); err != nil { | ||||||
|  | 		return false, err | ||||||
|  | 	} | ||||||
|  | 	for _, org := range user.OwnedOrgs { | ||||||
|  | 		if repo.OwnerID != org.ID && !org.HasForkedRepo(repo.ID) { | ||||||
|  | 			return true, nil | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return false, nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
| // CanEnablePulls returns true if repository meets the requirements of accepting pulls. | // CanEnablePulls returns true if repository meets the requirements of accepting pulls. | ||||||
| func (repo *Repository) CanEnablePulls() bool { | func (repo *Repository) CanEnablePulls() bool { | ||||||
| 	return !repo.IsMirror && !repo.IsBare | 	return !repo.IsMirror && !repo.IsBare | ||||||
|  | |||||||
| @ -337,6 +337,11 @@ func RepoAssignment() macaron.Handler { | |||||||
| 		ctx.Data["IsRepositoryAdmin"] = ctx.Repo.IsAdmin() | 		ctx.Data["IsRepositoryAdmin"] = ctx.Repo.IsAdmin() | ||||||
| 		ctx.Data["IsRepositoryWriter"] = ctx.Repo.IsWriter() | 		ctx.Data["IsRepositoryWriter"] = ctx.Repo.IsWriter() | ||||||
| 
 | 
 | ||||||
|  | 		if ctx.Data["CanSignedUserFork"], err = ctx.Repo.Repository.CanUserFork(ctx.User); err != nil { | ||||||
|  | 			ctx.Handle(500, "CanUserFork", err) | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
| 		ctx.Data["DisableSSH"] = setting.SSH.Disabled | 		ctx.Data["DisableSSH"] = setting.SSH.Disabled | ||||||
| 		ctx.Data["ExposeAnonSSH"] = setting.SSH.ExposeAnonymous | 		ctx.Data["ExposeAnonSSH"] = setting.SSH.ExposeAnonymous | ||||||
| 		ctx.Data["DisableHTTP"] = setting.Repository.DisableHTTPGit | 		ctx.Data["DisableHTTP"] = setting.Repository.DisableHTTPGit | ||||||
|  | |||||||
| @ -61,6 +61,8 @@ func getForkRepository(ctx *context.Context) *models.Repository { | |||||||
| 	ctx.Data["repo_name"] = forkRepo.Name | 	ctx.Data["repo_name"] = forkRepo.Name | ||||||
| 	ctx.Data["description"] = forkRepo.Description | 	ctx.Data["description"] = forkRepo.Description | ||||||
| 	ctx.Data["IsPrivate"] = forkRepo.IsPrivate | 	ctx.Data["IsPrivate"] = forkRepo.IsPrivate | ||||||
|  | 	canForkToUser := forkRepo.OwnerID != ctx.User.ID && !ctx.User.HasForkedRepo(forkRepo.ID) | ||||||
|  | 	ctx.Data["CanForkToUser"] = canForkToUser | ||||||
| 
 | 
 | ||||||
| 	if err = forkRepo.GetOwner(); err != nil { | 	if err = forkRepo.GetOwner(); err != nil { | ||||||
| 		ctx.Handle(500, "GetOwner", err) | 		ctx.Handle(500, "GetOwner", err) | ||||||
| @ -69,11 +71,23 @@ func getForkRepository(ctx *context.Context) *models.Repository { | |||||||
| 	ctx.Data["ForkFrom"] = forkRepo.Owner.Name + "/" + forkRepo.Name | 	ctx.Data["ForkFrom"] = forkRepo.Owner.Name + "/" + forkRepo.Name | ||||||
| 	ctx.Data["ForkFromOwnerID"] = forkRepo.Owner.ID | 	ctx.Data["ForkFromOwnerID"] = forkRepo.Owner.ID | ||||||
| 
 | 
 | ||||||
| 	if err := ctx.User.GetOrganizations(true); err != nil { | 	if err := ctx.User.GetOwnedOrganizations(); err != nil { | ||||||
| 		ctx.Handle(500, "GetOrganizations", err) | 		ctx.Handle(500, "GetOwnedOrganizations", err) | ||||||
| 		return nil | 		return nil | ||||||
| 	} | 	} | ||||||
| 	ctx.Data["Orgs"] = ctx.User.Orgs | 	var orgs []*models.User | ||||||
|  | 	for _, org := range ctx.User.OwnedOrgs { | ||||||
|  | 		if forkRepo.OwnerID != org.ID && !org.HasForkedRepo(forkRepo.ID) { | ||||||
|  | 			orgs = append(orgs, org) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	ctx.Data["Orgs"] = orgs | ||||||
|  | 
 | ||||||
|  | 	if canForkToUser { | ||||||
|  | 		ctx.Data["ContextUser"] = ctx.User | ||||||
|  | 	} else if len(orgs) > 0 { | ||||||
|  | 		ctx.Data["ContextUser"] = orgs[0] | ||||||
|  | 	} | ||||||
| 
 | 
 | ||||||
| 	return forkRepo | 	return forkRepo | ||||||
| } | } | ||||||
| @ -87,7 +101,6 @@ func Fork(ctx *context.Context) { | |||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	ctx.Data["ContextUser"] = ctx.User |  | ||||||
| 	ctx.HTML(200, tplFork) | 	ctx.HTML(200, tplFork) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -95,15 +108,16 @@ func Fork(ctx *context.Context) { | |||||||
| func ForkPost(ctx *context.Context, form auth.CreateRepoForm) { | func ForkPost(ctx *context.Context, form auth.CreateRepoForm) { | ||||||
| 	ctx.Data["Title"] = ctx.Tr("new_fork") | 	ctx.Data["Title"] = ctx.Tr("new_fork") | ||||||
| 
 | 
 | ||||||
|  | 	ctxUser := checkContextUser(ctx, form.UID) | ||||||
|  | 	if ctx.Written() { | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	forkRepo := getForkRepository(ctx) | 	forkRepo := getForkRepository(ctx) | ||||||
| 	if ctx.Written() { | 	if ctx.Written() { | ||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	ctxUser := checkContextUser(ctx, form.UID) |  | ||||||
| 	if ctx.Written() { |  | ||||||
| 		return |  | ||||||
| 	} |  | ||||||
| 	ctx.Data["ContextUser"] = ctxUser | 	ctx.Data["ContextUser"] = ctxUser | ||||||
| 
 | 
 | ||||||
| 	if ctx.HasError() { | 	if ctx.HasError() { | ||||||
|  | |||||||
| @ -32,7 +32,7 @@ | |||||||
| 						</div> | 						</div> | ||||||
| 						{{if .CanBeForked}} | 						{{if .CanBeForked}} | ||||||
| 							<div class="ui compact labeled button" tabindex="0"> | 							<div class="ui compact labeled button" tabindex="0"> | ||||||
| 								<a class="ui compact button {{if eq .OwnerID $.SignedUserID}}poping up{{end}}" {{if not (eq .OwnerID $.SignedUserID)}}href="{{AppSubUrl}}/repo/fork/{{.ID}}"{{else}} data-content="{{$.i18n.Tr "repo.fork_from_self"}}" data-position="top center" data-variation="tiny"{{end}}> | 								<a class="ui compact button {{if not $.CanSignedUserFork}}poping up{{end}}" {{if $.CanSignedUserFork}}href="{{AppSubUrl}}/repo/fork/{{.ID}}"{{else}} data-content="{{$.i18n.Tr "repo.fork_from_self"}}" data-position="top center" data-variation="tiny"{{end}}> | ||||||
| 									<i class="octicon octicon-repo-forked"></i>{{$.i18n.Tr "repo.fork"}} | 									<i class="octicon octicon-repo-forked"></i>{{$.i18n.Tr "repo.fork"}} | ||||||
| 								</a> | 								</a> | ||||||
| 								<a class="ui basic label" href="{{.Link}}/forks"> | 								<a class="ui basic label" href="{{.Link}}/forks"> | ||||||
|  | |||||||
| @ -19,18 +19,18 @@ | |||||||
| 							</span> | 							</span> | ||||||
| 							<i class="dropdown icon"></i> | 							<i class="dropdown icon"></i> | ||||||
| 							<div class="menu"> | 							<div class="menu"> | ||||||
|  | 								{{if .CanForkToUser}} | ||||||
| 									<div class="item" data-value="{{.SignedUser.ID}}"> | 									<div class="item" data-value="{{.SignedUser.ID}}"> | ||||||
| 										<img class="ui mini image" src="{{.SignedUser.RelAvatarLink}}"> | 										<img class="ui mini image" src="{{.SignedUser.RelAvatarLink}}"> | ||||||
| 										{{.SignedUser.ShortName 20}} | 										{{.SignedUser.ShortName 20}} | ||||||
| 									</div> | 									</div> | ||||||
|  | 								{{end}} | ||||||
| 								{{range .Orgs}} | 								{{range .Orgs}} | ||||||
| 									{{if and (.IsOwnedBy $.SignedUser.ID) (ne .ID $.ForkFromOwnerID)}} |  | ||||||
| 									<div class="item" data-value="{{.ID}}"> | 									<div class="item" data-value="{{.ID}}"> | ||||||
| 										<img class="ui mini image" src="{{.RelAvatarLink}}"> | 										<img class="ui mini image" src="{{.RelAvatarLink}}"> | ||||||
| 										{{.ShortName 20}} | 										{{.ShortName 20}} | ||||||
| 									</div> | 									</div> | ||||||
| 								{{end}} | 								{{end}} | ||||||
| 								{{end}} |  | ||||||
| 							</div> | 							</div> | ||||||
| 						</div> | 						</div> | ||||||
| 					</div> | 					</div> | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user