2018-01-05 19:56:52 +09:00
|
|
|
// Copyright 2018 The Gitea Authors.
|
|
|
|
// Copyright 2014 The Gogs Authors.
|
|
|
|
// All rights reserved.
|
2014-03-24 19:25:15 +09:00
|
|
|
// Use of this source code is governed by a MIT-style
|
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
|
|
|
package repo
|
|
|
|
|
|
|
|
import (
|
2021-06-07 23:52:59 +09:00
|
|
|
"errors"
|
2016-10-12 22:28:51 +09:00
|
|
|
"fmt"
|
2021-11-17 03:18:25 +09:00
|
|
|
"html"
|
2019-12-16 15:20:25 +09:00
|
|
|
"net/http"
|
2021-11-17 03:18:25 +09:00
|
|
|
"net/url"
|
|
|
|
"strconv"
|
2015-08-08 23:43:14 +09:00
|
|
|
"strings"
|
2020-01-17 15:03:40 +09:00
|
|
|
"time"
|
2015-08-08 23:43:14 +09:00
|
|
|
|
2016-11-11 01:24:48 +09:00
|
|
|
"code.gitea.io/gitea/models"
|
2022-08-25 11:31:57 +09:00
|
|
|
activities_model "code.gitea.io/gitea/models/activities"
|
2021-09-24 20:32:56 +09:00
|
|
|
"code.gitea.io/gitea/models/db"
|
2022-06-13 00:51:54 +09:00
|
|
|
git_model "code.gitea.io/gitea/models/git"
|
2022-06-13 18:37:59 +09:00
|
|
|
issues_model "code.gitea.io/gitea/models/issues"
|
2022-03-29 15:29:02 +09:00
|
|
|
"code.gitea.io/gitea/models/organization"
|
2022-05-11 19:09:36 +09:00
|
|
|
access_model "code.gitea.io/gitea/models/perm/access"
|
2022-06-11 23:44:20 +09:00
|
|
|
pull_model "code.gitea.io/gitea/models/pull"
|
2021-12-10 10:27:50 +09:00
|
|
|
repo_model "code.gitea.io/gitea/models/repo"
|
2021-11-10 04:57:58 +09:00
|
|
|
"code.gitea.io/gitea/models/unit"
|
2021-11-24 18:49:20 +09:00
|
|
|
user_model "code.gitea.io/gitea/models/user"
|
2016-11-11 01:24:48 +09:00
|
|
|
"code.gitea.io/gitea/modules/base"
|
|
|
|
"code.gitea.io/gitea/modules/context"
|
2019-03-27 18:33:00 +09:00
|
|
|
"code.gitea.io/gitea/modules/git"
|
2022-09-02 16:58:49 +09:00
|
|
|
issue_template "code.gitea.io/gitea/modules/issue/template"
|
2016-11-11 01:24:48 +09:00
|
|
|
"code.gitea.io/gitea/modules/log"
|
2017-01-29 00:59:58 +09:00
|
|
|
"code.gitea.io/gitea/modules/notification"
|
2016-11-11 01:24:48 +09:00
|
|
|
"code.gitea.io/gitea/modules/setting"
|
2020-06-07 09:45:12 +09:00
|
|
|
"code.gitea.io/gitea/modules/structs"
|
2020-10-05 14:49:33 +09:00
|
|
|
"code.gitea.io/gitea/modules/upload"
|
2019-01-21 20:45:32 +09:00
|
|
|
"code.gitea.io/gitea/modules/util"
|
2021-01-27 00:36:53 +09:00
|
|
|
"code.gitea.io/gitea/modules/web"
|
2021-01-30 17:55:53 +09:00
|
|
|
"code.gitea.io/gitea/modules/web/middleware"
|
2020-02-22 22:08:48 +09:00
|
|
|
"code.gitea.io/gitea/routers/utils"
|
2022-03-31 23:53:08 +09:00
|
|
|
asymkey_service "code.gitea.io/gitea/services/asymkey"
|
2022-06-11 23:44:20 +09:00
|
|
|
"code.gitea.io/gitea/services/automerge"
|
2021-04-07 04:44:05 +09:00
|
|
|
"code.gitea.io/gitea/services/forms"
|
2019-09-06 11:20:09 +09:00
|
|
|
"code.gitea.io/gitea/services/gitdiff"
|
2019-09-27 09:22:36 +09:00
|
|
|
pull_service "code.gitea.io/gitea/services/pull"
|
2019-10-26 15:54:11 +09:00
|
|
|
repo_service "code.gitea.io/gitea/services/repository"
|
2014-03-24 19:25:15 +09:00
|
|
|
)
|
|
|
|
|
2014-06-23 12:11:12 +09:00
|
|
|
const (
|
2016-11-24 16:04:31 +09:00
|
|
|
tplFork base.TplName = "repo/pulls/fork"
|
2019-06-08 05:29:29 +09:00
|
|
|
tplCompareDiff base.TplName = "repo/diff/compare"
|
2016-11-24 16:04:31 +09:00
|
|
|
tplPullCommits base.TplName = "repo/pulls/commits"
|
|
|
|
tplPullFiles base.TplName = "repo/pulls/files"
|
2016-02-18 07:21:31 +09:00
|
|
|
|
2016-11-24 16:04:31 +09:00
|
|
|
pullRequestTemplateKey = "PullRequestTemplate"
|
2016-02-18 07:21:31 +09:00
|
|
|
)
|
|
|
|
|
2022-01-21 02:46:10 +09:00
|
|
|
var pullRequestTemplateCandidates = []string{
|
|
|
|
"PULL_REQUEST_TEMPLATE.md",
|
2022-09-02 16:58:49 +09:00
|
|
|
"PULL_REQUEST_TEMPLATE.yaml",
|
|
|
|
"PULL_REQUEST_TEMPLATE.yml",
|
2022-01-21 02:46:10 +09:00
|
|
|
"pull_request_template.md",
|
2022-09-02 16:58:49 +09:00
|
|
|
"pull_request_template.yaml",
|
|
|
|
"pull_request_template.yml",
|
2022-01-21 02:46:10 +09:00
|
|
|
".gitea/PULL_REQUEST_TEMPLATE.md",
|
2022-09-02 16:58:49 +09:00
|
|
|
".gitea/PULL_REQUEST_TEMPLATE.yaml",
|
|
|
|
".gitea/PULL_REQUEST_TEMPLATE.yml",
|
2022-01-21 02:46:10 +09:00
|
|
|
".gitea/pull_request_template.md",
|
2022-09-02 16:58:49 +09:00
|
|
|
".gitea/pull_request_template.yaml",
|
|
|
|
".gitea/pull_request_template.yml",
|
2022-01-21 02:46:10 +09:00
|
|
|
".github/PULL_REQUEST_TEMPLATE.md",
|
2022-09-02 16:58:49 +09:00
|
|
|
".github/PULL_REQUEST_TEMPLATE.yaml",
|
|
|
|
".github/PULL_REQUEST_TEMPLATE.yml",
|
2022-01-21 02:46:10 +09:00
|
|
|
".github/pull_request_template.md",
|
2022-09-02 16:58:49 +09:00
|
|
|
".github/pull_request_template.yaml",
|
|
|
|
".github/pull_request_template.yml",
|
2022-01-21 02:46:10 +09:00
|
|
|
}
|
2014-06-23 12:11:12 +09:00
|
|
|
|
2021-12-10 10:27:50 +09:00
|
|
|
func getRepository(ctx *context.Context, repoID int64) *repo_model.Repository {
|
|
|
|
repo, err := repo_model.GetRepositoryByID(repoID)
|
2015-08-08 18:10:34 +09:00
|
|
|
if err != nil {
|
2021-12-10 10:27:50 +09:00
|
|
|
if repo_model.IsErrRepoNotExist(err) {
|
2018-01-11 06:34:17 +09:00
|
|
|
ctx.NotFound("GetRepositoryByID", nil)
|
2015-08-08 18:10:34 +09:00
|
|
|
} else {
|
2018-01-11 06:34:17 +09:00
|
|
|
ctx.ServerError("GetRepositoryByID", err)
|
2015-08-08 18:10:34 +09:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
2015-09-02 00:57:02 +09:00
|
|
|
|
2022-05-11 19:09:36 +09:00
|
|
|
perm, err := access_model.GetUserRepoPermission(ctx, repo, ctx.Doer)
|
2018-11-28 20:26:14 +09:00
|
|
|
if err != nil {
|
|
|
|
ctx.ServerError("GetUserRepoPermission", err)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2021-11-10 04:57:58 +09:00
|
|
|
if !perm.CanRead(unit.TypeCode) {
|
2019-11-12 00:15:29 +09:00
|
|
|
log.Trace("Permission Denied: User %-v cannot read %-v of repo %-v\n"+
|
|
|
|
"User in repo has Permissions: %-+v",
|
2022-03-22 16:03:22 +09:00
|
|
|
ctx.Doer,
|
2021-11-10 04:57:58 +09:00
|
|
|
unit.TypeCode,
|
2019-11-12 00:15:29 +09:00
|
|
|
ctx.Repo,
|
|
|
|
perm)
|
|
|
|
ctx.NotFound("getRepository", nil)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return repo
|
|
|
|
}
|
|
|
|
|
2021-12-10 10:27:50 +09:00
|
|
|
func getForkRepository(ctx *context.Context) *repo_model.Repository {
|
2019-11-12 00:15:29 +09:00
|
|
|
forkRepo := getRepository(ctx, ctx.ParamsInt64(":repoid"))
|
|
|
|
if ctx.Written() {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
if forkRepo.IsEmpty {
|
|
|
|
log.Trace("Empty repository %-v", forkRepo)
|
2018-01-11 06:34:17 +09:00
|
|
|
ctx.NotFound("getForkRepository", nil)
|
2015-09-02 00:57:02 +09:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-03-23 00:22:54 +09:00
|
|
|
if err := forkRepo.GetOwner(ctx); err != nil {
|
2018-01-11 06:34:17 +09:00
|
|
|
ctx.ServerError("GetOwner", err)
|
2015-08-08 18:10:34 +09:00
|
|
|
return nil
|
|
|
|
}
|
2020-06-07 09:45:12 +09:00
|
|
|
|
|
|
|
ctx.Data["repo_name"] = forkRepo.Name
|
|
|
|
ctx.Data["description"] = forkRepo.Description
|
|
|
|
ctx.Data["IsPrivate"] = forkRepo.IsPrivate || forkRepo.Owner.Visibility == structs.VisibleTypePrivate
|
2022-03-22 16:03:22 +09:00
|
|
|
canForkToUser := forkRepo.OwnerID != ctx.Doer.ID && !repo_model.HasForkedRepo(ctx.Doer.ID, forkRepo.ID)
|
2020-06-07 09:45:12 +09:00
|
|
|
|
2021-11-17 03:18:25 +09:00
|
|
|
ctx.Data["ForkRepo"] = forkRepo
|
2015-08-08 18:10:34 +09:00
|
|
|
|
2022-03-29 15:29:02 +09:00
|
|
|
ownedOrgs, err := organization.GetOrgsCanCreateRepoByUserID(ctx.Doer.ID)
|
2021-11-23 00:21:55 +09:00
|
|
|
if err != nil {
|
2021-11-25 14:03:03 +09:00
|
|
|
ctx.ServerError("GetOrgsCanCreateRepoByUserID", err)
|
2015-08-08 18:10:34 +09:00
|
|
|
return nil
|
|
|
|
}
|
2022-03-29 15:29:02 +09:00
|
|
|
var orgs []*organization.Organization
|
2021-11-23 00:21:55 +09:00
|
|
|
for _, org := range ownedOrgs {
|
2021-12-13 00:48:20 +09:00
|
|
|
if forkRepo.OwnerID != org.ID && !repo_model.HasForkedRepo(org.ID, forkRepo.ID) {
|
2017-10-16 00:06:07 +09:00
|
|
|
orgs = append(orgs, org)
|
|
|
|
}
|
|
|
|
}
|
2017-11-06 13:12:55 +09:00
|
|
|
|
2022-01-21 02:46:10 +09:00
|
|
|
traverseParentRepo := forkRepo
|
2017-11-06 13:12:55 +09:00
|
|
|
for {
|
2022-03-22 16:03:22 +09:00
|
|
|
if ctx.Doer.ID == traverseParentRepo.OwnerID {
|
2017-11-06 13:12:55 +09:00
|
|
|
canForkToUser = false
|
|
|
|
} else {
|
|
|
|
for i, org := range orgs {
|
|
|
|
if org.ID == traverseParentRepo.OwnerID {
|
|
|
|
orgs = append(orgs[:i], orgs[i+1:]...)
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if !traverseParentRepo.IsFork {
|
|
|
|
break
|
|
|
|
}
|
2021-12-10 10:27:50 +09:00
|
|
|
traverseParentRepo, err = repo_model.GetRepositoryByID(traverseParentRepo.ForkID)
|
2017-11-06 13:12:55 +09:00
|
|
|
if err != nil {
|
2018-01-11 06:34:17 +09:00
|
|
|
ctx.ServerError("GetRepositoryByID", err)
|
2017-11-06 13:12:55 +09:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx.Data["CanForkToUser"] = canForkToUser
|
2017-10-16 00:06:07 +09:00
|
|
|
ctx.Data["Orgs"] = orgs
|
|
|
|
|
|
|
|
if canForkToUser {
|
2022-03-22 16:03:22 +09:00
|
|
|
ctx.Data["ContextUser"] = ctx.Doer
|
2017-10-16 00:06:07 +09:00
|
|
|
} else if len(orgs) > 0 {
|
|
|
|
ctx.Data["ContextUser"] = orgs[0]
|
|
|
|
}
|
2015-08-08 18:10:34 +09:00
|
|
|
|
|
|
|
return forkRepo
|
|
|
|
}
|
|
|
|
|
2016-11-24 16:04:31 +09:00
|
|
|
// Fork render repository fork page
|
2016-03-12 01:56:52 +09:00
|
|
|
func Fork(ctx *context.Context) {
|
2015-08-08 18:10:34 +09:00
|
|
|
ctx.Data["Title"] = ctx.Tr("new_fork")
|
|
|
|
|
|
|
|
getForkRepository(ctx)
|
|
|
|
if ctx.Written() {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2021-04-06 00:30:52 +09:00
|
|
|
ctx.HTML(http.StatusOK, tplFork)
|
2015-08-08 18:10:34 +09:00
|
|
|
}
|
|
|
|
|
2016-11-24 16:04:31 +09:00
|
|
|
// ForkPost response for forking a repository
|
2021-01-27 00:36:53 +09:00
|
|
|
func ForkPost(ctx *context.Context) {
|
2021-04-07 04:44:05 +09:00
|
|
|
form := web.GetForm(ctx).(*forms.CreateRepoForm)
|
2015-08-08 18:10:34 +09:00
|
|
|
ctx.Data["Title"] = ctx.Tr("new_fork")
|
|
|
|
|
2017-10-16 00:06:07 +09:00
|
|
|
ctxUser := checkContextUser(ctx, form.UID)
|
2015-08-08 18:10:34 +09:00
|
|
|
if ctx.Written() {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-10-16 00:06:07 +09:00
|
|
|
forkRepo := getForkRepository(ctx)
|
2015-08-08 18:10:34 +09:00
|
|
|
if ctx.Written() {
|
|
|
|
return
|
|
|
|
}
|
2017-10-16 00:06:07 +09:00
|
|
|
|
2015-08-08 18:10:34 +09:00
|
|
|
ctx.Data["ContextUser"] = ctxUser
|
|
|
|
|
|
|
|
if ctx.HasError() {
|
2021-04-06 00:30:52 +09:00
|
|
|
ctx.HTML(http.StatusOK, tplFork)
|
2015-08-08 18:10:34 +09:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-11-06 13:12:55 +09:00
|
|
|
var err error
|
2022-01-21 02:46:10 +09:00
|
|
|
traverseParentRepo := forkRepo
|
2017-11-06 13:12:55 +09:00
|
|
|
for {
|
|
|
|
if ctxUser.ID == traverseParentRepo.OwnerID {
|
|
|
|
ctx.RenderWithErr(ctx.Tr("repo.settings.new_owner_has_same_repo"), tplFork, &form)
|
|
|
|
return
|
|
|
|
}
|
2021-12-13 00:48:20 +09:00
|
|
|
repo := repo_model.GetForkedRepo(ctxUser.ID, traverseParentRepo.ID)
|
2021-11-23 00:21:55 +09:00
|
|
|
if repo != nil {
|
2021-11-17 03:18:25 +09:00
|
|
|
ctx.Redirect(ctxUser.HomeLink() + "/" + url.PathEscape(repo.Name))
|
2017-11-06 13:12:55 +09:00
|
|
|
return
|
|
|
|
}
|
|
|
|
if !traverseParentRepo.IsFork {
|
|
|
|
break
|
|
|
|
}
|
2021-12-10 10:27:50 +09:00
|
|
|
traverseParentRepo, err = repo_model.GetRepositoryByID(traverseParentRepo.ForkID)
|
2017-11-06 13:12:55 +09:00
|
|
|
if err != nil {
|
2018-01-11 06:34:17 +09:00
|
|
|
ctx.ServerError("GetRepositoryByID", err)
|
2017-11-06 13:12:55 +09:00
|
|
|
return
|
|
|
|
}
|
2017-07-26 16:17:38 +09:00
|
|
|
}
|
|
|
|
|
2021-11-25 14:03:03 +09:00
|
|
|
// Check if user is allowed to create repo's on the organization.
|
2015-08-08 18:10:34 +09:00
|
|
|
if ctxUser.IsOrganization() {
|
2022-03-29 15:29:02 +09:00
|
|
|
isAllowedToFork, err := organization.OrgFromUser(ctxUser).CanCreateOrgRepo(ctx.Doer.ID)
|
2017-12-21 16:43:26 +09:00
|
|
|
if err != nil {
|
2021-11-25 14:03:03 +09:00
|
|
|
ctx.ServerError("CanCreateOrgRepo", err)
|
2017-12-21 16:43:26 +09:00
|
|
|
return
|
2021-11-25 14:03:03 +09:00
|
|
|
} else if !isAllowedToFork {
|
2021-04-06 00:30:52 +09:00
|
|
|
ctx.Error(http.StatusForbidden)
|
2015-08-08 18:10:34 +09:00
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-30 04:13:41 +09:00
|
|
|
repo, err := repo_service.ForkRepository(ctx, ctx.Doer, ctxUser, repo_service.ForkRepoOptions{
|
2021-08-28 17:37:14 +09:00
|
|
|
BaseRepo: forkRepo,
|
|
|
|
Name: form.RepoName,
|
|
|
|
Description: form.Description,
|
|
|
|
})
|
2015-08-08 18:10:34 +09:00
|
|
|
if err != nil {
|
2015-08-31 16:24:28 +09:00
|
|
|
ctx.Data["Err_RepoName"] = true
|
2015-08-08 18:10:34 +09:00
|
|
|
switch {
|
2021-12-13 00:48:20 +09:00
|
|
|
case repo_model.IsErrRepoAlreadyExist(err):
|
2016-11-24 16:04:31 +09:00
|
|
|
ctx.RenderWithErr(ctx.Tr("repo.settings.new_owner_has_same_repo"), tplFork, &form)
|
2021-11-24 18:49:20 +09:00
|
|
|
case db.IsErrNameReserved(err):
|
|
|
|
ctx.RenderWithErr(ctx.Tr("repo.form.name_reserved", err.(db.ErrNameReserved).Name), tplFork, &form)
|
|
|
|
case db.IsErrNamePatternNotAllowed(err):
|
|
|
|
ctx.RenderWithErr(ctx.Tr("repo.form.name_pattern_not_allowed", err.(db.ErrNamePatternNotAllowed).Pattern), tplFork, &form)
|
2015-08-08 18:10:34 +09:00
|
|
|
default:
|
2018-01-11 06:34:17 +09:00
|
|
|
ctx.ServerError("ForkPost", err)
|
2015-08-08 18:10:34 +09:00
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2015-08-08 23:43:14 +09:00
|
|
|
log.Trace("Repository forked[%d]: %s/%s", forkRepo.ID, ctxUser.Name, repo.Name)
|
2021-11-17 03:18:25 +09:00
|
|
|
ctx.Redirect(ctxUser.HomeLink() + "/" + url.PathEscape(repo.Name))
|
2015-08-08 18:10:34 +09:00
|
|
|
}
|
|
|
|
|
2022-06-13 18:37:59 +09:00
|
|
|
func checkPullInfo(ctx *context.Context) *issues_model.Issue {
|
|
|
|
issue, err := issues_model.GetIssueByIndex(ctx.Repo.Repository.ID, ctx.ParamsInt64(":index"))
|
2015-09-02 17:08:05 +09:00
|
|
|
if err != nil {
|
2022-06-13 18:37:59 +09:00
|
|
|
if issues_model.IsErrIssueNotExist(err) {
|
2018-01-11 06:34:17 +09:00
|
|
|
ctx.NotFound("GetIssueByIndex", err)
|
2015-09-02 17:08:05 +09:00
|
|
|
} else {
|
2018-01-11 06:34:17 +09:00
|
|
|
ctx.ServerError("GetIssueByIndex", err)
|
2015-09-02 17:08:05 +09:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
2018-12-14 00:55:43 +09:00
|
|
|
if err = issue.LoadPoster(); err != nil {
|
|
|
|
ctx.ServerError("LoadPoster", err)
|
|
|
|
return nil
|
|
|
|
}
|
2022-04-08 18:11:15 +09:00
|
|
|
if err := issue.LoadRepo(ctx); err != nil {
|
2019-10-24 02:54:13 +09:00
|
|
|
ctx.ServerError("LoadRepo", err)
|
|
|
|
return nil
|
|
|
|
}
|
2016-10-12 22:28:51 +09:00
|
|
|
ctx.Data["Title"] = fmt.Sprintf("#%d - %s", issue.Index, issue.Title)
|
2015-10-19 08:30:39 +09:00
|
|
|
ctx.Data["Issue"] = issue
|
2015-09-02 17:08:05 +09:00
|
|
|
|
2015-10-19 08:30:39 +09:00
|
|
|
if !issue.IsPull {
|
2018-01-11 06:34:17 +09:00
|
|
|
ctx.NotFound("ViewPullCommits", nil)
|
2015-09-02 17:08:05 +09:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2018-12-14 00:55:43 +09:00
|
|
|
if err = issue.LoadPullRequest(); err != nil {
|
|
|
|
ctx.ServerError("LoadPullRequest", err)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-04-28 20:48:48 +09:00
|
|
|
if err = issue.PullRequest.LoadHeadRepoCtx(ctx); err != nil {
|
2020-03-03 07:31:55 +09:00
|
|
|
ctx.ServerError("LoadHeadRepo", err)
|
2015-10-24 16:36:47 +09:00
|
|
|
return nil
|
2015-09-02 17:08:05 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
if ctx.IsSigned {
|
|
|
|
// Update issue-user.
|
2022-08-25 11:31:57 +09:00
|
|
|
if err = activities_model.SetIssueReadBy(ctx, issue.ID, ctx.Doer.ID); err != nil {
|
2018-01-11 06:34:17 +09:00
|
|
|
ctx.ServerError("ReadBy", err)
|
2015-09-02 17:08:05 +09:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-10-19 08:30:39 +09:00
|
|
|
return issue
|
2015-09-02 17:08:05 +09:00
|
|
|
}
|
|
|
|
|
2022-06-13 18:37:59 +09:00
|
|
|
func setMergeTarget(ctx *context.Context, pull *issues_model.PullRequest) {
|
2019-10-18 20:13:31 +09:00
|
|
|
if ctx.Repo.Owner.Name == pull.MustHeadUserName() {
|
2017-10-05 02:35:01 +09:00
|
|
|
ctx.Data["HeadTarget"] = pull.HeadBranch
|
|
|
|
} else if pull.HeadRepo == nil {
|
2019-10-18 20:13:31 +09:00
|
|
|
ctx.Data["HeadTarget"] = pull.MustHeadUserName() + ":" + pull.HeadBranch
|
2017-10-05 02:35:01 +09:00
|
|
|
} else {
|
2019-10-18 20:13:31 +09:00
|
|
|
ctx.Data["HeadTarget"] = pull.MustHeadUserName() + "/" + pull.HeadRepo.Name + ":" + pull.HeadBranch
|
2017-10-05 02:35:01 +09:00
|
|
|
}
|
|
|
|
ctx.Data["BaseTarget"] = pull.BaseBranch
|
2021-02-18 11:45:49 +09:00
|
|
|
ctx.Data["HeadBranchHTMLURL"] = pull.GetHeadBranchHTMLURL()
|
|
|
|
ctx.Data["BaseBranchHTMLURL"] = pull.GetBaseBranchHTMLURL()
|
2017-10-05 02:35:01 +09:00
|
|
|
}
|
|
|
|
|
2016-11-24 16:04:31 +09:00
|
|
|
// PrepareMergedViewPullInfo show meta information for a merged pull request view page
|
2022-06-13 18:37:59 +09:00
|
|
|
func PrepareMergedViewPullInfo(ctx *context.Context, issue *issues_model.Issue) *git.CompareInfo {
|
2016-08-17 02:19:09 +09:00
|
|
|
pull := issue.PullRequest
|
2015-09-02 22:26:56 +09:00
|
|
|
|
2017-10-05 02:35:01 +09:00
|
|
|
setMergeTarget(ctx, pull)
|
|
|
|
ctx.Data["HasMerged"] = true
|
|
|
|
|
2021-12-23 17:32:29 +09:00
|
|
|
var baseCommit string
|
|
|
|
// Some migrated PR won't have any Base SHA and lose history, try to get one
|
|
|
|
if pull.MergeBase == "" {
|
|
|
|
var commitSHA, parentCommit string
|
|
|
|
// If there is a head or a patch file, and it is readable, grab info
|
2021-12-23 22:44:00 +09:00
|
|
|
commitSHA, err := ctx.Repo.GitRepo.GetRefCommitID(pull.GetGitRefName())
|
2021-12-23 17:32:29 +09:00
|
|
|
if err != nil {
|
|
|
|
// Head File does not exist, try the patch
|
|
|
|
commitSHA, err = ctx.Repo.GitRepo.ReadPatchCommit(pull.Index)
|
|
|
|
if err == nil {
|
|
|
|
// Recreate pull head in files for next time
|
2021-12-23 22:44:00 +09:00
|
|
|
if err := ctx.Repo.GitRepo.SetReference(pull.GetGitRefName(), commitSHA); err != nil {
|
2021-12-23 17:32:29 +09:00
|
|
|
log.Error("Could not write head file", err)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// There is no history available
|
|
|
|
log.Trace("No history file available for PR %d", pull.Index)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if commitSHA != "" {
|
|
|
|
// Get immediate parent of the first commit in the patch, grab history back
|
2022-10-23 23:44:45 +09:00
|
|
|
parentCommit, _, err = git.NewCommand(ctx, "rev-list", "-1", "--skip=1").AddDynamicArguments(commitSHA).RunStdString(&git.RunOpts{Dir: ctx.Repo.GitRepo.Path})
|
2021-12-23 17:32:29 +09:00
|
|
|
if err == nil {
|
|
|
|
parentCommit = strings.TrimSpace(parentCommit)
|
|
|
|
}
|
|
|
|
// Special case on Git < 2.25 that doesn't fail on immediate empty history
|
|
|
|
if err != nil || parentCommit == "" {
|
|
|
|
log.Info("No known parent commit for PR %d, error: %v", pull.Index, err)
|
|
|
|
// bring at least partial history if it can work
|
|
|
|
parentCommit = commitSHA
|
|
|
|
}
|
|
|
|
}
|
|
|
|
baseCommit = parentCommit
|
|
|
|
} else {
|
|
|
|
// Keep an empty history or original commit
|
|
|
|
baseCommit = pull.MergeBase
|
|
|
|
}
|
|
|
|
|
2019-06-12 08:32:08 +09:00
|
|
|
compareInfo, err := ctx.Repo.GitRepo.GetCompareInfo(ctx.Repo.Repository.RepoPath(),
|
2022-01-18 16:45:43 +09:00
|
|
|
baseCommit, pull.GetGitRefName(), false, false)
|
2015-09-02 22:26:56 +09:00
|
|
|
if err != nil {
|
2020-10-20 21:52:54 +09:00
|
|
|
if strings.Contains(err.Error(), "fatal: Not a valid object name") || strings.Contains(err.Error(), "unknown revision or path not in the working tree") {
|
2018-08-01 12:00:35 +09:00
|
|
|
ctx.Data["IsPullRequestBroken"] = true
|
2020-03-03 07:31:55 +09:00
|
|
|
ctx.Data["BaseTarget"] = pull.BaseBranch
|
2018-01-19 15:18:51 +09:00
|
|
|
ctx.Data["NumCommits"] = 0
|
|
|
|
ctx.Data["NumFiles"] = 0
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-06-08 05:29:29 +09:00
|
|
|
ctx.ServerError("GetCompareInfo", err)
|
2018-01-19 15:18:51 +09:00
|
|
|
return nil
|
2015-09-02 22:26:56 +09:00
|
|
|
}
|
2021-08-10 03:08:51 +09:00
|
|
|
ctx.Data["NumCommits"] = len(compareInfo.Commits)
|
2019-06-12 08:32:08 +09:00
|
|
|
ctx.Data["NumFiles"] = compareInfo.NumFiles
|
2020-12-18 21:37:55 +09:00
|
|
|
|
2021-08-10 03:08:51 +09:00
|
|
|
if len(compareInfo.Commits) != 0 {
|
|
|
|
sha := compareInfo.Commits[0].ID.String()
|
2022-06-13 00:51:54 +09:00
|
|
|
commitStatuses, _, err := git_model.GetLatestCommitStatus(ctx, ctx.Repo.Repository.ID, sha, db.ListOptions{})
|
2020-12-18 21:37:55 +09:00
|
|
|
if err != nil {
|
|
|
|
ctx.ServerError("GetLatestCommitStatus", err)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
if len(commitStatuses) != 0 {
|
|
|
|
ctx.Data["LatestCommitStatuses"] = commitStatuses
|
2022-06-13 00:51:54 +09:00
|
|
|
ctx.Data["LatestCommitStatus"] = git_model.CalcCommitStatus(commitStatuses)
|
2020-12-18 21:37:55 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-12 08:32:08 +09:00
|
|
|
return compareInfo
|
2015-09-02 22:26:56 +09:00
|
|
|
}
|
|
|
|
|
2016-11-24 16:04:31 +09:00
|
|
|
// PrepareViewPullInfo show meta information for a pull request preview page
|
2022-06-13 18:37:59 +09:00
|
|
|
func PrepareViewPullInfo(ctx *context.Context, issue *issues_model.Issue) *git.CompareInfo {
|
2022-02-11 17:02:53 +09:00
|
|
|
ctx.Data["PullRequestWorkInProgressPrefixes"] = setting.Repository.PullRequest.WorkInProgressPrefixes
|
|
|
|
|
2015-09-02 17:08:05 +09:00
|
|
|
repo := ctx.Repo.Repository
|
2016-08-17 02:19:09 +09:00
|
|
|
pull := issue.PullRequest
|
2015-09-02 17:08:05 +09:00
|
|
|
|
2022-04-28 20:48:48 +09:00
|
|
|
if err := pull.LoadHeadRepoCtx(ctx); err != nil {
|
2020-03-03 07:31:55 +09:00
|
|
|
ctx.ServerError("LoadHeadRepo", err)
|
2015-10-24 16:36:47 +09:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-04-28 20:48:48 +09:00
|
|
|
if err := pull.LoadBaseRepoCtx(ctx); err != nil {
|
2020-03-03 07:31:55 +09:00
|
|
|
ctx.ServerError("LoadBaseRepo", err)
|
2020-01-08 02:06:14 +09:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2017-10-05 02:35:01 +09:00
|
|
|
setMergeTarget(ctx, pull)
|
|
|
|
|
2020-01-08 02:06:14 +09:00
|
|
|
if err := pull.LoadProtectedBranch(); err != nil {
|
2020-01-17 15:03:40 +09:00
|
|
|
ctx.ServerError("LoadProtectedBranch", err)
|
2019-09-18 14:39:45 +09:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
ctx.Data["EnableStatusCheck"] = pull.ProtectedBranch != nil && pull.ProtectedBranch.EnableStatusCheck
|
|
|
|
|
2022-01-20 08:26:57 +09:00
|
|
|
var baseGitRepo *git.Repository
|
|
|
|
if pull.BaseRepoID == ctx.Repo.Repository.ID && ctx.Repo.GitRepo != nil {
|
|
|
|
baseGitRepo = ctx.Repo.GitRepo
|
|
|
|
} else {
|
2022-03-30 04:13:41 +09:00
|
|
|
baseGitRepo, err := git.OpenRepository(ctx, pull.BaseRepo.RepoPath())
|
2022-01-20 08:26:57 +09:00
|
|
|
if err != nil {
|
|
|
|
ctx.ServerError("OpenRepository", err)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
defer baseGitRepo.Close()
|
2020-01-08 02:06:14 +09:00
|
|
|
}
|
2020-03-06 03:51:21 +09:00
|
|
|
|
|
|
|
if !baseGitRepo.IsBranchExist(pull.BaseBranch) {
|
|
|
|
ctx.Data["IsPullRequestBroken"] = true
|
|
|
|
ctx.Data["BaseTarget"] = pull.BaseBranch
|
|
|
|
ctx.Data["HeadTarget"] = pull.HeadBranch
|
2020-03-31 22:42:44 +09:00
|
|
|
|
|
|
|
sha, err := baseGitRepo.GetRefCommitID(pull.GetGitRefName())
|
|
|
|
if err != nil {
|
|
|
|
ctx.ServerError(fmt.Sprintf("GetRefCommitID(%s)", pull.GetGitRefName()), err)
|
|
|
|
return nil
|
|
|
|
}
|
2022-06-13 00:51:54 +09:00
|
|
|
commitStatuses, _, err := git_model.GetLatestCommitStatus(ctx, repo.ID, sha, db.ListOptions{})
|
2020-03-31 22:42:44 +09:00
|
|
|
if err != nil {
|
|
|
|
ctx.ServerError("GetLatestCommitStatus", err)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
if len(commitStatuses) > 0 {
|
|
|
|
ctx.Data["LatestCommitStatuses"] = commitStatuses
|
2022-06-13 00:51:54 +09:00
|
|
|
ctx.Data["LatestCommitStatus"] = git_model.CalcCommitStatus(commitStatuses)
|
2020-03-31 22:42:44 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
compareInfo, err := baseGitRepo.GetCompareInfo(pull.BaseRepo.RepoPath(),
|
2022-01-18 16:45:43 +09:00
|
|
|
pull.MergeBase, pull.GetGitRefName(), false, false)
|
2020-03-31 22:42:44 +09:00
|
|
|
if err != nil {
|
|
|
|
if strings.Contains(err.Error(), "fatal: Not a valid object name") {
|
|
|
|
ctx.Data["IsPullRequestBroken"] = true
|
|
|
|
ctx.Data["BaseTarget"] = pull.BaseBranch
|
|
|
|
ctx.Data["NumCommits"] = 0
|
|
|
|
ctx.Data["NumFiles"] = 0
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx.ServerError("GetCompareInfo", err)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2021-08-10 03:08:51 +09:00
|
|
|
ctx.Data["NumCommits"] = len(compareInfo.Commits)
|
2020-03-31 22:42:44 +09:00
|
|
|
ctx.Data["NumFiles"] = compareInfo.NumFiles
|
|
|
|
return compareInfo
|
2020-03-06 03:51:21 +09:00
|
|
|
}
|
|
|
|
|
2019-06-30 16:57:59 +09:00
|
|
|
var headBranchExist bool
|
2020-01-08 02:06:14 +09:00
|
|
|
var headBranchSha string
|
2019-06-30 16:57:59 +09:00
|
|
|
// HeadRepo may be missing
|
2015-10-05 09:54:06 +09:00
|
|
|
if pull.HeadRepo != nil {
|
2022-03-30 04:13:41 +09:00
|
|
|
headGitRepo, err := git.OpenRepository(ctx, pull.HeadRepo.RepoPath())
|
2015-10-05 09:54:06 +09:00
|
|
|
if err != nil {
|
2018-01-11 06:34:17 +09:00
|
|
|
ctx.ServerError("OpenRepository", err)
|
2015-10-05 09:54:06 +09:00
|
|
|
return nil
|
|
|
|
}
|
2019-11-13 16:01:19 +09:00
|
|
|
defer headGitRepo.Close()
|
2019-06-30 16:57:59 +09:00
|
|
|
|
2022-06-13 18:37:59 +09:00
|
|
|
if pull.Flow == issues_model.PullRequestFlowGithub {
|
2021-07-28 18:42:56 +09:00
|
|
|
headBranchExist = headGitRepo.IsBranchExist(pull.HeadBranch)
|
|
|
|
} else {
|
2021-12-01 05:06:32 +09:00
|
|
|
headBranchExist = git.IsReferenceExist(ctx, baseGitRepo.Path, pull.GetGitRefName())
|
2021-07-28 18:42:56 +09:00
|
|
|
}
|
2019-06-30 16:57:59 +09:00
|
|
|
|
|
|
|
if headBranchExist {
|
2022-06-13 18:37:59 +09:00
|
|
|
if pull.Flow != issues_model.PullRequestFlowGithub {
|
2021-07-28 18:42:56 +09:00
|
|
|
headBranchSha, err = baseGitRepo.GetRefCommitID(pull.GetGitRefName())
|
|
|
|
} else {
|
|
|
|
headBranchSha, err = headGitRepo.GetBranchCommitID(pull.HeadBranch)
|
|
|
|
}
|
2019-06-30 16:57:59 +09:00
|
|
|
if err != nil {
|
|
|
|
ctx.ServerError("GetBranchCommitID", err)
|
|
|
|
return nil
|
|
|
|
}
|
2020-01-08 02:06:14 +09:00
|
|
|
}
|
|
|
|
}
|
2019-06-30 16:57:59 +09:00
|
|
|
|
2020-01-25 11:48:22 +09:00
|
|
|
if headBranchExist {
|
2022-01-20 08:26:57 +09:00
|
|
|
var err error
|
2022-04-28 20:48:48 +09:00
|
|
|
ctx.Data["UpdateAllowed"], ctx.Data["UpdateByRebaseAllowed"], err = pull_service.IsUserAllowedToUpdate(ctx, pull, ctx.Doer)
|
2020-01-25 11:48:22 +09:00
|
|
|
if err != nil {
|
|
|
|
ctx.ServerError("IsUserAllowedToUpdate", err)
|
|
|
|
return nil
|
|
|
|
}
|
2022-01-20 08:26:57 +09:00
|
|
|
ctx.Data["GetCommitMessages"] = pull_service.GetSquashMergeCommitMessages(ctx, pull)
|
2022-08-03 13:56:59 +09:00
|
|
|
} else {
|
|
|
|
ctx.Data["GetCommitMessages"] = ""
|
2020-01-25 11:48:22 +09:00
|
|
|
}
|
|
|
|
|
2020-01-08 02:06:14 +09:00
|
|
|
sha, err := baseGitRepo.GetRefCommitID(pull.GetGitRefName())
|
|
|
|
if err != nil {
|
2020-06-09 03:07:41 +09:00
|
|
|
if git.IsErrNotExist(err) {
|
|
|
|
ctx.Data["IsPullRequestBroken"] = true
|
|
|
|
if pull.IsSameRepo() {
|
|
|
|
ctx.Data["HeadTarget"] = pull.HeadBranch
|
|
|
|
} else if pull.HeadRepo == nil {
|
|
|
|
ctx.Data["HeadTarget"] = "<deleted>:" + pull.HeadBranch
|
|
|
|
} else {
|
|
|
|
ctx.Data["HeadTarget"] = pull.HeadRepo.OwnerName + ":" + pull.HeadBranch
|
|
|
|
}
|
|
|
|
ctx.Data["BaseTarget"] = pull.BaseBranch
|
|
|
|
ctx.Data["NumCommits"] = 0
|
|
|
|
ctx.Data["NumFiles"] = 0
|
|
|
|
return nil
|
|
|
|
}
|
2020-01-08 02:06:14 +09:00
|
|
|
ctx.ServerError(fmt.Sprintf("GetRefCommitID(%s)", pull.GetGitRefName()), err)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-06-13 00:51:54 +09:00
|
|
|
commitStatuses, _, err := git_model.GetLatestCommitStatus(ctx, repo.ID, sha, db.ListOptions{})
|
2020-01-08 02:06:14 +09:00
|
|
|
if err != nil {
|
|
|
|
ctx.ServerError("GetLatestCommitStatus", err)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
if len(commitStatuses) > 0 {
|
|
|
|
ctx.Data["LatestCommitStatuses"] = commitStatuses
|
2022-06-13 00:51:54 +09:00
|
|
|
ctx.Data["LatestCommitStatus"] = git_model.CalcCommitStatus(commitStatuses)
|
2020-01-08 02:06:14 +09:00
|
|
|
}
|
2019-09-18 14:39:45 +09:00
|
|
|
|
2020-01-08 02:06:14 +09:00
|
|
|
if pull.ProtectedBranch != nil && pull.ProtectedBranch.EnableStatusCheck {
|
|
|
|
ctx.Data["is_context_required"] = func(context string) bool {
|
|
|
|
for _, c := range pull.ProtectedBranch.StatusCheckContexts {
|
|
|
|
if c == context {
|
|
|
|
return true
|
2019-09-18 14:39:45 +09:00
|
|
|
}
|
|
|
|
}
|
2020-01-08 02:06:14 +09:00
|
|
|
return false
|
2019-06-30 16:57:59 +09:00
|
|
|
}
|
2020-04-16 05:35:18 +09:00
|
|
|
ctx.Data["RequiredStatusCheckState"] = pull_service.MergeRequiredContextsCommitStatus(commitStatuses, pull.ProtectedBranch.StatusCheckContexts)
|
2015-09-02 17:08:05 +09:00
|
|
|
}
|
|
|
|
|
2020-01-08 02:06:14 +09:00
|
|
|
ctx.Data["HeadBranchMovedOn"] = headBranchSha != sha
|
|
|
|
ctx.Data["HeadBranchCommitID"] = headBranchSha
|
|
|
|
ctx.Data["PullHeadCommitID"] = sha
|
|
|
|
|
|
|
|
if pull.HeadRepo == nil || !headBranchExist || headBranchSha != sha {
|
2018-08-01 12:00:35 +09:00
|
|
|
ctx.Data["IsPullRequestBroken"] = true
|
2020-03-03 07:31:55 +09:00
|
|
|
if pull.IsSameRepo() {
|
|
|
|
ctx.Data["HeadTarget"] = pull.HeadBranch
|
2020-06-09 03:07:41 +09:00
|
|
|
} else if pull.HeadRepo == nil {
|
|
|
|
ctx.Data["HeadTarget"] = "<deleted>:" + pull.HeadBranch
|
2020-03-03 07:31:55 +09:00
|
|
|
} else {
|
2020-06-09 03:07:41 +09:00
|
|
|
ctx.Data["HeadTarget"] = pull.HeadRepo.OwnerName + ":" + pull.HeadBranch
|
2020-03-03 07:31:55 +09:00
|
|
|
}
|
2015-09-02 22:26:56 +09:00
|
|
|
}
|
|
|
|
|
2020-01-08 02:06:14 +09:00
|
|
|
compareInfo, err := baseGitRepo.GetCompareInfo(pull.BaseRepo.RepoPath(),
|
2022-01-18 16:45:43 +09:00
|
|
|
git.BranchPrefix+pull.BaseBranch, pull.GetGitRefName(), false, false)
|
2015-09-02 17:08:05 +09:00
|
|
|
if err != nil {
|
2016-07-23 19:35:16 +09:00
|
|
|
if strings.Contains(err.Error(), "fatal: Not a valid object name") {
|
2018-08-01 12:00:35 +09:00
|
|
|
ctx.Data["IsPullRequestBroken"] = true
|
2020-03-03 07:31:55 +09:00
|
|
|
ctx.Data["BaseTarget"] = pull.BaseBranch
|
2016-07-23 19:35:16 +09:00
|
|
|
ctx.Data["NumCommits"] = 0
|
|
|
|
ctx.Data["NumFiles"] = 0
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-06-08 05:29:29 +09:00
|
|
|
ctx.ServerError("GetCompareInfo", err)
|
2015-09-02 17:08:05 +09:00
|
|
|
return nil
|
|
|
|
}
|
2018-08-14 04:04:39 +09:00
|
|
|
|
2021-07-29 11:32:48 +09:00
|
|
|
if compareInfo.HeadCommitID == compareInfo.MergeBase {
|
|
|
|
ctx.Data["IsNothingToCompare"] = true
|
|
|
|
}
|
|
|
|
|
2018-08-14 04:04:39 +09:00
|
|
|
if pull.IsWorkInProgress() {
|
|
|
|
ctx.Data["IsPullWorkInProgress"] = true
|
|
|
|
ctx.Data["WorkInProgressPrefix"] = pull.GetWorkInProgressPrefix()
|
|
|
|
}
|
|
|
|
|
2019-02-05 20:54:49 +09:00
|
|
|
if pull.IsFilesConflicted() {
|
|
|
|
ctx.Data["IsPullFilesConflicted"] = true
|
|
|
|
ctx.Data["ConflictedFiles"] = pull.ConflictedFiles
|
|
|
|
}
|
|
|
|
|
2021-08-10 03:08:51 +09:00
|
|
|
ctx.Data["NumCommits"] = len(compareInfo.Commits)
|
2019-06-12 08:32:08 +09:00
|
|
|
ctx.Data["NumFiles"] = compareInfo.NumFiles
|
|
|
|
return compareInfo
|
2015-09-02 17:08:05 +09:00
|
|
|
}
|
|
|
|
|
2016-11-24 16:04:31 +09:00
|
|
|
// ViewPullCommits show commits for a pull request
|
2016-03-12 01:56:52 +09:00
|
|
|
func ViewPullCommits(ctx *context.Context) {
|
2016-08-14 19:32:24 +09:00
|
|
|
ctx.Data["PageIsPullList"] = true
|
2015-09-02 17:08:05 +09:00
|
|
|
ctx.Data["PageIsPullCommits"] = true
|
|
|
|
|
2016-08-17 02:19:09 +09:00
|
|
|
issue := checkPullInfo(ctx)
|
2015-09-02 17:08:05 +09:00
|
|
|
if ctx.Written() {
|
|
|
|
return
|
|
|
|
}
|
2016-08-17 02:19:09 +09:00
|
|
|
pull := issue.PullRequest
|
2015-09-02 17:08:05 +09:00
|
|
|
|
2019-12-19 03:37:44 +09:00
|
|
|
var prInfo *git.CompareInfo
|
2015-09-02 22:26:56 +09:00
|
|
|
if pull.HasMerged {
|
2019-12-19 03:37:44 +09:00
|
|
|
prInfo = PrepareMergedViewPullInfo(ctx, issue)
|
2015-09-02 22:26:56 +09:00
|
|
|
} else {
|
2019-12-19 03:37:44 +09:00
|
|
|
prInfo = PrepareViewPullInfo(ctx, issue)
|
2015-09-02 17:08:05 +09:00
|
|
|
}
|
|
|
|
|
2019-12-19 03:37:44 +09:00
|
|
|
if ctx.Written() {
|
|
|
|
return
|
|
|
|
} else if prInfo == nil {
|
|
|
|
ctx.NotFound("ViewPullCommits", nil)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx.Data["Username"] = ctx.Repo.Owner.Name
|
|
|
|
ctx.Data["Reponame"] = ctx.Repo.Repository.Name
|
2021-08-10 03:08:51 +09:00
|
|
|
|
2022-06-13 00:51:54 +09:00
|
|
|
commits := git_model.ConvertFromGitCommit(prInfo.Commits, ctx.Repo.Repository)
|
2015-09-02 22:26:56 +09:00
|
|
|
ctx.Data["Commits"] = commits
|
2021-08-10 03:08:51 +09:00
|
|
|
ctx.Data["CommitCount"] = len(commits)
|
2015-09-02 22:26:56 +09:00
|
|
|
|
2019-12-16 15:20:25 +09:00
|
|
|
getBranchData(ctx, issue)
|
2021-04-06 00:30:52 +09:00
|
|
|
ctx.HTML(http.StatusOK, tplPullCommits)
|
2015-09-02 17:08:05 +09:00
|
|
|
}
|
|
|
|
|
2016-11-24 16:04:31 +09:00
|
|
|
// ViewPullFiles render pull request changed files list page
|
2016-03-12 01:56:52 +09:00
|
|
|
func ViewPullFiles(ctx *context.Context) {
|
2016-08-14 19:32:24 +09:00
|
|
|
ctx.Data["PageIsPullList"] = true
|
2015-09-02 17:08:05 +09:00
|
|
|
ctx.Data["PageIsPullFiles"] = true
|
|
|
|
|
2016-08-17 02:19:09 +09:00
|
|
|
issue := checkPullInfo(ctx)
|
2015-09-02 17:08:05 +09:00
|
|
|
if ctx.Written() {
|
|
|
|
return
|
|
|
|
}
|
2016-08-17 02:19:09 +09:00
|
|
|
pull := issue.PullRequest
|
2015-09-02 17:08:05 +09:00
|
|
|
|
2015-09-02 22:26:56 +09:00
|
|
|
var (
|
|
|
|
startCommitID string
|
|
|
|
endCommitID string
|
2021-08-31 13:16:23 +09:00
|
|
|
gitRepo = ctx.Repo.GitRepo
|
2015-09-02 22:26:56 +09:00
|
|
|
)
|
|
|
|
|
2019-12-19 03:37:44 +09:00
|
|
|
var prInfo *git.CompareInfo
|
2015-09-02 22:26:56 +09:00
|
|
|
if pull.HasMerged {
|
2019-12-19 03:37:44 +09:00
|
|
|
prInfo = PrepareMergedViewPullInfo(ctx, issue)
|
2015-09-02 22:26:56 +09:00
|
|
|
} else {
|
2019-12-19 03:37:44 +09:00
|
|
|
prInfo = PrepareViewPullInfo(ctx, issue)
|
|
|
|
}
|
2015-09-02 17:08:05 +09:00
|
|
|
|
2019-12-19 03:37:44 +09:00
|
|
|
if ctx.Written() {
|
|
|
|
return
|
|
|
|
} else if prInfo == nil {
|
|
|
|
ctx.NotFound("ViewPullFiles", nil)
|
|
|
|
return
|
|
|
|
}
|
2015-09-02 17:08:05 +09:00
|
|
|
|
2019-12-19 03:37:44 +09:00
|
|
|
headCommitID, err := gitRepo.GetRefCommitID(pull.GetGitRefName())
|
|
|
|
if err != nil {
|
|
|
|
ctx.ServerError("GetRefCommitID", err)
|
|
|
|
return
|
|
|
|
}
|
2015-09-02 22:26:56 +09:00
|
|
|
|
2019-12-19 03:37:44 +09:00
|
|
|
startCommitID = prInfo.MergeBase
|
|
|
|
endCommitID = headCommitID
|
2017-06-21 07:25:38 +09:00
|
|
|
|
2019-12-19 03:37:44 +09:00
|
|
|
ctx.Data["Username"] = ctx.Repo.Owner.Name
|
|
|
|
ctx.Data["Reponame"] = ctx.Repo.Repository.Name
|
2019-11-15 11:52:59 +09:00
|
|
|
ctx.Data["AfterCommitID"] = endCommitID
|
2015-09-02 17:08:05 +09:00
|
|
|
|
2021-11-22 01:51:08 +09:00
|
|
|
fileOnly := ctx.FormBool("file-only")
|
|
|
|
|
|
|
|
maxLines, maxFiles := setting.Git.MaxGitDiffLines, setting.Git.MaxGitDiffFiles
|
|
|
|
files := ctx.FormStrings("files")
|
|
|
|
if fileOnly && (len(files) == 2 || len(files) == 1) {
|
|
|
|
maxLines, maxFiles = -1, -1
|
|
|
|
}
|
2022-05-08 03:28:10 +09:00
|
|
|
diffOptions := &gitdiff.DiffOptions{
|
|
|
|
BeforeCommitID: startCommitID,
|
|
|
|
AfterCommitID: endCommitID,
|
|
|
|
SkipTo: ctx.FormString("skip-to"),
|
|
|
|
MaxLines: maxLines,
|
|
|
|
MaxLineCharacters: setting.Git.MaxGitDiffLineCharacters,
|
|
|
|
MaxFiles: maxFiles,
|
|
|
|
WhitespaceBehavior: gitdiff.GetWhitespaceFlag(ctx.Data["WhitespaceBehavior"].(string)),
|
|
|
|
}
|
|
|
|
|
|
|
|
var methodWithError string
|
|
|
|
var diff *gitdiff.Diff
|
|
|
|
if !ctx.IsSigned {
|
|
|
|
diff, err = gitdiff.GetDiff(gitRepo, diffOptions, files...)
|
|
|
|
methodWithError = "GetDiff"
|
|
|
|
} else {
|
|
|
|
diff, err = gitdiff.SyncAndGetUserSpecificDiff(ctx, ctx.Doer.ID, pull, gitRepo, diffOptions, files...)
|
|
|
|
methodWithError = "SyncAndGetUserSpecificDiff"
|
|
|
|
}
|
2015-09-02 17:08:05 +09:00
|
|
|
if err != nil {
|
2022-05-08 03:28:10 +09:00
|
|
|
ctx.ServerError(methodWithError, err)
|
2015-09-02 17:08:05 +09:00
|
|
|
return
|
|
|
|
}
|
2018-08-06 13:43:22 +09:00
|
|
|
|
2022-05-08 03:28:10 +09:00
|
|
|
ctx.PageData["prReview"] = map[string]interface{}{
|
|
|
|
"numberOfFiles": diff.NumFiles,
|
|
|
|
"numberOfViewedFiles": diff.NumViewedFiles,
|
|
|
|
}
|
|
|
|
|
2022-03-22 16:03:22 +09:00
|
|
|
if err = diff.LoadComments(ctx, issue, ctx.Doer); err != nil {
|
2018-08-06 13:43:22 +09:00
|
|
|
ctx.ServerError("LoadComments", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2020-10-14 03:50:57 +09:00
|
|
|
if err = pull.LoadProtectedBranch(); err != nil {
|
|
|
|
ctx.ServerError("LoadProtectedBranch", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if pull.ProtectedBranch != nil {
|
|
|
|
glob := pull.ProtectedBranch.GetProtectedFilePatterns()
|
|
|
|
if len(glob) != 0 {
|
|
|
|
for _, file := range diff.Files {
|
|
|
|
file.IsProtected = pull.ProtectedBranch.IsProtectedFile(glob, file.Name)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-09-02 17:08:05 +09:00
|
|
|
ctx.Data["Diff"] = diff
|
2020-05-26 14:58:07 +09:00
|
|
|
ctx.Data["DiffNotAvailable"] = diff.NumFiles == 0
|
2015-09-02 17:08:05 +09:00
|
|
|
|
2019-09-16 18:03:22 +09:00
|
|
|
baseCommit, err := ctx.Repo.GitRepo.GetCommit(startCommitID)
|
|
|
|
if err != nil {
|
|
|
|
ctx.ServerError("GetCommit", err)
|
|
|
|
return
|
|
|
|
}
|
2015-09-02 22:26:56 +09:00
|
|
|
commit, err := gitRepo.GetCommit(endCommitID)
|
2015-09-02 17:08:05 +09:00
|
|
|
if err != nil {
|
2018-01-11 06:34:17 +09:00
|
|
|
ctx.ServerError("GetCommit", err)
|
2015-09-02 17:08:05 +09:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2022-03-22 16:03:22 +09:00
|
|
|
if ctx.IsSigned && ctx.Doer != nil {
|
2022-06-13 18:37:59 +09:00
|
|
|
if ctx.Data["CanMarkConversation"], err = issues_model.CanMarkConversation(issue, ctx.Doer); err != nil {
|
2020-04-18 22:50:25 +09:00
|
|
|
ctx.ServerError("CanMarkConversation", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-11-17 03:18:25 +09:00
|
|
|
setCompareContext(ctx, baseCommit, commit, ctx.Repo.Owner.Name, ctx.Repo.Repository.Name)
|
2019-09-16 18:03:22 +09:00
|
|
|
|
2018-08-06 13:43:22 +09:00
|
|
|
ctx.Data["RequireTribute"] = true
|
2022-06-06 17:01:49 +09:00
|
|
|
if ctx.Data["Assignees"], err = repo_model.GetRepoAssignees(ctx, ctx.Repo.Repository); err != nil {
|
2018-08-06 13:43:22 +09:00
|
|
|
ctx.ServerError("GetAssignees", err)
|
|
|
|
return
|
|
|
|
}
|
2020-12-22 00:39:28 +09:00
|
|
|
handleTeamMentions(ctx)
|
|
|
|
if ctx.Written() {
|
|
|
|
return
|
|
|
|
}
|
2022-05-07 14:35:12 +09:00
|
|
|
|
2022-06-13 18:37:59 +09:00
|
|
|
currentReview, err := issues_model.GetCurrentReview(ctx, ctx.Doer, issue)
|
|
|
|
if err != nil && !issues_model.IsErrReviewNotExist(err) {
|
2018-08-06 13:43:22 +09:00
|
|
|
ctx.ServerError("GetCurrentReview", err)
|
|
|
|
return
|
|
|
|
}
|
2022-05-07 14:35:12 +09:00
|
|
|
numPendingCodeComments := int64(0)
|
|
|
|
if currentReview != nil {
|
2022-06-13 18:37:59 +09:00
|
|
|
numPendingCodeComments, err = issues_model.CountComments(&issues_model.FindCommentsOptions{
|
|
|
|
Type: issues_model.CommentTypeCode,
|
2022-05-07 14:35:12 +09:00
|
|
|
ReviewID: currentReview.ID,
|
|
|
|
IssueID: issue.ID,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
ctx.ServerError("CountComments", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ctx.Data["CurrentReview"] = currentReview
|
|
|
|
ctx.Data["PendingCodeCommentNumber"] = numPendingCodeComments
|
|
|
|
|
2019-12-16 15:20:25 +09:00
|
|
|
getBranchData(ctx, issue)
|
2022-03-22 16:03:22 +09:00
|
|
|
ctx.Data["IsIssuePoster"] = ctx.IsSigned && issue.IsPoster(ctx.Doer.ID)
|
2020-04-04 14:39:48 +09:00
|
|
|
ctx.Data["HasIssuesOrPullsWritePermission"] = ctx.Repo.CanWriteIssuesOrPulls(issue.IsPull)
|
2021-06-15 10:12:33 +09:00
|
|
|
|
|
|
|
ctx.Data["IsAttachmentEnabled"] = setting.Attachment.Enabled
|
|
|
|
upload.AddUploadContext(ctx, "comment")
|
|
|
|
|
2021-04-06 00:30:52 +09:00
|
|
|
ctx.HTML(http.StatusOK, tplPullFiles)
|
2015-09-02 17:08:05 +09:00
|
|
|
}
|
2015-08-31 16:24:28 +09:00
|
|
|
|
2020-08-05 05:55:22 +09:00
|
|
|
// UpdatePullRequest merge PR's baseBranch into headBranch
|
2020-01-17 15:03:40 +09:00
|
|
|
func UpdatePullRequest(ctx *context.Context) {
|
|
|
|
issue := checkPullInfo(ctx)
|
|
|
|
if ctx.Written() {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if issue.IsClosed {
|
|
|
|
ctx.NotFound("MergePullRequest", nil)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if issue.PullRequest.HasMerged {
|
|
|
|
ctx.NotFound("MergePullRequest", nil)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2021-08-31 23:03:45 +09:00
|
|
|
rebase := ctx.FormString("style") == "rebase"
|
|
|
|
|
2022-04-28 20:48:48 +09:00
|
|
|
if err := issue.PullRequest.LoadBaseRepoCtx(ctx); err != nil {
|
2021-01-15 05:27:22 +09:00
|
|
|
ctx.ServerError("LoadBaseRepo", err)
|
2020-01-17 15:03:40 +09:00
|
|
|
return
|
|
|
|
}
|
2022-04-28 20:48:48 +09:00
|
|
|
if err := issue.PullRequest.LoadHeadRepoCtx(ctx); err != nil {
|
2021-01-15 05:27:22 +09:00
|
|
|
ctx.ServerError("LoadHeadRepo", err)
|
2020-01-17 15:03:40 +09:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2022-04-28 20:48:48 +09:00
|
|
|
allowedUpdateByMerge, allowedUpdateByRebase, err := pull_service.IsUserAllowedToUpdate(ctx, issue.PullRequest, ctx.Doer)
|
2020-01-17 15:03:40 +09:00
|
|
|
if err != nil {
|
|
|
|
ctx.ServerError("IsUserAllowedToMerge", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// ToDo: add check if maintainers are allowed to change branch ... (need migration & co)
|
2021-08-31 23:03:45 +09:00
|
|
|
if (!allowedUpdateByMerge && !rebase) || (rebase && !allowedUpdateByRebase) {
|
2020-01-17 15:03:40 +09:00
|
|
|
ctx.Flash.Error(ctx.Tr("repo.pulls.update_not_allowed"))
|
2021-11-17 03:18:25 +09:00
|
|
|
ctx.Redirect(issue.Link())
|
2020-01-17 15:03:40 +09:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// default merge commit message
|
|
|
|
message := fmt.Sprintf("Merge branch '%s' into %s", issue.PullRequest.BaseBranch, issue.PullRequest.HeadBranch)
|
|
|
|
|
2022-03-22 16:03:22 +09:00
|
|
|
if err = pull_service.Update(ctx, issue.PullRequest, ctx.Doer, message, rebase); err != nil {
|
2020-01-17 15:03:40 +09:00
|
|
|
if models.IsErrMergeConflicts(err) {
|
|
|
|
conflictError := err.(models.ErrMergeConflicts)
|
2021-12-15 15:59:57 +09:00
|
|
|
flashError, err := ctx.RenderToString(tplAlertDetails, map[string]interface{}{
|
2020-10-21 08:50:10 +09:00
|
|
|
"Message": ctx.Tr("repo.pulls.merge_conflict"),
|
|
|
|
"Summary": ctx.Tr("repo.pulls.merge_conflict_summary"),
|
|
|
|
"Details": utils.SanitizeFlashErrorString(conflictError.StdErr) + "<br>" + utils.SanitizeFlashErrorString(conflictError.StdOut),
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
ctx.ServerError("UpdatePullRequest.HTMLString", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
ctx.Flash.Error(flashError)
|
2021-11-17 03:18:25 +09:00
|
|
|
ctx.Redirect(issue.Link())
|
2020-01-17 15:03:40 +09:00
|
|
|
return
|
2021-09-05 18:30:40 +09:00
|
|
|
} else if models.IsErrRebaseConflicts(err) {
|
|
|
|
conflictError := err.(models.ErrRebaseConflicts)
|
2021-12-15 15:59:57 +09:00
|
|
|
flashError, err := ctx.RenderToString(tplAlertDetails, map[string]interface{}{
|
2021-09-05 18:30:40 +09:00
|
|
|
"Message": ctx.Tr("repo.pulls.rebase_conflict", utils.SanitizeFlashErrorString(conflictError.CommitSHA)),
|
|
|
|
"Summary": ctx.Tr("repo.pulls.rebase_conflict_summary"),
|
|
|
|
"Details": utils.SanitizeFlashErrorString(conflictError.StdErr) + "<br>" + utils.SanitizeFlashErrorString(conflictError.StdOut),
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
ctx.ServerError("UpdatePullRequest.HTMLString", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
ctx.Flash.Error(flashError)
|
2021-11-17 03:18:25 +09:00
|
|
|
ctx.Redirect(issue.Link())
|
2021-09-05 18:30:40 +09:00
|
|
|
return
|
|
|
|
|
2020-01-17 15:03:40 +09:00
|
|
|
}
|
|
|
|
ctx.Flash.Error(err.Error())
|
2021-11-17 03:18:25 +09:00
|
|
|
ctx.Redirect(issue.Link())
|
2020-08-04 05:50:29 +09:00
|
|
|
return
|
2020-01-17 15:03:40 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
time.Sleep(1 * time.Second)
|
|
|
|
|
|
|
|
ctx.Flash.Success(ctx.Tr("repo.pulls.update_branch_success"))
|
2021-11-17 03:18:25 +09:00
|
|
|
ctx.Redirect(issue.Link())
|
2020-01-17 15:03:40 +09:00
|
|
|
}
|
|
|
|
|
2016-11-24 16:04:31 +09:00
|
|
|
// MergePullRequest response for merging pull request
|
2021-01-27 00:36:53 +09:00
|
|
|
func MergePullRequest(ctx *context.Context) {
|
2021-04-07 04:44:05 +09:00
|
|
|
form := web.GetForm(ctx).(*forms.MergePullRequestForm)
|
2015-10-19 08:30:39 +09:00
|
|
|
issue := checkPullInfo(ctx)
|
2015-09-02 22:26:56 +09:00
|
|
|
if ctx.Written() {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2018-12-14 00:55:43 +09:00
|
|
|
pr := issue.PullRequest
|
2022-03-31 23:53:08 +09:00
|
|
|
pr.Issue = issue
|
|
|
|
pr.Issue.Repo = ctx.Repo.Repository
|
|
|
|
manuallMerge := repo_model.MergeStyle(form.Do) == repo_model.MergeStyleManuallyMerged
|
|
|
|
forceMerge := form.ForceMerge != nil && *form.ForceMerge
|
2015-09-02 22:26:56 +09:00
|
|
|
|
2022-05-04 04:46:28 +09:00
|
|
|
// start with merging by checking
|
2022-03-31 23:53:08 +09:00
|
|
|
if err := pull_service.CheckPullMergable(ctx, ctx.Doer, &ctx.Repo.Permission, pr, manuallMerge, forceMerge); err != nil {
|
|
|
|
if errors.Is(err, pull_service.ErrIsClosed) {
|
|
|
|
if issue.IsPull {
|
|
|
|
ctx.Flash.Error(ctx.Tr("repo.pulls.is_closed"))
|
|
|
|
ctx.Redirect(issue.Link())
|
|
|
|
} else {
|
|
|
|
ctx.Flash.Error(ctx.Tr("repo.issues.closed_title"))
|
|
|
|
ctx.Redirect(issue.Link())
|
|
|
|
}
|
|
|
|
} else if errors.Is(err, pull_service.ErrUserNotAllowedToMerge) {
|
|
|
|
ctx.Flash.Error(ctx.Tr("repo.pulls.update_not_allowed"))
|
|
|
|
ctx.Redirect(issue.Link())
|
|
|
|
} else if errors.Is(err, pull_service.ErrHasMerged) {
|
|
|
|
ctx.Flash.Error(ctx.Tr("repo.pulls.has_merged"))
|
|
|
|
ctx.Redirect(issue.Link())
|
|
|
|
} else if errors.Is(err, pull_service.ErrIsWorkInProgress) {
|
|
|
|
ctx.Flash.Error(ctx.Tr("repo.pulls.no_merge_wip"))
|
|
|
|
ctx.Redirect(issue.Link())
|
|
|
|
} else if errors.Is(err, pull_service.ErrNotMergableState) {
|
|
|
|
ctx.Flash.Error(ctx.Tr("repo.pulls.no_merge_not_ready"))
|
|
|
|
ctx.Redirect(issue.Link())
|
|
|
|
} else if models.IsErrDisallowedToMerge(err) {
|
|
|
|
ctx.Flash.Error(ctx.Tr("repo.pulls.no_merge_not_ready"))
|
|
|
|
ctx.Redirect(issue.Link())
|
|
|
|
} else if asymkey_service.IsErrWontSign(err) {
|
|
|
|
ctx.Flash.Error(err.Error()) // has not translation ...
|
|
|
|
ctx.Redirect(issue.Link())
|
|
|
|
} else if errors.Is(err, pull_service.ErrDependenciesLeft) {
|
|
|
|
ctx.Flash.Error(ctx.Tr("repo.issues.dependency.pr_close_blocked"))
|
|
|
|
ctx.Redirect(issue.Link())
|
|
|
|
} else {
|
|
|
|
ctx.ServerError("WebCheck", err)
|
|
|
|
}
|
2020-01-27 19:26:53 +09:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2021-03-04 12:41:23 +09:00
|
|
|
// handle manually-merged mark
|
2022-03-31 23:53:08 +09:00
|
|
|
if manuallMerge {
|
|
|
|
if err := pull_service.MergedManually(pr, ctx.Doer, ctx.Repo.GitRepo, form.MergeCommitID); err != nil {
|
2021-03-04 12:41:23 +09:00
|
|
|
if models.IsErrInvalidMergeStyle(err) {
|
|
|
|
ctx.Flash.Error(ctx.Tr("repo.pulls.invalid_merge_option"))
|
2021-11-17 03:18:25 +09:00
|
|
|
ctx.Redirect(issue.Link())
|
2021-03-04 12:41:23 +09:00
|
|
|
} else if strings.Contains(err.Error(), "Wrong commit ID") {
|
|
|
|
ctx.Flash.Error(ctx.Tr("repo.pulls.wrong_commit_id"))
|
2021-11-17 03:18:25 +09:00
|
|
|
ctx.Redirect(issue.Link())
|
2022-05-04 04:46:28 +09:00
|
|
|
} else {
|
|
|
|
ctx.ServerError("MergedManually", err)
|
2021-03-04 12:41:23 +09:00
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2021-11-17 03:18:25 +09:00
|
|
|
ctx.Redirect(issue.Link())
|
2021-03-04 12:41:23 +09:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2022-05-08 21:32:45 +09:00
|
|
|
message := strings.TrimSpace(form.MergeTitleField)
|
|
|
|
if len(message) == 0 {
|
|
|
|
var err error
|
|
|
|
message, err = pull_service.GetDefaultMergeMessage(ctx.Repo.GitRepo, pr, repo_model.MergeStyle(form.Do))
|
|
|
|
if err != nil {
|
|
|
|
ctx.ServerError("GetDefaultMergeMessage", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
form.MergeMessageField = strings.TrimSpace(form.MergeMessageField)
|
|
|
|
if len(form.MergeMessageField) > 0 {
|
|
|
|
message += "\n\n" + form.MergeMessageField
|
2018-07-18 06:23:58 +09:00
|
|
|
}
|
|
|
|
|
2022-06-11 23:44:20 +09:00
|
|
|
if form.MergeWhenChecksSucceed {
|
|
|
|
// delete all scheduled auto merges
|
|
|
|
_ = pull_model.DeleteScheduledAutoMerge(ctx, pr.ID)
|
|
|
|
// schedule auto merge
|
|
|
|
scheduled, err := automerge.ScheduleAutoMerge(ctx, ctx.Doer, pr, repo_model.MergeStyle(form.Do), message)
|
|
|
|
if err != nil {
|
|
|
|
ctx.ServerError("ScheduleAutoMerge", err)
|
|
|
|
return
|
|
|
|
} else if scheduled {
|
|
|
|
// nothing more to do ...
|
|
|
|
ctx.Flash.Success(ctx.Tr("repo.pulls.auto_merge_newly_scheduled"))
|
|
|
|
ctx.Redirect(fmt.Sprintf("%s/pulls/%d", ctx.Repo.RepoLink, pr.Index))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-05-08 21:32:45 +09:00
|
|
|
if err := pull_service.Merge(ctx, pr, ctx.Doer, ctx.Repo.GitRepo, repo_model.MergeStyle(form.Do), form.HeadCommitID, message); err != nil {
|
2018-01-06 03:56:50 +09:00
|
|
|
if models.IsErrInvalidMergeStyle(err) {
|
|
|
|
ctx.Flash.Error(ctx.Tr("repo.pulls.invalid_merge_option"))
|
2021-11-17 03:18:25 +09:00
|
|
|
ctx.Redirect(issue.Link())
|
2019-11-10 17:42:51 +09:00
|
|
|
} else if models.IsErrMergeConflicts(err) {
|
|
|
|
conflictError := err.(models.ErrMergeConflicts)
|
2021-12-15 15:59:57 +09:00
|
|
|
flashError, err := ctx.RenderToString(tplAlertDetails, map[string]interface{}{
|
2020-10-21 08:50:10 +09:00
|
|
|
"Message": ctx.Tr("repo.editor.merge_conflict"),
|
|
|
|
"Summary": ctx.Tr("repo.editor.merge_conflict_summary"),
|
|
|
|
"Details": utils.SanitizeFlashErrorString(conflictError.StdErr) + "<br>" + utils.SanitizeFlashErrorString(conflictError.StdOut),
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
ctx.ServerError("MergePullRequest.HTMLString", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
ctx.Flash.Error(flashError)
|
2021-11-17 03:18:25 +09:00
|
|
|
ctx.Redirect(issue.Link())
|
2019-11-10 17:42:51 +09:00
|
|
|
} else if models.IsErrRebaseConflicts(err) {
|
|
|
|
conflictError := err.(models.ErrRebaseConflicts)
|
2021-12-15 15:59:57 +09:00
|
|
|
flashError, err := ctx.RenderToString(tplAlertDetails, map[string]interface{}{
|
2020-10-22 04:59:12 +09:00
|
|
|
"Message": ctx.Tr("repo.pulls.rebase_conflict", utils.SanitizeFlashErrorString(conflictError.CommitSHA)),
|
|
|
|
"Summary": ctx.Tr("repo.pulls.rebase_conflict_summary"),
|
2020-10-21 08:50:10 +09:00
|
|
|
"Details": utils.SanitizeFlashErrorString(conflictError.StdErr) + "<br>" + utils.SanitizeFlashErrorString(conflictError.StdOut),
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
ctx.ServerError("MergePullRequest.HTMLString", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
ctx.Flash.Error(flashError)
|
2021-11-17 03:18:25 +09:00
|
|
|
ctx.Redirect(issue.Link())
|
2019-11-10 17:42:51 +09:00
|
|
|
} else if models.IsErrMergeUnrelatedHistories(err) {
|
|
|
|
log.Debug("MergeUnrelatedHistories error: %v", err)
|
|
|
|
ctx.Flash.Error(ctx.Tr("repo.pulls.unrelated_histories"))
|
2021-11-17 03:18:25 +09:00
|
|
|
ctx.Redirect(issue.Link())
|
2020-03-28 13:13:18 +09:00
|
|
|
} else if git.IsErrPushOutOfDate(err) {
|
2019-11-10 17:42:51 +09:00
|
|
|
log.Debug("MergePushOutOfDate error: %v", err)
|
|
|
|
ctx.Flash.Error(ctx.Tr("repo.pulls.merge_out_of_date"))
|
2021-11-17 03:18:25 +09:00
|
|
|
ctx.Redirect(issue.Link())
|
2021-12-20 09:32:54 +09:00
|
|
|
} else if models.IsErrSHADoesNotMatch(err) {
|
|
|
|
log.Debug("MergeHeadOutOfDate error: %v", err)
|
|
|
|
ctx.Flash.Error(ctx.Tr("repo.pulls.head_out_of_date"))
|
|
|
|
ctx.Redirect(issue.Link())
|
2020-03-28 13:13:18 +09:00
|
|
|
} else if git.IsErrPushRejected(err) {
|
2020-02-22 22:08:48 +09:00
|
|
|
log.Debug("MergePushRejected error: %v", err)
|
2020-03-28 13:13:18 +09:00
|
|
|
pushrejErr := err.(*git.ErrPushRejected)
|
2020-02-22 22:08:48 +09:00
|
|
|
message := pushrejErr.Message
|
|
|
|
if len(message) == 0 {
|
|
|
|
ctx.Flash.Error(ctx.Tr("repo.pulls.push_rejected_no_message"))
|
|
|
|
} else {
|
2021-12-15 15:59:57 +09:00
|
|
|
flashError, err := ctx.RenderToString(tplAlertDetails, map[string]interface{}{
|
2020-10-21 08:50:10 +09:00
|
|
|
"Message": ctx.Tr("repo.pulls.push_rejected"),
|
|
|
|
"Summary": ctx.Tr("repo.pulls.push_rejected_summary"),
|
|
|
|
"Details": utils.SanitizeFlashErrorString(pushrejErr.Message),
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
ctx.ServerError("MergePullRequest.HTMLString", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
ctx.Flash.Error(flashError)
|
2020-02-22 22:08:48 +09:00
|
|
|
}
|
2021-11-17 03:18:25 +09:00
|
|
|
ctx.Redirect(issue.Link())
|
2022-05-04 04:46:28 +09:00
|
|
|
} else {
|
|
|
|
ctx.ServerError("Merge", err)
|
2018-01-06 03:56:50 +09:00
|
|
|
}
|
2015-09-02 22:26:56 +09:00
|
|
|
return
|
|
|
|
}
|
2022-05-04 04:46:28 +09:00
|
|
|
log.Trace("Pull request merged: %d", pr.ID)
|
2015-09-02 22:26:56 +09:00
|
|
|
|
2022-03-22 16:03:22 +09:00
|
|
|
if err := stopTimerIfAvailable(ctx.Doer, issue); err != nil {
|
2019-02-05 20:38:11 +09:00
|
|
|
ctx.ServerError("CreateOrStopIssueStopwatch", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2015-09-02 22:26:56 +09:00
|
|
|
log.Trace("Pull request merged: %d", pr.ID)
|
2021-07-13 08:26:25 +09:00
|
|
|
|
|
|
|
if form.DeleteBranchAfterMerge {
|
2022-01-04 04:45:58 +09:00
|
|
|
// Don't cleanup when other pr use this branch as head branch
|
2022-06-13 18:37:59 +09:00
|
|
|
exist, err := issues_model.HasUnmergedPullRequestsByHeadInfo(ctx, pr.HeadRepoID, pr.HeadBranch)
|
2022-01-04 04:45:58 +09:00
|
|
|
if err != nil {
|
|
|
|
ctx.ServerError("HasUnmergedPullRequestsByHeadInfo", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if exist {
|
|
|
|
ctx.Redirect(issue.Link())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2021-07-13 08:26:25 +09:00
|
|
|
var headRepo *git.Repository
|
|
|
|
if ctx.Repo != nil && ctx.Repo.Repository != nil && pr.HeadRepoID == ctx.Repo.Repository.ID && ctx.Repo.GitRepo != nil {
|
|
|
|
headRepo = ctx.Repo.GitRepo
|
|
|
|
} else {
|
2022-03-30 04:13:41 +09:00
|
|
|
headRepo, err = git.OpenRepository(ctx, pr.HeadRepo.RepoPath())
|
2021-07-13 08:26:25 +09:00
|
|
|
if err != nil {
|
|
|
|
ctx.ServerError(fmt.Sprintf("OpenRepository[%s]", pr.HeadRepo.RepoPath()), err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
defer headRepo.Close()
|
|
|
|
}
|
|
|
|
deleteBranch(ctx, pr, headRepo)
|
|
|
|
}
|
|
|
|
|
2021-11-17 03:18:25 +09:00
|
|
|
ctx.Redirect(issue.Link())
|
2022-06-11 23:44:20 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
// CancelAutoMergePullRequest cancels a scheduled pr
|
|
|
|
func CancelAutoMergePullRequest(ctx *context.Context) {
|
|
|
|
issue := checkPullInfo(ctx)
|
|
|
|
if ctx.Written() {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := automerge.RemoveScheduledAutoMerge(ctx, ctx.Doer, issue.PullRequest); err != nil {
|
|
|
|
if db.IsErrNotExist(err) {
|
|
|
|
ctx.Flash.Error(ctx.Tr("repo.pulls.auto_merge_not_scheduled"))
|
|
|
|
ctx.Redirect(fmt.Sprintf("%s/pulls/%d", ctx.Repo.RepoLink, issue.Index))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
ctx.ServerError("RemoveScheduledAutoMerge", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
ctx.Flash.Success(ctx.Tr("repo.pulls.auto_merge_canceled_schedule"))
|
|
|
|
ctx.Redirect(fmt.Sprintf("%s/pulls/%d", ctx.Repo.RepoLink, issue.Index))
|
2015-09-02 22:26:56 +09:00
|
|
|
}
|
|
|
|
|
2022-06-13 18:37:59 +09:00
|
|
|
func stopTimerIfAvailable(user *user_model.User, issue *issues_model.Issue) error {
|
|
|
|
if issues_model.StopwatchExists(user.ID, issue.ID) {
|
|
|
|
if err := issues_model.CreateOrStopIssueStopwatch(user, issue); err != nil {
|
2019-02-05 20:38:11 +09:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2016-11-24 16:04:31 +09:00
|
|
|
// CompareAndPullRequestPost response for creating pull request
|
2021-01-27 00:36:53 +09:00
|
|
|
func CompareAndPullRequestPost(ctx *context.Context) {
|
2021-04-07 04:44:05 +09:00
|
|
|
form := web.GetForm(ctx).(*forms.CreateIssueForm)
|
2015-09-02 08:07:02 +09:00
|
|
|
ctx.Data["Title"] = ctx.Tr("repo.pulls.compare_changes")
|
|
|
|
ctx.Data["PageIsComparePull"] = true
|
|
|
|
ctx.Data["IsDiffCompare"] = true
|
2021-07-25 11:59:27 +09:00
|
|
|
ctx.Data["IsRepoToolbarCommits"] = true
|
|
|
|
ctx.Data["RequireTribute"] = true
|
2018-08-14 04:04:39 +09:00
|
|
|
ctx.Data["PullRequestWorkInProgressPrefixes"] = setting.Repository.PullRequest.WorkInProgressPrefixes
|
2020-10-05 14:49:33 +09:00
|
|
|
ctx.Data["IsAttachmentEnabled"] = setting.Attachment.Enabled
|
|
|
|
upload.AddUploadContext(ctx, "comment")
|
2021-11-10 04:57:58 +09:00
|
|
|
ctx.Data["HasIssuesOrPullsWritePermission"] = ctx.Repo.CanWrite(unit.TypePullRequests)
|
2015-09-02 08:07:02 +09:00
|
|
|
|
|
|
|
var (
|
|
|
|
repo = ctx.Repo.Repository
|
|
|
|
attachments []string
|
|
|
|
)
|
|
|
|
|
2021-09-27 21:19:34 +09:00
|
|
|
ci := ParseCompareInfo(ctx)
|
2021-08-31 16:43:31 +09:00
|
|
|
defer func() {
|
2021-10-01 04:31:02 +09:00
|
|
|
if ci != nil && ci.HeadGitRepo != nil {
|
2021-09-27 21:19:34 +09:00
|
|
|
ci.HeadGitRepo.Close()
|
2021-08-31 16:43:31 +09:00
|
|
|
}
|
|
|
|
}()
|
2015-09-02 08:07:02 +09:00
|
|
|
if ctx.Written() {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2021-01-27 00:36:53 +09:00
|
|
|
labelIDs, assigneeIDs, milestoneID, _ := ValidateRepoMetas(ctx, *form, true)
|
2015-09-02 08:07:02 +09:00
|
|
|
if ctx.Written() {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2020-08-18 13:23:45 +09:00
|
|
|
if setting.Attachment.Enabled {
|
2016-08-11 21:48:08 +09:00
|
|
|
attachments = form.Files
|
2015-09-02 08:07:02 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
if ctx.HasError() {
|
2021-01-30 17:55:53 +09:00
|
|
|
middleware.AssignForm(form, ctx.Data)
|
2016-08-16 06:04:44 +09:00
|
|
|
|
|
|
|
// This stage is already stop creating new pull request, so it does not matter if it has
|
|
|
|
// something to compare or not.
|
2021-09-27 21:19:34 +09:00
|
|
|
PrepareCompareDiff(ctx, ci,
|
2021-02-13 13:35:43 +09:00
|
|
|
gitdiff.GetWhitespaceFlag(ctx.Data["WhitespaceBehavior"].(string)))
|
2016-08-16 06:04:44 +09:00
|
|
|
if ctx.Written() {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2021-07-25 11:59:27 +09:00
|
|
|
if len(form.Title) > 255 {
|
|
|
|
var trailer string
|
|
|
|
form.Title, trailer = util.SplitStringAtByteN(form.Title, 255)
|
|
|
|
|
|
|
|
form.Content = trailer + "\n\n" + form.Content
|
|
|
|
}
|
|
|
|
middleware.AssignForm(form, ctx.Data)
|
|
|
|
|
2021-04-06 00:30:52 +09:00
|
|
|
ctx.HTML(http.StatusOK, tplCompareDiff)
|
2015-09-02 08:07:02 +09:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-01-21 20:45:32 +09:00
|
|
|
if util.IsEmptyString(form.Title) {
|
2021-09-27 21:19:34 +09:00
|
|
|
PrepareCompareDiff(ctx, ci,
|
2021-02-13 13:35:43 +09:00
|
|
|
gitdiff.GetWhitespaceFlag(ctx.Data["WhitespaceBehavior"].(string)))
|
2019-01-21 20:45:32 +09:00
|
|
|
if ctx.Written() {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-06-08 05:29:29 +09:00
|
|
|
ctx.RenderWithErr(ctx.Tr("repo.issues.new.title_empty"), tplCompareDiff, form)
|
2019-01-21 20:45:32 +09:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2022-09-02 16:58:49 +09:00
|
|
|
content := form.Content
|
|
|
|
if filename := ctx.Req.Form.Get("template-file"); filename != "" {
|
|
|
|
if template, err := issue_template.UnmarshalFromRepo(ctx.Repo.GitRepo, ctx.Repo.Repository.DefaultBranch, filename); err == nil {
|
|
|
|
content = issue_template.RenderToMarkdown(template, ctx.Req.Form)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-06-13 18:37:59 +09:00
|
|
|
pullIssue := &issues_model.Issue{
|
2015-09-02 08:07:02 +09:00
|
|
|
RepoID: repo.ID,
|
2021-11-17 03:18:25 +09:00
|
|
|
Repo: repo,
|
2016-08-14 19:32:24 +09:00
|
|
|
Title: form.Title,
|
2022-03-22 16:03:22 +09:00
|
|
|
PosterID: ctx.Doer.ID,
|
|
|
|
Poster: ctx.Doer,
|
2015-09-02 08:07:02 +09:00
|
|
|
MilestoneID: milestoneID,
|
|
|
|
IsPull: true,
|
2022-09-02 16:58:49 +09:00
|
|
|
Content: content,
|
2015-09-02 08:07:02 +09:00
|
|
|
}
|
2022-06-13 18:37:59 +09:00
|
|
|
pullRequest := &issues_model.PullRequest{
|
2022-04-29 00:45:33 +09:00
|
|
|
HeadRepoID: ci.HeadRepo.ID,
|
|
|
|
BaseRepoID: repo.ID,
|
|
|
|
HeadBranch: ci.HeadBranch,
|
|
|
|
BaseBranch: ci.BaseBranch,
|
|
|
|
HeadRepo: ci.HeadRepo,
|
|
|
|
BaseRepo: repo,
|
|
|
|
MergeBase: ci.CompareInfo.MergeBase,
|
2022-06-13 18:37:59 +09:00
|
|
|
Type: issues_model.PullRequestGitea,
|
2022-04-29 00:45:33 +09:00
|
|
|
AllowMaintainerEdit: form.AllowMaintainerEdit,
|
2016-02-24 21:56:54 +09:00
|
|
|
}
|
2016-08-16 06:04:44 +09:00
|
|
|
// FIXME: check error in the case two people send pull request at almost same time, give nice error prompt
|
|
|
|
// instead of 500.
|
2018-05-10 01:29:04 +09:00
|
|
|
|
2022-01-20 08:26:57 +09:00
|
|
|
if err := pull_service.NewPullRequest(ctx, repo, pullIssue, labelIDs, attachments, pullRequest, assigneeIDs); err != nil {
|
2022-06-13 18:37:59 +09:00
|
|
|
if repo_model.IsErrUserDoesNotHaveAccessToRepo(err) {
|
2021-04-06 00:30:52 +09:00
|
|
|
ctx.Error(http.StatusBadRequest, "UserDoesNotHaveAccessToRepo", err.Error())
|
2018-05-10 01:29:04 +09:00
|
|
|
return
|
2020-06-09 03:07:41 +09:00
|
|
|
} else if git.IsErrPushRejected(err) {
|
|
|
|
pushrejErr := err.(*git.ErrPushRejected)
|
|
|
|
message := pushrejErr.Message
|
|
|
|
if len(message) == 0 {
|
|
|
|
ctx.Flash.Error(ctx.Tr("repo.pulls.push_rejected_no_message"))
|
|
|
|
} else {
|
2021-12-15 15:59:57 +09:00
|
|
|
flashError, err := ctx.RenderToString(tplAlertDetails, map[string]interface{}{
|
2020-10-21 08:50:10 +09:00
|
|
|
"Message": ctx.Tr("repo.pulls.push_rejected"),
|
|
|
|
"Summary": ctx.Tr("repo.pulls.push_rejected_summary"),
|
|
|
|
"Details": utils.SanitizeFlashErrorString(pushrejErr.Message),
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
ctx.ServerError("CompareAndPullRequest.HTMLString", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
ctx.Flash.Error(flashError)
|
2020-06-09 03:07:41 +09:00
|
|
|
}
|
2021-11-17 03:18:25 +09:00
|
|
|
ctx.Redirect(pullIssue.Link())
|
2020-06-09 03:07:41 +09:00
|
|
|
return
|
2018-05-10 01:29:04 +09:00
|
|
|
}
|
2018-01-11 06:34:17 +09:00
|
|
|
ctx.ServerError("NewPullRequest", err)
|
2015-09-02 08:07:02 +09:00
|
|
|
return
|
2015-12-11 01:18:56 +09:00
|
|
|
}
|
|
|
|
|
2016-02-24 21:56:54 +09:00
|
|
|
log.Trace("Pull request created: %d/%d", repo.ID, pullIssue.ID)
|
2021-11-17 03:18:25 +09:00
|
|
|
ctx.Redirect(pullIssue.Link())
|
2014-03-24 19:25:15 +09:00
|
|
|
}
|
2015-10-24 16:36:47 +09:00
|
|
|
|
2017-06-21 10:00:03 +09:00
|
|
|
// CleanUpPullRequest responses for delete merged branch when PR has been merged
|
|
|
|
func CleanUpPullRequest(ctx *context.Context) {
|
|
|
|
issue := checkPullInfo(ctx)
|
|
|
|
if ctx.Written() {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2018-12-14 00:55:43 +09:00
|
|
|
pr := issue.PullRequest
|
2017-06-21 10:00:03 +09:00
|
|
|
|
2019-04-21 05:50:34 +09:00
|
|
|
// Don't cleanup unmerged and unclosed PRs
|
|
|
|
if !pr.HasMerged && !issue.IsClosed {
|
2018-01-11 06:34:17 +09:00
|
|
|
ctx.NotFound("CleanUpPullRequest", nil)
|
2017-06-21 10:00:03 +09:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2022-01-04 04:45:58 +09:00
|
|
|
// Don't cleanup when there are other PR's that use this branch as head branch.
|
2022-06-13 18:37:59 +09:00
|
|
|
exist, err := issues_model.HasUnmergedPullRequestsByHeadInfo(ctx, pr.HeadRepoID, pr.HeadBranch)
|
2022-01-04 04:45:58 +09:00
|
|
|
if err != nil {
|
|
|
|
ctx.ServerError("HasUnmergedPullRequestsByHeadInfo", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if exist {
|
|
|
|
ctx.NotFound("CleanUpPullRequest", nil)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2022-04-28 20:48:48 +09:00
|
|
|
if err := pr.LoadHeadRepoCtx(ctx); err != nil {
|
2020-03-03 07:31:55 +09:00
|
|
|
ctx.ServerError("LoadHeadRepo", err)
|
2017-06-21 10:00:03 +09:00
|
|
|
return
|
|
|
|
} else if pr.HeadRepo == nil {
|
|
|
|
// Forked repository has already been deleted
|
2018-01-11 06:34:17 +09:00
|
|
|
ctx.NotFound("CleanUpPullRequest", nil)
|
2017-06-21 10:00:03 +09:00
|
|
|
return
|
2022-04-28 20:48:48 +09:00
|
|
|
} else if err = pr.LoadBaseRepoCtx(ctx); err != nil {
|
2020-03-03 07:31:55 +09:00
|
|
|
ctx.ServerError("LoadBaseRepo", err)
|
2017-06-21 10:00:03 +09:00
|
|
|
return
|
2022-03-23 00:22:54 +09:00
|
|
|
} else if err = pr.HeadRepo.GetOwner(ctx); err != nil {
|
2018-01-11 06:34:17 +09:00
|
|
|
ctx.ServerError("HeadRepo.GetOwner", err)
|
2017-06-21 10:00:03 +09:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2022-05-11 19:09:36 +09:00
|
|
|
perm, err := access_model.GetUserRepoPermission(ctx, pr.HeadRepo, ctx.Doer)
|
2018-11-28 20:26:14 +09:00
|
|
|
if err != nil {
|
|
|
|
ctx.ServerError("GetUserRepoPermission", err)
|
|
|
|
return
|
|
|
|
}
|
2021-11-10 04:57:58 +09:00
|
|
|
if !perm.CanWrite(unit.TypeCode) {
|
2018-01-11 06:34:17 +09:00
|
|
|
ctx.NotFound("CleanUpPullRequest", nil)
|
2017-06-21 10:00:03 +09:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
fullBranchName := pr.HeadRepo.Owner.Name + "/" + pr.HeadBranch
|
|
|
|
|
2021-07-13 08:26:25 +09:00
|
|
|
var gitBaseRepo *git.Repository
|
|
|
|
|
|
|
|
// Assume that the base repo is the current context (almost certainly)
|
|
|
|
if ctx.Repo != nil && ctx.Repo.Repository != nil && ctx.Repo.Repository.ID == pr.BaseRepoID && ctx.Repo.GitRepo != nil {
|
|
|
|
gitBaseRepo = ctx.Repo.GitRepo
|
|
|
|
} else {
|
|
|
|
// If not just open it
|
2022-03-30 04:13:41 +09:00
|
|
|
gitBaseRepo, err = git.OpenRepository(ctx, pr.BaseRepo.RepoPath())
|
2021-07-13 08:26:25 +09:00
|
|
|
if err != nil {
|
|
|
|
ctx.ServerError(fmt.Sprintf("OpenRepository[%s]", pr.BaseRepo.RepoPath()), err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
defer gitBaseRepo.Close()
|
2017-06-21 10:00:03 +09:00
|
|
|
}
|
|
|
|
|
2021-07-13 08:26:25 +09:00
|
|
|
// Now assume that the head repo is the same as the base repo (reasonable chance)
|
|
|
|
gitRepo := gitBaseRepo
|
|
|
|
// But if not: is it the same as the context?
|
|
|
|
if pr.BaseRepoID != pr.HeadRepoID && ctx.Repo != nil && ctx.Repo.Repository != nil && ctx.Repo.Repository.ID == pr.HeadRepoID && ctx.Repo.GitRepo != nil {
|
|
|
|
gitRepo = ctx.Repo.GitRepo
|
|
|
|
} else if pr.BaseRepoID != pr.HeadRepoID {
|
|
|
|
// Otherwise just load it up
|
2022-03-30 04:13:41 +09:00
|
|
|
gitRepo, err = git.OpenRepository(ctx, pr.HeadRepo.RepoPath())
|
2021-07-13 08:26:25 +09:00
|
|
|
if err != nil {
|
|
|
|
ctx.ServerError(fmt.Sprintf("OpenRepository[%s]", pr.HeadRepo.RepoPath()), err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
defer gitRepo.Close()
|
2017-06-21 10:00:03 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
defer func() {
|
2021-04-06 00:30:52 +09:00
|
|
|
ctx.JSON(http.StatusOK, map[string]interface{}{
|
2021-11-17 03:18:25 +09:00
|
|
|
"redirect": issue.Link(),
|
2017-06-21 10:00:03 +09:00
|
|
|
})
|
|
|
|
}()
|
|
|
|
|
|
|
|
// Check if branch has no new commits
|
2018-01-30 21:29:39 +09:00
|
|
|
headCommitID, err := gitBaseRepo.GetRefCommitID(pr.GetGitRefName())
|
|
|
|
if err != nil {
|
2019-04-02 16:48:31 +09:00
|
|
|
log.Error("GetRefCommitID: %v", err)
|
2018-01-30 21:29:39 +09:00
|
|
|
ctx.Flash.Error(ctx.Tr("repo.branch.deletion_failed", fullBranchName))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
branchCommitID, err := gitRepo.GetBranchCommitID(pr.HeadBranch)
|
|
|
|
if err != nil {
|
2019-04-02 16:48:31 +09:00
|
|
|
log.Error("GetBranchCommitID: %v", err)
|
2018-01-30 21:29:39 +09:00
|
|
|
ctx.Flash.Error(ctx.Tr("repo.branch.deletion_failed", fullBranchName))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if headCommitID != branchCommitID {
|
|
|
|
ctx.Flash.Error(ctx.Tr("repo.branch.delete_branch_has_new_commits", fullBranchName))
|
|
|
|
return
|
2017-06-21 10:00:03 +09:00
|
|
|
}
|
|
|
|
|
2021-07-13 08:26:25 +09:00
|
|
|
deleteBranch(ctx, pr, gitRepo)
|
|
|
|
}
|
|
|
|
|
2022-06-13 18:37:59 +09:00
|
|
|
func deleteBranch(ctx *context.Context, pr *issues_model.PullRequest, gitRepo *git.Repository) {
|
2021-07-13 08:26:25 +09:00
|
|
|
fullBranchName := pr.HeadRepo.Owner.Name + "/" + pr.HeadBranch
|
2022-03-22 16:03:22 +09:00
|
|
|
if err := repo_service.DeleteBranch(ctx.Doer, pr.HeadRepo, gitRepo, pr.HeadBranch); err != nil {
|
2021-06-07 23:52:59 +09:00
|
|
|
switch {
|
|
|
|
case git.IsErrBranchNotExist(err):
|
|
|
|
ctx.Flash.Error(ctx.Tr("repo.branch.deletion_failed", fullBranchName))
|
|
|
|
case errors.Is(err, repo_service.ErrBranchIsDefault):
|
|
|
|
ctx.Flash.Error(ctx.Tr("repo.branch.deletion_failed", fullBranchName))
|
|
|
|
case errors.Is(err, repo_service.ErrBranchIsProtected):
|
|
|
|
ctx.Flash.Error(ctx.Tr("repo.branch.deletion_failed", fullBranchName))
|
|
|
|
default:
|
|
|
|
log.Error("DeleteBranch: %v", err)
|
|
|
|
ctx.Flash.Error(ctx.Tr("repo.branch.deletion_failed", fullBranchName))
|
|
|
|
}
|
2017-06-21 10:00:03 +09:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2022-06-13 18:37:59 +09:00
|
|
|
if err := issues_model.AddDeletePRBranchComment(ctx, ctx.Doer, pr.BaseRepo, pr.IssueID, pr.HeadBranch); err != nil {
|
2017-06-21 10:00:03 +09:00
|
|
|
// Do not fail here as branch has already been deleted
|
2019-04-02 16:48:31 +09:00
|
|
|
log.Error("DeleteBranch: %v", err)
|
2017-06-21 10:00:03 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
ctx.Flash.Success(ctx.Tr("repo.branch.deletion_success", fullBranchName))
|
|
|
|
}
|
2018-01-05 19:56:52 +09:00
|
|
|
|
|
|
|
// DownloadPullDiff render a pull's raw diff
|
|
|
|
func DownloadPullDiff(ctx *context.Context) {
|
2019-12-14 07:21:06 +09:00
|
|
|
DownloadPullDiffOrPatch(ctx, false)
|
2018-01-05 19:56:52 +09:00
|
|
|
}
|
2018-01-07 22:10:20 +09:00
|
|
|
|
|
|
|
// DownloadPullPatch render a pull's raw patch
|
|
|
|
func DownloadPullPatch(ctx *context.Context) {
|
2019-12-14 07:21:06 +09:00
|
|
|
DownloadPullDiffOrPatch(ctx, true)
|
|
|
|
}
|
|
|
|
|
|
|
|
// DownloadPullDiffOrPatch render a pull's raw diff or patch
|
|
|
|
func DownloadPullDiffOrPatch(ctx *context.Context, patch bool) {
|
2022-06-13 18:37:59 +09:00
|
|
|
pr, err := issues_model.GetPullRequestByIndex(ctx, ctx.Repo.Repository.ID, ctx.ParamsInt64(":index"))
|
2018-01-07 22:10:20 +09:00
|
|
|
if err != nil {
|
2022-06-13 18:37:59 +09:00
|
|
|
if issues_model.IsErrPullRequestNotExist(err) {
|
2021-10-05 23:41:48 +09:00
|
|
|
ctx.NotFound("GetPullRequestByIndex", err)
|
2018-01-07 22:10:20 +09:00
|
|
|
} else {
|
2021-10-05 23:41:48 +09:00
|
|
|
ctx.ServerError("GetPullRequestByIndex", err)
|
2018-01-07 22:10:20 +09:00
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2021-09-28 06:09:49 +09:00
|
|
|
binary := ctx.FormBool("binary")
|
2018-01-07 22:10:20 +09:00
|
|
|
|
2022-01-20 08:26:57 +09:00
|
|
|
if err := pull_service.DownloadDiffOrPatch(ctx, pr, ctx, patch, binary); err != nil {
|
2019-12-14 07:21:06 +09:00
|
|
|
ctx.ServerError("DownloadDiffOrPatch", err)
|
2018-01-07 22:10:20 +09:00
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
2019-12-16 15:20:25 +09:00
|
|
|
|
|
|
|
// UpdatePullRequestTarget change pull request's target branch
|
|
|
|
func UpdatePullRequestTarget(ctx *context.Context) {
|
|
|
|
issue := GetActionIssue(ctx)
|
|
|
|
pr := issue.PullRequest
|
|
|
|
if ctx.Written() {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if !issue.IsPull {
|
|
|
|
ctx.Error(http.StatusNotFound)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2022-03-22 16:03:22 +09:00
|
|
|
if !ctx.IsSigned || (!issue.IsPoster(ctx.Doer.ID) && !ctx.Repo.CanWriteIssuesOrPulls(issue.IsPull)) {
|
2019-12-16 15:20:25 +09:00
|
|
|
ctx.Error(http.StatusForbidden)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2021-07-29 10:42:15 +09:00
|
|
|
targetBranch := ctx.FormTrim("target_branch")
|
2019-12-16 15:20:25 +09:00
|
|
|
if len(targetBranch) == 0 {
|
|
|
|
ctx.Error(http.StatusNoContent)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2022-03-22 16:03:22 +09:00
|
|
|
if err := pull_service.ChangeTargetBranch(ctx, pr, ctx.Doer, targetBranch); err != nil {
|
2022-06-13 18:37:59 +09:00
|
|
|
if issues_model.IsErrPullRequestAlreadyExists(err) {
|
|
|
|
err := err.(issues_model.ErrPullRequestAlreadyExists)
|
2019-12-16 15:20:25 +09:00
|
|
|
|
|
|
|
RepoRelPath := ctx.Repo.Owner.Name + "/" + ctx.Repo.Repository.Name
|
Fix various typos (#20338)
* Fix various typos
Found via `codespell -q 3 -S ./options/locale,./options/license,./public/vendor -L actived,allways,attachements,ba,befores,commiter,pullrequest,pullrequests,readby,splitted,te,unknwon`
Co-authored-by: zeripath <art27@cantab.net>
2022-07-13 06:32:37 +09:00
|
|
|
errorMessage := ctx.Tr("repo.pulls.has_pull_request", html.EscapeString(ctx.Repo.RepoLink+"/pulls/"+strconv.FormatInt(err.IssueID, 10)), html.EscapeString(RepoRelPath), err.IssueID) // FIXME: Creates url inside locale string
|
2019-12-16 15:20:25 +09:00
|
|
|
|
|
|
|
ctx.Flash.Error(errorMessage)
|
|
|
|
ctx.JSON(http.StatusConflict, map[string]interface{}{
|
|
|
|
"error": err.Error(),
|
|
|
|
"user_error": errorMessage,
|
|
|
|
})
|
2022-06-13 18:37:59 +09:00
|
|
|
} else if issues_model.IsErrIssueIsClosed(err) {
|
2019-12-16 15:20:25 +09:00
|
|
|
errorMessage := ctx.Tr("repo.pulls.is_closed")
|
|
|
|
|
|
|
|
ctx.Flash.Error(errorMessage)
|
|
|
|
ctx.JSON(http.StatusConflict, map[string]interface{}{
|
|
|
|
"error": err.Error(),
|
|
|
|
"user_error": errorMessage,
|
|
|
|
})
|
|
|
|
} else if models.IsErrPullRequestHasMerged(err) {
|
|
|
|
errorMessage := ctx.Tr("repo.pulls.has_merged")
|
|
|
|
|
|
|
|
ctx.Flash.Error(errorMessage)
|
|
|
|
ctx.JSON(http.StatusConflict, map[string]interface{}{
|
|
|
|
"error": err.Error(),
|
|
|
|
"user_error": errorMessage,
|
|
|
|
})
|
|
|
|
} else if models.IsErrBranchesEqual(err) {
|
|
|
|
errorMessage := ctx.Tr("repo.pulls.nothing_to_compare")
|
|
|
|
|
|
|
|
ctx.Flash.Error(errorMessage)
|
|
|
|
ctx.JSON(http.StatusBadRequest, map[string]interface{}{
|
|
|
|
"error": err.Error(),
|
|
|
|
"user_error": errorMessage,
|
|
|
|
})
|
|
|
|
} else {
|
|
|
|
ctx.ServerError("UpdatePullRequestTarget", err)
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
2022-03-22 16:03:22 +09:00
|
|
|
notification.NotifyPullRequestChangeTargetBranch(ctx.Doer, pr, targetBranch)
|
2019-12-16 15:20:25 +09:00
|
|
|
|
|
|
|
ctx.JSON(http.StatusOK, map[string]interface{}{
|
|
|
|
"base_branch": pr.BaseBranch,
|
|
|
|
})
|
|
|
|
}
|
2022-04-29 00:45:33 +09:00
|
|
|
|
|
|
|
// SetAllowEdits allow edits from maintainers to PRs
|
|
|
|
func SetAllowEdits(ctx *context.Context) {
|
|
|
|
form := web.GetForm(ctx).(*forms.UpdateAllowEditsForm)
|
|
|
|
|
2022-06-13 18:37:59 +09:00
|
|
|
pr, err := issues_model.GetPullRequestByIndex(ctx, ctx.Repo.Repository.ID, ctx.ParamsInt64(":index"))
|
2022-04-29 00:45:33 +09:00
|
|
|
if err != nil {
|
2022-06-13 18:37:59 +09:00
|
|
|
if issues_model.IsErrPullRequestNotExist(err) {
|
2022-04-29 00:45:33 +09:00
|
|
|
ctx.NotFound("GetPullRequestByIndex", err)
|
|
|
|
} else {
|
|
|
|
ctx.ServerError("GetPullRequestByIndex", err)
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := pull_service.SetAllowEdits(ctx, ctx.Doer, pr, form.AllowMaintainerEdit); err != nil {
|
|
|
|
if errors.Is(pull_service.ErrUserHasNoPermissionForAction, err) {
|
|
|
|
ctx.Error(http.StatusForbidden)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
ctx.ServerError("SetAllowEdits", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx.JSON(http.StatusOK, map[string]interface{}{
|
|
|
|
"allow_maintainer_edit": pr.AllowMaintainerEdit,
|
|
|
|
})
|
|
|
|
}
|