kaspad/infrastructure/db/database/transaction_test.go
Svarog ad9c213a06
Restructure database to prevent double-slashes in keys, causing bugs in cursors (#1432)
* Add TestValidateAndInsertPruningPointWithSideBlocks

* Optimize infrastracture bucket paths

* Update infrastracture tests

* Refactor the consensus/database layer

* Remove utils/dbkeys

* Use consensus/database in consensus instead of infrastructure

* Fix a bug in dbBucketToDatabaseBucket and MakeBucket combination

Co-authored-by: Elichai Turkel <elichai.turkel@gmail.com>
Co-authored-by: stasatdaglabs <39559713+stasatdaglabs@users.noreply.github.com>
2021-01-19 14:19:08 +02:00

497 lines
13 KiB
Go

// All tests within this file should call testForAllDatabaseTypes
// over the actual test. This is to make sure that all supported
// database types adhere to the assumptions defined in the
// interfaces in this package.
package database_test
import (
"bytes"
"strings"
"testing"
"github.com/kaspanet/kaspad/infrastructure/db/database"
)
func TestTransactionPut(t *testing.T) {
testForAllDatabaseTypes(t, "TestTransactionPut", testTransactionPut)
}
func testTransactionPut(t *testing.T, db database.Database, testName string) {
// Begin a new transaction
dbTx, err := db.Begin()
if err != nil {
t.Fatalf("%s: Begin "+
"unexpectedly failed: %s", testName, err)
}
defer func() {
err := dbTx.RollbackUnlessClosed()
if err != nil {
t.Fatalf("%s: RollbackUnlessClosed "+
"unexpectedly failed: %s", testName, err)
}
}()
// Put value1 into the transaction
key := database.MakeBucket(nil).Key([]byte("key"))
value1 := []byte("value1")
err = dbTx.Put(key, value1)
if err != nil {
t.Fatalf("%s: Put "+
"unexpectedly failed: %s", testName, err)
}
// Put value2 into the transaction with the same key
value2 := []byte("value2")
err = dbTx.Put(key, value2)
if err != nil {
t.Fatalf("%s: Put "+
"unexpectedly failed: %s", testName, err)
}
// Commit the transaction
err = dbTx.Commit()
if err != nil {
t.Fatalf("%s: Commit "+
"unexpectedly failed: %s", testName, err)
}
// Make sure that the returned value is value2
returnedValue, err := db.Get(key)
if err != nil {
t.Fatalf("%s: Get "+
"unexpectedly failed: %s", testName, err)
}
if !bytes.Equal(returnedValue, value2) {
t.Fatalf("%s: Get "+
"returned wrong value. Want: %s, got: %s",
testName, string(value2), string(returnedValue))
}
}
func TestTransactionGet(t *testing.T) {
testForAllDatabaseTypes(t, "TestTransactionGet", testTransactionGet)
}
func testTransactionGet(t *testing.T, db database.Database, testName string) {
// Put a value into the database
key1 := database.MakeBucket(nil).Key([]byte("key1"))
value1 := []byte("value1")
err := db.Put(key1, value1)
if err != nil {
t.Fatalf("%s: Put "+
"unexpectedly failed: %s", testName, err)
}
// Begin a new transaction
dbTx, err := db.Begin()
if err != nil {
t.Fatalf("%s: Begin "+
"unexpectedly failed: %s", testName, err)
}
defer func() {
err := dbTx.RollbackUnlessClosed()
if err != nil {
t.Fatalf("%s: RollbackUnlessClosed "+
"unexpectedly failed: %s", testName, err)
}
}()
// Get the value back and make sure it's the same one
returnedValue, err := dbTx.Get(key1)
if err != nil {
t.Fatalf("%s: Get "+
"unexpectedly failed: %s", testName, err)
}
if !bytes.Equal(returnedValue, value1) {
t.Fatalf("%s: Get "+
"returned wrong value. Want: %s, got: %s",
testName, string(value1), string(returnedValue))
}
// Try getting a non-existent value and make sure
// the returned error is ErrNotFound
_, err = dbTx.Get(database.MakeBucket(nil).Key([]byte("doesn't exist")))
if err == nil {
t.Fatalf("%s: Get "+
"unexpectedly succeeded", testName)
}
if !database.IsNotFoundError(err) {
t.Fatalf("%s: Get "+
"returned wrong error: %s", testName, err)
}
// Put a new value into the database outside of the transaction
key2 := database.MakeBucket(nil).Key([]byte("key2"))
value2 := []byte("value2")
err = db.Put(key2, value2)
if err != nil {
t.Fatalf("%s: Put "+
"unexpectedly failed: %s", testName, err)
}
// Make sure that the new value exists inside the transaction
newValue2, err := dbTx.Get(key2)
if err != nil {
t.Fatalf("%s: Get "+
"unexpectedly failed: %v", testName, err)
}
if !bytes.Equal(value2, newValue2) {
t.Fatalf("Expected %x and %x to be the same", value2, newValue2)
}
// Put a new value into the transaction
key3 := database.MakeBucket(nil).Key([]byte("key3"))
value3 := []byte("value3")
err = dbTx.Put(key3, value3)
if err != nil {
t.Fatalf("%s: Put "+
"unexpectedly failed: %s", testName, err)
}
// Make sure that the new value doesn't exist outside the transaction
_, err = db.Get(key3)
if err == nil {
t.Fatalf("%s: Get "+
"unexpectedly succeeded", testName)
}
if !database.IsNotFoundError(err) {
t.Fatalf("%s: Get "+
"returned wrong error: %s", testName, err)
}
}
func TestTransactionHas(t *testing.T) {
testForAllDatabaseTypes(t, "TestTransactionHas", testTransactionHas)
}
func testTransactionHas(t *testing.T, db database.Database, testName string) {
// Put a value into the database
key1 := database.MakeBucket(nil).Key([]byte("key1"))
value1 := []byte("value1")
err := db.Put(key1, value1)
if err != nil {
t.Fatalf("%s: Put "+
"unexpectedly failed: %s", testName, err)
}
// Begin a new transaction
dbTx, err := db.Begin()
if err != nil {
t.Fatalf("%s: Begin "+
"unexpectedly failed: %s", testName, err)
}
defer func() {
err := dbTx.RollbackUnlessClosed()
if err != nil {
t.Fatalf("%s: RollbackUnlessClosed "+
"unexpectedly failed: %s", testName, err)
}
}()
// Make sure that Has returns true for the value we just put
exists, err := dbTx.Has(key1)
if err != nil {
t.Fatalf("%s: Has "+
"unexpectedly failed: %s", testName, err)
}
if !exists {
t.Fatalf("%s: Has "+
"unexpectedly returned that the value does not exist", testName)
}
// Make sure that Has returns false for a non-existent value
exists, err = dbTx.Has(database.MakeBucket(nil).Key([]byte("doesn't exist")))
if err != nil {
t.Fatalf("%s: Has "+
"unexpectedly failed: %s", testName, err)
}
if exists {
t.Fatalf("%s: Has "+
"unexpectedly returned that the value exists", testName)
}
// Put a new value into the database outside of the transaction
key2 := database.MakeBucket(nil).Key([]byte("key2"))
value2 := []byte("value2")
err = db.Put(key2, value2)
if err != nil {
t.Fatalf("%s: Put "+
"unexpectedly failed: %s", testName, err)
}
// Make sure that the new value exists inside the transaction
exists, err = dbTx.Has(key2)
if err != nil {
t.Fatalf("%s: Has "+
"unexpectedly failed: %s", testName, err)
}
if !exists {
t.Fatalf("%s: Has "+
"unexpectedly returned that the value doesn't exists", testName)
}
}
func TestTransactionDelete(t *testing.T) {
testForAllDatabaseTypes(t, "TestTransactionDelete", testTransactionDelete)
}
func testTransactionDelete(t *testing.T, db database.Database, testName string) {
// Put a value into the database
key := database.MakeBucket(nil).Key([]byte("key"))
value := []byte("value")
err := db.Put(key, value)
if err != nil {
t.Fatalf("%s: Put "+
"unexpectedly failed: %s", testName, err)
}
// Begin two new transactions
dbTx1, err := db.Begin()
if err != nil {
t.Fatalf("%s: Begin "+
"unexpectedly failed: %s", testName, err)
}
dbTx2, err := db.Begin()
if err != nil {
t.Fatalf("%s: Begin "+
"unexpectedly failed: %s", testName, err)
}
defer func() {
err := dbTx1.RollbackUnlessClosed()
if err != nil {
t.Fatalf("%s: RollbackUnlessClosed "+
"unexpectedly failed: %s", testName, err)
}
err = dbTx2.RollbackUnlessClosed()
if err != nil {
t.Fatalf("%s: RollbackUnlessClosed "+
"unexpectedly failed: %s", testName, err)
}
}()
// Delete the value in the first transaction
err = dbTx1.Delete(key)
if err != nil {
t.Fatalf("%s: Delete "+
"unexpectedly failed: %s", testName, err)
}
// Commit the first transaction
err = dbTx1.Commit()
if err != nil {
t.Fatalf("%s: Commit "+
"unexpectedly failed: %s", testName, err)
}
// Make sure that Has returns false for the deleted value
exists, err := db.Has(key)
if err != nil {
t.Fatalf("%s: Has "+
"unexpectedly failed: %s", testName, err)
}
if exists {
t.Fatalf("%s: Has "+
"unexpectedly returned that the value exists", testName)
}
// Make sure that the second transaction is also affected
exists, err = dbTx2.Has(key)
if err != nil {
t.Fatalf("%s: Has "+
"unexpectedly failed: %s", testName, err)
}
if exists {
t.Fatalf("%s: Has "+
"unexpectedly returned that the value exists", testName)
}
}
func TestTransactionCommit(t *testing.T) {
testForAllDatabaseTypes(t, "TestTransactionCommit", testTransactionCommit)
}
func testTransactionCommit(t *testing.T, db database.Database, testName string) {
// Begin a new transaction
dbTx, err := db.Begin()
if err != nil {
t.Fatalf("%s: Begin "+
"unexpectedly failed: %s", testName, err)
}
defer func() {
err := dbTx.RollbackUnlessClosed()
if err != nil {
t.Fatalf("%s: RollbackUnlessClosed "+
"unexpectedly failed: %s", testName, err)
}
}()
// Put a value into the transaction
key := database.MakeBucket(nil).Key([]byte("key"))
value := []byte("value")
err = dbTx.Put(key, value)
if err != nil {
t.Fatalf("%s: Put "+
"unexpectedly failed: %s", testName, err)
}
// Commit the transaction
err = dbTx.Commit()
if err != nil {
t.Fatalf("%s: Commit "+
"unexpectedly failed: %s", testName, err)
}
// Make sure that the returned value exists and is as expected
returnedValue, err := db.Get(key)
if err != nil {
t.Fatalf("%s: Get "+
"unexpectedly failed: %s", testName, err)
}
if !bytes.Equal(returnedValue, value) {
t.Fatalf("%s: Get "+
"returned wrong value. Want: %s, got: %s",
testName, string(value), string(returnedValue))
}
// Make sure that further operations on the transaction return an error
_, err = dbTx.Get(key)
if err == nil {
t.Fatalf("%s: Get "+
"unexpectedly succeeded", testName)
}
expectedError := "closed transaction"
if !strings.Contains(err.Error(), expectedError) {
t.Fatalf("%s: Get "+
"returned wrong error. Want: %s, got: %s",
testName, expectedError, err)
}
}
func TestTransactionRollback(t *testing.T) {
testForAllDatabaseTypes(t, "TestTransactionRollback", testTransactionRollback)
}
func testTransactionRollback(t *testing.T, db database.Database, testName string) {
// Begin a new transaction
dbTx, err := db.Begin()
if err != nil {
t.Fatalf("%s: Begin "+
"unexpectedly failed: %s", testName, err)
}
defer func() {
err := dbTx.RollbackUnlessClosed()
if err != nil {
t.Fatalf("%s: RollbackUnlessClosed "+
"unexpectedly failed: %s", testName, err)
}
}()
// Put a value into the transaction
key := database.MakeBucket(nil).Key([]byte("key"))
value := []byte("value")
err = dbTx.Put(key, value)
if err != nil {
t.Fatalf("%s: Put "+
"unexpectedly failed: %s", testName, err)
}
// Rollback the transaction
err = dbTx.Rollback()
if err != nil {
t.Fatalf("%s: Rollback "+
"unexpectedly failed: %s", testName, err)
}
// Make sure that the returned value did not get added to the database
_, err = db.Get(key)
if err == nil {
t.Fatalf("%s: Get "+
"unexpectedly succeeded", testName)
}
if !database.IsNotFoundError(err) {
t.Fatalf("%s: Get "+
"returned wrong error", testName)
}
// Make sure that further operations on the transaction return an error
_, err = dbTx.Get(key)
if err == nil {
t.Fatalf("%s: Get "+
"unexpectedly succeeded", testName)
}
expectedError := "closed transaction"
if !strings.Contains(err.Error(), expectedError) {
t.Fatalf("%s: Get "+
"returned wrong error. Want: %s, got: %s",
testName, expectedError, err)
}
}
func TestTransactionRollbackUnlessClosed(t *testing.T) {
testForAllDatabaseTypes(t, "TestTransactionRollbackUnlessClosed", testTransactionRollbackUnlessClosed)
}
func testTransactionRollbackUnlessClosed(t *testing.T, db database.Database, testName string) {
// Begin a new transaction
dbTx, err := db.Begin()
if err != nil {
t.Fatalf("%s: Begin "+
"unexpectedly failed: %s", testName, err)
}
defer func() {
err := dbTx.RollbackUnlessClosed()
if err != nil {
t.Fatalf("%s: RollbackUnlessClosed "+
"unexpectedly failed: %s", testName, err)
}
}()
// Put a value into the transaction
key := database.MakeBucket(nil).Key([]byte("key"))
value := []byte("value")
err = dbTx.Put(key, value)
if err != nil {
t.Fatalf("%s: Put "+
"unexpectedly failed: %s", testName, err)
}
// RollbackUnlessClosed the transaction
err = dbTx.RollbackUnlessClosed()
if err != nil {
t.Fatalf("%s: RollbackUnlessClosed "+
"unexpectedly failed: %s", testName, err)
}
// Make sure that the returned value did not get added to the database
_, err = db.Get(key)
if err == nil {
t.Fatalf("%s: Get "+
"unexpectedly succeeded", testName)
}
if !database.IsNotFoundError(err) {
t.Fatalf("%s: Get "+
"returned wrong error", testName)
}
// Make sure that further operations on the transaction return an error
_, err = dbTx.Get(key)
if err == nil {
t.Fatalf("%s: Get "+
"unexpectedly succeeded", testName)
}
expectedError := "closed transaction"
if !strings.Contains(err.Error(), expectedError) {
t.Fatalf("%s: Get "+
"returned wrong error. Want: %s, got: %s",
testName, expectedError, err)
}
// Make sure that further calls to RollbackUnlessClosed don't return an error
err = dbTx.RollbackUnlessClosed()
if err != nil {
t.Fatalf("%s: RollbackUnlessClosed "+
"unexpectedly failed: %s", testName, err)
}
}