2020-12-08 16:41:50 -05:00
|
|
|
// Copyright 2020 The Gitea Authors. All rights reserved.
|
|
|
|
// Use of this source code is governed by a MIT-style
|
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
|
|
|
package task
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2023-04-29 12:16:18 -04:00
|
|
|
"regexp"
|
2020-12-08 16:41:50 -05:00
|
|
|
"strings"
|
|
|
|
|
|
|
|
"code.gitea.io/sdk/gitea"
|
|
|
|
"code.gitea.io/tea/modules/config"
|
2021-09-22 11:48:21 -04:00
|
|
|
"code.gitea.io/tea/modules/context"
|
2020-12-08 16:41:50 -05:00
|
|
|
local_git "code.gitea.io/tea/modules/git"
|
|
|
|
"code.gitea.io/tea/modules/print"
|
|
|
|
"code.gitea.io/tea/modules/utils"
|
|
|
|
)
|
|
|
|
|
2023-04-29 12:16:18 -04:00
|
|
|
var (
|
|
|
|
spaceRegex = regexp.MustCompile(`[\s_-]+`)
|
|
|
|
noSpace = regexp.MustCompile(`^[^a-zA-Z\s]*`)
|
|
|
|
consecutive = regexp.MustCompile(`[\s]{2,}`)
|
|
|
|
)
|
|
|
|
|
2020-12-08 16:41:50 -05:00
|
|
|
// CreatePull creates a PR in the given repo and prints the result
|
2022-09-27 11:36:36 -04:00
|
|
|
func CreatePull(ctx *context.TeaContext, base, head string, allowMaintainerEdits bool, opts *gitea.CreateIssueOption) (err error) {
|
2020-12-08 16:41:50 -05:00
|
|
|
// default is default branch
|
|
|
|
if len(base) == 0 {
|
2021-09-22 11:48:21 -04:00
|
|
|
base, err = GetDefaultPRBase(ctx.Login, ctx.Owner, ctx.Repo)
|
2020-12-08 16:41:50 -05:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// default is current one
|
|
|
|
if len(head) == 0 {
|
2021-09-22 11:48:21 -04:00
|
|
|
if ctx.LocalRepo == nil {
|
|
|
|
return fmt.Errorf("no local git repo detected, please specify head branch")
|
|
|
|
}
|
|
|
|
headOwner, headBranch, err := GetDefaultPRHead(ctx.LocalRepo)
|
2020-12-08 16:41:50 -05:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2021-09-22 11:48:21 -04:00
|
|
|
head = GetHeadSpec(headOwner, headBranch, ctx.Owner)
|
2020-12-08 16:41:50 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// head & base may not be the same
|
|
|
|
if head == base {
|
|
|
|
return fmt.Errorf("can't create PR from %s to %s", head, base)
|
|
|
|
}
|
|
|
|
|
|
|
|
// default is head branch name
|
2021-03-08 06:48:03 -05:00
|
|
|
if len(opts.Title) == 0 {
|
|
|
|
opts.Title = GetDefaultPRTitle(head)
|
2020-12-08 16:41:50 -05:00
|
|
|
}
|
|
|
|
// title is required
|
2021-03-08 06:48:03 -05:00
|
|
|
if len(opts.Title) == 0 {
|
2021-09-22 11:48:21 -04:00
|
|
|
return fmt.Errorf("title is required")
|
2020-12-08 16:41:50 -05:00
|
|
|
}
|
|
|
|
|
2022-09-27 11:36:36 -04:00
|
|
|
client := ctx.Login.Client()
|
|
|
|
|
|
|
|
pr, _, err := client.CreatePullRequest(ctx.Owner, ctx.Repo, gitea.CreatePullRequestOption{
|
2021-03-08 06:48:03 -05:00
|
|
|
Head: head,
|
|
|
|
Base: base,
|
|
|
|
Title: opts.Title,
|
|
|
|
Body: opts.Body,
|
|
|
|
Assignees: opts.Assignees,
|
|
|
|
Labels: opts.Labels,
|
|
|
|
Milestone: opts.Milestone,
|
|
|
|
Deadline: opts.Deadline,
|
2020-12-08 16:41:50 -05:00
|
|
|
})
|
|
|
|
if err != nil {
|
2021-09-22 11:48:21 -04:00
|
|
|
return fmt.Errorf("could not create PR from %s to %s:%s: %s", head, ctx.Owner, base, err)
|
2020-12-08 16:41:50 -05:00
|
|
|
}
|
|
|
|
|
2022-09-27 11:36:36 -04:00
|
|
|
if pr.AllowMaintainerEdit != allowMaintainerEdits {
|
|
|
|
pr, _, err = client.EditPullRequest(ctx.Owner, ctx.Repo, pr.Index, gitea.EditPullRequestOption{
|
|
|
|
AllowMaintainerEdit: gitea.OptionalBool(allowMaintainerEdits),
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("could not enable maintainer edit on pull: %v", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-12-16 12:16:50 -05:00
|
|
|
print.PullDetails(pr, nil, nil)
|
2020-12-08 16:41:50 -05:00
|
|
|
|
|
|
|
fmt.Println(pr.HTMLURL)
|
|
|
|
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetDefaultPRBase retrieves the default base branch for the given repo
|
|
|
|
func GetDefaultPRBase(login *config.Login, owner, repo string) (string, error) {
|
|
|
|
meta, _, err := login.Client().GetRepo(owner, repo)
|
|
|
|
if err != nil {
|
|
|
|
return "", fmt.Errorf("could not fetch repo meta: %s", err)
|
|
|
|
}
|
|
|
|
return meta.DefaultBranch, nil
|
|
|
|
}
|
|
|
|
|
2021-03-05 05:27:09 -05:00
|
|
|
// GetDefaultPRHead uses the currently checked out branch, tries to find a remote
|
|
|
|
// that has a branch with the same name, and extracts the owner from its URL.
|
|
|
|
// If no remote matches, owner is empty, meaning same as head repo owner.
|
2020-12-08 16:41:50 -05:00
|
|
|
func GetDefaultPRHead(localRepo *local_git.TeaRepo) (owner, branch string, err error) {
|
2022-10-24 18:38:39 -04:00
|
|
|
var sha string
|
|
|
|
if branch, sha, err = localRepo.TeaGetCurrentBranchNameAndSHA(); err != nil {
|
2020-12-08 16:41:50 -05:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2022-10-24 18:38:39 -04:00
|
|
|
remote, err := localRepo.TeaFindBranchRemote(branch, sha)
|
2020-12-08 16:41:50 -05:00
|
|
|
if err != nil {
|
|
|
|
err = fmt.Errorf("could not determine remote for current branch: %s", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if remote == nil {
|
2021-03-05 05:27:09 -05:00
|
|
|
// if no remote branch is found for the local branch,
|
|
|
|
// we leave owner empty, meaning "use same repo as head" to gitea.
|
2020-12-08 16:41:50 -05:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
url, err := local_git.ParseURL(remote.Config().URLs[0])
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
2021-10-18 08:09:27 -04:00
|
|
|
owner, _ = utils.GetOwnerAndRepo(url.Path, "")
|
2020-12-08 16:41:50 -05:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetHeadSpec creates a head string as expected by gitea API
|
|
|
|
func GetHeadSpec(owner, branch, baseOwner string) string {
|
|
|
|
if len(owner) != 0 && owner != baseOwner {
|
|
|
|
return fmt.Sprintf("%s:%s", owner, branch)
|
|
|
|
}
|
|
|
|
return branch
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetDefaultPRTitle transforms a string like a branchname to a readable text
|
2023-04-29 12:16:18 -04:00
|
|
|
func GetDefaultPRTitle(header string) string {
|
|
|
|
// Extract the part after the last colon in the input string
|
|
|
|
colonIndex := strings.LastIndex(header, ":")
|
|
|
|
if colonIndex != -1 {
|
|
|
|
header = header[colonIndex+1:]
|
2020-12-08 16:41:50 -05:00
|
|
|
}
|
2023-04-29 12:16:18 -04:00
|
|
|
|
|
|
|
title := noSpace.ReplaceAllString(header, "")
|
|
|
|
title = spaceRegex.ReplaceAllString(title, " ")
|
|
|
|
title = strings.TrimSpace(title)
|
2020-12-08 16:41:50 -05:00
|
|
|
title = strings.Title(strings.ToLower(title))
|
2023-04-29 12:16:18 -04:00
|
|
|
title = consecutive.ReplaceAllString(title, " ")
|
|
|
|
|
2020-12-08 16:41:50 -05:00
|
|
|
return title
|
|
|
|
}
|