erigon-pulse/ethdb/mdbx/env_test.go
2021-03-15 09:41:02 +07:00

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)
}
}