mirror of
https://github.com/kaspanet/kaspad.git
synced 2025-03-30 15:08:33 +00:00

* [DEV-81] Overwrite maxOpenFiles for testInterface to force tests to check the LRU-mechanism in openFile * [DEV-81] Added database.UseLogger test * [DEV-81] Completed coverage of reconcileDB() * [DEV-81] Added some tests for dbcache * [DEV-81] Moved init and UseLogger to separate file to make them more easily-testable + added tests * [DEV-81] Added tests for deleteFile * [DEV-81] Added tests to cursor.Delete + made sure it returns error when transaction is not writable * [DEV-81] Moved database/error_test.go from database_test package to database package + added test for IsErrorCode * [DEV-81] Added tests for handleRollback error-cases * [DEV-81] Added tests for cursor.skipPendingUpdates * [DEV-81] Added tests for various cursor edge-cases * [DEV-81] tx.putKey no longer returns error, because there is no case when it does * [DEV-81] Added tests to CreateBucket error cases * [DEV-81] Added tests to bucket.Get and .Delete error cases + .Delete now returns error on empty key * [DEV-81] Added test for ForEachBucket * [DEV-81] Added tests to StoreBlock * [DEV-81] Added test for deleting a double nested bucket * [DEV-81] Removed log_test, as it is no longer necessary with the logging system re-design * [DEV-81] Added test to some of writePendingAndCommit error-cases * [DEV-81] Update references from btcutil to btcd/util * [DEV-81] Add tests for dbCacheIterator{.Next(), .Prev(), .Key, .Value()} in cases when iterator is exhausted * [DEV-81] Added tests for ldbIterator placeholder functions * [DEV-81] Added test name to Error messsages in TestSkipPendingUpdates * [DEV-81] Begin writing TestSkipPendingUpdatesCache * [DEV-81] Added error-cases for DBCache.flush() and DBCache.commitTreaps() * [DEV-81] Use monkey.patch from bou.ke and not from github * [DEV-81] Rewrote IsErrorCode in both database and txscript packages to be more concise * [DEV-81] Rename any database.Tx to dbTx instead of tx - to remove confusion with coin Tx * [DEV-81] Fix typo * [DEV-81] Use os.TempDir() instead of /tmp/ to be cross-platform * [DEV-81] use SimNet for database tests + Error if testDB exists after deleting it * [DEV-81] Removed useLogger - it's redundant * [DEV-81] Added comment on how CRC32 checksums are calculated in reconcile_test.go * [DEV-81] Added comment that explains what setWriteRow does * [DEV-81] Use constant instead of hard-coded value * [DEV-81] Fixed some typo's + better formatting
291 lines
8.2 KiB
Go
291 lines
8.2 KiB
Go
// Copyright (c) 2015-2016 The btcsuite developers
|
|
// Use of this source code is governed by an ISC
|
|
// license that can be found in the LICENSE file.
|
|
|
|
package ffldb_test
|
|
|
|
import (
|
|
"fmt"
|
|
"os"
|
|
"path/filepath"
|
|
"reflect"
|
|
"runtime"
|
|
"testing"
|
|
|
|
"github.com/daglabs/btcd/dagconfig"
|
|
"github.com/daglabs/btcd/database"
|
|
"github.com/daglabs/btcd/database/ffldb"
|
|
"github.com/daglabs/btcd/util"
|
|
)
|
|
|
|
// dbType is the database type name for this driver.
|
|
const dbType = "ffldb"
|
|
|
|
// TestCreateOpenFail ensures that errors related to creating and opening a
|
|
// database are handled properly.
|
|
func TestCreateOpenFail(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
// Ensure that attempting to open a database that doesn't exist returns
|
|
// the expected error.
|
|
wantErrCode := database.ErrDbDoesNotExist
|
|
_, err := database.Open(dbType, "noexist", blockDataNet)
|
|
if !checkDbError(t, "Open", err, wantErrCode) {
|
|
return
|
|
}
|
|
|
|
// Ensure that attempting to open a database with the wrong number of
|
|
// parameters returns the expected error.
|
|
wantErr := fmt.Errorf("invalid arguments to %s.Open -- expected "+
|
|
"database path and block network", dbType)
|
|
_, err = database.Open(dbType, 1, 2, 3)
|
|
if err.Error() != wantErr.Error() {
|
|
t.Errorf("Open: did not receive expected error - got %v, "+
|
|
"want %v", err, wantErr)
|
|
return
|
|
}
|
|
|
|
// Ensure that attempting to open a database with an invalid type for
|
|
// the first parameter returns the expected error.
|
|
wantErr = fmt.Errorf("first argument to %s.Open is invalid -- "+
|
|
"expected database path string", dbType)
|
|
_, err = database.Open(dbType, 1, blockDataNet)
|
|
if err.Error() != wantErr.Error() {
|
|
t.Errorf("Open: did not receive expected error - got %v, "+
|
|
"want %v", err, wantErr)
|
|
return
|
|
}
|
|
|
|
// Ensure that attempting to open a database with an invalid type for
|
|
// the second parameter returns the expected error.
|
|
wantErr = fmt.Errorf("second argument to %s.Open is invalid -- "+
|
|
"expected block network", dbType)
|
|
_, err = database.Open(dbType, "noexist", "invalid")
|
|
if err.Error() != wantErr.Error() {
|
|
t.Errorf("Open: did not receive expected error - got %v, "+
|
|
"want %v", err, wantErr)
|
|
return
|
|
}
|
|
|
|
// Ensure that attempting to create a database with the wrong number of
|
|
// parameters returns the expected error.
|
|
wantErr = fmt.Errorf("invalid arguments to %s.Create -- expected "+
|
|
"database path and block network", dbType)
|
|
_, err = database.Create(dbType, 1, 2, 3)
|
|
if err.Error() != wantErr.Error() {
|
|
t.Errorf("Create: did not receive expected error - got %v, "+
|
|
"want %v", err, wantErr)
|
|
return
|
|
}
|
|
|
|
// Ensure that attempting to create a database with an invalid type for
|
|
// the first parameter returns the expected error.
|
|
wantErr = fmt.Errorf("first argument to %s.Create is invalid -- "+
|
|
"expected database path string", dbType)
|
|
_, err = database.Create(dbType, 1, blockDataNet)
|
|
if err.Error() != wantErr.Error() {
|
|
t.Errorf("Create: did not receive expected error - got %v, "+
|
|
"want %v", err, wantErr)
|
|
return
|
|
}
|
|
|
|
// Ensure that attempting to create a database with an invalid type for
|
|
// the second parameter returns the expected error.
|
|
wantErr = fmt.Errorf("second argument to %s.Create is invalid -- "+
|
|
"expected block network", dbType)
|
|
_, err = database.Create(dbType, "noexist", "invalid")
|
|
if err.Error() != wantErr.Error() {
|
|
t.Errorf("Create: did not receive expected error - got %v, "+
|
|
"want %v", err, wantErr)
|
|
return
|
|
}
|
|
|
|
// Ensure operations against a closed database return the expected
|
|
// error.
|
|
dbPath := filepath.Join(os.TempDir(), "ffldb-createfail")
|
|
_ = os.RemoveAll(dbPath)
|
|
db, err := database.Create(dbType, dbPath, blockDataNet)
|
|
if err != nil {
|
|
t.Errorf("Create: unexpected error: %v", err)
|
|
return
|
|
}
|
|
defer os.RemoveAll(dbPath)
|
|
db.Close()
|
|
|
|
wantErrCode = database.ErrDbNotOpen
|
|
err = db.View(func(dbTx database.Tx) error {
|
|
return nil
|
|
})
|
|
if !checkDbError(t, "View", err, wantErrCode) {
|
|
return
|
|
}
|
|
|
|
wantErrCode = database.ErrDbNotOpen
|
|
err = db.Update(func(dbTx database.Tx) error {
|
|
return nil
|
|
})
|
|
if !checkDbError(t, "Update", err, wantErrCode) {
|
|
return
|
|
}
|
|
|
|
wantErrCode = database.ErrDbNotOpen
|
|
_, err = db.Begin(false)
|
|
if !checkDbError(t, "Begin(false)", err, wantErrCode) {
|
|
return
|
|
}
|
|
|
|
wantErrCode = database.ErrDbNotOpen
|
|
_, err = db.Begin(true)
|
|
if !checkDbError(t, "Begin(true)", err, wantErrCode) {
|
|
return
|
|
}
|
|
|
|
wantErrCode = database.ErrDbNotOpen
|
|
err = db.Close()
|
|
if !checkDbError(t, "Close", err, wantErrCode) {
|
|
return
|
|
}
|
|
}
|
|
|
|
// TestPersistence ensures that values stored are still valid after closing and
|
|
// reopening the database.
|
|
func TestPersistence(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
// Create a new database to run tests against.
|
|
dbPath := filepath.Join(os.TempDir(), "ffldb-persistencetest")
|
|
_ = os.RemoveAll(dbPath)
|
|
db, err := database.Create(dbType, dbPath, blockDataNet)
|
|
if err != nil {
|
|
t.Errorf("Failed to create test database (%s) %v", dbType, err)
|
|
return
|
|
}
|
|
defer os.RemoveAll(dbPath)
|
|
defer db.Close()
|
|
|
|
// Create a bucket, put some values into it, and store a block so they
|
|
// can be tested for existence on re-open.
|
|
bucket1Key := []byte("bucket1")
|
|
storeValues := map[string]string{
|
|
"b1key1": "foo1",
|
|
"b1key2": "foo2",
|
|
"b1key3": "foo3",
|
|
}
|
|
genesisBlock := util.NewBlock(dagconfig.MainNetParams.GenesisBlock)
|
|
genesisHash := dagconfig.MainNetParams.GenesisHash
|
|
err = db.Update(func(dbTx database.Tx) error {
|
|
metadataBucket := dbTx.Metadata()
|
|
if metadataBucket == nil {
|
|
return fmt.Errorf("Metadata: unexpected nil bucket")
|
|
}
|
|
|
|
bucket1, err := metadataBucket.CreateBucket(bucket1Key)
|
|
if err != nil {
|
|
return fmt.Errorf("CreateBucket: unexpected error: %v",
|
|
err)
|
|
}
|
|
|
|
for k, v := range storeValues {
|
|
err := bucket1.Put([]byte(k), []byte(v))
|
|
if err != nil {
|
|
return fmt.Errorf("Put: unexpected error: %v",
|
|
err)
|
|
}
|
|
}
|
|
|
|
if err := dbTx.StoreBlock(genesisBlock); err != nil {
|
|
return fmt.Errorf("StoreBlock: unexpected error: %v",
|
|
err)
|
|
}
|
|
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
t.Errorf("Update: unexpected error: %v", err)
|
|
return
|
|
}
|
|
|
|
// Close and reopen the database to ensure the values persist.
|
|
db.Close()
|
|
db, err = database.Open(dbType, dbPath, blockDataNet)
|
|
if err != nil {
|
|
t.Errorf("Failed to open test database (%s) %v", dbType, err)
|
|
return
|
|
}
|
|
defer db.Close()
|
|
|
|
// Ensure the values previously stored in the 3rd namespace still exist
|
|
// and are correct.
|
|
err = db.View(func(dbTx database.Tx) error {
|
|
metadataBucket := dbTx.Metadata()
|
|
if metadataBucket == nil {
|
|
return fmt.Errorf("Metadata: unexpected nil bucket")
|
|
}
|
|
|
|
bucket1 := metadataBucket.Bucket(bucket1Key)
|
|
if bucket1 == nil {
|
|
return fmt.Errorf("Bucket1: unexpected nil bucket")
|
|
}
|
|
|
|
for k, v := range storeValues {
|
|
gotVal := bucket1.Get([]byte(k))
|
|
if !reflect.DeepEqual(gotVal, []byte(v)) {
|
|
return fmt.Errorf("Get: key '%s' does not "+
|
|
"match expected value - got %s, want %s",
|
|
k, gotVal, v)
|
|
}
|
|
}
|
|
|
|
genesisBlockBytes, _ := genesisBlock.Bytes()
|
|
gotBytes, err := dbTx.FetchBlock(genesisHash)
|
|
if err != nil {
|
|
return fmt.Errorf("FetchBlock: unexpected error: %v",
|
|
err)
|
|
}
|
|
if !reflect.DeepEqual(gotBytes, genesisBlockBytes) {
|
|
return fmt.Errorf("FetchBlock: stored block mismatch")
|
|
}
|
|
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
t.Errorf("View: unexpected error: %v", err)
|
|
return
|
|
}
|
|
}
|
|
|
|
// TestInterface performs all interfaces tests for this database driver.
|
|
func TestInterface(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
// Create a new database to run tests against.
|
|
dbPath := filepath.Join(os.TempDir(), "ffldb-interfacetest")
|
|
_ = os.RemoveAll(dbPath)
|
|
db, err := database.Create(dbType, dbPath, blockDataNet)
|
|
if err != nil {
|
|
t.Errorf("Failed to create test database (%s) %v", dbType, err)
|
|
return
|
|
}
|
|
defer os.RemoveAll(dbPath)
|
|
defer db.Close()
|
|
|
|
// Ensure the driver type is the expected value.
|
|
gotDbType := db.Type()
|
|
if gotDbType != dbType {
|
|
t.Errorf("Type: unepxected driver type - got %v, want %v",
|
|
gotDbType, dbType)
|
|
return
|
|
}
|
|
|
|
// Run all of the interface tests against the database.
|
|
runtime.GOMAXPROCS(runtime.NumCPU())
|
|
|
|
// Change the maximum file size to a small value to force multiple flat
|
|
// files with the test data set.
|
|
// Change maximum open files to small value to force shifts in the LRU
|
|
// mechanism
|
|
ffldb.TstRunWithMaxBlockFileSizeAndMaxOpenFiles(db, 2048, 10, func() {
|
|
testInterface(t, db)
|
|
})
|
|
}
|