mirror of
				https://github.com/go-gitea/gitea.git
				synced 2025-11-03 17:24:22 +00: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:
		@@ -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,17 +19,17 @@
 | 
				
			|||||||
							</span>
 | 
												</span>
 | 
				
			||||||
							<i class="dropdown icon"></i>
 | 
												<i class="dropdown icon"></i>
 | 
				
			||||||
							<div class="menu">
 | 
												<div class="menu">
 | 
				
			||||||
								<div class="item" data-value="{{.SignedUser.ID}}">
 | 
													{{if .CanForkToUser}}
 | 
				
			||||||
									<img class="ui mini image" src="{{.SignedUser.RelAvatarLink}}">
 | 
														<div class="item" data-value="{{.SignedUser.ID}}">
 | 
				
			||||||
									{{.SignedUser.ShortName 20}}
 | 
															<img class="ui mini image" src="{{.SignedUser.RelAvatarLink}}">
 | 
				
			||||||
								</div>
 | 
															{{.SignedUser.ShortName 20}}
 | 
				
			||||||
 | 
														</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>
 | 
				
			||||||
 
 | 
				
			|||||||
		Reference in New Issue
	
	Block a user