396 lines
9 KiB
Go
396 lines
9 KiB
Go
|
// Copyright 2019 Drone.IO Inc. All rights reserved.
|
||
|
// Use of this source code is governed by the Drone Non-Commercial License
|
||
|
// that can be found in the LICENSE file.
|
||
|
|
||
|
package batch2
|
||
|
|
||
|
import (
|
||
|
"context"
|
||
|
"database/sql"
|
||
|
"testing"
|
||
|
|
||
|
"github.com/drone/drone/core"
|
||
|
"github.com/drone/drone/store/perm"
|
||
|
"github.com/drone/drone/store/repos"
|
||
|
"github.com/drone/drone/store/shared/db"
|
||
|
"github.com/drone/drone/store/shared/db/dbtest"
|
||
|
"github.com/drone/drone/store/user"
|
||
|
)
|
||
|
|
||
|
var noContext = context.TODO()
|
||
|
|
||
|
func TestBatch(t *testing.T) {
|
||
|
conn, err := dbtest.Connect()
|
||
|
if err != nil {
|
||
|
t.Error(err)
|
||
|
return
|
||
|
}
|
||
|
defer func() {
|
||
|
dbtest.Reset(conn)
|
||
|
dbtest.Disconnect(conn)
|
||
|
}()
|
||
|
|
||
|
batcher := New(conn).(*batchUpdater)
|
||
|
repos := repos.New(conn)
|
||
|
perms := perm.New(conn)
|
||
|
|
||
|
user, err := seedUser(batcher.db)
|
||
|
if err != nil {
|
||
|
t.Error(err)
|
||
|
}
|
||
|
|
||
|
t.Run("Insert", testBatchInsert(batcher, repos, perms, user))
|
||
|
t.Run("Update", testBatchUpdate(batcher, repos, perms, user))
|
||
|
t.Run("Delete", testBatchDelete(batcher, repos, perms, user))
|
||
|
t.Run("DuplicateID", testBatchDuplicateID(batcher, repos, perms, user))
|
||
|
t.Run("DuplicateSlug", testBatchDuplicateSlug(batcher, repos, perms, user))
|
||
|
t.Run("DuplicateRename", testBatchDuplicateRename(batcher, repos, perms, user))
|
||
|
t.Run("DuplicateRecreateRename", testBatchDuplicateRecreateRename(batcher, repos, perms, user))
|
||
|
|
||
|
}
|
||
|
|
||
|
func testBatchInsert(
|
||
|
batcher core.Batcher,
|
||
|
repos core.RepositoryStore,
|
||
|
perms core.PermStore,
|
||
|
user *core.User,
|
||
|
) func(t *testing.T) {
|
||
|
return func(t *testing.T) {
|
||
|
batch := &core.Batch{
|
||
|
Insert: []*core.Repository{
|
||
|
{
|
||
|
UserID: 1,
|
||
|
UID: "42",
|
||
|
Namespace: "octocat",
|
||
|
Name: "hello-world",
|
||
|
Slug: "octocat/hello-world",
|
||
|
Private: false,
|
||
|
Visibility: "public",
|
||
|
},
|
||
|
},
|
||
|
}
|
||
|
err := batcher.Batch(noContext, user, batch)
|
||
|
if err != nil {
|
||
|
t.Error(err)
|
||
|
}
|
||
|
|
||
|
repo, err := repos.FindName(noContext, "octocat", "hello-world")
|
||
|
if err != nil {
|
||
|
t.Errorf("Want repository, got error %q", err)
|
||
|
}
|
||
|
|
||
|
_, err = perms.Find(noContext, repo.UID, user.ID)
|
||
|
if err != nil {
|
||
|
t.Errorf("Want permissions, got error %q", err)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func testBatchUpdate(
|
||
|
batcher core.Batcher,
|
||
|
repos core.RepositoryStore,
|
||
|
perms core.PermStore,
|
||
|
user *core.User,
|
||
|
) func(t *testing.T) {
|
||
|
return func(t *testing.T) {
|
||
|
before, err := repos.FindName(noContext, "octocat", "hello-world")
|
||
|
if err != nil {
|
||
|
t.Errorf("Want repository, got error %q", err)
|
||
|
}
|
||
|
|
||
|
batch := &core.Batch{
|
||
|
Update: []*core.Repository{
|
||
|
{
|
||
|
ID: before.ID,
|
||
|
UserID: 1,
|
||
|
UID: "42",
|
||
|
Namespace: "octocat",
|
||
|
Name: "hello-world",
|
||
|
Slug: "octocat/hello-world",
|
||
|
Private: true,
|
||
|
},
|
||
|
},
|
||
|
}
|
||
|
|
||
|
err = batcher.Batch(noContext, user, batch)
|
||
|
if err != nil {
|
||
|
t.Error(err)
|
||
|
}
|
||
|
|
||
|
after, err := repos.FindName(noContext, "octocat", "hello-world")
|
||
|
if err != nil {
|
||
|
t.Errorf("Want repository, got error %q", err)
|
||
|
}
|
||
|
|
||
|
if got, want := after.Private, true; got != want {
|
||
|
t.Errorf("Want repository Private %v, got %v", want, got)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func testBatchDelete(
|
||
|
batcher core.Batcher,
|
||
|
repos core.RepositoryStore,
|
||
|
perms core.PermStore,
|
||
|
user *core.User,
|
||
|
) func(t *testing.T) {
|
||
|
return func(t *testing.T) {
|
||
|
repo, err := repos.FindName(noContext, "octocat", "hello-world")
|
||
|
if err != nil {
|
||
|
t.Errorf("Want repository, got error %q", err)
|
||
|
}
|
||
|
|
||
|
_, err = perms.Find(noContext, repo.UID, user.ID)
|
||
|
if err != nil {
|
||
|
t.Errorf("Want permissions, got error %q", err)
|
||
|
}
|
||
|
|
||
|
batch := &core.Batch{
|
||
|
Revoke: []*core.Repository{
|
||
|
{
|
||
|
ID: repo.ID,
|
||
|
UserID: 1,
|
||
|
UID: "42",
|
||
|
Namespace: "octocat",
|
||
|
Name: "hello-world",
|
||
|
Slug: "octocat/hello-world",
|
||
|
Private: true,
|
||
|
},
|
||
|
},
|
||
|
}
|
||
|
|
||
|
err = batcher.Batch(noContext, user, batch)
|
||
|
if err != nil {
|
||
|
t.Error(err)
|
||
|
}
|
||
|
|
||
|
_, err = perms.Find(noContext, repo.UID, user.ID)
|
||
|
if err != sql.ErrNoRows {
|
||
|
t.Errorf("Want sql.ErrNoRows got %v", err)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func testBatchDuplicateID(
|
||
|
batcher core.Batcher,
|
||
|
repos core.RepositoryStore,
|
||
|
perms core.PermStore,
|
||
|
user *core.User,
|
||
|
) func(t *testing.T) {
|
||
|
return func(t *testing.T) {
|
||
|
before, err := repos.FindName(noContext, "octocat", "hello-world")
|
||
|
if err != nil {
|
||
|
t.Errorf("Want repository, got error %q", err)
|
||
|
}
|
||
|
|
||
|
batchDuplicate := &core.Batch{
|
||
|
Insert: []*core.Repository{
|
||
|
{
|
||
|
ID: 0,
|
||
|
UserID: 1,
|
||
|
UID: "43", // Updated ID
|
||
|
Namespace: "octocat",
|
||
|
Name: "hello-world",
|
||
|
Slug: "octocat/hello-world",
|
||
|
},
|
||
|
{
|
||
|
ID: 0,
|
||
|
UserID: 1,
|
||
|
UID: "43", // Updated ID
|
||
|
Namespace: "octocat",
|
||
|
Name: "hello-world",
|
||
|
Slug: "octocat/hello-world",
|
||
|
},
|
||
|
},
|
||
|
}
|
||
|
|
||
|
err = batcher.Batch(noContext, user, batchDuplicate)
|
||
|
if err != nil {
|
||
|
t.Error(err)
|
||
|
return
|
||
|
}
|
||
|
|
||
|
batch := &core.Batch{
|
||
|
Insert: []*core.Repository{
|
||
|
{
|
||
|
ID: 0,
|
||
|
UserID: 1,
|
||
|
UID: "64778136",
|
||
|
Namespace: "octocat",
|
||
|
Name: "linguist",
|
||
|
Slug: "octocat/linguist",
|
||
|
},
|
||
|
},
|
||
|
Update: []*core.Repository{
|
||
|
{
|
||
|
ID: before.ID,
|
||
|
UserID: 1,
|
||
|
UID: "44", // Updated ID
|
||
|
Namespace: "octocat",
|
||
|
Name: "hello-world",
|
||
|
Slug: "octocat/hello-world",
|
||
|
Private: true,
|
||
|
},
|
||
|
},
|
||
|
}
|
||
|
|
||
|
err = batcher.Batch(noContext, user, batch)
|
||
|
if err != nil {
|
||
|
t.Error(err)
|
||
|
return
|
||
|
}
|
||
|
|
||
|
added, err := repos.FindName(noContext, "octocat", "linguist")
|
||
|
if err != nil {
|
||
|
t.Errorf("Want inserted repository, got error %q", err)
|
||
|
}
|
||
|
|
||
|
if got, want := added.UID, "64778136"; got != want {
|
||
|
t.Errorf("Want inserted repository UID %v, got %v", want, got)
|
||
|
}
|
||
|
|
||
|
renamed, err := repos.FindName(noContext, "octocat", "hello-world")
|
||
|
if err != nil {
|
||
|
t.Errorf("Want renamed repository, got error %q", err)
|
||
|
}
|
||
|
|
||
|
if got, want := renamed.UID, "44"; got != want {
|
||
|
t.Errorf("Want renamed repository UID %v, got %v", want, got)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// the purpose of this unit test is to understand what happens
|
||
|
// when a repository is deleted, re-created with the same name,
|
||
|
// but has a different unique identifier.
|
||
|
func testBatchDuplicateSlug(
|
||
|
batcher core.Batcher,
|
||
|
repos core.RepositoryStore,
|
||
|
perms core.PermStore,
|
||
|
user *core.User,
|
||
|
) func(t *testing.T) {
|
||
|
return func(t *testing.T) {
|
||
|
_, err := repos.FindName(noContext, "octocat", "hello-world")
|
||
|
if err != nil {
|
||
|
t.Errorf("Want repository, got error %q", err)
|
||
|
return
|
||
|
}
|
||
|
|
||
|
batch := &core.Batch{
|
||
|
Insert: []*core.Repository{
|
||
|
{
|
||
|
ID: 0,
|
||
|
UserID: 1,
|
||
|
UID: "99", // Updated ID
|
||
|
Namespace: "octocat",
|
||
|
Name: "hello-world",
|
||
|
Slug: "octocat/hello-world",
|
||
|
},
|
||
|
},
|
||
|
}
|
||
|
err = batcher.Batch(noContext, user, batch)
|
||
|
if err != nil {
|
||
|
t.Error(err)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// the purpose of this unit test is to understand what happens
|
||
|
// when a repository is deleted, re-created with a different name,
|
||
|
// renamed to the original name, but has a different unique identifier.
|
||
|
func testBatchDuplicateRecreateRename(
|
||
|
batcher core.Batcher,
|
||
|
repos core.RepositoryStore,
|
||
|
perms core.PermStore,
|
||
|
user *core.User,
|
||
|
) func(t *testing.T) {
|
||
|
return func(t *testing.T) {
|
||
|
_, err := repos.FindName(noContext, "octocat", "hello-world")
|
||
|
if err != nil {
|
||
|
t.Errorf("Want repository, got error %q", err)
|
||
|
return
|
||
|
}
|
||
|
|
||
|
batch := &core.Batch{
|
||
|
Update: []*core.Repository{
|
||
|
{
|
||
|
ID: 0,
|
||
|
UserID: 1,
|
||
|
UID: "8888", // Updated ID
|
||
|
Namespace: "octocat",
|
||
|
Name: "hello-world",
|
||
|
Slug: "octocat/hello-world",
|
||
|
},
|
||
|
},
|
||
|
}
|
||
|
err = batcher.Batch(noContext, user, batch)
|
||
|
if err != nil {
|
||
|
t.Error(err)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// the purpose of this unit test is to understand what happens
|
||
|
// when a repository is deleted, re-created with a new name, and
|
||
|
// then updated back to the old name.
|
||
|
//
|
||
|
// TODO(bradrydzewski) for sqlite consider UPDATE OR REPLACE.
|
||
|
// TODO(bradrydzewski) for mysql consider UPDATE IGNORE.
|
||
|
// TODO(bradrydzewski) consider breaking rename into a separate set of logic that checks for existing records.
|
||
|
func testBatchDuplicateRename(
|
||
|
batcher core.Batcher,
|
||
|
repos core.RepositoryStore,
|
||
|
perms core.PermStore,
|
||
|
user *core.User,
|
||
|
) func(t *testing.T) {
|
||
|
return func(t *testing.T) {
|
||
|
batch := &core.Batch{
|
||
|
Insert: []*core.Repository{
|
||
|
{
|
||
|
ID: 0,
|
||
|
UserID: 1,
|
||
|
UID: "200",
|
||
|
Namespace: "octocat",
|
||
|
Name: "test-1",
|
||
|
Slug: "octocat/test-1",
|
||
|
},
|
||
|
{
|
||
|
ID: 0,
|
||
|
UserID: 1,
|
||
|
UID: "201",
|
||
|
Namespace: "octocat",
|
||
|
Name: "test-2",
|
||
|
Slug: "octocat/test-2",
|
||
|
},
|
||
|
},
|
||
|
}
|
||
|
err := batcher.Batch(noContext, user, batch)
|
||
|
if err != nil {
|
||
|
t.Error(err)
|
||
|
return
|
||
|
}
|
||
|
|
||
|
before, err := repos.FindName(noContext, "octocat", "test-2")
|
||
|
if err != nil {
|
||
|
t.Errorf("Want repository, got error %q", err)
|
||
|
return
|
||
|
}
|
||
|
before.Name = "test-1"
|
||
|
before.Slug = "octocat/test-1"
|
||
|
|
||
|
batch = &core.Batch{
|
||
|
Update: []*core.Repository{before},
|
||
|
}
|
||
|
err = batcher.Batch(noContext, user, batch)
|
||
|
if err != nil {
|
||
|
t.Skip(err)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func seedUser(db *db.DB) (*core.User, error) {
|
||
|
out := &core.User{Login: "octocat"}
|
||
|
err := user.New(db).Create(noContext, out)
|
||
|
return out, err
|
||
|
}
|