ShareDAV/store_test.go

314 lines
9.0 KiB
Go

package main
import (
"testing"
uuid "github.com/satori/go.uuid"
"github.com/tidwall/buntdb"
)
func TestStoreUserHandling(t *testing.T) {
store, err := NewDBStore(":memory:")
if err != nil {
t.Fatalf("cannot create store: %v", err)
}
defer store.Close()
t.Run("store should be empty initially", func(t *testing.T) {
users, err := store.GetUsers()
if err != nil {
t.Errorf("no error should have been returned: %v", err)
}
if len(users) != 0 {
t.Errorf("there should be no users")
}
})
t.Run("user that doesn't exist should return error", func(t *testing.T) {
_, err := store.GetUser("someuser")
if err != ErrUserNotFound {
t.Errorf("unexpected error: %v", err)
}
})
t.Run("adding users should work", func(t *testing.T) {
if err := store.AddUser(User{Username: "myuser", Password: "mypass", Role: GlobalRoleUser}); err != nil {
t.Errorf("cannot add user: %v", err)
}
t.Run("retrieving that single user should work", func(t *testing.T) {
user, err := store.GetUser("myuser")
if err != nil {
t.Errorf("cannot retrieve user: %v", err)
}
if user.Username != "myuser" {
t.Errorf("retrieved user contains unexpected username")
}
if user.Role != GlobalRoleUser {
t.Errorf("retrieved user contains unexpected role")
}
})
t.Run("retrieving multiple users should work", func(t *testing.T) {
users, err := store.GetUsers()
if err != nil {
t.Errorf("cannot retrieve user list: %v", err)
}
if len(users) != 1 {
t.Errorf("there should be only one user")
}
if users[0].Username != "myuser" {
t.Errorf("retrieved user contains unexpected username")
}
if users[0].Role != GlobalRoleUser {
t.Errorf("retrieved user contains unexpected role")
}
})
t.Run("deleting that user should work", func(t *testing.T) {
err := store.RemoveUser("myuser")
if err != nil {
t.Errorf("cannot delete user: %v", err)
}
t.Run("user should no longer be found", func(t *testing.T) {
_, err := store.GetUser("myuser")
if err != ErrUserNotFound {
t.Errorf("unexpected error: %v", err)
}
})
t.Run("user should no longer be listed", func(t *testing.T) {
users, err := store.GetUsers()
if err != nil {
t.Errorf("unexpected error: %v", err)
}
if len(users) != 0 {
t.Errorf("there should be no users")
}
})
})
})
t.Run("database should be empty now", func(t *testing.T) {
// checks that we properly deleted all keys
if err := store.db.View(func(tx *buntdb.Tx) error {
return tx.Ascend("", func(key, value string) bool {
t.Errorf("there should be no keys left")
return false
})
}); err != nil {
t.Errorf("iterating keys failed: %v", err)
}
})
}
func TestStoreShareHandling(t *testing.T) {
store, err := NewDBStore(":memory:")
if err != nil {
t.Fatalf("cannot create store: %v", err)
}
defer store.Close()
t.Run("store should be empty initially", func(t *testing.T) {
shares, err := store.GetShares()
if err != nil {
t.Errorf("no error should have been returned: %v", err)
}
if len(shares) != 0 {
t.Errorf("there should be no shares")
}
})
t.Run("creating a share should work", func(t *testing.T) {
share, err := store.CreateShare()
if err != nil {
t.Errorf("error creating share: %v", err)
return
}
var emptyUUID uuid.UUID
if share.UUID == emptyUUID {
t.Errorf("UUID is empty")
return
}
if share.Name != "" || share.Description != "" {
t.Errorf("share should not have attributes set (yet)")
}
t.Run("share attributes can be set", func(t *testing.T) {
share.Name = "a name"
share.Description = "some desc"
if err := store.UpdateShareAttributes(share); err != nil {
t.Errorf("cannot set attributes of share: %v", err)
}
})
t.Run("cannot set attributes of unknown share", func(t *testing.T) {
otherShare := Share{UUID: uuid.NewV4()}
otherShare.Name = "foo"
otherShare.Description = "bar"
err := store.UpdateShareAttributes(otherShare)
if err == nil {
t.Errorf("an error should have been returned")
} else if err != ErrShareNotFound {
t.Errorf("wrong error has been returned: %v", err)
}
})
t.Run("share can be listed", func(t *testing.T) {
shares, err := store.GetShares()
if err != nil {
t.Errorf("error getting shares: %v", err)
return
}
if len(shares) != 1 {
t.Errorf("invalid number of shares: %d", len(shares))
return
}
if shares[0].UUID != share.UUID {
t.Errorf("unexpected uuid")
}
if shares[0].Name != "a name" {
t.Errorf("unexpected name")
}
if shares[0].Description != "some desc" {
t.Errorf("unexpected description")
}
})
t.Run("share can be removed", func(t *testing.T) {
if err := store.RemoveShare(share.UUID); err != nil {
t.Errorf("removing share failed: %v", err)
}
})
})
t.Run("share access should work", func(t *testing.T) {
share1, _ := store.CreateShare()
share2, _ := store.CreateShare()
share3, _ := store.CreateShare()
user1 := User{Username: "user1"}
user2 := User{Username: "user2"}
_ = store.AddUser(user1)
_ = store.AddUser(user2)
defer func() {
_ = store.RemoveUser(user1.Username)
_ = store.RemoveUser(user2.Username)
}()
t.Run("multiple shares should exist", func(t *testing.T) {
shares, _ := store.GetShares()
if len(shares) != 3 {
t.Errorf("3 shares should exist")
}
})
t.Run("can add users to shares", func(t *testing.T) {
if err := store.AddUserToShare(share1, user1.Username, ShareRoleAdmin); err != nil {
t.Errorf("cannot add user1 to share1: %v", err)
}
if err := store.AddUserToShare(share1, user2.Username, ShareRoleReader); err != nil {
t.Errorf("cannot add user2 to share1: %v", err)
}
if err := store.AddUserToShare(share2, user2.Username, ShareRoleAdmin); err != nil {
t.Errorf("cannot add user2 to share2: %v", err)
}
t.Run("cannot add login if user doesn't exist", func(t *testing.T) {
if err := store.AddLogin(share3, user1.Username, Login{"foo", "", false}); err == nil {
t.Errorf("an error should have been returned")
} else if err != ErrUserNotFound {
t.Errorf("wrong error has been returned: %v", err)
}
})
t.Run("logins can be added", func(t *testing.T) {
login1 := Login{LoginName: "login1"}
login2 := Login{LoginName: "login2"}
login3 := Login{LoginName: "login3"}
if err := store.AddLogin(share1, user1.Username, login1); err != nil {
t.Errorf("adding login returned error: %v", err)
}
if err := store.AddLogin(share1, user2.Username, login2); err != nil {
t.Errorf("adding login returned error: %v", err)
}
if err := store.AddLogin(share1, user2.Username, login3); err != nil {
t.Errorf("adding login returned error: %v", err)
}
t.Run("duplicate login not allowed", func(t *testing.T) {
// Different share, but same user:login pair as above. Must be a share where
// that user is already assigned, though.
if err := store.AddLogin(share2, user2.Username, login2); err != ErrLoginDuplicate {
t.Errorf("unexpected error: %v", err)
}
})
t.Run("share is found by login", func(t *testing.T) {
share, err := store.FindShareByLogin(user1.Username, login1.LoginName)
if err != nil {
t.Errorf("unexpected error: %v", err)
}
if share.UUID != share1.UUID {
t.Errorf("wrong share returned")
}
})
t.Run("unknown login/share combination returns error", func(t *testing.T) {
if _, err := store.FindShareByLogin(user1.Username, login3.LoginName); err != ErrShareNotFound {
t.Errorf("unexpected error: %v", err)
}
})
t.Run("login can be removed", func(t *testing.T) {
if err := store.RemoveLogin(share1, user1.Username, login1.LoginName); err != nil {
t.Errorf("unexpected error: %v", err)
}
if _, err := store.FindShareByLogin(user1.Username, login1.LoginName); err != ErrShareNotFound {
t.Errorf("share should not be found now, but returned: %v", err)
}
})
t.Run("user can be removed", func(t *testing.T) {
if err := store.RemoveUserFromShare(share1, user2.Username); err != nil {
t.Errorf("unexpected error: %v", err)
}
if _, err := store.FindShareByLogin(user2.Username, login2.LoginName); err != ErrShareNotFound {
t.Errorf("share should not be found now, but returned: %v", err)
}
})
})
})
t.Run("can remove shares", func(t *testing.T) {
if err := store.RemoveShare(share1.UUID); err != nil {
t.Errorf("cannot remove share1: %v", err)
}
if err := store.RemoveShare(share2.UUID); err != nil {
t.Errorf("cannot remove share2: %v", err)
}
if err := store.RemoveShare(share3.UUID); err != nil {
t.Errorf("cannot remove share3: %v", err)
}
})
})
t.Run("database should be empty now", func(t *testing.T) {
// checks that we properly deleted all keys
if err := store.db.View(func(tx *buntdb.Tx) error {
return tx.Ascend("", func(key, value string) bool {
t.Errorf("leftover key found: %v", key)
return true
})
}); err != nil {
t.Errorf("iterating keys failed: %v", err)
}
})
}