mirror of
https://github.com/harness/drone.git
synced 2025-05-17 09:30:00 +08:00
add alternative batch syncer
This commit is contained in:
parent
1e0f028166
commit
05be17ff76
420
store/batch2/batch.go
Normal file
420
store/batch2/batch.go
Normal file
@ -0,0 +1,420 @@
|
|||||||
|
// Copyright 2019 Drone IO, Inc.
|
||||||
|
//
|
||||||
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
// you may not use this file except in compliance with the License.
|
||||||
|
// You may obtain a copy of the License at
|
||||||
|
//
|
||||||
|
// http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
//
|
||||||
|
// Unless required by applicable law or agreed to in writing, software
|
||||||
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
// See the License for the specific language governing permissions and
|
||||||
|
// limitations under the License.
|
||||||
|
|
||||||
|
package batch2
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"fmt"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/drone/drone/core"
|
||||||
|
"github.com/drone/drone/store/repos"
|
||||||
|
"github.com/drone/drone/store/shared/db"
|
||||||
|
)
|
||||||
|
|
||||||
|
// New returns a new Batcher.
|
||||||
|
func New(db *db.DB) core.Batcher {
|
||||||
|
return &batchUpdater{db}
|
||||||
|
}
|
||||||
|
|
||||||
|
type batchUpdater struct {
|
||||||
|
db *db.DB
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *batchUpdater) Batch(ctx context.Context, user *core.User, batch *core.Batch) error {
|
||||||
|
return b.db.Update(func(execer db.Execer, binder db.Binder) error {
|
||||||
|
now := time.Now().Unix()
|
||||||
|
|
||||||
|
//
|
||||||
|
// the repository list API does not return permissions, which means we have
|
||||||
|
// no way of knowing if permissions are current or not. We therefore mark all
|
||||||
|
// permissions stale in the database, so that each one must be individually
|
||||||
|
// verified at runtime.
|
||||||
|
//
|
||||||
|
|
||||||
|
stmt := permResetStmt
|
||||||
|
switch b.db.Driver() {
|
||||||
|
case db.Postgres:
|
||||||
|
stmt = permResetStmtPostgres
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err := execer.Exec(stmt, now, user.ID)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("batch: cannot reset permissions: %s", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// if the repository exists with the same name,
|
||||||
|
// but a different unique identifier, attempt to
|
||||||
|
// delete the previous entry.
|
||||||
|
var insert []*core.Repository
|
||||||
|
var update []*core.Repository
|
||||||
|
for _, repo := range append(batch.Insert, batch.Update...) {
|
||||||
|
params := repos.ToParams(repo)
|
||||||
|
stmt, args, err := binder.BindNamed(repoDeleteDeleted, params)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
res, err := execer.Exec(stmt, args...)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("batch: cannot remove duplicate repository: %s: %s: %s", repo.Slug, repo.UID, err)
|
||||||
|
}
|
||||||
|
rows, _ := res.RowsAffected()
|
||||||
|
if rows > 0 {
|
||||||
|
insert = append(insert, repo)
|
||||||
|
} else if repo.ID > 0 {
|
||||||
|
update = append(update, repo)
|
||||||
|
} else {
|
||||||
|
insert = append(insert, repo)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, repo := range insert {
|
||||||
|
|
||||||
|
//
|
||||||
|
// insert repository
|
||||||
|
// TODO: group inserts in batches of N
|
||||||
|
//
|
||||||
|
|
||||||
|
stmt := repoInsertIgnoreStmt
|
||||||
|
switch b.db.Driver() {
|
||||||
|
case db.Mysql:
|
||||||
|
stmt = repoInsertIgnoreStmtMysql
|
||||||
|
case db.Postgres:
|
||||||
|
stmt = repoInsertIgnoreStmtPostgres
|
||||||
|
}
|
||||||
|
|
||||||
|
params := repos.ToParams(repo)
|
||||||
|
stmt, args, err := binder.BindNamed(stmt, params)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
_, err = execer.Exec(stmt, args...)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("batch: cannot insert repository: %s: %s: %s", repo.Slug, repo.UID, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// insert permissions
|
||||||
|
// TODO: group inserts in batches of N
|
||||||
|
//
|
||||||
|
|
||||||
|
stmt = permInsertIgnoreStmt
|
||||||
|
switch b.db.Driver() {
|
||||||
|
case db.Mysql:
|
||||||
|
stmt = permInsertIgnoreStmtMysql
|
||||||
|
case db.Postgres:
|
||||||
|
stmt = permInsertIgnoreStmtPostgres
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err = execer.Exec(stmt,
|
||||||
|
user.ID,
|
||||||
|
repo.UID,
|
||||||
|
now,
|
||||||
|
now,
|
||||||
|
)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("batch: cannot insert permissions: %s: %s: %s", repo.Slug, repo.UID, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// update existing repositories
|
||||||
|
// TODO: group updates in batches of N
|
||||||
|
//
|
||||||
|
|
||||||
|
for _, repo := range update {
|
||||||
|
params := repos.ToParams(repo)
|
||||||
|
|
||||||
|
// // if the repository exists with the same name,
|
||||||
|
// // but a different unique identifier, attempt to
|
||||||
|
// // delete the previous entry.
|
||||||
|
// stmt, args, err := binder.BindNamed(repoDeleteDeleted, params)
|
||||||
|
// if err != nil {
|
||||||
|
// return err
|
||||||
|
// }
|
||||||
|
// res, err := execer.Exec(stmt, args...)
|
||||||
|
// if err != nil {
|
||||||
|
// return fmt.Errorf("batch: cannot remove duplicate repository: %s: %s: %s", repo.Slug, repo.UID, err)
|
||||||
|
// }
|
||||||
|
// rows, _ := res.RowsAffected()
|
||||||
|
// if rows > 0 {
|
||||||
|
// stmt := repoInsertIgnoreStmt
|
||||||
|
// switch b.db.Driver() {
|
||||||
|
// case db.Mysql:
|
||||||
|
// stmt = repoInsertIgnoreStmtMysql
|
||||||
|
// case db.Postgres:
|
||||||
|
// stmt = repoInsertIgnoreStmtPostgres
|
||||||
|
// }
|
||||||
|
|
||||||
|
// params := repos.ToParams(repo)
|
||||||
|
// stmt, args, err := binder.BindNamed(stmt, params)
|
||||||
|
// if err != nil {
|
||||||
|
// return err
|
||||||
|
// }
|
||||||
|
// _, err = execer.Exec(stmt, args...)
|
||||||
|
// if err != nil {
|
||||||
|
// return fmt.Errorf("batch: cannot insert repository: %s: %s: %s", repo.Slug, repo.UID, err)
|
||||||
|
// }
|
||||||
|
// } else {
|
||||||
|
stmt, args, err := binder.BindNamed(repoUpdateRemoteStmt, params)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
_, err = execer.Exec(stmt, args...)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("batch: cannot update repository: %s: %s: %s", repo.Slug, repo.UID, err)
|
||||||
|
}
|
||||||
|
// }
|
||||||
|
|
||||||
|
stmt = permInsertIgnoreStmt
|
||||||
|
switch b.db.Driver() {
|
||||||
|
case db.Mysql:
|
||||||
|
stmt = permInsertIgnoreStmtMysql
|
||||||
|
case db.Postgres:
|
||||||
|
stmt = permInsertIgnoreStmtPostgres
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err = execer.Exec(stmt,
|
||||||
|
user.ID,
|
||||||
|
repo.UID,
|
||||||
|
now,
|
||||||
|
now,
|
||||||
|
)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("batch: cannot insert permissions: %s: %s: %s", repo.Slug, repo.UID, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// revoke permissions
|
||||||
|
// TODO: group deletes in batches of N
|
||||||
|
//
|
||||||
|
|
||||||
|
for _, repo := range batch.Revoke {
|
||||||
|
stmt := permRevokeStmt
|
||||||
|
switch b.db.Driver() {
|
||||||
|
case db.Postgres:
|
||||||
|
stmt = permRevokeStmtPostgres
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err = execer.Exec(stmt, user.ID, repo.UID)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("batch: cannot revoking permissions: %s: %s: %s", repo.Slug, repo.UID, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
const stmtInsertBase = `
|
||||||
|
(
|
||||||
|
repo_uid
|
||||||
|
,repo_user_id
|
||||||
|
,repo_namespace
|
||||||
|
,repo_name
|
||||||
|
,repo_slug
|
||||||
|
,repo_scm
|
||||||
|
,repo_clone_url
|
||||||
|
,repo_ssh_url
|
||||||
|
,repo_html_url
|
||||||
|
,repo_active
|
||||||
|
,repo_private
|
||||||
|
,repo_visibility
|
||||||
|
,repo_branch
|
||||||
|
,repo_counter
|
||||||
|
,repo_config
|
||||||
|
,repo_timeout
|
||||||
|
,repo_trusted
|
||||||
|
,repo_protected
|
||||||
|
,repo_no_forks
|
||||||
|
,repo_no_pulls
|
||||||
|
,repo_synced
|
||||||
|
,repo_created
|
||||||
|
,repo_updated
|
||||||
|
,repo_version
|
||||||
|
,repo_signer
|
||||||
|
,repo_secret
|
||||||
|
) VALUES (
|
||||||
|
:repo_uid
|
||||||
|
,:repo_user_id
|
||||||
|
,:repo_namespace
|
||||||
|
,:repo_name
|
||||||
|
,:repo_slug
|
||||||
|
,:repo_scm
|
||||||
|
,:repo_clone_url
|
||||||
|
,:repo_ssh_url
|
||||||
|
,:repo_html_url
|
||||||
|
,:repo_active
|
||||||
|
,:repo_private
|
||||||
|
,:repo_visibility
|
||||||
|
,:repo_branch
|
||||||
|
,:repo_counter
|
||||||
|
,:repo_config
|
||||||
|
,:repo_timeout
|
||||||
|
,:repo_trusted
|
||||||
|
,:repo_protected
|
||||||
|
,:repo_no_forks
|
||||||
|
,:repo_no_pulls
|
||||||
|
,:repo_synced
|
||||||
|
,:repo_created
|
||||||
|
,:repo_updated
|
||||||
|
,:repo_version
|
||||||
|
,:repo_signer
|
||||||
|
,:repo_secret
|
||||||
|
)
|
||||||
|
`
|
||||||
|
|
||||||
|
const repoInsertIgnoreStmt = `
|
||||||
|
INSERT OR IGNORE INTO repos ` + stmtInsertBase
|
||||||
|
|
||||||
|
const repoInsertIgnoreStmtMysql = `
|
||||||
|
INSERT IGNORE INTO repos ` + stmtInsertBase
|
||||||
|
|
||||||
|
const repoInsertIgnoreStmtPostgres = `
|
||||||
|
INSERT INTO repos ` + stmtInsertBase + ` ON CONFLICT DO NOTHING`
|
||||||
|
|
||||||
|
const repoUpdateRemoteStmt = `
|
||||||
|
UPDATE repos SET
|
||||||
|
repo_namespace=:repo_namespace
|
||||||
|
,repo_name=:repo_name
|
||||||
|
,repo_slug=:repo_slug
|
||||||
|
,repo_clone_url=:repo_clone_url
|
||||||
|
,repo_ssh_url=:repo_ssh_url
|
||||||
|
,repo_html_url=:repo_html_url
|
||||||
|
,repo_private=:repo_private
|
||||||
|
,repo_branch=:repo_branch
|
||||||
|
,repo_updated=:repo_updated
|
||||||
|
WHERE repo_id=:repo_id
|
||||||
|
`
|
||||||
|
|
||||||
|
const repoUpdateRemoteStmtPostgres = `
|
||||||
|
UPDATE repos SET
|
||||||
|
repo_namespace=$1
|
||||||
|
,repo_name=$2
|
||||||
|
,repo_slug=$3
|
||||||
|
,repo_clone_url=$4
|
||||||
|
,repo_ssh_url=$5
|
||||||
|
,repo_html_url=$6
|
||||||
|
,repo_private=$7
|
||||||
|
,repo_branch=$8
|
||||||
|
,repo_updated=$9
|
||||||
|
WHERE repo_id=$10
|
||||||
|
`
|
||||||
|
|
||||||
|
const permInsertIgnoreStmt = `
|
||||||
|
INSERT OR IGNORE INTO perms (
|
||||||
|
perm_user_id
|
||||||
|
,perm_repo_uid
|
||||||
|
,perm_read
|
||||||
|
,perm_write
|
||||||
|
,perm_admin
|
||||||
|
,perm_synced
|
||||||
|
,perm_created
|
||||||
|
,perm_updated
|
||||||
|
) values (
|
||||||
|
?
|
||||||
|
,?
|
||||||
|
,1
|
||||||
|
,0
|
||||||
|
,0
|
||||||
|
,0
|
||||||
|
,?
|
||||||
|
,?
|
||||||
|
)
|
||||||
|
`
|
||||||
|
|
||||||
|
const permInsertIgnoreStmtMysql = `
|
||||||
|
INSERT IGNORE INTO perms (
|
||||||
|
perm_user_id
|
||||||
|
,perm_repo_uid
|
||||||
|
,perm_read
|
||||||
|
,perm_write
|
||||||
|
,perm_admin
|
||||||
|
,perm_synced
|
||||||
|
,perm_created
|
||||||
|
,perm_updated
|
||||||
|
) values (
|
||||||
|
?
|
||||||
|
,?
|
||||||
|
,1
|
||||||
|
,0
|
||||||
|
,0
|
||||||
|
,0
|
||||||
|
,?
|
||||||
|
,?
|
||||||
|
)
|
||||||
|
`
|
||||||
|
|
||||||
|
const permInsertIgnoreStmtPostgres = `
|
||||||
|
INSERT INTO perms (
|
||||||
|
perm_user_id
|
||||||
|
,perm_repo_uid
|
||||||
|
,perm_read
|
||||||
|
,perm_write
|
||||||
|
,perm_admin
|
||||||
|
,perm_synced
|
||||||
|
,perm_created
|
||||||
|
,perm_updated
|
||||||
|
) values (
|
||||||
|
$1
|
||||||
|
,$2
|
||||||
|
,true
|
||||||
|
,false
|
||||||
|
,false
|
||||||
|
,0
|
||||||
|
,$3
|
||||||
|
,$4
|
||||||
|
) ON CONFLICT DO NOTHING
|
||||||
|
`
|
||||||
|
|
||||||
|
// this statement deletes a repository that was
|
||||||
|
// deleted in version control and then re-created
|
||||||
|
// with the same name (and thus has a different
|
||||||
|
// unique identifier)
|
||||||
|
const repoDeleteDeleted = `
|
||||||
|
DELETE FROM repos
|
||||||
|
WHERE repo_slug = :repo_slug
|
||||||
|
AND repo_uid != :repo_uid
|
||||||
|
`
|
||||||
|
|
||||||
|
// this resets the synced date indicating that
|
||||||
|
// the system should refresh the permissions next
|
||||||
|
// time the user attempts to access the resource
|
||||||
|
const permResetStmt = `
|
||||||
|
UPDATE perms SET
|
||||||
|
perm_updated = ?
|
||||||
|
,perm_synced = 0
|
||||||
|
WHERE perm_user_id = ?
|
||||||
|
`
|
||||||
|
|
||||||
|
const permResetStmtPostgres = `
|
||||||
|
UPDATE perms SET
|
||||||
|
perm_updated = $1
|
||||||
|
,perm_synced = 0
|
||||||
|
WHERE perm_user_id = $2
|
||||||
|
`
|
||||||
|
|
||||||
|
const permRevokeStmt = `
|
||||||
|
DELETE FROM perms
|
||||||
|
WHERE perm_user_id = ?
|
||||||
|
AND perm_repo_uid = ?
|
||||||
|
`
|
||||||
|
|
||||||
|
const permRevokeStmtPostgres = `
|
||||||
|
DELETE FROM perms
|
||||||
|
WHERE perm_user_id = $1
|
||||||
|
AND perm_repo_uid = $2
|
||||||
|
`
|
395
store/batch2/batch_test.go
Normal file
395
store/batch2/batch_test.go
Normal file
@ -0,0 +1,395 @@
|
|||||||
|
// 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
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user