mirror of
https://github.com/go-gitea/gitea
synced 2025-01-25 16:57:52 +01:00
115 lines
3.4 KiB
Go
115 lines
3.4 KiB
Go
// Copyright 2024 The Gitea Authors. All rights reserved.
|
|
// SPDX-License-Identifier: MIT
|
|
|
|
package repo
|
|
|
|
import (
|
|
"net/http"
|
|
|
|
git_model "code.gitea.io/gitea/models/git"
|
|
issues_model "code.gitea.io/gitea/models/issues"
|
|
access_model "code.gitea.io/gitea/models/perm/access"
|
|
repo_model "code.gitea.io/gitea/models/repo"
|
|
unit_model "code.gitea.io/gitea/models/unit"
|
|
"code.gitea.io/gitea/modules/git"
|
|
"code.gitea.io/gitea/modules/gitrepo"
|
|
"code.gitea.io/gitea/modules/web"
|
|
"code.gitea.io/gitea/routers/utils"
|
|
"code.gitea.io/gitea/services/context"
|
|
"code.gitea.io/gitea/services/forms"
|
|
repo_service "code.gitea.io/gitea/services/repository"
|
|
)
|
|
|
|
func CreateBranchFromIssue(ctx *context.Context) {
|
|
if ctx.HasError() { // form binding error check
|
|
ctx.JSONError(ctx.GetErrMsg())
|
|
return
|
|
}
|
|
|
|
issue := GetActionIssue(ctx)
|
|
if ctx.Written() {
|
|
return
|
|
}
|
|
|
|
if issue.IsPull {
|
|
ctx.Flash.Error(ctx.Tr("repo.issues.create_branch_from_issue_error_is_pull"))
|
|
ctx.JSONRedirect(issue.Link())
|
|
return
|
|
}
|
|
|
|
form := web.GetForm(ctx).(*forms.NewBranchForm)
|
|
repo := ctx.Repo.Repository
|
|
gitRepo := ctx.Repo.GitRepo
|
|
// if create branch in a forked repository
|
|
if form.RepoID > 0 && form.RepoID != repo.ID {
|
|
var err error
|
|
repo, err = repo_model.GetRepositoryByID(ctx, form.RepoID)
|
|
if err != nil {
|
|
ctx.ServerError("GetRepositoryByID", err)
|
|
return
|
|
}
|
|
gitRepo, err = gitrepo.OpenRepository(ctx, repo)
|
|
if err != nil {
|
|
ctx.ServerError("OpenRepository", err)
|
|
return
|
|
}
|
|
defer gitRepo.Close()
|
|
}
|
|
|
|
perm, err := access_model.GetUserRepoPermission(ctx, repo, ctx.Doer)
|
|
if err != nil {
|
|
ctx.ServerError("GetUserRepoPermission", err)
|
|
return
|
|
}
|
|
|
|
canCreateBranch := perm.CanWrite(unit_model.TypeCode) && repo.CanCreateBranch()
|
|
if !canCreateBranch {
|
|
ctx.Error(http.StatusForbidden, "No permission to create branch in this repository")
|
|
return
|
|
}
|
|
|
|
if err := repo_service.CreateNewBranch(ctx, ctx.Doer, repo, gitRepo, form.SourceBranchName, form.NewBranchName); err != nil {
|
|
switch {
|
|
case git_model.IsErrBranchAlreadyExists(err) || git.IsErrPushOutOfDate(err):
|
|
ctx.JSONError(ctx.Tr("repo.branch.branch_already_exists", form.NewBranchName))
|
|
case git_model.IsErrBranchNameConflict(err):
|
|
e := err.(git_model.ErrBranchNameConflict)
|
|
ctx.JSONError(ctx.Tr("repo.branch.branch_name_conflict", form.NewBranchName, e.BranchName))
|
|
case git_model.IsErrBranchNotExist(err):
|
|
ctx.JSONError(ctx.Tr("repo.branch.branch_not_exist", form.SourceBranchName))
|
|
case git.IsErrPushRejected(err):
|
|
e := err.(*git.ErrPushRejected)
|
|
if len(e.Message) == 0 {
|
|
ctx.Flash.Error(ctx.Tr("repo.editor.push_rejected_no_message"))
|
|
} else {
|
|
flashError, err := ctx.RenderToHTML(tplAlertDetails, map[string]any{
|
|
"Message": ctx.Tr("repo.editor.push_rejected"),
|
|
"Summary": ctx.Tr("repo.editor.push_rejected_summary"),
|
|
"Details": utils.SanitizeFlashErrorString(e.Message),
|
|
})
|
|
if err != nil {
|
|
ctx.ServerError("UpdatePullRequest.HTMLString", err)
|
|
return
|
|
}
|
|
ctx.JSONError(flashError)
|
|
}
|
|
default:
|
|
ctx.ServerError("CreateNewBranch", err)
|
|
}
|
|
return
|
|
}
|
|
|
|
if err := issues_model.CreateIssueDevLink(ctx, &issues_model.IssueDevLink{
|
|
IssueID: issue.ID,
|
|
LinkType: issues_model.IssueDevLinkTypeBranch,
|
|
LinkedRepoID: repo.ID,
|
|
LinkIndex: form.NewBranchName,
|
|
}); err != nil {
|
|
ctx.ServerError("CreateIssueDevLink", err)
|
|
return
|
|
}
|
|
|
|
ctx.Flash.Success(ctx.Tr("repo.issues.create_branch_from_issue_success", form.NewBranchName))
|
|
ctx.JSONRedirect(issue.Link())
|
|
}
|