2018-10-12 14:06:03 +02:00
|
|
|
package commands
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"strings"
|
2019-02-11 12:30:27 +02:00
|
|
|
|
|
|
|
"github.com/go-errors/errors"
|
2019-12-06 14:38:18 +02:00
|
|
|
"github.com/jesseduffield/lazygit/pkg/config"
|
2018-10-12 14:06:03 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
// Service is a service that repository is on (Github, Bitbucket, ...)
|
|
|
|
type Service struct {
|
2021-06-26 11:49:49 +02:00
|
|
|
Name string
|
|
|
|
pullRequestURLIntoDefaultBranch func(owner string, repository string, from string) string
|
2021-07-27 12:02:52 +02:00
|
|
|
pullRequestURLIntoTargetBranch func(owner string, repository string, from string, to string) string
|
2018-10-12 14:06:03 +02:00
|
|
|
}
|
|
|
|
|
2019-12-06 16:39:24 +02:00
|
|
|
// NewService builds a Service based on the host type
|
|
|
|
func NewService(typeName string, repositoryDomain string, siteDomain string) *Service {
|
2020-02-25 21:18:22 +02:00
|
|
|
var service *Service
|
|
|
|
|
2019-12-06 16:39:24 +02:00
|
|
|
switch typeName {
|
|
|
|
case "github":
|
2020-02-25 21:18:22 +02:00
|
|
|
service = &Service{
|
2021-04-21 11:56:14 +02:00
|
|
|
Name: repositoryDomain,
|
2021-06-26 11:49:49 +02:00
|
|
|
pullRequestURLIntoDefaultBranch: func(owner string, repository string, from string) string {
|
|
|
|
return fmt.Sprintf("https://%s/%s/%s/compare/%s?expand=1", siteDomain, owner, repository, from)
|
|
|
|
},
|
|
|
|
pullRequestURLIntoTargetBranch: func(owner string, repository string, from string, to string) string {
|
|
|
|
return fmt.Sprintf("https://%s/%s/%s/compare/%s...%s?expand=1", siteDomain, owner, repository, to, from)
|
2021-04-21 11:56:14 +02:00
|
|
|
},
|
2019-12-06 16:39:24 +02:00
|
|
|
}
|
|
|
|
case "bitbucket":
|
2020-02-25 21:18:22 +02:00
|
|
|
service = &Service{
|
2021-04-21 13:23:36 +02:00
|
|
|
Name: repositoryDomain,
|
2021-06-26 11:49:49 +02:00
|
|
|
pullRequestURLIntoDefaultBranch: func(owner string, repository string, from string) string {
|
|
|
|
return fmt.Sprintf("https://%s/%s/%s/pull-requests/new?source=%s&t=1", siteDomain, owner, repository, from)
|
|
|
|
},
|
|
|
|
pullRequestURLIntoTargetBranch: func(owner string, repository string, from string, to string) string {
|
|
|
|
return fmt.Sprintf("https://%s/%s/%s/pull-requests/new?source=%s&dest=%s&t=1", siteDomain, owner, repository, from, to)
|
2021-04-21 11:56:14 +02:00
|
|
|
},
|
2019-12-06 16:39:24 +02:00
|
|
|
}
|
|
|
|
case "gitlab":
|
2020-02-25 21:18:22 +02:00
|
|
|
service = &Service{
|
2021-04-21 13:23:36 +02:00
|
|
|
Name: repositoryDomain,
|
2021-06-26 11:49:49 +02:00
|
|
|
pullRequestURLIntoDefaultBranch: func(owner string, repository string, from string) string {
|
|
|
|
return fmt.Sprintf("https://%s/%s/%s/merge_requests/new?merge_request[source_branch]=%s", siteDomain, owner, repository, from)
|
|
|
|
},
|
|
|
|
pullRequestURLIntoTargetBranch: func(owner string, repository string, from string, to string) string {
|
|
|
|
return fmt.Sprintf("https://%s/%s/%s/merge_requests/new?merge_request[source_branch]=%s&merge_request[target_branch]=%s", siteDomain, owner, repository, from, to)
|
2021-04-21 11:56:14 +02:00
|
|
|
},
|
2019-12-06 16:39:24 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-25 21:18:22 +02:00
|
|
|
return service
|
2019-12-06 16:39:24 +02:00
|
|
|
}
|
|
|
|
|
2021-06-26 11:49:49 +02:00
|
|
|
func (s *Service) PullRequestURL(owner string, repository string, from string, to string) string {
|
|
|
|
if to == "" {
|
|
|
|
return s.pullRequestURLIntoDefaultBranch(owner, repository, from)
|
|
|
|
} else {
|
|
|
|
return s.pullRequestURLIntoTargetBranch(owner, repository, from, to)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-07-27 12:02:52 +02:00
|
|
|
// PullRequest opens a link in browser to create new pull request
|
|
|
|
// with selected branch
|
|
|
|
type PullRequest struct {
|
|
|
|
GitServices []*Service
|
|
|
|
GitCommand *GitCommand
|
|
|
|
}
|
|
|
|
|
|
|
|
// RepoInformation holds some basic information about the repo
|
|
|
|
type RepoInformation struct {
|
|
|
|
Owner string
|
|
|
|
Repository string
|
|
|
|
}
|
|
|
|
|
2019-12-06 14:38:18 +02:00
|
|
|
func getServices(config config.AppConfigurer) []*Service {
|
|
|
|
services := []*Service{
|
2019-12-06 16:39:24 +02:00
|
|
|
NewService("github", "github.com", "github.com"),
|
|
|
|
NewService("bitbucket", "bitbucket.org", "bitbucket.org"),
|
|
|
|
NewService("gitlab", "gitlab.com", "gitlab.com"),
|
2018-10-12 14:06:03 +02:00
|
|
|
}
|
2019-12-06 14:38:18 +02:00
|
|
|
|
2020-10-03 06:54:55 +02:00
|
|
|
configServices := config.GetUserConfig().Services
|
2019-12-06 14:38:18 +02:00
|
|
|
|
2019-12-06 16:39:24 +02:00
|
|
|
for repoDomain, typeAndDomain := range configServices {
|
|
|
|
splitData := strings.Split(typeAndDomain, ":")
|
2020-02-25 21:18:22 +02:00
|
|
|
if len(splitData) != 2 {
|
|
|
|
// TODO log this misconfiguration
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
service := NewService(splitData[0], repoDomain, splitData[1])
|
|
|
|
if service == nil {
|
|
|
|
// TODO log this unsupported service
|
|
|
|
continue
|
2019-12-06 16:39:24 +02:00
|
|
|
}
|
2020-02-25 21:18:22 +02:00
|
|
|
|
|
|
|
services = append(services, service)
|
2019-12-06 14:38:18 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return services
|
2018-10-12 14:06:03 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewPullRequest creates new instance of PullRequest
|
2018-10-15 11:00:19 +02:00
|
|
|
func NewPullRequest(gitCommand *GitCommand) *PullRequest {
|
2018-10-12 14:06:03 +02:00
|
|
|
return &PullRequest{
|
2019-12-06 14:38:18 +02:00
|
|
|
GitServices: getServices(gitCommand.Config),
|
2018-10-12 14:06:03 +02:00
|
|
|
GitCommand: gitCommand,
|
2018-10-15 11:00:19 +02:00
|
|
|
}
|
2018-10-12 14:06:03 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Create opens link to new pull request in browser
|
2021-07-27 12:02:52 +02:00
|
|
|
func (pr *PullRequest) Create(from string, to string) (string, error) {
|
2021-04-21 13:23:36 +02:00
|
|
|
pullRequestURL, err := pr.getPullRequestURL(from, to)
|
2020-11-10 21:57:50 +02:00
|
|
|
if err != nil {
|
2021-04-10 09:31:23 +02:00
|
|
|
return "", err
|
2020-11-10 21:57:50 +02:00
|
|
|
}
|
|
|
|
|
2021-04-10 09:31:23 +02:00
|
|
|
return pullRequestURL, pr.GitCommand.OSCommand.OpenLink(pullRequestURL)
|
2020-11-10 21:57:50 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// CopyURL copies the pull request URL to the clipboard
|
2021-07-27 12:02:52 +02:00
|
|
|
func (pr *PullRequest) CopyURL(from string, to string) (string, error) {
|
2021-04-21 13:23:36 +02:00
|
|
|
pullRequestURL, err := pr.getPullRequestURL(from, to)
|
2020-11-10 21:57:50 +02:00
|
|
|
if err != nil {
|
2021-04-10 09:31:23 +02:00
|
|
|
return "", err
|
2020-11-10 21:57:50 +02:00
|
|
|
}
|
|
|
|
|
2021-04-10 09:31:23 +02:00
|
|
|
return pullRequestURL, pr.GitCommand.OSCommand.CopyToClipboard(pullRequestURL)
|
2020-11-10 21:57:50 +02:00
|
|
|
}
|
|
|
|
|
2021-07-27 12:02:52 +02:00
|
|
|
func (pr *PullRequest) getPullRequestURL(from string, to string) (string, error) {
|
2021-04-21 13:23:36 +02:00
|
|
|
branchExistsOnRemote := pr.GitCommand.CheckRemoteBranchExists(from)
|
2018-10-17 14:20:15 +02:00
|
|
|
|
|
|
|
if !branchExistsOnRemote {
|
2020-11-10 21:57:50 +02:00
|
|
|
return "", errors.New(pr.GitCommand.Tr.NoBranchOnRemote)
|
2018-10-17 14:20:15 +02:00
|
|
|
}
|
|
|
|
|
2018-10-12 14:06:03 +02:00
|
|
|
repoURL := pr.GitCommand.GetRemoteURL()
|
|
|
|
var gitService *Service
|
|
|
|
|
|
|
|
for _, service := range pr.GitServices {
|
|
|
|
if strings.Contains(repoURL, service.Name) {
|
|
|
|
gitService = service
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if gitService == nil {
|
2020-11-10 21:57:50 +02:00
|
|
|
return "", errors.New(pr.GitCommand.Tr.UnsupportedGitService)
|
2018-10-12 14:06:03 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
repoInfo := getRepoInfoFromURL(repoURL)
|
2021-07-27 12:02:52 +02:00
|
|
|
|
|
|
|
pullRequestURL := gitService.PullRequestURL(repoInfo.Owner, repoInfo.Repository, from, to)
|
2020-11-10 21:57:50 +02:00
|
|
|
|
|
|
|
return pullRequestURL, nil
|
2018-10-12 14:06:03 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func getRepoInfoFromURL(url string) *RepoInformation {
|
|
|
|
isHTTP := strings.HasPrefix(url, "http")
|
|
|
|
|
|
|
|
if isHTTP {
|
|
|
|
splits := strings.Split(url, "/")
|
2019-12-06 16:40:02 +02:00
|
|
|
owner := strings.Join(splits[3:len(splits)-1], "/")
|
2018-10-17 14:20:15 +02:00
|
|
|
repo := strings.TrimSuffix(splits[len(splits)-1], ".git")
|
2018-10-12 14:06:03 +02:00
|
|
|
|
|
|
|
return &RepoInformation{
|
|
|
|
Owner: owner,
|
|
|
|
Repository: repo,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
tmpSplit := strings.Split(url, ":")
|
|
|
|
splits := strings.Split(tmpSplit[1], "/")
|
2019-12-06 16:40:02 +02:00
|
|
|
owner := strings.Join(splits[0:len(splits)-1], "/")
|
|
|
|
repo := strings.TrimSuffix(splits[len(splits)-1], ".git")
|
2018-10-12 14:06:03 +02:00
|
|
|
|
|
|
|
return &RepoInformation{
|
|
|
|
Owner: owner,
|
|
|
|
Repository: repo,
|
|
|
|
}
|
|
|
|
}
|