* 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 organizationtags/v1.21.12.1
| @@ -46,7 +46,7 @@ func testPullCreate(t *testing.T, session *TestSession, user, repo, branch strin | |||
| func TestPullCreate(t *testing.T) { | |||
| prepareTestEnv(t) | |||
| session := loginUser(t, "user1") | |||
| testRepoFork(t, session) | |||
| testRepoFork(t, session, "user2", "repo1", "user1", "repo1") | |||
| testEditFile(t, session, "user1", "repo1", "master", "README.md") | |||
| 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) { | |||
| prepareTestEnv(t) | |||
| session := loginUser(t, "user1") | |||
| testRepoFork(t, session) | |||
| testRepoFork(t, session, "user2", "repo1", "user1", "repo1") | |||
| testEditFile(t, session, "user1", "repo1", "master", "README.md") | |||
| resp := testPullCreate(t, session, "user1", "repo1", "master") | |||
| @@ -61,7 +61,7 @@ func TestPullMerge(t *testing.T) { | |||
| func TestPullCleanUpAfterMerge(t *testing.T) { | |||
| prepareTestEnv(t) | |||
| session := loginUser(t, "user1") | |||
| testRepoFork(t, session) | |||
| testRepoFork(t, session, "user2", "repo1", "user1", "repo1") | |||
| testEditFileToNewBranch(t, session, "user1", "repo1", "master", "feature/test", "README.md") | |||
| resp := testPullCreate(t, session, "user1", "repo1", "feature/test") | |||
| @@ -5,19 +5,24 @@ | |||
| package integrations | |||
| import ( | |||
| "fmt" | |||
| "net/http" | |||
| "testing" | |||
| "code.gitea.io/gitea/models" | |||
| "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 | |||
| req := NewRequest(t, "GET", "/user1/repo1") | |||
| req := NewRequestf(t, "GET", "/%s/%s", forkOwnerName, forkRepoName) | |||
| resp := session.MakeRequest(t, req, http.StatusNotFound) | |||
| // 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) | |||
| // Step2: click the fork button | |||
| @@ -31,15 +36,17 @@ func testRepoFork(t *testing.T, session *TestSession) *TestResponse { | |||
| htmlDoc = NewHTMLParser(t, resp.Body) | |||
| link, exists = htmlDoc.doc.Find("form.ui.form[action^=\"/repo/fork/\"]").Attr("action") | |||
| 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{ | |||
| "_csrf": htmlDoc.GetCSRF(), | |||
| "uid": "1", | |||
| "repo_name": "repo1", | |||
| "uid": fmt.Sprintf("%d", forkOwner.ID), | |||
| "repo_name": forkRepoName, | |||
| }) | |||
| resp = session.MakeRequest(t, req, http.StatusFound) | |||
| // 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) | |||
| return resp | |||
| @@ -48,5 +55,19 @@ func testRepoFork(t *testing.T, session *TestSession) *TestResponse { | |||
| func TestRepoFork(t *testing.T) { | |||
| prepareTestEnv(t) | |||
| 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) | |||
| } | |||
| // 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. | |||
| func (repo *Repository) CanEnablePulls() bool { | |||
| return !repo.IsMirror && !repo.IsBare | |||
| @@ -337,6 +337,11 @@ func RepoAssignment() macaron.Handler { | |||
| ctx.Data["IsRepositoryAdmin"] = ctx.Repo.IsAdmin() | |||
| 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["ExposeAnonSSH"] = setting.SSH.ExposeAnonymous | |||
| 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["description"] = forkRepo.Description | |||
| 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 { | |||
| 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["ForkFromOwnerID"] = forkRepo.Owner.ID | |||
| if err := ctx.User.GetOrganizations(true); err != nil { | |||
| ctx.Handle(500, "GetOrganizations", err) | |||
| if err := ctx.User.GetOwnedOrganizations(); err != nil { | |||
| ctx.Handle(500, "GetOwnedOrganizations", err) | |||
| 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 | |||
| } | |||
| @@ -87,7 +101,6 @@ func Fork(ctx *context.Context) { | |||
| return | |||
| } | |||
| ctx.Data["ContextUser"] = ctx.User | |||
| ctx.HTML(200, tplFork) | |||
| } | |||
| @@ -95,15 +108,16 @@ func Fork(ctx *context.Context) { | |||
| func ForkPost(ctx *context.Context, form auth.CreateRepoForm) { | |||
| ctx.Data["Title"] = ctx.Tr("new_fork") | |||
| forkRepo := getForkRepository(ctx) | |||
| ctxUser := checkContextUser(ctx, form.UID) | |||
| if ctx.Written() { | |||
| return | |||
| } | |||
| ctxUser := checkContextUser(ctx, form.UID) | |||
| forkRepo := getForkRepository(ctx) | |||
| if ctx.Written() { | |||
| return | |||
| } | |||
| ctx.Data["ContextUser"] = ctxUser | |||
| if ctx.HasError() { | |||
| @@ -32,7 +32,7 @@ | |||
| </div> | |||
| {{if .CanBeForked}} | |||
| <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"}} | |||
| </a> | |||
| <a class="ui basic label" href="{{.Link}}/forks"> | |||
| @@ -19,17 +19,17 @@ | |||
| </span> | |||
| <i class="dropdown icon"></i> | |||
| <div class="menu"> | |||
| <div class="item" data-value="{{.SignedUser.ID}}"> | |||
| <img class="ui mini image" src="{{.SignedUser.RelAvatarLink}}"> | |||
| {{.SignedUser.ShortName 20}} | |||
| </div> | |||
| {{if .CanForkToUser}} | |||
| <div class="item" data-value="{{.SignedUser.ID}}"> | |||
| <img class="ui mini image" src="{{.SignedUser.RelAvatarLink}}"> | |||
| {{.SignedUser.ShortName 20}} | |||
| </div> | |||
| {{end}} | |||
| {{range .Orgs}} | |||
| {{if and (.IsOwnedBy $.SignedUser.ID) (ne .ID $.ForkFromOwnerID)}} | |||
| <div class="item" data-value="{{.ID}}"> | |||
| <img class="ui mini image" src="{{.RelAvatarLink}}"> | |||
| {{.ShortName 20}} | |||
| </div> | |||
| {{end}} | |||
| <div class="item" data-value="{{.ID}}"> | |||
| <img class="ui mini image" src="{{.RelAvatarLink}}"> | |||
| {{.ShortName 20}} | |||
| </div> | |||
| {{end}} | |||
| </div> | |||
| </div> | |||