/commands/merge.go
http://github.com/defunkt/hub · Go · 100 lines · 82 code · 17 blank · 1 comment · 20 complexity · 4c5266102e796e335f59637880c23e3c MD5 · raw file
- package commands
- import (
- "fmt"
- "regexp"
- "github.com/github/hub/v2/github"
- "github.com/github/hub/v2/utils"
- )
- var cmdMerge = &Command{
- Run: merge,
- GitExtension: true,
- Usage: "merge <PULLREQ-URL>",
- Long: `Merge a pull request locally with a message like the GitHub Merge Button.
- This creates a local merge commit in the current branch, but does not actually
- change the state of the pull request. However, the pull request will get
- auto-closed and marked as "merged" as soon as the newly created merge commit is
- pushed to the default branch of the remote repository.
- ## Examples:
- $ hub merge https://github.com/jingweno/gh/pull/73
- > git fetch origin refs/pull/73/head
- > git merge FETCH_HEAD --no-ff -m "Merge pull request #73 from jingweno/feature..."
- ## See also:
- hub-checkout(1), hub(1), git-merge(1)
- `,
- }
- func init() {
- CmdRunner.Use(cmdMerge)
- }
- func merge(command *Command, args *Args) {
- if !args.IsParamsEmpty() {
- err := transformMergeArgs(args)
- utils.Check(err)
- }
- }
- func transformMergeArgs(args *Args) error {
- words := args.Words()
- if len(words) == 0 {
- return nil
- }
- mergeURL := words[0]
- url, err := github.ParseURL(mergeURL)
- if err != nil {
- return nil
- }
- pullURLRegex := regexp.MustCompile("^pull/(\\d+)")
- projectPath := url.ProjectPath()
- if !pullURLRegex.MatchString(projectPath) {
- return nil
- }
- id := pullURLRegex.FindStringSubmatch(projectPath)[1]
- gh := github.NewClient(url.Project.Host)
- pullRequest, err := gh.PullRequest(url.Project, id)
- if err != nil {
- return err
- }
- repo, err := github.LocalRepo()
- if err != nil {
- return err
- }
- remote, err := repo.RemoteForRepo(pullRequest.Base.Repo)
- if err != nil {
- return err
- }
- branch := pullRequest.Head.Ref
- headRepo := pullRequest.Head.Repo
- if headRepo == nil {
- return fmt.Errorf("Error: that fork is not available anymore")
- }
- args.Before("git", "fetch", remote.Name, fmt.Sprintf("refs/pull/%s/head", id))
- // Remove pull request URL
- idx := args.IndexOfParam(mergeURL)
- args.RemoveParam(idx)
- mergeMsg := fmt.Sprintf("Merge pull request #%s from %s/%s\n\n%s", id, headRepo.Owner.Login, branch, pullRequest.Title)
- args.AppendParams("FETCH_HEAD", "-m", mergeMsg)
- if args.IndexOfParam("--ff-only") == -1 && args.IndexOfParam("--squash") == -1 && args.IndexOfParam("--ff") == -1 {
- i := args.IndexOfParam("-m")
- args.InsertParam(i, "--no-ff")
- }
- return nil
- }