mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-20 09:21:11 +00:00
635 lines
12 KiB
Go
635 lines
12 KiB
Go
package mdbx
|
|
|
|
import (
|
|
"fmt"
|
|
"io/ioutil"
|
|
"os"
|
|
"strings"
|
|
"syscall"
|
|
"testing"
|
|
)
|
|
|
|
func TestEnv_Path_notOpen(t *testing.T) {
|
|
env, err := NewEnv()
|
|
if err != nil {
|
|
t.Fatalf("create: %v", err)
|
|
}
|
|
defer env.Close()
|
|
|
|
// before Open the Path method returns "" and a non-nil error.
|
|
path, err := env.Path()
|
|
if err == nil {
|
|
t.Errorf("no error returned before Open")
|
|
}
|
|
if path != "" {
|
|
t.Errorf("non-zero path returned before Open")
|
|
}
|
|
}
|
|
|
|
func TestEnv_Path(t *testing.T) {
|
|
env, err := NewEnv()
|
|
if err != nil {
|
|
t.Fatalf("create: %v", err)
|
|
}
|
|
|
|
// open an environment
|
|
dir, err := ioutil.TempDir("", "mdb_test")
|
|
if err != nil {
|
|
t.Fatalf("tempdir: %v", err)
|
|
}
|
|
defer os.RemoveAll(dir)
|
|
|
|
err = env.Open(dir, 0, 0644)
|
|
defer env.Close()
|
|
if err != nil {
|
|
t.Errorf("open: %v", err)
|
|
}
|
|
path, err := env.Path()
|
|
if err != nil {
|
|
t.Errorf("path: %v", err)
|
|
}
|
|
if path != dir {
|
|
t.Errorf("path: %q (!= %q)", path, dir)
|
|
}
|
|
}
|
|
|
|
func TestEnv_Open_notExist(t *testing.T) {
|
|
env, err := NewEnv()
|
|
if err != nil {
|
|
t.Fatalf("create: %s", err)
|
|
}
|
|
defer env.Close()
|
|
|
|
// ensure that opening a non-existent path fails.
|
|
err = env.Open("/path/does/not/exist/aoeu", 0, 0664)
|
|
if !IsNotExist(err) {
|
|
t.Errorf("open: %v", err)
|
|
}
|
|
}
|
|
|
|
func TestEnv_Open(t *testing.T) {
|
|
env, err1 := NewEnv()
|
|
if err1 != nil {
|
|
t.Error(err1)
|
|
return
|
|
}
|
|
defer func() {
|
|
if err := env.Close(); err != nil {
|
|
t.Error(err)
|
|
}
|
|
}()
|
|
|
|
// open an environment at a temporary path.
|
|
path, err := ioutil.TempDir("", "mdb_test")
|
|
if err != nil {
|
|
t.Fatalf("tempdir: %v", err)
|
|
}
|
|
defer os.RemoveAll(path)
|
|
err = env.Open(path, 0, 0664)
|
|
if err != nil {
|
|
t.Errorf("open: %s", err)
|
|
}
|
|
}
|
|
|
|
func TestEnv_FD(t *testing.T) {
|
|
env, err1 := NewEnv()
|
|
if err1 != nil {
|
|
t.Error(err1)
|
|
return
|
|
}
|
|
defer func() {
|
|
if err := env.Close(); err != nil {
|
|
t.Error(err)
|
|
}
|
|
}()
|
|
|
|
fd, err := env.FD()
|
|
if err != nil && !strings.Contains(err.Error(), "operation not permitted") {
|
|
t.Errorf("fd: %x (%v)", fd, err)
|
|
}
|
|
|
|
// open an environment at a temporary path.
|
|
path, err := ioutil.TempDir("", "mdb_test")
|
|
if err != nil {
|
|
t.Fatalf("tempdir: %v", err)
|
|
}
|
|
defer os.RemoveAll(path)
|
|
err = env.Open(path, 0, 0664)
|
|
if err != nil {
|
|
t.Errorf("open: %s", err)
|
|
}
|
|
|
|
fd, err = env.FD()
|
|
if err != nil {
|
|
t.Errorf("fd error: %v", err)
|
|
}
|
|
if fd == 0 {
|
|
t.Errorf("fd: %x", fd)
|
|
}
|
|
}
|
|
|
|
func TestEnv_Flags(t *testing.T) {
|
|
env := setup(t)
|
|
defer clean(env, t)
|
|
|
|
flags, err := env.Flags()
|
|
if err != nil {
|
|
t.Error(err)
|
|
return
|
|
}
|
|
|
|
if flags&NoTLS == 0 {
|
|
t.Errorf("NoTLS is not set")
|
|
}
|
|
if flags&SafeNoSync != 0 {
|
|
t.Errorf("NoSync is set")
|
|
}
|
|
|
|
err = env.SetFlags(SafeNoSync)
|
|
if err != nil {
|
|
t.Error(err)
|
|
}
|
|
|
|
flags, err = env.Flags()
|
|
if err != nil {
|
|
t.Error(err)
|
|
}
|
|
if flags&SafeNoSync == 0 {
|
|
t.Error("NoSync is not set")
|
|
}
|
|
|
|
err = env.UnsetFlags(SafeNoSync)
|
|
if err != nil {
|
|
t.Error(err)
|
|
}
|
|
|
|
flags, err = env.Flags()
|
|
if err != nil {
|
|
t.Error(err)
|
|
}
|
|
if flags&SafeNoSync != 0 {
|
|
t.Error("NoSync is set")
|
|
}
|
|
}
|
|
|
|
func TestEnv_SetMaxReader(t *testing.T) {
|
|
dir, err := ioutil.TempDir("", "test-env-setmaxreaders-")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer os.RemoveAll(dir)
|
|
|
|
env, err := NewEnv()
|
|
if err != nil {
|
|
t.Error(err)
|
|
}
|
|
|
|
maxreaders := 246
|
|
err = env.SetMaxReaders(maxreaders)
|
|
if err != nil {
|
|
t.Error(err)
|
|
}
|
|
_maxreaders, err := env.MaxReaders()
|
|
if err != nil {
|
|
t.Error(err)
|
|
}
|
|
if _maxreaders < maxreaders {
|
|
t.Errorf("unexpected MaxReaders: %v (< %v)", _maxreaders, maxreaders)
|
|
}
|
|
|
|
err = env.Open(dir, 0, 0644)
|
|
defer env.Close()
|
|
if err != nil {
|
|
env.Close()
|
|
t.Error(err)
|
|
}
|
|
|
|
err = env.SetMaxReaders(126)
|
|
if !IsErrnoSys(err, syscall.EPERM) {
|
|
t.Errorf("unexpected error: %v (!= %v)", err, syscall.EPERM)
|
|
}
|
|
_maxreaders, err = env.MaxReaders()
|
|
if err != nil {
|
|
t.Error(err)
|
|
}
|
|
if _maxreaders < maxreaders {
|
|
t.Errorf("unexpected MaxReaders: %v (!= %v)", _maxreaders, maxreaders)
|
|
}
|
|
}
|
|
|
|
func TestEnv_SetDebug(t *testing.T) {
|
|
dir, err := ioutil.TempDir("", "test-env-setmdebug-")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer os.RemoveAll(dir)
|
|
|
|
env, err := NewEnv()
|
|
if err != nil {
|
|
t.Error(err)
|
|
}
|
|
|
|
err = env.SetDebug(LogLvlDoNotChange, DbgLegacyTxOverlap, LoggerDoNotChange)
|
|
if err != nil {
|
|
t.Error(err)
|
|
}
|
|
}
|
|
|
|
//func TestEnv_SetMapSize(t *testing.T) {
|
|
// env := setup(t)
|
|
// defer clean(env, t)
|
|
//
|
|
// const minsize = 100 << 20 // 100MB
|
|
// err := env.SetMapSize(minsize)
|
|
// if err != nil {
|
|
// t.Error(err)
|
|
// }
|
|
//
|
|
// err = env.Update(func(txn *Txn) (err error) {
|
|
// return nil
|
|
// })
|
|
// if err != nil {
|
|
// t.Error(err)
|
|
// }
|
|
//
|
|
// info, err := env.Info()
|
|
// if err != nil {
|
|
// t.Error(err)
|
|
// } else if info.MapSize < minsize {
|
|
// t.Errorf("unexpected mapsize: %v (< %v)", info.MapSize, minsize)
|
|
// }
|
|
//}
|
|
|
|
//func TestEnv_ReaderList(t *testing.T) {
|
|
// env := setup(t)
|
|
// defer clean(env, t)
|
|
//
|
|
// var numreaders = 2
|
|
//
|
|
// var fin sync.WaitGroup
|
|
// defer fin.Wait()
|
|
// ready := make(chan struct{})
|
|
// done := make(chan struct{})
|
|
// defer close(done)
|
|
//
|
|
// t.Logf("starting")
|
|
//
|
|
// for i := 0; i < numreaders; i++ {
|
|
// fin.Add(1)
|
|
// go func(i int) {
|
|
// defer fin.Done()
|
|
// err := env.View(func(txn *Txn) (err error) {
|
|
// t.Logf("reader %v: ready", i)
|
|
// ready <- struct{}{}
|
|
//
|
|
// <-done
|
|
// t.Logf("reader %v: done", i)
|
|
// return nil
|
|
// })
|
|
// if err != nil {
|
|
// t.Errorf("reader %d: %q", i, err)
|
|
// }
|
|
// }(i)
|
|
//
|
|
// // wait for each reader to become ready
|
|
// <-ready
|
|
// }
|
|
//
|
|
// var readers []string
|
|
// _ = env.ReaderList(func(msg string) error {
|
|
// t.Logf("reader: %q", msg)
|
|
// readers = append(readers, msg)
|
|
// return nil
|
|
// })
|
|
// if len(readers) != numreaders+1 {
|
|
// t.Errorf("unexpected reader list size: %d (!= %d)", len(readers), numreaders)
|
|
// }
|
|
//}
|
|
|
|
//func TestEnv_ReaderList_error(t *testing.T) {
|
|
// env := setup(t)
|
|
// defer clean(env, t)
|
|
//
|
|
// var numreaders = 2
|
|
//
|
|
// var fin sync.WaitGroup
|
|
// defer fin.Wait()
|
|
// ready := make(chan struct{})
|
|
// done := make(chan struct{})
|
|
// defer close(done)
|
|
//
|
|
// t.Logf("starting")
|
|
//
|
|
// for i := 0; i < numreaders; i++ {
|
|
// fin.Add(1)
|
|
// go func(i int) {
|
|
// defer fin.Done()
|
|
// err := env.View(func(txn *Txn) (err error) {
|
|
// t.Logf("reader %v: ready", i)
|
|
// ready <- struct{}{}
|
|
//
|
|
// <-done
|
|
// t.Logf("reader %v: done", i)
|
|
// return nil
|
|
// })
|
|
// if err != nil {
|
|
// t.Errorf("reader %d: %q", i, err)
|
|
// }
|
|
// }(i)
|
|
//
|
|
// // wait for each reader to become ready
|
|
// <-ready
|
|
// }
|
|
//
|
|
// e := fmt.Errorf("testerror")
|
|
// var readers []string
|
|
// err := env.ReaderList(func(msg string) error {
|
|
// readers = append(readers, msg)
|
|
// return e
|
|
// })
|
|
// if err == nil {
|
|
// t.Errorf("expected error")
|
|
// }
|
|
// if !errors.Is(err, e) {
|
|
// t.Errorf("unexpected error: %q (!= %q)", err, e)
|
|
// }
|
|
// if len(readers) != 1 {
|
|
// t.Errorf("unexpected reader list size: %d (!= %d)", len(readers), 1)
|
|
// }
|
|
//}
|
|
//
|
|
//func TestEnv_ReaderList_envInvalid(t *testing.T) {
|
|
// err := (&Env{}).ReaderList(func(msg string) error {
|
|
// t.Logf("%s", msg)
|
|
// return nil
|
|
// })
|
|
// if err == nil {
|
|
// t.Errorf("expected error")
|
|
// }
|
|
//}
|
|
//
|
|
//func TestEnv_ReaderList_nilFunc(t *testing.T) {
|
|
// env, err := NewEnv()
|
|
// if err != nil {
|
|
// t.Fatal(err)
|
|
// }
|
|
// err = env.ReaderList(nil)
|
|
// if err == nil {
|
|
// t.Errorf("expected error")
|
|
// }
|
|
//}
|
|
|
|
func TestEnv_ReaderCheck(t *testing.T) {
|
|
env := setup(t)
|
|
defer clean(env, t)
|
|
|
|
numDead, err := env.ReaderCheck()
|
|
if err != nil {
|
|
t.Error(err)
|
|
}
|
|
if numDead != 0 {
|
|
t.Errorf("unexpected dead readers: %v (!= %v)", numDead, 0)
|
|
}
|
|
}
|
|
|
|
//func TestEnv_Copy(t *testing.T) {
|
|
// testEnvCopy(t, 0, false, false)
|
|
//}
|
|
//
|
|
//func TestEnv_CopyFlags(t *testing.T) {
|
|
// testEnvCopy(t, CopyCompact, true, false)
|
|
//}
|
|
//
|
|
//func TestEnv_CopyFlags_zero(t *testing.T) {
|
|
// testEnvCopy(t, 0, true, false)
|
|
//}
|
|
//
|
|
//func TestEnv_CopyFD(t *testing.T) {
|
|
// testEnvCopy(t, 0, false, true)
|
|
//}
|
|
//
|
|
//func TestEnv_CopyFDFlags(t *testing.T) {
|
|
// testEnvCopy(t, CopyCompact, true, true)
|
|
//}
|
|
//
|
|
//func TestEnv_CopyFDFlags_zero(t *testing.T) {
|
|
// testEnvCopy(t, 0, true, true)
|
|
//}
|
|
//
|
|
//func testEnvCopy(t *testing.T, flags uint, useflags bool, usefd bool) {
|
|
// dircp, err := ioutil.TempDir("", "test-env-copy-")
|
|
// if err != nil {
|
|
// t.Fatal(err)
|
|
// }
|
|
// defer os.RemoveAll(dircp)
|
|
//
|
|
// var fd uintptr
|
|
// if usefd {
|
|
// path := filepath.Join(dircp, "data.mdb")
|
|
// f, err := os.Create(path)
|
|
// if err != nil {
|
|
// t.Error(err)
|
|
// return
|
|
// }
|
|
// fd = f.Fd()
|
|
// defer f.Close()
|
|
// }
|
|
//
|
|
// env := setup(t)
|
|
// defer clean(env, t)
|
|
//
|
|
// item := struct{ k, v []byte }{
|
|
// []byte("k0"),
|
|
// []byte("v0"),
|
|
// }
|
|
//
|
|
// err = env.Update(func(txn *Txn) (err error) {
|
|
// db, err := txn.OpenRoot(0)
|
|
// if err != nil {
|
|
// return err
|
|
// }
|
|
// return txn.Put(db, item.k, item.v, 0)
|
|
// })
|
|
// if err != nil {
|
|
// t.Error(err)
|
|
// }
|
|
//
|
|
// switch {
|
|
// case usefd && useflags:
|
|
// err = env.CopyFDFlag(fd, flags)
|
|
// case usefd && !useflags:
|
|
// err = env.CopyFD(fd)
|
|
// case !usefd && useflags:
|
|
// err = env.CopyFlag(dircp, flags)
|
|
// case !usefd && !useflags:
|
|
// err = env.Copy(dircp)
|
|
// }
|
|
// if err != nil {
|
|
// t.Error(err)
|
|
// }
|
|
//
|
|
// envcp, err := NewEnv()
|
|
// if err != nil {
|
|
// t.Error(err)
|
|
// }
|
|
// err = envcp.Open(dircp, 0, 0644)
|
|
// defer envcp.Close()
|
|
// if err != nil {
|
|
// t.Error(err)
|
|
// return
|
|
// }
|
|
//
|
|
// err = envcp.View(func(txn *Txn) (err error) {
|
|
// db, err := txn.OpenRoot(0)
|
|
// if err != nil {
|
|
// return err
|
|
// }
|
|
// v, err := txn.Get(db, item.k)
|
|
// if err != nil {
|
|
// return err
|
|
// }
|
|
// if !bytes.Equal(v, item.v) {
|
|
// return fmt.Errorf("unexpected value: %q (!= %q)", v, "v0")
|
|
// }
|
|
// return nil
|
|
// })
|
|
// if err != nil {
|
|
// t.Error(err)
|
|
// }
|
|
//}
|
|
|
|
func TestEnv_Sync(t *testing.T) {
|
|
env := setupFlags(t, SafeNoSync)
|
|
defer clean(env, t)
|
|
|
|
item := struct{ k, v []byte }{[]byte("k0"), []byte("v0")}
|
|
|
|
err := env.Update(func(txn *Txn) (err error) {
|
|
db, err := txn.OpenRoot(0)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return txn.Put(db, item.k, item.v, 0)
|
|
})
|
|
if err != nil {
|
|
t.Error(err)
|
|
}
|
|
|
|
err = env.Sync(true, false)
|
|
if err != nil {
|
|
t.Error(err)
|
|
}
|
|
}
|
|
|
|
func setup(t T) *Env {
|
|
return setupFlags(t, 0)
|
|
}
|
|
|
|
func setupFlags(t T, flags uint) *Env {
|
|
env, err := NewEnv()
|
|
if err != nil {
|
|
t.Fatalf("env: %s", err)
|
|
}
|
|
path, err := ioutil.TempDir("", "mdb_test")
|
|
if err != nil {
|
|
t.Fatalf("tempdir: %v", err)
|
|
}
|
|
err = os.MkdirAll(path, 0770)
|
|
if err != nil {
|
|
t.Fatalf("mkdir: %s", path)
|
|
}
|
|
err = env.SetMaxDBs(64 << 10)
|
|
if err == nil {
|
|
t.Fatalf("expecting error")
|
|
}
|
|
err = env.SetMaxDBs(1024)
|
|
if err != nil {
|
|
t.Fatalf("setmaxdbs: %v", err)
|
|
}
|
|
flags |= LifoReclaim
|
|
err = env.Open(path, flags, 0664)
|
|
if err != nil {
|
|
t.Fatalf("open: %s", err)
|
|
}
|
|
|
|
return env
|
|
}
|
|
|
|
type T interface {
|
|
Errorf(format string, vals ...interface{})
|
|
Fatalf(format string, vals ...interface{})
|
|
}
|
|
|
|
func clean(env *Env, t T) {
|
|
path, err := env.Path()
|
|
if err != nil {
|
|
t.Errorf("path: %v", err)
|
|
}
|
|
err = env.Close()
|
|
if err != nil {
|
|
t.Errorf("close: %s", err)
|
|
}
|
|
if path != "" {
|
|
err = os.RemoveAll(path)
|
|
if err != nil {
|
|
t.Errorf("remove: %v", err)
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestEnvCopy(t *testing.T) {
|
|
env := setup(t)
|
|
defer clean(env, t)
|
|
}
|
|
|
|
func TestEnv_MaxKeySize(t *testing.T) {
|
|
env := setup(t)
|
|
defer clean(env, t)
|
|
|
|
n := env.MaxKeySize()
|
|
if n <= 0 {
|
|
t.Errorf("invaild maxkeysize: %d", n)
|
|
}
|
|
}
|
|
|
|
func TestEnv_MaxKeySize_nil(t *testing.T) {
|
|
var env *Env
|
|
n := env.MaxKeySize()
|
|
if n < -1 {
|
|
t.Errorf("invaild maxkeysize: %d", n)
|
|
}
|
|
t.Logf("mdb_env_get_maxkeysize: %d", n)
|
|
}
|
|
|
|
func TestEnv_CloseDBI(t *testing.T) {
|
|
env := setup(t)
|
|
defer clean(env, t)
|
|
|
|
const numdb = 1000
|
|
for i := 0; i < numdb; i++ {
|
|
dbname := fmt.Sprintf("db%d", i)
|
|
|
|
var dbi DBI
|
|
err := env.Update(func(txn *Txn) (err error) {
|
|
dbi, err = txn.CreateDBI(dbname)
|
|
return err
|
|
})
|
|
if err != nil {
|
|
t.Errorf("%s", err)
|
|
}
|
|
|
|
env.CloseDBI(dbi)
|
|
}
|
|
|
|
stat, err := env.Stat()
|
|
if err != nil {
|
|
t.Errorf("%s", err)
|
|
return
|
|
}
|
|
|
|
//nolint:goerr113
|
|
if stat.Entries != numdb {
|
|
t.Errorf("unexpected entries: %d (not %d)", stat.Entries, numdb)
|
|
}
|
|
}
|