2021-12-28 04:58:09 +02:00
|
|
|
package hosting_service
|
|
|
|
|
|
|
|
import (
|
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/jesseduffield/lazygit/pkg/i18n"
|
|
|
|
"github.com/jesseduffield/lazygit/pkg/test"
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
|
|
)
|
|
|
|
|
|
|
|
func TestGetPullRequestURL(t *testing.T) {
|
|
|
|
type scenario struct {
|
|
|
|
testName string
|
|
|
|
from string
|
|
|
|
to string
|
|
|
|
remoteUrl string
|
|
|
|
configServiceDomains map[string]string
|
|
|
|
test func(url string, err error)
|
|
|
|
expectedLoggedErrors []string
|
|
|
|
}
|
|
|
|
|
|
|
|
scenarios := []scenario{
|
|
|
|
{
|
|
|
|
testName: "Opens a link to new pull request on bitbucket",
|
|
|
|
from: "feature/profile-page",
|
|
|
|
remoteUrl: "git@bitbucket.org:johndoe/social_network.git",
|
|
|
|
test: func(url string, err error) {
|
|
|
|
assert.NoError(t, err)
|
2022-01-15 12:46:37 +02:00
|
|
|
assert.Equal(t, "https://bitbucket.org/johndoe/social_network/pull-requests/new?source=feature%2Fprofile-page&t=1", url)
|
2021-12-28 04:58:09 +02:00
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
testName: "Opens a link to new pull request on bitbucket with http remote url",
|
|
|
|
from: "feature/events",
|
|
|
|
remoteUrl: "https://my_username@bitbucket.org/johndoe/social_network.git",
|
|
|
|
test: func(url string, err error) {
|
|
|
|
assert.NoError(t, err)
|
2022-01-15 12:46:37 +02:00
|
|
|
assert.Equal(t, "https://bitbucket.org/johndoe/social_network/pull-requests/new?source=feature%2Fevents&t=1", url)
|
2021-12-28 04:58:09 +02:00
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
testName: "Opens a link to new pull request on github",
|
|
|
|
from: "feature/sum-operation",
|
|
|
|
remoteUrl: "git@github.com:peter/calculator.git",
|
|
|
|
test: func(url string, err error) {
|
|
|
|
assert.NoError(t, err)
|
2022-01-15 12:46:37 +02:00
|
|
|
assert.Equal(t, "https://github.com/peter/calculator/compare/feature%2Fsum-operation?expand=1", url)
|
2021-12-28 04:58:09 +02:00
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
testName: "Opens a link to new pull request on bitbucket with specific target branch",
|
|
|
|
from: "feature/profile-page/avatar",
|
|
|
|
to: "feature/profile-page",
|
|
|
|
remoteUrl: "git@bitbucket.org:johndoe/social_network.git",
|
|
|
|
test: func(url string, err error) {
|
|
|
|
assert.NoError(t, err)
|
2022-01-15 12:46:37 +02:00
|
|
|
assert.Equal(t, "https://bitbucket.org/johndoe/social_network/pull-requests/new?source=feature%2Fprofile-page%2Favatar&dest=feature%2Fprofile-page&t=1", url)
|
2021-12-28 04:58:09 +02:00
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
testName: "Opens a link to new pull request on bitbucket with http remote url with specified target branch",
|
|
|
|
from: "feature/remote-events",
|
|
|
|
to: "feature/events",
|
|
|
|
remoteUrl: "https://my_username@bitbucket.org/johndoe/social_network.git",
|
|
|
|
test: func(url string, err error) {
|
|
|
|
assert.NoError(t, err)
|
2022-01-15 12:46:37 +02:00
|
|
|
assert.Equal(t, "https://bitbucket.org/johndoe/social_network/pull-requests/new?source=feature%2Fremote-events&dest=feature%2Fevents&t=1", url)
|
2021-12-28 04:58:09 +02:00
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
testName: "Opens a link to new pull request on github with specific target branch",
|
|
|
|
from: "feature/sum-operation",
|
|
|
|
to: "feature/operations",
|
|
|
|
remoteUrl: "git@github.com:peter/calculator.git",
|
|
|
|
test: func(url string, err error) {
|
|
|
|
assert.NoError(t, err)
|
2022-01-15 12:46:37 +02:00
|
|
|
assert.Equal(t, "https://github.com/peter/calculator/compare/feature%2Foperations...feature%2Fsum-operation?expand=1", url)
|
2021-12-28 04:58:09 +02:00
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
testName: "Opens a link to new pull request on gitlab",
|
|
|
|
from: "feature/ui",
|
|
|
|
remoteUrl: "git@gitlab.com:peter/calculator.git",
|
|
|
|
test: func(url string, err error) {
|
|
|
|
assert.NoError(t, err)
|
2022-01-15 12:46:37 +02:00
|
|
|
assert.Equal(t, "https://gitlab.com/peter/calculator/merge_requests/new?merge_request[source_branch]=feature%2Fui", url)
|
2021-12-28 04:58:09 +02:00
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
testName: "Opens a link to new pull request on gitlab in nested groups",
|
|
|
|
from: "feature/ui",
|
|
|
|
remoteUrl: "git@gitlab.com:peter/public/calculator.git",
|
|
|
|
test: func(url string, err error) {
|
|
|
|
assert.NoError(t, err)
|
2022-01-15 12:46:37 +02:00
|
|
|
assert.Equal(t, "https://gitlab.com/peter/public/calculator/merge_requests/new?merge_request[source_branch]=feature%2Fui", url)
|
2021-12-28 04:58:09 +02:00
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
testName: "Opens a link to new pull request on gitlab with specific target branch",
|
|
|
|
from: "feature/commit-ui",
|
|
|
|
to: "epic/ui",
|
|
|
|
remoteUrl: "git@gitlab.com:peter/calculator.git",
|
|
|
|
test: func(url string, err error) {
|
|
|
|
assert.NoError(t, err)
|
2022-01-15 12:46:37 +02:00
|
|
|
assert.Equal(t, "https://gitlab.com/peter/calculator/merge_requests/new?merge_request[source_branch]=feature%2Fcommit-ui&merge_request[target_branch]=epic%2Fui", url)
|
2021-12-28 04:58:09 +02:00
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
testName: "Opens a link to new pull request on gitlab with specific target branch in nested groups",
|
|
|
|
from: "feature/commit-ui",
|
|
|
|
to: "epic/ui",
|
|
|
|
remoteUrl: "git@gitlab.com:peter/public/calculator.git",
|
|
|
|
test: func(url string, err error) {
|
|
|
|
assert.NoError(t, err)
|
2022-01-15 12:46:37 +02:00
|
|
|
assert.Equal(t, "https://gitlab.com/peter/public/calculator/merge_requests/new?merge_request[source_branch]=feature%2Fcommit-ui&merge_request[target_branch]=epic%2Fui", url)
|
2021-12-28 04:58:09 +02:00
|
|
|
},
|
|
|
|
},
|
2022-08-06 22:54:57 +02:00
|
|
|
{
|
|
|
|
testName: "Opens a link to new pull request on bitbucket with a custom SSH username",
|
|
|
|
from: "feature/profile-page",
|
|
|
|
remoteUrl: "john@bitbucket.org:johndoe/social_network.git",
|
|
|
|
test: func(url string, err error) {
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, "https://bitbucket.org/johndoe/social_network/pull-requests/new?source=feature%2Fprofile-page&t=1", url)
|
|
|
|
},
|
|
|
|
},
|
2021-12-30 17:04:49 +02:00
|
|
|
{
|
|
|
|
testName: "Opens a link to new pull request on Azure DevOps (SSH)",
|
|
|
|
from: "feature/new",
|
|
|
|
remoteUrl: "git@ssh.dev.azure.com:v3/myorg/myproject/myrepo",
|
|
|
|
test: func(url string, err error) {
|
|
|
|
assert.NoError(t, err)
|
2022-03-16 11:47:39 +02:00
|
|
|
assert.Equal(t, "https://dev.azure.com/myorg/myproject/_git/myrepo/pullrequestcreate?sourceRef=feature%2Fnew", url)
|
2021-12-30 17:04:49 +02:00
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
2022-04-03 22:19:15 +02:00
|
|
|
testName: "Opens a link to new pull request on Azure DevOps (SSH) with specific target",
|
2021-12-30 17:04:49 +02:00
|
|
|
from: "feature/new",
|
|
|
|
to: "dev",
|
|
|
|
remoteUrl: "git@ssh.dev.azure.com:v3/myorg/myproject/myrepo",
|
|
|
|
test: func(url string, err error) {
|
|
|
|
assert.NoError(t, err)
|
2022-03-16 11:47:39 +02:00
|
|
|
assert.Equal(t, "https://dev.azure.com/myorg/myproject/_git/myrepo/pullrequestcreate?sourceRef=feature%2Fnew&targetRef=dev", url)
|
2021-12-30 17:04:49 +02:00
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
testName: "Opens a link to new pull request on Azure DevOps (HTTP)",
|
|
|
|
from: "feature/new",
|
|
|
|
remoteUrl: "https://myorg@dev.azure.com/myorg/myproject/_git/myrepo",
|
|
|
|
test: func(url string, err error) {
|
|
|
|
assert.NoError(t, err)
|
2022-03-16 11:47:39 +02:00
|
|
|
assert.Equal(t, "https://dev.azure.com/myorg/myproject/_git/myrepo/pullrequestcreate?sourceRef=feature%2Fnew", url)
|
2021-12-30 17:04:49 +02:00
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
2022-04-03 22:19:15 +02:00
|
|
|
testName: "Opens a link to new pull request on Azure DevOps (HTTP) with specific target",
|
2021-12-30 17:04:49 +02:00
|
|
|
from: "feature/new",
|
|
|
|
to: "dev",
|
|
|
|
remoteUrl: "https://myorg@dev.azure.com/myorg/myproject/_git/myrepo",
|
|
|
|
test: func(url string, err error) {
|
|
|
|
assert.NoError(t, err)
|
2022-03-16 11:47:39 +02:00
|
|
|
assert.Equal(t, "https://dev.azure.com/myorg/myproject/_git/myrepo/pullrequestcreate?sourceRef=feature%2Fnew&targetRef=dev", url)
|
2021-12-30 17:04:49 +02:00
|
|
|
},
|
|
|
|
},
|
2022-04-10 15:26:31 +02:00
|
|
|
{
|
|
|
|
testName: "Opens a link to new pull request on Bitbucket Server (SSH)",
|
|
|
|
from: "feature/new",
|
|
|
|
remoteUrl: "ssh://git@mycompany.bitbucket.com/myproject/myrepo.git",
|
|
|
|
configServiceDomains: map[string]string{
|
|
|
|
// valid configuration for a bitbucket server URL
|
|
|
|
"mycompany.bitbucket.com": "bitbucketServer:mycompany.bitbucket.com",
|
|
|
|
},
|
|
|
|
test: func(url string, err error) {
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, "https://mycompany.bitbucket.com/projects/myproject/repos/myrepo/pull-requests?create&sourceBranch=feature%2Fnew", url)
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
testName: "Opens a link to new pull request on Bitbucket Server (SSH) with specific target",
|
|
|
|
from: "feature/new",
|
|
|
|
to: "dev",
|
|
|
|
remoteUrl: "ssh://git@mycompany.bitbucket.com/myproject/myrepo.git",
|
|
|
|
configServiceDomains: map[string]string{
|
|
|
|
// valid configuration for a bitbucket server URL
|
|
|
|
"mycompany.bitbucket.com": "bitbucketServer:mycompany.bitbucket.com",
|
|
|
|
},
|
|
|
|
test: func(url string, err error) {
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, "https://mycompany.bitbucket.com/projects/myproject/repos/myrepo/pull-requests?create&targetBranch=dev&sourceBranch=feature%2Fnew", url)
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
testName: "Opens a link to new pull request on Bitbucket Server (HTTP)",
|
|
|
|
from: "feature/new",
|
|
|
|
remoteUrl: "https://mycompany.bitbucket.com/scm/myproject/myrepo.git",
|
|
|
|
configServiceDomains: map[string]string{
|
|
|
|
// valid configuration for a bitbucket server URL
|
|
|
|
"mycompany.bitbucket.com": "bitbucketServer:mycompany.bitbucket.com",
|
|
|
|
},
|
|
|
|
test: func(url string, err error) {
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, "https://mycompany.bitbucket.com/projects/myproject/repos/myrepo/pull-requests?create&sourceBranch=feature%2Fnew", url)
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
testName: "Opens a link to new pull request on Bitbucket Server (HTTP) with specific target",
|
|
|
|
from: "feature/new",
|
|
|
|
to: "dev",
|
|
|
|
remoteUrl: "https://mycompany.bitbucket.com/scm/myproject/myrepo.git",
|
|
|
|
configServiceDomains: map[string]string{
|
|
|
|
// valid configuration for a bitbucket server URL
|
|
|
|
"mycompany.bitbucket.com": "bitbucketServer:mycompany.bitbucket.com",
|
|
|
|
},
|
|
|
|
test: func(url string, err error) {
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, "https://mycompany.bitbucket.com/projects/myproject/repos/myrepo/pull-requests?create&targetBranch=dev&sourceBranch=feature%2Fnew", url)
|
|
|
|
},
|
|
|
|
},
|
2021-12-28 04:58:09 +02:00
|
|
|
{
|
|
|
|
testName: "Throws an error if git service is unsupported",
|
|
|
|
from: "feature/divide-operation",
|
|
|
|
remoteUrl: "git@something.com:peter/calculator.git",
|
|
|
|
test: func(url string, err error) {
|
|
|
|
assert.EqualError(t, err, "Unsupported git service")
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
testName: "Does not log error when config service domains are valid",
|
|
|
|
from: "feature/profile-page",
|
|
|
|
remoteUrl: "git@bitbucket.org:johndoe/social_network.git",
|
|
|
|
configServiceDomains: map[string]string{
|
|
|
|
// valid configuration for a custom service URL
|
|
|
|
"git.work.com": "gitlab:code.work.com",
|
|
|
|
},
|
|
|
|
test: func(url string, err error) {
|
|
|
|
assert.NoError(t, err)
|
2022-01-15 12:46:37 +02:00
|
|
|
assert.Equal(t, "https://bitbucket.org/johndoe/social_network/pull-requests/new?source=feature%2Fprofile-page&t=1", url)
|
2021-12-28 04:58:09 +02:00
|
|
|
},
|
|
|
|
expectedLoggedErrors: nil,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
testName: "Logs error when config service domain is malformed",
|
|
|
|
from: "feature/profile-page",
|
|
|
|
remoteUrl: "git@bitbucket.org:johndoe/social_network.git",
|
|
|
|
configServiceDomains: map[string]string{
|
|
|
|
"noservice.work.com": "noservice.work.com",
|
|
|
|
},
|
|
|
|
test: func(url string, err error) {
|
|
|
|
assert.NoError(t, err)
|
2022-01-15 12:46:37 +02:00
|
|
|
assert.Equal(t, "https://bitbucket.org/johndoe/social_network/pull-requests/new?source=feature%2Fprofile-page&t=1", url)
|
2021-12-28 04:58:09 +02:00
|
|
|
},
|
|
|
|
expectedLoggedErrors: []string{"Unexpected format for git service: 'noservice.work.com'. Expected something like 'github.com:github.com'"},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
testName: "Logs error when config service domain uses unknown provider",
|
|
|
|
from: "feature/profile-page",
|
|
|
|
remoteUrl: "git@bitbucket.org:johndoe/social_network.git",
|
|
|
|
configServiceDomains: map[string]string{
|
|
|
|
"invalid.work.com": "noservice:invalid.work.com",
|
|
|
|
},
|
|
|
|
test: func(url string, err error) {
|
|
|
|
assert.NoError(t, err)
|
2022-01-15 12:46:37 +02:00
|
|
|
assert.Equal(t, "https://bitbucket.org/johndoe/social_network/pull-requests/new?source=feature%2Fprofile-page&t=1", url)
|
2021-12-28 04:58:09 +02:00
|
|
|
},
|
2022-04-10 15:26:31 +02:00
|
|
|
expectedLoggedErrors: []string{"Unknown git service type: 'noservice'. Expected one of github, bitbucket, gitlab, azuredevops, bitbucketServer"},
|
2021-12-28 04:58:09 +02:00
|
|
|
},
|
2022-01-17 13:17:59 +02:00
|
|
|
{
|
2022-01-18 15:05:48 +02:00
|
|
|
testName: "Escapes reserved URL characters in from branch name",
|
|
|
|
from: "feature/someIssue#123",
|
|
|
|
to: "master",
|
|
|
|
remoteUrl: "git@gitlab.com:me/public/repo-with-issues.git",
|
2022-01-17 13:17:59 +02:00
|
|
|
test: func(url string, err error) {
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, "https://gitlab.com/me/public/repo-with-issues/merge_requests/new?merge_request[source_branch]=feature%2FsomeIssue%23123&merge_request[target_branch]=master", url)
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
2022-01-18 15:05:48 +02:00
|
|
|
testName: "Escapes reserved URL characters in to branch name",
|
|
|
|
from: "yolo",
|
|
|
|
to: "archive/never-ending-feature#666",
|
|
|
|
remoteUrl: "git@gitlab.com:me/public/repo-with-issues.git",
|
2022-01-17 13:17:59 +02:00
|
|
|
test: func(url string, err error) {
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, "https://gitlab.com/me/public/repo-with-issues/merge_requests/new?merge_request[source_branch]=yolo&merge_request[target_branch]=archive%2Fnever-ending-feature%23666", url)
|
|
|
|
},
|
|
|
|
},
|
2021-12-28 04:58:09 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
for _, s := range scenarios {
|
2022-01-08 06:46:35 +02:00
|
|
|
s := s
|
2021-12-28 04:58:09 +02:00
|
|
|
t.Run(s.testName, func(t *testing.T) {
|
|
|
|
tr := i18n.EnglishTranslationSet()
|
|
|
|
log := &test.FakeFieldLogger{}
|
|
|
|
hostingServiceMgr := NewHostingServiceMgr(log, &tr, s.remoteUrl, s.configServiceDomains)
|
|
|
|
s.test(hostingServiceMgr.GetPullRequestURL(s.from, s.to))
|
|
|
|
log.AssertErrors(t, s.expectedLoggedErrors)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|