mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-07 19:41:19 +00:00
1dd283efe2
* history_early_stop * auto_increment * auto_increment * tx_table
1482 lines
26 KiB
Go
1482 lines
26 KiB
Go
//nolint:goconst
|
|
package mdbx
|
|
|
|
import (
|
|
"bytes"
|
|
"encoding/binary"
|
|
"fmt"
|
|
"os"
|
|
"runtime"
|
|
"syscall"
|
|
"testing"
|
|
"time"
|
|
)
|
|
|
|
func TestTxn_ID(t *testing.T) {
|
|
env := setup(t)
|
|
defer clean(env, t)
|
|
|
|
var id0, id1, id2, id3 uintptr
|
|
var txnInvalid *Txn
|
|
err := env.View(func(txn *Txn) (err error) {
|
|
id0 = txn.ID()
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
t.Error(err)
|
|
}
|
|
if id0 != 3 {
|
|
t.Errorf("unexpected readonly id (before update): %v (!= %v)", id0, 3)
|
|
}
|
|
|
|
txnCached, err := env.BeginTxn(nil, Readonly)
|
|
if err != nil {
|
|
t.Error(err)
|
|
return
|
|
}
|
|
defer txnCached.Abort()
|
|
if txnCached.ID() != 3 {
|
|
t.Errorf("unexpected readonly id (before update): %v (!= %v)", txnCached.ID(), 3)
|
|
}
|
|
if txnCached.getID() != txnCached.ID() {
|
|
t.Errorf("unexpected readonly id (before update): %v (!= %v)", txnCached.ID(), txnCached.getID())
|
|
}
|
|
txnCached.Reset()
|
|
if txnCached.getID() != txnCached.ID() {
|
|
t.Errorf("unexpected reset id: %v (!= %v)", txnCached.ID(), txnCached.getID())
|
|
}
|
|
|
|
err = env.Update(func(txn *Txn) (err error) {
|
|
dbi, err := txn.OpenDBI("test", Create, nil, nil)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
id1 = txn.ID()
|
|
return txn.Put(dbi, []byte("key"), []byte("val"), 0)
|
|
})
|
|
if err != nil {
|
|
t.Errorf("%v", err)
|
|
return
|
|
}
|
|
err = env.View(func(txn *Txn) (err error) {
|
|
id2 = txn.ID()
|
|
txnInvalid = txn
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
t.Errorf("%v", err)
|
|
return
|
|
}
|
|
id3 = txnInvalid.ID()
|
|
|
|
// The ID of txnCached will actually change during the call to
|
|
// txnCached.Renew(). It's imperative that any ID cached in the Txn object
|
|
// does not diverge.
|
|
if txnCached.ID() != txnCached.getID() {
|
|
t.Errorf("unexpected invalid id: %v (!= %v)", txnCached.ID(), txnCached.getID())
|
|
}
|
|
err = txnCached.Renew()
|
|
if err != nil {
|
|
t.Error(err)
|
|
return
|
|
}
|
|
if txnCached.ID() != txnCached.getID() {
|
|
t.Errorf("unexpected invalid id: %v (!= %v)", txnCached.ID(), txnCached.getID())
|
|
}
|
|
|
|
t.Logf("ro txn id:: %v", id1)
|
|
t.Logf("txn id: %v", id1)
|
|
t.Logf("ro txn id: %v", id2)
|
|
t.Logf("bad txn id: %v", id3)
|
|
if id1 != id2 {
|
|
t.Errorf("unexpected readonly id: %v (!= %v)", id2, id1)
|
|
}
|
|
if id3 != 0 {
|
|
t.Errorf("unexpected invalid id: %v (!= %v)", id3, 0)
|
|
}
|
|
if id1 != txnCached.ID() {
|
|
t.Errorf("unexpected invalid id: %v (!= %v)", txnCached.ID(), 0)
|
|
}
|
|
}
|
|
|
|
func TestTxn_errLogf(t *testing.T) {
|
|
env := setup(t)
|
|
defer clean(env, t)
|
|
|
|
runtime.LockOSThread()
|
|
defer runtime.UnlockOSThread()
|
|
|
|
txn, err := env.BeginTxn(nil, 0)
|
|
if err != nil {
|
|
t.Error(err)
|
|
} else {
|
|
defer txn.Abort()
|
|
txn.errf("this is just a test")
|
|
}
|
|
}
|
|
|
|
func TestTxn_finalizer(t *testing.T) {
|
|
env := setup(t)
|
|
defer clean(env, t)
|
|
|
|
runtime.LockOSThread()
|
|
defer runtime.UnlockOSThread()
|
|
|
|
called := make(chan struct{})
|
|
func() {
|
|
txn, err := env.BeginTxn(nil, 0)
|
|
if err != nil {
|
|
t.Error(err)
|
|
} else {
|
|
txn.errLogf = func(string, ...interface{}) {
|
|
close(called)
|
|
}
|
|
}
|
|
}()
|
|
|
|
// make sure that finalizer has a chance to get called. it seems like this
|
|
// may not be consistent across versions of go.
|
|
runtime.GC()
|
|
runtime.Gosched()
|
|
|
|
select {
|
|
case <-called:
|
|
case <-time.After(time.Second):
|
|
t.Errorf("error logging function was not called")
|
|
}
|
|
}
|
|
|
|
func TestTxn_Drop(t *testing.T) {
|
|
env := setup(t)
|
|
defer clean(env, t)
|
|
|
|
db, err := openDBI(env, "db", Create)
|
|
if err != nil {
|
|
t.Error(err)
|
|
return
|
|
}
|
|
|
|
err = env.Update(func(txn *Txn) (err error) {
|
|
return txn.Put(db, []byte("k"), []byte("v"), 0)
|
|
})
|
|
if err != nil {
|
|
t.Error(err)
|
|
return
|
|
}
|
|
|
|
err = env.Update(func(txn *Txn) (err error) {
|
|
return txn.Drop(db, false)
|
|
})
|
|
if err != nil {
|
|
t.Error(err)
|
|
return
|
|
}
|
|
|
|
err = env.View(func(txn *Txn) (err error) {
|
|
_, err = txn.Get(db, []byte("k"))
|
|
return err
|
|
})
|
|
if !IsNotFound(err) {
|
|
t.Error(err)
|
|
return
|
|
}
|
|
|
|
err = env.Update(func(txn *Txn) (err error) {
|
|
return txn.Drop(db, true)
|
|
})
|
|
if err != nil {
|
|
t.Error(err)
|
|
return
|
|
}
|
|
|
|
err = env.View(func(txn *Txn) (err error) {
|
|
_, err = txn.Get(db, []byte("k"))
|
|
return err
|
|
})
|
|
if !IsErrno(err, BadDBI) {
|
|
t.Errorf("mdb_get: %v", err)
|
|
}
|
|
}
|
|
|
|
func TestTxn_Del(t *testing.T) {
|
|
env := setup(t)
|
|
defer clean(env, t)
|
|
|
|
var db DBI
|
|
err := env.Update(func(txn *Txn) (err error) {
|
|
db, err = txn.OpenDBISimple("text", Create)
|
|
return err
|
|
})
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
err = env.Update(func(txn *Txn) (err error) {
|
|
return txn.Put(db, []byte("k"), []byte("v"), 0)
|
|
})
|
|
if err != nil {
|
|
t.Error(err)
|
|
}
|
|
|
|
err = env.Update(func(txn *Txn) (err error) {
|
|
return txn.Del(db, []byte("k"), nil)
|
|
})
|
|
if err != nil {
|
|
t.Error(err)
|
|
}
|
|
|
|
err = env.View(func(txn *Txn) (err error) {
|
|
_, err = txn.Get(db, []byte("k"))
|
|
return err
|
|
})
|
|
if !IsNotFound(err) {
|
|
t.Errorf("mdb_txn_get: %v", err)
|
|
}
|
|
}
|
|
|
|
func TestTxn_Del_dup(t *testing.T) {
|
|
env := setup(t)
|
|
defer clean(env, t)
|
|
|
|
var db DBI
|
|
err := env.Update(func(txn *Txn) (err error) {
|
|
db, err = txn.OpenDBISimple("text", Create|DupSort)
|
|
return err
|
|
})
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
err = env.Update(func(txn *Txn) (err error) {
|
|
return txn.Put(db, []byte("k"), []byte("v"), 0)
|
|
})
|
|
if err != nil {
|
|
t.Error(err)
|
|
}
|
|
|
|
err = env.Update(func(txn *Txn) (err error) {
|
|
return txn.Del(db, []byte("k"), []byte("valignored"))
|
|
})
|
|
if !IsNotFound(err) {
|
|
t.Errorf("mdb_del: %v", err)
|
|
}
|
|
|
|
err = env.View(func(txn *Txn) (err error) {
|
|
v, err := txn.Get(db, []byte("k"))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if string(v) != "v" {
|
|
return fmt.Errorf("unexpected value: %q", v)
|
|
}
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
t.Error(err)
|
|
}
|
|
}
|
|
|
|
func TestTexn_Put_emptyValue(t *testing.T) {
|
|
env := setup(t)
|
|
defer clean(env, t)
|
|
|
|
var db DBI
|
|
err := env.Update(func(txn *Txn) (err error) {
|
|
db, err = txn.OpenDBISimple("text", Create)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
err = txn.Put(db, []byte("k"), nil, 0)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
v, err := txn.Get(db, []byte("k"))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if len(v) != 0 {
|
|
t.Errorf("value: %q (!= \"\")", v)
|
|
}
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
t.Error(err)
|
|
return
|
|
}
|
|
}
|
|
|
|
func TestTxn_PutReserve(t *testing.T) {
|
|
env := setup(t)
|
|
defer clean(env, t)
|
|
|
|
var db DBI
|
|
err := env.Update(func(txn *Txn) (err error) {
|
|
db, err = txn.OpenRoot(0)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
val := "v"
|
|
err = txn.Put(db, []byte("k"), []byte(val), 0)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
p, err := txn.PutReserve(db, []byte("k"), len(val), 0)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
copy(p, val)
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
t.Error(err)
|
|
return
|
|
}
|
|
|
|
err = env.View(func(txn *Txn) (err error) {
|
|
v, err := txn.Get(db, []byte("k"))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if string(v) != "v" {
|
|
return fmt.Errorf("value: %q", v)
|
|
}
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
t.Error(err)
|
|
return
|
|
}
|
|
}
|
|
|
|
func TestTxn_bytesBuffer(t *testing.T) {
|
|
env := setup(t)
|
|
defer clean(env, t)
|
|
|
|
db, err := openRoot(env, 0)
|
|
if err != nil {
|
|
t.Error(err)
|
|
return
|
|
}
|
|
|
|
err = env.Update(func(txn *Txn) (err error) {
|
|
k := new(bytes.Buffer)
|
|
k.WriteString("hello")
|
|
v := new(bytes.Buffer)
|
|
v.WriteString("world")
|
|
return txn.Put(db, k.Bytes(), v.Bytes(), 0)
|
|
})
|
|
if err != nil {
|
|
t.Error(err)
|
|
return
|
|
}
|
|
|
|
err = env.View(func(txn *Txn) (err error) {
|
|
k := new(bytes.Buffer)
|
|
k.WriteString("hello")
|
|
v, err := txn.Get(db, k.Bytes())
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if !bytes.Equal(v, []byte("world")) {
|
|
return fmt.Errorf("unexpected value: %q", v)
|
|
}
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
t.Error(err)
|
|
return
|
|
}
|
|
}
|
|
|
|
func TestTxn_Put_overwrite(t *testing.T) {
|
|
env := setup(t)
|
|
defer clean(env, t)
|
|
|
|
db, err := openRoot(env, 0)
|
|
if err != nil {
|
|
t.Error(err)
|
|
return
|
|
}
|
|
|
|
err = env.Update(func(txn *Txn) (err error) {
|
|
k := []byte("hello")
|
|
v := []byte("world")
|
|
err = txn.Put(db, k, v, 0)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
copy(k, "bye!!")
|
|
copy(v, "toodles")
|
|
err = txn.Put(db, k, v, 0)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
t.Error(err)
|
|
return
|
|
}
|
|
|
|
err = env.View(func(txn *Txn) (err error) {
|
|
v, err := txn.Get(db, []byte("hello"))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if !bytes.Equal(v, []byte("world")) {
|
|
return fmt.Errorf("unexpected value")
|
|
}
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
t.Error(err)
|
|
return
|
|
}
|
|
}
|
|
|
|
func TestTxn_OpenDBI_emptyName(t *testing.T) {
|
|
env := setup(t)
|
|
defer clean(env, t)
|
|
|
|
err := env.View(func(txn *Txn) (err error) {
|
|
_, err = txn.OpenDBISimple("", 0)
|
|
return err
|
|
})
|
|
if !IsErrno(err, NotFound) {
|
|
t.Errorf("mdb_dbi_open: %v", err)
|
|
}
|
|
|
|
err = env.View(func(txn *Txn) (err error) {
|
|
_, err = txn.OpenDBISimple("", Create)
|
|
return err
|
|
})
|
|
if !IsErrnoSys(err, syscall.EACCES) {
|
|
t.Errorf("mdb_dbi_open: %v", err)
|
|
}
|
|
}
|
|
|
|
func TestTxn_OpenDBI_zero(t *testing.T) {
|
|
env := setup(t)
|
|
defer clean(env, t)
|
|
|
|
err := env.View(func(txn *Txn) (err error) {
|
|
_, err = txn.OpenRoot(0)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
_, err = txn.Get(0, []byte("k"))
|
|
return err
|
|
})
|
|
if !IsErrno(err, BadDBI) {
|
|
t.Errorf("mdb_dbi_open: %v", err)
|
|
}
|
|
}
|
|
|
|
func TestTxn_Commit_managed(t *testing.T) {
|
|
env := setup(t)
|
|
defer clean(env, t)
|
|
|
|
err := env.View(func(txn *Txn) (err error) {
|
|
defer func() {
|
|
if e := recover(); e == nil {
|
|
t.Errorf("expected panic: %v", err)
|
|
}
|
|
}()
|
|
_, err2 := txn.Commit()
|
|
return err2
|
|
})
|
|
if err != nil {
|
|
t.Error(err)
|
|
}
|
|
|
|
err = env.View(func(txn *Txn) (err error) {
|
|
defer func() {
|
|
if e := recover(); e == nil {
|
|
t.Errorf("expected panic: %v", err)
|
|
}
|
|
}()
|
|
txn.Abort()
|
|
return fmt.Errorf("abort")
|
|
})
|
|
if err != nil {
|
|
t.Error(err)
|
|
}
|
|
|
|
err = env.View(func(txn *Txn) (err error) {
|
|
defer func() {
|
|
if e := recover(); e == nil {
|
|
t.Errorf("expected panic: %v", err)
|
|
}
|
|
}()
|
|
return txn.Renew()
|
|
})
|
|
if err != nil {
|
|
t.Error(err)
|
|
}
|
|
|
|
err = env.View(func(txn *Txn) (err error) {
|
|
defer func() {
|
|
if e := recover(); e == nil {
|
|
t.Errorf("expected panic: %v", err)
|
|
}
|
|
}()
|
|
txn.Reset()
|
|
return fmt.Errorf("reset")
|
|
})
|
|
if err != nil {
|
|
t.Error(err)
|
|
}
|
|
}
|
|
|
|
func TestTxn_Commit(t *testing.T) {
|
|
env := setup(t)
|
|
defer clean(env, t)
|
|
|
|
runtime.LockOSThread()
|
|
defer runtime.UnlockOSThread()
|
|
|
|
txn, err := env.BeginTxn(nil, 0)
|
|
if err != nil {
|
|
t.Error(err)
|
|
return
|
|
}
|
|
txn.Abort()
|
|
_, err = txn.Commit()
|
|
if !IsErrnoSys(err, syscall.EINVAL) {
|
|
t.Errorf("mdb_txn_commit: %v", err)
|
|
}
|
|
}
|
|
|
|
func TestTxn_Update(t *testing.T) {
|
|
env := setup(t)
|
|
defer clean(env, t)
|
|
|
|
var db DBI
|
|
err := env.Update(func(txn *Txn) (err error) {
|
|
db, err = txn.OpenRoot(Create)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
err = txn.Put(db, []byte("mykey"), []byte("myvalue"), 0)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
t.Errorf("update: %v", err)
|
|
return
|
|
}
|
|
|
|
err = env.View(func(txn *Txn) (err error) {
|
|
v, err := txn.Get(db, []byte("mykey"))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if string(v) != "myvalue" {
|
|
return fmt.Errorf("value: %q", v)
|
|
}
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
t.Errorf("view: %v", err)
|
|
return
|
|
}
|
|
}
|
|
|
|
func TestTxn_View_noSubTxn(t *testing.T) {
|
|
env := setup(t)
|
|
defer clean(env, t)
|
|
|
|
// view transactions cannot create subtransactions. were it possible, they
|
|
// would provide no utility.
|
|
var executed bool
|
|
err := env.View(func(txn *Txn) (err error) {
|
|
return txn.Sub(func(txn *Txn) error {
|
|
executed = true
|
|
return nil
|
|
})
|
|
})
|
|
if err == nil {
|
|
t.Errorf("view: %v", err)
|
|
}
|
|
if executed {
|
|
t.Errorf("view executed: %v", err)
|
|
}
|
|
}
|
|
|
|
func TestTxn_Sub(t *testing.T) {
|
|
env := setup(t)
|
|
defer clean(env, t)
|
|
|
|
var errSubAbort = fmt.Errorf("aborted subtransaction")
|
|
var db DBI
|
|
err := env.Update(func(txn *Txn) (err error) {
|
|
db, err = txn.OpenRoot(Create)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// set the key in the root transaction
|
|
err = txn.Put(db, []byte("mykey"), []byte("myvalue"), 0)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// set the key in a sub transaction
|
|
err = txn.Sub(func(txn *Txn) (err error) {
|
|
return txn.Put(db, []byte("mykey"), []byte("yourvalue"), 0)
|
|
})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// set the key before aborting a subtransaction
|
|
err = txn.Sub(func(txn *Txn) (err error) {
|
|
err = txn.Put(db, []byte("mykey"), []byte("badvalue"), 0)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return errSubAbort
|
|
})
|
|
//nolint:goerr113
|
|
if err != errSubAbort {
|
|
return fmt.Errorf("expected abort: %v", err)
|
|
}
|
|
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
t.Errorf("update: %v", err)
|
|
return
|
|
}
|
|
|
|
err = env.View(func(txn *Txn) (err error) {
|
|
v, err := txn.Get(db, []byte("mykey"))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if string(v) != "yourvalue" {
|
|
return fmt.Errorf("value: %q", v)
|
|
}
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
t.Errorf("view: %v", err)
|
|
return
|
|
}
|
|
}
|
|
|
|
func TestTxn_Flags(t *testing.T) {
|
|
env := setup(t)
|
|
path, err := env.Path()
|
|
if err != nil {
|
|
_ = env.Close()
|
|
t.Error(err)
|
|
return
|
|
}
|
|
defer os.RemoveAll(path)
|
|
|
|
dbflags := uint(ReverseKey | ReverseDup | DupSort | DupFixed)
|
|
err = env.Update(func(txn *Txn) (err error) {
|
|
db, err := txn.OpenDBI("testdb", dbflags|Create, nil, nil)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
err = txn.Put(db, []byte("bcd"), []byte("exval1"), 0)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
err = txn.Put(db, []byte("abcda"), []byte("exval3"), 0)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
err = txn.Put(db, []byte("abcda"), []byte("exval2"), 0)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
cur, err := txn.OpenCursor(db)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer cur.Close()
|
|
k, v, err := cur.Get(nil, nil, Next)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if string(k) != "abcda" { // ReverseKey does not do what one might expect
|
|
return fmt.Errorf("unexpected first key: %q", k)
|
|
}
|
|
if string(v) != "exval2" {
|
|
return fmt.Errorf("unexpected first value: %q", v)
|
|
}
|
|
return nil
|
|
})
|
|
_ = env.Close()
|
|
if err != nil {
|
|
t.Error(err)
|
|
return
|
|
}
|
|
|
|
// opening the database after it is created inherits the original flags.
|
|
env, err = NewEnv()
|
|
if err != nil {
|
|
t.Error(err)
|
|
return
|
|
}
|
|
err = env.SetMaxDBs(1)
|
|
if err != nil {
|
|
t.Error(err)
|
|
return
|
|
}
|
|
defer env.Close()
|
|
err = env.Open(path, 0, 0644)
|
|
if err != nil {
|
|
t.Error(err)
|
|
return
|
|
}
|
|
err = env.View(func(txn *Txn) (err error) {
|
|
db, err := txn.OpenDBI("testdb", 0, nil, nil)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
flags, err := txn.Flags(db)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if flags != dbflags {
|
|
return fmt.Errorf("unexpected flags")
|
|
}
|
|
cur, err := txn.OpenCursor(db)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
k, v, err := cur.Get(nil, nil, Next)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if string(k) != "abcda" {
|
|
return fmt.Errorf("unexpected first key: %q", k)
|
|
}
|
|
if string(v) != "exval2" {
|
|
return fmt.Errorf("unexpected first value: %q", v)
|
|
}
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
t.Error(err)
|
|
return
|
|
}
|
|
}
|
|
|
|
func TestTxn_Renew(t *testing.T) {
|
|
env := setup(t)
|
|
path, err := env.Path()
|
|
if err != nil {
|
|
_ = env.Close()
|
|
t.Error(err)
|
|
return
|
|
}
|
|
defer os.RemoveAll(path)
|
|
defer env.Close()
|
|
|
|
// It is not necessary to call runtime.LockOSThread in this test because
|
|
// the only unmanaged Txn is Readonly.
|
|
|
|
var dbroot DBI
|
|
err = env.Update(func(txn *Txn) (err error) {
|
|
dbroot, err = txn.OpenRoot(0)
|
|
return err
|
|
})
|
|
if err != nil {
|
|
t.Error(err)
|
|
return
|
|
}
|
|
|
|
txn, err := env.BeginTxn(nil, Readonly)
|
|
if err != nil {
|
|
t.Error(err)
|
|
return
|
|
}
|
|
defer txn.Abort()
|
|
_, err = txn.Get(dbroot, []byte("k"))
|
|
if !IsNotFound(err) {
|
|
t.Errorf("get: %v", err)
|
|
}
|
|
|
|
err = env.Update(func(txn *Txn) (err error) {
|
|
dbroot, err = txn.OpenRoot(0)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return txn.Put(dbroot, []byte("k"), []byte("v"), 0)
|
|
})
|
|
if err != nil {
|
|
t.Error(err)
|
|
}
|
|
|
|
_, err = txn.Get(dbroot, []byte("k"))
|
|
if !IsNotFound(err) {
|
|
t.Errorf("get: %v", err)
|
|
}
|
|
txn.Reset()
|
|
|
|
err = txn.Renew()
|
|
if err != nil {
|
|
t.Error(err)
|
|
}
|
|
val, err := txn.Get(dbroot, []byte("k"))
|
|
if err != nil {
|
|
t.Error(err)
|
|
}
|
|
if string(val) != "v" {
|
|
t.Errorf("unexpected value: %q", val)
|
|
}
|
|
}
|
|
|
|
func TestTxn_Reset_doubleReset(t *testing.T) {
|
|
env := setup(t)
|
|
path, err := env.Path()
|
|
if err != nil {
|
|
env.Close()
|
|
t.Error(err)
|
|
return
|
|
}
|
|
defer os.RemoveAll(path)
|
|
defer env.Close()
|
|
|
|
txn, err := env.BeginTxn(nil, Readonly)
|
|
if err != nil {
|
|
t.Error(err)
|
|
return
|
|
}
|
|
defer txn.Abort()
|
|
|
|
txn.Reset()
|
|
txn.Reset()
|
|
}
|
|
|
|
// This test demonstrates that Reset/Renew have no effect on writable
|
|
// transactions. The transaction may be committed after Reset/Renew are called.
|
|
func TestTxn_Reset_writeTxn(t *testing.T) {
|
|
env := setup(t)
|
|
path, err := env.Path()
|
|
if err != nil {
|
|
env.Close()
|
|
t.Error(err)
|
|
return
|
|
}
|
|
defer os.RemoveAll(path)
|
|
defer env.Close()
|
|
|
|
runtime.LockOSThread()
|
|
defer runtime.UnlockOSThread()
|
|
|
|
txn, err := env.BeginTxn(nil, 0)
|
|
if err != nil {
|
|
t.Error(err)
|
|
return
|
|
}
|
|
defer txn.Abort()
|
|
|
|
db, err := txn.OpenRoot(0)
|
|
if err != nil {
|
|
t.Error(err)
|
|
}
|
|
err = txn.Put(db, []byte("k"), []byte("v"), 0)
|
|
if err != nil {
|
|
t.Error(err)
|
|
}
|
|
|
|
// Reset is a noop and Renew will always error out.
|
|
txn.Reset()
|
|
err = txn.Renew()
|
|
if !IsErrnoSys(err, syscall.EINVAL) {
|
|
t.Errorf("renew: %v", err)
|
|
}
|
|
|
|
_, err = txn.Commit()
|
|
if err != nil {
|
|
t.Errorf("commit: %v", err)
|
|
}
|
|
|
|
err = env.View(func(txn *Txn) (err error) {
|
|
val, err := txn.Get(db, []byte("k"))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if string(val) != "v" {
|
|
return fmt.Errorf("unexpected value: %q", val)
|
|
}
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
t.Error(err)
|
|
}
|
|
}
|
|
|
|
func TestTxn_UpdateLocked(t *testing.T) {
|
|
env := setup(t)
|
|
path, err := env.Path()
|
|
if err != nil {
|
|
env.Close()
|
|
t.Error(err)
|
|
return
|
|
}
|
|
defer os.RemoveAll(path)
|
|
defer env.Close()
|
|
|
|
runtime.LockOSThread()
|
|
defer runtime.UnlockOSThread()
|
|
|
|
var dbi DBI
|
|
err = env.UpdateLocked(func(txn *Txn) (err error) {
|
|
dbi, err = txn.OpenRoot(0)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return txn.Put(dbi, []byte("k0"), []byte("v0"), 0)
|
|
})
|
|
if err != nil {
|
|
t.Error(err)
|
|
}
|
|
|
|
err = env.View(func(txn *Txn) (err error) {
|
|
v, err := txn.Get(dbi, []byte("k0"))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if string(v) != "v0" {
|
|
return fmt.Errorf("unexpected value: %q (!= %q)", v, "v0")
|
|
}
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
t.Error(err)
|
|
}
|
|
}
|
|
|
|
func TestTxn_RunTxn(t *testing.T) {
|
|
env := setup(t)
|
|
path, err := env.Path()
|
|
if err != nil {
|
|
env.Close()
|
|
t.Error(err)
|
|
return
|
|
}
|
|
defer os.RemoveAll(path)
|
|
defer env.Close()
|
|
|
|
var dbi DBI
|
|
err = env.RunTxn(0, func(txn *Txn) (err error) {
|
|
dbi, err = txn.OpenRoot(0)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return txn.Put(dbi, []byte("k0"), []byte("v0"), 0)
|
|
})
|
|
if err != nil {
|
|
t.Error(err)
|
|
}
|
|
|
|
err = env.RunTxn(Readonly, func(txn *Txn) (err error) {
|
|
v, err := txn.Get(dbi, []byte("k0"))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if string(v) != "v0" {
|
|
return fmt.Errorf("unexpected value: %q (!= %q)", v, "v0")
|
|
}
|
|
err = txn.Put(dbi, []byte("k1"), []byte("v1"), 0)
|
|
if err == nil {
|
|
return fmt.Errorf("allowed to Put in a readonly Txn")
|
|
}
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
t.Error(err)
|
|
}
|
|
}
|
|
|
|
func TestTxn_Stat(t *testing.T) {
|
|
env := setup(t)
|
|
path, err := env.Path()
|
|
if err != nil {
|
|
env.Close()
|
|
t.Error(err)
|
|
return
|
|
}
|
|
defer os.RemoveAll(path)
|
|
defer env.Close()
|
|
|
|
var dbi DBI
|
|
err = env.Update(func(txn *Txn) (err error) {
|
|
dbi, err = txn.OpenDBISimple("testdb", Create)
|
|
return err
|
|
})
|
|
if err != nil {
|
|
t.Errorf("%s", err)
|
|
return
|
|
}
|
|
|
|
err = env.Update(func(txn *Txn) (err error) {
|
|
put := func(k, v []byte) {
|
|
if err == nil {
|
|
err = txn.Put(dbi, k, v, 0)
|
|
}
|
|
}
|
|
put([]byte("a"), []byte("1"))
|
|
put([]byte("b"), []byte("2"))
|
|
put([]byte("c"), []byte("3"))
|
|
return err
|
|
})
|
|
if err != nil {
|
|
t.Errorf("%s", err)
|
|
}
|
|
|
|
var stat *Stat
|
|
err = env.View(func(txn *Txn) (err error) {
|
|
stat, err = txn.StatDBI(dbi)
|
|
return err
|
|
})
|
|
if err != nil {
|
|
t.Errorf("%s", err)
|
|
return
|
|
}
|
|
if stat.Entries != 3 {
|
|
t.Errorf("unexpected entries: %d (expected %d)", stat.Entries, 3)
|
|
}
|
|
}
|
|
|
|
func TestSequence(t *testing.T) {
|
|
env := setup(t)
|
|
path, err := env.Path()
|
|
if err != nil {
|
|
env.Close()
|
|
t.Error(err)
|
|
return
|
|
}
|
|
defer os.RemoveAll(path)
|
|
defer env.Close()
|
|
|
|
var dbi1 DBI
|
|
var dbi2 DBI
|
|
err = env.Update(func(txn *Txn) (err error) {
|
|
dbi1, err = txn.OpenDBISimple("testdb", Create)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
dbi2, err = txn.OpenDBISimple("testdb2", Create)
|
|
return err
|
|
})
|
|
if err != nil {
|
|
t.Errorf("%s", err)
|
|
return
|
|
}
|
|
|
|
err = env.Update(func(txn *Txn) (err error) {
|
|
v, err := txn.Sequence(dbi1, 0) // 0 accepted, validate on app level
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if v != 0 {
|
|
t.Errorf("unexpected value: %d (expected %d)", v, 0)
|
|
}
|
|
v, err = txn.Sequence(dbi2, 2)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if v != 0 {
|
|
t.Errorf("unexpected value: %d (expected %d)", v, 1)
|
|
}
|
|
|
|
v, err = txn.Sequence(dbi1, 3)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if v != 0 {
|
|
t.Errorf("unexpected value: %d (expected %d)", 0, 0)
|
|
}
|
|
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
t.Errorf("%s", err)
|
|
}
|
|
|
|
err = env.View(func(txn *Txn) (err error) {
|
|
v, err := txn.Sequence(dbi1, 0)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if v != 3 {
|
|
t.Errorf("unexpected value: %d (expected %d)", v, 3)
|
|
}
|
|
|
|
_, err = txn.Sequence(dbi1, 3) // error if > 0 in read tx
|
|
if err == nil {
|
|
t.Errorf("error expected")
|
|
}
|
|
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
t.Errorf("%s", err)
|
|
return
|
|
}
|
|
}
|
|
|
|
func BenchmarkTxn_Sub_commit(b *testing.B) {
|
|
env := setup(b)
|
|
path, err := env.Path()
|
|
if err != nil {
|
|
env.Close()
|
|
b.Error(err)
|
|
return
|
|
}
|
|
defer os.RemoveAll(path)
|
|
defer env.Close()
|
|
|
|
err = env.Update(func(txn *Txn) (err error) {
|
|
b.ResetTimer()
|
|
defer b.StopTimer()
|
|
for i := 0; i < b.N; i++ {
|
|
err = txn.Sub(func(txn *Txn) (err error) { return nil })
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
b.Error(err)
|
|
return
|
|
}
|
|
}
|
|
|
|
func BenchmarkTxn_Sub_abort(b *testing.B) {
|
|
env := setup(b)
|
|
path, err := env.Path()
|
|
if err != nil {
|
|
env.Close()
|
|
b.Error(err)
|
|
return
|
|
}
|
|
defer os.RemoveAll(path)
|
|
defer env.Close()
|
|
|
|
var e = fmt.Errorf("abort")
|
|
|
|
err = env.Update(func(txn *Txn) (err error) {
|
|
b.ResetTimer()
|
|
defer b.StopTimer()
|
|
for i := 0; i < b.N; i++ {
|
|
_ = txn.Sub(func(txn *Txn) (err error) { return e })
|
|
}
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
b.Error(err)
|
|
return
|
|
}
|
|
}
|
|
|
|
func BenchmarkTxn_abort(b *testing.B) {
|
|
env := setup(b)
|
|
path, err := env.Path()
|
|
if err != nil {
|
|
env.Close()
|
|
b.Error(err)
|
|
return
|
|
}
|
|
defer os.RemoveAll(path)
|
|
defer env.Close()
|
|
|
|
var e = fmt.Errorf("abort")
|
|
|
|
b.ResetTimer()
|
|
for i := 0; i < b.N; i++ {
|
|
_ = env.Update(func(txn *Txn) error { return e })
|
|
}
|
|
}
|
|
|
|
func BenchmarkTxn_commit(b *testing.B) {
|
|
env := setup(b)
|
|
path, err := env.Path()
|
|
if err != nil {
|
|
env.Close()
|
|
b.Error(err)
|
|
return
|
|
}
|
|
defer os.RemoveAll(path)
|
|
defer env.Close()
|
|
|
|
b.ResetTimer()
|
|
for i := 0; i < b.N; i++ {
|
|
err := env.Update(func(txn *Txn) error { return nil })
|
|
if err != nil {
|
|
b.Error(err)
|
|
return
|
|
}
|
|
}
|
|
}
|
|
|
|
func BenchmarkTxn_ro(b *testing.B) {
|
|
env := setup(b)
|
|
path, err := env.Path()
|
|
if err != nil {
|
|
_ = env.Close()
|
|
b.Error(err)
|
|
return
|
|
}
|
|
defer os.RemoveAll(path)
|
|
defer env.Close()
|
|
|
|
b.ResetTimer()
|
|
for i := 0; i < b.N; i++ {
|
|
err := env.View(func(txn *Txn) error { return nil })
|
|
if err != nil {
|
|
b.Error(err)
|
|
return
|
|
}
|
|
}
|
|
}
|
|
|
|
func BenchmarkTxn_unmanaged_abort(b *testing.B) {
|
|
env := setup(b)
|
|
path, err := env.Path()
|
|
if err != nil {
|
|
_ = env.Close()
|
|
b.Error(err)
|
|
return
|
|
}
|
|
defer os.RemoveAll(path)
|
|
defer env.Close()
|
|
|
|
runtime.LockOSThread()
|
|
defer runtime.UnlockOSThread()
|
|
|
|
b.ResetTimer()
|
|
for i := 0; i < b.N; i++ {
|
|
txn, err := env.BeginTxn(nil, 0)
|
|
if err != nil {
|
|
b.Error(err)
|
|
return
|
|
}
|
|
txn.Abort()
|
|
}
|
|
}
|
|
|
|
func BenchmarkTxn_unmanaged_commit(b *testing.B) {
|
|
env := setup(b)
|
|
path, err := env.Path()
|
|
if err != nil {
|
|
_ = env.Close()
|
|
b.Error(err)
|
|
return
|
|
}
|
|
defer os.RemoveAll(path)
|
|
defer env.Close()
|
|
|
|
runtime.LockOSThread()
|
|
defer runtime.UnlockOSThread()
|
|
|
|
b.ResetTimer()
|
|
for i := 0; i < b.N; i++ {
|
|
txn, err := env.BeginTxn(nil, 0)
|
|
if err != nil {
|
|
b.Error(err)
|
|
return
|
|
}
|
|
txn.Abort()
|
|
}
|
|
}
|
|
|
|
func BenchmarkTxn_unmanaged_ro(b *testing.B) {
|
|
env := setup(b)
|
|
path, err := env.Path()
|
|
if err != nil {
|
|
_ = env.Close()
|
|
b.Error(err)
|
|
return
|
|
}
|
|
defer os.RemoveAll(path)
|
|
defer env.Close()
|
|
|
|
// It is not necessary to call runtime.LockOSThread here because the txn is
|
|
// Readonly
|
|
|
|
b.ResetTimer()
|
|
for i := 0; i < b.N; i++ {
|
|
txn, err := env.BeginTxn(nil, Readonly)
|
|
if err != nil {
|
|
b.Error(err)
|
|
return
|
|
}
|
|
txn.Abort()
|
|
}
|
|
}
|
|
|
|
func BenchmarkTxn_renew(b *testing.B) {
|
|
env := setup(b)
|
|
path, err := env.Path()
|
|
if err != nil {
|
|
env.Close()
|
|
b.Error(err)
|
|
return
|
|
}
|
|
defer os.RemoveAll(path)
|
|
defer env.Close()
|
|
|
|
// It is not necessary to call runtime.LockOSThread here because the txn is
|
|
// Readonly
|
|
|
|
txn, err := env.BeginTxn(nil, Readonly)
|
|
if err != nil {
|
|
b.Error(err)
|
|
return
|
|
}
|
|
defer txn.Abort()
|
|
|
|
b.ResetTimer()
|
|
for i := 0; i < b.N; i++ {
|
|
txn.Reset()
|
|
err = txn.Renew()
|
|
if err != nil {
|
|
b.Error(err)
|
|
return
|
|
}
|
|
}
|
|
}
|
|
|
|
func BenchmarkTxn_Put_append(b *testing.B) {
|
|
env := setup(b)
|
|
path, err := env.Path()
|
|
if err != nil {
|
|
_ = env.Close()
|
|
b.Error(err)
|
|
return
|
|
}
|
|
defer os.RemoveAll(path)
|
|
defer env.Close()
|
|
err = env.SetGeometry(-1, -1, 1024*1024, -1, -1, 4096)
|
|
if err != nil {
|
|
b.Error(err)
|
|
return
|
|
}
|
|
|
|
var db DBI
|
|
|
|
err = env.Update(func(txn *Txn) (err error) {
|
|
db, err = txn.OpenRoot(0)
|
|
return err
|
|
})
|
|
if err != nil {
|
|
b.Errorf("dbi: %v", err)
|
|
return
|
|
}
|
|
|
|
b.ResetTimer()
|
|
err = env.Update(func(txn *Txn) (err error) {
|
|
for i := 0; i < b.N; i++ {
|
|
var k [8]byte
|
|
binary.BigEndian.PutUint64(k[:], uint64(i))
|
|
err = txn.Put(db, k[:], k[:], Append)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
b.StopTimer()
|
|
defer b.StartTimer()
|
|
|
|
return txn.Drop(db, false)
|
|
})
|
|
if err != nil {
|
|
b.Errorf("put: %v", err)
|
|
}
|
|
}
|
|
|
|
func BenchmarkTxn_Put_append_noflag(b *testing.B) {
|
|
env := setup(b)
|
|
path, err := env.Path()
|
|
if err != nil {
|
|
_ = env.Close()
|
|
b.Error(err)
|
|
return
|
|
}
|
|
defer os.RemoveAll(path)
|
|
defer env.Close()
|
|
err = env.SetGeometry(-1, -1, 1024*1024, -1, -1, 4096)
|
|
if err != nil {
|
|
b.Fatalf("Cannot set mapsize: %s", err)
|
|
}
|
|
|
|
var db DBI
|
|
|
|
err = env.Update(func(txn *Txn) (err error) {
|
|
db, err = txn.OpenRoot(0)
|
|
return err
|
|
})
|
|
if err != nil {
|
|
b.Errorf("dbi: %v", err)
|
|
return
|
|
}
|
|
|
|
err = env.Update(func(txn *Txn) (err error) {
|
|
for i := 0; i < b.N; i++ {
|
|
var k [8]byte
|
|
binary.BigEndian.PutUint64(k[:], uint64(i))
|
|
err = txn.Put(db, k[:], k[:], 0)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
b.StopTimer()
|
|
defer b.StartTimer()
|
|
return txn.Drop(db, false)
|
|
})
|
|
if err != nil {
|
|
b.Errorf("put: %v", err)
|
|
}
|
|
}
|
|
|
|
func openRoot(env *Env, flags uint) (DBI, error) {
|
|
var db DBI
|
|
dotxn := env.View
|
|
if flags != 0 {
|
|
dotxn = env.Update
|
|
}
|
|
err := dotxn(func(txn *Txn) (err error) {
|
|
db, err = txn.OpenRoot(flags)
|
|
return err
|
|
})
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
return db, nil
|
|
}
|
|
|
|
func openDBI(env *Env, key string, flags uint) (DBI, error) {
|
|
var db DBI
|
|
err := env.Update(func(txn *Txn) (err error) {
|
|
db, err = txn.OpenDBI(key, flags, nil, nil)
|
|
return err
|
|
})
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
return db, nil
|
|
}
|
|
|
|
func TestNameToNano(t *testing.T) {
|
|
if toDuration(65536).Milliseconds() != 1_000 {
|
|
t.Error("unexpected result")
|
|
}
|
|
|
|
if toDuration(65536*1_000).Milliseconds() != 1_000*1_000 {
|
|
t.Error("unexpected result")
|
|
}
|
|
}
|