2019-07-23 20:16:52 +03:00
|
|
|
package gitlab
|
2019-08-13 21:51:39 +03:00
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
"math/rand"
|
|
|
|
"os"
|
|
|
|
"strconv"
|
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/xanzy/go-gitlab"
|
|
|
|
|
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
|
|
|
|
"github.com/MichaelMure/git-bug/bridge/core"
|
2019-12-08 23:15:06 +03:00
|
|
|
"github.com/MichaelMure/git-bug/bridge/core/auth"
|
2019-08-13 21:51:39 +03:00
|
|
|
"github.com/MichaelMure/git-bug/bug"
|
|
|
|
"github.com/MichaelMure/git-bug/cache"
|
|
|
|
"github.com/MichaelMure/git-bug/repository"
|
|
|
|
"github.com/MichaelMure/git-bug/util/interrupt"
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
testRepoBaseName = "git-bug-test-gitlab-exporter"
|
|
|
|
)
|
|
|
|
|
|
|
|
type testCase struct {
|
2019-08-19 15:09:31 +03:00
|
|
|
name string
|
|
|
|
bug *cache.BugCache
|
|
|
|
numOp int // number of original operations
|
|
|
|
numOpExp int // number of operations after export
|
|
|
|
numOpImp int // number of operations after import
|
2019-08-13 21:51:39 +03:00
|
|
|
}
|
|
|
|
|
2019-12-08 23:15:06 +03:00
|
|
|
func testCases(t *testing.T, repo *cache.RepoCache) []*testCase {
|
2019-08-13 21:51:39 +03:00
|
|
|
// simple bug
|
|
|
|
simpleBug, _, err := repo.NewBug("simple bug", "new bug")
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
// bug with comments
|
|
|
|
bugWithComments, _, err := repo.NewBug("bug with comments", "new bug")
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
_, err = bugWithComments.AddComment("new comment")
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
// bug with label changes
|
|
|
|
bugLabelChange, _, err := repo.NewBug("bug label change", "new bug")
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
_, _, err = bugLabelChange.ChangeLabels([]string{"bug"}, nil)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
_, _, err = bugLabelChange.ChangeLabels([]string{"core"}, nil)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
_, _, err = bugLabelChange.ChangeLabels(nil, []string{"bug"})
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
// bug with comments editions
|
|
|
|
bugWithCommentEditions, createOp, err := repo.NewBug("bug with comments editions", "new bug")
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
_, err = bugWithCommentEditions.EditComment(createOp.Id(), "first comment edited")
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
commentOp, err := bugWithCommentEditions.AddComment("first comment")
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
_, err = bugWithCommentEditions.EditComment(commentOp.Id(), "first comment edited")
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
// bug status changed
|
|
|
|
bugStatusChanged, _, err := repo.NewBug("bug status changed", "new bug")
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
_, err = bugStatusChanged.Close()
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
_, err = bugStatusChanged.Open()
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
// bug title changed
|
|
|
|
bugTitleEdited, _, err := repo.NewBug("bug title edited", "new bug")
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
_, err = bugTitleEdited.SetTitle("bug title edited again")
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
return []*testCase{
|
|
|
|
&testCase{
|
2019-08-19 15:09:31 +03:00
|
|
|
name: "simple bug",
|
|
|
|
bug: simpleBug,
|
|
|
|
numOp: 1,
|
|
|
|
numOpExp: 2,
|
|
|
|
numOpImp: 1,
|
2019-08-13 21:51:39 +03:00
|
|
|
},
|
|
|
|
&testCase{
|
2019-08-19 15:09:31 +03:00
|
|
|
name: "bug with comments",
|
|
|
|
bug: bugWithComments,
|
|
|
|
numOp: 2,
|
|
|
|
numOpExp: 4,
|
|
|
|
numOpImp: 2,
|
2019-08-13 21:51:39 +03:00
|
|
|
},
|
|
|
|
&testCase{
|
2019-08-19 15:09:31 +03:00
|
|
|
name: "bug label change",
|
|
|
|
bug: bugLabelChange,
|
|
|
|
numOp: 4,
|
|
|
|
numOpExp: 8,
|
|
|
|
numOpImp: 4,
|
2019-08-13 21:51:39 +03:00
|
|
|
},
|
|
|
|
&testCase{
|
2019-08-19 15:09:31 +03:00
|
|
|
name: "bug with comment editions",
|
|
|
|
bug: bugWithCommentEditions,
|
|
|
|
numOp: 4,
|
|
|
|
numOpExp: 8,
|
|
|
|
numOpImp: 2,
|
2019-08-13 21:51:39 +03:00
|
|
|
},
|
|
|
|
&testCase{
|
2019-08-19 15:09:31 +03:00
|
|
|
name: "bug changed status",
|
|
|
|
bug: bugStatusChanged,
|
|
|
|
numOp: 3,
|
|
|
|
numOpExp: 6,
|
|
|
|
numOpImp: 3,
|
2019-08-13 21:51:39 +03:00
|
|
|
},
|
|
|
|
&testCase{
|
2019-08-19 15:09:31 +03:00
|
|
|
name: "bug title edited",
|
|
|
|
bug: bugTitleEdited,
|
|
|
|
numOp: 2,
|
|
|
|
numOpExp: 4,
|
|
|
|
numOpImp: 2,
|
2019-08-13 21:51:39 +03:00
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-09 01:56:32 +03:00
|
|
|
func TestGitlabPushPull(t *testing.T) {
|
2019-08-13 21:51:39 +03:00
|
|
|
// token must have 'repo' and 'delete_repo' scopes
|
2019-12-08 23:15:06 +03:00
|
|
|
envToken := os.Getenv("GITLAB_API_TOKEN")
|
|
|
|
if envToken == "" {
|
2019-08-13 21:51:39 +03:00
|
|
|
t.Skip("Env var GITLAB_API_TOKEN missing")
|
|
|
|
}
|
|
|
|
|
|
|
|
// create repo backend
|
2020-10-04 20:56:16 +03:00
|
|
|
repo := repository.CreateGoGitTestRepo(false)
|
2020-06-23 19:02:54 +03:00
|
|
|
defer repository.CleanupTestRepos(repo)
|
2019-08-13 21:51:39 +03:00
|
|
|
|
|
|
|
backend, err := cache.NewRepoCache(repo)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
// set author identity
|
2020-02-04 02:25:27 +03:00
|
|
|
login := "test-identity"
|
2019-08-13 21:51:39 +03:00
|
|
|
author, err := backend.NewIdentity("test identity", "test@test.org")
|
|
|
|
require.NoError(t, err)
|
2020-02-04 02:25:27 +03:00
|
|
|
author.SetMetadata(metaKeyGitlabLogin, login)
|
2020-02-09 00:05:13 +03:00
|
|
|
err = author.Commit()
|
|
|
|
require.NoError(t, err)
|
2019-08-13 21:51:39 +03:00
|
|
|
|
|
|
|
err = backend.SetUserIdentity(author)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
defer backend.Close()
|
|
|
|
interrupt.RegisterCleaner(backend.Close)
|
|
|
|
|
2020-02-12 20:32:01 +03:00
|
|
|
token := auth.NewToken(target, envToken)
|
2020-02-08 19:35:35 +03:00
|
|
|
token.SetMetadata(auth.MetaKeyLogin, login)
|
2020-02-10 00:17:10 +03:00
|
|
|
token.SetMetadata(auth.MetaKeyBaseURL, defaultBaseURL)
|
2019-12-08 23:15:06 +03:00
|
|
|
err = auth.Store(repo, token)
|
|
|
|
require.NoError(t, err)
|
2019-08-13 21:51:39 +03:00
|
|
|
|
2020-02-08 19:35:35 +03:00
|
|
|
tests := testCases(t, backend)
|
|
|
|
|
2019-08-13 21:51:39 +03:00
|
|
|
// generate project name
|
|
|
|
projectName := generateRepoName()
|
|
|
|
|
|
|
|
// create target Gitlab repository
|
|
|
|
projectID, err := createRepository(context.TODO(), projectName, token)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
fmt.Println("created repository", projectName)
|
|
|
|
|
|
|
|
// Make sure to remove the Gitlab repository when the test end
|
|
|
|
defer func(t *testing.T) {
|
|
|
|
if err := deleteRepository(context.TODO(), projectID, token); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
fmt.Println("deleted repository:", projectName)
|
|
|
|
}(t)
|
|
|
|
|
|
|
|
interrupt.RegisterCleaner(func() error {
|
|
|
|
return deleteRepository(context.TODO(), projectID, token)
|
|
|
|
})
|
|
|
|
|
2020-02-15 17:39:49 +03:00
|
|
|
ctx := context.Background()
|
|
|
|
|
2019-08-13 21:51:39 +03:00
|
|
|
// initialize exporter
|
|
|
|
exporter := &gitlabExporter{}
|
2020-02-15 17:39:49 +03:00
|
|
|
err = exporter.Init(ctx, backend, core.Configuration{
|
2020-02-15 04:55:19 +03:00
|
|
|
confKeyProjectID: strconv.Itoa(projectID),
|
|
|
|
confKeyGitlabBaseUrl: defaultBaseURL,
|
2020-02-23 16:05:03 +03:00
|
|
|
confKeyDefaultLogin: login,
|
2019-08-13 21:51:39 +03:00
|
|
|
})
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
start := time.Now()
|
|
|
|
|
|
|
|
// export all bugs
|
2019-08-19 14:50:48 +03:00
|
|
|
exportEvents, err := exporter.ExportAll(ctx, backend, time.Time{})
|
2019-08-13 21:51:39 +03:00
|
|
|
require.NoError(t, err)
|
|
|
|
|
2019-08-19 14:50:48 +03:00
|
|
|
for result := range exportEvents {
|
2019-08-13 21:51:39 +03:00
|
|
|
require.NoError(t, result.Err)
|
|
|
|
}
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
fmt.Printf("test repository exported in %f seconds\n", time.Since(start).Seconds())
|
|
|
|
|
2020-10-04 20:56:16 +03:00
|
|
|
repoTwo := repository.CreateGoGitTestRepo(false)
|
2020-06-23 19:02:54 +03:00
|
|
|
defer repository.CleanupTestRepos(repoTwo)
|
2019-08-13 21:51:39 +03:00
|
|
|
|
|
|
|
// create a second backend
|
|
|
|
backendTwo, err := cache.NewRepoCache(repoTwo)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
importer := &gitlabImporter{}
|
2020-02-15 17:39:49 +03:00
|
|
|
err = importer.Init(ctx, backend, core.Configuration{
|
2020-02-15 04:55:19 +03:00
|
|
|
confKeyProjectID: strconv.Itoa(projectID),
|
|
|
|
confKeyGitlabBaseUrl: defaultBaseURL,
|
2020-02-23 16:05:03 +03:00
|
|
|
confKeyDefaultLogin: login,
|
2019-08-13 21:51:39 +03:00
|
|
|
})
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
// import all exported bugs to the second backend
|
2019-08-19 14:50:48 +03:00
|
|
|
importEvents, err := importer.ImportAll(ctx, backendTwo, time.Time{})
|
2019-08-13 21:51:39 +03:00
|
|
|
require.NoError(t, err)
|
|
|
|
|
2019-08-19 14:50:48 +03:00
|
|
|
for result := range importEvents {
|
|
|
|
require.NoError(t, result.Err)
|
|
|
|
}
|
|
|
|
|
2019-08-13 21:51:39 +03:00
|
|
|
require.Len(t, backendTwo.AllBugsIds(), len(tests))
|
|
|
|
|
|
|
|
for _, tt := range tests {
|
|
|
|
t.Run(tt.name, func(t *testing.T) {
|
2020-07-26 12:29:20 +03:00
|
|
|
if tt.name == "bug changed status" {
|
|
|
|
t.Skip("test known as broken, see https://github.com/MichaelMure/git-bug/issues/435 and complain to gitlab")
|
|
|
|
// TODO: fix, somehow, someday, or drop support.
|
|
|
|
}
|
|
|
|
|
2019-08-13 21:51:39 +03:00
|
|
|
// for each operation a SetMetadataOperation will be added
|
|
|
|
// so number of operations should double
|
2019-08-19 15:09:31 +03:00
|
|
|
require.Len(t, tt.bug.Snapshot().Operations, tt.numOpExp)
|
2019-08-13 21:51:39 +03:00
|
|
|
|
|
|
|
// verify operation have correct metadata
|
|
|
|
for _, op := range tt.bug.Snapshot().Operations {
|
|
|
|
// Check if the originals operations (*not* SetMetadata) are tagged properly
|
|
|
|
if _, ok := op.(*bug.SetMetadataOperation); !ok {
|
2019-11-10 19:48:13 +03:00
|
|
|
_, haveIDMetadata := op.GetMetadata(metaKeyGitlabId)
|
2019-08-13 21:51:39 +03:00
|
|
|
require.True(t, haveIDMetadata)
|
|
|
|
|
2019-11-10 19:48:13 +03:00
|
|
|
_, haveURLMetada := op.GetMetadata(metaKeyGitlabUrl)
|
2019-08-13 21:51:39 +03:00
|
|
|
require.True(t, haveURLMetada)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// get bug gitlab ID
|
2019-11-10 19:48:13 +03:00
|
|
|
bugGitlabID, ok := tt.bug.Snapshot().GetCreateMetadata(metaKeyGitlabId)
|
2019-08-13 21:51:39 +03:00
|
|
|
require.True(t, ok)
|
|
|
|
|
|
|
|
// retrieve bug from backendTwo
|
2019-11-10 19:48:13 +03:00
|
|
|
importedBug, err := backendTwo.ResolveBugCreateMetadata(metaKeyGitlabId, bugGitlabID)
|
2019-08-13 21:51:39 +03:00
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
// verify bug have same number of original operations
|
2019-08-19 15:09:31 +03:00
|
|
|
require.Len(t, importedBug.Snapshot().Operations, tt.numOpImp)
|
2019-08-13 21:51:39 +03:00
|
|
|
|
2020-02-08 19:35:35 +03:00
|
|
|
// verify bugs are tagged with origin=gitlab
|
2019-11-10 19:48:13 +03:00
|
|
|
issueOrigin, ok := importedBug.Snapshot().GetCreateMetadata(core.MetaKeyOrigin)
|
2019-08-13 21:51:39 +03:00
|
|
|
require.True(t, ok)
|
|
|
|
require.Equal(t, issueOrigin, target)
|
|
|
|
|
|
|
|
//TODO: maybe more tests to ensure bug final state
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func generateRepoName() string {
|
|
|
|
rand.Seed(time.Now().UnixNano())
|
|
|
|
var letterRunes = []rune("abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ")
|
|
|
|
b := make([]rune, 8)
|
|
|
|
for i := range b {
|
|
|
|
b[i] = letterRunes[rand.Intn(len(letterRunes))]
|
|
|
|
}
|
|
|
|
return fmt.Sprintf("%s-%s", testRepoBaseName, string(b))
|
|
|
|
}
|
|
|
|
|
|
|
|
// create repository need a token with scope 'repo'
|
2019-12-08 23:15:06 +03:00
|
|
|
func createRepository(ctx context.Context, name string, token *auth.Token) (int, error) {
|
2020-02-10 00:17:10 +03:00
|
|
|
client, err := buildClient(defaultBaseURL, token)
|
2019-12-10 22:30:29 +03:00
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
|
2019-08-13 21:51:39 +03:00
|
|
|
project, _, err := client.Projects.CreateProject(
|
|
|
|
&gitlab.CreateProjectOptions{
|
|
|
|
Name: gitlab.String(name),
|
|
|
|
},
|
|
|
|
gitlab.WithContext(ctx),
|
|
|
|
)
|
2019-11-05 23:15:21 +03:00
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
2019-08-13 21:51:39 +03:00
|
|
|
|
2020-07-26 12:29:20 +03:00
|
|
|
// fmt.Println("Project URL:", project.WebURL)
|
|
|
|
|
2019-11-05 23:15:21 +03:00
|
|
|
return project.ID, nil
|
2019-08-13 21:51:39 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// delete repository need a token with scope 'delete_repo'
|
2019-12-08 23:15:06 +03:00
|
|
|
func deleteRepository(ctx context.Context, project int, token *auth.Token) error {
|
2020-02-10 00:17:10 +03:00
|
|
|
client, err := buildClient(defaultBaseURL, token)
|
2019-12-10 22:30:29 +03:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
_, err = client.Projects.DeleteProject(project, gitlab.WithContext(ctx))
|
2019-08-13 21:51:39 +03:00
|
|
|
return err
|
|
|
|
}
|